STM32F0系列串口DMA收发数据

关于STM32F0系列串口DMA收发数据详解

这里用的库函数版本,芯片型号为stm32f030c8t6.在用到串口DMA时,要按以下几个步骤进行。

1.确定使用的串口号,这里,我用的是usart2,对应复用的引脚是PA2,PA3。再根据数据手册查询其对应DMA通道。
在这里插入图片描述 对应使用是DMA1_CH4(USART2_TX)与DMA1_CH5(USART2_RX)

2.先进行串口初始化配置

void USART2_Init()
{
	GPIO_InitTypeDef  GPIO_InitStructure;
	USART_InitTypeDef USART_InitStructure;
	NVIC_InitTypeDef  NVIC_InitStructure;
	
	
	RCC_AHBPeriphClockCmd(RCC_AHBPeriph_GPIOA, ENABLE);
	RCC_APB1PeriphClockCmd(RCC_APB1Periph_USART2, ENABLE);
	
	GPIO_PinAFConfig(GPIOA, GPIO_PinSource2, GPIO_AF_1);		//Tx
	GPIO_PinAFConfig(GPIOA, GPIO_PinSource3, GPIO_AF_1);		//Rx
	GPIO_InitStructure.GPIO_Pin   = GPIO_Pin_2 |GPIO_Pin_3;
	GPIO_InitStructure.GPIO_Mode  = GPIO_Mode_AF;	
	GPIO_InitStructure.GPIO_OType = GPIO_OType_PP;
	GPIO_InitStructure.GPIO_PuPd  = GPIO_PuPd_UP;
	GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
	GPIO_Init(GPIOA, &GPIO_InitStructure);
    //配置串口中断优先级
	NVIC_InitStructure.NVIC_IRQChannel                   = USART2_IRQn; 
	NVIC_InitStructure.NVIC_IRQChannelPriority           = 2;
	NVIC_InitStructure.NVIC_IRQChannelCmd                = ENABLE;
	NVIC_Init(&NVIC_InitStructure);

    USART_DeInit(USART2); 
	USART_InitStructure.USART_BaudRate            = bound;                              //波特率
	USART_InitStructure.USART_WordLength          = USART_WordLength_8b;                //数据长度8
	USART_InitStructure.USART_StopBits            = USART_StopBits_1;                   //1个停止位
	USART_InitStructure.USART_Parity              = USART_Parity_No;                    //无奇偶校验
	USART_InitStructure.USART_HardwareFlowControl = USART_HardwareFlowControl_None;     //无硬件流控制
	USART_InitStructure.USART_Mode                = USART_Mode_Tx | USART_Mode_Rx;
	USART_Init(USART2, &USART_InitStructure);
			
	USART_ITConfig(USART2, USART_IT_IDLE, ENABLE);//使能串口空闲中断
	USART_ClearITPendingBit(USART2, USART_IT_IDLE);//清除串口空闲中断标志位
	USART_Cmd(USART2, ENABLE);//使能串口
	
}

3.再配置DMA初始化


