Toggle navigation
首页
问答
文章
积分商城
专家
专区
更多专区...
文档中心
返回主站
搜索
提问
会员
中心
登录
注册
DMA
RoboMaster
串口
基于 RT-Thread 的 RoboMaster 电控框架(五)
发布于 2023-10-28 11:14:24 浏览:428
订阅该版
[tocm] # 基于 RT-Thread 的 RoboMaster 电控框架(五) *由于 RT-Thread 稳定高效的内核,丰富的文档教程,积极活跃的社区氛围,以及设备驱动框架、Kconfig、Scons、日志系统、海量的软件包......很难不选择 RT-Thread 进行项目开发。但也正是因为这些优点的覆盖面较广,很多初学者会觉得无从下手,但只要步入 RT-Thread 的大门,你就发现她的美好。这系列文档将作为本人基于 RT-Thread 开发 RoboMaster 电控框架的记录与分享,希望能帮助到更多初识 RT-Thread 的小伙伴,也欢迎大家交流分享,指正不足,共同进步。* ## 背景 使用的开发板为大疆的 RoboMaster-C 型开发板,基础工程为 rt-thread>bsp>stm32f407-robomaster-c ## 遥控器模块开发 在 C 板上是提供了针对大疆遥控器的 DBUS 接口,但本片文章是基于 SBUS 进行遥控。 - **DBUS**:100k波特率,8位数据位,1位停止位,偶校验(EVEN),无控流,18个字节; - **SBUS**:100k波特率,8位数据位,2位停止位,偶校验(EVEN),无控流,25个字节。 SBUS 和 DBUS 主要区别就是停止位不同,两者都需要硬件取反电路,因此 SBUS 的接收机也是可以直接插在 C 板提供的 DBUS 接口上进行使用的,只需要在软件层面修改数据解析处理即可。 ### 串口DMA双缓冲 这里使用的是空闲中断 + DMA双缓冲的方案,改方案能够极大限度的提高处理高速数据的效率和稳定性。但STM32不是所有芯片都支持DMA双缓冲,虽然也可以通过DMA半满中断实现双缓冲的效果,但是这样程序的兼容性是较差的;因此针对遥控器接收机的串口,选择不使用 RT-Thread 的串口驱动框架,也不是对其驱动框架进行改动。而是使用 HAL 库实现,但不会影响其他串口使用 RT-Thread 的串口驱动框架。 ### 代码实现 首先是串口和 DMA 的初始化: ```c /* DMA controller clock enable */ __HAL_RCC_DMA1_CLK_ENABLE(); /* DMA1_Stream1_IRQn interrupt configuration */ HAL_NVIC_SetPriority(DMA1_Stream1_IRQn, 0, 0); HAL_NVIC_EnableIRQ(DMA1_Stream1_IRQn); huart3.Instance = USART3; huart3.Init.BaudRate = 100000; huart3.Init.WordLength = UART_WORDLENGTH_9B; huart3.Init.StopBits = UART_STOPBITS_2; huart3.Init.Parity = UART_PARITY_EVEN; huart3.Init.Mode = UART_MODE_TX_RX; huart3.Init.HwFlowCtl = UART_HWCONTROL_NONE; huart3.Init.OverSampling = UART_OVERSAMPLING_16; HAL_UART_Init(&huart3); ``` 以及 DMA 双缓冲功能的配置: ```c /** * @brief 串口 DMA 双缓冲初始化 * @param rx1_buf 缓冲区1 * @param rx2_buf 缓冲区2 * @param dma_buf_num DMA缓冲区大小 */ static void rc_doub_dma_init(uint8_t *rx1_buf, uint8_t *rx2_buf, uint16_t dma_buf_num) { //使能DMA串口接收 SET_BIT(huart3.Instance->CR3, USART_CR3_DMAR); //使能空闲中断 __HAL_UART_ENABLE_IT(&huart3, UART_IT_IDLE); //失效DMA __HAL_DMA_DISABLE(&hdma_usart3_rx); while(hdma_usart3_rx.Instance->CR & DMA_SxCR_EN) { __HAL_DMA_DISABLE(&hdma_usart3_rx); } hdma_usart3_rx.Instance->PAR = (uint32_t) & (USART3->DR); //内存缓冲区1 hdma_usart3_rx.Instance->M0AR = (uint32_t)(rx1_buf); //内存缓冲区2 hdma_usart3_rx.Instance->M1AR = (uint32_t)(rx2_buf); //数据长度 hdma_usart3_rx.Instance->NDTR = dma_buf_num; //使能双缓冲区 SET_BIT(hdma_usart3_rx.Instance->CR, DMA_SxCR_DBM); //使能DMA __HAL_DMA_ENABLE(&hdma_usart3_rx); } ``` 以及 CubeMX 的一些基本配置,这里就不细说了,设置完这些,串口空闲中断 + DMA双缓冲就开起来了,接下来就是要到串口中断处理函数里进行 DMA 双缓冲的接收和数据的解析处理了: ```c void USART3_IRQHandler(void) { if(huart3.Instance->SR & UART_FLAG_RXNE) { __HAL_UART_CLEAR_PEFLAG(&huart3); } else if(USART3->SR & UART_FLAG_IDLE) { static uint16_t this_time_rx_len = 0; __HAL_UART_CLEAR_PEFLAG(&huart3); if ((hdma_usart3_rx.Instance->CR & DMA_SxCR_CT) == RESET) { /* Current memory buffer used is Memory 0 */ //失效DMA __HAL_DMA_DISABLE(&hdma_usart3_rx); //get receive data length, length = set_data_length - remain_length //获取接收数据长度,长度 = 设定长度 - 剩余长度 this_time_rx_len = SBUS_RX_BUF_NUM - hdma_usart3_rx.Instance->NDTR; //重新设定数据长度 hdma_usart3_rx.Instance->NDTR = SBUS_RX_BUF_NUM; //设定缓冲区1 hdma_usart3_rx.Instance->CR |= DMA_SxCR_CT; //使能DMA __HAL_DMA_ENABLE(&hdma_usart3_rx); if(this_time_rx_len == SBUS_FRAME_SIZE) { //处理遥控器数据 sbus_rc_decode(sbus_rx_buf[0]); rt_timer_start(rc_timer); } } else { /* Current memory buffer used is Memory 1 */ //失效DMA __HAL_DMA_DISABLE(&hdma_usart3_rx); //get receive data length, length = set_data_length - remain_length //获取接收数据长度,长度 = 设定长度 - 剩余长度 this_time_rx_len = SBUS_RX_BUF_NUM - hdma_usart3_rx.Instance->NDTR; //重新设定数据长度 hdma_usart3_rx.Instance->NDTR = SBUS_RX_BUF_NUM; //设定缓冲区0 DMA1_Stream1->CR &= ~(DMA_SxCR_CT); //使能DMA __HAL_DMA_ENABLE(&hdma_usart3_rx); if(this_time_rx_len == SBUS_FRAME_SIZE) { //处理遥控器数据 sbus_rc_decode(sbus_rx_buf[1]); rt_timer_start(rc_timer); } } } } ``` 到这一步,已经可以顺利的接收并解析处理 SBUS 遥控数据了。通过空闲中断我们可以确保完整的接收数据帧,而且使用 DMA 双缓冲以后,相较于普通的 DMA 接收处理高速数据更加高效快速,在处理一个缓冲区的数据之前,先将 DMA 切换到另外一个缓冲区,这样在处理数据的时候就不会影响到 DMA 数据的接收,而且针对遥控器这种实时要求高且解析简单的数据,就可以在中断处理函数中 DMA 缓冲区切换后直接进行解析处理。STM32F4 系列是支持 DMA 双缓冲功能的,但是对于其他一些不支持双缓冲的芯片,也想要使用 pingpong 缓冲的话,就可以通过 DMA 半满中断实现。 ### 抽象设备 这里将遥控器数据就简单的抽象为遥控器设备: ```c typedef struct { int16_t ch1; //右侧左右 int16_t ch2; //右侧上下 int16_t ch3; //左侧上下 int16_t ch4; //左侧左右 int16_t ch5; //左侧非线性旋钮 int16_t ch6; //右侧非线性旋钮 uint8_t sw1; //右侧长拨杆 uint8_t sw2; //左侧长拨杆 uint8_t sw3; //右侧短拨杆 uint8_t sw4; //左侧短拨杆 } rc_obj_t; ``` 接收到数据存储在 rc_sbus.c 的 rc_data[2] 中: ```c rc_obj_t rc_data[2]; // [0]:当前数据NOW,[1]:上一次的数据LAST ``` 通过调用 sbus_rc_init() 即可获得遥控器数据的地址,使用示例如下: ```c rc_obj_t rc_data[2]; // [0]:当前数据NOW,[1]:上一次的数据LAST rc_data = sbus_rc_init(); ```
0
条评论
默认排序
按发布时间排序
登录
注册新账号
关于作者
螺丝松掉的人
这家伙很懒,什么也没写!
文章
42
回答
0
被采纳
0
关注TA
发私信
相关文章
1
串口DMA发送数据时,数据被覆盖
2
关于串口DMA模式下rt_device_close问题
3
利用stm32f427实现usb转串口,电脑端什么也没有识别到
4
finsh 控制台 适配 RS 485请大神指点????
5
uart_sample.c 中,读串口设备时偏移量pos要设置为-1而不是0?
6
【结贴】at_device软件包中对串口接收数据缺少判断导致数据接收异常
7
串口无法接受数据,但可以发送
8
串口如何有效的清除掉接收缓冲,而不必一个一个的去读取
9
串口接收使用方式问题
10
雅特力FINSH问题
推荐文章
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
USB
DMA
文件系统
RT-Thread
SCons
RT-Thread Nano
线程
MQTT
STM32
RTC
FAL
rt-smart
ESP8266
I2C_IIC
WIZnet_W5500
UART
ota在线升级
PWM
cubemx
freemodbus
flash
packages_软件包
BSP
潘多拉开发板_Pandora
定时器
ADC
GD32
flashDB
socket
中断
Debug
编译报错
SFUD
msh
rt_mq_消息队列_msg_queue
keil_MDK
ulog
MicroPython
C++_cpp
本月问答贡献
出出啊
1517
个答案
342
次被采纳
小小李sunny
1443
个答案
289
次被采纳
张世争
805
个答案
174
次被采纳
crystal266
547
个答案
161
次被采纳
whj467467222
1222
个答案
148
次被采纳
本月文章贡献
出出啊
1
篇文章
4
次点赞
小小李sunny
1
篇文章
1
次点赞
张世争
1
篇文章
1
次点赞
crystal266
2
篇文章
2
次点赞
whj467467222
2
篇文章
1
次点赞
回到
顶部
发布
问题
投诉
建议
回到
底部