基于STM32定时器实现毫秒延时函数

2020-05-23来源: eefocus关键字:STM32  定时器  毫秒延时函数

STM32定时器包含基本定时器、通用定时器和高级定时器,其中TIM6和TIM7是STM32当中的基本定时器,作为初学者,先从最基本的学起最容易,下面我们用这个定时器实现毫秒延时函数来入门STM32定时器的应用。


学习单片机,就是学习使用它的寄存器。即便你用库函数,寄存器也是必须要学习的。


TIM6 TIM7的寄存器如下所示:

基于STM32定时器实现毫秒延时函数
 

先略览一下寄存器,CR1和CR1是控制寄存器,SR是状态寄存器,ARR就是溢出值寄存器,CNT就是计数器的当前值,PSC是预分频寄存器。预分频寄存器?听的傻眼了吧,前面几个个寄存器听的还能理解,一听到预分频寄存器,好像不知道是干嘛用的。瑞生来给你解释一下吧,你可以给预分频寄存器里面写一个从0~65535的值,这个值+1,就是定时器运行的时钟。举个例子,比如单片机工作在主频72MHz,预分频寄存器写0,预分频系数就是0+1=1,定时器的时钟就是72MHz/1=72MHz;再举个例子,比如单片机还是工作在主频72MHz,预分频寄存器写71,预分频系数就是71+1=72,定时器的时钟就是72MHz/72=1MHz。知道定时器的时钟有什么用?相信很多初学者不清楚,定时器的时钟关乎定时器计数器CNT递增的时间间隔,根据频率和周期的公式f=1/T,定时器计数器递增的时间间隔就是1/定时器的时钟,例如当定时器时钟为1MHz时,定时器计数器递增的时间间隔就是1/1MHz=1微秒,这时,如果你把溢出值设置为1000,就是1000*1us=1ms溢出。


1.直接操作寄存器

下面,我们先用直接操作寄存器的方式,写一个毫秒延时函数:

voiddelay_ms(uint16_tms){TIM6->PSC=35999;TIM6->ARR=ms*2;TIM6->CR1|=(1<<3);TIM6->CR1|=0x1;while((TIM6->SR&0X1)==0);TIM6->SR=0;}

第一条语句,设置预分频系数为35999+1=36000,所以定时器的时钟为72000000/36000=2000Hz,那么定时时间间隔就是1/2000=0.0005秒,即0.5毫秒。

第二条语句,设置溢出值为ms乘以2,假如要延时1秒,函数的参数ms就是1000,溢出值就是1000*2=2000,2000*0.5毫秒=1000毫秒,即1秒。这时候,有人会说,为什么不干脆把预分频值PSC设置为71999,即预分频系数为72000,定时器的时钟就是72000000/72000=1000Hz,定时时间就是1毫秒,那么直接把函数的参数ms给了溢出值寄存器ARR就可以了,就不必乘以2了。想法是可以,但是你得知道,定时器都是16位的,所以PSC的值最大到65535,到不了71999。这下你明白了吧?


第三条语句,CR1寄存器bit3写1,由寄存器定义得知,这是把定时器设置为一旦发生溢出,就停止定时器,因为我们做的是延时函数,延时到了以后,就没有必要让定时器再不断递增了,所以要这样设置。

基于STM32定时器实现毫秒延时函数

第四条语句,CR1寄存器bit0写1,打开定时器,定时器计数器开始从0递增。

基于STM32定时器实现毫秒延时函数

第五条语句,检测状态寄存器SR中的bit0UIF是否置1,置1的时候,定时值就达到溢出值了,说明定时时间到了。


第六条语句,清除状态寄存器SR中刚才溢出造成的UIF位。


2.使用库函数

下面,我们看看怎么使用库函数实现毫秒延时函数:

