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.rhkb.cn/news/426090.html

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

kali里面搭建docker容器

注意事项&#xff1a;kali版本&#xff0c;镜像源 &#xff08;1&#xff09;权限为管理员&#xff1a; sudo su (2) 更新软件包列表并升级已安装的软件包 apt-get update apt-get upgrade 出错了&#xff0c;应该是更新源出问题了。 &#xff08;3&#xff09;更换镜像源&am…

java日志框架之JUL(Logging)

文章目录 一、JUL简介1、JUL组件介绍 二、Logger快速入门三、Logger日志级别1、日志级别2、默认级别info3、原理分析4、自定义日志级别5、日志持久化&#xff08;保存到磁盘&#xff09; 三、Logger父子关系四、Logger配置文件 一、JUL简介 JUL全程Java Util Logging&#xff…

Git提交类型

说明&#xff1a;Git提交类型指的是代码commit时&#xff0c;写在comment前面的标志&#xff0c;表示此次commit的提交类型&#xff0c;如下&#xff1a; Git提交类型 常见的Git提交类型有&#xff1a; feat&#xff1a;新特性、新功能或优化&#xff1b; fix&#xff1a;修复…

C++ | Leetcode C++题解之第413题等差数列划分

题目&#xff1a; 题解&#xff1a; class Solution { public:int numberOfArithmeticSlices(vector<int>& nums) {int n nums.size();if (n 1) {return 0;}int d nums[0] - nums[1], t 0;int ans 0;// 因为等差数列的长度至少为 3&#xff0c;所以可以从 i2 开…

Mysql表的增删查改(基础)

1.CRUD 1. 在SQL中&#xff0c;我们可以在 -- 后写注释的内容。 2. CRUD即增加(Create),查询(Retrieve),更新(Update),删除(Delete)的英文首字母缩写。 2.增加&#xff08;Create&#xff09; SQL语句&#xff1a; insert into 表名 (列名&#xff0c;列明...) values (值…

Java | Leetcode Java题解之第414题第三大的数

题目&#xff1a; 题解&#xff1a; class Solution {public int thirdMax(int[] nums) {Integer a null, b null, c null;for (int num : nums) {if (a null || num > a) {c b;b a;a num;} else if (a > num && (b null || num > b)) {c b;b num;…

海康威视相机在QTcreate上的环境配置教程(qt+opencv+海康SDK)

环境配置教程 前言&#xff1a;环境配置&#xff1a;1.海康SDK2.opencv 参考导入文件 前言&#xff1a; 配置环境是编程的第一步&#xff0c;所以写这篇文章来指导环境的配置。如果已经配置好了&#xff0c;想在qt上使用海康的摄像头&#xff0c;可以参考这篇文章&#xff1a;…

MFC实现对话框与控件的自适应调节

一.对话框属性设置 先将对话框的Border属性由Dialog Frame设置为Resizing。 二.添加消息处理函数 类向导 -> 消息&#xff08;WM-SIZE&#xff09;添加处理函数->void COnshowDlg::OnSize(UINT nType, int cx, int cy)。 三.添加代码 在.h文件中声明&#xff1a; vo…

影刀RPE学习——自动化

下载网址&#xff1a;影刀RPA - 影刀官网 (yingdao.com) 傻瓜式安装进入界面&#xff1a; 官方教程&#xff1a;影刀RPA零基础入门教程&#xff08;2024最新版&#xff09;&#xff1a;01 引入课-影刀初相识_哔哩哔哩_bilibili

[数据集][目标检测]高铁受电弓检测数据集VOC+YOLO格式1245张2类别

数据集格式&#xff1a;Pascal VOC格式YOLO格式(不包含分割路径的txt文件&#xff0c;仅仅包含jpg图片以及对应的VOC格式xml文件和yolo格式txt文件) 图片数量(jpg文件个数)&#xff1a;1245 标注数量(xml文件个数)&#xff1a;1245 标注数量(txt文件个数)&#xff1a;1245 标注…

