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. 运行结果

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

相关推荐
意疏8 小时前
【Linux 篇】Docker 容器星河与镜像灯塔:Linux 系统下解锁应用部署奇幻征程
linux·docker
朱包林8 小时前
k8s-Pod基础管理,标签管理,rc控制器及重启策略实战
linux·运维·云原生·容器·kubernetes·云计算
勇闯逆流河8 小时前
【Linux】linux进程概念(环境变量详解)
linux·运维·服务器
_evenif9 小时前
MySql 8 一机多实例部署
linux·数据库·mysql
程序员差不多先生9 小时前
小米Openvela生态成熟度如何?
linux
Yupureki9 小时前
《Linux系统编程》12.基础IO
linux·运维·c语言·开发语言·数据库·c++
淮北4949 小时前
bash下好用的快捷键以及linux常用指令
linux·开发语言·ubuntu·bash
顶点多余9 小时前
深度剖析Linux 线程概念
java·linux·jvm
小马学嵌入式~10 小时前
linux开发深度学习-时钟
linux·arm开发·嵌入式硬件·学习
青瓦梦滋10 小时前
Linux进程间通信(IPC)——system V
linux·服务器·c++·文件