HAL库与STM32CubeMX结合详解UART串口通信
单片机与嵌入式','单片机与嵌入式 2024-01-30

分享本文学习STM32CubeMX串口的操作,以及HAL库串口的配置,我们会详细的讲解各个模块的使用和具体功能,并且基于HAL库实现Printf函数功能重定向,UART中断接收,本系列教程将HAL库与STM32CubeMX结合在一起讲解,使您可以更快速的学会各个模块的使用。


所用工具

1、芯片:STM32F407ZET6

2、STM32CubeMx软件

3、IDE:MDK-Keil软件

4、STM32F1xx/STM32F4xxHAL库

5、串口:使用USART1 PA9,PA10

知识概括

  • STM32CubeMX创建串口例程

  • HAL库UATR函数库

  • 重定义printf函数

  • HAL库,UART中断接收

  • HAL库UATR接收与发送例程

下面介绍工程的创建过程。

设置RCC

设置高速外部时钟HSE 选择外部时钟源。

设置串口

  • 点击USATR1

  • 设置MODE为异步通信(Asynchronous)

  • 基础参数波特率为115200 Bits/s。传输数据长度为8 Bit。奇偶检验无,停止位1 ,接收和发送都使能

  • 4GPIO引脚设置 USART1_RX/USART_TX

  • 5 NVIC Settings 一栏使能接收中断

设置时钟

我用的外部晶振为8MHz。

  • 1选择外部时钟HSE 8MHz

  • 2PLL锁相环倍频72倍

  • 3系统时钟来源选择为PLL

  • 4设置APB1分频器为 /2

如果不懂STM32的时钟树框图的话,请看《【STM32】系统时钟RCC详解(超详细,超全面)》


项目文件设置

  • 1 设置项目名称

  • 2 设置存储路径

  • 3 选择所用IDE

创建工程文件

然后点击GENERATE CODE  创建工程。

配置下载工具

新建的工程所有配置都是默认的  我们需要自行选择下载模式,勾选上下载后复位运行。


下面介绍下HAL库UART函数库。

UART结构体定义

UART_HandleTypeDef huart1;

UART的名称定义,这个结构体中存放了UART所有用到的功能,后面的别名就是我们所用的uart串口的别名,默认为huart1,可以自行修改。

串口发送/接收函数如下。

  • HAL_UART_Transmit();串口发送数据,使用超时管理机制

  • HAL_UART_Receive();串口接收数据,使用超时管理机制

  • HAL_UART_Transmit_IT();串口中断模式发送

  • HAL_UART_Receive_IT();串口中断模式接收

  • HAL_UART_Transmit_DMA();串口DMA模式发送

  • HAL_UART_Transmit_DMA();串口DMA模式接收

这几个函数的参数基本都是一样的,我们挑两个讲解一下。

串口发送数据:

HAL_UART_Transmit(UART_HandleTypeDef *huart, uint8_t *pData, uint16_t Size, uint32_t Timeout)

功能:串口发送指定长度的数据。如果超时没发送完成,则不再发送,返回超时标志(HAL_TIMEOUT)。

参数:

  • UART_HandleTypeDef *huart 是UATR的别名,如 :

     UART_HandleTypeDef huart1; //别名就是huart1
  • *pData      需要发送的数据

  • Size    发送的字节数

  • Timeout   最大发送时间,发送数据超过该时间退出发送

举例:

HAL_UART_Transmit(&huart1, (uint8_t *)ZZX, 3, 0xffff); //串口发送三个字节数据,最大传输时间0xffff


中断接收数据:

HAL_UART_Receive_IT(UART_HandleTypeDef *huart, uint8_t *pData, uint16_t Size)

功能:串口中断接收,以中断方式接收指定长度数据。

大致过程是,设置数据存放位置,接收数据长度,然后使能串口接收中断。接收到数据时,会触发串口中断。再然后,串口中断函数处理,直到接收到指定长度数据,而后关闭中断,进入中断接收回调函数,不再触发接收中断。(只触发一次中断)

参数:

  • UART_HandleTypeDef *huart 是UATR的别名,如 :

    UART_HandleTypeDef huart1;   //别名就是huart1
  • *pData      接收到的数据存放地址

  • Size    接收的字节数

举例:

HAL_UART_Receive_IT(&huart1,(uint8_t *)&value,1); //中断接收一个字符,存储到value中

如下是串口中断函数。

  • HAL_UART_IRQHandler(UART_HandleTypeDef *huart);  //串口中断处理函数

  • HAL_UART_TxCpltCallback(UART_HandleTypeDef *huart);  //串口发送中断回调函数

  • HAL_UART_TxHalfCpltCallback(UART_HandleTypeDef *huart);  //串口发送一半中断回调函数(用的较少)

  • HAL_UART_RxCpltCallback(UART_HandleTypeDef *huart);  //串口接收中断回调函数

  • HAL_UART_RxHalfCpltCallback(UART_HandleTypeDef *huart);//串口接收一半回调函数(用的较少)

  • HAL_UART_ErrorCallback();串口接收错误函数

