stm32开发之串口空闲中断和环形数组的最简单的组合使用

前言

  1. 本次使用的是lwrb开源的源码;
  2. 测试环境使用的是stm32f407zgt6
  3. 这里不介绍lwrb的内容,如有需要请自行去查阅.
  4. 这里会使用到rt_container_of的宏定义(相关介绍请参考rt_thread或linux源码相关的宏定义,其表达的内容是一致的)
  5. 这里使用的是threadx做为os
  6. 本次驱动存在一个数据拷贝的问题(待优化,最简单的方式就是重写HAL库自定义的中断处理函数)
  7. 本次只涉及到中断的方式,dma的话对应的回调函数都是一致的。

串口驱动编写

头文件

/**** Copyright (c) 2024-2024,KHYX******************************************************************************* @file           : bsp_uart.h* @author         : shchl* @brief          : None* @attention      : None* @crete date     : 24-9-17******************************************************************************* Change Logs:* Date           Author           Notes* 24-9-17       shchl      */#ifndef BSP_UART_H
#define BSP_UART_H
#define UART1_RX_BUF_SIZE 1*1024
#define UART1_FRAME_MAX_SIZE 256
#include <lwrb/lwrb.h>struct stm32_uart_device {UART_HandleTypeDef handle; // 串口句柄struct lwrb rx_lwrb; // 接收环形数组// 主要是给HAL串口接收使用的缓存uint8_t *cache_buf; // 缓存uint16_t cache_len; // 缓存长度void (*tx_complete_cb)(void *); // 发送完成回调
};
/*** @brief  串口初始化* @retval None*/
int bsp_uart_init(void);
/*** @brief  串口接收数据* @param  uart: 串口设备* @param  pdata: 接收数据指针* @param  len: 接收数据长度* @retval None*/
int uart_receive(const USART_TypeDef *uart, void *pdata, int len);
/*** @brief  串口发送数据(阻塞)* @param  uart: 串口设备* @param  pdata: 发送数据指针* @param  len: 发送数据长度* @retval None*/
int uart_send_block(const USART_TypeDef *uart, const void *pdata, int len);/*** @brief  串口发送数据(非阻塞)* @param  uart: 串口设备* @param  pdata: 发送数据指针* @param  len: 发送数据长度* @param  tx_complete_cb: 发送完成回调函数* @retval None*/
int uart_send_async(const USART_TypeDef *uart, const void *pdata, int len, void (*tx_complete_cb)(void *));#endif //BSP_UART_H

源文件 (这里只实现串口1)

