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

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

相关推荐
0xDevNull21 小时前
Linux切换JDK版本详细教程
linux
进击的丸子21 小时前
虹软人脸服务器版SDK(Linux/ARM Pro)多线程调用及性能优化
linux·数据库·后端
Johny_Zhao2 天前
OpenClaw安装部署教程
linux·人工智能·ai·云计算·系统运维·openclaw
chlk1234 天前
Linux文件权限完全图解:读懂 ls -l 和 chmod 755 背后的秘密
linux·操作系统
舒一笑4 天前
Ubuntu系统安装CodeX出现问题
linux·后端
改一下配置文件4 天前
Ubuntu24.04安装NVIDIA驱动完整指南(含Secure Boot解决方案)
linux
深紫色的三北六号4 天前
Linux 服务器磁盘扩容与目录迁移:rsync + bind mount 实现服务无感迁移(无需修改配置)
linux·扩容·服务迁移
SudosuBash4 天前
[CS:APP 3e] 关于对 第 12 章 读/写者的一点思考和题解 (作业 12.19,12.20,12.21)
linux·并发·操作系统(os)
哈基咪怎么可能是AI5 天前
为什么我就想要「线性历史 + Signed Commits」GitHub 却把我当猴耍 🤬🎙️
linux·github
十日十行6 天前
Linux和window共享文件夹
linux