【问题标题】:MikroC, UART. Requesting multiple bytesMikroC,UART。请求多个字节
【发布时间】:2015-12-23 10:43:23
【问题描述】:
  • 微控制器:dsPIC33EP512MU810
  • 编译器:MikroC

我正在尝试通过UART 从远程设备请求多个字节。 要获取所需的信息,您发送一个请求字节以接收单个数据字节。 请求多个字节时,我无法识别数据字节。

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
  }
 }

UART 发送例程:

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?我们需要更多信息和代码来帮助您。
  • 我添加了信息。谢谢!
  • 为什么每次收到一个字节就重置缓冲区索引?
  • 我没有得到你。我想您应该将if (ready0 == 1) 更改为while (ready0 == 0)。这允许您在单个请求后等待每个单个字符。
  • 如果你想增加缓冲区大小,你必须实现一个假脱机程序。然后是中断中的输入索引和消费者的输出索引。在这种情况下,你的缓冲区变成了一个循环队列。

标签: c communication microcontroller uart dspic


【解决方案1】:

用头尾指针(索引)实现两个circular queues,大到足以适应吞吐率。确定响应者在请求之间所需的最短时间并尊重它。每次发送请求时,将其添加到第一个队列,即“请求发送队列”。当响应到达时,将其添加到中断例程中的第二个队列并增加头指针。这是您的“响应接收队列”。

然后您可以在主循环中异步处理事务。比较第二个队列的头部和尾部。每次第二个队列变得不平衡(头部不等于尾部)时,您至少收到一个响应。第一个队列告诉您它响应的请求。每次处理完响应后,两个队列的尾指针都会前移。

如果您没有收到响应(如果队列长时间保持不平衡),您可能需要重置队列,并确保队列不会被覆盖。

【讨论】:

  • 我正在努力解决这个问题,因为我还不熟悉循环队列。如果我理解正确,我需要这样做:- 每次发送请求时,将其添加到“请求队列” - 每次“接收队列”由于发生 UART Rx 中断而提前;从两个队列中出列或“弹出”以获取具有相应数据字节的请求字节。
  • @Felix 是的,中断例程会在新数据到达时不断将新数据推入队列,无论先前交换的处理是否已完成。维护请求队列会告诉您弹出的下一个响应属于哪个请求。
  • 好的。我将需要一些时间来实施和测试。这听起来像是一个很好的解决方案。认为大小应该等于不同请求的总数?
  • @Felix 不,大小由吞吐率决定。考虑只有一种类型的请求的情况,但响应者的回复速度比请求者处理回复的速度要快。如果多个请求发送得太快,队列将缓冲交换,直到请求者赶上。
  • 我在原始帖子中添加了我对循环缓冲区结构的第一次尝试。虽然还没有机会测试它。你怎么看?
猜你喜欢
  • 1970-01-01
  • 2013-02-23
  • 1970-01-01
  • 2013-12-20
  • 1970-01-01
  • 2022-08-17
  • 1970-01-01
  • 2015-12-04
  • 1970-01-01
相关资源
最近更新 更多