这里只记录注意要点:

1,要开启串口 全局中断 和对应的接收DMA 中断,两个中断必须同时开

2,裸机程序需要在主循环外调用一次 这个函数

 HAL_UARTEx_ReceiveToIdle_DMA(&huart2, rx_buff, BUFF_SIZE);

3,要在串口中断处理函数中 添加这个函数 


 HAL_UARTEx_ReceiveToIdle_DMA(&huart2, rx_buff, BUFF_SIZE);

4,重写空闲中断回调函数 ,它是个弱函数,需要程序员自己实现

void HAL_UARTEx_RxEventCallback(UART_HandleTypeDef * huart, uint16_t Size)

在这个函数中,可以做一些事情,比如记录收到了多少个字节数据,或者对接收数据处理。

5,HAL_UARTEx_ReceiveToIdle_DMA 函数的作用:

        1,把接收类型设置成HAL_UART_RECEPTION_TOIDLE,
        2,调用开启DMA接收函数;
        3,调用完再清除一次IDLEF标志位
        4,然后再设置IDLEIE标志位

6,上面第5点可以看出,这个函数需要反复调用,一般放在串口中断函数里面就可以了,不能放在回调函数里,因为一旦出现异常没有进中断,那么在没有其它地方调用HAL_UARTEx_ReceiveToIdle_DMA的话 ,就再也不进空闲中断了,DMA 也不接收了

7,如果是了 OS , 那么可以用这种办法:在任务中反复调用这个函数:HAL_UARTEx_ReceiveToIdle_DMA ,而无需在中断函数里调用 ,不过本质上还是一样的 :

while(1)
{
    status = HAL_UARTEx_ReceiveToIdle_DMA(&huart4, uart4_rx_buffer, UART_BUFFER_SIZE);
		while (status != HAL_OK) {
			osDelay(1);
			status =     HAL_UARTEx_ReceiveToIdle_DMA(&huart4,uart4_rx_buffer,UART_BUFFER_SIZE);
		}
		
    osSemaphoreAcquire(sem_uart4_rxHandle, osWaitForever); //这里是等待空闲中断到来
		HAL_UART_AbortReceive(&huart4);


     //do sometings 

}




...

可以在 空闲中断函数中添加 发送信号量

	if(huart->Instance == UART4) {
			uart4_rx_len = Size;
			osSemaphoreRelease(sem_uart4_rxHandle);
	}

本站无任何商业行为
个人在线分享 » STM32串口DMA 空闲中断使用笔记
E-->