草庐IT

STM32F103C8T6的学习(5)——定时器

小顺要加油 2023-05-11 原文

:  本教程基于up主江科大自化协——“STM32入门教程”记录的个人学习笔记

跳转链接:STM32入门教程-2022持续更新中_哔哩哔哩_bilibili

1.简介

TIMTimer)定时器

定时器可以对输入的时钟进行计数,并在计数值达到设定值时触发中断

16位计数器、预分频器、自动重装寄存器的时基单元,在72MHz计数时钟下可以实现最大59.65s(1/72/65536/65536)的定时

不仅具备基本的定时中断功能,而且还包含内外时钟源选择、输入捕获、输出比较、编码器接口、主从触发模式等多种功能

根据复杂度和应用场景分为了高级定时器、通用定时器、基本定时器三种类型

 (注:高级定时器中,重复计数器,死区生成,互补输出,刹车输入等是针对于三相无刷电机的驱动设计的)

STM32F103C8T6定时器资源:TIM1,TIM2,TIM3,TIM4

2.定时中断基本结构

 

配置流程(内部时钟)

  1. RCC开启时钟                 (此时基准时钟和整个外设的工作时钟都被打开)
  2. 选择时基单元的时钟源   (内部时钟模式)
  3. 配置时基单元                 (自动重装器,预分频器,计数器通过结构体配置即可)
  4. 配置输出中断控制,允许更新中断输出到NVIC
  5. 配置NVIC                      (打开中断通道,并分配优先级)

TIM库函数

TIM_DeInit(TIM_TypeDef* TIMx);

//TIM初始配置

TIM_TimeBaseInit(TIM_TypeDef* TIMx, TIM_TimeBaseInitTypeDef* TIM_TimeBaseInitStruct);

//配置时基单元,TIMx选择某个定时器,TIM_TimeBaseInitStruct结构体

关于时基单元结构体的补充

TIM_TimeBaseInitStruct.TIM_ClockDivision =;
TIM_TimeBaseInitStruct.TIM_CounterMode =;
TIM_TimeBaseInitStruct.TIM_Period =;
TIM_TimeBaseInitStruct.TIM_Prescaler =;
TIM_TimeBaseInitStruct.TIM_RepetitionCounter =;

TIM_ClockDivision:时钟分割,其参数为 TIM_Clock_Division_CKD

TIM_CKD_DIV1:不分割,

TIM_CKD_DIV2:2分频

TIM_CKD_DIV4:4分频

TIM_CounterMode:指定计数器模式

TIM_CounterMode_Up                                        向上计数

TIM_CounterMode_Down                                   向下计数
TIM_CounterMode_CenterAligned1                   中央对齐1
TIM_CounterMode_CenterAligned2                   中央对齐2
TIM_CounterMode_CenterAligned3                   中央对齐3

TIM_Period:ARR自动重装值(总数)

TIM_Prescaler:PSC预分频器的值(一周期计多少次)

上述两个函数决定定时时间

        定时频率=72M/(PSC+1)/(ARR+1)

        例如:定时1秒,表示定时频率为1Hz,则PSC为7200,ARR为10000,其参数再均减1

        因为预分频器与计数器都有1个数的偏差

如果PSC的值比较小,ARR的值比较大,就是表示是一个比较高的频率计比较多的数

注意:PSC,ARR的取值需要在0~65535之间

计数器计数频率:CK_CNT = CK_PSC / (PSC + 1)

•计数器溢出频率:CK_CNT_OV = CK_CNT / (ARR + 1)

                                                     = CK_PSC / (PSC + 1) / (ARR + 1)                            

TIM_RepetitionCounter:重复计数器的值(高级TIM才有,低级默认为0)

注:没有CNT计数器的值,后续通过SetCount和GetCount两个函数来操作

续接TIM库函数

TIM_TimeBaseStructInit(TIM_TimeBaseInitTypeDef* TIM_TimeBaseInitStruct);

//把结构体变量赋一个默认值

TIM_Cmd(TIM_TypeDef* TIMx, FunctionalState NewState);

//使能计数器 (运行控制

TIM_ITConfig(TIM_TypeDef* TIMx, uint16_t TIM_IT, FunctionalState NewState);

//使能中断输出信号(中断输出控制)(TIMx选择定时器,TIM_IT选择要配置那个中断输出)

 

NVIC用NVIC_Init函数

TIM_InternalClockConfig(TIM_TypeDef* TIMx);

//选择内部时钟-内部时钟模式-时基单元
TIM_ITRxExternalClockConfig(TIM_TypeDef* TIMx, uint16_t TIM_InputTriggerSource);

//ITRx-外部时钟模式-时基单元
TIM_TIxExternalClockConfig(TIM_TypeDef* TIMx, uint16_t TIM_TIxExternalCLKSource,
                                uint16_t TIM_ICPolarity, uint16_t ICFilter);

