STM32单片机HAL库串口接收失效问题的解决

当收发频繁,且发送同时产生接收中断,HAL库函数的收发函数互锁以及ORE错误中断造成接收不能重启
摘要由CSDN通过智能技术生成

      近日一个项目,发现STM32单片机的串口在运行 一段时间后就不再能继续收到数据,但发送又正常的情况,我的串口采用的是阻塞式发送和DMA接收+空闲中断的处理方式。

    经过查询资料和测试,发现第1种原因是正在发送数据的情况下又收到接收空闲中断,造成了空闲中断处理出错,未能正常启动DMA接收:

1)串口发送采用的函数HAL_UART_Transmit里有执行 __HAL_LOCK(huart)。

2)此时,如有接收空闲中断,则开始执行:

    a)停止DMA接收: HAL_UART_DMAStop(huart)

    b)获取接收数据

    c)重启DMA操作:HAL_UART_Receive_DMA

       该函数又执行一次 __HAL_LOCK(huart),根据该宏定义可知,产生阻塞,返回HAL_BUSY.因而未执行后续代码,如UART_Start_Receive_DMA等,所以串口的DMA接收和中断都未被打开。后续不再接收。

解决办法:

   1)每次发送时检查上次发送的时间,做到间隔2ms以上,尽量避免串口发送时又正好收到对方的数据。

   2)不再调用HAL_UART_Transmit函数。

自己编写如下串口发送函数:

 static uint32_t last_send_time = 0;

HAL_StatusTypeDef USART_SendData(UART_HandleTypeDef *huart, uint8_t *pData, uint8_t Size, uint32_t Timeout)

{

  • 4
    点赞
  • 22
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
你好!关于STM32单片机HAL库串口通信,我可以为你提供一些基本的信息。 HAL(Hardware Abstraction Layer)库是STMicroelectronics为STM32单片机系列开发的一套硬件抽象层库,它提供了一套统一的API接口,方便开发者在不同的STM32系列芯片上进行开发。其中包含了串口通信功能。 首先,你需要在工程中包含相应的HAL库文件,并初始化串口相关的引脚和时钟。然后,你可以使用HAL库提供的函数来进行串口的配置和操作。 以下是一个简单的示例代码,展示了如何使用HAL库进行串口通信: ```c #include "stm32f4xx_hal.h" UART_HandleTypeDef huart2; void SystemClock_Config(void); static void MX_GPIO_Init(void); static void MX_USART2_UART_Init(void); int main(void) { HAL_Init(); SystemClock_Config(); MX_GPIO_Init(); MX_USART2_UART_Init(); uint8_t txData[] = "Hello, World!"; uint8_t rxData[20]; while (1) { HAL_UART_Transmit(&huart2, txData, sizeof(txData), HAL_MAX_DELAY); HAL_UART_Receive(&huart2, rxData, sizeof(rxData), HAL_MAX_DELAY); } } void SystemClock_Config(void) { RCC_OscInitTypeDef RCC_OscInitStruct = {0}; RCC_ClkInitTypeDef RCC_ClkInitStruct = {0}; __HAL_RCC_PWR_CLK_ENABLE(); __HAL_PWR_VOLTAGESCALING_CONFIG(PWR_REGULATOR_VOLTAGE_SCALE1); RCC_OscInitStruct.OscillatorType = RCC_OSCILLATORTYPE_HSI; RCC_OscInitStruct.HSIState = RCC_HSI_ON; RCC_OscInitStruct.HSICalibrationValue = RCC_HSICALIBRATION_DEFAULT; RCC_OscInitStruct.PLL.PLLState = RCC_PLL_ON; RCC_OscInitStruct.PLL.PLLSource = RCC_PLLSOURCE_HSI; RCC_OscInitStruct.PLL.PLLM = 8; RCC_OscInitStruct.PLL.PLLN = 168; RCC_OscInitStruct.PLL.PLLP = RCC_PLLP_DIV2; RCC_OscInitStruct.PLL.PLLQ = 4; if (HAL_RCC_OscConfig(&RCC_OscInitStruct) != HAL_OK) { Error_Handler(); } if (HAL_PWREx_EnableOverDrive() != HAL_OK) { Error_Handler(); } RCC_ClkInitStruct.ClockType = RCC_CLOCKTYPE_HCLK | RCC_CLOCKTYPE_SYSCLK | RCC_CLOCKTYPE_PCLK1 | RCC_CLOCKTYPE_PCLK2; RCC_ClkInitStruct.SYSCLKSource = RCC_SYSCLKSOURCE_PLLCLK; RCC_ClkInitStruct.AHBCLKDivider = RCC_SYSCLK_DIV1; RCC_ClkInitStruct.APB1CLKDivider = RCC_HCLK_DIV4; RCC_ClkInitStruct.APB2CLKDivider = RCC_HCLK_DIV2; if (HAL_RCC_ClockConfig(&RCC_ClkInitStruct, FLASH_LATENCY_5) != HAL_OK) { Error_Handler(); } } void MX_GPIO_Init(void) { GPIO_InitTypeDef GPIO_InitStruct = {0}; __HAL_RCC_GPIOA_CLK_ENABLE(); HAL_GPIO_WritePin(GPIOA, GPIO_PIN_5, GPIO_PIN_RESET); GPIO_InitStruct.Pin = GPIO_PIN_5; GPIO_InitStruct.Mode = GPIO_MODE_OUTPUT_PP; GPIO_InitStruct.Pull = GPIO_NOPULL; GPIO_InitStruct.Speed = GPIO_SPEED_FREQ_LOW; HAL_GPIO_Init(GPIOA, &GPIO_InitStruct); } void MX_USART2_UART_Init(void) { huart2.Instance = USART2; huart2.Init.BaudRate = 115200; huart2.Init.WordLength = UART_WORDLENGTH_8B; huart2.Init.StopBits = UART_STOPBITS_1; huart2.Init.Parity = UART_PARITY_NONE; huart2

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值