你的浏览器版本过低,可能导致网站不能正常访问!
为了你能正常使用网站功能,请使用这些浏览器。

基于STM32F1的CAN通信之定时器

[复制链接]
攻城狮Melo 发布时间:2023-10-23 17:55
一、什么是定时器
关于什么是定时器,简单来讲,就是是用来定时的。STM32F103ZET6有两个基本定时器TIM6和TIM7,四个通用定时器TIM2~TIM5和两个高级定时器TIM1,TIM8。每一个定时器都是完全独立的,不共享任何资源。

根据中文参考手册介绍,基本定时器最为简单,类似于51单片机的定时器。通用定时器在基本定时器的基础上增加了输入捕获和输出比较功能。高级定时器相比通用定时器,又增加了可编程死区互补输出,重复计数器等功能。

STM32F103ZET6的通用定时器是一个通过可编程预分频器驱动的16位自动装载计数器构成。使用定时器预分频器和RCC时钟控制器预分频器,脉冲长度和波形周期可以在几个微秒到几个毫秒间调整。

这里介绍一下对于定时器的个人理解。定时器的定时原理实际可以理解为按照固定的频率数数。按照固定频率就说明定时器一定要有输入时钟。比如输入为一个1KHz的时钟,那么数一个数的时间就是1ms。另外,数数也不是无限地数,数值有一个上限。可以规定是从0开始数到上限值,还是从上限值数到0。而且每次数到头,需要重新开始。比如,需要控制灯亮200ms。那么只需要在点亮LED之后,等到数到200时熄灭即可。当数到上限值或者数到0时,重新开始数。


二、定时器有什么用
定时器有许多用途,以通用定时器为例。它可以测量输入信号的脉冲宽度,产生PWM波。此外定时器也可以用于触发ADC采集,按键检测等方面。

中文参考手册介绍如下

微信图片_20231023175513.png

中文参考手册对于通用定时器功能的描述


三、通用定时器详细介绍
速成选手可以线跳过这一部分,直接看下面,后来再返回来仔细看。

3.1 时钟来源
根据中文参考手册,通用定时器的时钟来源有四个。
• 内部CK_INT
• 外部触发时钟输入TIMx_ETR(外部时钟模式2)
• 内部触发输入(ITRx):使用一个定时器作为另一个定时器的预分频器
• 外部引脚输入(外部时钟模式1)

微信图片_20231023175507.png

通用定时器时钟来源



通过配置TIMx_SMCR寄存器来选择,关于寄存器这里就不再详细介绍了,大家可以去看中文参考手册。

根据中文参考手册关于时钟的介绍,通用定时器挂接在APB1总线。对于APB1总线的时钟如下

微信图片_20231023175503.png

APB1时钟介绍



如果APB1的预分频系数为1,那么通用定时器的输入时钟频率为36MHz,否则为72MHz。但是通常APB1总线的预分频系数我们不会设置成1,所以通用定时器的时钟频率为72MHz。

3.2 预分频器,计数器,自动重装载寄存器

微信图片_20231023175452.png

预分频器,计数器和自动重装载寄存器



3.2.1 预分频器
预分频器是对时钟进行分频,范围是1~65536。比如通用定时器输入时钟频率为72MHz,此时,将预分频值设置为72,那么最终计数时的时钟频率为72MHz / 72 = 1MHz。

3.2.2 计数器
计数器就是用来计数的,计数值取值范围是0~65535。有三种计数方式:向上计数,向下计数,中央对齐模式(向上/向下计数)。

在向上计数模式中,计数器从0计数到自动加载值(TIMx_ARR计数器的内容),然后重新从0开始计数并且产生一个计数器溢出事件。

在向下计数模式中,计数器从自动装入的值(TIMx_ARR计数器的值)开始向下计数到0,然后从自动装入的值重新开始并且产生一个计数器向下溢出事件。

在中央对齐模式,计数器从0开始计数到自动加载的值(TIMx_ARR寄存器)−1,产生一个计数器溢出事件,然后向下计数到1并且产生一个计数器下溢事件;然后再从0开始重新计数。

3.2.3 自动重装载寄存器
比如,选择向下计数模式,初始值为2000。当计数到0时,会再次从2000开始向下计数。这就叫重装载。但是实际起作用的并不是这里的自动重装载寄存器,而是影子寄存器。关于影子寄存器这里就不再做介绍了,大家可以自行了解。

