STM32双缓冲机制初始化(使用STM32CubeMX)

2019-05-24来源: eefocus关键字:STM32  双缓冲机制  初始化

1.使用STM32CubeMX配置的串口引脚设置和dma的设置会生成在usart.c。


  1)如果DMA接收想采用循环缓冲区的方式,可以直接将RX-DMA设置成Circle方式,然后数据就会硬件上自动实现环形缓冲区的功能,省了不少时间。


  2)DMA在采用Normal模式的时候,当一次任务完成后,DMA->DMA_BufferSize自动清零,并且DMA自动停止。如果想再次设置DMA的BufferSize的话,必须要进行如下操作:


   step1:DMA_CMD(DMAx_Channely,DISABLE);


   step2: 设置DMA_BufferLen


   step3:DMA_CMD(DMAx_Channely,ENABLE)


  3)DMA采用Circle模式的时候,在发送或者接受完成之后,仍然保存着BufferSize,并且DMA还处于使能状态,一直连续工作,直到用户停止DMA


else if(uartHandle->Instance==USART1)

  {

  /* USER CODE BEGIN USART1_MspInit 0 */

 

  /* USER CODE END USART1_MspInit 0 */

    /* USART1 clock enable */

    __HAL_RCC_USART1_CLK_ENABLE();

  

    /**USART1 GPIO Configuration    

    PA9     ------> USART1_TX

    PA10     ------> USART1_RX 

    */

    GPIO_InitStruct.Pin = GPIO_PIN_9|GPIO_PIN_10;

    GPIO_InitStruct.Mode = GPIO_MODE_AF_PP;

    GPIO_InitStruct.Pull = GPIO_PULLUP;

    GPIO_InitStruct.Speed = GPIO_SPEED_FREQ_VERY_HIGH;

    GPIO_InitStruct.Alternate = GPIO_AF7_USART1;

    HAL_GPIO_Init(GPIOA, &GPIO_InitStruct);

 

    /* USART1 DMA Init */

    /* USART1_RX Init */

    hdma_usart1_rx.Instance = DMA2_Stream2;

    hdma_usart1_rx.Init.Channel = DMA_CHANNEL_4;

    hdma_usart1_rx.Init.Direction = DMA_PERIPH_TO_MEMORY;

    hdma_usart1_rx.Init.PeriphInc = DMA_PINC_DISABLE;

    hdma_usart1_rx.Init.MemInc = DMA_MINC_ENABLE;

    hdma_usart1_rx.Init.PeriphDataAlignment = DMA_PDATAALIGN_BYTE;

    hdma_usart1_rx.Init.MemDataAlignment = DMA_MDATAALIGN_BYTE;

    hdma_usart1_rx.Init.Mode = DMA_NORMAL; //这里设成DMA_CIRCULAR和DMA_NORMAL好像没有对双缓冲造成影响

    hdma_usart1_rx.Init.Priority = DMA_PRIORITY_LOW;

    hdma_usart1_rx.Init.FIFOMode = DMA_FIFOMODE_DISABLE;

    if (HAL_DMA_Init(&hdma_usart1_rx) != HAL_OK)

    {

      _Error_Handler(__FILE__, __LINE__);

    }

 

    __HAL_LINKDMA(uartHandle,hdmarx,hdma_usart1_rx);

 

    /* USART1_TX Init */

    hdma_usart1_tx.Instance = DMA2_Stream7;

    hdma_usart1_tx.Init.Channel = DMA_CHANNEL_4;

    hdma_usart1_tx.Init.Direction = DMA_MEMORY_TO_PERIPH;

    hdma_usart1_tx.Init.PeriphInc = DMA_PINC_DISABLE;

    hdma_usart1_tx.Init.MemInc = DMA_MINC_ENABLE;

    hdma_usart1_tx.Init.PeriphDataAlignment = DMA_PDATAALIGN_BYTE;

    hdma_usart1_tx.Init.MemDataAlignment = DMA_MDATAALIGN_BYTE;

    hdma_usart1_tx.Init.Mode = DMA_NORMAL;

    hdma_usart1_tx.Init.Priority = DMA_PRIORITY_LOW;

    hdma_usart1_tx.Init.FIFOMode = DMA_FIFOMODE_DISABLE;

    if (HAL_DMA_Init(&hdma_usart1_tx) != HAL_OK)

    {

      _Error_Handler(__FILE__, __LINE__);

    }

 

    __HAL_LINKDMA(uartHandle,hdmatx,hdma_usart1_tx);

 

    /* USART1 interrupt Init */

    HAL_NVIC_SetPriority(USART1_IRQn, 5, 0);

    HAL_NVIC_EnableIRQ(USART1_IRQn);

  /* USER CODE BEGIN USART1_MspInit 1 */

 

  /* USER CODE END USART1_MspInit 1 */

  }

 


