흔히들,
빙챗이나 ChatGPT를 이용하여 프로그래밍에 이용하면서
답변으로 아두이노 샘플 코드를 많이 제안 받습니다.
즉, 해당 칩셋의 해당 SDK의 API를 이용하여
코드 제안을 받기는 어려운 상황인 것 같고
아두이노 샘플코드가 가장 일반적이어서 그런 것 같기도 합니다.
그럼, 이 제안 받은 아두이코 코드를 기반으로
자신의 칩셋과 거기에 맞는 SDK의 API로 실제 업무에 적용하기 위해 코딩을 하려면,,,
해당 칩셋의 SDK의 API는 어느정도 익숙하다고 할때
기존 빙챗으로 제안받은
아두이노 코드를 원하는 프로젝트 업무에 포팅을 하면 됩니다.
그러기 위해서 전제조건이
제안 받은 아두이노 코드의 아두이노 API의 내부 구현 코드를 확인하고
이 내부 코드를 자신의 프로젝트의 칩셋 기반의 SDK API로 대체하면 될 것 입니다.
예를들어,
I2C 아두이노 라이브러리인 Wire라이브러리의
TwoWire::write, TwoWire::available, TwoWire::read
등의 함수 내부를 확인하면 아래와 같은 링크의 내용의 코드로 구현된 것을 알 수 있습니다.
ArduinoCore-avr/Wire.cpp at master · arduino/ArduinoCore-avr (github.com)
만약, 아래와 같은 I2C통신을 위한 아두이노 코드를
빙챗이나 챗GPT에게서 제안 받았다면,
#include <Wire.h>
void setup() {
Wire.begin(); // join i2c bus (address optional for master)
}
void loop() {
Wire.beginTransmission(8); // transmit to device #8
Wire.write("hello "); // sends five bytes
Wire.write("world");
Wire.endTransmission(); // stop transmitting
delay(500);
}
위의 제안 받은 코드를 자신의 칩셉의 SDK API로 구현해야 합니다.
이때 이미 언급드린 위 링크에서
begin, beginTransmission, write, endTransmission 함수등을 내부에 아두이노 기반의 AVR 기반의
API로 구현된 것을 알 수 있습니다.
예를들면,
Wire.beginTransmission은 다음과 같고,
void TwoWire::beginTransmission(uint8_t address)
{
// indicate that we are transmitting
transmitting = 1;
// set address of targeted slave
txAddress = address;
// reset tx buffer iterator vars
txBufferIndex = 0;
txBufferLength = 0;
}
void TwoWire::beginTransmission(int address)
{
beginTransmission((uint8_t)address);
}
Wire.write 함수는 이렇게,
size_t TwoWire::write(uint8_t data)
{
if(transmitting){
// in master transmitter mode
// don't bother if buffer is full
if(txBufferLength >= BUFFER_LENGTH){
setWriteError();
return 0;
}
// put byte in tx buffer
txBuffer[txBufferIndex] = data;
++txBufferIndex;
// update amount in buffer
txBufferLength = txBufferIndex;
}else{
// in slave send mode
// reply to master
twi_transmit(&data, 1);
}
return 1;
}
// must be called in:
// slave tx event callback
// or after beginTransmission(address)
size_t TwoWire::write(const uint8_t *data, size_t quantity)
{
if(transmitting){
// in master transmitter mode
for(size_t i = 0; i < quantity; ++i){
write(data[i]);
}
}else{
// in slave send mode
// reply to master
twi_transmit(data, quantity);
}
return quantity;
}
Wire.endTransmission 함수는 다음과 같이 AVR API를 사용하여 코딩되어 있습니다.
uint8_t TwoWire::endTransmission(uint8_t sendStop)
{
// transmit buffer (blocking)
uint8_t ret = twi_writeTo(txAddress, txBuffer, txBufferLength, 1, sendStop);
// reset tx buffer iterator vars
txBufferIndex = 0;
txBufferLength = 0;
// indicate that we are done transmitting
transmitting = 0;
return ret;
}
이렇게, 아두이노 내부코드는 AVR I2C API인,
twi_transmit, twi_writeTo
등의 함수로 구현되어 있습니다.
이 함수들을 자신의 칩셋 SDK API로 대체하여 구현해서
예를들면,
자신의 I2C 라이브러리가 halI2CWriteBytes() 와 halI2CReadBytes()로 제공된다면,
twi_transmit / twi_writeTo 함수를 halI2CWriteBytes 함수로 대체하여 구현하면 된다는 것입니다.
그러면,
초기에 빙챗이나 ChatGPT로 제안받은 아두이노 코드를 이용하면서
자신의 업무에 적용하면 됩니다.
(추가1)
물론,
twi_transmit / twi_writeTo 함수를 halI2CWriteBytes 함수로 대체하여 구현할 때
일대일로 매핑되어 대체되긴 힘들고 어느정도 workaround를 가지고 포팅을 하는 것을 의미합니다.
그럼,
공유합니다.
'개발 이야기 > 임베디드 개발' 카테고리의 다른 글
아두이노 i2c 라이브러리인 Wire 라이브러리를 사용하지 않고 i2c_write / i2c_read 함수 만들기 (0) | 2023.04.09 |
---|---|
[KOCW] 오픈소스 드론 - 드론의 개념과 이론, 제작, 운용, 항공법규의 이해 (0) | 2023.04.06 |
화웨이 자체적으로 14nm EDA개발 완료 (0) | 2023.03.27 |
efr32mg13 에서 efr32xg22 로의 포팅에 필요한 수정사항 (0) | 2023.03.24 |
실리콘랩스에서 나오는 칩셋들 정리 (0) | 2023.03.23 |