voidTIM6_Delay_ms(uint16_tms){/*定义一个定时器基本定时初始化结构体变量*/TIM_TimeBaseInitTypeDefTIM_TimeBaseInitStruct;/*时钟预分频数为36000,在主频72M时,计数器每500us加1*/TIM_TimeBaseInitStruct.TIM_Prescaler=35999;/*自动重装载寄存器值*/TIM_TimeBaseInitStruct.TIM_Period=ms*2;/*把上面的值配置到寄存器*/TIM_TimeBaseInit(TIM6,&TIM_TimeBaseInitStruct);/*设置定时时间到了以后停止定时器计数*/TIM_SelectOnePulseMode(TIM6,TIM_OPMode_Single);/*清除SR中的UIF标志*/TIM_ClearFlag(TIM6,TIM_IT_Update);/*打开定时器6*/TIM_Cmd(TIM6,ENABLE);/*检测定时时间是否到来*/while(TIM_GetFlagStatus(TIM6,TIM_IT_Update)==RESET);/*软件清除更新标志*/TIM_ClearFlag(TIM6,TIM_IT_Update);}

你可以细细观察一下上面的库函数,实际上,和直接操作寄存器是一样的。比如说,我们看打开定时器的库函数TIM_Cmd(TIM6,ENABLE),我们打开这个函数,如下所示:

voidTIM_Cmd(TIM_TypeDef*TIMx,FunctionalStateNewState){/*Checktheparameters*/assert_param(IS_TIM_ALL_PERIPH(TIMx));assert_param(IS_FUNCTIONAL_STATE(NewState));if(NewState!=DISABLE){/*EnabletheTIMCounter*/TIMx->CR1|=TIM_CR1_CEN;}else{/*DisabletheTIMCounter*/TIMx->CR1&=(uint16_t)(~((uint16_t)TIM_CR1_CEN));}}

把参数TIM6和ENABLE带进去,你会发现,实际上,就是我们直接操作寄存器的TIM6->CR1|=0X1。


其它的库函数,你可以自己打开研究一下。


另外,库函数和我们直接操作寄存器有个不同的地方,就是TIM_TimeBaseInit()这个函数中的最后一条语句,它给TIMx->EGR寄存器的bit0写了1,使得SR中的bit0UIF置1,所以在执行完TIM_TimeBaseInit()这个函数之后,我们后面用TIM_ClearFlag(TIM6, TIM_IT_Update);函数把UIF清0,要不然的话,你在后面会直接检测到UIF置1,但是,这个置1不是溢出产生的,而是刚才给EGR写1产生的,就达不到延时的效果了。


总结,在使用以上两个延时函数的时候,记得先允许TIM6外设,即给APB1ENR寄存器的bit4写1。或者直接用库函数RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM6 , ENABLE)。从上面库函数和直接操作寄存器的延时函数,你可以看出,使用库函数,即使不用注释,也可以看出来你是在干嘛,而直接操作寄存器,就看不出来了。在实际使用中,没必要纠结库函数和直接操作寄存器的选择,我就喜欢一会用库函数,一会儿直接操作寄存器,你想咋地?不过,公司的老板肯定是希望使用库函数,因为如果你辞职了,接替你的人看到一堆操作寄存器的代码,让他情何以堪?

关键字:STM32  定时器  毫秒延时函数 编辑:什么鱼 引用地址:http://news.eeworld.com.cn/mcu/ic498010.html 本网站转载的所有的文章、图片、音频视频文件等资料的版权归版权所有人所有,本站采用的非本站原创文章及图片等内容无法一一联系确认版权者。如果本网所选内容的文章作者及编辑认为其作品不宜公开自由传播,或不应无偿使用,请及时通过电子邮件或电话通知我们,以迅速采取适当措施,避免给双方造成不必要的经济损失。

上一篇:基于STM32单片机低功耗模式机制详细解析
下一篇:最后一页

关注eeworld公众号 快捷获取更多信息
关注eeworld公众号
快捷获取更多信息
关注eeworld服务号 享受更多官方福利
关注eeworld服务号
享受更多官方福利

推荐阅读

