stm32基本定时器timer6的原理与使用

摘要:
基本计时器的核心是时基。还有通用计时器和高级计时器。Typedefstruct{uint16_tTIM_Pescaler;//预分频器uint16_tTIM_CounterMode;//计数模式uint32_tTIM_Period;//计时器周期uint16.tTIM_ClockDivision;//时钟分频器uit8_tTIM_RepetionCounter;//重复计数器}TIM_TimeBaseInitTTypeDef;1234567TIM_预分频器:设置定时器预分频器。时钟源是通过预分频器的定时器时钟。它设置TIMx_ PSC寄存器的值。因为基本计时器只能向上计数,并且没有配置计数模式的寄存器,所以默认值为向上。
/********************基本定时器 TIM 参数定义,只限 TIM6、7************/
/* 一、定时器分类 STM32F1 系列中,除了互联型的产品,共有 8 个定时器,分为基本定时器,通用定时器和高级定时器。基本定时器 TIM6 和 TIM7 是一个 16 位的只能向上计数的定时器,只能定时,没有外部 IO。通用定时器 TIM2/3/4/5 是一个 16 位的可以向上/下计数的定时器,可以定时,可以输出比较,可以输入捕捉,每个定时器有四个外部 IO。高级定时器 TIM1/8是一个 16 位的可以向上/下计数的定时器,可以定时,可以输出比较,可以输入捕捉,还可以有三相电机互补输出信号,每个定时器有 8 个外部 IO。 基本定时器的核心是时基,通用计时器和高级定时器也有。 1、时钟源 定时器时钟TIMxCLK,即内部时钟CK_INT,经APB1预分频器后分频提供,如果APB1 预分频系数等于 1,则频率不变,否则频率乘以 2,库函数中 APB1 预分频的系数是 2,即 PCLK1=36M,所以定时器时钟 TIMxCLK=36*2=72M 。 2、计数器时钟 定时器时钟经过 PSC 预分频器之后,即 CK_CNT,用来驱动计数器计数。PSC 是一个16 位的预分频器,可以对定时器时钟 TIMxCLK 进行 1~65536 之间的任何一个数进行分频。 具体计算方式为:CK_CNT=TIMxCLK/(PSC+1)。 3.计数器 计数器 CNT 是一个 16 位的计数器,只能往上计数,最大计数值为 65535。当计数达到自动重装载寄存器的时候产生更新事件,并清零从头开始计数。 4、自动重装载寄存器 自动重装载寄存器 ARR 是一个 16 位的寄存器,这里面装着计数器能计数的最大数值。当计数到这个值的时候,如果使能了中断的话,定时器就产生溢出中断。 5. 定时时间的计算 定时器的定时时间等于计数器的中断周期乘以中断的次数。计数器在 CK_CNT 的驱动下,计一个数的时间则是 CK_CLK 的倒数,等于:1/(TIMxCLK/(PSC+1)),产生一次中断的时间则等于:1/(CK_CLK * ARR)。如果在中断服务程序里面设置一个变量 time,用来记录中断的次数,那么就可以计算出我们需要的定时时间等于: 1/CK_CLK *(ARR+1)*time。 三、定时器初始化结构体详解 在标准库函数头文件stm32f10x_tim.h中对定时器外设建立了四个初始化结构体,基本定时器只用到其中一个即TIM_TimeBaseInitTypeDef,其他三个在高级定时器章节讲解。 typedef struct { uint16_t TIM_Prescaler; // 预分频器 uint16_t TIM_CounterMode; // 计数模式 uint32_t TIM_Period; // 定时器周期 uint16_t TIM_ClockDivision; // 时钟分频 uint8_t TIM_RepetitionCounter; // 重复计算器 } TIM_TimeBaseInitTypeDef;1234567 (1) TIM_Prescaler:定时器预分频器设置,时钟源经该预分频器才是定时器时钟,它设定TIMx_PSC 寄存器的值。可设置范围为 0 至 65535,实现 1至 65536 分频。 (2) TIM_CounterMode:定时器计数方式,可是在为向上计数、向下计数以及三种中心对齐模式。基本定时器只能是向上计数,即 TIMx_CNT只能从 0开始递增,并且无需初始化。 (3) TIM_Period:定时器周期,实际就是设定自动重载寄存器的值,在事件生成时更新到影子寄存器。可设置范围为 0至 65535。 (4) TIM_ClockDivision:时钟分频,设置定时器时钟 CK_INT 频率与数字滤波器采样时钟频率分频比,基本定时器没有此功能,不用设置。 (5) TIM_RepetitionCounter:重复计数器,属于高级控制寄存器专用寄存器位,利用它可以非常容易控制输出 PWM 的个数。这里不用设置。 虽然定时器基本初始化结构体有 5 个成员,但对于基本定时器只需设置其中两个就可以。 四、基本定时器实验 本实验利用基本定时器 TIM6/7 定时 1s,1s 时间到 LED 翻转一次。基本定时器是单片机内部的资源,没有外部 IO,不需要接外部电路,现只需要一个 LED 即可 。 软件设计 编写两个定时器驱动文件,bsp_TiMbase.h 和bsp_TiMbase.h,用来配置定时器中断优先级和和初始化定时器 。 1、 编程要点 (1) 开定时器时钟 TIMx_CLK, x[6,7] ; (2) 初始化时基初始化结构体 ; (3) 使能 TIMx, x[6,7] update 中断; (4) 打开定时器; (5) 编写中断服务程序 通用定时器和高级定时器的定时编程要点跟基本定时器差不多,只是还要再选择下计数器的计数模式,是向上还是向下。因为基本定时器只能向上计数,且没有配置计数模式的寄存器,默认是向上。 2.、软件分析 基本 定时器宏定义 */ #define BASIC_TIM6 // 如果使用 TIM7,注释掉这个宏即可 #ifdef BASIC_TIM6 // 使用基本定时器 TIM6 #define BASIC_TIM TIM6 #define BASIC_TIM_APBxClock_FUN RCC_APB1PeriphClockCmd #define BASIC_TIM_CLK RCC_APB1Periph_TIM6 #define BASIC_TIM_IRQ TIM6_IRQn #define BASIC_TIM_IRQHandler TIM6_IRQHandler #else // 使用基本定时器 TIM7 #define BASIC_TIM TIM7 #define BASIC_TIM_APBxClock_FUN RCC_APB1PeriphClockCmd #define BASIC_TIM_CLK RCC_APB1Periph_TIM7 #define BASIC_TIM_IRQ TIM7_IRQn #define BASIC_TIM_IRQHandler TIM7_IRQHandler #endif /* 基本定时器有 TIM6 和 TIM7,我们可以有选择的使用,为了提高代码的可移植性,我们把当需要修改定时器时需要修改的代码定义成宏,默认使用的是定时器 6,如果想修改成定时器 7,只需要把宏 BASIC_TIM6 注释掉即可。 基本定时器设定 */ void BASIC_TIM_Config(void) { TIM_TimeBaseInitTypeDef TIM_TimeBaseStructure; // 开启定时器时钟,即内部时钟 CK_INT=72M BASIC_TIM_APBxClock_FUN(BASIC_TIM_CLK, ENABLE); // 自动重装载寄存器周的值(计数值) TIM_TimeBaseStructure.TIM_Period=1000; // 累计 TIM_Period 个频率后产生一个更新或者中断 // 时钟预分频数为 71,则驱动计数器的时钟 CK_CNT = CK_INT / (71+1)=1M TIM_TimeBaseStructure.TIM_Prescaler= 71; // 时钟分频因子 ,基本定时器没有,不用管 //TIM_TimeBaseStructure.TIM_ClockDivision=TIM_CKD_DIV1; // 计数器计数模式,基本定时器只能向上计数,没有计数模式的设置 //TIM_TimeBaseStructure.TIM_CounterMode=TIM_CounterMode_Up; // 重复计数器的值,基本定时器没有,不用管 //TIM_TimeBaseStructure.TIM_RepetitionCounter=0; // 初始化定时器 TIM_TimeBaseInit(BASIC_TIM, &TIM_TimeBaseStructure); // 清除计数器中断标志位 TIM_ClearFlag(BASIC_TIM, TIM_FLAG_Update); // 开启计数器中断 TIM_ITConfig(BASIC_TIM,TIM_IT_Update,ENABLE); // 使能计数器 TIM_Cmd(BASIC_TIM, ENABLE); // 暂时关闭定时器的时钟,等待使用 BASIC_TIM_APBxClock_FUN(BASIC_TIM_CLK, DISABLE) } /* 我们把定时器设置自动重装载寄存器 ARR 的值为 1000,设置时钟预分频器为 71,则驱动计数器的时钟:CK_CNT = CK_INT / (71+1)=1M,则计数器计数一次的时间等于:1/CK_CNT=1us,当计数器计数到 ARR 的值 1000 时,产生一次中断,则中断一次的时间为:1/CK_CNT*ARR=1ms。 在初始化定时器的时候,我们定义了一个结构体:TIM_TimeBaseInitTypeDef,TIM_TimeBaseInitTypeDef 结构体里面有 5 个成员,TIM6 和 TIM7 的寄存器里面只有TIM_Prescaler 和 TIM_Period,另外三个成员基本定时器是没有的,所以使用 TIM6 和TIM7的时候只需初始化这两个成员即可, 另外三个成员是通用定时器和高级定时器才有,具体说明如下: --------------------- 本文来自 Yuk丶Han 的CSDN 博客 ,全文地址请点击:https://blog.csdn.net/zxh1592000/article/details/78770064?utm_source=copy */ typedef struct { TIM_Prescaler // 都有 TIM_CounterMode // TIMx,x[6,7]没有,其他都有 TIM_Period // 都有 TIM_ClockDivision // TIMx,x[6,7]没有,其他都有 TIM_RepetitionCounter // TIMx,x[1,8,15,16,17]才有 } TIM_TimeBaseInitTypeDef; // 中断优先级配置 void BASIC_TIM_NVIC_Config(void) { NVIC_InitTypeDef NVIC_InitStructure; // 设置中断组为 0 NVIC_PriorityGroupConfig(NVIC_PriorityGroup_0); // 设置中断来源 NVIC_InitStructure.NVIC_IRQChannel = BASIC_TIM_IRQ ; // 设置主优先级为 0 NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 0; // 设置抢占优先级为 3 NVIC_InitStructure.NVIC_IRQChannelSubPriority = 3; NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE; NVIC_Init(&NVIC_InitStructure); } //定时器中断服务程序 void BASIC_TIM_IRQHandler (void) { if ( TIM_GetITStatus( BASIC_TIM, TIM_IT_Update) != RESET ) { time++; TIM_ClearITPendingBit(BASIC_TIM, TIM_FLAG_Update); } } /* 定时器中断一次的时间是 1ms,我们定义一个全局变量 time,每当进一次中断的时候,让 time 来记录进入中断的次数。如果我们想实现一个 1s 的定时,我们只需要判断time 是否等于 1000 即可,1000 个 1ms 就是 1s。然后把 time 清 0,重新计数,以此循环往复。在中断服务程序的最后,要把相应的中断标志位清除掉,切记。 主函数 */ int main(void) { /* led 端口配置 */ LED_GPIO_Config(); /* 基本定时器 TIMx,x[6,7] 定时配置 */ BASIC_TIM_Config(); /* 配置基本定时器 TIMx,x[6,7]的中断优先级 */ BASIC_TIM_NVIC_Config(); /* 基本定时器 TIMx,x[6,7] 重新开时钟,开始计时 */ BASIC_TIM_APBxClock_FUN(BASIC_TIM_CLK, ENABLE); while (1) { if ( time == 1000 ) /* 1000 * 1 ms = 1s 时间到 */ { time = 0; /* LED1 取反 */ LED1_TOGGLE; } } } /* 函数做一些必须的初始化,然后在一个死循环中不断的判断 time 的值,time 的值在定时器中断改变,每加一次表示定时器过了 1ms,当 time 等于 1000 时,1s 时间到,LED1翻转一次,并把 time 清 0。 四、思考 1. 计算基本定时器一次最长定时时间,如果需要使用基本定时器产生 100s 周期事件有什么办法实现? 2. 修改实验程序,在保使其每 0.5s 翻转一次 LED1的同时在每 10s 翻转 LED2。 */

总结:

基本定时器timer6、timer7的设置过程

这两个定时器这在大容量F103中才有。

 

使用目的:使用TIM定时器让小灯每0.5秒翻转一次亮灭

编程过程:    

    1-配置时基初始化结构体

    2-开启定时器更新中断(即定时时间到了)

    3-配置中断优先级

    4-使能定时器

    5-编写中断服务函数

    6-编写main函数

免责声明:文章转载自《stm32基本定时器timer6的原理与使用》仅用于学习参考。如对内容有疑问,请及时联系本站处理。

上篇centos中网卡的配置Let’s Encrypt配置ssl证书自动更新下篇

宿迁高防,2C2G15M,22元/月;香港BGP,2C5G5M,25元/月 雨云优惠码:MjYwNzM=

相关文章

AVR单片机教程——走向高层

本文隶属于AVR单片机教程系列。   在系列教程的最后一篇中,我将向你推荐3个可以深造的方向:C++、事件驱动、RTOS。掌握这些技术可以帮助你更快、更好地开发更大的项目。 本文涉及到许多概念性的内容,如果你有不同意见,欢迎讨论。 关于高层 这一篇教程叫作“走向高层”。什么是高层? 我认为,如果寥寥几行代码就能实现一个复杂功能,或者一行代码可以对应到几百句...

转载:嵌入式C语言面试题(二)

BSS段 是“Block Started by Symbol”的缩写,意为“以符号开始的块”。   BSS是Unix链接器产生的未初始化数据段。其他的段分别是包含程序代码的“text”段和包含已初始化数据的“data”段。BSS段的变量只有名称和大小却没有值。此名后来被许多文件格式使用,包括PE。“以符号开始的块”指的是编译器处理未初始化数据的地方。BS...

android 定时器的使用

1、android中通常是使用AlarmManager来定时启动一个单次或重复多次操作的。具体的说就是我们通过AlarmManager设定一个时间和注册一个intent到系统中,然后在该时间到来时,系统为我们发送一个广播,即执行我们设定的Intent(要执行的操作),通常我们使用 PendingIntent来实现“要执行的操作”,PendingIntent...

stm32的双向io口

# stm32的io端口双向设置问题,须将io端口模式配置为 open-drain mode > 在io端口配置为输出模式时,输入通道上的施密特触发器一直是打开的,所以读取IDR是能检测到端口电平的 > 排除 push-pull mode 模式的依据是,推挽输出是强输出电流模式,在此模式下的输出通道上的推挽结构MOS管,属于强上拉和强下拉的,这...

IOS创建一个单件实例

Foundation和Application Kit框架中的一些类只允许创建单件对象,即这些类在当前进程中的唯一实例。举例来说,NSFileManager和NSWorkspace类在使用时都是基于进程进行单件对象的实例化。当您向这些类请求实例的时候,它们会向您传递单一实例的一个引用,如果该实例还不存在,则首先进行实例的分配和初始化。 单件对象充当控制中心的...

MIPI DSI 和 D-PHY 初始化序列

MIPI DSI 和 D-PHY 初始化序列 2015-12-29 深圳 南山平山村 曾剑锋 参考文档: i.MX 6Dual/6Quad Multimedia Appl...