STM32F4串口DMA配置

在使用串口时,一般采用查询发送,中断接收。但当要接收一串很长的数据时,每收到一个字节进入一次串口中断,有可能会导致中断占用时间过长。如果有一种方式,能够让串口收完一串数据,才进一次中断,那将是对写底层驱动的人来说,是极其好的一件事。经过查资料看手册,发现可以采用串口空闲中断和DMA接收来实现这个功能。具体更详细的说明后续补充,现只贴出代码,以供参考。
调试的过程中发现几个问题:
1、要串口初始化放在DMA初始化之前,否则会出现DMA发送和接收使用不了的问题;
2、DMA接收配置中DMA模式要配置为循环模式,如果配置成为正常模式会导致只能接收到一次数据,问题未知;
如果有某位大神知道这两个问题的原因,望不吝赐教。

画布多少,直接上代码:

//usart.h 头文件中要定义如下:

//该结构主要用来存放所有的串口相关的配置参数
typedef struct
{
    uint32_t baud_rate;
    uint16_t gpio_tx;
    uint16_t gpio_rx;
    uint8_t  gpio_tx_pin_source;
  uint8_t  gpio_rx_pin_source;  
    GPIO_TypeDef * usart_tx_port;
    GPIO_TypeDef * usart_rx_port;
    USART_TypeDef * usart_base;
    uint32_t  gpio_tx_rcc;
    uint32_t  gpio_rx_rcc;
    uint32_t apb_periph_rcc ;
    uint8_t  gpio_af;
    uint32_t dma_rx_rcc;
    DMA_Stream_TypeDef* dma_rx_stream;
    uint32_t dma_rx_channel;
    uint32_t dma_tx_rcc;
    DMA_Stream_TypeDef* dma_tx_stream;
    uint32_t dma_tx_channel;

    uint8_t tx_buff[48];
    uint8_t rx_buff[48];
    uint8_t len;
}UsartParameter_Typedef;

/******************************USART6***********************************/
#define USART6_GPIO_TX       GPIO_Pin_9
#define USART6_TX_PIN_SOURCE GPIO_PinSource9
#define USART6_GPIO_RX       GPIO_Pin_14
#define USART6_RX_PIN_SOURCE GPIO_PinSource14
#define USART6_TX_PORT       GPIOG
#define USART6_RX_PORT       GPIOG
#define USART6_GPIO_TX_RCC   RCC_AHB1Periph_GPIOG
#define USART6_GPIO_RX_RCC   RCC_AHB1Periph_GPIOG
#define USART6_APBPeriph_RCC RCC_APB2Periph_USART6
#define USART6_GPIO_AF       GPIO_AF_USART6


#define USART6_RX_DMA_RCC      RCC_AHB1Periph_DMA2
#define USART6_RX_DMA_STREAM   DMA2_Stream1
#define USART6_RX_DMA_CHANNEL  DMA_Channel_5

#define USART6_TX_DMA_RCC      RCC_AHB1Periph_DMA2
#define USART6_TX_DMA_STREAM   DMA2_Stream6
#define USART6_TX_DMA_CHANNEL  DMA_Channel_5
#define USART6_DMA_IT_TCIF     DMA_IT_TCIF6

#define USE_USART6_TX_DMA //使能发送DMA
#define USE_USART6_RX_DMA //使能接收DMA
//usart.c

