目录
使用的是正点原子的探索者开发板进行学习,芯片:STM32F407ZGTx
学习说明此文档为本人的学习笔记,注重实践,关于理论部分会给出相应的学习链接。
本文参考了《正点原子的寄存器开发指南》、《STM32F4参考手册》
详细介绍的时钟的配置与,定时器时钟如何从时钟树而来的。
包括TIM1和TIM8高级控制定时器、TIM2-5和TIM9-14通用定时器、TIM6和TIM7基本定时器。


STM32F4 的定时器除了 TIM6 和 7。其他的定时器都可以用来产生 PWM 输出。其中高级定时器 TIM1 和 TIM8 可以同时产生多达 7 路的 PWM 输出。而通用定时器也能同时产生多达 4路的 PWM 输出!这里我们仅使用 TIM14 的 CH1 产生一路 PWM 输出。
关于PWM在FPGA上的产生更加有助于理解
二、15【FPGA】呼吸灯实现_追逐者-桥的博客-CSDN博客_fpga 呼吸灯
除了使用到了定时器中断的那几个寄存器外还用到了以下寄存器

在 t1~t2 之间,可能产生 N 次定时器溢出,这就要求我们对定时器溢出,做处理,防止高电平太长,导致数据不准确。如图15.1.1所示,t1~t2之间,CNT计数的次数等于:N*ARR+CCRx2,有了这个计数次数,再乘以 CNT 的计数周期,即可得到 t2-t1 的时间长度,即高电平持续时间。
STM32F4 的定时器,除了 TIM6 和 TIM7,其他定时器都有输入捕获功能。
- //设置时钟频率:HSE=8M M=8 N=336 P=2 Q=7
- //PLL=HSE/M=1M PLLCLK=PLL*N/P=168M USB=PLL*N/Q=48M
- Stm32_Clock_Init(336,8,2,7);

想CFGR[12:10]位写5,即4分频,因此APB1=PLLCLK/4=42M
RCC->CFGR|=(0<<4)|(5<<10)|(4<<13);

TIM3挂载在APB1时钟总线上,
TIM3由于在配置APB1的时候使用了4分频,所以定时器进行了2倍频,为84MHz。
RCC->APB1ENR |= 1<<1;
时间计算公式:Tout = ( (arr+1) * (psc+1) ) / Tclk;
- TIM3->ARR=arr; //设置TIM3的自动装载值
- TIM3->PSC=psc; //预分频器设置

TIM3->DIER|=1<<0;
TIM3->CR1|=0x01;
使能定时计数器器,且为递增计数
MY_NVIC_Init(1,3,TIM3_IRQn,2);
TIMX-SR状态寄存器:

- void TIM3_IRQHandler(void)
- {
- if(TIM3->SR&0X0001) //中断发生,最低位由硬件置1
- {
- LED1=!LED1;
- }
- TIM3->SR&=~(1<<0); //必须将最低位软件置0,等待下次中断的到来
- }
TIM(定时器)=84MHz psc(分频系数)=8400 arr(重装在值)=5000
定时器计数频率 = 84MHz / 8400 =10KHz
Tout = 5000 / 10KHz = 0.5s
- int main(void)
- {
- Stm32_Clock_Init(336,8,2,7); //pll=1M pllclk=168M
- delay_init(168);
- LED_Init();
- TIM3_Int_Init(5000-1,8400-1); //arr psc
- while(1)
- {
- LED0=!LED0;
- delay_ms(200);
- };
- }
- RCC->APB1ENR|=1<<8; //使能TIM13定时器的时钟
- RCC->AHB1ENR|=1<<5; //使能端口FA9的端口时钟
- GPIO_Set(GPIOF,PIN9,GPIO_MODE_AF,GPIO_OTYPE_PP,GPIO_SPEED_100M,GPIO_PUPD_PU);
- GPIO_AF_Set(GPIOF,9,9); //PF9,AF9
- TIM14->ARR=arr; //重装载值
- TIM14->PSC=psc; //分频系数

#define LED0_PWM_VAL TIM14->CCR1
- int main(void)
- {
- u16 led0pwmval=0; //TIM14-CCR1的值,及占空比的计数值
- u8 dir=1;
- Stm32_Clock_Init(336,8,2,7);
- delay_init(168);
- TIM14_PWM_Init(500-1,84-1); //1MHz的计数频率,PWM频率2KHz, T=0.5ms
- while(1)
- {
- delay_ms(10); //10ms重载一次占空比,20次PWM的输出
- if(dir)led0pwmval++; //占空比增加
- else led0pwmval--; //占空比减小
- if(led0pwmval>300)dir=0;
- if(led0pwmval==0)dir=1;
- LED0_PWM_VAL=led0pwmval; //重新装载占空比值
- }
- }
- RCC->APB1ENR|=1<<3;
- RCC->AHB1ENR|=1<<0;
- GPIO_Set(GPIOA,PIN0,GPIO_MODE_AF,GPIO_OTYPE_PP,GPIO_SPEED_100M,GPIO_PUPD_PD);
- GPIO_AF_Set(GPIOA,0,2); //PA0,AF2 (TIM5-CH1)
计数分频系数与计数器数设置,在定时器中断有相关描述
- TIM5->ARR=arr;
- TIM5->PSC=psc;
TIM5-CCMR1捕获比较寄存器1相关位描述


