原创 STM32笔记(一)TIM模块定时器向上溢出 & 输出比较

2008-12-13 10:28 14307 8 11 分类: MCU/ 嵌入式

首先我们必须肯定ST公司的实力,也承认STM32的确是一款非常不错的Cortex-M3核单片机,但是,他的手册实在是让人觉得无法理解,尤其是其中
的TIM模块,没有条理可言,看了两天几乎还是不知所云,让人很是郁闷。同时配套的固件库的说明也很难和手册上的寄存器对应起来,研究起来非常费劲!功能
强大倒是真的,但至少也应该配套一个让人看的明白的说明吧~~


  两天时间研究了STM32定时器的最最基础的部分,把定时器最基础的两个功能实现了,余下的功能有待继续学习。

  首先有一点需要注意:FWLib固件库目前的最新版应该是V2.0.x,V1.0.x版本固件库中,TIM1模块被独立出来,调用的函数与其他定时器不同;在V2.0系列版本中,取消了TIM1.h,所有的TIM模块统一调用TIM.h即可。网络上流传的各种代码有许多是基于v1版本的固件库,在移植到v2版本固件库时,需要做些修改。本文的所有程序都是基于V2.0固件库。




  以下是定时器向上溢出示例代码:




//Step1.时钟设置:启动TIM1


RCC_APB2PeriphClockCmd(RCC_APB2Periph_TIM1, ENABLE);



//Step2.中断NVIC设置:允许中断,设置优先级


NVIC_InitStructure.NVIC_IRQChannel =
TIM1_UP_IRQChannel;   
//更新事件

NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 0;  
//抢占优先级0

NVIC_InitStructure.NVIC_IRQChannelSubPriority = 1;         
//响应优先级1

NVIC_InitStructure.NVIC_IRQChannelCmd =
ENABLE;            
//允许中断

NVIC_Init(&NVIC_InitStructure);                            
//写入设置



//Step3.TIM1模块设置


void
TIM_Configuration(void)

{

TIM_TimeBaseInitTypeDef TIM_BaseInitStructure;

TIM_OCInitTypeDef TIM_OCInitStructure;





//TIM1
使用内部时钟


//TIM_InternalClockConfig(TIM1);




//TIM1基本设置


//设置预分频器分频系数71,即APB2=72M,
TIM1_CLK=72/72=1MHz


//TIM_Period(TIM1_ARR)=1000,计数器向上计数到1000后产生更新事件,计数值归零


//向上计数模式

//TIM_RepetitionCounter(TIM1_RCR)=0,每次向上溢出都产生更新事件


TIM_BaseInitStructure.TIM_Period = 1000;

TIM_BaseInitStructure.TIM_Prescaler = 71;

TIM_BaseInitStructure.TIM_ClockDivision = 0;

TIM_BaseInitStructure.TIM_CounterMode = TIM_CounterMode_Up;

TIM_BaseInitStructure.TIM_RepetitionCounter = 0;

TIM_TimeBaseInit(TIM1,
&TIM_BaseInitStructure);



//清中断,以免一启用中断后立即产生中断


TIM_ClearFlag(TIM1, TIM_FLAG_Update);

//使能TIM1中断源


TIM_ITConfig(TIM1, TIM_IT_Update, ENABLE);



//TIM1总开关:开启


TIM_Cmd(TIM1, ENABLE);

}



//Step4.中断服务子程序:


void
TIM1_UP_IRQHandler(void)

{

GPIOC->ODR ^= (1<<4);                        
//闪灯

TIM_ClearITPendingBit(TIM1, TIM_FLAG_Update); //清中断

}




  下面是输出比较功能实现TIM1_CH1管脚输出指定频率的脉冲:




//Step1.启动TIM1,同时还要注意给相应功能管脚启动时钟


RCC_APB2PeriphClockCmd(RCC_APB2Periph_TIM1, ENABLE);

RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE);



//Step2.
PA.8口设置为TIM1的OC1输出口



GPIO_InitStructure.GPIO_Pin = GPIO_Pin_8;

GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;

GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;

GPIO_Init(GPIOA, &GPIO_InitStructure);



//Step3.使能TIM1的输出比较匹配中断


NVIC_InitStructure.NVIC_IRQChannel = TIM1_CC_IRQChannel;

NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 1;

NVIC_InitStructure.NVIC_IRQChannelSubPriority = 1;

NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;

NVIC_Init(&NVIC_InitStructure);



//Step4.
TIM模块设置



void
TIM_Configuration(void)