//TIx-外部时钟模式-时基单元  TIM_TIxExternalCLKSource 选择TIx具体的某个引脚   TIM_ICPolarity输入的极性  ICFilter滤波器
TIM_ETRClockMode1Config(TIM_TypeDef* TIMx, uint16_t TIM_ExtTRGPrescaler, uint16_t TIM_ExtTRGPolarity,
                             uint16_t ExtTRGFilter);

//选择ETR-外部时钟模式1-时基单元   TIM_ExtTRGPrescaler外部触发预分频器
TIM_ETRClockMode2Config(TIM_TypeDef* TIMx, uint16_t TIM_ExtTRGPrescaler, 
                             uint16_t TIM_ExtTRGPolarity, uint16_t ExtTRGFilter);

//选择ETR-外部时钟模式2-时基单元
TIM_ETRConfig(TIM_TypeDef* TIMx, uint16_t TIM_ExtTRGPrescaler, uint16_t TIM_ExtTRGPolarity,
                   uint16_t ExtTRGFilter);

//单独用来配置ETR引脚的预分频器、极性、滤波器这些参数

TIM_PrescalerConfig(TIM_TypeDef* TIMx, uint16_t Prescaler, uint16_t

TIM_PSCReloadMode);

//单独写预分频值的函数
TIM_CounterModeConfig(TIM_TypeDef* TIMx, uint16_t TIM_CounterMode);

//改变计数器的计数模式(TIM_CounterMode 选择新的计数器模式)
TIM_ARRPreloadConfig(TIM_TypeDef* TIMx, FunctionalState NewState);

//自动重装器预装功能配置

TIM_SetCounter(TIM_TypeDef* TIMx, uint16_t Counter);

//给计数器手动写入一个值

TIM_SetAutoreload(TIM_TypeDef* TIMx, uint16_t Autoreload);

//给自动重装器手动写入一个值

TIM_GetCounter(TIM_TypeDef* TIMx);

//获取当前计数器的值
TIM_GetPrescaler(TIM_TypeDef* TIMx);

//获取当前的预分频器的值

FlagStatus TIM_GetFlagStatus(TIM_TypeDef* TIMx, uint16_t TIM_FLAG);
void TIM_ClearFlag(TIM_TypeDef* TIMx, uint16_t TIM_FLAG);
ITStatus TIM_GetITStatus(TIM_TypeDef* TIMx, uint16_t TIM_IT);
void TIM_ClearITPendingBit(TIM_TypeDef* TIMx, uint16_t TIM_IT);

//获取标志位,清除标志位

3.相关配置函数

配置GPIO

    RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM2,ENABLE);

选择定时器

	TIM_InternalClockConfig(TIM2);

配置时基单元

    TIM_TimeBaseInitTypeDef TIM_TimeBaseInitStruct;
	TIM_TimeBaseInitStruct.TIM_ClockDivision =TIM_CKD_DIV1;
	TIM_TimeBaseInitStruct.TIM_CounterMode =TIM_CounterMode_Up;
	TIM_TimeBaseInitStruct.TIM_Period =10000-1;
	TIM_TimeBaseInitStruct.TIM_Prescaler =7200-1;
	TIM_TimeBaseInitStruct.TIM_RepetitionCounter =0;
	TIM_TimeBaseInit(TIM2,&TIM_TimeBaseInitStruct);

为了避免刚一上电就立刻进入中断,需要在TIM_TimeBaseInit的后面,进入中断的前面,手动调用TIM_ClearFlag来解决这个问题

手动把“更新中断标志位”清除就ok了

    TIM_ClearFlag(TIM2,TIM_FLAG_Update);

配置中断输出控制

    TIM_ITConfig(TIM2,TIM_IT_Update,ENABLE);

配置NVIC

    NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2);
	
	NVIC_InitTypeDef NVIC_InitStruct;
	NVIC_InitStruct.NVIC_IRQChannel=TIM2_IRQn;
	NVIC_InitStruct.NVIC_IRQChannelCmd=ENABLE;
	NVIC_InitStruct.NVIC_IRQChannelPreemptionPriority=2;
	NVIC_InitStruct.NVIC_IRQChannelSubPriority=1;
	NVIC_Init(&NVIC_InitStruct);

配置运行控制

    TIM_Cmd(TIM2,ENABLE);

4.中断函数

在start--startup_stm32f10x_md.s文件中寻找TIM2_IRQHandler,TIM2中断函数

  1. 判断中断标志位
  2. 书写用户代码
  3. 清除标志位
void TIM2_IRQHandler(void)  //当定时器产生更新中断时,此函数自动执行
{
	
}

5.书写主函数(main函数)

