STM32通定时器时间设置步骤和计算公式方法寄存器值

发布者:innovator7最新更新时间:2022-06-01 来源: eefocus关键字:STM32  时间设置 手机看文章 扫描二维码
随时随地手机看文章

一、STM32通用定时器原理                        

STM32 系列的CPU,有多达8个定时器,其中TIM1和TIM8是能够产生三对PWM互补输出的高级定时器,常用于三相电机的驱动,它们的时钟由APB2的输出产生。其它6个为普通定时器,时钟由APB1的输出产生。


下图是STM32参考手册上时钟分配图中,有关定时器时钟部分的截图:

从图中可以看出,定时器的时钟不是直接来自APB1或APB2,而是来自于输入为APB1或APB2的一个倍频器,图中的蓝色部分。


下面以通用定时器2的时钟说明这个倍频器的作用:当APB1的预分频系数为1时,这个倍频器不起作用,定时器的时钟频率等于APB1的频率;当APB1的预分频系数为其它数值(即预分频系数为2、4、8或16)时,这个倍频器起作用,定时器的时钟频率等于APB1的频率两倍。


可能有同学还是有点不理解,OK,我们举一个例子说明。假定AHB=36MHz,因为APB1允许的最大频率为36MHz,所以APB1的预分频系数可以取任意数值;

当预分频系数=1时,APB1=36MHz,TIM2~7的时钟频率=36MHz(倍频器不起作用);

当预分频系数=2时,APB1=18MHz,在倍频器的作用下,TIM2~7的时钟频率=36MHz。


有人会问,既然需要TIM2~7的时钟频率=36MHz,为什么不直接取APB1的预分频系数=1?答案是:APB1不但要为TIM2~7提供时钟,而且还要为其它外设提供时钟;设置这个倍频器可以在保证其它外设使用较低时钟频率时。


Stm32外设用户手册,如图:

再举个例子:当AHB=72MHz时,APB1的预分频系数必须大于2,因为APB1的最大频率只能为36MHz。如果APB1的预分频系数=2,则因为这个倍频器,TIM2~7仍然能够得到72MHz的时钟频率。能够使用更高的时钟频率,无疑提高了定时器的分辨率,这也正是设计这个倍频器的初衷。

              

二、STM32通用定时器编程

定时器编程,就是中断的编程。因为使用定时器必定要使用到中断。

  

步骤一:RCC_Configuration();//设置系统时钟,包括时钟RCC的配置,倍频到72MHZ。

      

步骤二: GPIO的配置,使用函数为GPIO_cfg();,该函数的实现如下:

 

void GPIO_cfg()

 

{

 

       GPIO_InitTypeDef GPIO_InitStructure;

 

      RCC_APB2PeriphClockCmd( RCC_APB2Periph_USART1 |RCC_APB2Periph_GPIOA | RCC_APB2Periph_GPIOB | RCC_APB2Periph_GPIOC | RCC_APB2Periph_GPIOD , ENABLE);

 

       

 

       GPIO_InitStructure.GPIO_Pin = GPIO_Pin_6;                 //选择引脚6

 

       GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz; //输出频率最大50MHz

 

      GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP; //带上拉电阻输出

 

       GPIO_Init(GPIOC,&GPIO_InitStructure);

 

}


实际上定时器的讲解,不需要配置GPIO的引脚,只是我们在定时器实验中,

使用每隔一秒点亮一次LED灯来做实验,所以需要配置对应GPIO的引脚。

 

步骤三:嵌套中断控制器的配置,我们照样使用函数NVIC_Config();只是初始化的过程略有不同。这里我们也把函数实现列出来:


从以上函数实现来看,实际上只是改动了结构体成员NVIC_IRQChannel的值,现在需要的通道是TIM2的通道,因此初始化值为TIM2_IRQChannel。从这里也可以看出,这个函数实际上可以看做一个模型,可以拿去别的程序中改动后直接使用。

 

void NVIC_cfg()

 

