STM32学习笔记(4):通用定时器基本定时功能

发布者:SerendipityGlow最新更新时间:2015-09-07 来源: eefocus关键字:STM32  学习笔记  通用定时器  定时功能 手机看文章 扫描二维码
随时随地手机看文章
1.     STM32的Timer简介

STM32中一共有11个定时器,其中2个高级控制定时器,4个普通定时器和2个基本定时器,以及2个看门狗定时器和1个系统嘀嗒定时器。其中系统嘀嗒定时器是前文中所描述的SysTick,看门狗定时器以后再详细研究。今天主要是研究剩下的8个定时器。

其中TIM1和TIM8是能够产生3对PWM互补输出的高级登时其,常用于三相电机的驱动,时钟由APB2的输出产生。TIM2-TIM5是普通定时器,TIM6和TIM7是基本定时器,其时钟由APB1输出产生。由于STM32的TIMER功能太复杂了,所以只能一点一点的学习。因此今天就从最简单的开始学习起,也就是TIM2-TIM5普通定时器的定时功能。

2.     普通定时器TIM2-TIM5

2.1    时钟来源

计数器时钟可以由下列时钟源提供:

·内部时钟(CK_INT)

·外部时钟模式1:外部输入脚(TIx)

·外部时钟模式2:外部触发输入(ETR)

       ·内部触发输入(ITRx):使用一个定时器作为另一个定时器的预分频器,如可以配置一个定时器Timer1而作为另一个定时器Timer2的预分频器。

    由于今天的学习是最基本的定时功能,所以采用内部时钟。TIM2-TIM5的时钟不是直接来自于APB1,而是来自于输入为APB1的一个倍频器。这个倍频器的作用是:当APB1的预分频系数为1时,这个倍频器不起作用,定时器的时钟频率等于APB1的频率;当APB1的预分频系数为其他数值时(即预分频系数为2、4、8或16),这个倍频器起作用,定时器的时钟频率等于APB1的频率的2倍。APB1的分频在STM32_SYSTICK的学习笔记中有详细描述。通过倍频器给定时器时钟的好处是:APB1不但要给TIM2-TIM5提供时钟,还要为其他的外设提供时钟;设置这个倍频器可以保证在其他外设使用较低时钟频率时,TIM2-TIM5仍然可以得到较高的时钟频率。

2.2    计数器模式

TIM2-TIM5可以由向上计数、向下计数、向上向下双向计数。向上计数模式中,计数器从0计数到自动加载值(TIMx_ARR计数器内容),然后重新从0开始计数并且产生一个计数器溢出事件。在向下模式中,计数器从自动装入的值(TIMx_ARR)开始向下计数到0,然后从自动装入的值重新开始,并产生一个计数器向下溢出事件。而中央对齐模式(向上/向下计数)是计数器从0开始计数到自动装入的值-1,产生一个计数器溢出事件,然后向下计数到1并且产生一个计数器溢出事件;然后再从0开始重新计数。

2.3    编程步骤

1.       配置系统时钟;

2.       配置NVIC;

3.       配置GPIO;

4.       配置TIMER;

其中,前3项在前面的笔记中已经给出,在此就不再赘述了。第4项配置TIMER有如下配置:

(1)       利用TIM_DeInit()函数将Timer设置为默认缺省值;

(2)       TIM_InternalClockConfig()选择TIMx来设置内部时钟源;

(3)       TIM_Perscaler来设置预分频系数;

(4)       TIM_ClockDivision来设置时钟分割;

(5)       TIM_CounterMode来设置计数器模式;

(6)       TIM_Period来设置自动装入的值

(7)       TIM_ARRPerloadConfig()来设置是否使用预装载缓冲器

(8)       TIM_ITConfig()来开启TIMx的中断

其中(3)-(6)步骤中的参数由TIM_TimerBaseInitTypeDef结构体给出。步骤(3)中的预分频系数用来确定TIMx所使用的时钟频率,具体计算方法为:CK_INT/(TIM_Perscaler+1)。CK_INT是内部时钟源的频率,是根据2.1中所描述的APB1的倍频器送出的时钟,TIM_Perscaler是用户设定的预分频系数,其值范围是从0 – 65535。

步骤(4)中的时钟分割定义的是在定时器时钟频率(CK_INT)与数字滤波器(ETR,TIx)使用的采样频率之间的分频比例。TIM_ClockDivision的参数如下表:​