//指定初始化结构
UsartParameter_Typedef Usart6_parameter=
{
    .baud_rate = USART6_BaudRate,
    .gpio_tx = USART6_GPIO_TX,
    .gpio_rx = USART6_GPIO_RX,
    .gpio_tx_pin_source = USART6_TX_PIN_SOURCE,
  .gpio_rx_pin_source = USART6_RX_PIN_SOURCE,   
    .usart_tx_port = USART6_TX_PORT,
    .usart_rx_port = USART6_RX_PORT,
    .usart_base = USART6,
    .gpio_tx_rcc = USART6_GPIO_TX_RCC,
    .gpio_rx_rcc = USART6_GPIO_RX_RCC,
    .apb_periph_rcc = USART6_APBPeriph_RCC ,
    .gpio_af = USART6_GPIO_AF,
    .dma_rx_rcc = USART6_RX_DMA_RCC,
    .dma_rx_stream =USART6_RX_DMA_STREAM ,
    .dma_rx_channel = USART6_RX_DMA_CHANNEL,
    .dma_tx_rcc = USART6_TX_DMA_RCC,
    .dma_tx_stream = USART6_TX_DMA_STREAM,
    .dma_tx_channel = USART6_TX_DMA_CHANNEL,    

};
//串口初始化模板函数
static void InitUsart(const UsartParameter_Typedef* usart)
{
    GPIO_InitTypeDef GPIO_InitStructure;
    USART_InitTypeDef USART_InitStructure;
    RCC_APB2PeriphClockCmd(usart->apb_periph_rcc, ENABLE);
    RCC_AHB1PeriphClockCmd( usart->gpio_tx_rcc, ENABLE);
    RCC_AHB1PeriphClockCmd( usart->gpio_rx_rcc, ENABLE);

    USART_InitStructure.USART_BaudRate = usart->baud_rate;
    USART_InitStructure.USART_WordLength = USART_WordLength_8b;
    USART_InitStructure.USART_StopBits = USART_StopBits_1;
    USART_InitStructure.USART_Parity = USART_Parity_No;
    USART_InitStructure.USART_Mode = USART_Mode_Tx | USART_Mode_Rx;
    USART_InitStructure.USART_HardwareFlowControl = USART_HardwareFlowControl_None;
    USART_Init(usart->usart_base, &USART_InitStructure);

//  USART_ITConfig(usart->usart_base, USART_IT_RXNE, ENABLE);

    USART_ITConfig(usart->usart_base,USART_IT_IDLE,ENABLE);
    USART_Cmd(usart->usart_base, ENABLE);

    /*初始化串口后再,配置Io口,防止串口初始化中会发一个无效的0x00或者0xfe*/
    GPIO_InitStructure.GPIO_Pin = usart->gpio_tx ;
    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(usart->usart_tx_port, &GPIO_InitStructure);
    GPIO_PinAFConfig(usart->usart_tx_port, usart->gpio_tx_pin_source, usart->gpio_af);

    GPIO_InitStructure.GPIO_Pin = usart->gpio_rx ;
    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(usart->usart_tx_port, &GPIO_InitStructure);
    GPIO_PinAFConfig(usart->usart_rx_port, usart->gpio_rx_pin_source, usart->gpio_af);

    USART_GetFlagStatus(usart->usart_base,USART_FLAG_TC);//清除TXE,防止第一个数据没有发送出来

}