{

 

       NVIC_InitTypeDef NVIC_InitStructure;

 

        //选择中断分组1

 

        NVIC_PriorityGroupConfig(NVIC_PriorityGroup_1);

 

 

 

        //选择TIM2的中断通道

 

        NVIC_InitStructure.NVIC_IRQChannel = TIM2_IRQChannel;     

 

        //抢占式中断优先级设置为0

 

        NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 0;

 

       //响应式中断优先级设置为0

 

        NVIC_InitStructure.NVIC_IRQChannelSubPriority = 0;

 

        //使能中断

 

        NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;

 

        NVIC_Init(&NVIC_InitStructure);

 

}


步骤四:定时器的初始化配置,使用Timer_Config();。OK,关键部分出来了。

我们来看下实现过程:


TIMER_cfg(); //定时器的配置

 

       //开启定时器2

 

 TIM_Cmd(TIM2,ENABLE);

 

voidTimer_Config(void)

 

    RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM2,ENABLE); 

 

    TIM_DeInit(TIM2);

 

    TIM_TimeBaseStructure.TIM_Period=2000-1;  //自动重装载寄存器的值

 

    TIM_TimeBaseStructure.TIM_Prescaler=(36000-1);         //时钟预分频数

 

    TIM_TimeBaseStructure.TIM_ClockDivision=TIM_CKD_DIV1;  //采样分频

 

     TIM_TimeBaseStructure.TIM_CounterMode=TIM_CounterMode_Up; //向上计数模式

    TIM_TimeBaseInit(TIM2,&TIM_TimeBaseStructure);

 

    TIM_ClearFlag(TIM2,TIM_FLAG_Update);               //清除溢出中断标志

 

    TIM_ITConfig(TIM2,TIM_IT_Update,ENABLE);

 

     TIM_Cmd(TIM2,ENABLE);                              /开启时钟

 

}


我们每个语句都来解释一下。首先我们想使用定时器,就必须使能定时器的时钟,这就是函数 RCC_APB1PeriphClockCmd();,通过它开启 RCC_APB1Periph_TIM2。


TIM_DeInit(TIM2);该函数主要用于复位TIM2定时器,使之进入初始状态。

然后我们对自动重装载寄存器赋值,TIM_Period的大小实际上表示的是需要经过TIM_Period次计数后才会发生一次更新或中断。接下来需要设置时钟预分频数TIM_Prescaler,这里有一个公式,我们举例来说明:例如时钟频率=72MHZ/(时钟预分频+1)。说明当前设置的这个TIM_Prescaler,直接决定定时器的时钟频率。通俗点说,就是一秒钟能计数多少次。比如算出来的时钟频率是2000,也就是

一秒钟会计数2000次,而此时如果TIM_Period设置为4000,即4000次计数后就会中断一次。由于时钟频率是一秒钟计数2000次,因此只要2秒钟,就会中断一次。

再往后的代码,还有一个需要注意的,TIM_TimeBaseStructure.TIM_CounterMode=TIM_CounterMode_Up;就是我们一般采用向上计数模式,即每次计数就会加1,直到寄存器溢出发生中断为止。最后别忘了,需要使能定时器!!


发生中断时间=(TIM_Prescaler+1)* (TIM_Period+1)/FLK


用上述公式可算出:发生中断时间 (2000-1+1)*(36000-1+1)/72000000=1 秒

步骤五:编写中断服务程序。同样需要注意的,一进入中断服务程序,第一步要做的,就是清除掉中断标志位。由于我们使用的是向上溢出模式,因此使用

的函数应该是:TIM_ClearITPendingBit(TIM2,TIM_FLAG_Update);。


STM32开发板实现的中断服务程序如下:

每隔一秒,发生中断时,进入此中断函数执行程序,让LED闪一下,此中断程序所在文件stm32f10x_it.c


/*******************************************************************************

* Function Name  : TIM2_IRQHandler

* Description    : This function handles TIM2 global interrupt request.

* Input          : None

* Output         : None

* Return         : None

******************************************************************************/

 

 

 

void TIM2_IRQHandler(void)

 

