Linux串口接收0x0D莫名转换为0x0A问题的根本原因分析

本文内容参考:

Linux串口接收字符莫名转换问题_linux 串口接收 将0x0d转换为0x0a-CSDN博客

特此致谢!

目录

一、问题现象

二、问题分析

[1. 排查问题](#1. 排查问题)

[2. 具体分析](#2. 具体分析)

三、问题解决

[1. 代码修改](#1. 代码修改)

[2. 运行结果](#2. 运行结果)


一、问题现象

笔者最近在实际项目调试时,发现了一个比较奇怪的现象。下位机(MCU)明明发送的数据是对的,上位机却总是不能校验通过,从而导致帧被丢弃。经过检查,大多数情况下是没问题的,但是当有0x55 0xAA 0x00 0x0D ......的时候,上位机(SoC)收到的却总是0x55 0xAA 0x00 0x0A ......,从而导致接收的帧校验总是不能通过。

二、问题分析

1. 排查问题

一开始笔者认为是下位机回复的本身就有问题,本身回复的就是0x0A。结果笔者在下位机代码中写死发送0x0D,经过检查,上位机Soc侧实际收到的确实就是0x0A。这就排除了下位机(MCU)代码的问题,将问题锁定在了上位机(SoC)本身、尤其是Linux串口自身机制上。

2. 具体分析

既然明确了是Linux串口自身的问题,而这个问题又是笔者之前从未遇到过的,那么最直接快捷的方法就是在百度上(现在也可以在豆包上)搜索,看看是否有前人遇到过此问题,又是怎样解决的。经过搜索,其实遇到此问题的人还是比较多的,其中选了一篇讲解得最为清楚的(就是本文开头致谢的那篇博文)。此问题的根本原因是:

串口配置时未禁用输入模式的字符转换标志位,导致系统驱动自动修改了接收数据。

具体来说是,串口的c_iflag(输入模式标志)中,默认可能开启了ICRNL标志(字符转换标志位) ------ 该标志会将接收的CR(0x0D)自动转换为NL(0x0A)。若代码中未明确清除这个标志位,将会导致数据被篡改:

1)发送数据包含0x0D → 串口驱动检测到CR;

2)因ICRNL开启,驱动自动将其替换为NL(0x0A);

3)最终应用层收到的就是0x0A而非原始的0x0D。

三、问题解决

1. 代码修改

既然明确了问题的根因,那么解决起来就比较容易了。在SoC侧Linux串口代码中做以下修改(添加红色矩形框中的一行):

实际上就是清除默认开启的ICRNL标志位。

2. 运行结果

作以上修改之后,再次编译、烧录(替换)、运行上位机可执行程序后,工作正常,问题解决。

相关推荐
UP_Continue2 小时前
Linux--动静态库
linux·运维·服务器
CheungChunChiu2 小时前
Linux 音频系统全景解析:PipeWire、PulseAudio 与 ALSA 的层次关系
linux·运维·服务器·audio
小张成长计划..2 小时前
【linux】5:编译器-gcc/g++的使用
linux
你真是饿了2 小时前
7.进程间通信
linux·运维·服务器
TEC_INO2 小时前
Linux_12:通过多线程获取VENC的H264码流数据
linux·运维·服务器
无证驾驶梁嗖嗖3 小时前
git_lab_事故恢复全过程(ubuntu_22
linux·git·ubuntu
小程同学>o<3 小时前
Linux 应用层开发入门(二十)| 获取输入系统设备数据
linux·嵌入式软件·嵌入式应用层·应用层开发·linux应用层开发
yuanmenghao4 小时前
Linux 性能实战 | 第 17 篇:strace 系统调用分析与性能调优 [特殊字符]
linux·python·性能优化
hweiyu004 小时前
Linux 命令:setfacl
linux·运维·服务器