Toggle navigation
首页
问答
文章
积分商城
专家
专区
更多专区...
文档中心
返回主站
搜索
提问
会员
中心
登录
注册
DMA
STM32H743
UART
10
stm32h7 串口 dma 发送长时间运行似乎会卡死
发布于 2025-04-28 09:02:26 浏览:54
订阅该版
最近在公司内部的一个使用了 stm32h743 的模块在长时间运行好几个小时过后,出现了奇怪的现象。原本正常定时外发协议数据的一个串口,突然不发送数据了。 结合调试打印,感觉似乎是发送线程在进入 `rt_device_write`的逻辑后, 就没有再次退出了。但 shell等其他正常的, shell 输入 ps 后的结果也挺奇怪的,居然有2个线程同时进入了 `RUNING` 的状态 (shell 和这个串口的发送线程),这显然不正常。请问大家觉得可能是什么原因? 谢谢! 异常情况下的 ps 结果:  与重新上电恢复后的正常情况对比:  部分业务逻辑展示: 为了便于使用,我们公司的前辈封装了一个 Serial 类,用于异步发送串口数据,因为部分逻辑涉及公司内部协议,这里只展示了部分串口发送相关的逻辑: ```CXX #include "rtthread.h" #include "rtdevice.h" #define LOG_TAG "app.serial" #define LOG_LVL LOG_LVL_DBG #include "ulog.h" struct serial_app_config { char usname[RT_NAME_MAX]; int baudrate; // ..... }; class Serial { public: Serial(struct serial_app_config serial_cfg); ~Serial(); bool StartTxProcess(); void TxDone(); private: char _us_name[RT_NAME_MAX]; rt_device_t dev_serial; rt_ringbuffer Serial_Tx_ringbuffer; rt_uint8_t *_serial_tx_bufTransfer; rt_uint8_t *_serial_tx_buf; rt_sem_t serial_tx_sem; rt_sem_t serial_txRing_sem; bool SendData(rt_uint8_t *data, int len); rt_mutex_t dynamic_mutex = RT_NULL; bool isread_exit; }; void Serial::TxDone() { log_d("Tx Done %s", _us_name); rt_sem_release(serial_tx_sem); } bool Serial::SendData(rt_uint8_t * data, int len) { bool result = false; if(rt_mutex_take(dynamic_mutex, 10) == RT_EOK) { int buff_left =rt_ringbuffer_space_len(&Serial_Tx_ringbuffer); if(buff_left == 0 || len > buff_left) { rt_ringbuffer_reset(&Serial_Tx_ringbuffer); buff_left =rt_ringbuffer_space_len(&Serial_Tx_ringbuffer); } if(len > buff_left) { len = buff_left; } if(len <= buff_left) { int putlen = rt_ringbuffer_put(&Serial_Tx_ringbuffer,data,len); if(putlen != len) { rt_ringbuffer_reset(&Serial_Tx_ringbuffer); } else { result = true; } } rt_mutex_release(dynamic_mutex); } if(result == true) { rt_sem_release(serial_txRing_sem); } return result; } bool Serial::StartTxProcess() { bool result = false; rt_uint16_t waitTime = 2000; while(!isread_exit) { rt_sem_take(serial_txRing_sem,waitTime); if(rt_sem_take(serial_tx_sem,500) == RT_EOK) { result = rt_mutex_take(dynamic_mutex, 10); if(result == RT_EOK) { int len = rt_ringbuffer_cpy(&Serial_Tx_ringbuffer, _serial_tx_buf, len); rt_mutex_release(dynamic_mutex); if(len > 0) { // 发送疑似进入后就没有再退出。。。 rt_uint16_t len_write = rt_device_write(dev_serial,0,_serial_tx_buf,len); if(len != len_write) { log_e("%s device write err:%d %d",_us_name,len,len_write); waitTime = 10; } else { waitTime = 2000; } if(rt_mutex_take(dynamic_mutex, 10) == RT_EOK) { rt_ringbuffer_del(&Serial_Tx_ringbuffer, len_write); rt_mutex_release(dynamic_mutex); } result = true; } else { rt_sem_release(serial_tx_sem); } } else { log_e("%s mutex err",_us_name); rt_sem_release(serial_tx_sem); } } else { log_e("not Tx Com Sem %s",_us_name); rt_ringbuffer_reset(&Serial_Tx_ringbuffer); } } return result; } static rt_err_t _tx_done(rt_device_t dev, void *buffer) { Serial *serial = (Serial *)dev->user_data; if (serial != RT_NULL) { serial->TxDone(); } return RT_EOK; } void _tx_thread(void *p) { Serial *serial = (Serial *)p; serial->StartTxProcess(); } Serial::Serial(struct serial_app_config serial_cfg) { rt_strncpy(_us_name, serial_cfg.usname, sizeof(_us_name)); dev_serial = rt_device_find(_us_name); if (dev_serial == RT_NULL) { log_e("Device %s not found!", _us_name); return; } // Initialize other members and resources // ... rt_device_set_tx_complete(dev_serial, _tx_done); rt_device_open(dev_serial, RT_DEVICE_OFLAG_RDWR | RT_DEVICE_FLAG_INT_RX | RT_DEVICE_FLAG_DMA_TX); // .... rt_thread_t th; th = rt_thread_create("tx_thread", _tx_thread, this, 1024, 25, 10); rt_thread_startup(th); } ```
查看更多
0
个回答
默认排序
按发布时间排序
暂无答案,快来添加答案吧
撰写答案
登录
注册新账号
关注者
0
被浏览
54
关于作者
a1012112796
这家伙很懒,什么也没写!
提问
30
回答
294
被采纳
35
关注TA
发私信
相关问题
1
rt thread 2.0.2 usart 接收缓存问题
2
关于STM32串口通信的问题
3
STM32F1+RTT串口接收终端数据丢失问题
4
UART TX丢数据?
5
RTT打开串口的时候如何自定义波特率呢?
6
STM32F4的USART数据接收问题
7
串口1234使用问题
8
串口接收回调函数
9
LPC18xx UART问题讨论
10
x1000串口配置的失败问题
推荐文章
1
RT-Thread应用项目汇总
2
玩转RT-Thread系列教程
3
国产MCU移植系列教程汇总,欢迎查看!
4
机器人操作系统 (ROS2) 和 RT-Thread 通信
5
五分钟玩转RT-Thread新社区
6
【技术三千问】之《玩转ART-Pi》,看这篇就够了!干货汇总
7
关于STM32H7开发板上使用SDIO接口驱动SD卡挂载文件系统的问题总结
8
STM32的“GPU”——DMA2D实例详解
9
RT-Thread隐藏的宝藏之completion
10
【ART-PI】RT-Thread 开启RTC 与 Alarm组件
最新文章
1
CherryUSB STM32 glue 新增初始化
2
在LPC1114上移植 RT-Thread Nano 3.1.5
3
求助can通讯下怎末运用485
4
STM32F407ZGT适配串口V2驱动
5
【CFP】2025 RT-Thread全球技术大会演讲征集开始啦!
热门标签
RT-Thread Studio
串口
Env
LWIP
SPI
AT
Bootloader
Hardfault
CAN总线
FinSH
ART-Pi
DMA
USB
文件系统
RT-Thread
SCons
RT-Thread Nano
线程
MQTT
STM32
RTC
rt-smart
FAL
I2C_IIC
cubemx
UART
ESP8266
WIZnet_W5500
BSP
ota在线升级
PWM
flash
packages_软件包
freemodbus
潘多拉开发板_Pandora
ADC
GD32
定时器
flashDB
编译报错
keil_MDK
socket
中断
rt_mq_消息队列_msg_queue
Debug
ulog
SFUD
msh
C++_cpp
at_device
本月问答贡献
聚散无由
9
个答案
6
次被采纳
RTT_逍遥
10
个答案
2
次被采纳
a1012112796
5
个答案
2
次被采纳
三世执戟
4
个答案
2
次被采纳
加缪
2
个答案
2
次被采纳
本月文章贡献
wake_mirco
2
篇文章
7
次点赞
mushroom
1
篇文章
9
次点赞
张世争
1
篇文章
7
次点赞
RTT_逍遥
1
篇文章
6
次点赞
sakumisu
1
篇文章
5
次点赞
回到
顶部
发布
问题
分享
好友
手机
浏览
扫码手机浏览
投诉
建议
回到
底部