STM32 串口协议简明教程

前言

本文旨在介绍STM32单片机串口协议的使用。
主要是为了个人复习,一段时间没用,就容易忘记。因此在文章中也不会出现串口的原理等讲解。
本文的重点是利用CubeMX实现一个最基本的串口模板,从而能够在往后的各个项目中得到运用。

本文使用单片机是STM32F407VET6核心板。因为是第一篇关于STM32的文章,我就浅浅讲一下新工程的创建

你将解决以下问题

  • 创建一个最基本的工程模板
  • 完成串口发送消息(使用HAL_UART_Transmit 和 printf)
  • 完成串口回显(中断回调函数的配置)

创建新工程

打开Cube,选择你对应型号的单片机,正常情况下,你会看到这个
在这里插入图片描述
首先配置SystemCore,Debug务必选择SerialWire,否则无法重复烧录,你的单片机成一次性的了
在这里插入图片描述
接下来配置RCC,选择HSE并找到Crystal Resonator选项,就是选择外部高速晶振。你选择了这个,就意味着时钟树配置的时候你要修改Input Frequency为这个外部高速晶振的频率。
在这里插入图片描述
最后,找到NVIC选项,勾选RCC全局中断
在这里插入图片描述
完事之后配置一下GPIO,点亮一个LED啥的,来确保代码是正确的。本代码不包括FreeRTOS,我们直接生成裸机代码。
在这里插入图片描述
时钟树关键要调节这几个部分。调节完毕直接回车。这里面有几个比较重要的我说一下.
第一个是输入频率。这个为什么是8MHz?
关于Input Frequency,一方面,你可以直接查阅芯片手册,上面会有高速外部晶振的推荐频率。另一方面,也是最实际的,你可以直接询问你购入芯片的商家。大多数商家会直接把这个晶振标注在外面。就不用你费心思查阅了。
第二个是你
一定要选择HSE和PLLCLK
,也就是选择高速外部时钟和锁相环倍增。HCLK直接和下面推荐的max设定一样,来确保单片机发挥了最大性能。
在这里插入图片描述

我这边用的是优信电子的F407VET6核心板,他很贴心地帮我标好了。一般407都是这个外部晶振频率
在这里插入图片描述
配置完时钟树后,我们可以开始生成工程代码,并测试生成工程的正确性
后面我把A0的初始化改成A1了,所以在main函数的while里面扔如下代码,只要led闪烁了,就说明工程生成是正确的

while (1)
{
  /* USER CODE END WHILE */
  HAL_GPIO_TogglePin(GPIOA,GPIO_PIN_1);
  HAL_Delay(500);
  /* USER CODE BEGIN 3 */
}

使用CubeMX配置串口

在这里插入图片描述
找到Connectivity中的UART5选项,并选择Mode为Async…,也就是异步模式。大多数的串口都是异步串口。在上面图片的右侧,展示了这个串口的具体引脚。分别是D2和C12。
串口的波特率、字长等都保持默认。
一般项目中只会涉及波特率的修改,这里BaudRate是115200。
另外,我们的串口是依赖中断来实现接收的,接收也是本文的重点。因此我们需要配置中断,打勾UART5 global interrupt即可
在这里插入图片描述
到此为止,你可以生成代码了

串口收发信息

单单完成一个初始化只能直接使用信息的发送。如果要使用信息的接受,则需要在MX_UART5_Init();函数下面添加一个串口接收中断。
HAL_UART_Receive_IT这个函数的作用是,等待串口收到一个rx_data的数据,只要串口收到了rx_data大小的数据,就会触发一次串口回调的中断函数。这个中断函数,咱们后面再谈。
请注意“等待”这个词。意思是只要代码经过了这个函数,串口就会开启一次等待。等待的时候这个任务是挂起的,不会阻塞,因而不会影响到后续代码的执行。
中断回调函数里面写着我们用户自定义的代码,例如实现回显,或者解析数据流等等操作。

MX_UART5_Init();
/* USER CODE BEGIN 2 */
HAL_UART_Receive_IT(&huart5, &rx_data, sizeof(rx_data));
/* USER CODE END 2 */

这里rx_data定义在usart.c中,并extern出来。这个头文件是cube自动生成的,可以在User里面直接找到

