关于STM32的串口用DMA接收

306 阅读3分钟

原来都用CUBE直接生成使用DMA接收的串口接收程序,现在自己写调用函数才发现每个DMA通道对应的有相应的功能

如下图           对DMA比较形象的描述

配置代码:一般在发送中断里面清除标志位,接收一般是串口空闲中断和DMA接收一起完成,即先关闭接收中断,再使能空闲中断(接收到一帧数据就进入空闲中断)USART_ITConfig(USART3, USART_IT_RXNE, DISABLE);    
USART_ITConfig(USART3, USART_IT_IDLE, ENABLE);    //使能空闲中断

要注意的坑:在串口初始化中如果有使能接收中断,再在dma初始化中失能接收中断,那么这两个中断时间尽量间隔时间短一些,否则容易一直进串口中断,而串口中断只进行了清除空闲中断标志位而没有清除串口接收中断,所以会卡死在里面

//这里的传输形式是固定的,这点要根据不同的情况来修改
//从存储器->外设模式/8位数据宽度/存储器增量模式
//DMA_CHx:DMA通道CHx
//cpar:外设地址
//cmar:存储器地址
//cndtr:数据传输量 
void _dma_Recv_init(DMA_Channel_TypeDef* DMA_CHx,u32 cpar,u32 cmar,u16 cndtr)
{ 
	RCC_AHBPeriphClockCmd(RCC_AHBPeriph_DMA1, ENABLE);	//使能DMA传输		
	DMA_DeInit(DMA_CHx);   
	DMA_InitStructure.DMA_PeripheralBaseAddr = cpar;  //DMA外设基地址	
	DMA_InitStructure.DMA_MemoryBaseAddr = cmar;  //DMA内存基地址	
	DMA_InitStructure.DMA_DIR = DMA_DIR_PeripheralSRC;  //数据传输方向	DMA_DIR_PeripheralDST是双相
	DMA_InitStructure.DMA_BufferSize = cndtr;  //DMA通道的DMA缓存的大小	
	DMA_InitStructure.DMA_PeripheralInc = DMA_PeripheralInc_Disable;  //外设地址寄存器不变	
	DMA_InitStructure.DMA_MemoryInc = DMA_MemoryInc_Enable;  //内存地址寄存器递增	
	DMA_InitStructure.DMA_PeripheralDataSize = DMA_PeripheralDataSize_Byte;  //数据宽度为8位	
	DMA_InitStructure.DMA_MemoryDataSize = DMA_MemoryDataSize_Byte; //数据宽度为8位	
	DMA_InitStructure.DMA_Mode = DMA_Mode_Normal;  //工作在正常模式	
	DMA_InitStructure.DMA_Priority = DMA_Priority_Medium; //DMA通道 x拥有中优先级 	
	DMA_InitStructure.DMA_M2M = DMA_M2M_Disable;  //DMA通道x没有设置为内存到内存传输	
	DMA_Init(DMA_CHx, &DMA_InitStructure);  //根据DMA_InitStruct中指定的参数初始化DMA的通道USART1_Tx_DMA_Channel所标识的寄存器	
	DMA_ITConfig(DMA_CHx, DMA_IT_TC, ENABLE);
	DMA_Cmd(DMA_CHx,ENABLE);  
	
}

