现有串口实现变长数据接收的方式有两种方式:
- 配合UART的空闲中断(IDLE)实现
- 配合定时器或者systick实现
上述两种方式都需要配合中断函数或者其它外设实现,在一些对CPU的效率要求不高的情况下,使用上面两种方式实现方式比较麻烦。
为了追求简单,可以只使用HAL库函数重新封装一下轮询方式的接收函数,也能实现变长数据的接收。
代码
#define BUF_SIZE 200 //接收缓存区的大小 #define FRAME_BYTE_TIMEOUT 20 //在收到数据后,超过这个时间没有收到新的字节数据则认为一帧数据结束 /* 功能: 接收变长数据 参数: buf - 用于保存接收的数据 timeout - 等待接收到第一个字节数据的超时时间 返回值: 接收到数据的长度 */ uint16_t recv_undef_len_data(uint8_t *buf,uint32_t timeout) { uint16_t len=0; HAL_StatusTypeDef status; status = HAL_UART_Receive(&huart6,buf,1,timeout); if(status == HAL_OK){ //如果在timeout时间内收到数据,则确认一帧数据到来,接着接收后面的数据 len = 1; while(1){ status = HAL_UART_Receive(&huart6,buf+len,1,FRAME_BYTE_TIMEOUT); //每个字节接收时间不得超过FRAME_BYTE_TIMEOUT,否则认为一帧数据结束 if(status == HAL_OK){ len++; if(len == BUF_SIZE){ //接收的数据长度已达缓冲区最大值 return len; } }else if(status == HAL_TIMEOUT){ return len; } } }else{ return 0; } }
流程图
总结
使用上面的函数,不需要使能额外的中断函数和外设,就可以接收变长的数据,用法简单。已测试能够正常使用。测试环境是STM32与ESP-01S通信,能够很好的接收到ESP对AT指令的回复。
原创文章,作者:ItWorker,如若转载,请注明出处:https://blog.ytso.com/290662.html