TIM5_CCER捕获/比较使能寄存器相关位描述


- TIM5->CCMR1|=1<<0; //CCIS位,将CC1设置成输入且映射到TI1
- TIM5->CCMR1|=0<<4; //IC1PSC,无分频
- TIM5->CCMR1|=0<<10; //IC1F,无滤波器
- TIM5->CCER|=0<<1; //CC1P,无反向,上升沿触发
- TIM5->CCER|=1<<0; //CC1E,使能输入捕获

- TIM5->DIER|=1<<1;
- TIM5->DIER|=1<<0;
- //设置软件控制产生更新事件,是载入PSC值立刻生效,否则要等到溢出后才生效
- TIM5->EGR=1<<0;
TIM5->CR1|=0x01;
MY_NVIC_Init(2,0,TIM5_IRQn,2)
- u8 TIM5CH1_CAPTURE_STA=0; //捕获标志0x40上升沿捕获,0x80下降沿捕获
- u32 TIM5CH1_CAPTURE_VAL; //捕获计数值存储空间
- void TIM5_IRQHandler(void)
- {
- u16 tsr;
- tsr=TIM5->SR; //定时器状态寄存器
- if((TIM5CH1_CAPTURE_STA&0X80)==0) //捕获未完成
- {
- if(tsr&0X01) //定时器溢出标志
- {
- if(TIM5CH1_CAPTURE_STA&0X40) //上升沿被捕获到后
- {
- if((TIM5CH1_CAPTURE_STA&0X3F)==0X3F) //达到计数器溢出值
- {
- TIM5CH1_CAPTURE_STA|=0X80; //标记捕获到了一个定时计数器周期
- TIM5CH1_CAPTURE_VAL=0XFFFFFFFF; //保存一个完成的计数周期
- }else TIM5CH1_CAPTURE_STA++; //计数器未溢出
- }
- }
- if(tsr&0x02) //发生上升沿捕获事件进入中断
- {
- if(TIM5CH1_CAPTURE_STA&0X40) //上升沿捕获已完成,下降沿捕获中断进入
- {
- TIM5CH1_CAPTURE_STA|=0X80; //已经捕获到下降沿标志
- TIM5CH1_CAPTURE_VAL=TIM5->CCR1; //储存捕获计数器值
- TIM5->CCER&=~(1<<1); //设置为上升沿捕获进入中断
- }else //将计数器清零,重新计数
- {
- TIM5CH1_CAPTURE_STA=0;
- TIM5CH1_CAPTURE_VAL=0;
- TIM5CH1_CAPTURE_STA|=0X40; //已经捕获到上升沿标志
- TIM5->CR1&=~(1<<0); //使能定时器
- TIM5->CNT=0; //清空计数器
- TIM5->CCER|=1<<1; //设置为下降沿捕获进入中断
- TIM5->CR1|=0x01; //开启定时器
- }
- }
- }
- TIM5->SR=0;
- }
- extern u8 TIM5CH1_CAPTURE_STA;
- extern u32 TIM5CH1_CAPTURE_VAL;
- int main(void)
- {
- long long temp=0;
- Stm32_Clock_Init(336,8,2,7);
- delay_init(168);
- uart_init(84,115200);
- TIM14_PWM_Init(500-1,84-1);
- TIM5_CH1_Cap_Init(0XFFFFFFFF,84-1);
- while(1)
- {
- delay_ms(10);
- LED0_PWM_VAL++;
- if(LED0_PWM_VAL==300)LED0_PWM_VAL=0;
- if(TIM5CH1_CAPTURE_STA&0X80) //捕获到了一个定时器计数周期或者下降沿
- {
- temp=TIM5CH1_CAPTURE_STA&0X3F; //提取捕获值
- temp*=0XFFFFFFFF; //计算捕获时间
- temp+=TIM5CH1_CAPTURE_VAL; //加入定时器周期时间和下降沿捕获时间
- printf("HIGH:%lld us\r\n",temp); //输出捕获总时间
- TIM5CH1_CAPTURE_STA=0; //将捕获标志位清0,进入下一次捕获
- }
- }
- }
电容触摸按键:电容按键是接触式的,点一下就松开(与微动开关类似),因此需要消除抖。在之前的微动开关使用时间延迟判断两次,状态相同时才认为是按下。这里同理,这里使用的是输入捕获即手指接触一定时间后到达B认为电容按键按下。

正点原子中的文档内容描述:

对于电容按键用的不是特别多,这里不进行实验,有需要的可以自己去看一下正点原子的相关实验。