{

 

   u8 ReadValue;

 

 

 

   if(TIM_GetITStatus(TIM2, TIM_IT_Update) != RESET)    //检测是否发生溢出更新事件

 

     {

 

        TIM_ClearITPendingBit(TIM2 , TIM_FLAG_Update);//清除TIM2的中断待处理位

 

              //UART2_TX485_Puts("123450");

 

 

 

              /*调试定时器测试*/

 

              ReadValue = GPIO_ReadOutputDataBit(GPIOC,GPIO_Pin_6);

 

        if(ReadValue == 0)                   //若该端口当前为低电平,

 

         {

 

            GPIO_SetBits(GPIOC,GPIO_Pin_6);//将其改为高电平输出 ;

 

         }    

 

 

 

         else                            //若该端口当前为高电平,

 

         {

 

            GPIO_ResetBits(GPIOC,GPIO_Pin_6);   //将其改为低电平输出;

 

         }

 

        

 

     }        

 

}


普通定时器工作原理图

 

编译完成的代码下载在我的资源:http://download.csdn.net/detail/yx_l128125/4508425

https://blog.csdn.net/yx_l128125/article/details/7879506


关键字:STM32  时间设置 引用地址:STM32通定时器时间设置步骤和计算公式方法寄存器值

上一篇:STM32F103的启动配置
下一篇:stm32定时器时钟源时钟选择,重点是外部时钟源1模式的理解

推荐阅读最新更新时间:2024-11-12 23:56

