好久没有写东西了,工作太忙都快没时间学习了。今天调试了STM32的定时器功能,STM32定时器比较多,但调试都是一样的,寄存器都是一一对应的。就拿tiM2举例说明。在网上搜了好多关于定时器的设置,但大多数都是一个版本,而且都是针对库函数操作的,让人看起来一头雾水,对于初学者很是不利(我也是初学者)。下面我将自己的定时器设置过程一一记录下来,以供大家参考,我们共同学习……
首先定义定时器头文件,也就是定义寄存器以供操作:
#define TIM2_CR1 (*((volatile unsigned long *)0x40000000))
#define TIM2_CR2 (*((volatile unsigned long *)0x40000004))
#define TIM2_DIER (*((volatile unsigned long *)0x4000000C))
#define TIM2_SR (*((volatile unsigned long *)0x40000010))
#define TIM2_EGR (*((volatile unsigned long *)0x40000014))
#define TIM2_CNT (*((volatile unsigned long *)0x40000024))
#define TIM2_PSC (*((volatile unsigned long *)0x40000028))
#define TIM2_ARR (*((volatile unsigned long *)0x4000002C))
然后配置定时器TIM2:
void TIM2_Configuration(void)
{
RCC_APB1ENR|=0x00000001; //位0 IM2EN TIM2 定时器时钟使能
TIM2_PSC |=19999; //设置预分频,实现1ms,20M ,20000/20M
TIM2_ARR = 1000; //1s产生一次中断
TIM2_DIER |=0X1; //允许tim2更新中断
SETENA0|=0x10000000; //允许T2中断
TIM2_CR1 |=0X1;//启动tim2 相当于//TIM_Cmd(TIM2, ENABLE); //是能定时器
}
注:RCC_APB1ENR,SETENA0前面的文章已经定义过
中断函数处理:
void TIM2_IRQHandler(void)
{
if(TIM1_SR&0x01)
{
TIM1_SR&=0xFFFE; //清中断标志
USART1_DR=0xAA;
while((USART1_SR)&0x80==0);
}
解释:1S中断来一次,向串口发送一字节数据0xAA;
接下来就在main()函数中添加配置定时器就可以了:
int main()
{
SystemInit0(); //系统(时钟)初始化
stm32_GpioSetup (); //GPIO初始化
stm32_UsartSetup (); //串口初始化
TIM2_Configuration(); //TIM1初始化
while(1)
{
GPIO_PORTB_ODR|=(1<<5); //LED闪烁
delay(10); //延时
GPIO_PORTB_ODR&=~(1<<5);
delay(10);
}
}
当然以上TIM2的配置必须在系统(时钟)初始化以后,因为这些外设的工作要有时钟来源的,知道了时钟来源,和频率才能计算出定时的时间等参数,本程序是基于所有时钟都设置成20M来执行的。前面的几个初始化函数以及寄存器设置都是调用的前面文章中设置好的。
好了,查看串口(9600),调试成功,1S发送1个0xAA,这样看起来是很简单,但调试的过程是艰难的,要一次次的实验,一个寄存器一个寄存器的进行配置,学习本来就是一个缓慢而且艰辛的过程,但是当你看到LED在闪烁,串口上收到数据时,说有的付出都是值得的,天道酬勤,一切困难都是纸老虎!
|