数字滤波器(ETR,TIx)是为了将ETR进来的分频后的信号滤波,保证通过信号频率不超过某个限定。

步骤(7)中需要禁止使用预装载缓冲器。当预装载缓冲器被禁止时,写入自动装入的值(TIMx_ARR)的数值会直接传送到对应的影子寄存器;如果使能预加载寄存器,则写入ARR的数值会在更新事件时,才会从预加载寄存器传送到对应的影子寄存器。

ARM中,有的逻辑寄存器在物理上对应2个寄存器,一个是程序员可以写入或读出的寄存器,称为preload register(预装载寄存器),另一个是程序员看不见的、但在操作中真正起作用的寄存器,称为shadow register(影子寄存器);设计preload register和shadow register的好处是,所有真正需要起作用的寄存器(shadow register)可以在同一个时间(发生更新事件时)被更新为所对应的preload register的内容,这样可以保证多个通道的操作能够准确地同步。如果没有shadow register,或者preload register和shadow register是直通的,即软件更新preload register时,同时更新了shadow register,因为软件不可能在一个相同的时刻同时更新多个寄存器,结果造成多个通道的时序不能同步,如果再加上其它因素(例如中断),多个通道的时序关系有可能是不可预知的。

 

3.     程序源代码

本例实现的是通过TIM2的定时功能,使得LED灯按照1s的时间间隔来闪烁

 

#include "stm32f10x_lib.h"

 

void RCC_cfg();

void TIMER_cfg();

void NVIC_cfg();

void GPIO_cfg();

 

int main()

{

       RCC_cfg();

       NVIC_cfg();

       GPIO_cfg();

       TIMER_cfg();

 

       //开启定时器2

       TIM_Cmd(TIM2,ENABLE);

 

       while(1);

}

 

void RCC_cfg()

{

      

       //定义错误状态变量

       ErrorStatus HSEStartUpStatus;

      

       //将RCC寄存器重新设置为默认值

       RCC_DeInit();

 

       //打开外部高速时钟晶振

       RCC_HSEConfig(RCC_HSE_ON);

 

       //等待外部高速时钟晶振工作

       HSEStartUpStatus = RCC_WaitForHSEStartUp();

       if(HSEStartUpStatus == SUCCESS)

       {

              //设置AHB时钟(HCLK)为系统时钟

              RCC_HCLKConfig(RCC_SYSCLK_Div1);

 

              //设置高速AHB时钟(APB2)为HCLK时钟

              RCC_PCLK2Config(RCC_HCLK_Div1);

 

              //设置低速AHB时钟(APB1)为HCLK的2分频

              RCC_PCLK1Config(RCC_HCLK_Div2);

             

              //设置FLASH代码延时

              FLASH_SetLatency(FLASH_Latency_2);

[page]

              //使能预取指缓存

              FLASH_PrefetchBufferCmd(FLASH_PrefetchBuffer_Enable);

 

              //设置PLL时钟,为HSE的9倍频 8MHz * 9 = 72MHz

              RCC_PLLConfig(RCC_PLLSource_HSE_Div1, RCC_PLLMul_9);

 

              //使能PLL

              RCC_PLLCmd(ENABLE);

 

              //等待PLL准备就绪

              while(RCC_GetFlagStatus(RCC_FLAG_PLLRDY) == RESET);

 

              //设置PLL为系统时钟源

              RCC_SYSCLKConfig(RCC_SYSCLKSource_PLLCLK);

 

              //判断PLL是否是系统时钟

              while(RCC_GetSYSCLKSource() != 0x08);

       }

 

       //允许TIM2的时钟

       RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM2,ENABLE);

       //允许GPIO的时钟

       RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB,ENABLE);

 

}

 

void TIMER_cfg()