void _dma_Send_init(DMA_Channel_TypeDef* DMA_CHx,u32 cpar,u32 cmar,u16 cndtr)
{	
	RCC_AHBPeriphClockCmd(RCC_AHBPeriph_DMA1, ENABLE);	//使能DMA传输  
	DMA_DeInit(DMA_CHx);   //将DMA的通道1寄存器重设为缺省值	
	DMA1_MEM_LEN=cndtr;	
	DMA_InitStructure.DMA_PeripheralBaseAddr = cpar;  //DMA外设基地址
	DMA_InitStructure.DMA_MemoryBaseAddr = cmar;  //DMA内存基地址	
	DMA_InitStructure.DMA_DIR = DMA_DIR_PeripheralDST;  //数据传输方向,从内存读取发送到外设	
	DMA_InitStructure.DMA_BufferSize = cndtr;  //DMA通道的DMA缓存的大小	
	DMA_InitStructure.DMA_PeripheralInc = DMA_PeripheralInc_Disable;  //外设地址寄存器不变	
	DMA_InitStructure.DMA_MemoryInc = DMA_MemoryInc_Enable;  //内存地址寄存器递增	
	DMA_InitStructure.DMA_PeripheralDataSize = DMA_PeripheralDataSize_Byte;  //数据宽度为8位	
	DMA_InitStructure.DMA_MemoryDataSize = DMA_MemoryDataSize_Byte; //数据宽度为8位	
	DMA_InitStructure.DMA_Mode = DMA_Mode_Normal;  //工作在正常模式	
	DMA_InitStructure.DMA_Priority = DMA_Priority_High; //DMA通道 x拥有中优先级 
	DMA_InitStructure.DMA_M2M = DMA_M2M_Disable;  //DMA通道x没有设置为内存到内存传输
	DMA_Init(DMA_CHx, &DMA_InitStructure); 
  DMA_ITConfig(DMA_CHx, DMA_IT_TC, ENABLE);
	
}
void _dma_config(void)
{	
	USART_ITConfig(USART3, USART_IT_TC,DISABLE);	
	USART_ITConfig(USART3, USART_IT_RXNE, DISABLE);	
	USART_ITConfig(USART3, USART_IT_IDLE, ENABLE);	//使能空闲中断
	_dma_Recv_init(DMA1_Channel3,(u32)&USART3->DR,(u32)_dma_RXBuff_3,DMA_REC_LEN);//RX	
	_dma_Send_init(DMA1_Channel2,(u32)&USART3->DR,(u32)_dma_TXBuff_3,DMA_REC_LEN);//TX
	USART_DMACmd(USART3,USART_DMAReq_Rx,ENABLE);//使能DMA收
	USART_DMACmd(USART3, USART_DMAReq_Tx, ENABLE);	//使能DMA发
	
	USART_ITConfig(USART2, USART_IT_TC,DISABLE);	
	USART_ITConfig(USART2, USART_IT_RXNE, DISABLE);	
	USART_ITConfig(USART2, USART_IT_IDLE, ENABLE);	//使能空闲中断
	_dma_Recv_init(DMA1_Channel6,(u32)&USART2->DR,(u32)_dma_RXBuff_2,DMA_REC_LEN);//RX	
	_dma_Send_init(DMA1_Channel7,(u32)&USART2->DR,(u32)_dma_TXBuff_2,DMA_REC_LEN);//TX
	USART_DMACmd(USART2,USART_DMAReq_Rx,ENABLE);//使能DMA收
	USART_DMACmd(USART2, USART_DMAReq_Tx, ENABLE);	//使能DMA发
	
	USART_ITConfig(USART1, USART_IT_TC,DISABLE);	
	USART_ITConfig(USART1, USART_IT_RXNE, DISABLE);	
	USART_ITConfig(USART1, USART_IT_IDLE, ENABLE);	//使能空闲中断
	_dma_Recv_init(DMA1_Channel5,(u32)&USART1->DR,(u32)_dma_RXBuff_1,DMA_REC_LEN);//RX	
	_dma_Send_init(DMA1_Channel4,(u32)&USART1->DR,(u32)_dma_TXBuff_1,DMA_REC_LEN);//TX
	USART_DMACmd(USART1,USART_DMAReq_Rx,ENABLE);//使能DMA收
	USART_DMACmd(USART1, USART_DMAReq_Tx, ENABLE);	//使能DMA发
}
void DMA1_Channel2_IRQHandler(void)
{
		if(DMA_GetITStatus(DMA1_IT_TC2)!=RESET)	//判断通道2传输完成
		{
			DMA_ClearFlag(DMA1_IT_TC2);//清除通道2传输完成标志
		}
}

注意:如果多个串口都用到DMA收发,那么收发的缓存区要单独分开