/* USER CODE BEGIN Includes */
extern uint8_t rx_data;
/* USER CODE END Includes */

另外,为了使用printf,我们在usart.c里面添加fputc函数。这个函数在stdio.c中,是一个弱定义。因此要记得在usart.c中引用这个头文件"stdio.h"

/* USER CODE BEGIN 0 */
#include "stdio.h"
uint8_t rx_data;
int fputc(int ch, FILE *f)
{
  HAL_UART_Transmit(&huart5, (uint8_t *)&ch, 1, 0xFFFF);
  return ch;
}
/* USER CODE END 0 */

值得注意的是,如果你要使用printf,你就必须要勾选这个UseMicroLIB的选项,否则printf将无法正常使用。
在这里插入图片描述
接下来,我们分别使用HAL_UART_Transmit和printf来实现串口消息打印。
这里值得注意的是,HAL_UART_Transmit在HAL库中有两种,它们的函数原型如下

HAL_StatusTypeDef HAL_UART_Transmit(UART_HandleTypeDef *huart, const uint8_t *pData, uint16_t Size, uint32_t Timeout);
HAL_StatusTypeDef HAL_UART_Transmit_IT(UART_HandleTypeDef *huart, const uint8_t *pData, uint16_t Size);

它们的区别在于阻塞和非阻塞。换言之,就是没有IT的会有一个Timeout变量,只要经过了这么长时间,还是没有发送成功,那么单片机就不会因为这个任务毁了整个流程,过了Timeout时长,就不会继续卡住了。
HAL_UART_Transmit_IT则不然,只要信息发送成功,他就会来一个串口中断。但是没啥卵用。我们一般将串口中断用在接收中。
因此我们会喜欢去使用阻塞发送,非阻塞接收
我们用如下代码实现串口消息发送

int main(void)
{
  /* USER CODE BEGIN 1 */
  const uint8_t test_word[20]="hello!";
  /* USER CODE END 1 */

  /* MCU Configuration--------------------------------------------------------*/

  /* Reset of all peripherals, Initializes the Flash interface and the Systick. */
  HAL_Init();

  /* USER CODE BEGIN Init */

  /* USER CODE END Init */

  /* Configure the system clock */
  SystemClock_Config();

  /* USER CODE BEGIN SysInit */

  /* USER CODE END SysInit */

  /* Initialize all configured peripherals */
  MX_GPIO_Init();
  MX_UART5_Init();
  /* USER CODE BEGIN 2 */
  HAL_UART_Receive_IT(&huart5, &rx_data, sizeof(rx_data));

  HAL_UART_Transmit(&huart5, test_word, sizeof(test_word), 1000);
  printf("good!");
  /* USER CODE END 2 */

  /* Infinite loop */
  /* USER CODE BEGIN WHILE */
  while (1)
  {
    /* USER CODE END WHILE */

    /* USER CODE BEGIN 3 */
    HAL_GPIO_TogglePin(GPIOA, GPIO_PIN_1);
    HAL_Delay(500);
  }
  /* USER CODE END 3 */
}

测试结果毫无问题
在这里插入图片描述
接着测试串口接收。
找到stm32f4xx_it.c,划到最后面,添加这一段函数。不要忘记在这个c文件的前面引用#include "usart.h"

/* USER CODE BEGIN 1 */
void HAL_UART_RxCpltCallback(UART_HandleTypeDef *huart){  
  if(huart==&huart5){
    printf("%c\n",rx_data);
    HAL_UART_Receive_IT(&huart5, &rx_data, sizeof(rx_data));
  }
}
/* USER CODE END 1 */

为啥这样做就好了?HAL_UART_RxCpltCallback在HAL库中是一个弱定义,这里需要用户去具体实现。串口最终会调用这个中断服务函数。那么要怎么知道具体是哪个串口的中断服务函数?只需要加一个if作为判断即可,写法就类似于if(huart==&huart5)
其中rx_data就是接收到的数据。注意,接受完数据后,如果要持续接收数据,那就必须要继续使用HAL_UART_Receive_IT(&huart5, &rx_data, sizeof(rx_data));
这样,即使你在串口助手里面输入一堆字符串,它也会一个一个处理,一个一个显示出来。因为你的每一个字符都会触发一次中断回调函数。
在这里插入图片描述