3.3 触发控制器
从图中的右上角可以注意到,有一个触发控制器。它可以用来触发一些外设,比如触发ADC采集,也可以用来给其他定时器提供时钟。

四、PWM
4.1 什么是PWM

PWM(脉冲宽度调制),它是一种利用微处理器的数字输出来对模拟电路进行控制的技术,也可以理解为是对模拟信号电平进行数字编码的方法。PWM可被应用于电机驱动,调光,通信等方面。

4.2 什么是占空比
一个PWM是有固定频率的,也就意味着周期一定,一个周期内有效电平持续时间占整个周期的比例可以称为占空比。比如一个周期100ms,其中50ms持续为有效电平,那么占空比就是50%。正是通过调节占空比,来调节电机转速,或者用不同占空比代表不同信号,用于通信。

4.3 STM32F1 PWM介绍
STM32F1系列单片机,除了基本定时器TIM6和TIM7外,都可以产生PWM输出。其中高级定时器TIM1和TIM8可以同时产生高达7路PWM输出。PWM输出其实就是对外输出占空比可调的方波,信号频率由自动重装载寄存器ARR的值决定,占空比由比较寄存器CCR的值决定。假设高电平为有效电平,见下图。ARR决定了周期(频率),CCR调节占空比。

微信图片_20231023175433.png

PWM示意图



根据中文参考手册介绍,STM32F1的PWM比较输出模式共有8种。脉冲宽度调制模式可以产生一个由TIMx_ARR寄存器确定频率、由TIMx_CCRx寄存器确定占空比的信号。在TIMx_CCMRx寄存器中的OCxM位写入’110’(PWM模式1)或’111’(PWM模式2),能够独立地设置每个OCx输出通道产生一路PWM。有关寄存器的内容,这里就不不再做详细介绍。

这里介绍一下8种输出模式中比较常用的两种PWM输出模式,PWM1和PWM2。其实这两种输出模式差别不大,只不过输出电平的极性不同。

