STM32串口丢失数据

STM32串口发送必须先检测状态,否则第一个字节无法发出,发送完毕,必须检测发送状态是否完成,否则,发送不成功,使用stm32f10x调试串口通讯时,发现一个出错的现象,硬件复位重启之后,发送测试数据0x01 0x02 0x03 0x04…接收端收到的数据为:0x02 0x03 0x04,第一个数据丢失。换成发送别的数值的数据,如0x06 0x0ff,则接收到0x0ff,0x06丢失。错误依旧。

故障排除过程:
1、刚开始怀疑是接收端的错误,我是使用电脑串口,运行串口辅助调试工具接收,换成其他软件后,发现故障依旧,而且电脑软件一直是开启状态,不像和电脑软件有关。
2、使用单步调试,单步运行各个发送指令,都正常。能收到0x01 0x02 0x03 0x04的数据。间接的排除了不是电脑软件的问题,而是其他的错误。
3、单步调试运行虽然正常了,但连续运行时,错误依旧。现在有点摸不到头绪了,单步运行正常,看起来编程没有出错,那故障在哪里呢?测试程序如下

  USART_SendData(USART2, 0x01);                                 //A 
  while(USART_GetFlagStatus(USART2, USART_FLAG_TC) == RESET);   //B 
  USART_SendData(USART2, 0x02);                                 //C 
  while(USART_GetFlagStatus(USART2, USART_FLAG_TC) == RESET); 
  USART_SendData(USART2, 0x03); 
  while(USART_GetFlagStatus(USART2, USART_FLAG_TC) == RESET); 
  USART_SendData(USART2, 0x04); 
  while(USART_GetFlagStatus(USART2, USART_FLAG_TC) == RESET); 

4、猜测,也许是因为某个特殊原因,使第二个数据覆盖了首个数据,使得首个数据丢失。假设:在执行B指令时,USART的 TC 状态位==SET,那么就会紧接着执行C指令,也就有可能发生数据的覆盖。于是,在A指令前,加入如下指令:USART_ClearFlag(USART2,USART_FLAG_TC);
5、加入上一条指令后,运行,错误消失了。说明上一个假设,应该是成立的。
6、查阅stm32f10x参考手册,找到这样一句话:
TC:发送完成
当包含有数据的一帧发送完成后,由硬件将该位置位。如果USART_CR1中的TCIE为1,则产生中断。由软件序列清除该位(先读USART_SR,然后写入USART_DR)。TC位 也可以通过写入0来清除,只有在多缓存通讯中才推荐这种清除程序。
0:发送还未完成;1:发送完成。
7、注意到这一句:由软件序列清除该位(先读USART_SR,然后写入USART_DR)。 也就是说,要先read USART_SR,然后write USART_DR,才能完成TC状态位的清除。而硬件复位后,串口发送的首个数据之前没有read SR的操作,是直接write DR,也就是说,TC没有被清除掉。 说明第4步的猜测是对的。
8、那么,应该把指令A前面加的USART_ClearFlag(USART2,USART_FLAG_TC); 改为USART_GetFlagStatus(USART2, USART_FLAG_TC);,应该也能消除错误。测试后证实,确实如此,在发送首个数据之前,先读取一下USART_SR,那么就不会出现首个数据丢失的情况了。
9、总结:硬件复位后,串口发送首个数据之前,先读取一下USART_SR,则能够保证首个数据发送时,不出现覆盖的情况。当然,也有别的方法,比如先清除TC状态位,或是,在write USART_DR之后,加入一个小延时,让数据发送完毕,应该也能间接排除这个错误。

或者终极解决办法

int hal_uart_sendbuf(int uart_id, uint8_t *buf, uint16_t len)
{
    for(i=0; i<len; i++)
    {
      while((USART1->SR&USART_FLAG_TXE)==0);  //while(USART_GetFlagStatus(USART2,USART_FLAG_TXE)==RESET);
	    USART_SendData(USART1,buf[i]);
      while((USART1->SR&USART_FLAG_TC)==0); //while(USART_GetFlagStatus(USART2,USART_FLAG_TC)==RESET); 			
    }
   
}

