STM32学习----通用定时器的应用(PWM)

这篇具有很好参考价值的文章主要介绍了STM32学习----通用定时器的应用(PWM)。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

STM32的通用定时器

STM32单片机的通用定时器,有TIM2、TIM3、TIM4、TIM5这4个。

通用定时器的功能,用的比较多的有下面几种:

计数功能:向上计数,向下计数,向上/向下计数;

输入捕获:测量信号的周期和占空比;

输出比较:

PWM生成:

通用定时器框图

通用定时器功能多了,框图也就复杂起来了

stm32 定时器3 pwm,STM32学习,stm32,单片机,嵌入式硬件

通用定时器框图分解

stm32 定时器3 pwm,STM32学习,stm32,单片机,嵌入式硬件

第一部分:时钟来源

stm32 定时器3 pwm,STM32学习,stm32,单片机,嵌入式硬件

时钟来源有很多:

CK_INT、TIMx_ETR、ITR0、ITR1、ITR2、ITR3、TI1F_ED、TI1FP1、TI2FP2这9个,可以分下类;

CK_INT,就是定时器外设的时钟,比如72MHz,属于内部时钟

TIMx_ETR,就是定时器所对应的外部输入,这个外部输入信号经过相位检测、边缘检测和分频之后的信号就是ETRP,ETRP经过输入滤波之后又分成两路,一路直接用于时钟计数(跟CK_INT一样)这就是外部模式2(ETR2);一部分是作为出发信号给从模式控制器,就是外部模式1(ETR1)。

ITR0、ITR1、ITR2、ITR3,这几个的信号之后都是传给从模式控制器,就是两个定时器串联着使用,前面的那个定时器可以看成后面那个定时器的预分频器。

第二部分:时基单元

stm32 定时器3 pwm,STM32学习,stm32,单片机,嵌入式硬件

第三部分:输入捕获

stm32 定时器3 pwm,STM32学习,stm32,单片机,嵌入式硬件

TIMx_CH1 是定时器的输入引脚1

TIMx_CH2是定时器的输入引脚2

TIMx_CH3是定时器的输入引脚3

TIMx_CH4是定时器的输入引脚4

TI1FP1 是来自于通道TI1,经过输入滤波和边沿检测后送给捕获比较通道IC1;

TI1FP2是来自于通道TI1,经过输入滤波和边沿检测后送给捕获比较通道IC2;

TI1FP1和TI1FP2这两个信号的特性是一模一样的,都是通道1的信号经过滤波和边沿检测之后产生的信号,只是TI1FP1是送给捕获比较通道IC1,TI1FP2是送给捕获比较通道IC2。

TI2FP1是来自于通道TI2,经过输入滤波和边沿检测后送给捕获比较通道IC1;

TI2FP2是来自于通道TI2,经过输入滤波和边沿检测后送给捕获比较通道IC2;

TI2FP1和TI2FP2这两个信号的特性是一模一样的,都是通道2的信号经过滤波和边沿检测之后产生的信号,只是TI2FP1是送给捕获比较通道IC1,TI2FP2是送给捕获比较通道IC2。

TI3FP3是来自于通道TI3,经过输入滤波和边沿检测后送给捕获比较通道IC3;

TI3FP4是来自于通道TI3,经过输入滤波和边沿检测后送给捕获比较通道IC4;

TI3FP3和TI3FP4这两个信号的特性是一模一样的,都是通道3的信号经过滤波和边沿检测之后产生的信号,只是TI3FP3是送给捕获比较通道IC3,TI3FP4是送给捕获比较通道IC4。

TI4FP3是来自于通道TI4,经过输入滤波和边沿检测后送给捕获比较通道IC3;

TI4FP4是来自于通道TI4,经过输入滤波和边沿检测后送给捕获比较通道IC4;

TI4FP3和TI4FP4这两个信号的特性是一模一样的,都是通道4的信号经过滤波和边沿检测之后产生的信号,只是TI4FP3是送给捕获比较通道IC3,TI4FP4是送给捕获比较通道IC4。

这一块的意思大概是这样的:

通用定时器都有4个通道,CH1、CH2、CH3、CH4,分别都有外部的引脚对应;

外部的信号进入通道后,进行滤波的边沿检测,并生成两路一模一样的信号TIxFPx;

CH1和CH2可以看成一伙的,它们生成的两路信号是,你给我一个,我给你一个;

CH3和CH4也可以看成一伙的,它们生成的两路信号也是,你给一个我,我给一个你;

最后这些信号都是经过分频之后给捕获比较寄存器。

所谓捕获比较寄存器,对输入信号来说就是捕获寄存器,对输出信号来说就是比较寄存器。

第四部分:输出比较

stm32 定时器3 pwm,STM32学习,stm32,单片机,嵌入式硬件

