stm32串口中断接收失效原因-HAL库锁死问题

串口使用HAL_UART_Receive_IT(UART_HandleTypeDef *huart, uint8_t *pData, uint16_t Size)使能中断接收,

使用HAL_UART_Transmit(UART_HandleTypeDef *huart, uint8_t *pData, uint16_t Size, uint32_t Timeout)发送数据,

会出现串口能发送,但是中断接收有时会失效问题,原因在于HAL_UART_Receive_IT和HAL_UART_Transmit都使用了

__HAL_LOCK(huart);

__HAL_UNLOCK(huart);

这两个函数,当HAL_UART_Transmit执行过程中执行__HAL_LOCK(huart)加锁,还未__HAL_UNLOCK(huart)解锁时,

被中断接收打断并且调用HAL_UART_Receive_IT来启动下一次接收中断,HAL_UART_Receive_IT在调用__HAL_LOCK(huart)时就会返回HAL_BUSY,

所以HAL_UART_Receive_IT开启接收中断失败,自然接收中断就会失败。

解决方法:注释HAL_UART_Receive_IT或HAL_UART_Transmit的加锁和解锁语句即可。在使用的时候注意线程安全避免冲突就可以了

同理可以想象其它接口函数比如iiC,spi等也会存在此问题,使用的时候也应该注意此问题

具体看下面代码

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

{

uint16_t *tmp;

uint32_t tickstart = 0U;

/* Check that a Tx process is not already ongoing */

if (huart->gState == HAL_UART_STATE_READY)

{

if ((pData == NULL) || (Size == 0U))

{

return HAL_ERROR;

}

/* Process Locked */

__HAL_LOCK(huart);

//未解锁前被中断接收打断,此时中断处理中调用HAL_UART_Receive_IT必然会失败

/* Process Unlocked */

__HAL_UNLOCK(huart);

}

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

{

/* Check that a Rx process is not already ongoing */

if (huart->RxState == HAL_UART_STATE_READY)

{

if ((pData == NULL) || (Size == 0U))

{

return HAL_ERROR;

}

/* Process Locked */

//__HAL_LOCK(huart);//可以注释掉这个加锁和解锁代码来处理这个问题

huart->pRxBuffPtr = pData;

huart->RxXferSize = Size;

huart->RxXferCount = Size;

huart->ErrorCode = HAL_UART_ERROR_NONE;

huart->RxState = HAL_UART_STATE_BUSY_RX;

/* Process Unlocked */

//__HAL_UNLOCK(huart);

/* Enable the UART Parity Error Interrupt */

__HAL_UART_ENABLE_IT(huart, UART_IT_PE);

/* Enable the UART Error Interrupt: (Frame error, noise error, overrun error) */

__HAL_UART_ENABLE_IT(huart, UART_IT_ERR);

/* Enable the UART Data Register not empty Interrupt */

__HAL_UART_ENABLE_IT(huart, UART_IT_RXNE);

return HAL_OK;

}

else

{

return HAL_BUSY;

}

}

  • 4
    点赞
  • 7
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
您可以使用HAL库中的UART接收中断回调函数来实现接收指定字符串的功能。具体步骤如下: 1. 在main函数中初始化UART,并开启接收中断: ``` UART_HandleTypeDef huart1; void MX_USART1_UART_Init(void) { huart1.Instance = USART1; huart1.Init.BaudRate = 115200; huart1.Init.WordLength = UART_WORDLENGTH_8B; huart1.Init.StopBits = UART_STOPBITS_1; huart1.Init.Parity = UART_PARITY_NONE; huart1.Init.Mode = UART_MODE_TX_RX; huart1.Init.HwFlowCtl = UART_HWCONTROL_NONE; huart1.Init.OverSampling = UART_OVERSAMPLING_16; HAL_UART_Init(&huart1); HAL_UART_Receive_IT(&huart1, (uint8_t*)rx_buffer, RX_BUFFER_SIZE); //开启接收中断 } ``` 2. 在接收中断回调函数中,判断接收到的字符串是否为指定字符串,如果是则执行相应的操作,如打印调试信息或控制外设: ``` #define TARGET_STRING "hello" uint8_t rx_buffer[RX_BUFFER_SIZE]; uint8_t rx_data; uint32_t rx_index = 0; void HAL_UART_RxCpltCallback(UART_HandleTypeDef *huart) { if(huart == &huart1) { if(rx_data == '\n') //接收到换行符,表示接收完成 { rx_buffer[rx_index] = '\0'; //字符串结尾添加'\0' rx_index = 0; //重置接收缓存区索引 if(strcmp((char*)rx_buffer, TARGET_STRING) == 0) //判断接收到的字符串是否为指定字符串 { //执行相应的操作,如打印调试信息或控制外设 printf("Received target string!\r\n"); } HAL_UART_Receive_IT(&huart1, &rx_data, 1); //重新开启接收中断 } else { rx_buffer[rx_index] = rx_data; //将接收到的字符存入接收缓存区 rx_index++; if(rx_index >= RX_BUFFER_SIZE) //接收缓存区已满,丢弃之前的数据 { rx_index = 0; } HAL_UART_Receive_IT(&huart1, &rx_data, 1); //继续开启接收中断 } } } ``` 其中,TARGET_STRING为指定字符串,rx_buffer为接收缓存区,RX_BUFFER_SIZE为接收缓存区大小。在接收中断回调函数中,通过判断接收到的字符是否为换行符来判断接收是否完成,如果是则将接收缓存区中的数据与指定字符串进行比较,如果相等则执行相应的操作,并重新开启接收中断。如果接收缓存区已满,则丢弃之前的数据,重置接收缓存区索引,并继续开启接收中断
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值