{

   
TIM_TimeBaseInitTypeDef TIM_BaseInitStructure;

   
TIM_OCInitTypeDef TIM_OCInitStructure;





   //TIM1基本计数器设置


   
TIM_BaseInitStructure.TIM_Period = 0xffff;                     
//这里必须是65535 


   
TIM_BaseInitStructure.TIM_Prescaler = 71;                      
//预分频71,即72分频,得1M


   
TIM_BaseInitStructure.TIM_ClockDivision = 0;

   
TIM_BaseInitStructure.TIM_CounterMode = TIM_CounterMode_Up;

   
TIM_BaseInitStructure.TIM_RepetitionCounter = 0;

   
TIM_TimeBaseInit(TIM1,
&TIM_BaseInitStructure);



   //TIM1_OC1模块设置


   
TIM_OCStructInit(&
TIM_OCInitStructure); 

   
TIM_OCInitStructure.TIM_OCMode =
TIM_OCMode_Toggle;            
//管脚输出模式:翻转


   
TIM_OCInitStructure.TIM_Pulse = 2000;                          
//翻转周期:2000个脉冲


   
TIM_OCInitStructure.TIM_OutputState =
TIM_OutputState_Enable;  
//使能TIM1_CH1通道


   
TIM_OCInitStructure.TIM_OCPolarity =
TIM_OCPolarity_High;      
//输出为正逻辑

   
TIM_OC1Init(TIM1,
&TIM_OCInitStructure);                       
//写入配置



   //清中断

   
TIM_ClearFlag(TIM1, TIM_FLAG_CC1);



   //TIM1中断源设置,开启相应通道的捕捉比较中断


   
TIM_ITConfig(TIM1, TIM_IT_CC1, ENABLE);



   //TIM1开启

   
TIM_Cmd(TIM1, ENABLE);

   //通道输出使能

   
TIM_CtrlPWMOutputs(TIM1, ENABLE);

}



Step5.中断服务子程序


void
TIM1_CC_IRQHandler(void)

{

    u16
capture;

   if(TIM_GetITStatus(TIM1,
TIM_IT_CC1) == SET)

    {

       
TIM_ClearITPendingBit(TIM1, TIM_IT_CC1 );

       
capture = TIM_GetCapture1(TIM1);

       
TIM_SetCompare1(TIM1, capture + 2000);

       
//这里解释下:

       
//将TIM1_CCR1的值增加2000,使得下一个TIM事件也需要2000个脉冲,


       
//另一种方式是清零脉冲计数器


       
//TIM_SetCounter(TIM2,0x0000);


    }

}



  关于TIM的操作,要注意的是STM32处理器因为低功耗的需要,各模块需要分别独立开启时钟,所以,一定不要忘记给用到的模块和管脚使能时钟,因为这个原因,浪费了我好多时间阿~~!


下一回,将介绍TIM模块PWM的功能!
PARTNER CONTENT

文章评论3条评论)

登录后参与讨论

用户1588142 2011-12-24 18:23

mark

用户1670261 2009-8-8 16:41

学stm32,在网上搜到好多九九的文章,呵呵,给了我很多帮助,我是初学者,九九的文章写得很清楚,真是十分感谢呀

用户213184 2009-8-6 09:53

void TIM1_UP_IRQHandler(void)函数中的GPIOC->ODR ^= (1<<4); 这个函数我太理解,我想问一下,你这个是不是让GPIOC_Pin_4管脚去点亮外设的LED?如果是的话,那并未见你将此引脚与外设LED相连,如果不是的话,请多多指教!谢谢!
相关推荐阅读
用户699237 2009-06-04 14:36
发布TBDML Plus用户手册和驱动包
九九的TBDML Plus已经于2009年5月1日正式发售在EDN和21IC上发布TBDML Plus用户使用手册和用户驱动包欢迎各位用户下载使用TBDML Plus用户包下载:https://sta...
用户699237 2009-05-30 16:39
低价供应Freescale飞思卡尔 BDM仿真器 调试器 兼容8/16/32位单片机
本店最新产品的飞思卡尔三合一BDM仿真器TBDML Plus支持S12 S08 ColdFireV1三种内核的仿真调试USB接口 最新JB16+12MHz晶振硬件 仿真速度是原BDM的2倍!超小体积 ...
用户699237 2009-05-20 15:12
隆重发布高速模拟视频解码模块(智能车可用)
关于智能车摄像头的选择,CCD以其高感光度、能在运动时获取较为清晰的图像而成为智能车摄像头的首选,这是一个趋势。详情请搜索《CCD与CMOS摄像头在智能车竞赛中的选择》  同时摄像头的前瞻越来越大,很...
用户699237 2009-05-15 16:45
智能车竞赛中CCD与CMOS摄像头的选择
https://static.assets-stash.eet-china.com/album/old-resources/2009/5/15/3cb54ae1-ee52-4f05-9629-86c2...
用户699237 2009-05-12 21:03
智能车上位机:串口图像显示程序
这是我很久以前写的程序功能很不完善如果要使用 请先把串口调整到COM1 不能在设置窗口中设置串口号 否则会出错同时连拍功能没有实现 嘿嘿 不好意思哦具体如何发送 在readme中有详细解释没空改程序了...
用户699237 2009-05-12 21:02
OV7620 OV6620等CMOS摄像头图像采集的方法
本文由九九原创,发表于《电子技术应用》2008年第9期,版权归作者与出版社所有,转载请务必注明作者出处,谢谢http://shop35388432.taobao.com九九小铺,供应:MC9S12DG...
EE直播间
更多
我要评论
3
8
关闭 站长推荐上一条 /1 下一条