解决STM32G474单片机_HAL_UART_Transmit_IT()连续发送之问题

在使用STM32G474单片机的HAL库时,使用"在中断服务程序中发送数据"和"在中断程序中接收数据",是一种很常用的方法,特别是RS485通讯中。首次使用,肯定会踩坑。要么出现第一个数据收不到,要么出现连续发送,要么发送象是乱码,总之怪象连篇,搜索网络,发现能解决问题的成功案例很少。

1、串口初始化

//将PA9复用为USART1_TX
//将PA10复用为USART1_RX

UART_HandleTypeDef HardwareUSART1;

uint8_t USART1_RX_Data;

uint8_t USART1_RX_Buffer[USART1_RX_Buffer_Size];
//USART1接收缓冲区数组

uint8_t USART1_RX_Buffer_Load_Index;
//USART1_RX_Buffer[]的装载索引值

uint8_t USART1_TX_Buffer[USART1_TX_Buffer_Size];
//USART1发送缓冲区数组

uint8_t USART1_TX_Buffer_Load_Index;
//USART1_TX_Buffer[]的装载索引值

uint8_t *pUSART1_TX_Buffer;

void USART1_Init(uint32_t baudrate);
void print_USART1_Receive_Data(void);
void Start_USART1_Send_Data(void);

void USART1_Init(uint32_t baudrate)

{

HardwareUSART1.Instance = USART1;

HardwareUSART1.Init.BaudRate = baudrate; //波特率

HardwareUSART1.Init.WordLength = UART_WORDLENGTH_8B; //字长为8位数据格式

HardwareUSART1.Init.StopBits = UART_STOPBITS_1; //一个停止位

HardwareUSART1.Init.Parity = UART_PARITY_NONE; //无奇偶校验位

HardwareUSART1.Init.Mode = UART_MODE_TX_RX; //收发模式

HardwareUSART1.Init.HwFlowCtl = UART_HWCONTROL_NONE; //无硬件流控

HardwareUSART1.Init.OverSampling = UART_OVERSAMPLING_16; //过采样率

HardwareUSART1.Init.OneBitSampling = UART_ONE_BIT_SAMPLE_DISABLE;

HardwareUSART1.AdvancedInit.AdvFeatureInit = UART_ADVFEATURE_NO_INIT;

HardwareUSART1.AdvancedInit.AutoBaudRateEnable = UART_ADVFEATURE_AUTOBAUDRATE_DISABLE;

HardwareUSART1.AdvancedInit.AutoBaudRateMode = UART_ADVFEATURE_AUTOBAUDRATE_ONSTARTBIT;

HardwareUSART1.Init.ClockPrescaler = UART_PRESCALER_DIV1;

HardwareUSART1.FifoMode=UART_FIFOMODE_DISABLE;

if (HAL_UART_Init(&HardwareUSART1) != HAL_OK)

{

Error_Handler();

}

__HAL_UART_CLEAR_FLAG(&HardwareUSART1, UART_CLEAR_TCF);

//Transmission Complete Clear Flag

__HAL_UART_ENABLE_IT(&HardwareUSART1,UART_IT_RXNE);
//使能USART1接收到数据时产生中断
//"UART read data register" not empty interruption

__HAL_UART_DISABLE_IT(&HardwareUSART1, UART_IT_TXE);
//串口发送数据时,不使能"串口发送数据寄存器为空"产生中断(位TXE=0)
//Disable the UART Transmit Complete Interrupt

__HAL_UART_DISABLE_IT(&HardwareUSART1,UART_IT_TC);
//串口发送数据时,不使能"串口发送完成"产生中断(位TC=1)
// __HAL_UART_ENABLE_IT(&HardwareUSART1, UART_IT_TXE);
// __HAL_UART_ENABLE_IT(&HardwareUSART1,UART_IT_TC);
//使能USART1发送完成时产生中断

HAL_NVIC_EnableIRQ(USART1_IRQn);

HAL_NVIC_SetPriority(USART1_IRQn,8,0);
//设置NVIC中断分组4:4位抢占优先级,0位响应优先级
//选择中断优先级组4,即抢占优先级为4位,取值为0~15,响应优先级组为0位,取值为0

USART1_RX_Buffer_Load_Index=0;

}

//HAL_UART_Init()执行时,会先调用HAL_UART_MspInit()
void HAL_UART_MspInit(UART_HandleTypeDef* uartHandle)

{

GPIO_InitTypeDef GPIO_InitStruct = {0};

RCC_PeriphCLKInitTypeDef PeriphClkInit = {0};

if(uartHandle->Instance==USART1)

{

PeriphClkInit.PeriphClockSelection = RCC_PERIPHCLK_USART1;

PeriphClkInit.Usart1ClockSelection = RCC_USART1CLKSOURCE_PCLK2;

if (HAL_RCCEx_PeriphCLKConfig(&PeriphClkInit) != HAL_OK)

{ //HAL_RCCEx_PeriphCLKConfig()初始化USART1外设时钟
//Initializes the peripherals clocks

Error_Handler();

}

__HAL_RCC_USART1_CLK_ENABLE(); //使能USART1外设时钟

__HAL_RCC_GPIOA_CLK_ENABLE(); //使能GPIOA外设时钟

GPIO_InitStruct.Pin = GPIO_PIN_9|GPIO_PIN_10;//选择引脚编号9和10

GPIO_InitStruct.Mode = GPIO_MODE_AF_PP; //复用功能推挽模式

GPIO_InitStruct.Pull = GPIO_NOPULL; //不用上拉

GPIO_InitStruct.Speed = GPIO_SPEED_FREQ_LOW; //引脚速度为低速

GPIO_InitStruct.Alternate = GPIO_AF7_USART1; //将引脚复用为USART1

HAL_GPIO_Init(GPIOA, &GPIO_InitStruct);

}

}