{

       TIM_TimeBaseInitTypeDef TIM_TimeBaseStructure;

 

       //重新将Timer设置为缺省值

       TIM_DeInit(TIM2);

       //采用内部时钟给TIM2提供时钟源

       TIM_InternalClockConfig(TIM2);

       //预分频系数为36000-1,这样计数器时钟为72MHz/36000 = 2kHz

       TIM_TimeBaseStructure.TIM_Prescaler = 36000 - 1;

       //设置时钟分割

       TIM_TimeBaseStructure.TIM_ClockDivision = TIM_CKD_DIV1;

       //设置计数器模式为向上计数模式

       TIM_TimeBaseStructure.TIM_CounterMode = TIM_CounterMode_Up;

       //设置计数溢出大小,每计2000个数就产生一个更新事件

       TIM_TimeBaseStructure.TIM_Period = 2000 - 1;

       //将配置应用到TIM2中

       TIM_TimeBaseInit(TIM2,&TIM_TimeBaseStructure);

 

       //清除溢出中断标志

       TIM_ClearFlag(TIM2, TIM_FLAG_Update);

       //禁止ARR预装载缓冲器

       TIM_ARRPreloadConfig(TIM2, DISABLE);

       //开启TIM2的中断

       TIM_ITConfig(TIM2,TIM_IT_Update,ENABLE);

}

 

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);

}

 

void GPIO_cfg()

{

       GPIO_InitTypeDef GPIO_InitStructure;

 

      

       GPIO_InitStructure.GPIO_Pin = GPIO_Pin_5;                 //选择引脚5

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

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

       GPIO_Init(GPIOB,&GPIO_InitStructure);

}

在stm32f10x_it.c中,我们找到函数TIM2_IRQHandler(),并向其中添加代码

void TIM2_IRQHandler(void)

{

       u8 ReadValue;

       //检测是否发生溢出更新事件

       if(TIM_GetITStatus(TIM2, TIM_IT_Update) != RESET)

       {

              //清除TIM2的中断待处理位

              TIM_ClearITPendingBit(TIM2 , TIM_FLAG_Update);

              //将PB.5管脚输出数值写入ReadValue

              ReadValue = GPIO_ReadOutputDataBit(GPIOB,GPIO_Pin_5);

             

              if(ReadValue == 0)

              {

                     GPIO_SetBits(GPIOB,GPIO_Pin_5);

              }    

              else

              {

                     GPIO_ResetBits(GPIOB,GPIO_Pin_5);      

              }

       }

 

}

关键字:STM32  学习笔记  通用定时器  定时功能 引用地址:STM32学习笔记(4):通用定时器基本定时功能

上一篇:STM32学习笔记(5):通用定时器PWM输出
下一篇:STM32学习笔记(3):系统时钟和SysTick定时器

推荐阅读最新更新时间:2024-03-16 14:30

