| **银杏科技有限公司旗下技术文档发布平台** |||| |技术支持电话|**0379-69926675-801**||| |技术支持邮件|Gingko@vip.163.com||| ^ 版本 ^ 日期 ^ 作者 ^ 修改内容 ^ | V1.0 | 2020-07-04 | gingko | 初次建立 | ===== 实验十二:通用定时器实验——定时点亮LED ===== ==== 一、 实验目的与意义 ==== - 了解STM32 TIMER结构。 - 了解STM32 TIMER特征。 - 掌握通用定时器的使用方法。 - 掌握STM32 HAL库中TIMER属性的配置方法。 - 掌握KEIL MDK 集成开发环境使用方法。 ==== 二、 实验设备及平台 ==== - iCore4 双核心板[[https://item.taobao.com/item.htm?spm=a1z10.1-c-s.w4004-22598974120.15.5923532fsFrHiE&id=551864196684|点击购买]]。 - JLINK(或相同功能)仿真器[[https://item.taobao.com/item.htm?id=554869837940|点击购买]]。 - Micro USB线缆。 - Keil MDK 开发平台。 - STM32CubeMX开发平台。 - 装有WIN XP(及更高版本)系统的计算机。 ==== 三、 实验原理 ==== === 1、用定时器简介 === * STM32F767的通用定时器包含一个16位或32位自动重载计数器(CNT),该计数器由可编程预分频器(PSC)驱动。STM32F767的通用定时器可以被用于:测量输入信号的脉冲长度(输入捕获)或者产生输出波形(输出比较和PWM)等。使用定时器预分频器和RCC时钟控制器预分频器,脉冲长度和波形周期可以在几个微秒到几个毫秒间调整。STM32F767的每个通用定时器都是完全独立的,没有互相共享的任何资源。 * STM32的通用TIMx(TIM2~TIM5和TIM9~TIM14)定时器功能包括: * (1)16位/32位(仅TIM2和TIM5)向上、向下、向上/向下自动装载计数(TIMx_CNT),注意:TIM9~TIM14只支持向上(递增)计数方式。 * (2)16位可编程(可以实时修改)预分频器(TIMx_PSC),计数器时钟频率的分频系数为1~65535之间的任意数值。 * (3)4个独立通道(TIMx_CH1~4,TIM9~TIM14最多2个通道),这些通道可以用来作为: * A.输入捕获 * B.输出比较 * C.PWM生成(边缘或中间对齐模式),注意:TIM9~TIM14不支持中间对齐模式 * D.单脉冲模式输出 * (4)可使用外部信号(TIMx_ETR)控制定时器和定时器互连(可以用1个定时器控制另外一个定时器)的同步电路。 * (5)如下事件发生时产生中断/DMA(TIM9~TIM14不支持DMA): * A.更新:计数器向上溢出/向下溢出,计数器初始化(通过软件或者内部/外部触发) * B.触发事件(计数器启动、停止、初始化或者由内部/外部触发计数) * C.输入捕获 * D.输出比较 * E.支持针对定位的增量(正交)编码器和霍尔传感器电路(TIM9~TIM14不支持) * F.触发输入作为外部时钟或者按周期的电流管理(TIM9~TIM14不支持) === 2、时基单元 === * 可编程定时器的主要模块由一个16位/32位计数器及其相关的自动重装寄存器组成。计数器可递增计数、递减计数或同时递增和递减计数。计数器的时钟可通过预分频器进行分频。计数器、自动重载寄存器和预分频器寄存器可通过软件进行读写。即使在计数器运行时也可执行读写操作。 * 时基单元包括: *  计数器寄存器(TIMx_CNT) *  预分频器寄存器(TIMx_PSC) *  自动重载寄存器(TIMx_ARR) * 自动重载寄存器是预装载的。对自动重载寄存器执行写入或读取操作时会访问预装载寄存器。预装载寄存器的内容既可以直接传送到影子寄存器,也可以在每次发生更新事件(UEV)时传送到影子寄存器,这取决于TIMx_CR1寄存器中的自动重载预装载使能位(ARPE)。当计数器达到上溢值(或者在递减计数时达到下溢值)并且TIMx_CR1寄存器中的UDIS位为0时,将发送更新事件。该更新事件也可由软件产生。计数器由预分频器输出CK_CNT提供时钟,仅当TIMx_CR1寄存器中的计数器启动位(CEN)置1时,才会启动计数器。注意,实际的计数器使能信号CNT_EN在CEN置1的一个时钟周期后被置1。 === 3、相关寄存器 === * (1)控制寄存器1(TIMx_CR1),该寄存器的各位描述如图所示: {{ :icore4:icore4_arm_hal_12_1.png?direct |}} * 在本实验中,我们只用到了TIMx_CR1的最低位,也就是计数器使能位,该位必须置 1,才能让定时器开始计数。 * (2)DMA/中断使能寄存器(TIMx_DIER),该寄存器是一个16位的寄存器,其各位描述如图所示: {{ :icore4:icore4_arm_hal_12_2.png?direct |}} * 这里我们同样仅关心它的第0位,该位是更新中断允许位,实验中用到的是定时器的更新中断,所以该位要设置为1,来允许由于更新事件所产生的中断。 * (3)预分频寄存器(TIMx_PSC)。该寄存器用来设置对时钟进行分频,然后提供给计数器,作为计数器的时钟。该寄存器的各位描述如图所示: {{ :icore4:icore4_arm_hal_12_3.png?direct |}} * 这里,定时器的时钟来源有4个: *  内部时钟(CK_INT) *  外部时钟模式1:外部输入脚(TIx) *  外部时钟模式2:外部触发输入(ETR),仅适用于TIM2、TIM3、TIM4 *  内部触发输入(ITRx):使用A定时器作为B定时器的预分频器(A为B提供时钟)。 * 这些时钟,具体选择哪个可以通过TIMx_SMCR寄存器的相关位来设置。这里的CK_INT时钟是从APB1倍频的来的,除非APB1的时钟分频数设置为1(一般都不会是1),否则通用定时器TIMx的时钟是APB1时钟的2倍,当APB1的时钟不分频的时候,通用定时器TIMx的时钟就等于APB1的时钟。这里还要注意的就是高级定时器以及TIM9~TIM11的时钟不是来自APB1,而是来自APB2的。 * (4)自动重装载寄存器(TIMx_ARR),该寄存器在物理上实际对应着2个寄存器。一个是程序员可以直接操作的,另外一个是程序员看不到的,这个看不到的寄存器在《STM32F7中文参考手册》里面被叫做影子寄存器。事实上真正起作用的是影子寄存器。根据TIMx_CR1寄存器中APRE位的设置:APRE=0时,预装载寄存器的内容可以随时传送到影子寄存器,此时2者是连通的;而APRE=1时,在每一次更新事件(UEV)时,才把预装载寄存器(ARR)的内容传送到影子寄存器。自动重装载寄存器的各位描述如图所示: {{ :icore4:icore4_arm_hal_12_4.png?direct |}} * (5)状态寄存器(TIMx_SR),该寄存器用来标记当前与定时器相关的各种事件/中断是否发生。该寄存器的各位描述如图所示: {{ :icore4:icore4_arm_hal_12_5.png?direct |}} * 只要对以上几个寄存器进行简单的设置,我们就可以使用通用定时器了,并且可以产生中断。 * 本实验中,通过STM32的三个GPIO口来驱动LED灯的三个通道,设定GPIO为推挽输出模式,采用灌电流的方式与LED连接,输出高电平LED灭,输出低电平LED亮,通过通用定时器TIM3实现500ms定时,每500ms变换一次LED颜色。 ==== 四、 实验程序 ==== === 1、主函数 === int main(void) { system_clock.initialize();//初始化系统时钟 led.initialize(); //LED初始化 timer3.initialize(); //timer3初始化 while(1); } === 2、定时器初始化 === static int initialize(void) { //定时1s htim3.Instance = TIM3; htim3.Init.Prescaler = 21599; //预分频系数 htim3.Init.CounterMode = TIM_COUNTERMODE_UP; //向上计数模式 htim3.Init.Period = 4999; //自动装载值 htim3.Init.ClockDivision = TIM_CLOCKDIVISION_DIV1; //时钟分频因子 htim3.Init.AutoReloadPreload=TIM_AUTORELOAD_PRELOAD_ENABLE;//自动装载使能 if (HAL_TIM_Base_Init(&htim3) != HAL_OK) //初始化定时器参数 { while(1); } //使能定时器更新中断和使能定时器 HAL_TIM_Base_Start_IT(&htim3); return 0; } HAL_StatusTypeDef HAL_TIM_Base_Start_IT(TIM_HandleTypeDef *htim) { /* 检查参数 */ assert_param(IS_TIM_INSTANCE(htim->Instance)); /* 使能定时器更新中断 */ __HAL_TIM_ENABLE_IT(htim, TIM_IT_UPDATE); /* 使能外围设备 */ __HAL_TIM_ENABLE(htim); /* 返回函数状态*/ return HAL_OK; } === 3、TIM3中断优先级设置 === * HAL中定时器使能是通过宏定义标识符来实现对相关寄存器操作的。在定时器中断使能之后,因为要产生中断,必不可少的要设置 NVIC 相关寄存器,设置中断优先级。和串口等其他外设一样, HAL库为定时器初始化定义了回调函数 HAL_TIM_Base_MspInit。一般情况下,与MCU有关的时钟使能,以及中断优先级配置我们都会放在该回调函数内部。 void HAL_TIM_Base_MspInit(TIM_HandleTypeDef* tim_baseHandle) { if(tim_baseHandle->Instance==TIM3) { /* TIM3时钟使能 */ __HAL_RCC_TIM3_CLK_ENABLE(); /* 初始化中断 */ HAL_NVIC_SetPriority(TIM3_IRQn, 1, 3);//设置中断优先级 HAL_NVIC_EnableIRQ(TIM3_IRQn);//开启中断 } } === 4、中断服务函数 === * 我们通过重写中断回调函数,来处理定时器产生的相关中断。 void HAL_TIM_PeriodElapsedCallback(TIM_HandleTypeDef *htim) { static int counter = 0; if(counter % 3 == 0){ LED_RED_ON; LED_BLUE_OFF; LED_GREEN_OFF; }else if(counter % 3 == 1){ LED_RED_OFF; LED_BLUE_ON; LED_GREEN_OFF; }else if(counter % 3 == 2){ LED_RED_OFF; LED_BLUE_OFF; LED_GREEN_ON; } counter ++; } ==== 五、 实验步骤 ==== - 把仿真器与iCore4的SWD调试口相连(直接相连或者通过转接器相连); - 把iCore4通过Micro USB线与计算机相连,为iCore4供电; - 打开Keil MDK 开发环境,并打开本实验工程; - 烧写程序到iCore4上; - 也可以进入Debug 模式,单步运行或设置断点验证程序逻辑。 ==== 六、 实验现象 ==== * 三色LED每隔1s循环点亮。