STM32CubeMX 配置串口通信 HAL库

一、STM32CubeMX 配置串口

每个外设生成独立的 '.c/.h' 文件

不勾:所有初始化代码都生成在 main.c

勾选:初始化代码生成在对应的外设文件。 如 GPIO 初始化代码生成在 gpio.c 中。

二、重写fputc函数

cpp 复制代码
​
#include <stdio.h>
 
 #ifdef __GNUC__
     #define PUTCHAR_PROTOTYPE int _io_putchar(int ch)
 #else
     #define PUTCHAR_PROTOTYPE int fputc(int ch, FILE *f)
 #endif /* __GNUC__*/
 
 /******************************************************************
     *@brief  Retargets the C library printf  function to the USART.
     *@param  None
     *@retval None
 ******************************************************************/
 PUTCHAR_PROTOTYPE
 {
     HAL_UART_Transmit(&huart1, (uint8_t *)&ch,1,0xFFFF);
     return ch;
 }

​

三、配置接收中断函数

cpp 复制代码
​
#define RXBUFFERSIZE    1                       /* 缓存大小 */
uint8_t  g_usart_rx_buf[200];  //接收缓冲,最大200个字节
uint16_t g_usart_rx_len;   	   //接收长度
uint8_t g_usart_rx_flag=0;	   //接收完成标志
uint8_t g_rx_buffer[RXBUFFERSIZE];       /* HAL库USART接收Buffer */

在串口初始化函数中使能接收中断
/* 该函数会开启接收中断:标志位UART_IT_RXNE,并且设置接收缓冲以及接收缓冲接收最大数据量 */
HAL_UART_Receive_IT(&huart1, (uint8_t *)g_rx_buffer, RXBUFFERSIZE);

void HAL_UART_RxCpltCallback(UART_HandleTypeDef *huart)
{  
	if(huart->Instance == USART1)     
    {
		g_usart_rx_buf[g_usart_rx_len] = g_rx_buffer[0] ;
        g_usart_rx_len++;
		if(g_rx_buffer[0]==0x0a)
		{
			g_usart_rx_flag=1;
		}
		HAL_UART_Receive_IT(&huart1, (uint8_t *)g_rx_buffer, RXBUFFERSIZE);
	}
}

​

四、串口空闲中断接收数据

cpp 复制代码
__HAL_UART_ENABLE_IT(&huart1,UART_IT_IDLE);//空闲中断
cpp 复制代码
#define USART1_MAX_RECV_LEN		1000				//最大接收缓存字节数
char USART1_RX_BUF[USART1_MAX_RECV_LEN]; 				//接收缓冲,最大USART3_MAX_RECV_LEN个字节
unsigned short USART1_RX_STA=0;   	



/* USER CODE BEGIN 1 */
void USART1_IRQHandler(void)
{
	uint8_t res = 0;
	
	//接收中断
	if(__HAL_UART_GET_FLAG(&huart1,UART_FLAG_RXNE) != RESET)
	{
		HAL_UART_Receive(&huart1,&res,1,1000);
		//将数据放入缓冲区
		if( (USART1_RX_STA&0x7fff) < USART1_MAX_RECV_LEN)
		{
			USART1_RX_BUF[USART1_RX_STA] = res;
			USART1_RX_STA++;
		}
		
		__HAL_UART_CLEAR_FLAG(&huart1,UART_FLAG_RXNE);
	}
	
	//空闲中断
	if(__HAL_UART_GET_FLAG(&huart1,UART_FLAG_IDLE) != RESET)
	{
		//一帧数据接收完成
		//USART1_IdleCallback(USART1_RX_BUF,USART1_RX_STA&0x7fff);
		USART1_RX_BUF[ USART1_RX_STA &0x7fff] = 0;
		USART1_RX_STA |= 1 << 15;  
		
		__HAL_UART_CLEAR_IDLEFLAG(&huart1);
	}
}
cpp 复制代码
	if(USART1_RX_STA& 0x8000)
	{
		printf("%s\r\n",USART1_RX_BUF);
		USART1_RX_STA=0;
	}
	

五、串口乱码原因之一

乱码一般是配置串口波特率和上位机串口波特率不一致所致,如果两者一致,则说明系统时钟有问题,比如下面的外部晶振不匹配

开发板上外部时钟的晶振与代码中的外部时钟的晶振不一致,使用cubemx配置时钟时要格外注意,选好芯片之后再看外部晶振是啥,不然随便设置可能乱码

相关推荐
iCxhust5 小时前
8255 PORTC 按键输入测试
单片机·嵌入式硬件·微机原理
d111111111d8 小时前
在STM32函数指针是什么,怎么使用还有典型应用场景。
笔记·stm32·单片机·嵌入式硬件·学习·算法
♛识尔如昼♛9 小时前
计算机组成原理(21) 第五章 - 总线性能指标
单片机·嵌入式硬件
快乐的划水a10 小时前
windows用户态到内核态
stm32·单片机·嵌入式硬件
硬汉嵌入式10 小时前
将STM32H7的SPI MISO和MOSI短接回环测试配置问题也是非常方便的
stm32·miso·h7-tool·spi回环·mosi
Hy行者勇哥10 小时前
从零搭建小智 AI 音箱 MCP 开发环境:自定义智能家居控制技能实战指南
人工智能·嵌入式硬件·硬件工程·智能家居
richxu2025100112 小时前
嵌入式学习之路>单片机核心原理篇>(11) 存储器(Flash & SRam)
单片机·嵌入式硬件·学习
@good_good_study12 小时前
STM32 TIM编码器模式配置函数及实验
stm32·单片机
云山工作室13 小时前
基于单片机的客车载客状况自动检测系统设计(论文+源码)
stm32·单片机·嵌入式硬件·毕业设计
Rorsion14 小时前
第二章(2.5):微控制器8051的硬件结构---时钟、复位和MCU工作方式
单片机·嵌入式硬件·备考ing