stm32程序下载方式总结
程序下载是一切的基础,对于搞单片机的人最开始学的就是程序下载,本人在刚开始学习51单片机的时候就只是知道串口下载,以至于什么usb-转串口的芯片的作用也没有去了解,直到现在学习使用了stm32都快两年了,才第一次深入的考虑这些问题,想总结一下,当然这个都是参考网络上的一些牛人,借鉴一下别人的,然后自己整理了一下,为刚入门单片机的新手们简单解释一下,这些完全是本人的理解,不对之处,还请指正。 先总体说一下,stm32的程序下载方式(这里我主要讲解采用jlink,isp(串口在线编程),而用u-link,st-link应该也是可以的,不过本人没有真实的接触过), 大概有一下三种方式: (1)j-flash下载(需要用到j-link),
[单片机]
ST推出新款的图形处理性能强大的STM32微控制器
2016年2月2日 横跨多重电子应用领域、全球领先的半导体供应商意法半导体(ST)推出新款的STM32F767/769微控制器(MCU),内置功能丰富的存储器、图形处理器和通信外设,让ARM Cortex -M7处理器的强大性能和高能效拓展至更广泛的应用产品,例如便携或穿戴式消费电子产品、智能楼宇和工业控制器、智能家电、个人医疗设备、保健点医疗设备。 阵容强大的STM32系列微控制器的最新产品基于内置双精度浮点单元且支持DSP指令的216MHz/462DMIPS/1082 EEMBC CoreMark Cortex-M7内核,集成容量最高2MB的双区闪存、图形处理性能强大的Chrom-ART加速器、硬件JPEG加速器、TFT-L
[单片机]
ST推出新款的图形处理性能强大的<font color='red'>STM32</font>微控制器
STM32 flash 读写操作
一、Flash简介   通过对stm32内部的flash的读写可以实现对stm32的编程操作。   stm32的内置可编程Flash在许多场合具有十分重要的意义。如其支持ICP(In Circuit Programming,在电路编程;在线编程)特性使得开发人员对stm32可以警醒调试开发,可以通过JTAG和SWD接口对stm32进行程序烧写;支持IAP(In Application Programming,在应用中编程)使得开发人员可以在stm32运行程序的时候对其内部程序进行更新操作。对一些对数据安全有要求的场合,可编程FLASH可以结合stm32内部唯一的身份标识实现各种各样的防破解方案。并且stm32的FLASH在一
[单片机]
STM32中的GPIO结构
1、GPIO内部结构 2、GPIO输入模式 (1)浮空输入模式 (2)输入上拉模式 (3)输入下拉模式 (4)模拟输入模式 3、GPIO输出模式 (1)开漏输出模式 (2)开漏复用输出模式 (3)推挽输出模式 (4)推挽复用输出模式 4、3.GPIO初始化步骤 具体寄存器需要参考STM32F1x手册 (1)确定GPIO的工作模式和工作速度 通过配置寄存器GPIOx_CRL和GPIOx_CRH (2)具体要输出的内容 置位复位寄存器:BSRR和数据输出寄存器:ODR 5、总结 (1)推挽输出 a、可以输出高低电平,用于连接数字器件,高电平由VDD决定,低电平由V
[单片机]
<font color='red'>STM32</font>中的GPIO结构
STM32 407 iar fpu 加入浮点
之前想玩一下FPU 可惜那个DSP库看不懂 也找不到教程 我研究了2个晚上总算会用那个FFT了.... 我的环境是 IAR 6.2 板子是STM32F4DISCOVERY 第一 首先下载官方的 STM32F4xx_DSP_StdPeriph_Lib_V1.0.0 没有的同学请自己去官网下一个 DSP库在 STM32F4xx_DSP_StdPeriph_Lib_V1.0.0LibrariesCMSIS 然后我们使用IAR建立工作空间 这个教程到处都有 我就不说了 在编译器里面加入 $PROJ_DIR$......LibrariesCMSISDSP_LibSource ARM_MATH_CM4 ARM_MATH_MATRIX_CH
[单片机]
STM32之ADC多通道连续例程
#include stm32f10x.h /* RCC时钟配置 */ void RCC_config(void) { ErrorStatus HSEStartUpStatus; /* RCC寄存器设置为默认配置 */ RCC_DeInit(); /* 打开外部高速时钟 */ RCC_HSEConfig(RCC_HSE_ON); /* 等待外部高速时钟稳定 */ HSEStartUpStatus = RCC_WaitForHSEStartUp(); if(HSEStartUpStatus == SUCCESS) { /* 设置HCLK = SYSCLK */ RCC_HCLKConfig(RCC_SYSCLK_D
[单片机]
STM32外设有哪些?
STM32系列是ST公司基于专为要求高性能、低成本、低功耗的嵌入式应用专门设计的ARM Cortex-M3内核的32位单片机。按内核架构分为不同产品: 其中STM32F系列有:STM32F103“增强型”系列、STM32F101“基本型”系列、STM32F105、STM32F107“互联型”系列。 增强型系列时钟频率达到72MHz,是同类产品中性能最高的产品;基本型时钟频率为36MHz,以16位产品的价格得到比16位产品大幅提升的性能,是32位产品用户的最佳选择。两个系列都内置32K到128K的闪存,不同的是SRAM的最大容量和外设接口的组合。时钟频率72MHz时,从闪存执行代码,STM32功耗36mA,相当于0.5mA/
[嵌入式]
STM32中断和事件的对比差异分析
事件是中断的触发源,开放了对应的中断屏蔽位,则事件可以触发相应的中断。 事件还是其它一些操作的触发源,比如DMA,还有TIM中影子寄存器的传递与更新;而中断是不能触发这些操作的,所以要把事件与中断区分开。当你只要产生中断而不想触发其它操作时,就可以用事件屏蔽寄存器实现。 在STM32中,中断与事件不是等价的,一个中断肯定对应一个事件,但一个事件不一定对应一个中断。 这张图是一条外部中断线或外部事件线的示意图,图中信号线上划有一条斜线,旁边标志19字样的注释,表示这样的线路共有19套.图中的蓝色虚线箭头,标出了外部中断信号的传输路径,首先外部信号从编号1的芯片管脚进入,经过编号2的边沿检测电路,通过编号3的或门进入中断挂起请
[单片机]
小广播
添点儿料...
无论热点新闻、行业分析、技术干货……
设计资源 培训 开发板 精华推荐

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

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

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