`
hulianwang2014
  • 浏览: 687291 次
文章分类
社区版块
存档分类
最新评论
  • bcworld: 排版成这样,一点看的欲望都没有了
    jfinal

CC2430 定时器溢出中断 详解

 
阅读更多

定时器1使用总结——溢出中断

1 目的说明

实现定时器最简单的溢出中断,结合我手头的开发板,使得位于P10的LED灯,以2HZ的速度不断闪烁。这样的实验还是非常亲切的,让我想起了第一次在51上实现了这样的代码,自己第一次在CC2430上实现,依然非常激动。

2 使用方法概述

需要使用定时器的中断,需要知道如何操作才可以产生这个中断请求。数据手册中提到需要两个条件,第一IEN1.T1EN需要置位,第二TIMIF.OVFIM需要置位。代码中使用modulo、模式,使用该模式可以改变定时器溢出的频率。

3 代码总览

先来看看所有的代码,然后再分步解释。

//头文件 

#include "hal.h"

//函数声明

void Timer1_Init();

//主函数

void main(){

         //初始化外部时钟

         SET_MAIN_CLOCK_SOURCE(CRYSTAL);

         //P1_0 输出

         IO_DIR_PORT_PIN(1,0,IO_OUT);

         //初始化定时器1

         Timer1_Init();

         while(1){

         }

}

 

void Timer1_Init(){

         //定时器1复位

         TIMER1_INIT();        

 

         //设定定时器相关参数

  //128分频0000 1100

  T1CTL = 0x0c;     

         //溢出值低8位 

         T1CC0L=0x24; 

         //溢出值高8位

  T1CC0H=0xF4;      

 

         //定时器T1溢出中断使能

         TIMER1_ENABLE_OVERFLOW_INT(TRUE);

         //定时器T1中断使能

         INT_ENABLE(INUM_T1,INT_ON);

 

         //全局中断使能

         INT_GLOBAL_ENABLE(INT_ON);

  

       //启动定时器1

         TIMER1_RUN(TRUE);

}

 

//定时器1中断函数

#pragma vector=T1_VECTOR

__interrupt void Timer1_ISR(void)

{

  //检查中断标志位

  if(T1CTL & 0x10){

  //LED灯反转  

  P1_0 = !P1_0;       

  //清中断标志

  T1CTL &= ~0x10; 

  }

}


4 主函数说明

//初始化外部时钟

SET_MAIN_CLOCK_SOURCE(CRYSTAL);

//P1_0 输出

IO_DIR_PORT_PIN(1,0,IO_OUT);

//初始化定时器1

Timer1_Init();

操作CC2430之前,先指定系统时钟,这是一个好习惯。由于定时器时钟和系统时钟频率有关,所以必须要设定好系统的时钟。在SET_MAIN_CLOCK_SOURCE()在这个动作宏中,把系统时钟设定为32MHz。(该宏前面的文章已经提到,不多做说明)

请注意定时器的时钟频率 默认为16MHz,而不是32MHz。

请注意CLKCON的5:3位, 该3位组成了一个定时器时钟的分频器,该参数决定了定时器的时钟频率。在定时器1的相关操作中还有定时器时钟的分频系数设置,那是定时器1特有的,这里的定时器分频参数是分频了定时器1,3,4的时钟。相见数据手册或下图:

为了操作IO口,定义LED相关的IO口为输出。IO_DIR_PORT_PIN()的相关操作如下面的代码所示:

#define IO_DIR_PORT_PIN(port, pin, dir)  \

   do {                                  \

      if (dir == IO_OUT)                 \

         P##port##DIR |= (0x01<<(pin));  \

      else                               \

         P##port##DIR &= ~(0x01<<(pin)); \

   }while(0)


该宏操作了PXDIR寄存器,定义了IO口的方向。

5 定时器初始化操作

TIMER1_INIT()把定时器1的寄存器全部复位。具体的代码如下:

#define TIMER1_INIT()   \

   do {                 \

      T1CTL  = 0x00;    \

      T1CCTL0 = 0x00;   \

      T1CCTL1 = 0x00;   \

      T1CCTL2 = 0x00;   \

      TIMIF &= ~0x40;   \

   } while (0)

从这个代码中也可以看出定时器1的操作和哪些寄存器有关。具体的定义可以查看数据手册,这里不多做说明。

6 设定定时器中断频率

操作代码如下

  //128分频0000 1100

  T1CTL = 0x0c;     

  //溢出值低8位 

  T1CC0L=0x24; 

  //溢出值高8位

  T1CC0H=0xF4;  


由于CC2430的运行速度比较快,所以需要对定时器1进行分频。由于T1CTL在前面的函数中已经被全部复位,所以可以舒服的操作T1CTL寄存器。在这里把系统时钟设定为128分频,定时器T1的运行速度只有125K。这个速度对于0.5闪烁来说,还是非常快的。

接着设定T1CC0寄存器。这个寄存器还是非常特殊的。请注意,T1CC0在modulo模式和up-down模式中,始终作为定时器T1计数的最大值。数据手册上说定时器1有3个比较匹配中断,其实这个和AVR的定时器1有的两个比较匹配时一样的,因为CC2430没有一个专用寄存器储存计数的最大值,那么定时器1的比较通道0就“牺牲”了比较通道的作用。所以要产生两路频率指定的PWM波的时候,T1CC0作为最大值决定PWM的频率,而T1CC1和T1CC2决定PWM的相位。

下面再讲讲计数值的计算方法。我是从分频的角度思考的,写出这个等式:

Ftimer/(N*T1CC0) = Fdesi。

其中Ftimer为定时器的运行时钟,此处为16,000,000Hz;N为分频系数,此处为128;T1CC0为定时器的计数值;Fdesi为期望溢出频率,此处为2Hz。带入这个等式可以计算出T1CC0的值为62500,写成16进制为F424。如果计算出来的结果大于65536,那么只能进一步降低定时器的运行频率,在这里只能调整Ftimer了。

7 使能该使能的内容

         //定时器T1溢出中断使能

         TIMER1_ENABLE_OVERFLOW_INT(TRUE);

         //定时器T1中断使能

         INT_ENABLE(INUM_T1,INT_ON);

         //全局中断使能

         INT_GLOBAL_ENABLE(INT_ON);


开篇的时候就说了需要操作哪两个寄存器——第一IEN1.T1EN,第二TIMIF.OVFIM。操作时分别使用了以下两个宏。具体的代码如下:

#define TIMER1_ENABLE_OVERFLOW_INT(val) \

   (TIMIF =  (val) ? TIMIF | 0x40 : TIMIF & ~0x40)

#define INT_ENABLE(inum, on)                        \

   do {                                             \

      if      (inum==INUM_RFERR) { RFERRIE = on; }  \

      else if (inum==INUM_ADC)   { ADCIE   = on; }  \

      else if (inum==INUM_URX0)  { URX0IE  = on; }  \

      else if (inum==INUM_URX1)  { URX1IE  = on; }  \

      else if (inum==INUM_ENC)   { ENCIE   = on; }  \

      else if (inum==INUM_ST)    { STIE    = on; }  \

      else if (inum==INUM_P2INT) { (on) ? (IEN2 |= 0x02) : (IEN2 &= ~0x02); } \

      else if (inum==INUM_UTX0)  { (on) ? (IEN2 |= 0x04) : (IEN2 &= ~0x04); } \

      else if (inum==INUM_DMA)   { DMAIE   = on; }  \

      else if (inum==INUM_T1)    { T1IE    = on; }  \

      else if (inum==INUM_T2)    { T2IE    = on; }  \

      else if (inum==INUM_T3)    { T3IE    = on; }  \

      else if (inum==INUM_T4)    { T4IE    = on; }  \

      else if (inum==INUM_P0INT) { P0IE    = on; }  \

      else if (inum==INUM_UTX1)  { (on) ? (IEN2 |= 0x08) : (IEN2 &= ~0x08); } \

      else if (inum==INUM_P1INT) { (on) ? (IEN2 |= 0x10) : (IEN2 &= ~0x10); } \

      else if (inum==INUM_RF)    { (on) ? (IEN2 |= 0x01) : (IEN2 &= ~0x01); } \

      else if (inum==INUM_WDT)   { (on) ? (IEN2 |= 0x20) : (IEN2 &= ~0x20); } \

   } while (0)


最后还要操作一个“总”中断,这个是51中断的老大——全局中断EA。有点基础的一定知道这个东西,代码如下:

// Global interrupt enables

#define INT_GLOBAL_ENABLE(on) EA=(!!on)

(不知道为什么来个双重否定????)

8 启动定时器T1

准备好所有的初始化代码之后,才开始启动定时器。在启动定时器就是选择工作方式(也搞不明白为什么CC2430不来一个定时器启动相关的寄存器),把工作方式定义为modulo模式。具体的代码如下:

#define TIMER1_RUN(value)      (T1CTL = (value) ? T1CTL|0x02 : T1CTL&~0x03)

由于默认的初始值为00(从第0位开始),把第1位置位就相当于选择了modulo模式,清零保留默认模式。

9 定时器中断

#pragma vector=T1_VECTOR

__interrupt void Timer1_ISR(void)

{

  //检查中断标志位

  if(T1CTL & 0x10){

  //LED灯反转  

  P1_0 = !P1_0;       

  //清中断标志

  T1CTL &= ~0x10; 

  }

}

所有的中断都有固定的写法,这个大家必须牢牢记住。先使用伪命令定义中断入口地址:#pragma vector=T1_VECTOR,然后定义函数名称__interrupt void Timer1_ISR(void)。通过检测OVFIM(T1CTL 第4位)来判断是否发生了溢出中断,最后通过软件清除中断标志位。(但是我发现不清除中断标志位同样可以再次进入中断,最后查看数据手册,如何清除这个中断标志位还和同时处理多个中断的关系有关,在这里也不多做解释)。

总结

在这里使用了定时器T1的modulo模式,进行了定时器溢出的相关操作,在这里需要掌握的是定时器溢出频率的计算和中断服务函数的书写。后面还会讲定时器比较匹配的相关内容,如果有兴趣的话,请关注。

分享到:
评论

相关推荐

Global site tag (gtag.js) - Google Analytics