本文内容参考:
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. 运行结果
作以上修改之后,再次编译、烧录(替换)、运行上位机可执行程序后,工作正常,问题解决。