STM32处理器输入捕获分析
前言: 1.博文基于ARM Cortex-M3内核的STM32F103ZET6芯片和标准3.5.0库; 2.如有不足之处,还请多多指教; * 一 基本知识 * 1. 输入捕获的功能:用来测量脉宽或者测量信号频率; 2. 输了TIM6和TIM7外,其他定时器都有输入捕获功能; 3. 通用定时器输入捕获中断和定时器更新中断公用同一个中断函数; 二 侧脉宽工作原理 如何获取一个脉冲的宽度(比如高电平): 1. 开启并设置好定时器的时钟源 ,频率为F; 2. 输入需要检测的脉冲; 3. 利用CNT计数器计算一个脉冲的上升沿和下降沿之间的脉宽 三 相关寄存器 TIMx_ARR,TIMx_PSC,TIMx_CCMRx,TIMx_CCERx,T
[单片机]
<font color='red'>STM32</font>处理器输入捕获分析
STM32上电启动过程分析(START_TEST代码实例)
一、概述 1、说明 每一款芯片的启动文件都值得去研究,因为它可是你的程序跑的最初一段路,不可以不知道。通过了解启动文件,我们可以体会到处理器的架构、指令集、中断向量安排等内容,是非常值得玩味的。 STM32作为一款高端 Cortex-M3系列单片机,有必要了解它的启动文件。打好基础,为以后优化程序,写出高质量的代码最准备。 本文以一个实际测试代码--START_TEST为例进行阐述。 2、整体过程概括 STM32整个启动过程是指从上电开始,一直到运行到 main函数之间的这段过程,步骤为(以使用微库为例): ①上电后硬件设置SP、PC ②设置系统时钟 ③软件设置SP ④加载.data、.bss,并初始化栈区 ⑤跳转到C文件
[单片机]
<font color='red'>STM32</font>上电启动过程分析(START_TEST代码实例)
STM32 LED跑马灯-位操作
一,什么是位操作   之前我们已经介绍了库函数和寄存器控制LED跑马灯,其实无论使用哪一种方法(包括操作BSRR,BRR寄存器的方式)最终都是通过操作GPIO_ODR寄存器(32位寄存器只使用低16位)响应的位为该IO口赋值     那么什么是位操作?我们知道GPIO_ODR寄存器的每一位对应一个IO口的电平操作,而每一位实际是一个IO口地址的映射,位操作就是跨越寄存器映射,直接为这个地址进行赋值      在LED跑马灯-位操作的实验中我们将使用位操作的方式控制IO口输出高低电平 二,位与别名映射关系 1,支持位操作的两个内存区范围: 0x2000_0000-0x200F_FFFF // SRAM区中的最低1M 0x
[单片机]
<font color='red'>STM32</font> LED跑马灯-位操作
STM32的PVD、POR、PDR简介
先看一下这2张截自STM32参考手册的图片: 下面这张表截自STM32F103x8/B的数据手册,对上图的参数给出了具体数值: 下面对上面2张图和表格中的数据做一个简要的解释: 1)PVD = Programmable Votage Detector 可编程电压监测器 它的作用是监视供电电压,在供电电压下降到给定的阀值以下时,产生一个中断,通知软件做紧急处理。在给出表格的上半部分就是可编程的监视阀值数据。当供电电压又恢复到给定的阀值以上时,也会产生一个中断,通知软件供电恢复。供电下降的阀值与供电上升的PVD阀值有一个固定的差值,这就是表中的VPVDhyst(PVD迟滞)这个参数,通过列出的PVD阀值数据
[单片机]
<font color='red'>STM32</font>的PVD、POR、PDR简介
STM32经keil mdk编译产生的.map文件分析
昨天遇到堆大小不足的问题,扩大Heap的size觉得有必要去分析下整个程序的内存占用的详细情况,仅仅从生成的Code + RO + RW +ZI提示太笼统,无法清楚的看见我写的每一个函数,定义的每一个全局变量、静态变量的具体位置、大小等等,经各路大神的提示,我去仔细研究了下.map文件,上网查阅了相关的资料,本文主要受博主 非著名码农 的启示,原文地址 http://blog.csdn.net/ropai/article/details/7493168。 第一部分 Section Cross Reference 文件引用关系分析 主要是各个源文件生成的模块之间相互引用的关系 如第一句: startup_stm
[单片机]
<font color='red'>STM32</font>经keil mdk编译产生的.map文件分析
STM32再学习——实时时钟(RTC)
RTC,是RealTimeCounter的缩写。在MCU中,其实是一组连续计数的计数器,各个产品及系列各不相同。因此,在相应的软件配置下,可提供时钟日历的功能。当然,修改这个计数器的值也就重新设置了系统当前的时间和日期。 本文引用地址:http://www.eepw.com.cn/article/182407.htm RTC只是一个时钟,但与RTC相连的有两个系统时钟,一个是APB1接口的PCLK1另一个是RTC时钟 。这样,RTC功能也就分为两个部分:第一部分,APB1接口部分,与APB1总线相连,MCU也就是通过这条总线对其进行读写操作。另一部,RTC核,由一系列可编程计数器组成,这部分又再细分为两个组件:预分频模块与
[单片机]
<font color='red'>STM32</font>再学习——实时时钟(RTC)
025_STM32之MDK5软件仿真之查看io口输出
(一)模拟仿真或者STLINK仿真那些就需要相应的配置对应的地方 DARMSTM.DLL-pSTM32F103ZE (二) 如果是模拟仿真的话注意要配置好时钟 (三) 配置监测的引脚 输入配置的引脚之后点击空白处就行了 PORTA.0 如果弹出unknown signal的话,看第一点是否配置好,或者所配置的型号是否支持仿真,如:F429是不支持的 (四) 点击右键,选择输出的类型,像我们PWM的话就选bit的类型的 (五)运行看运行状态,点击stop或者start
[单片机]
025_STM32之MDK5软件仿真之查看io口输出
STM32 ucos 下添加CAN模块
1.CAN IO RX TX 的设定和重映射 GPIO_PinRemapConfig(GPIO_Remap2_CAN, ENABLE); //端口重映射到PD0,PD1 2.clock CAN IO CLOCK 的开启 、*-------gpio for can------------*/ RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOD | RCC_APB2Periph_AFIO, ENABLE); CAN 自身CLOCK 的开启、 /* CAN Periph clock enable */ RCC_APB1PeriphClockCmd(RCC_APB1Periph_CAN
[单片机]
小广播
设计资源 培训 开发板 精华推荐

最新单片机文章
何立民专栏 单片机及嵌入式宝典

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

换一换 更多 相关热搜器件

 
EEWorld订阅号

 
EEWorld服务号

 
汽车开发圈

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