void UsartDmaTxConfig(const UsartParameter_Typedef* usart)
{
  DMA_InitTypeDef DMA_InitStructure;
  /*开启DMA时钟*/
  RCC_AHB1PeriphClockCmd(usart->dma_tx_rcc, ENABLE);


//————————————————————————————————发送————————————————————//  

  /* 复位初始化DMA数据流 */
  DMA_DeInit(usart->dma_tx_stream);

  /* 确保DMA数据流复位完成 */
  while (DMA_GetCmdStatus(usart->dma_tx_stream) != DISABLE)  {
  }

  /*usart1 tx对应dma2,通道4,数据流7*/  
  DMA_InitStructure.DMA_Channel = usart->dma_tx_channel;  
  /*设置DMA源:串口数据寄存器地址*/
  DMA_InitStructure.DMA_PeripheralBaseAddr = (uint32_t)(&usart->usart_base->DR);     
  /*内存地址(要传输的变量的指针)*/
  DMA_InitStructure.DMA_Memory0BaseAddr = (uint32_t)usart->tx_buff;
  /*方向:从内存到外设*/     
  DMA_InitStructure.DMA_DIR = DMA_DIR_MemoryToPeripheral;   
  /*传输大小DMA_BufferSize=SENDBUFF_SIZE*/  
  DMA_InitStructure.DMA_BufferSize = MAX_DATA_LEN;
  /*外设地址不增*/        
  DMA_InitStructure.DMA_PeripheralInc = DMA_PeripheralInc_Disable; 
  /*内存地址自增*/
  DMA_InitStructure.DMA_MemoryInc = DMA_MemoryInc_Enable;   
  /*外设数据单位*/    
  DMA_InitStructure.DMA_PeripheralDataSize = DMA_PeripheralDataSize_Byte;
  /*内存数据单位 8bit*/
  DMA_InitStructure.DMA_MemoryDataSize = DMA_MemoryDataSize_Byte;   
  /*DMA模式:普通模式*/
  DMA_InitStructure.DMA_Mode = DMA_Mode_Normal;  
  /*优先级:中*/ 
  DMA_InitStructure.DMA_Priority = DMA_Priority_High;      
  /*禁用FIFO*/
  DMA_InitStructure.DMA_FIFOMode = DMA_FIFOMode_Disable;        
  DMA_InitStructure.DMA_FIFOThreshold = DMA_FIFOThreshold_Full;    
  /*存储器突发传输 16个节拍*/
  DMA_InitStructure.DMA_MemoryBurst = DMA_MemoryBurst_Single;    
  /*外设突发传输 1个节拍*/
  DMA_InitStructure.DMA_PeripheralBurst = DMA_PeripheralBurst_Single;    
  /*配置DMA2的数据流6*/          
  DMA_Init(usart->dma_tx_stream, &DMA_InitStructure);
    /*配置发送完成中断*/
  DMA_ITConfig(usart->dma_tx_stream, DMA_IT_TC,ENABLE);  //此处必须开启中断,否则DMA_GetITStatus(DMA2_Stream6,DMA_IT_TCIF6)一直为0
  /*使能DMA*/
  DMA_Cmd(usart->dma_tx_stream, ENABLE);    
    USART_DMACmd(usart->usart_base,USART_DMAReq_Tx,ENABLE);    //使能DMA发送
}


//DMA TX RX配置模板函数
static void UsartDmaRxConfig(const UsartParameter_Typedef* usart)
{
    DMA_InitTypeDef DMA_InitStructure;
/*-----------------------接收------------------------------------*/
    RCC_AHB1PeriphClockCmd(usart->dma_rx_rcc, ENABLE);
//  RCC_AHB1PeriphResetCmd(usart->dma_rx_rcc, ENABLE);
  /* 复位初始化DMA数据流 */
  DMA_DeInit(usart->dma_rx_stream);

  /* 确保DMA数据流复位完成 */
  while (DMA_GetCmdStatus(usart->dma_rx_stream) != DISABLE)  {
  }

  /*usart6 rx对应dma2,通道5,数据流1*/  
  DMA_InitStructure.DMA_Channel = usart->dma_rx_channel;  
  /*设置DMA源:串口数据寄存器地址*/
  DMA_InitStructure.DMA_PeripheralBaseAddr =(uint32_t)(&usart->usart_base->DR)/*((uint32_t)usart->usart_base+0x04)usart->usart_base->DR*/;   
  /*内存地址(要传输的变量的指针)*/
  DMA_InitStructure.DMA_Memory0BaseAddr = (u32)usart->rx_buff;
  /*方向:从内存到外设*/     
  DMA_InitStructure.DMA_DIR = DMA_DIR_PeripheralToMemory;   
  /*传输大小DMA_BufferSize=SENDBUFF_SIZE*/  
  DMA_InitStructure.DMA_BufferSize = MAX_DATA_LEN;
  /*外设地址不增*/        
  DMA_InitStructure.DMA_PeripheralInc = DMA_PeripheralInc_Disable; 
  /*内存地址自增*/
  DMA_InitStructure.DMA_MemoryInc = DMA_MemoryInc_Enable;   
  /*外设数据单位*/    
  DMA_InitStructure.DMA_PeripheralDataSize = DMA_PeripheralDataSize_Byte;
  /*内存数据单位 8bit*/
  DMA_InitStructure.DMA_MemoryDataSize = DMA_MemoryDataSize_Byte;   
  /*DMA模式:普通模式*/
  DMA_InitStructure.DMA_Mode = /*DMA_Mode_Normal*/DMA_Mode_Circular;     
  /*优先级:中*/ 
  DMA_InitStructure.DMA_Priority = DMA_Priority_High;      
  /*禁用FIFO*/
  DMA_InitStructure.DMA_FIFOMode = DMA_FIFOMode_Disable;        
  DMA_InitStructure.DMA_FIFOThreshold = DMA_FIFOThreshold_Full;    
  /*存储器突发传输 16个节拍*/
  DMA_InitStructure.DMA_MemoryBurst = DMA_MemoryBurst_Single;    
  /*外设突发传输 1个节拍*/
  DMA_InitStructure.DMA_PeripheralBurst = DMA_PeripheralBurst_Single;    
  /*配置DMA2的数据流6*/          
  DMA_Init(usart->dma_rx_stream, &DMA_InitStructure);
      /*使能DMA*/
  DMA_Cmd(usart->dma_rx_stream, ENABLE);
  USART_DMACmd(usart->usart_base,USART_DMAReq_Rx,ENABLE);    //使能DMA接收
}