TGA6C5PL18VL~RRI8{~94NG.png

4.4 PWM频率计算
频率 = (主时钟频率(72MHz) / (分频系数 + 1)) / 自动重装载值(单位为Hz)


五、通用定时器输出引脚

F44RWE@L[DPR%3U4AL8W[PR.png

其他几个定时器如下

微信图片_20231023175428.png

TIM2的PWM引脚



微信图片_20231023175415.png

TIM4的PWM引脚



微信图片_20231023175405.png

TIM5的PWM引脚

六、实战项目
这里以一个经典项目——呼吸灯,来一起熟悉一下定时器的配置和使用,要求灭—>亮—>灭,时间为2.5s。

6.1 呼吸灯
呼吸灯是指灯能够像人的呼吸一样,实现由暗到亮或由亮到暗的变化,通常用于消息提示功能,或者作为系统正在运行的提示。之前一篇博文介绍过三种呼吸灯的实现方式,这里针对普中核心板,来介绍一下如果实现呼吸灯。

6.2 实现思路
这里用两种方法来实现一下呼吸灯。分别是定时器的溢出中断和PWM。其实第一种和PWM类似,我非就是控制LED点亮时间。
• 定时器中断实现 配置好预分频系数和重装载值,使每0.25ms进入一次定时器中断,记录进入中断次数(count)。当进入次数满100次之后(2.5ms),控制LED点亮的变量(t)值加1。主函数的while(1)轮询中,如果t小于等于count的时候,LED点亮,否则LED熄灭。t的值累计100加次后(2.5s),开始递减,LED由亮到灭。控制t是递增还是递减的是一个标志位(flag),初始值为0,具体可以看程序设计。

• PWM 利用PWM实现呼吸灯就更加简单了,只需要不断调节占空比即可。

66.3 定时器配置
配置通用定时器,有以下步骤
• 使能定时器时钟
• 初始化定时器参数,包括自动重装载值,分频系数,计数方式等
• 设置中断类型,并使能
• 设置中断优先级,使能定时器中断通道
• 开启定时器
• 编写定时器中断服务函数

需要注意的是,配置预分频系数时,比如设置为6,实际是6 + 1。

定时时间T = 自动重装载值 * ((预分频系数 + 1) / 主时钟频率)。主时钟频率为72MHz。(为了避免误导,这里写的主时钟频率为72MHz是APB1总线分频系数不是1的前提下。)

6.4 定时器中断实现呼吸灯
定时器配置程序如下,使用定时器2,控制LED1实现呼吸灯,由灭—>亮—>灭,时间为5秒。
  1. /*
  2. *==============================================================================
  3. *函数名称:TIM2_Iint
  4. *函数功能:初始化定时器2
  5. *输入参数:per:自动重装载值;psc:预分频系数
  6. *返回值:无
  7. *备  注:无
  8. *==============================================================================
  9. */
  10. void TIM2_Iint (u16 per,u16 psc)
  11. {
  12.     // 结构体定义
  13.     TIM_TimeBaseInitTypeDef TIM_TimeBaseInitStructure;
  14.     NVIC_InitTypeDef NVIC_InitStructure;
  15.    
  16.     RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM2,ENABLE);   // 使能TIM2时钟
  17.    
  18.     TIM_TimeBaseInitStructure.TIM_Period = per;   // 自动装载值
  19.     TIM_TimeBaseInitStructure.TIM_Prescaler = psc;   // 分频系数
  20.     TIM_TimeBaseInitStructure.TIM_ClockDivision = TIM_CKD_DIV1;   // 不分频
  21.     TIM_TimeBaseInitStructure.TIM_CounterMode=TIM_CounterMode_Up;   // 设置向上计数模式
  22.     TIM_TimeBaseInit(TIM2,&TIM_TimeBaseInitStructure);
  23.    
  24.     TIM_ITConfig(TIM2,TIM_IT_Update,ENABLE);   // 开启定时器中断
  25.     TIM_ClearITPendingBit(TIM2,TIM_IT_Update);   // 使能更新中断
  26.    
  27.     NVIC_InitStructure.NVIC_IRQChannel = TIM2_IRQn;   // 定时器中断通道
  28.     NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority=2;   // 抢占优先级
  29.     NVIC_InitStructure.NVIC_IRQChannelSubPriority =3;   // 子优先级
  30.     NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;   // IRQ通道使能
  31.     NVIC_Init(&NVIC_InitStructure);
  32.    
  33.     TIM_Cmd(TIM2,ENABLE);   // 使能定时器
  34. }
复制代码

初始化时定时器的程序如下
  1. TIM2_Iint(250,71);   // TIM2初始化
复制代码

预分频系数为71 + 1 = 72,计数到250进入一次中断,也就是0.25ms进入一次中断。累计进入100次(25ms)中断开始调节一点LED的亮度。由灭到亮,累计调节100次(2.5s)。主函数和中断服务函数如下
  1. u8 gTimIrqCunt = 0;   // 进入中断次数计数变量
  2. u8 gLedLightCtrl = 0;   // LED亮度控制变量
  3. u8 gLedFlag = 0;   // LED亮灭控制标志位,0:灭—>亮;1:亮—>灭

  4. int main(void)
  5. {
  6.     Med_Mcu_Iint();   // 系统初始化
  7.    
  8.     while(1)
  9.   {
  10.         if (gLedLightCtrl <= gTimIrqCunt)
  11.         {
  12.             Med_Led_StateCtrl (LED1,LED_OFF);   // 熄灭LED1
  13.         }
  14.         if (gLedLightCtrl > gTimIrqCunt)
  15.         {
  16.             Med_Led_StateCtrl (LED1,LED_ON);   // 熄灭LED1
  17.         }
  18.     }
  19. }

  20. // TIM2中断服务函数
  21. void TIM2_IRQHandler(void)   // TIM2中断
  22. {
  23.     // 产生更新中断
  24.     if (TIM_GetITStatus(TIM2, TIM_IT_Update) != RESET)
  25.     {
  26.         gTimIrqCunt = gTimIrqCunt + 1;   // 进入中断次数加1
  27.         
  28.         // 累计进入100次中断,且是由灭到亮
  29.         if (gTimIrqCunt >= 100 && gLedFlag < 100)
  30.         {
  31.             gTimIrqCunt = 0;   // 清零进入中断计数变量
  32.             gLedLightCtrl = gLedLightCtrl + 1;   // LED亮度控制变量加1
  33.             gLedFlag = gLedFlag + 1;   // LED亮灭控制标志位加1
  34.         }
  35.         
  36.         // 累计进入100次中断,且是由亮到灭
  37.         if (gTimIrqCunt >= 100 && gLedFlag >= 100)
  38.         {
  39.             gTimIrqCunt = 0;   // 清零进入中断计数变量
  40.             gLedLightCtrl = gLedLightCtrl - 1;   // LED亮度控制变量加1
  41.             gLedFlag = gLedFlag + 1;   // LED亮灭控制标志位加1
  42.         }
  43.         
  44.         // 一个亮灭周期结束
  45.         if (gLedFlag >= 200)
  46.         {
  47.             gLedFlag = 0;   // 清零LED亮灭控制标志位
  48.         }
  49.     }
  50.    
  51.     TIM_ClearITPendingBit(TIM2, TIM_IT_Update);   // 清除TIM2更新中断标志
  52. }
复制代码

6.5 使用PWM实现呼吸灯

PWM配置步骤
• 使能定时器以及GPIO时钟,设置引脚复用映射
• 初始化定时器参数,包括自动重装载值,分频系数,计数方式等
• 初始化PWM输出参数,包括PWM模式,输出极性,使能等
• 开启定时器
• 修改CCRx的值来修改占空比
• 使能TIMx在CCRx上的预装载寄存器
• 使能TIMx在ARR上的预装载寄存器允许位

TIM3的通道1配置程序如下这里对引脚进行了重映射。
  1. /*
  2. *==============================================================================
  3. *函数名称:TIM3_CH1_PWM_Init
  4. *函数功能:初始化定时器3的PWM通道1
  5. *输入参数:per:自动重装载值;psc:预分频系数
  6. *返回值:无
  7. *备  注:无
  8. *==============================================================================
  9. */
  10. void TIM3_CH1_PWM_Init (u16 per,u16 psc)
  11. {
  12.     // 结构体定义
  13.     TIM_TimeBaseInitTypeDef TIM_TimeBaseInitStructure;
  14.     TIM_OCInitTypeDef TIM_OCInitStructure;
  15.     GPIO_InitTypeDef GPIO_InitStructure;
  16.    
  17.     // 开启时钟
  18.     RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOC,ENABLE);
  19.     RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM3,ENABLE);
  20.     RCC_APB2PeriphClockCmd(RCC_APB2Periph_AFIO,ENABLE);
  21.    
  22.     // 初始化GPIO
  23.     GPIO_InitStructure.GPIO_Pin = GPIO_Pin_6;
  24.     GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
  25.     GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;   // 复用推挽输出
  26.     GPIO_Init(GPIOC,&GPIO_InitStructure);
  27.    
  28.     GPIO_PinRemapConfig(GPIO_FullRemap_TIM3,ENABLE);   // 改变指定管脚的映射
  29.    
  30.     // 初始化定时器参数
  31.     TIM_TimeBaseInitStructure.TIM_Period = per;   // 自动装载值
  32.     TIM_TimeBaseInitStructure.TIM_Prescaler = psc;   // 分频系数
  33.     TIM_TimeBaseInitStructure.TIM_ClockDivision = TIM_CKD_DIV1;
  34.     TIM_TimeBaseInitStructure.TIM_CounterMode = TIM_CounterMode_Up;   // 设置向上计数模式
  35.     TIM_TimeBaseInit(TIM3,&TIM_TimeBaseInitStructure);
  36.    
  37.     // 初始化PWM参数
  38.     TIM_OCInitStructure.TIM_OCMode = TIM_OCMode_PWM1;   // 比较输出模式
  39.     TIM_OCInitStructure.TIM_OCPolarity = TIM_OCPolarity_Low;   // 输出极性
  40.     TIM_OCInitStructure.TIM_OutputState = TIM_OutputState_Enable;   // 输出使能
  41.     TIM_OC1Init(TIM3,&TIM_OCInitStructure);   // 输出比较通道1初始化
  42.    
  43.     TIM_OC1PreloadConfig(TIM3,TIM_OCPreload_Enable);   // 使能TIMx在 CCR1 上的预装载寄存器
  44.     TIM_ARRPreloadConfig(TIM3,ENABLE);   // 使能预装载寄存器
  45.    
  46.     TIM_Cmd(TIM3,ENABLE);   // 使能定时器
  47.         
  48. }