```c
u8 DMA_Rx[Temp_BufSize_LEN] = {0};
u8 DMA_Tx[Temp_BufSize_LEN] = {0};

void DMA1_Config(void)

{
    
	DMA_InitTypeDef DMA_InitStructure;
    NVIC_InitTypeDef    NVIC_InitStructure;	
    RCC_AHBPeriphClockCmd(RCC_AHBPeriph_DMA1,ENABLE); 
	  
	  /*****DMA发送配置 Memory->Peripheral *****************/
	DMA_DeInit(DMA1_Channel4);           //将DMA1_Channel4信道寄存器初始化为默认的重置值。
	DMA_InitStructure.DMA_BufferSize = DMA1_MEM_LEN_RX;//设置DMA1的Buffer缓冲区大小,DMA1_MEM_LEN_RX为自定义的变量 
    DMA_InitStructure.DMA_M2M = DMA_M2M_Disable;       //关闭内存到内存
    DMA_InitStructure.DMA_Mode = DMA_Mode_Normal;      //正常模式
    DMA_InitStructure.DMA_DIR = DMA_DIR_PeripheralDST;  //内存到外设
    DMA_InitStructure.DMA_Priority = DMA_Priority_VeryHigh; //优先级非常高
    DMA_InitStructure.DMA_MemoryInc = DMA_MemoryInc_Enable; //内存地址递增
    DMA_InitStructure.DMA_PeripheralBaseAddr = (uint32_t)&USART2->TDR;   //设置外设接收地址 
    DMA_InitStructure.DMA_PeripheralInc =  DMA_PeripheralInc_Disable;    //外设地址不变
    DMA_InitStructure.DMA_MemoryDataSize = DMA_MemoryDataSize_Byte;      //设置内存数据长度以Byte为单位
    DMA_InitStructure.DMA_MemoryBaseAddr = (uint32_t)DMA_Tx;             //设置内存发送地址,DMA_Tx为自定义变量
    DMA_InitStructure.DMA_PeripheralDataSize = DMA_PeripheralDataSize_Byte; //设置外设数据长度以Byte为单位
    DMA_Init(DMA1_Channel4,&DMA_InitStructure);
    
    DMA_ClearITPendingBit(DMA1_IT_TC4);  //清除DMA_ch4传输完成中断标志 		
    DMA_ITConfig(DMA1_Channel4,DMA_IT_TC,ENABLE);	//使能DMA1_CH4传输完成中断		
	USART_DMACmd(USART2,USART_DMAReq_Tx,ENABLE);    //使能串口与DMA发送关联
		
		
		/*****DMA接收配置 Peripheral->Memory*****************/
    DMA_DeInit(DMA1_Channel5);
    DMA_InitStructure.DMA_BufferSize = DMA1_MEM_LEN_RX;  
    DMA_InitStructure.DMA_M2M = DMA_M2M_Disable;    
    DMA_InitStructure.DMA_Mode = DMA_Mode_Normal;   
    DMA_InitStructure.DMA_DIR = DMA_DIR_PeripheralSRC;  
    DMA_InitStructure.DMA_Priority = DMA_Priority_High; 
    DMA_InitStructure.DMA_MemoryInc = DMA_MemoryInc_Enable;
    DMA_InitStructure.DMA_PeripheralBaseAddr = (uint32_t)&USART2->RDR;    
    DMA_InitStructure.DMA_PeripheralInc =  DMA_PeripheralInc_Disable;
    DMA_InitStructure.DMA_MemoryDataSize = DMA_MemoryDataSize_Byte; 
    DMA_InitStructure.DMA_MemoryBaseAddr = (uint32_t)DMA_Rx;  
    DMA_InitStructure.DMA_PeripheralDataSize = DMA_PeripheralDataSize_Byte;
    DMA_Init(DMA1_Channel5,&DMA_InitStructure);
    
    DMA_ClearITPendingBit(DMA1_IT_TC5); 
    DMA_ITConfig(DMA1_Channel5,DMA_IT_TC,ENABLE);
    USART_DMACmd(USART2,USART_DMAReq_Rx,ENABLE);
	DMA_Cmd(DMA1_Channel5,ENABLE);  //使能DMA通道5
		
    NVIC_InitStructure.NVIC_IRQChannel = DMA1_Channel4_5_IRQn;
    NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;
    NVIC_InitStructure.NVIC_IRQChannelPriority = 1;
		
    NVIC_Init(&NVIC_InitStructure);
      
}
//注意:DMA初始化没有使能DMA_CH4,我们只需在要发送数据时使能即可

4.用DMA接收数据

void USART2_IRQHandler(void)
{
	u32 id = 0;
	u8 FIFO_Level = 0,i = 0;
	uint16_t Temp_Len=0; 
	//OverRun Error处理
   if(USART_GetFlagStatus(USART2, USART_FLAG_ORE) != RESET)
    {
      USART_ReceiveData(USART2);
      USART_ClearFlag(USART2, USART_FLAG_ORE);      
    }
    //接收空闲中断处理
	if(USART_GetITStatus(USART2, USART_IT_IDLE) != RESET) 
	{
	  USART_ClearITPendingBit(USART2, USART_IT_IDLE);	//清除中断标志位	     
	  DMA_Cmd(DMA1_Channel5, DISABLE);            
      Temp_Len= DMA1_MEM_LEN_RX - DMA_GetCurrDataCounter(DMA1_Channel5);//获得传输的数据个数		 
	  DMA_SetCurrDataCounter(DMA1_Channel5, DMA1_MEM_LEN_RX);
      DMA_Cmd(DMA1_Channel5, ENABLE);
      
      //此部分为自定义的传输协议
	  id = ((u32)DMA_Rx[0] << 24) | ((u32)DMA_Rx[1] << 16) | ((u32)DMA_Rx[2] << 8) | (u32)DMA_Rx[3];
      if(((id >> 13) & 0xff) != 0xff && ((id >> 13) & 0xff) != MY485_deviceinf[14])   
         return;            
      FIFO_Level = RS485_SetRxFifoFillLevel(RS485_FIFO.RS485_RCE);       
      if(FIFO_Level > 3)
         return;
      RS485_FIFO.FIFO_RxBuff_DLen[FIFO_Level-1] = Temp_Len;            
      for(i = 0; i < RS485_FIFO.FIFO_RxBuff_DLen[FIFO_Level-1]; i++)
         RS485_FIFO.RS485_FIFO_RXBUFF[FIFO_Level-1][i] = DMA_Rx[i];
         
	  memset(&DMA_Rx, 0, sizeof(DMA_Rx));		//将DMA中Buffer清空
	}
	
}


5.用DMA发送数据

void DMA_Enable_TX(DMA_Channel_TypeDef *DMA_CHx, u8 len)
{
    RS485_TX_Enable();    //由于用的是rs485,故需先置高bussy信号
	 DMA_Cmd(DMA_CHx, ENABLE);  //使能通道
    //DMA_Cmd(DMA_CHx, DISABLE);
    //DMA_SetCurrDataCounter(DMA_CHx, len);
    //DMA_Cmd(DMA_CHx, ENABLE);
}

int main(void)
{
	int i = 0;
	
	delay_init()
	USART2_Init();
	RS485_Busy_init();
	DMA1_Config();
		
	while(1)
	{						
		DMA_Enable_TX(DMA1_Channel4,DMA1_MEM_LEN_RX);
		DMA_Tx[0]='H';   //例如发送H字母
		
		
	}
}


以上是我对串口DMA的使用理解,希望能帮助到大家,谢谢观看。

  • 9
    点赞
  • 38
    收藏
    觉得还不错? 一键收藏
  • 2
    评论
### 回答1: STM32F4系列微控制器具有丰富的外设接口,其中包括多个串口接口以实现串口通信的功能。使用DMA(直接内存访问)模式可以提高串口通信的效率,实现数据的高速收发。 首先,需要初始化串口DMA相关的寄存器。通过配置相关的寄存器,设置波特率、数据位、停止位等参数,并使能串口接口和DMA功能。接着,为DMA配置通道和相关的内存地址。通常情况下,DMA的通道与串口接口对应,可以通过寄存器的设置来实现。 在接收数据时,可以配置DMA串口接收到的数据直接传输到指定的内存地址。通过设置DMA接收通道的内存地址寄存器,将接收到的数据直接存储到指定的内存空间中。此外,还需要设置DMA传输的数据大小和传输完成后的操作。 在发送数据时,类似地,可以将需要发送的数据存储在指定的内存地址中,然后通过配置DMA发送通道的内存地址寄存器,将数据从内存传输到串口发送寄存器中,实现数据的发送。 当收发数据完成后,可以通过DMA传输完成中断来通知处理器,以进行后续的数据处理操作。 总之,使用STM32F4系列微控制器的串口DMA功能,可以实现高效、稳定的串口通信。此外,对于更高级的应用,还可以使用DMA的双缓冲区功能来同时实现并行的数据收发。 ### 回答2: STM32F4系列的微控制器具有强大的DMA(直接内存访问)功能,可以实现高效的串口数据收发。对于串口数据收发,我们通常会使用USART(通用同步/异步收发器)模块,并结合DMA来实现数据传输的快速和可靠。 首先,我们需要初始化USART模块,设置相应的波特率、数据位、停止位等参数。然后,我们需要启用DMA功能,并配置DMA通道的源和目标地址。在收发数据时,我们可以通过修改对应的寄存器,向USART发送数据或从USART接收数据。 使用DMA来进行串口数据收发时,我们可以在初始化时设置好DMA通道的源和目标地址,然后通过修改USART的寄存器来触发数据传输。在数据传输过程中,DMA将自动将数据从源地址传输到目标地址,无需CPU的干预。这样可以大大减少了CPU的负担,提高了数据传输的效率。 在使用DMA进行串口数据收发时,我们还可以利用DMA的中断功能,实现接收完成中断或发送完成中断的回调函数。这样我们可以及时地处理接收到的数据或发送完成的状态。 总之,使用STM32F4系列微控制器的串口DMA功能可以实现快速、可靠的数据收发。合理配置和使用DMA通道,结合中断回调函数的实现,可以进一步提高串口数据传输的效率和可靠性。 ### 回答3: STM32F4系列微控制器具有内置的DMA (直接存储器访问) 控制器,可以在串口通信中使用DMA来实现数据收发。 在STM32F4中,我们可以使用USART(通用同步/异步收发器)模块来实现串口通信。USART模块具有多个寄存器用于配置和控制串口的工作方式。 首先,需要配置USART模块进行串口通信的初始化设置。例如,可以选择串口的波特率、数据位数、停止位数和校验位等参数。初始化完成后,可以使能USART模块。 然后,可以配置DMA控制器来进行串口数据收发。首先,需要选择合适的DMA通道,并配置其源地址(发送数据的存储器地址)和目的地址(接收数据的存储器地址)。然后,配置DMA的传输大小,即每次传输的字节长度。可以选择单个字节、半字或全字等传输大小。 接下来,需要配置DMA的传输模式。在串口收发中,常用的传输模式为循环模式,即当一次传输完成后,自动重新开始下一次传输。可以选择DMA的循环模式、传输方向(发送或接收)和传输方式(单次传输或连续传输)。 此外,还可以选择DMA的传输触发方式。可以选择硬件触发,即由外部事件触发DMA传输,例如USART的发送或接收完成事件;或者选择软件触发,即由软件控制手动触发DMA传输。 最后,使能USART的发送(TXE)和接收(RXNE)中断,并在中断服务函数中进行数据的处理。当USART发送或接收到数据时,会触发相应的中断并执行中断服务函数。 通过上述的配置和设置,可以实现串口通信中的数据收发操作。使用DMA进行数据传输可以提高系统的效率,减少CPU的负载。同时,可以利用USART的中断功能实时处理收发数据。 需要注意的是,具体的配置方法和步骤可能会因具体的STM32F4系列微控制器型号和开发环境的不同而略有差异,需要参考相应的技术手册和开发工具的文档进行详细设置。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值