STM32 DMA传输出错的防错机制

一、DMA  中断

对于每个 DMA 数据流,可在发生以下事件时产生中断:
● 达到半传输(每次传输都会触发,属于正常触发)
● 传输完成
● 传输错误
● FIFO 错误(上溢、下溢或 FIFO 级别错误)
● 直接模式错误
可以使用单独的中断使能位以实现灵活性,如 表 43 所示。

上诉为F4系列,F1系列中断只有三个,所以只需处理三个即可。见下图:

二、常规操作

我们正常使用DMA传输,比如串口只需要在DMA中断中判断传输完成(TCIF)中断标志位,并做出相应程序处理以及清除传输完成(TCIF)标志位即可。代码如下:

void DMA2_Stream7_IRQHandler(void)
{
	if(DMA_GetITStatus(DMA2_Stream7,DMA_IT_TCIF7) != RESET)
	{
		DMA_ClearITPendingBit(DMA2_Stream7,DMA_FLAG_TCIF7); 
		DMA_Cmd(DMA2_Stream7, DISABLE);
		DMA_ClearFlag(DMA2_Stream7,DMA_FLAG_TCIF7); 
		USART_DMACmd(USART1, USART_DMAReq_Tx, DISABLE); 
		RS485_LI_EN = 0;
		NotGet_LiCMD_count = 0;
	}
}

三、安全操作

测试过程中对目标板进行了强干扰试验,发现串口失效。经过仿真查找,得出结论为:强干扰下DMA传输出错,即产生出错的中断请求,然而常规操作中并没有清除错误中断标志位,导致DMA失效,串口无法正常工作。以下代码做出全部的中断标志位清除操作,经验证可行,并提高程序稳定性。

void DMA1_Stream6_IRQHandler(void)
{  
	//if(DMA_GetFlagStatus(DMA1_Stream6,DMA_FLAG_TCIF6) != 	RESET)//DMA_FLAG_TCIF6
	if(DMA_GetITStatus(DMA1_Stream6,DMA_IT_TCIF6) != 	RESET)//DMA_FLAG_TCIF6
	{
		DMA_ClearITPendingBit(DMA1_Stream6,DMA_IT_TCIF6); 
		DMA_Cmd(DMA1_Stream6, DISABLE);
		DMA_ClearFlag(DMA1_Stream6,DMA_FLAG_TCIF6); 
		USART_DMACmd(USART2, USART_DMAReq_Tx, DISABLE); 
		NotGet_INVCMD_count = 0;
	}
	if(DMA_GetITStatus(DMA1_Stream6,DMA_IT_HTIF6) != 	RESET)
	{
		DMA_ClearITPendingBit(DMA1_Stream6,DMA_IT_HTIF6); 
		DMA_ClearFlag(DMA1_Stream6,DMA_FLAG_HTIF6); 
	}
	if(DMA_GetITStatus(DMA1_Stream6,DMA_IT_TEIF6) != 	RESET)
	{
		DMA_ClearITPendingBit(DMA1_Stream6,DMA_IT_TEIF6); 
		DMA_ClearFlag(DMA1_Stream6,DMA_FLAG_TEIF6); 
	}
	if(DMA_GetITStatus(DMA1_Stream6,DMA_IT_DMEIF6) != 	RESET)
	{
		DMA_ClearITPendingBit(DMA1_Stream6,DMA_IT_DMEIF6); 
		DMA_ClearFlag(DMA1_Stream6,DMA_FLAG_DMEIF6); 
	}
			
	if(DMA_GetITStatus(DMA1_Stream6,DMA_IT_FEIF6) != 	RESET)
	{
		DMA_ClearITPendingBit(DMA1_Stream6,DMA_IT_FEIF6); 
		DMA_ClearFlag(DMA1_Stream6,DMA_FLAG_FEIF6); 
	}
}

猜你喜欢

转载自blog.csdn.net/qq_27747359/article/details/93776435