/**** Copyright (c) 2024-2024,KHYX******************************************************************************* @file           : bsp_uart.c* @author         : shchl* @brief          : None* @attention      : None* @crete date     : 24-9-17******************************************************************************* Change Logs:* Date           Author           Notes* 24-9-17       shchl      */#include "bsp.h"
#define khyx_container_of(ptr, type, member) \(type *)((char *)(ptr) - offsetof(type, member))
enum {UART1_IDX = 0,UART_MAX_NUMBER
} UART_IDX;static uint8_t uart1_rx_buf[UART1_RX_BUF_SIZE];
static struct stm32_uart_device *uart_map[UART_MAX_NUMBER];int bsp_uart_init(void) {// 串口1初始化{static struct stm32_uart_device uart1 = {0};uart1.handle.Instance = USART1;uart1.handle.Init.BaudRate = 115200;uart1.handle.Init.WordLength = UART_WORDLENGTH_8B;uart1.handle.Init.StopBits = UART_STOPBITS_1;uart1.handle.Init.Parity = UART_PARITY_NONE;uart1.handle.Init.Mode = UART_MODE_TX_RX;uart1.handle.Init.HwFlowCtl = UART_HWCONTROL_NONE;// 初始化串口HAL_UART_Init(&uart1.handle);// 初始化缓存lwrb_init(&uart1.rx_lwrb, uart1_rx_buf, UART1_RX_BUF_SIZE);// 串口1 数据帧最大长度static uint8_t uart1_cache_buf[UART1_FRAME_MAX_SIZE];uart1.cache_buf = uart1_cache_buf;uart1.cache_len = sizeof(uart1_cache_buf);// 开启串口空闲中断HAL_UARTEx_ReceiveToIdle_IT(&uart1.handle, uart1.cache_buf, uart1.cache_len);// 添加uart_map[UART1_IDX] = &uart1;}return 0;
}struct stm32_uart_device *usart_to_dev(const USART_TypeDef *uart) {if (uart == USART1) {return uart_map[UART1_IDX];}return NULL;
}/*** @brief  串口接收数据* @param  uart: 串口设备* @param  pdata: 接收数据指针* @param  len: 接收数据长度* @retval 接收数据长度,0 表示没有数据*/
int uart_receive(const USART_TypeDef *uart, void *pdata, int len) {struct stm32_uart_device *dev = usart_to_dev(uart);return dev ? lwrb_read(&dev->rx_lwrb, pdata, len) : 0;
}/*** @brief  串口发送数据* @param  uart: 串口设备* @param  pdata: 发送数据指针* @param  len: 发送数据长度* @retval HAL_OK 表示发送成功*/
int uart_send_block(const USART_TypeDef *uart, const void *pdata, const int len) {struct stm32_uart_device *dev = usart_to_dev(uart);return dev ? HAL_UART_Transmit(&dev->handle, pdata, len, 100) : HAL_OK;
}int uart_send_async(const USART_TypeDef *uart, const void *pdata, const int len, void (*tx_complete_cb)(void *)) {struct stm32_uart_device *dev = usart_to_dev(uart);if (dev == NULL) return HAL_ERROR;dev->tx_complete_cb = tx_complete_cb;return HAL_UART_Transmit_IT(&dev->handle, pdata, len);
}// 串口1中断处理函数
void USART1_IRQHandler(void) {HAL_UART_IRQHandler(&uart_map[UART1_IDX]->handle);
}void HAL_UART_RxCpltCallback(UART_HandleTypeDef *huart) {struct stm32_uart_device *uart = khyx_container_of(huart, struct stm32_uart_device, handle);// 串口1接收完成// 将缓存中的数据写入接收缓存lwrb_write(&uart->rx_lwrb, uart->cache_buf, uart->cache_len);// 重新开启接收中断HAL_UARTEx_ReceiveToIdle_IT(&uart->handle, uart->cache_buf, uart->cache_len);
}// 串口1 传输完成
void HAL_UART_TxCpltCallback(UART_HandleTypeDef *huart) {struct stm32_uart_device *uart = khyx_container_of(huart, struct stm32_uart_device, handle);// 执行回调(如果有)if (uart->tx_complete_cb) uart->tx_complete_cb(uart);
}void HAL_UARTEx_RxEventCallback(UART_HandleTypeDef *huart, uint16_t Size) {// 串口接收到数据struct stm32_uart_device *uart = khyx_container_of(huart, struct stm32_uart_device, handle);// 将缓存中的数据写入接收缓存(todo 这里可以进行优化,优化的方式根据共享数据的方式,只更新索引位置)lwrb_write(&uart->rx_lwrb, uart->cache_buf, Size);// 重新开启接收中断HAL_UARTEx_ReceiveToIdle_IT(&uart->handle, uart->cache_buf, uart->cache_len);
}// 串口接收错误
void HAL_UART_ErrorCallback(const UART_HandleTypeDef *huart) {// 串口1接收错误struct stm32_uart_device *uart = khyx_container_of(huart, struct stm32_uart_device, handle);// 重新开启接收中断HAL_UARTEx_ReceiveToIdle_IT(&uart->handle, uart->cache_buf, uart->cache_len);
}

引脚初始化