sqlgun靶场训练

1.看到php&#xff1f;id &#xff0c;然后刚好有个框&#xff0c;直接测试sql注入 2.发现输入1 union select 1,2,3#的时候在2处有回显 3.查看表名 -1 union select 1,group_concat(table_name),3 from information_schema.tables where table_schemadatabase()# 4.查看列名…

安装node 报错需要:glibc >= 2.28

--> 解决依赖关系完成 错误&#xff1a;软件包&#xff1a;2:nodejs-18.20.4-1nodesource.x86_64 (nodesource-nodejs) 需要&#xff1a;libm.so.6(GLIBC_2.27)(64bit) 错误&#xff1a;软件包&#xff1a;2:nodejs-18.20.4-1nodesource.x86_64 (nodesource-nodej…

【AcWing】【双指针算法】799. 最长连续不重复子序列

最长连续不重复子序列 这是一道基于双指针算法的题目&#xff0c;但是想解这道题需要一点额外的思路&#xff0c;第一遍我没想出来&#xff0c;故在此对这道题的思路进行记录。 题目描述与输入输出 思路 这道题目的要求是寻找不包含重复的数的最长子序列&#xff0c;从题目来…

Python | Leetcode Python题解之第413题等差数列划分

题目&#xff1a; 题解&#xff1a; class Solution:def numberOfArithmeticSlices(self, nums: List[int]) -> int:n len(nums)if n 1:return 0d, t nums[0] - nums[1], 0ans 0# 因为等差数列的长度至少为 3&#xff0c;所以可以从 i2 开始枚举for i in range(2, n):i…

OpenCV calcHist()函数及其用法详解

OpenCV calcHist()函数原型共有三个&#xff0c;如下&#xff1a; 该函数计算一个或多个数组的直方图。用于递增直方图箱的元组的元素取自同一位置的相应输入数组。 函数参数&#xff1a; images 源&#xff08;图像&#xff09;数组。它们都应具有相同的深度、CV_8U、CV_16U…

若依-原理

1.代码生成器 1.1源码分析 代码生成器分为两个部分&#xff1a; 第一部分涉及将业务表结构导入到系统中 第二部分是点击生成按钮&#xff0c;系统将根据表结构生成相应的前后端代码&#xff0c;并提供下载。 1.表结构说明 gen_table&#xff1a;存储业务表的基本信息 &am…

硬件工程师笔试面试——无线通讯模块

目录 15、无线通讯模块 15.1 基础 无线通讯模块实物图 15.1.1 概念 15.1.2 常见的无线通讯模块及其特点 15.1.3 无线通讯模块参数 15.1.4 无线通讯模块工作原理 15.2 相关问题 15.2.1 如何根据项目需求选择合适的无线通讯模块? 15.2.2 无线通讯模块的安全性如何,如…

[机器学习]聚类算法

1 聚类算法简介 # 导包 from sklearn.datasets import make_blobs import matplotlib.pyplot as plt from sklearn.cluster import KMeans from sklearn.metrics import calinski_harabasz_score # 构建数据 x,ymake_blobs(n_samples1000,n_features2,centers[[-1,-1],[0,0],[1…

硬件工程师笔试面试——电机

目录 18、电机 18.1 基础 电机原理图 电机实物图 18.1.1 概念 18.1.2 电机的一些基本分类和特点 18.2 相关问题 18.2.1 不同类型的电机在实际应用中有哪些具体的优势和劣势 18.2.2 在设计一个电机系统时,我应该如何考虑电机的选型和配置? 18.2.3 对于需要频繁启停的…

[SWPUCTF 2021 新生赛]Do_you_know_http

很基础的一题&#xff0c;就是修改发送的数据包 1.拿到题目&#xff0c;他让我们使用这个WLLM浏览器&#xff0c;那我们就用bp抓包&#xff0c;修改成User-Agent:WLLM 2.得到响应有个a.php文件&#xff0c;那我们就访问一下&#xff0c;发现请求权限不够&#xff0c;ip地址不对…