2.下面是关于DMA双缓冲机制的配置,特别说明一下,我先__HAL_DMA_DISABLE(hdma); 再配置后面的参数。DMA才正常接收数据了。


static HAL_StatusTypeDef DMAEx_MultiBufferStart_IT(DMA_HandleTypeDef *hdma,

                                                   uint32_t SrcAddress,

                                                   uint32_t DstAddress,

                                                   uint32_t SecondMemAddress,

                                                   uint32_t DataLength)

{

  HAL_StatusTypeDef status = HAL_OK;

  

  /* Memory-to-memory transfer not supported in double buffering mode */

  if (hdma->Init.Direction == DMA_MEMORY_TO_MEMORY)

  {

    hdma->ErrorCode = HAL_DMA_ERROR_NOT_SUPPORTED;

    return HAL_ERROR;

  }

  

  /* Set the UART DMA transfer complete callback */

  /* Current memory buffer used is Memory 1 callback */

  hdma->XferCpltCallback   = dma_m0_rxcplt_callback; //第一个缓冲区填满后会调用这个函数

  /* Current memory buffer used is Memory 0 callback */

  hdma->XferM1CpltCallback = dma_m1_rxcplt_callback; ////第二个缓冲区填满后会调用这个函数

  

  /* Check callback functions */

  if ((NULL == hdma->XferCpltCallback) || (NULL == hdma->XferM1CpltCallback))

  {

    hdma->ErrorCode = HAL_DMA_ERROR_PARAM;

    return HAL_ERROR;

  }

  

  /* Process locked */

  __HAL_LOCK(hdma);

    /* Enable the peripheral */

    __HAL_DMA_DISABLE(hdma);   //先要禁止DMA后面的设置才会生效

  

  //if(HAL_DMA_STATE_READY == hdma->State)

  //{

    /* Change DMA peripheral state */

    hdma->State = HAL_DMA_STATE_BUSY;

    /* Initialize the error code */

    hdma->ErrorCode = HAL_DMA_ERROR_NONE;

    /* Enable the Double buffer mode */

    hdma->Instance->CR |= (uint32_t)DMA_SxCR_DBM;

    /* Configure DMA Stream destination address */

    hdma->Instance->M1AR = SecondMemAddress;

    

    /* Configure DMA Stream data length */

    hdma->Instance->NDTR = DataLength;

    /* Configure the source, destination address */

    if((hdma->Init.Direction) == DMA_MEMORY_TO_PERIPH)

    {

      hdma->Instance->PAR = DstAddress;

      hdma->Instance->M0AR = SrcAddress;

    }

    else

    {

      hdma->Instance->PAR = SrcAddress;

      hdma->Instance->M0AR = DstAddress;

    }

    

    /* Clear TC flags */

    __HAL_DMA_CLEAR_FLAG (hdma, __HAL_DMA_GET_TC_FLAG_INDEX(hdma));

    /* Enable TC interrupts*/

    hdma->Instance->CR  |= DMA_IT_TC;

    

    /* Enable the peripheral */

    __HAL_DMA_ENABLE(hdma);

    

    /* Change the DMA state */

    hdma->State = HAL_DMA_STATE_READY;

  //}

  //else

  //{

  //  /* Return error status */

  //  status = HAL_BUSY;

  //}

  

  /* Process unlocked */

  __HAL_UNLOCK(hdma);

  

  return status; 

}

 

void debug_uart_init(void)

{

  //open uart idle it

  __HAL_UART_CLEAR_IDLEFLAG(&DEBUG_HUART);

  __HAL_UART_ENABLE_IT(&DEBUG_HUART, UART_IT_IDLE);//设置了空闲中断。进入空闲中断就可以考虑处理数据了。如果数据量大就放到专门的任务里处理。

  

  // Enable the DMA transfer for the receiver request

  SET_BIT(DEBUG_HUART.Instance->CR3, USART_CR3_DMAR);

  

  DMAEx_MultiBufferStart_IT(DEBUG_HUART.hdmarx,

                           (uint32_t)&DEBUG_HUART.Instance->DR,

                           (uint32_t)debug_dma_rxbuff[0],

                           (uint32_t)debug_dma_rxbuff[1],

                           UART_RX_DMA_SIZE);

  

}

 


关键字:STM32  双缓冲机制  初始化 编辑:什么鱼 引用地址:http://news.eeworld.com.cn/mcu/ic462832.html 本网站转载的所有的文章、图片、音频视频文件等资料的版权归版权所有人所有,本站采用的非本站原创文章及图片等内容无法一一联系确认版权者。如果本网所选内容的文章作者及编辑认为其作品不宜公开自由传播,或不应无偿使用,请及时通过电子邮件或电话通知我们,以迅速采取适当措施,避免给双方造成不必要的经济损失。