至此,一个串口的收发基本模板已经形成,可以开始放进你的具体工程实践了。

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

老龄化对投资意味着什么?

1月15日,国务院办公厅印发《关于发展银发经济增进老年人福祉的意见》从4个方面提出26项举措,为我国首个以“银发经济”命名的政策文件。 近期,国信证券分析师王开发布题为《银发经济再思考:老龄化对投资的影响》的报告&#xff0…

Java8-Stream 流基本应用-groupBy进行分组

groupBy进行分组 Testpublic void testStreamGroupBy(){List<UserInfoModel> resultnew ArrayList<>();for (int i 0; i < 10; i) {UserInfoModel usernew UserInfoModel();user.setUserId(i"");user.setUserName("kangshihang");result.a…

探索设计模式的魅力:深入了解适配器模式-优雅地解决接口不匹配问题

设计模式专栏&#xff1a;http://t.csdnimg.cn/nolNS 目录 一、引言 1. 概述 2. 为什么需要适配器模式 3. 本文的目的和结构 二、简价 1. 适配器模式的定义和特点 定义 特点 2. 适配器模式的作用和适用场景 作用 适用场景 3. 适配器模式与其他设计模式的比较 三、适配…

代码增强LLM

大模型时代的语言模型&#xff08;LLM&#xff09;不仅在尺寸上变得更大了&#xff0c;而且训练数据也同时包含了自然语言和形式语言&#xff08;代码&#xff09;。作为人类和计算机之间的媒介&#xff0c;代码可以将高级目标转换为可执行的中间步骤&#xff0c;具有语法标准、…

Java 与 JavaScript的区别

Java 与 JavaScript的区别 Java 与 JavaScript&#xff1a;概述Java的特点JavaScript 的起源JavaScript 的特点Java 与 JavaScript&#xff0c;哪个更好&#xff1f;JavaScript 与 Java 相似吗&#xff1f;Java 与 JavaScript 的区别JavaScript 在服务器端的运行方式是怎样的&a…

线程锁多线程的复习

线程 实现方式3种乐观锁&悲观锁线程池线程池总结 进程:是正在运行的程序 线程:是进程中的单个顺序控制流,是一条执行路径 实现方式3种 1.Thread //步骤一:定义一个继承Thread的类 //步骤二:再定义的类中重写run()方法 //步骤三:创建定义类对象 //步骤四:启动线程 class M…

【数据分析】numpy基础第一天

文章目录 前言本文代码&#xff1a;使用jupyter notebook打开本文的代码操作示例步骤1.打开Anaconda Powershell Prompt步骤2.复制代码文件地址步骤3.在Anaconda Powershell Prompt中打开jupyter notebook步骤3.5.解决一个可能的问题步骤4.在浏览器中查看ipynb文件步骤5.运行代…

85.网游逆向分析与插件开发-物品使用-物品使用的逆向分析与C++代码的封装

内容参考于&#xff1a;易道云信息技术研究院VIP课 上一个内容&#xff1a;项目需求与需求拆解-CSDN博客 码云地址&#xff08;ui显示角色数据 分支&#xff09;&#xff1a;https://gitee.com/dye_your_fingers/sro_-ex.git 码云版本号&#xff1a;453dd83d54140d2e1ee65c9…

量化交易学习3(量化择时策略)

1 什么是量化择时 量化择时策略&#xff0c;简单来说&#xff0c;就是采用数量化分析方法&#xff0c;利用单个或多个技术指标的组合&#xff0c;来对交易标的股票或股票指数进行低买高卖的操作&#xff0c;期望获得超越简单买入持有策略的收益风险表现。 量化择时策略的核心…

网络防御安全知识(第三版)

配置黑洞路由 --- 黑洞路由即空接口路由&#xff0c;在NAT地址池中的地址&#xff0c;建议配置达到这个地址指 向空接口的路由&#xff0c;不然&#xff0c;在特定环境下会出现环路。&#xff08;主要针对地址池中的地址和出接口地址 不再同一个网段中的场景。&#xff09; …

二手交易|校园二手交易小程序|基于微信小程序的闲置物品交易平台设计与实现(源码+数据库+文档)

校园二手交易小程序目录 目录 基于微信小程序的闲置物品交易平台设计与实现 一、前言 二、系统功能设计 三、系统实现 1、用户信息管理 2、商品信息管理 3、公告信息管理 4、论坛信息管理 四、数据库设计 1、实体ER图 五、核心代码 六、论文参考 七、最新计算机毕…

档案数字化转型面临问题

档案数字化转型面临以下问题&#xff1a; 1. 技术问题&#xff1a;档案数字化需要借助先进的技术手段和设备&#xff0c;包括扫描仪、存储设备和数据管理软件等。这些技术的成本高、操作复杂&#xff0c;需要专业的人员进行操作和维护。 2. 安全问题&#xff1a;档案数字化后的…

重写Sylar基于协程的服务器(0、搭建开发环境以及项目框架 || 下载编译简化版Sylar)

重写Sylar基于协程的服务器&#xff08;0、搭建开发环境以及项目框架 || 下载编译简化版Sylar&#xff09; 重写Sylar基于协程的服务器系列&#xff1a; 重写Sylar基于协程的服务器&#xff08;0、搭建开发环境以及项目框架 || 下载编译简化版Sylar&#xff09; 前言 sylar是…

[C语言][C++][时间复杂度详解分析]二分查找——杨氏矩阵查找数字详解!!!

一&#xff0c;题目 遇到的一道算法题&#xff1a; 1&#xff0c;已知有一个数字矩阵&#xff08;row行&#xff0c;col列&#xff09;&#xff0c;矩阵的每行 从左到右 递增&#xff0c;每列 从上到下 递增。 2&#xff0c;现输入一个数字 num &#xff0c;判断数字矩阵中…

Python列表中的append功能及用法举例

Python列表中的append功能及用法举例 &#x1f335;文章目录&#x1f335; &#x1f333;引言&#x1f333;&#x1f333;append()&#x1f333;&#x1f340;功能介绍&#x1f340;&#x1f340;语法&#x1f340;&#x1f340;示例&#x1f340;&#x1f340;注意事项&#x…

【Go-Zero】Windows启动rpc服务报错panic:context deadline exceeded解决方案

【Go-Zero】Windows启动rpc服务报错panic:context deadline exceeded解决方案 大家好 我是寸铁&#x1f44a; 总结了一篇Windows11下启动rpc服务报错panic解决方案的文章✨ 喜欢的小伙伴可以点点关注 &#x1f49d; 问题来源 今天在编写完proto文件后&#xff0c;使用goctl生成…

jenkins pipeline配置maven可选参数

1、在Manage Jenkins下的Global Tool Configuration下对应的maven项添加我们要用得到的不同版本的maven安装项 2、pipeline文件内容具体如下 我们maven是单一的&#xff0c;所以我们都是配置单选参数 pipeline {agent anyparameters {gitParameter(name: BRANCH_TAG, type: …

【算法Hot100系列】合并区间

💝💝💝欢迎来到我的博客,很高兴能够在这里和您见面!希望您在这里可以感受到一份轻松愉快的氛围,不仅可以获得有趣的内容和知识,也可以畅所欲言、分享您的想法和见解。 推荐:kwan 的首页,持续学习,不断总结,共同进步,活到老学到老导航 檀越剑指大厂系列:全面总结 jav…

【Redis】关于它为什么快?使用场景?以及使用方式?为何引入多线程?

目录 1.既然redis那么快&#xff0c;为什么不用它做主数据库&#xff0c;只用它做缓存&#xff1f; 2.Redis 一般在什么场合下使用&#xff1f; 3.redis为什么这么快&#xff1f; 4.Redis为什么要引入了多线程&#xff1f; 1.既然redis那么快&#xff0c;为什么不用它做主数据…

电路笔记 :MOS场效应晶体管+红外遥控+AMS1117 电源模块

三极管&#xff08;BJT&#xff0c;Bipolar Junction Transistor&#xff09;和 MOSFET&#xff08;Metal-Oxide-Semiconductor Field-Effect Transistor&#xff09;是两种不同类型的晶体管&#xff0c;它们在工作原理、性能特性和应用方面有一些重要的区别。 结构和工作原理…