之前有说过,串口初始化后,TC标志位为1,这样会导致第一个for循环中的while循环直接跳出进入第二个for循环,进而导致第一个数组中的第一个数据刚写入到USART_DR寄存器,还没来得及转到移位寄存器,数组中的第二个数据又写入到USART_DR寄存器,进而覆盖了第一个数据。
可能会有人问,那第二个、第三个…,为什么没有被覆盖,这里解释以下:
在这里插入图片描述
红线框着的内容说明了如何清除TC标志位的其中一种方法,而while(USART_GetFlagStatus(USARTx, USART_FLAG_TC) == RESET);这一条语句就是读USART_SR寄存器,USART_SendData(USARTx, Array[i]);这一条语句就是写USART_DR寄存器。这样,每一次for循环的结束,和下一次for循环的开始,其结合作用就是清除TC标志位,所以这也就是为什么数组中的其他位数据不被覆盖的原因
在这里插入图片描述
注释掉的3句话任意一种都可以满足要求

  • 2
    点赞
  • 11
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: 在STM32中,使用串口接收数据需要借助中断或者轮询方式来实现。下面分别介绍这两种方式的实现方法。 ### 中断方式 1. 配置串口 ```c #include "stm32f10x.h" void uart_init(void) { USART_InitTypeDef usart_init; // 串口配置 usart_init.USART_BaudRate = 115200; usart_init.USART_WordLength = USART_WordLength_8b; usart_init.USART_StopBits = USART_StopBits_1; usart_init.USART_Parity = USART_Parity_No; usart_init.USART_Mode = USART_Mode_Rx; usart_init.USART_HardwareFlowControl = USART_HardwareFlowControl_None; USART_Init(USART1, &usart_init); // 使能串口接收中断 USART_ITConfig(USART1, USART_IT_RXNE, ENABLE); // 使能串口 USART_Cmd(USART1, ENABLE); } ``` 2. 编写中断服务函数 ```c void USART1_IRQHandler(void) { if (USART_GetITStatus(USART1, USART_IT_RXNE) != RESET) { uint8_t data = USART_ReceiveData(USART1); // 处理接收到的数据 USART_ClearITPendingBit(USART1, USART_IT_RXNE); } } ``` 在中断服务函数中,我们首先判断接收到数据的中断标志是否被置位,如果被置位则说明有数据到来,我们可以通过`USART_ReceiveData`函数获取接收到的数据,并进行处理。最后,需要清除中断标志位。 ### 轮询方式 1. 配置串口 ```c #include "stm32f10x.h" void uart_init(void) { USART_InitTypeDef usart_init; // 串口配置 usart_init.USART_BaudRate = 115200; usart_init.USART_WordLength = USART_WordLength_8b; usart_init.USART_StopBits = USART_StopBits_1; usart_init.USART_Parity = USART_Parity_No; usart_init.USART_Mode = USART_Mode_Rx; usart_init.USART_HardwareFlowControl = USART_HardwareFlowControl_None; USART_Init(USART1, &usart_init); // 使能串口 USART_Cmd(USART1, ENABLE); } ``` 2. 轮询接收数据 ```c void uart_receive(void) { uint8_t data; while (1) { if (USART_GetFlagStatus(USART1, USART_FLAG_RXNE) == SET) { // 接收到数据 data = USART_ReceiveData(USART1); // 处理接收到的数据 } } } ``` 在轮询方式下,我们在一个死循环中不断检查串口接收标志位是否被置位,如果被置位则说明有数据到来,我们可以通过`USART_ReceiveData`函数获取接收到的数据,并进行处理。 ### 回答2: stm32串口接收数据包括以下步骤: 1. 设置串口参数:首先需要配置串口的波特率,数据位,停止位和奇偶校验位等参数。可以通过寄存器来设置串口参数,例如BRR寄存器设置波特率。 2. 使能串口接收功能:通过设置控制寄存器来使能串口的接收功能,例如使能接收数据的中断。 3. 中断处理函数:当有数据到达串口时,会触发接收数据中断。在中断处理函数中,可以读取接收数据寄存器来获取接收到的数据。 4. 数据处理:读取到的数据可以进行进一步的处理,例如判断数据是否满足特定条件,然后进行相关的操作。可以将数据存储到缓冲区中,以供后续处理使用。 5. 清除标志位:在读取完数据后,需要清除接收数据寄存器中的标志位,以便接收下一组数据。 6. 关闭串口接收功能:如果不再需要接收数据,可以通过清除控制寄存器中的接收使能位来关闭串口接收功能。 总结起来,stm32串口接收数据的流程包括设置串口参数、使能串口接收功能、中断处理函数、数据处理、清除标志位和关闭接收功能等步骤。通过这些步骤,可以实现stm32串口数据接收功能。 ### 回答3: 在STM32单片机中,串口接收数据的过程可以通过以下步骤实现: 1. 初始化串口:首先,需要对串口进行初始化设置,包括波特率、数据位、停止位和奇偶校验等参数的配置。可以使用STM32提供的库函数进行初始化。 2. 接收数据中断:在串口初始化之后,需要打开接收数据中断使能。当串口接收到数据时,会触发相应的中断服务程序。 3. 中断服务程序:当串口接收到数据时,会执行中断服务程序。在中断服务程序中,可以通过读取数据寄存器的值来获取接收到的数据。 4. 数据处理:接收到的数据可以根据需求进行进一步的处理。例如,可以解析数据包,提取有效数据,进行数据校验等。 5. 缓冲区管理:为了确保数据的完整性和可靠性,在接收数据过程中,可以创建一个接收缓冲区,将接收到的数据存储在缓冲区中。可以使用循环队列等数据结构来管理接收缓冲区,以便于数据的读取和写入。 6. 数据读取:一旦接收到的数据被处理完毕,可以从接收缓冲区中读取数据,并进行下一步的操作。 需要注意的是,串口接收数据的过程是异步的,即接收到的数据不一定是连续的。因此,在处理和读取数据时,应该考虑数据的完整性和顺序性,避免数据丢失或混淆。 以上是关于STM32串口接收数据的基本过程和注意事项的简要说明。具体的代码实现可以参考STM32的官方文档和相关教程。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值