void HAL_UART_MspInit(UART_HandleTypeDef *huart) {GPIO_InitTypeDef GPIO_InitStruct = {0};if (huart->Instance == USART1) {__HAL_RCC_USART1_CLK_ENABLE();__HAL_RCC_GPIOA_CLK_ENABLE();GPIO_InitStruct.Pin = GPIO_PIN_9 | GPIO_PIN_10;GPIO_InitStruct.Mode = GPIO_MODE_AF_PP;GPIO_InitStruct.Pull = GPIO_NOPULL;GPIO_InitStruct.Speed = GPIO_SPEED_FREQ_VERY_HIGH;GPIO_InitStruct.Alternate = GPIO_AF7_USART1;HAL_GPIO_Init(GPIOA, &GPIO_InitStruct);// 设置串口为中断模式HAL_NVIC_SetPriority(USART1_IRQn, 0, 0);HAL_NVIC_EnableIRQ(USART1_IRQn);}
}void HAL_UART_MspDeInit(UART_HandleTypeDef *huart) {if (huart->Instance == USART1) {__HAL_RCC_USART1_CLK_DISABLE();HAL_GPIO_DeInit(GPIOA, GPIO_PIN_9 | GPIO_PIN_10);HAL_NVIC_DisableIRQ(USART1_IRQn);}
}#endif

驱动的使用(发送什么回什么)

初始化

khyx_int bsp_init(void) {// 配置系统时钟SystemClock_Config();bsp_uart_init();return KHYX_SUCCESS;
}

测试任务

/**** Copyright (c) 2024-2024,KHYX******************************************************************************* @file           : task_serial_monitor.c* @author         : shchl* @brief          : None* @attention      : None* @crete date     : 24-9-15******************************************************************************* Change Logs:* Date           Author           Notes* 24-9-15       shchl      */#include "app_task.h"
#define SERIAL_MONITOR_THREAD_STACK_SIZE 1024
#define SERIAL_MONITOR_THREAD_PRIORITY   2
#define SERIAL_MONITOR_THREAD_SLICE      1static TX_THREAD serial_monitor_thread;static void serial_monitor_thread_entry(ULONG entry_input) {static uint8_t rx_buf[128];while (1) {const int len = uart_receive(USART1, rx_buf,128);if (len) {uart_send_block( USART1, rx_buf, len);}tx_thread_sleep(10);}
}
/*** @brief  串口监视器线程初始化* @param* @retval None*/
khyx_int tx_serial_monitor_thread_init(void) {// 创建串口监视器线程tx_thread_create(&serial_monitor_thread, "serial_monitor_thread",serial_monitor_thread_entry, 0,tx_malloc(SERIAL_MONITOR_THREAD_STACK_SIZE), SERIAL_MONITOR_THREAD_STACK_SIZE,SERIAL_MONITOR_THREAD_PRIORITY, SERIAL_MONITOR_THREAD_PRIORITY,TX_NO_TIME_SLICE, TX_AUTO_ACTIVATE);return KHYX_SUCCESS;
}KHYX_APP_INIT_EXPORT(tx_serial_monitor_thread_init);

结果

在这里插入图片描述

注意,如果出现丢包,说明是数据出现覆盖的情况,需要调整一些lwrb的数据指针的大小

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.xdnf.cn/news/1537750.html

如若内容造成侵权/违法违规/事实不符,请联系一条长河网进行投诉反馈,一经查实,立即删除!

相关文章

Java调用数据库 笔记05

一. 数据库&#xff08;通过各种驱动来实现调用&#xff09;&#xff1a; &#xff08;应用程序通过接口控制的各种数据库驱动来调用数据库-->jdbc方法&#xff09; 1.创建Java的普通class类 2.加载驱动 Class.forName("com.mysql.jdbc.Driver"); 3.驱动管理类…

TCP并发服务器的实现