STM32MxCube配置串口
基于上一次将第一章:STM32MxCube 基本使用方法,本章直接讲叙述STM32配置串口2的。查看STM32F407电路图:可得USART2接在PA2、PA3。下面新建STM32MxCube工程,开始配置。这里我们配置IO(定义PF9、PF10引脚为LED)、RCC(使用HSE)、USART(串口)。配置USART(串口)USART的配置有好几种,也是最常见的一种”UART异步串行通信”配置来讲述。选择USART2模式为“asynchronous”(异步通信)对应的引脚自动转换为USART。对于系统时钟应该都知道它的作用,就是驱动整个芯片工作的心脏,如果没有了它,就等于人没有了心跳。时钟的快慢也就决定了程序的快慢。STM32
发表于 2020-05-04
STM32MxCube配置串口
STM32CubeMX学习教程之一:GPIO输出之跑马灯
完整源码下载:https://github.com/simonliu009/STM32CubeMX-GPIO-Control软件版本:STM32CubeMX V4.25.0 System Workbench V2.4固件库版本:STM32Cube FW_F1 V1.6.1硬件:OneNet 麒麟座V2.3在STM32CubeMX中新建项目,选择正确的MCU型号首先设置RCC和SYS,如下图然后根据板子实际情况设置时钟(麒麟座外部晶振是12M,STM32F103x的最高主频是72M),如下图GPIO设置 PC7, PC8, PA12和 PC10为GPIO_OUTPUT, (这是麒麟座V2.3的四个LED管脚),如下图
发表于 2020-05-04
STM32CubeMX学习教程之一:GPIO输出之跑马灯
STM32CubeMX学习--(3)串口通信
Cube配置USART配置在Connectivity中选中USART1MODE = AsynchronousHardware Flow Control = DisableParameter Settings中配置Baud Rate = 115200Word Length = 8bitParity = NoneStop Bits =1Data Direction = Receive and TransmitOver Sampling = 16 SamplesNVIC Setting勾选Enable,Preemption Priority =2生成代码修改代码生成代码后,即可使用HAL_UART_Transmit_IT(&
发表于 2020-04-29
STM32CubeMX系列教程 5.0版本环境开发——2.Uart串行通信功能
;                         PS:在明白原理的情况下建议自己手写一次模拟UART。明白了之后,代码重复的事就由STM32芯片的硬件功能来实现就可以了,你只需要会看逻辑分析仪或者示波器分析数据便可。 所谓硬件功能 就是你把数据填充到寄存器,然后配置好相关参数,他会自动帮你发送出去。通俗点说,自己手写整个协议代码实现实现就像你自己把一个快件送到到别人那里再回来,而硬件功能则是你把快件给快递员,快递员帮你送过去。  明显 后者会帮你节省大量时间和精力。它是一个硬件组成,并且你也为此付费
发表于 2020-04-29
STM32CubeMX系列教程 5.0版本环境开发——2.Uart串行通信功能
串口通信(用CubeMX学习STM32)
前言: STM32串口介绍串行通信是单片机与外部设备或其他计算机交换信息的一个方式, 数据一位一位的按顺序传送, 其优点是只需要一条传输线, 协议简单, 但是缺点就是传送速度较慢。串口是单片机上非常便捷的一个工具, 当写程序需要调试的时候, 它可以很方便的提供调试方法, 只要在一些关键代码执行的地方, 通过串口给串口调试助手发送相关信息, 就可以使我们很方便的查看代码在这个位置的执行情况。下面看一下我所使用的单片机上串口的原理图接线外部的发送端TXD就是单片机串口的接收端USART_RX,   外部接收端RXD就是单片机串口的发送端USART_TXTXD : Transmit(TX) Data(D)  Receive(RX
发表于 2020-04-29
【STM32CubeMX】 串口通信(USART) Printf重定向
STM32CubeMX: Version 4.26.1MDK-ARM: Version 5.24.2开发板: 中移onenet 麒麟座MINI板芯片: STM32F103CBT61.STM32CubeMX设置设置外部时钟源设置Debug设置串口 使用USART1 PA9,PA10设置时钟频率设置USART1详细参数生成MDK-ARM工程2.重定向Printf 及 Scanf主要用到两个函数HAL_StatusTypeDef HAL_UART_Transmit(UART_HandleTypeDef *huart, uint8_t *pData, uint16_t Size, uint32_t Timeout
发表于 2020-04-29
【STM32CubeMX】 串口通信(USART) Printf重定向
何立民专栏 单片机及嵌入式宝典

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

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