//串口初始化函数

void init_usart(void)
{
     InitUsart(&Usart6_parameter);  
#ifdef USE_USART6_TX_DMA
       UsartDmaTxConfig(&Usart6_parameter);
#endif              
#ifdef USE_USART6_RX_DMA
        UsartDmaRxConfig(&Usart6_parameter);
#endif
}


//发送函数
static void Usart6Send(const uint8_t *buf,const uint8_t len)
{

#ifdef USE_USART6_TX_DMA
    UsartDMASend(buf,len,&Usart6_parameter);
#else
    UsartSend(buf,len,&Usart6_parameter);
#endif  
}


//中断接收函数

void USART6_IRQHandler(void)
{
#ifdef USE_USART6_RX_DMA
     if(USART_GetITStatus(USART6, USART_IT_IDLE) != RESET)  
    {  
        Usart6_parameter.usart_base->SR;  
        Usart6_parameter.usart_base->DR; //清USART_IT_IDLE标志  
        //关闭DMA  
        DMA_Cmd(Usart6_parameter.dma_rx_stream,DISABLE);  
        //清除标志位  
        DMA_ClearFlag(Usart6_parameter.dma_rx_stream,DMA_FLAG_TCIF6);  
        //获得接收帧帧长  由于SnDTR寄存器用于指示要传输的剩余数据选项,每传输一次,递减1
        Usart6_parameter.len = MAX_DATA_LEN - DMA_GetCurrDataCounter(Usart6_parameter.dma_rx_stream);  
    //      Usart6.onrx(NULL,Usart6_parameter.rx_buff,Usart6_parameter.len);

        DMA_Cmd(Usart6_parameter.dma_rx_stream,ENABLE);  
            Usart6Send(Usart6_parameter.rx_buff,Usart6_parameter.len );
             memset(Usart6_parameter.rx_buff,0,sizeof(Usart6_parameter.rx_buff));   //清空接收缓冲区
            Usart6_parameter.len = 0; 
        //  Usart6DmaTx(TmpBuff,len);
    }   
#else
        uint8_t temp = 0;
    if(USART_GetFlagStatus(Usart6_parameter.usart_base,USART_FLAG_RXNE)==SET)
    {
        USART_ClearITPendingBit(Usart6_parameter.usart_base, USART_IT_RXNE);
        temp = USART_ReceiveData(Usart6_parameter.usart_base);
        Usart6Send(&temp,1);
    //  Usart6ReceivedProsess();
    }
#endif  
}
  • 0
    点赞
  • 11
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
### 回答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系列微控制器型号和开发环境的不同而略有差异,需要参考相应的技术手册和开发工具的文档进行详细设置。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值