头文件添加(#include "Timer.h")

主函数调用XXX—Init,初始化定时器(Timer_Init();)

//配置流程(外部时钟ETRC,需要调用GPIO)

 配置GPIO

    RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM2,ENABLE);

    RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA,ENABLE);
	
	GPIO_InitTypeDef GPIO_InitStruct;
	GPIO_InitStruct.GPIO_Mode=GPIO_Mode_IPU;
	GPIO_InitStruct.GPIO_Pin=GPIO_Pin_0;
	GPIO_InitStruct.GPIO_Speed=GPIO_Speed_50MHz;
	GPIO_Init(GPIOA,&GPIO_InitStruct);

选择定时器

    TIM_ETRClockMode2Config(TIM2, TIM_ExtTRGPSC_OFF, 
                             TIM_ExtTRGPolarity_NonInverted, 0x00);

配置时基单元

    TIM_TimeBaseInitTypeDef TIM_TimeBaseInitStruct;
	TIM_TimeBaseInitStruct.TIM_ClockDivision =TIM_CKD_DIV1;
	TIM_TimeBaseInitStruct.TIM_CounterMode =TIM_CounterMode_Up;
	TIM_TimeBaseInitStruct.TIM_Period =10000-1;
	TIM_TimeBaseInitStruct.TIM_Prescaler =7200-1;
	TIM_TimeBaseInitStruct.TIM_RepetitionCounter =0;
	TIM_TimeBaseInit(TIM2,&TIM_TimeBaseInitStruct);

为了避免刚一上电就立刻进入中断,需要在TIM_TimeBaseInit的后面,进入中断的前面,手动调用TIM_ClearFlag来解决这个问题

手动把“更新中断标志位”清除就ok了

    TIM_ClearFlag(TIM2,TIM_FLAG_Update);

配置中断输出控制

    TIM_ITConfig(TIM2,TIM_IT_Update,ENABLE);

配置NVIC

    NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2);
	
	NVIC_InitTypeDef NVIC_InitStruct;
	NVIC_InitStruct.NVIC_IRQChannel=TIM2_IRQn;
	NVIC_InitStruct.NVIC_IRQChannelCmd=ENABLE;
	NVIC_InitStruct.NVIC_IRQChannelPreemptionPriority=2;
	NVIC_InitStruct.NVIC_IRQChannelSubPriority=1;
	NVIC_Init(&NVIC_InitStruct);

配置运行控制

    TIM_Cmd(TIM2,ENABLE);

如果我们想要实时看一下CNT计数器的值,封装一下函数 

uint16_t Timer_GetCounterr(void)
{
	return TIM_GetCounter(TIM2);
}

