处理串口接收,数据帧接一次性接收不完整问题

import time
import binascii
import utils


def data_recv(self):
    while (True):
        midd_data = []
        while True:
            rev_num = self.ser.inWaiting()
            recv_buffer = b''
            while rev_num:
                recv_buffer += self.ser.read(rev_num)
                time.sleep(0.1)
                rev_num = self.ser.inWaiting()
            if recv_buffer:
                str_buffer = ''
                for c in binascii.hexlify(recv_buffer):
                    str_buffer += chr(c)
                queue_pkt = utils.hexstr_to_int_list(str_buffer)
                if queue_pkt[0:2] == [0xcc, 0xcc] and queue_pkt[-1] == 0xff:
                    print("cctt recv: %s" % utils.int_list_to_hexstr(queue_pkt))
                    midd_data = []
                    self.data_que.put(queue_pkt)
                else:
                    midd_data.extend(queue_pkt)

                    def foo(inner_midd_data):
                        nonlocal midd_data
                        for index, item in enumerate(zip(inner_midd_data[:], inner_midd_data[1:])):
                            if item == (0xcc, 0xcc):
                                payload_len = inner_midd_data[index + 2]
                                fixed_len = 10
                                data_len = payload_len + fixed_len
                                inner_queue_pkt = inner_midd_data[index: index + data_len]
                                if len(inner_queue_pkt) == data_len:
                                    if inner_queue_pkt[-1] == 0xff:
                                        print("cctt recv: %s" % utils.int_list_to_hexstr(inner_queue_pkt))
                                        self.data_que.put(inner_queue_pkt)
                                        if len(inner_midd_data[index + data_len:]) > 0:
                                            foo(inner_midd_data[index + data_len:])
                                        break
                                else:
                                    midd_data = inner_queue_pkt

                    foo(midd_data)

数据帧格式

0xcc				# 1 byte 
0xcc				# 1 byte
payload_len			# payload 字节数	, 1 byte
中间略				# 	4 byte
payload				#  payload_len byte
crc校验				# 2 byte
0xff  				# 1 byte

代码解析

这里的完成性判断并没有做crc校验

对于前导数据和结束数据符合格式的,直接认为数据帧完整。

if queue_pkt[0:2] == [0xcc, 0xcc] and queue_pkt[-1] == 0xff:

对于前导数据和结束数据符不合格式的,把上帧数据多余部分与当前帧合并,遍历到指定前导符符合的数据位置,然后根据 payload长度+ 固定数据长度,获取当前数据帧数据,然后判断是否符合数据格式。这里主要是foo的递归使用。

midd_data.extend(queue_pkt)
def foo(inner_midd_data):

数据帧样例

数据帧1: CC CC 0B FD 07 00 01 00 29 01 01 00 91 11 00 00 03 21 92 9D FF CC CC 0B FD 07 00 

数据帧2: 01 00 29 01 02 00 91 11 00 00 04 14 10 AF FF CC CC 0B FD 07 00 01 00 29 01 03 00 91 11 00 00 03 32 52 89 FF
  • 1
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
在STM32串口接收数据错位的问题中,有几个可能的原因。首先,引用\[1\]中的代码片段显示了一个USART2_IRQHandler中断处理函数,该函数处理串口接收中断。在该函数中,如果接收到空闲中断(USART_IT_IDLE),则执行一系列操作,包括禁用DMA通道、清除标志位、获取剩余数据长度等。然后,根据接收到的数据进行判断,如果是有效的帧,则进行解析和存储,否则进行错误处理。这段代码中没有明显的错位问题。 然而,引用\[2\]中的代码片段显示了一个发送和接收数据问题。在这个问题中,发送数据正常,但接收数据出现错位。根据描述,当发送数据"12345678"时,第一次接收到的数据只有一个'1',再次发送相同的数据时,接收到的数据变成了"81234567"。这可能是由于接收缓冲区的长度与传输数据的长度不匹配导致的。在这种情况下,建议将接收缓冲区的长度设置为大于传输数据的长度,以确保接收完整数据。 另外,引用\[3\]中提到了使用DMA进行通信时出现数据错位的问题。在这个问题中,发送端和接收端都使用了DMA进行数据传输。在之前的情况下,通过延时重置DMA来解决了错位问题。然而,在这次的情况下,采用相同的方法没有解决问题。这可能是由于其他因素导致的,比如中断处理函数中的代码执行时间过长。为了解决这个问题,可以尝试优化中断处理函数的代码,减少占用的空间时间。 综上所述,解决STM32串口接收数据错位的问题可以考虑以下几个方面: 1. 确保接收缓冲区的长度大于传输数据的长度,以避免数据截断或错位。 2. 优化中断处理函数的代码,减少占用的空间时间,以确保数据的正确接收处理。 3. 检查其他可能的因素,如硬件连问题或其他软件配置问题,以确保数据传输的稳定性和准确性。 #### 引用[.reference_title] - *1* [【STM32】DMA+串口空闲中断接收定长数据(解决接收错位问题)](https://blog.csdn.net/Corner_L/article/details/105312646)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v91^insertT0,239^v3^insert_chatgpt"}} ] [.reference_item] - *2* [stm32f4 串口空闲中断+DMA遇到的奇怪错位问题](https://blog.csdn.net/wxc971231/article/details/88387124)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v91^insertT0,239^v3^insert_chatgpt"}} ] [.reference_item] - *3* [STM32串口DMA接收数据错位——暴力解决方法](https://blog.csdn.net/qq_40464014/article/details/105590051)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v91^insertT0,239^v3^insert_chatgpt"}} ] [.reference_item] [ .reference_list ]

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值