上一篇:STM32F407各定时器的时钟频率
下一篇:STM32之TIM 舵机控制PWM

关注eeworld公众号 快捷获取更多信息
关注eeworld公众号
快捷获取更多信息
关注eeworld服务号 享受更多官方福利
关注eeworld服务号
享受更多官方福利

推荐阅读

4*4键盘程序代码 基于STM32
这是自己写的扫描第一行按键的程序代码。PE的位8~位11设置为下拉输入。PE的位12~位15设置为推挽输出其中PE的位11是4*4键盘的第一列,PE的为键盘的第一行。先将第一行设置为高电平,检测列中是否有高电平u8 KEY_Scan(u8 mode){                staticu8 key_up=1;//按键按松开标志         if(mode)key_up=1;  //支持连按        &nbs
发表于 2019-10-09
stm32中ADC初始化程序
拟通道输入引脚 ->ADC_IN10 */                       GPIO_InitStructure.GPIO_Pin = GPIO_Pin_0;GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN; //模拟输入引脚GPIO_Init(GPIOC, &GPIO_InitStructure); /* ADC初始化*/ //CR1寄存器设置tmpreg1 = ADC1->CR1;tmpreg1 &
发表于 2019-10-09
STM32F0 ADC学习
开始时候使用的是stdlib的库,最近发现cube库用的越来越广泛了,遂开始使用cube库来完成ADC的多通道采集实验。ADC 的driver 在STM32F0XX_HAL_DRIVER当中,有stm32f0xx_hal_adc.c文件中,我们可以在stm32f0xx_hal_conf.h中开启 宏定义 ADC 模块。ADC有三种工作模式,polling interruptDMA我这里使用了polling的方式来获取多通道的数据。首先是要声明两个参数设置的结构体ADC_HandleTypeDef             AdcHandle
发表于 2019-10-09
怎样用STM32 ADC测量电压(中断方式)
ADC 概述ADC是模数转换的缩写,是将连续的模拟信号转换为离散的数字信号,在通信,自动控制等多个领域有着广泛的应用,利用各种传感器,能将现实世界中的模拟量转换为机器能够识别的数字量,机器有了ADC,就像人有了各种感官,能够感知周围的世界并做出反应。STM32F10x  ADC特点l 12位逐次逼近型的模拟数字转换器。l 最多带3个ADC控制器l 最多支持18个通道,可最多测量16个外部和2个内部信号源。l 支持单次和连续转换模式l  转换结束,注入转换结束,和发生模拟看门狗事件时产生中断。l  通道0到通道n的自动扫描模式l  自动校准l  采样间隔可以按通道编程l&nbs
发表于 2019-10-09
怎样用STM32 ADC测量电压(中断方式)
STM32_ADC单通道单次采集
数位于在adc.c文件下面;调用这个接口就可以采集电压值。函数使用单通道单次,软件触发采样电压值,这里采样8次(更加自己情况可以选择多次),算平均,最后得出电压值(1000倍值)。五、主函数应用该函数位于在main.c文件下面;主要就是采集电压,通过串口打印出来(1000倍值)。 六、揭晓ADC123_IN2上面的问题有自己想明白了的吗?其实很简单的,ADC123_IN2顾名思义,它就是包含了ADC1、ADC2、ADC3的IN2的意思。也就是说,你们使用ADC2的通道2也是这个“ADC123_IN2”通道。方便大家学习,我把两个工程都上传至360云盘,不行的朋友可以亲自下载代码试试。对比的软件工程下载地址:https
发表于 2019-10-09
STM32_ADC单通道单次采集
STM32的ADC基本配置
(1)模/数转换工作于单通道还是多通道模式(2)工作于单次还是连续模式。(3)外部触发转换还是软件使能转换。(4)数据对齐方式,右对齐还是左对齐。(5)A/D转换的通道数目。(6)设置A/D通道的转换顺序及采样时间。其中转换时间为T.conv=采样时间+12.5个周期(7)⑥使能DMA启动传输⑦使能ADC⑧校准ADC,ADC的校准用到以下代码:/*重置ADC1的校准寄存器关/ADC_ ResetCal ibration( ADC1);/*获取ADC重置校准寄存器的状态*/while(ADC_ GetResetCal ibrat ionStatus(ADC1));ADC_ StartCal ibration(ADC1);/*开始校准
发表于 2019-10-09
小广播
何立民专栏 单片机及嵌入式宝典

北京航空航天大学教授,20余年来致力于单片机与嵌入式系统推广工作。

电子工程世界版权所有 京ICP证060456号 京ICP备10001474号 电信业务审批[2006]字第258号函 京公海网安备110108001534 Copyright © 2005-2019 EEWORLD.com.cn, Inc. All rights reserved