有关STM32F103C8T6的学习(5)——定时器的更多相关文章

  1. STM32读取串口传感器数据(颗粒物传感器,主动上传) - 2

    文章目录1.开发板选择*用到的资源2.串口通信(个人理解)3.代码分析(注释比较详细)1.主函数2.串口1配置3.串口2配置以及中断函数4.注意问题5.源码链接1.开发板选择我用的是STM32F103RCT6的板子,不过代码大概在F103系列的板子上都可以运行,我试过在野火103的霸道板上也可以,主要看一下串口对应的引脚一不一样就行了,不一样的就更改一下。*用到的资源keil5软件这里用到了两个串口资源,采集数据一个,串口通信一个,板子对应引脚如下:串口1,TX:PA9,RX:PA10串口2,TX:PA2,RX:PA32.串口通信(个人理解)我就从串口采集传感器数据这个过程说一下我自己的理解,

  2. STM32的HAL和LL库区别和性能对比 - 2

    LL库和HAL库简介LL:Low-Layer,底层库HAL:HardwareAbstractionLayer,硬件抽象层库LL库和hal库对比,很精简,这实际上是一个精简的库。LL库的配置选择如下:在STM32CUBEMX中,点击菜单的“ProjectManager”–>“AdvancedSettings”,在下面的界面中选择“AdvancedSettings”,然后在每个模块后面选择使用的库总结:1、如果使用的MCU是小容量的,那么STM32CubeLL将是最佳选择;2、如果结合可移植性和优化,使用STM32CubeHAL并使用特定的优化实现替换一些调用,可保持最大的可移植性。另外HAL和L

  3. ESP32学习入门:WiFi连接网络 - 2

    目录一、ESP32简单介绍二、ESP32Wi-Fi模块介绍三、ESP32Wi-Fi编程模型四、ESP32Wi-Fi事件处理流程 五、ESP32Wi-Fi开发环境六、ESP32Wi-Fi具体代码七、ESP32Wi-Fi代码解读6.1主程序app_main7.2自定义代码wifi_init_sta()八、ESP32Wi-Fi连接验证8.1测试方法8.2服务器模拟工具sscom58.3测试代码8.4测试结果前言为了开发一款亚马逊物联网产品,开始入手ESP32模块。为了能够记录自己的学习过程,特记录如下操作过程。一、ESP32简单介绍ESP32是一套Wi-Fi(2.4GHz)和蓝牙(4.2)双模解决方

  4. Spring Security 6.0系列【32】授权服务器篇之默认过滤器 - 2

    有道无术,术尚可求,有术无道,止于术。本系列SpringBoot版本3.0.4本系列SpringSecurity版本6.0.2本系列SpringAuthorizationServer版本1.0.2源码地址:https://gitee.com/pearl-organization/study-spring-security-demo文章目录前言1.OAuth2AuthorizationServerMetadataEndpointFilter2.OAuth2AuthorizationEndpointFilter3.OidcProviderConfigurationEndpointFilter4.N

  5. ruby - 摘要::CRC32 与 Zlib - 2

    在我的代码中,我需要使用各种算法(包括CRC32)对文件进行哈希处理。因为我还在Digest系列中使用其他加密哈希函数,所以我认为为它们维护一个一致的接口(interface)会很好。为了记录,我确实找到了digest-crc,一颗完全符合我要求的gem。问题是,Zlib是标准库的一部分,并且有一个我想重用的CRC32工作实现。此外,它是用C编写的,因此它应该提供与digest-crc相关的卓越性能,后者是纯ruby​​实现。实现Digest::CRC32一开始看起来非常简单:%w(digestzlib).each{|f|requiref}classDigest::CRC32一切正常:

  6. ruby - 安装gem : Couldn't reserve space for cygwin's heap, Win32错误487错误 - 2

    我正在尝试在我的机器上安装win32-apigem,但在构建native扩展时我遇到了一些问题:$geminstallwin32-api--no-ri--rdocTemporarilyenhancingPATHtoincludeDevKit...Buildingnativeextensions.Thiscouldtakeawhile...C:\Programs\dev_kit\bin\make.exe:***Couldn'treservespaceforcygwin'sheap,Win32error0ERROR:Errorinstallingwin32-api:ERROR:Failed

  7. Ruby 1.9 - 没有这样的文件可以加载 'win32/open3' - 2

    我在Windows上运行ruby​​1.9.2并试图移植在Ruby1.8中工作的代码。该代码使用以前运行良好的Open4.popen4。对于1.9.2,我做了以下事情:通过geminstallPOpen4安装了POpen4需要POpen4通过require'popen4'尝试像这样使用POpen4:Open4.popen4("cmd"){|io_in,io_out,io_er|...}当我这样做时,我得到了错误:nosuchfiletoload--win32/open3如果我尝试安装win32-open3(geminstallwin32-open3),我会收到错误消息:win32-op

  8. Dell Inspiron 5488加内存32G - 2

    DellInspiron5488加内存32G 原装内置内存仅仅8G,目前看,真的太小了! 1.内存型号Dell5488内存型号:DDR42666。笔记本有两个内存插槽,原装占了一个,还能扩展一个。 2.买内存如果买Dell原装笔记本内存,8G就得500块左右。 我咨询了一下,三星的笔记本内存,可以兼容。16G,299块(2023年2月23日,京东价) Dell5488内存组合,最多只能插两根16G内存。 我于是买了两根三星16G内存。装上,很爽😄 跑国产系统统信UOS,再也看不到用交换区了,32G内存,爽!  

  9. 蓝桥杯 stm32 MCP4017 - 2

    本文代码使用HAL库。文章目录前言一、MCP4017的重要特性二、MCP4017计算RBW阻值三、MCP4017地址四、MCP4017读写函数五、CubeMX创建工程(利用ADC测量MCP4017电压)、对应代码:总结前言一、MCP4017的重要特性蓝桥杯板子上的是MCP4017T-104ELT,如图1。MCP4017是一个可编程电阻,通过写入的数值可以改变电阻的大小。重点在于6引脚(W),5引脚(B&#

  10. STM32 OTA应用开发——通过USB实现OTA升级 - 2

    STM32OTA应用开发——通过USB实现OTA升级目录STM32OTA应用开发——通过USB实现OTA升级前言1环境搭建2功能描述3BootLoader的制作4APP的制作5烧录下载配置6运行测试结束语前言什么是OTA?百度百科:空中下载技术(Over-the-AirTechnology;OTA),是通过移动通信的空中接口实现对移动终端设备及SIM卡数据进行远程管理的技术。经过公网多年的应用与发展,已十分成熟,网络运营商通过OTA技术实现SIM卡远程管理,还能提供移动化的新业务下载功能。实际上,现在我们所说的OTA比百度百科的定义还要更广泛,OTA的形式已经不再局限于手机和SIM卡,只要涉及

随机推荐