//HAL_UART_DeInit()执行时,会先调用HAL_UART_MspDeInit()
void HAL_UART_MspDeInit(UART_HandleTypeDef* uartHandle)

{

if(uartHandle->Instance==USART1)

{

__HAL_RCC_USART1_CLK_DISABLE();//不使能USART1外设时钟

HAL_GPIO_DeInit(GPIOA, GPIO_PIN_9|GPIO_PIN_10);

}

}

2、启动发送

void Start_USART1_Send_Data(void)

{

char i;

uint16_t k;

for(i=0;i<USART1_RX_Buffer_Size;i++)//串口接送测试

{

if(USART1_RX_Buffer[i-1]=='\r'&&USART1_RX_Buffer[i]=='\n')//收到结束符号

{

USART1_RX_Buffer_Load_Index = 0;

memset(USART1_TX_Buffer,0,USART1_TX_Buffer_Size);

strcpy((char*)USART1_TX_Buffer,(char*)USART1_RX_Buffer);

memset(USART1_RX_Buffer,0,USART1_RX_Buffer_Size);

k=strlen((char*)USART1_TX_Buffer);

USART1_TX_Buffer_Load_Index = k;

__HAL_UART_ENABLE_IT(&HardwareUSART1,UART_IT_TC);

pUSART1_TX_Buffer=USART1_TX_Buffer;//启动发送

}

}

}

3、接收函数

//UART_RxISR_8BIT()会调用HAL_UART_RxCpltCallback()处理接收到的数据
//HAL_UART_Receive_IT()会调用UART_RxISR_8BIT()
//在中断中调用HAL_UART_Receive_IT()
void HAL_UART_RxCpltCallback(UART_HandleTypeDef *huart)

{

USART1_RX_Buffer[USART1_RX_Buffer_Load_Index]=USART1_RX_Data;

USART1_RX_Buffer_Load_Index++;

if(USART1_RX_Buffer_Load_Index>=USART1_RX_Buffer_Size-2)

{//USART1_RX_Buffer[]防止溢出

USART1_RX_Buffer[USART1_RX_Buffer_Load_Index]='\0';

USART1_RX_Buffer_Load_Index=0;

}

if(USART1_RX_Buffer[USART1_RX_Buffer_Load_Index-2]=='\r'&&USART1_RX_Buffer[USART1_RX_Buffer_Load_Index-1]=='\n')

{//收到"\r\n"结束符号

USART1_RX_Buffer[USART1_RX_Buffer_Load_Index]='\0';

USART1_RX_Buffer_Load_Index++;

}

}

4、接收中断函数

void USART1_IRQHandler(void)

{

HAL_UART_IRQHandler(&HardwareUSART1);

HAL_UART_Receive_IT(&HardwareUSART1, &USART1_RX_Data, sizeof(USART1_RX_Data));

//从USART1接收一个字节

HAL_UART_Transmit_IT(&HardwareUSART1, pUSART1_TX_Buffer, USART1_TX_Buffer_Load_Index);

pUSART1_TX_Buffer=NULL;//USART1_TX_Buffer[]被发送完成后,停止发送
// USART1_TX_Buffer_Load_Index=0;//USART1_TX_Buffer[]被发送完成后,停止发送

//实测,发现HAL_UART_Transmit_IT()会在中断中将**USART1_TX_Buffer[]**全部发完,才会退出HAL_UART_Transmit_IT();

}

HAL_StatusTypeDef HAL_UART_Transmit_IT(UART_HandleTypeDef *huart, uint8_t *pData, uint16_t Size)

在STM32G474的HAL库中,HAL_UART_Transmit_IT()会将所有数据发送完了,才会退出中断,具体HAL库是怎么做到的,不必浪费时间分析。若要防止HAL_UART_Transmit_IT()连续发送,要么将pData设置为NULL,要么将Size设置为0。否则,串口会不停往外发送数据,十分恼火。搞了好几天才搞好。

总结:HAL库是在没有办法的情况下,不得不用。让HAL_UART_Transmit_IT()停下来,有点耍流氓,还有没有更好的办法?

测试结果如下:

相关推荐
幼儿园老大*44 分钟前
【系统架构】如何设计一个秒杀系统?
java·经验分享·后端·微服务·系统架构
电工小王(全国可飞)1 小时前
STM32F407 内部参考电压校准实现 HAL库
stm32·单片机·嵌入式硬件
gyeolhada1 小时前
计算机组成原理(计算机系统3)--实验七:新增指令实验
单片机·嵌入式硬件
嵌入式小强工作室2 小时前
STM32更新程序OTA
stm32·单片机·嵌入式硬件
Pandaconda3 小时前
【Golang 面试题】每日 3 题(三十九)
开发语言·经验分享·笔记·后端·面试·golang·go
fwjzm3 小时前
SMT32 FatFs,RTC,记录文件操作时间
stm32
gyeolhada4 小时前
计算机组成原理(计算机系统3)--实验八:处理器结构拓展实验
java·前端·数据库·嵌入式硬件
andylauren11 小时前
(5)STM32 USB设备开发-USB键盘
stm32·嵌入式硬件·计算机外设
Ronin-Lotus12 小时前
嵌入式硬件篇---ADC模拟-数字转换
笔记·stm32·单片机·嵌入式硬件·学习·低代码·模块测试
Jason Yan12 小时前
【经验分享】ARM Linux-RT内核实时系统性能评估工具
linux·arm开发·经验分享