一请求一线程 问题 当客户端数量较多时&#xff0c;使用单独线程为每个客户端处理请求可能导致系统资源的消耗过大和性能瓶颈。 资源消耗&#xff1a; 线程创建和管理开销&#xff1a;每个线程都有其创建和销毁的开销&#xff0c;特别是在高并发环境中&#xff0c;这种开销…

开源 AI 智能名片链动 2+1 模式 O2O 商城小程序在社群活动中的应用与时机选择

摘要&#xff1a;本文探讨了开源 AI 智能名片链动 21 模式 O2O 商城小程序在社群经济中的重要性&#xff0c;着重分析了如何借助该小程序适时举办大型活动以维持和引爆社群活跃度。通过对活动时机选择的研究&#xff0c;强调了针对社群用户量身定制活动时机的必要性&#xff0c…

简单了解微服务--黑马(在更)

认识微服务 单体架构 不适合大型复杂项目 微服务架构 将单体结构的各个功能模块拆分为多个独立的项目 拆取的独立项目分别开发&#xff0c;在部署的时候也要分别去编译打包&#xff0c;分别去部署&#xff0c;不同的模块部署在不同的服务器上&#xff0c;对外提供不同的功能…

渗透测试入门学习——php表单form与POST、GET请求练习

最终效果&#xff1a; 必填项为空报错提示&#xff1a; 代码&#xff1a; <!DOCTYPE html> <html lang"en"> <head><meta charset"UTF-8"><title>php表单练习</title> </head> <body> <?php//php中的…

UE5学习笔记22-武器瞄准和武器自动开火

0、一些疑问的记录 1.UUserWidget类和AHUD类的区别。两者都是关于界面显示的类。 实践&#xff1a; 想让界面和用户有交互使用UUserWidget&#xff0c;如果不要交互只是显示使用AHUD类&#xff0c;例如使用UUserWidget类制作开始界面&#xff0c;游戏开始&#xff0c;游戏设置&…

计算机人工智能前沿进展-大语言模型方向-2024-09-17

计算机人工智能前沿进展-大语言模型方向-2024-09-17 1. Large Language Models in Biomedical and Health Informatics: A Review with Bibliometric Analysis H Yu, L Fan, L Li, J Zhou, Z Ma, L Xian, W Hua, S He… - Journal of Healthcare …, 2024 生物医学和健康信息…

部分动态铜皮的孤岛无法删除。报错

(SPMHCI-1): Cannot break shape into fragments. 网上寻找了很多答案&#xff0c;都不太理想&#xff0c;不是我想要的方法。 终于功夫不负有心人&#xff0c;在Cadence官方论坛找到了蛛丝马迹。 Breaking Static shape into fragments - PCB Design - PCB Design & IC …

深入解析Transformer原理

在自然语言处理&#xff08;NLP&#xff09;领域&#xff0c;Transformer架构的出现无疑是一个里程碑式的进展。从Google的BERT到OpenAI的GPT系列&#xff0c;Transformer已经成为许多前沿AI模型的核心。今天&#xff0c;我们就来深入探讨Transformer的原理&#xff0c;帮助你更…

优惠充值话费api对接如何选择对接平台?

优惠充值话费接口通常由电信运营商、第三方支付平台或专业的充值服务提供商提供。这些平台通过API接口允许开发者将话费充值功能集成到应用程序或网站中。 选择哪个平台比较好&#xff0c;取决于以下几个因素&#xff1a; 覆盖范围&#xff1a;选择能够覆盖你需要服务的地区和…

深度学习-生成式检索-论文速读-2024-09-14

深度学习-生成式检索-论文速读-2024-09-14 前言: 生成式检索&#xff08;Generative Retrieval&#xff0c; GR&#xff09;是一种结合了生成模型和检索系统的人工智能技术方法。这种方法在处理信息检索任务时&#xff0c;不仅依赖于已有数据的检索&#xff0c;还能生成新的、…

c++基类和派生类对象的赋值转换——赋值兼容规则

