- Микроконтроллер:
dsPIC33EP512MU810
- Компилятор:
MikroC
Я пытаюсь запросить несколько байт из удаленного устройства через UART
. Чтобы получить необходимую информацию, вы отправляете один байт запроса, чтобы получить один байт данных. У меня возникают трудности с идентификацией байтов данных при запросе более одного.MikroC, UART. Запрашивающая несколько байт
УАПП Принимающий обрабатывается с помощью прерывания:
void UART2RXInterrupt() iv IVT_ADDR_U2RXINTERRUPT
{
uart_rd2[LoopVar0] = UART2_Read(); //Read into buffer
LoopVar0++;
if (LoopVar0 >= 1)
{
LoopVar0 = 0;
ready0 = 1;
}
U2RXIF_bit = 0; //Reset interrupt flag
}
Мой код для одного байта данных выглядит следующим образом:
UART2_Write(0x11); //Request byte
if (ready0 == 1) //Data received and ready
{
//Parse data byte
ready0 = 0; //Reset data received bit
}
Это прекрасно работает.
Если мне нужно больше, чем просто байт один данных я делаю следующим образом
UART2_Write(0x11); //Request byte 11h
if (ready0 == 1) //Data received and ready
{
//Parse data byte
ready0 = 0; //Reset data received bit
}
UART2_Write(0x14); //Request byte 14h
if (ready0 == 1) //Data received and ready
{
//Parse data byte
ready0 = 0; //Reset data received bit
}
Проблема этого метода состоит в том, что следующий запрос байт может быть уже передан не будучи уверенным, что последний байт данных правильно получено.
Это приводит к тому, что данные становятся «скремблированными». Это означает, что байт данных из последнего запроса может быть проанализирован в текущем запросе и так далее.
Я не могу найти подходящее решение для этого. Любые предложения были бы замечательными!
Круговая реализация буфера:
UART прерываний:
void UART2RXInterrupt() iv IVT_ADDR_U2RXINTERRUPT
{
uart_rcv[write_buffer_pointer0] = UART2_Read(); // put received char in circular buffer
write_buffer_pointer0++; // increment pointer
if (write_buffer_pointer0 > 511) {
write_buffer_pointer0 = 0; // reset pointer
}
U2RXIF_bit = 0;
}
Основная обработка петли:
void service_UART()
{
UART_send(0x14); //MAP request
UART_send(0x1D); //ECT request
if (read_buffer_pointer0 != write_buffer_pointer0) { // new data in circular buffer
ser_in(); // handle incoming serial data
}
}
УАПП отправить рутина:
void UART_send(volatile char data0)
{
uart_snd[write_buffer_pointer1] = data0;
write_buffer_pointer1++; // increment pointer
if (write_buffer_pointer1 > 511) {
write_buffer_pointer1 = 0; // reset pointer
}
UART2_Write(data0);
}
Обработка полученных данных:
void ser_in()
{
volatile char rqst;
volatile char resp;
resp = uart_rcv[read_buffer_pointer0]; // read next character from buffer
read_buffer_pointer0++; // increment read buffer pointer
if (read_buffer_pointer0 > 511) { // if pointer past end of buffer
read_buffer_pointer0 = 0; // reset pointer
}
rqst = uart_snd[read_buffer_pointer1];
read_buffer_pointer1++;
if (read_buffer_pointer1 > 511) {
read_buffer_pointer1 = 0;
}
// do something with the data here.
if (rqst == 0x14) //If MAP request
{
//Handle MAP Data
}
if (rqst == 0x1D) //If ECT request
{
//Handle ECT Data
}
}
Что вы за платформа? Какова ваша платформа/SDK? Как и где управляется 'ready0'? Нам нужно больше информации и кода, которые помогут вам. – LPs
Я добавил информацию. Thnx! – Felix
Зачем вам сбросить свой индекс буфера каждый раз, когда получен байтов? – LPs