串口接收中断回调函数:

HAL_UART_RxCpltCallback(UART_HandleTypeDef *huart);

功能:HAL库的中断进行完之后,并不会直接退出,而是会进入中断回调函数中,用户可以在其中设置代码, 串口中断接收完成之后,会进入该函数,该函数为空函数,用户需自行修改。

参数:

  • UART_HandleTypeDef *huart 是UATR的别名,如 :


    UART_HandleTypeDef huart1;   //别名就是huart1


串口中断处理函数

HAL_UART_IRQHandler(UART_HandleTypeDef *huart);

功能:对接收到的数据进行判断和处理 判断是发送中断还是接收中断,然后进行数据的发送和接收,在中断服务函数中使用。

如果接收数据,则会进行接收中断处理函数。

/* UART in mode Receiver -----------------------------*/if((tmp_flag != RESET) && (tmp_it_source != RESET)){UART_Receive_IT(huart);}

如果发送数据,则会进行发送中断处理函数。

/* UART in mode Transmitter --------------------------*/if (((isrflags & USART_SR_TXE) != RESET) && ((cr1its & USART_CR1_TXEIE) != RESET)){UART_Transmit_IT(huart);return;}


串口查询函数

HAL_UART_GetState(); //判断UART的接收是否结束,或者发送数据是否忙碌

举例:

while(HAL_UART_GetState(&huart4) == HAL_UART_STATE_BUSY_TX) //检测UART发送结束

USART接收与发送
重新定义printf函数,在 stm32f4xx_hal.c中包含#include

#include"stm32f4xx_hal.h"#includeextern UART_HandleTypeDef huart1; //声明串口

在 stm32f4xx_hal.c 中重写fget和fput函数。

在main.c中添加如下代码。

#define RXBUFFERSIZE  256char RxBuffer[RXBUFFERSIZE];while (1){/* USER CODE END WHILE */printf("Z小旋测试\n");HAL_Delay(1000);/* USER CODE BEGIN 3 */}

之后便可以使用Printf函数和Scanf,getchar函数。

关于UART接收中断,因为中断接收函数只能触发一次接收中断,所以我们需要在中断回调函数中再调用一次中断接收函数具体流程:


  • 初始化串口

  • 在main中第一次调用接收中断函数

  • 进入接收中断,接收完数据  进入中断回调函数

  • 修改HAL_UART_RxCpltCallback中断回调函数,处理接收的数据

  • 回调函数中要调用一次HAL_UART_Receive_IT函数,使得程序可以重新触发接收中断


函数流程:

HAL_UART_Receive_IT(中断接收函数) -> USART2_IRQHandler(void)(中断服务函数)    ->    HAL_UART_IRQHandler(UART_HandleTypeDef *huart)(中断处理函数)    ->    UART_Receive_IT(UART_HandleTypeDef *huart) (接收函数)   ->    HAL_UART_RxCpltCallback(huart);(中断回调函数)

HAL_UART_RxCpltCallback函数就是用户要重写在main.c里的回调函数。

代码实现:

并在main.c中添加下列定义:

#include#define RXBUFFERSIZE  256 //最大接收字节数char RxBuffer[RXBUFFERSIZE]; //接收数据uint8_t aRxBuffer; //接收中断缓冲uint8_t Uart1_Rx_Cnt = 0; //接收缓冲计数

在main()主函数中,调用一次接收中断函数。

/* USER CODE BEGIN 2 */HAL_UART_Receive_IT(&huart1, (uint8_t *)&aRxBuffer, 1);/* USER CODE END 2 */

在main.c下方添加中断回调函数

