Toggle navigation
首页
问答
文章
积分商城
专家
专区
更多专区...
文档中心
返回主站
搜索
提问
会员
中心
登录
注册
DMA
STM32
UART串口空闲中断
rt-thread+stm32 使用串口dma空闲中断接收不定长数据
发布于 2021-04-20 12:57:40 浏览:2597
订阅该版
[tocm] # 开发环境 IDE: RT-Thread Studio v2.1.0 系统版本:v4.0.2 芯片: STM32F407VG # 一、工程配置 工程创建完成后,双击工程目录下的 RT-Thread Setting ![在这里插入图片描述](https://oss-club.rt-thread.org/uploads/20220714/a5fc28923d2f2c80f02254efb0f61d13be79e4b9.png?x-oss-process=image/watermark,type_ZmFuZ3poZW5naGVpdGk,shadow_10,text_aHR0cHM6Ly9ibG9nLmNzZG4ubmV0L2d5endfbXg=,size_16,color_FFFFFF,t_70#pic_center) 选择右下角的更多配置 ![在这里插入图片描述](https://oss-club.rt-thread.org/uploads/20220714/0ad223cd1bc037651d3518ed2d04ad85f37bf139.png?x-oss-process=image/watermark,type_ZmFuZ3poZW5naGVpdGk,shadow_10,text_aHR0cHM6Ly9ibG9nLmNzZG4ubmV0L2d5endfbXg=,size_16,color_FFFFFF,t_70#pic_center) 勾选使能串口DMA模式,设置缓冲区大小,ctrl+s保存配置后自动生成代码,然后退出配置页面。 ![在这里插入图片描述](https://oss-club.rt-thread.org/uploads/20220714/cab46a810b8726909819dc85e803279de967dca4.png?x-oss-process=image/watermark,type_ZmFuZ3poZW5naGVpdGk,shadow_10,text_aHR0cHM6Ly9ibG9nLmNzZG4ubmV0L2d5endfbXg=,size_16,color_FFFFFF,t_70#pic_center) 打开 drivers/board.h ,按图示添加代码,引脚配置根据图示自行修改。 # 二、添加代码并测试 新建一对 uartdma.c && uartdma.h文件,在 uartdma.h中添加 Uart类定义,并声明UART1 和UART2对象。 ``` #include "board.h" typedef struct Uart { rt_device_t serial; rt_mailbox_t mb; rt_size_t (*send)(char *, rt_size_t); rt_size_t (*recv)(char *, rt_int32_t); rt_err_t (*input)(rt_device_t, rt_size_t); int (*init)(uint32_t); } Uart; #ifdef BSP_UART1_RX_USING_DMA extern Uart UART1; #endif #ifdef BSP_UART2_RX_USING_DMA extern Uart UART2; #endif ``` 在 uartdma.c 中添加代码 ``` #include "rtthread.h" #include "uartdma.h" #define RT_SERIAL_CONFIG_DEFAULT \ { \ 115200, /* 115200 bits/s */ \ 8, /* 8 databits */ \ 0, /* 1 stopbit */ \ 0, /* No parity */ \ 0, /* LSB first sent */ \ 0, /* Normal mode */ \ 256, /* Buffer size */ \ 0 \ } struct serial_configure { rt_uint32_t baud_rate; rt_uint32_t data_bits :4; rt_uint32_t stop_bits :2; rt_uint32_t parity :2; rt_uint32_t bit_order :1; rt_uint32_t invert :1; rt_uint32_t bufsz :16; rt_uint32_t reserved :6; }; #ifdef BSP_UART1_RX_USING_DMA #define UART1_NAME "uart1" /* 串口设备名称 */ static rt_size_t uart1_send(char *data, rt_size_t size); static rt_size_t uart1_recv(char *buffer, rt_int32_t timeout); static rt_err_t uart1_input(rt_device_t dev, rt_size_t size); static int uart1_init(uint32_t baud_rate); Uart UART1 = { RT_NULL, RT_NULL, uart1_send, uart1_recv, uart1_input, uart1_init }; static rt_size_t uart1_send(char *data, rt_size_t size) { return rt_device_write(UART1.serial, 0, data, size); } static rt_size_t uart1_recv(char *buffer, rt_int32_t timeout) { rt_size_t len; if (rt_mb_recv(UART1.mb, &len, timeout) != RT_EOK) { return 0; } len = rt_device_read(UART1.serial, 0, buffer, len); return len; } /* 接收数据回调函数 */ static rt_err_t uart1_input(rt_device_t dev, rt_size_t size) { /* 发送邮件 */ return rt_mb_send(UART1.mb, size); } static int uart1_init(uint32_t baud_rate) { struct serial_configure config = RT_SERIAL_CONFIG_DEFAULT; /* 初始化配置参数 */ UART1.serial = rt_device_find(UART1_NAME); if (!UART1.serial) { rt_kprintf("find %s failed!\n", UART1_NAME); return RT_ERROR; } if (UART1.mb == RT_NULL) { UART1.mb = rt_mb_create("uart1_mb", 1, RT_IPC_FLAG_FIFO); if (UART1.mb == RT_NULL) { return RT_ERROR; } } /* step2:修改串口配置参数 */ config.baud_rate = baud_rate; //修改波特率 /* step3:控制串口设备。通过控制接口传入命令控制字,与控制参数 */ rt_device_control(UART1.serial, RT_DEVICE_CTRL_CONFIG, &config); rt_device_set_rx_indicate(UART1.serial, UART1.input); rt_device_open(UART1.serial, RT_DEVICE_FLAG_DMA_RX); return RT_EOK; } #endif #ifdef BSP_UART2_RX_USING_DMA #define UART2_NAME "uart2" /* 串口设备名称 */ static rt_size_t uart2_send(char *data, rt_size_t size); static rt_size_t uart2_recv(char *buffer, rt_int32_t timeout); static rt_err_t uart2_input(rt_device_t dev, rt_size_t size); static int uart2_init(uint32_t baud_rate); Uart UART2 = { RT_NULL, RT_NULL, uart2_send, uart2_recv, uart2_input, uart2_init }; static rt_size_t uart2_send(char *data, rt_size_t size) { return rt_device_write(UART2.serial, 0, data, size); } static rt_size_t uart2_recv(char *buffer, rt_int32_t timeout) { rt_size_t len; if (rt_mb_recv(UART2.mb, &len, timeout) != RT_EOK) { return 0; } len = rt_device_read(UART2.serial, 0, buffer, len); return len; } /* 接收数据回调函数 */ static rt_err_t uart2_input(rt_device_t dev, rt_size_t size) { /* 发送邮件 */ return rt_mb_send(UART2.mb, size); } static int uart2_init(uint32_t baud_rate) { struct serial_configure config = RT_SERIAL_CONFIG_DEFAULT; /* 初始化配置参数 */ UART2.serial = rt_device_find(UART2_NAME); if (!UART2.serial) { rt_kprintf("find %s failed!\n", UART2_NAME); return RT_ERROR; } if (UART2.mb == RT_NULL) { UART2.mb = rt_mb_create("uart2_mb", 1, RT_IPC_FLAG_FIFO); if (UART2.mb == RT_NULL) { return RT_ERROR; } } /* step2:修改串口配置参数 */ config.baud_rate = baud_rate; //修改波特率 /* step3:控制串口设备。通过控制接口传入命令控制字,与控制参数 */ rt_device_control(UART2.serial, RT_DEVICE_CTRL_CONFIG, &config); rt_device_set_rx_indicate(UART2.serial, UART2.input); rt_device_open(UART2.serial, RT_DEVICE_FLAG_DMA_RX); return RT_EOK; } #endif int uart_init(void) { #ifdef BSP_UART1_RX_USING_DMA if (UART1.init(115200) == RT_EOK) { rt_kprintf("uart1 dma init successful!\r\n"); } #endif #ifdef BSP_UART2_RX_USING_DMA if (UART2.init(115200) == RT_EOK) { rt_kprintf("uart2 dma init successful!\r\n"); } #endif return RT_EOK; } INIT_ENV_EXPORT(uart_init); /* 使用组件自动初始化机制 */ ``` 编译下载运行 ![在这里插入图片描述](https://oss-club.rt-thread.org/uploads/20220714/43aebc001ace0a027109c91d219afb05a90c9acb.png#pic_center) 控制台打印出 串口 dma 初始化成功信息。 之后可以在应用层线程引入 uartdma.h ,调用 UART1.recv 和 UART1.recv,或者 UART2.send 和 UART2.recv进行收发数据。 但是在使用过程中,出现接收数据断帧问题。 ![在这里插入图片描述](https://oss-club.rt-thread.org/uploads/20220714/6891d83a4bfdd11219f7e52e42d5a83b80995960.png?x-oss-process=image/watermark,type_ZmFuZ3poZW5naGVpdGk,shadow_10,text_aHR0cHM6Ly9ibG9nLmNzZG4ubmV0L2d5endfbXg=,size_16,color_FFFFFF,t_70#pic_center) 接收数据有大概10%的概率出现错误。 # 三、解决问题 参考博文 https://blog.csdn.net/coderdd/article/details/108264369 ![在这里插入图片描述](https://oss-club.rt-thread.org/uploads/20220714/727bcb883db5d79454547020a8fdbf2744ea6944.png?x-oss-process=image/watermark,type_ZmFuZ3poZW5naGVpdGk,shadow_10,text_aHR0cHM6Ly9ibG9nLmNzZG4ubmV0L2d5endfbXg=,size_16,color_FFFFFF,t_70#pic_center) 根据上述博文连接描述进行代码修改。 ![在这里插入图片描述](https://oss-club.rt-thread.org/uploads/20220714/e6ff3514587afc39220b849124129c9ff492de6e.png?x-oss-process=image/watermark,type_ZmFuZ3poZW5naGVpdGk,shadow_10,text_aHR0cHM6Ly9ibG9nLmNzZG4ubmV0L2d5endfbXg=,size_16,color_FFFFFF,t_70#pic_center) 修改完成后重新测试,以200ms一帧连续测试30分钟,未出现断帧问题。
0
条评论
默认排序
按发布时间排序
登录
注册新账号
关于作者
gyzw
这家伙很懒,什么也没写!
文章
3
回答
0
被采纳
0
关注TA
发私信
相关文章
1
串口DMA发送数据时,数据被覆盖
2
关于串口DMA模式下rt_device_close问题
3
stm32L4 lpuart1DMA下不能打开的问题
4
UART驱动开启DMA后编译报错,UART DMA配置中加入了没有定义的域
5
UART DMA 设计问题
6
UART DMA 拆包问题
7
串口如何有效的清除掉接收缓冲,而不必一个一个的去读取
8
串口接收使用方式问题
9
在studio中打开串口的DMA后,编译出错
10
官方DMA串口例程,使用时数据打印错误
推荐文章
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组件
热门标签
RT-Thread Studio
串口
Env
LWIP
SPI
AT
Bootloader
Hardfault
CAN总线
FinSH
ART-Pi
DMA
USB
文件系统
RT-Thread
SCons
RT-Thread Nano
线程
MQTT
STM32
RTC
FAL
rt-smart
I2C_IIC
ESP8266
UART
cubemx
WIZnet_W5500
ota在线升级
PWM
BSP
flash
freemodbus
packages_软件包
潘多拉开发板_Pandora
定时器
ADC
GD32
flashDB
socket
编译报错
中断
Debug
rt_mq_消息队列_msg_queue
keil_MDK
ulog
SFUD
msh
C++_cpp
MicroPython
本月问答贡献
RTT_逍遥
7
个答案
2
次被采纳
三世执戟
7
个答案
1
次被采纳
KunYi
6
个答案
1
次被采纳
winfeng
2
个答案
1
次被采纳
chenyaxing
2
个答案
1
次被采纳
本月文章贡献
出出啊
1
篇文章
4
次点赞
小小李sunny
1
篇文章
1
次点赞
张世争
1
篇文章
3
次点赞
crystal266
2
篇文章
2
次点赞
whj467467222
2
篇文章
2
次点赞
回到
顶部
发布
问题
投诉
建议
回到
底部