通用定时器都有4个通道,CH1、CH2、CH3、CH4,这4个通道既可以是输入,也可以是输出;

用作输入的话就是输入捕获通道,用作输出的话就是输出比较通道。东西还是同一个东西,用法不同,名字就不同了。

常见应用一:PWM输出

PWM模式其实也是输出比较模式的一种,它的工作原理也比较简单

1、以向上计数来说,设置时基单元的周期(也就是自动重装载寄存器的值)为1000,每次计数都是从0数到1000,一直循环;

2、在捕获比较寄存器中设置一个值,看你用的那个通道(一共就4个通道),比如用通道CH1,那就在CCR1中设置一个值,比如400;

3、现在就比较计数单元中计数寄存器的值和CCR1中的值(400)的大小了;

4、因为是向上计数,从0 开始的,那一开始肯定比400(CCR1)小了,这个时候可以让输出通道CH1对应的管脚输出某种电平。(这个时候新的概念就要出来了,PWM模式),假设输出极性为高,PWM2模式,那一开始就是输出低电平,在CNT从0计到400那段时间一直输出低电平,从400到1000就输出高电平。

输出极性与PWM模式的关系

stm32 定时器3 pwm,STM32学习,stm32,单片机,嵌入式硬件

个人觉得数字比字母方便理解一些

stm32 定时器3 pwm,STM32学习,stm32,单片机,嵌入式硬件

所以呢,PWM输出是不是很简单,不要怕定时器那个大框图,你只要拿你需要的那一小部分就可以了,针对PWM输出,只需要了解时钟源、时基单元、捕获比较寄存器这三个就可以了。

先输出一个固定频率和占空比的PWM信号,周期1000(是ms还是us自己根据需要设置),占空比40%(高电平的时间/周期,周期1000,高电平就是400)

步骤:

1、设置时基单元的参数

设置预分频系数;

设置计数模式;

设置周期(自动重装载寄存器的值);

2、设置时钟源

可以用内部时钟源CK_CNT,外部时钟源ETR,一般用内部时钟源

3、设置输出比较

设置输出比较模式:PWM1,PWM2;

设置Pulse(有的人叫这个位占空比,其实不合适,因为占空比指的是高脉冲的比例,个人觉得Pulse叫切换点合适些,反正就是比较寄存器里面的值,用这个值去跟CNT比大小);

设置PWM输出极性

4、设置输出管脚

既然是PWM输出,那肯定需要实在的引脚

5、都设置好了,就开始运行

还是用STM32CubeMX结合HAL库开做吧(CubeMX是好用,个人还是推荐HAL库+寄存器同时开发,这样又是一片新的天地)

使用定时器TIM3的第一通道(CH1)来输出固定占空比的PWM信号

tim源文件


#include "tim.h"

TIM_HandleTypeDef htim3;

/* TIM3 init function */
void MX_TIM3_Init(void)
{

  TIM_ClockConfigTypeDef sClockSourceConfig = {0};

  TIM_OC_InitTypeDef sConfigOC = {0};


  htim3.Instance = TIM3;  //使用定时器3
  htim3.Init.Prescaler = 72-1; //72MHz时钟72分频,1MHz
  htim3.Init.CounterMode = TIM_COUNTERMODE_UP; //向上计数
  htim3.Init.Period = 1000-1; //周期1000
  htim3.Init.ClockDivision = TIM_CLOCKDIVISION_DIV1;
  htim3.Init.AutoReloadPreload = TIM_AUTORELOAD_PRELOAD_DISABLE;
  if (HAL_TIM_Base_Init(&htim3) != HAL_OK)
  {
    Error_Handler();
  }
  
  sClockSourceConfig.ClockSource = TIM_CLOCKSOURCE_INTERNAL;  //选择内部时钟作为时钟源
  if (HAL_TIM_ConfigClockSource(&htim3, &sClockSourceConfig) != HAL_OK)
  {
    Error_Handler();
  }
  
  if (HAL_TIM_PWM_Init(&htim3) != HAL_OK)
  {
    Error_Handler();
  }
  
  sConfigOC.OCMode = TIM_OCMODE_PWM1;//PWM模式1
  sConfigOC.Pulse = 400; //占空比400/1000
  sConfigOC.OCPolarity = TIM_OCPOLARITY_HIGH; //输出极性设置为高
  sConfigOC.OCFastMode = TIM_OCFAST_DISABLE;
  if (HAL_TIM_PWM_ConfigChannel(&htim3, &sConfigOC, TIM_CHANNEL_1) != HAL_OK)
  {
    Error_Handler();
  }
  HAL_TIM_MspPostInit(&htim3);
}