1.引出 如下场景&#xff1a; 由于b是double类型&#xff0c;所以赋值给int类型的引用前&#xff0c;要先进行隐式类型转换&#xff0c;这中间会生成临时对象&#xff0c;类是对象具有常性&#xff0c;所以int&之前应该加上const。 但是下面的场景&#xff1a; 没有出现报…

Python3网络爬虫开发实战(16)分布式爬虫(第一版)

文章目录 一、分布式爬虫原理1.1 分布式爬虫架构1.2 维护爬取队列1.3 怎样来去重1.4 防止中断1.5 架构实现 二、Scrapy-Redis 源码解析2.1 获取源码2.2 爬取队列2.3 去重过滤2.4 调度器 三、Scrapy 分布式实现3.1 准备工作3.2 搭建 Redis 服务器3.3 部署代理池和 Cookies 池3.4…

python画图|中秋到了,尝试画个月亮(球体画法)

学习了一段时间的画图&#xff0c;已经掌握了一些3D图的画法&#xff0c;部分链接如下&#xff1a; python画图|极坐标下的3D surface-CSDN博客 python画图|3D参数化图形输出-CSDN博客 我们今天尝试一下月亮的画法。 【1】官网教程 首先还是到达官网教程学习&#xff1a; …

# 深度学习笔记(6)Hugginface -Transformer

深度学习笔记&#xff08;6&#xff09;Hugginface -Transformer 文章目录 深度学习笔记&#xff08;6&#xff09;Hugginface -Transformer一、工具包二、 Tokenizer三、 模型加载四、 输出五&#xff0c;padding的作用5.1 attention_mask5.2 不同padding方法 六&#xff0c;数…

基于yolov8+lprnet的中文车牌识别系统python源码+pytorch模型+精美GUI界面

【算法介绍】 基于YOLOv8和LPRNet的中文车牌识别系统是一种高效且准确的解决方案&#xff0c;结合了目标检测与字符识别的先进技术。YOLOv8作为最新的实时目标检测算法&#xff0c;以其高速度和精确度著称&#xff0c;能够迅速在图像或视频中定位车牌位置。LPRNet则是一种专为…

反向传播(Back Propagation,简称BP)

反向传播算法是用于训练神经网络的核心算法之一&#xff0c;它通过计算损失函数&#xff08;如均方误差或交叉熵&#xff09;相对于每个权重参数的梯度&#xff0c;来优化神经网络的权重。 1.前向传播&#xff08;Forward Propagation&#xff09; 步骤 输入层&#xff1a;接…

孙怡带你深度学习(2)--PyTorch框架认识

文章目录 PyTorch框架认识1. Tensor张量定义与特性创建方式 2. 下载数据集下载测试展现下载内容 3. 创建DataLoader&#xff08;数据加载器&#xff09;4. 选择处理器5. 神经网络模型构建模型 6. 训练数据训练集数据测试集数据 7. 提高模型学习率 总结 PyTorch框架认识 PyTorc…

计算机人工智能前沿进展-大语言模型方向-2024-09-13

计算机人工智能前沿进展-大语言模型方向-2024-09-13 1. OneEdit: A Neural-Symbolic Collaboratively Knowledge Editing System Authors: Ningyu Zhang, Zekun Xi, Yujie Luo, Peng Wang, Bozhong Tian, Yunzhi Yao, Jintian Zhang, Shumin Deng, Mengshu Sun, Lei Liang, Z…

基于51单片机的直流数字电流表proteus仿真

地址&#xff1a; https://pan.baidu.com/s/1adZbhgOBvvg0KsCO6_ZiAw 提取码&#xff1a;1234 仿真图&#xff1a; 芯片/模块的特点&#xff1a; AT89C52/AT89C51简介&#xff1a; AT89C52/AT89C51是一款经典的8位单片机&#xff0c;是意法半导体&#xff08;STMicroelectro…