STM32单片机示例:使用定时器触发DMA

前言

单片机中使用定时器触发DMA是比较好用的一种操作,这里将对此做个示例说明。

基础说明

ST官方的例程中有一个 TIM_DMA 的例程,其功能是启用一个定时器,设置其中一路通道输出PWM,使用 TIM_CH 事件来触发DMA,然后用DMA搬运数据到 CCR 寄存器,从而来改变PWM的占空比。该示例是借鉴官方示例而来。

该示例演示通过 TIM_UP 请求来触发DMA搬运数据。本例中通过DMA从内存中搬运数据修改 GPIOx->BSRR 寄存器的值,从而实现修改GPIO口输出电平。使用该方式下可以实现同步控制一组GPIO口的输出,可以当作方波或是PWM输出等功能。除默认生成的代码,只在 main.c 文件添加几行手动编写的代码。本例中输出引脚为 PB0、PB1、PB2、PB3 。

关键配置与代码

这里只贴出关键的配置与代码,完整示例可以通过文后链接查看。

本示例中关键配置就是下图的通过 TIM_UP 请求来触发DMA了:
在这里插入图片描述

除了配置生成的代码,需要手动添加的代码就几行:
HAL库

int main(void)
{
  HAL_Init();

  SystemClock_Config();

  MX_GPIO_Init();
  MX_DMA_Init();
  MX_TIM3_Init();

  // BSRR寄存器数据,高16位用于清除端口,低16位用于设置端口
  uint32_t buff[5] ={0x000F0000, 0x00000008, 0x00000004, 0x00000002, 0x00000001};

  // 以下步骤参考函数 HAL_TIM_Base_Start_DMA ,但该函数中外设地址是定时器的ARR寄存器,不符合我这里需求,所以根据该函数来编写下面代码
  HAL_DMA_Start(&hdma_tim3_up, (uint32_t)buff, (uint32_t)&GPIOB->BSRR, 5); // 设置并启动DMA
  __HAL_TIM_ENABLE_DMA(&htim3, TIM_DMA_UPDATE); // 使能定时器触发DMA
  HAL_TIM_Base_Start(&htim3); // 启动定时器

  while (1)
  {
  }
}

LL库

int main(void)
{
  LL_APB4_GRP1_EnableClock(LL_APB4_GRP1_PERIPH_SYSCFG);
  NVIC_SetPriorityGrouping(NVIC_PRIORITYGROUP_4);
  NVIC_SetPriority(SysTick_IRQn, NVIC_EncodePriority(NVIC_GetPriorityGrouping(),15, 0));
  SystemClock_Config();
  MX_GPIO_Init();
  MX_DMA_Init();
  MX_TIM3_Init();

  // BSRR寄存器数据,高16位用于清除端口,低16位用于设置端口
  uint32_t buff[5] ={0x000F0000, 0x00000008, 0x00000004, 0x00000002, 0x00000001};

  LL_DMA_SetPeriphAddress(DMA1, LL_DMA_STREAM_0, (uint32_t)&GPIOB->BSRR); // 设置外设地址
  LL_DMA_SetMemoryAddress(DMA1, LL_DMA_STREAM_0, (uint32_t)&buff); // 设备内存地址
  LL_DMA_SetDataLength(DMA1, LL_DMA_STREAM_0, 5); // 设置传输长度

  LL_DMA_EnableStream(DMA1, LL_DMA_STREAM_0); // 开启DMA传输

  LL_TIM_EnableDMAReq_UPDATE(TIM3); // 使能TIM_UP请求DMA

  LL_TIM_EnableCounter(TIM3); // 启动定时器

  while (1)
  {
  }
}

本示例演示结果可以通过逻辑分析仪方便的观察:
在这里插入图片描述

示例链接

仓库地址: https://github.com/NaisuXu/STM32_MCU_Examples

本示例为仓库中 TIM_UP_DMA_HAL_H750TIM_UP_DMA_LL_H750

  • 1
    点赞
  • 20
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
可以通过以下步骤实现 STM32 定时器触发 DMA: 1. 配置定时器 TIM 的计数模式和时钟源,使其能够按照一定频率产生定时中断。 2. 配置 DMA 的通道和传输方向,设置传输数据的大小和目的地址。 3. 配置 DMA触发源为定时器 TIM 的更新事件,即在 TIM 定时中断时触发 DMA 传输。 4. 在 TIM 的中断服务函数中启动 DMA 传输,将数据从源地址传输到目的地址。 以下是一个示例代码: ```c /* 配置 TIM 定时器 */ TIM_HandleTypeDef htim; htim.Instance = TIMx; htim.Init.Prescaler = xxx; htim.Init.Period = xxx; htim.Init.ClockDivision = TIM_CLOCKDIVISION_DIV1; htim.Init.CounterMode = TIM_COUNTERMODE_UP; HAL_TIM_Base_Init(&htim); HAL_TIM_Base_Start_IT(&htim); /* 配置 DMA 传输 */ DMA_HandleTypeDef hdma; hdma.Instance = DMAx_Channelx; hdma.Init.Direction = DMA_MEMORY_TO_MEMORY; hdma.Init.PeriphInc = DMA_PINC_ENABLE; hdma.Init.MemInc = DMA_MINC_ENABLE; hdma.Init.PeriphDataAlignment = DMA_PDATAALIGN_HALFWORD; hdma.Init.MemDataAlignment = DMA_MDATAALIGN_HALFWORD; hdma.Init.Mode = DMA_NORMAL; hdma.Init.Priority = DMA_PRIORITY_HIGH; HAL_DMA_Init(&hdma); __HAL_DMA_ENABLE_IT(&hdma, DMA_IT_TC); /* 配置 DMA 触发源为 TIM 更新事件 */ __HAL_DMA_DISABLE(&hdma); HAL_DMA_Start_IT(&hdma, (uint32_t)src_addr, (uint32_t)dst_addr, data_size); __HAL_DMA_ENABLE(&hdma); __HAL_TIM_ENABLE_DMA(&htim, TIM_DMA_UPDATE); /* 在 TIM 中断服务函数中启动 DMA 传输 */ void HAL_TIM_PeriodElapsedCallback(TIM_HandleTypeDef *htim) { if (htim->Instance == TIMx) { HAL_DMA_Start_IT(&hdma, (uint32_t)src_addr, (uint32_t)dst_addr, data_size); } } ``` 以上代码仅供参考,具体实现还需根据实际需求进行调整。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

Naisu Xu

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值