复制代码

实现呼吸灯时,只需要在main函数中不断调整占空比即可,调整占空比的函数为
  1. void TIM_SetCompare1(TIM_TypeDef* TIMx, uint16_t Compare1)
复制代码


这里main函数就不在列出来了。需要注意的是,设置的CCR的值,不能超过自动重装载值 - 1。


七、拓展

之前介绍按键检测时,介绍过检测按键长短按的方法。当时比较简单粗暴,这里介绍另一种,使用定时器来判断按键WK UP的长短按。假设规定,按下时间在10ms~500ms之间为短按,按下时间大于等于1s,为长按。短按LED1点亮,长按LED1熄灭。之前是利用delay来实现时间控制,现在改用定时器实现时间控制,但是基本思路都是相同的。

关于按键部分的程序这里就不再做介绍了。首先配置定时器,10ms进入一次更新中断,预分频系数为72,自动重装载值为10000。使用TIM2,定时器配置程序和上面一样,初始化程序如下填写
  1. TIM2_Iint(10000,71);   // TIM2初始化
复制代码

main函数以及中断服务函数如下
  1. u32 gKeyDownTimeCunt = 0;   // 按键按下时间计数变量
  2. u8 gKeyLongFlag = 0;   // 按键长按标志位
  3. u8 gKeyShotFlag = 0;   // 按键短按标志位

  4. int main(void)
  5. {
  6.     Med_Mcu_Iint();   // 系统初始化
  7.    
  8.     while(1)
  9.   {
  10.         // 短按
  11.         if (gKeyShotFlag == 1)
  12.         {
  13.             Med_Led_StateCtrl (LED1,LED_ON);   // 点亮LED1
  14.             gKeyShotFlag = 0;   // 清除短按标志位
  15.         }
  16.         
  17.         // 长按
  18.         if (gKeyLongFlag == 1)
  19.         {
  20.             Med_Led_StateCtrl (LED1,LED_OFF);   // 熄灭LED1
  21.             gKeyLongFlag = 0;   // 清除长按标志位
  22.         }
  23.     }
  24. }

  25. // TIM2中断服务函数
  26.     void TIM2_IRQHandler(void)   // TIM2中断
  27.     {
  28.         // 产生更新中断
  29.         if (TIM_GetITStatus(TIM2, TIM_IT_Update) != RESET)
  30.         {
  31.             if (KEY_UP == 1)
  32.             {
  33.                 gKeyDownTimeCunt = gKeyDownTimeCunt + 1;   // 时间计数变量加1
  34.             }
  35.             // 松开后
  36.             else
  37.             {
  38.                 // 短按
  39.                 if (1 <= gKeyDownTimeCunt && gKeyDownTimeCunt <= 50)
  40.                 {
  41.                     gKeyDownTimeCunt = 0;   // 清除时间计数变量
  42.                     gKeyShotFlag = 1;   // 短按标志位置1
  43.                 }
  44.                 // 长按
  45.                 if (gKeyDownTimeCunt >= 100)
  46.                 {
  47.                     gKeyDownTimeCunt = 0;   // 清除时间计数变量
  48.                     gKeyLongFlag = 1;   // 长按标志位置1
  49.                 }
  50.             }
  51.         }
  52.         
  53.         TIM_ClearITPendingBit(TIM2, TIM_IT_Update);   // 清除TIM2更新中断标志
  54.     }
复制代码


转载自:二土
如有侵权请联系删除


收藏 评论0 发布时间:2023-10-23 17:55

举报

0个回答

所属标签

相似分享

官网相关资源

关于
我们是谁
投资者关系
意法半导体可持续发展举措
创新与技术
意法半导体官网
联系我们
联系ST分支机构
寻找销售人员和分销渠道
社区
媒体中心
活动与培训
隐私策略
隐私策略
Cookies管理
行使您的权利
官方最新发布
STM32N6 AI生态系统
STM32MCU,MPU高性能GUI
ST ACEPACK电源模块
意法半导体生物传感器
STM32Cube扩展软件包
关注我们
st-img 微信公众号
st-img 手机版