void HAL_TIM_Base_MspInit(TIM_HandleTypeDef* tim_baseHandle)
{
  if(tim_baseHandle->Instance==TIM3)
  {
    __HAL_RCC_TIM3_CLK_ENABLE(); //使能定时器的时钟
  }
}
void HAL_TIM_MspPostInit(TIM_HandleTypeDef* timHandle)
{
  GPIO_InitTypeDef GPIO_InitStruct = {0};
  if(timHandle->Instance==TIM3)
  {
    __HAL_RCC_GPIOA_CLK_ENABLE();
    
    //TIM3 PWM 输出引脚配置
    GPIO_InitStruct.Pin = GPIO_PIN_6;  //PA6对应CH1
    GPIO_InitStruct.Mode = GPIO_MODE_AF_PP;
    GPIO_InitStruct.Speed = GPIO_SPEED_FREQ_LOW;
    HAL_GPIO_Init(GPIOA, &GPIO_InitStruct);
  }
}

void HAL_TIM_Base_MspDeInit(TIM_HandleTypeDef* tim_baseHandle)
{
  if(tim_baseHandle->Instance==TIM3)
  {
    __HAL_RCC_TIM3_CLK_DISABLE();
  }
}

在main函数中加一行代码开启PWM输出即可,当然,也可以把这一行代码放在定时器的初始化函数中。


HAL_TIM_PWM_Start(&htim3,TIM_CHANNEL_1);

使用定时器TIM3输出一个占空比可调的PWM信号

如果要输出一个占空比可调的PWM信号,其实也很简单,就是改变对应通道的比较捕获寄存器的值就可以了,其他的跟固定占空比信号的设置一样就行。

这个改变占空比就是改变CCR寄存器的值,

CH1对应的是CCR1(捕获比较寄存器1)

CH2对应的是CCR2(捕获比较寄存器2)

CH3对应的是CCR3(捕获比较寄存器3)

CH3对应的是CCR4(捕获比较寄存器4)

HAL库正好也提供了设置CCR寄存器的函数

__HAL_TIM_SET_COMPARE(__HANDLE__, __CHANNEL__, __COMPARE__)


//__HANDLE__,就是使用的定时器,比如&htim3
//__CHANNEL__,就是对应的通道,比如TIM_CHANNEL_1
//__COMPARE__,就是要设置到比较捕获寄存器的值
#define __HAL_TIM_SET_COMPARE(__HANDLE__, __CHANNEL__, __COMPARE__) \
  (((__CHANNEL__) == TIM_CHANNEL_1) ? ((__HANDLE__)->Instance->CCR1 = (__COMPARE__)) :\
   ((__CHANNEL__) == TIM_CHANNEL_2) ? ((__HANDLE__)->Instance->CCR2 = (__COMPARE__)) :\
   ((__CHANNEL__) == TIM_CHANNEL_3) ? ((__HANDLE__)->Instance->CCR3 = (__COMPARE__)) :\
   ((__HANDLE__)->Instance->CCR4 = (__COMPARE__)))

在前面例程的基础上,在main函数的大循环里面加一段代码


int main(void)
{
  HAL_Init();

  SystemClock_Config();

  MX_GPIO_Init();
  MX_TIM3_Init();

  HAL_TIM_PWM_Start(&htim3,TIM_CHANNEL_1);

  while (1)
  {
    //设置捕获比较寄存器的值,每隔10ms加1,实现占空比一直动态调节
    __HAL_TIM_SET_COMPARE(&htim3,TIM_CHANNEL_1,cmp++); 
    
    //占空比到1000后,有从0开始,慢慢增加
    if(cmp ==1000)
    {
      cmp = 0;
    }
    //延时10ms
    HAL_Delay(10);
  }

}

工程链接:

STM32F103ZET6定时器TIM3,通道1,PWM输出,占空比可调资源-CSDN文库

如果用中断的方式改变占空比,就需要在main函数中调用PWM中断开始函数,然后编写回调函数,基本代码如下,细节部分自己去研究一下。


HAL_TIM_PWM_Start_IT(&htim3,TIM_CHANNEL_1);


extern uint16_t cmp;
void HAL_TIM_PWM_PulseFinishedCallback(TIM_HandleTypeDef *htim)
{
  __HAL_TIM_SET_COMPARE(&htim3, TIM_CHANNEL_1, cmp++);
  if(cmp == 1000)
  {
    cmp = 0;
  }
}

STM32单片机的定时器内容太多,参考手册总共1134页,定时器的内容就有200页,可见内容之多。主要是因为定时器的应用领域太多,功能太强大,还是要慢慢的学下去。。。。。。文章来源地址https://www.toymoban.com/news/detail-761246.html