/* USER CODE BEGIN 4 */voidHAL_UART_RxCpltCallback(UART_HandleTypeDef *huart){/* Prevent unused argument(s) compilation warning */UNUSED(huart);/* NOTE: This function Should not be modified, when the callback is needed,the HAL_UART_TxCpltCallback could be implemented in the user file*/if(Uart1_Rx_Cnt >= 255) //溢出判断{Uart1_Rx_Cnt = 0;memset(RxBuffer,0x00,sizeof(RxBuffer));HAL_UART_Transmit(&huart1, (uint8_t *)"数据溢出", 10,0xFFFF);}else{RxBuffer[Uart1_Rx_Cnt++] = aRxBuffer; //接收数据转存if((RxBuffer[Uart1_Rx_Cnt-1] == 0x0A)&&(RxBuffer[Uart1_Rx_Cnt-2] == 0x0D)) //判断结束位{HAL_UART_Transmit(&huart1, (uint8_t *)&RxBuffer, Uart1_Rx_Cnt,0xFFFF); //将收到的信息发送出去while(HAL_UART_GetState(&huart1) == HAL_UART_STATE_BUSY_TX);//检测UART发送结束Uart1_Rx_Cnt = 0;memset(RxBuffer,0x00,sizeof(RxBuffer)); //清空数组}}HAL_UART_Receive_IT(&huart1, (uint8_t *)&aRxBuffer, 1); //再开启接收中断}/* USER CODE END 4 */

发送数据被正常返回。


声明: 本文转载自其它媒体或授权刊载,目的在于信息传递,并不代表本站赞同其观点和对其真实性负责,如有新闻稿件和图片作品的内容、版权以及其它问题的,请联系我们及时删除。(联系我们,邮箱:evan.li@aspencore.com )
0
评论
  • 【7.24 深圳】2025国际AI+IoT生态发展大会/2025全球 MCU及嵌入式技术论坛


  • 相关技术文库
  • 单片机
  • 嵌入式
  • MCU
  • STM
  • 3AT89C51单片机引脚说明及引脚图

    AT89C51是一种带4K字节闪烁可编程可擦除只读存储器的低电压,高性能CMOS8位微处理器,俗称单片机。该器件采用ATMEL高密度非易失存储器制造技术制造,与工业标准的MCS-51指令集和输出管脚相兼容。由于将多功能8位CPU...

    昨天
  • 51单片机对LCD1602液晶的驱动设计

    51单片机——LCD1602 1、1602液晶读写时序 (1)、读状态 RS=L,R/W=H,E=H。(判断忙完毕后释放总线) (2)、读数据 RS=H,R/W=H,E=H。 (3)、写指令 RS=L,R/W=L,D0~D7=指令码,E=高脉冲 (4)、写数据 RS=H,R/W=L,D0~D...

    昨天
  • 单片机串口如何接收不定长数据的?

    我们在使用其他STM32的单片机的时候,会发现有些困难,会发现常用的方法并不能用,在还没有接收完数据的时候,就解决不了。于是,只能用通用的方法来解决了。 这个通用的方法,其实原理和使用IDLE的原理一样:...

    昨天
  • ARM处理器的选型原则

    鉴于ARM微处理器的众多优点,随着国内外嵌入式应用领域的逐步发展,ARM微处理器必然会获得广泛的重视和应用。但是,由于ARM微处理器有多达十几种的内核结构,几十个芯片生产厂家,以及千变万化的内部功能配置组合,...

    前天
  • 有哪些低功耗设计方法?单片机系统低功耗设计要点介绍

    功耗,已经是一个老生常谈的话题了。对于功耗,大家多多少少有所了解。目前,很多产品的宣传里便带有低功耗噱头。为增进大家对功耗的认识,本文将基于两点介绍功耗:1.低功耗主要设计方法,2.单片机系统低功耗设计...

    前天
  • 8位32位MCU如何选择?如何选择合适的MCU?

    MCU,对于普通人而言,是一个高大上的存在。但是,在工业中,MCU确实常见产品。为增进大家对MCU的认识,本文将基于两点介绍MCU:1.8位MCU和32位MCU如何选择?2.如何选择合适的MCU。如果你对MCU具有兴趣,不妨继续往...

    07-09
  • ARM开发:一 ARM微处理器概述

    1.1ARM-Advanced RISC Machines ARM(Advanced RISC Machines),既可以认为是一个公司的名字,也可以认为是对一类微处理器的通称,还可以认为是一种技术的名字。 1991年ARM公司成立于英国剑桥,主要出售芯片设计技术...

    07-08
  • 分析C51单片机的一些误区和注意事项

    简介:常看见初学者要求使用_at_,这是一种谬误,把C当作ASM看待了。在C中变量的定位是编译器的事情,初学者只要定义变量和变量的作 用域,编译器就把一个固定地址给这个变量。怎么取得这个变量的地址?要用指针。 1) C...

    07-08
  • 51单片机几个延时程序

    简介:51单片机几个精确延时程序:在精确延时的计算当中,最容易让人忽略的是计算循环外的那部分延时,在对时间要求不高的场合,这部分对程序不会造成影响. 一. 500ms延时子程序(晶振12MHz,一个机器周期1us.) 程...

    07-08
  • 总结单片机软件抗干扰的几种办法

    简介:在提高硬件系统抗干扰能力的同时,软件抗干扰以其设计灵活、节省硬件资源、可靠性好越来越受到重视。下面以MCS-51单片机系统为例,对微机系统软件抗干扰方法进行研究。 1、软件抗干扰方法的研究 在工程实践中...

    07-08
  • 基于C51单片机实现汽车座椅自动控制系统的软硬件设计

    引言 随着人们生活水平的提高,对汽车座椅的舒适性要求也越来越高,要求对汽车座椅地调节能够更加简单、方便、快捷。目前,汽车座椅位置的调节多采用基于手动调节方式的机械和电动控制两种方式。汽车座椅位置的调节...

    07-02
  • MCS51单片机程序设计时堆栈的计算方法解析

    用C语言进行MCS51系列单片机程序设计是单片机开发和应用的必然趋势。Keil公司的C51编译器支持经典8051和8051派生产品的版本,通称为Cx51。应该说,Cx51是C语言在MCS51单片机上的扩展,既有C语言的共性,又有它自己...

    07-02
下载排行榜
更多
评测报告
更多
广告