到了这里,关于STM32学习----通用定时器的应用(PWM)的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • stm32f407关于通用定时器各种函数——PWM(二)

            定时器产生PWM:在计数器频率固定时,PWM 频率由 自动重载寄存器(TIMx_ARR) 的值决定,其占空比由 捕获/比较寄存器(TIMx_CCRx) 的值决定         定时器工作在递增计数模式,纵轴是计数器的计数值 CNT,横轴表示时。当 CNT=CCRx 时,IO 输出高电平(逻辑 1);

    2024年02月08日
    浏览(31)
  • STM32CubeMX教程6 TIM 通用定时器 - 生成PWM波

    开发板(STM32F407G-DISC1) STM32CubeMX软件(Version 6.10.0) keil µVision5 IDE(MDK-Arm) ST-LINK/V2驱动 逻辑分析仪nanoDLA 使用STM32CubeMX软件配置STM32F407 通用定时器生成可变占空比PWM波形,并将其输出到LED灯引脚实现呼吸灯效果 STM32F407有10个通用定时器,其中TIM2、TIM3、TIM4和TIM5有4个捕获

    2024年02月03日
    浏览(19)
  • STM32学习笔记(六)丨TIM定时器及其应用(输入捕获丨测量PWM波形的频率和占空比)

    ​  本次课程采用单片机型号为STM32F103C8T6。 ​  课程链接:江科大自化协 STM32入门教程   往期笔记链接:   STM32学习笔记(一)丨建立工程丨GPIO 通用输入输出   STM32学习笔记(二)丨STM32程序调试丨OLED的使用   STM32学习笔记(三)丨中断系统丨EXTI外部中断

    2023年04月19日
    浏览(24)
  • STM32 CubeMX学习实验13:定时器输入捕获获取PWM频率

    使用定时器5作为输入捕获定时器,将通道1(PA0)设置为输入捕获,设置预分频器和计数值,这里设置为1us计数一次,最大可以捕获周期为0xFFFFFFFFus的PWM,所以一般不需要考虑溢出的问题,使能自动重装载。 使能定时器中断,选择合适的优先级,   将引脚设置下拉,保证没

    2024年03月14日
    浏览(39)
  • STM32 HAL库 Timer(定时器)+DMA输出PWM底层配置过程学习

    本文使用的芯片型号是STM32G030,写本文的目前是想记录学习下Timer借助DMA生成可变占空比PWM时的底层配置过程。 使用TIM1,配置就只改了图上的配置,系统时钟用的16M,分频选择15(16-1),自动重装载寄存器ARR选择999(1000-1),那么生成的就是1kHz的PWM,这里为什么要减1,因为这俩是

    2024年04月09日
    浏览(35)
  • 定时器详解 -- 定时器中断、PWM输出 --stm32

    STM32F103系列芯片拥有多种定时器,包括基本定时器、通用定时器和高级定时器,每种定时器都具有一些特定的功能。 向上计数:计数器从0计数到自动重装载值(ARR),然后重新从0开始计数并且产生一个计数器溢出事件。 向下计数:计数器从自动重装载值(ARR)开始向下计数

    2024年02月11日
    浏览(27)
  • STM32学习笔记 & 通用定时器TIM3~TIM5 13

    通用定时器是一个通过可编程预分频器驱动的 16 位自动装载计数器构成。 它适用于多种场合,包括测量输入信号的脉冲长度 ( 输入捕获 ) 或者产生输出波形 ( 输出比较和PWM)。 使用定时器预分频器和 RCC 时钟控制器预分频器,脉冲长度和波形周期可以在几个微秒到几个毫秒间

    2024年02月05日
    浏览(41)
  • STM32学习笔记(4) 高级定时器-两路互补的PWM输出(带死区和刹车控制)

    目录 1.实验目的 2.实验效果 3.理论部分 3.1时钟源 3.2时基单元 3.3输入捕获 4.程序流程 4.1GPIO初始化结构体 4.2时基初始化结构体 4.3输出比较结构体 4.4刹车和死区结构体的初始化 5.程序源码 使用高级定时器,输出两路互补的PWM输出,需要有带死区和不带死区两种情况 图1:不带

    2024年02月13日
    浏览(16)
  • STM32定时器输出PWM脉冲

    一、什么是PWM脉冲? PWM (Pulse Width Modulation) 脉冲宽度调制,通常应用于惯性系统中,( 不是不能即停的设备,因为如果设备不具有关心,那么运动是断断续续的,不具有连续性 ),通过脉宽调节输出不同的波形作用域受控对象。 二、 PWM中三个重要的参数 1.频率=1/Ts ;这个很

    2024年02月15日
    浏览(25)
  • 【STM32】定时器PWM模式详解

    PWM模式: PWM模式1,向上计数时,PWM信号从 有效电平 变为 无效电平 PWM模式2,向上计数时,PWM信号从 无效电平 变为 有效电平 PWM极性: 极性为高时, 高电平为有效电平 ,低电平为无效电平 极性为低时, 低电平为有效电平 ,高电平为无效电平 中心对齐模式(先向上再向下

    2024年02月09日
    浏览(21)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包