s3c2410/s3c2440 定时器的一些东西

来源:互联网 发布:sql server导入oracle 编辑:程序博客网 时间:2024/05/29 16:50

   近两天做事效率不高,一些事,一些人,还有就是马上就要回家了。。。

    已然也觉得时间已经不多,所以今天认真的阅读了说明书,查了些资料。。。把那个AD驱动之后的定时器驱动补上。

    网上关于定时器驱动源程序很难找,不过好些好心人对S3C2410的PWM TIMER作了很详细的说明,内容和说明书上的相差不大,不过有简短的程序可以参考,简单的举例让自己更好的理解。到目前位置项目的驱动貌似都已经完备,捣腾驱动的这几天,对ARM有了新的理解,但是也发现诸多的问题。也明白这些问题在系统深入学习之前,弄明白很困难,暂且寄希望于以后实践中慢慢体会。

    下面是网上一牛人博客上的文章,觉得写得不错。

 *****************************************************

   s3c2410提供了516位的TimerTimer0~Timer4),其中Timer0~Timer3支持Pulse Width    Modulation—— PWM(脉宽调制 )。Timer4是一个内部定时器(internal timer),他没有输出引脚(output pins)。 
    
下面是Timer的工作原理图。

 

    如上图所示,PCLKTimer的信号源,我们通过设置每个Timer相应的PrescalerClock DividerPCLK转换成输入时钟信号传送给各个Timer的逻辑控制单元(Control Logic),事实上每个Timer都有一个称为输入时钟频率(Timer input clock Frequency)的参数,这个频率就是通过PCLKPrescalerClock Divider确定下来的,每个Timer 的逻辑控制单元就是以这个频率在工作。下面给出输入时钟频率的公式:

Timer input clock Frequency = PCLK / {prescaler value+1} / {clock divider }
{prescaler value} = 0~255
{ clock divider } = 2, 4, 8, 16

    然而并不是每一个Timer 都有对应的Prescaler Clock Divider ,从上面的原理图我们可以看到Timer0,Timer1共用一对PrescalerClock Divider,Timer2,Timer3,Timer4共用另一对PrescalerClock Dividers3c2410的整个时钟系统模块只存在两对PrescalerClock Divider 
    
我曾经在讨论watchdog的文章中提到,watchdog也是一种定时器,他的工作就是在一个单位时间内对一个给定的数值进行递减和比较的操作,而我们这篇文章讨论的定时器他的工作内容和watchdog在本质上是一样的。定时器在一个工作周期(Timer input clock cycle)内的具体工作内容主要有3个。分别是:

  1. 对一个数值进行递减操作

  2. 把递减后的数值和另一个数值进行比较操作

  3. 产生中断或执行DMA操作

     在启用Timer之前我们会对Timer进行一系列初始化操作,这些操作包括上面提到的设置PrescalerClock Divider,其中还有一个非常重要的就是要给Timer两个数值,我们分别称之为Counter(变量,用于递减)Comparer(定值,用于比较),Counter会被Timer 加载到COUNT BUFFER REGISTERTCNTB,Comparer会被Timer 加载到和COMPARE BUFFER REGISTERTCMPB),每个Timer都有这样两个寄存器。当我们设置完毕启动Timer之后,Timer在一个工作周期内所做的就是先把TCNTB中的数值(Counter)减1,之后把TCNTB中的数值和TCMPB中的数值(Comparer)进行对比,若Counter已经被递减到等于Comparer,发生计数超出,则Timer产生中断信号(或是执行DMA操作)并自动把Counter重新装入TCNTB(刷新TCNTB以重新进行递减)。以上就是Timer的工作原理。

下面我们结合代码具体说明如何对Timer0进行初始化并开启它。 
首先我假设我的PCLK50700000Hz

// define Timer register
#define rTCFG0 (*(volatile unsigned int *)0x51000000)
#define rTCFG1 (*(volatile unsigned int *)0x51000004)
#define rTCON (*(volatile unsigned int *)0x51000008)
#define rTCNTB0 (*(volatile unsigned int *)0x5100000C)
#define rTCMPB0 (*(volatile unsigned int *)0x51000010)
#define rTCNTO0 (*(volatile unsigned int *)0x51000014)
#define rTCNTB1 (*(volatile unsigned int *)0x51000018)
#define rTCMPB1 (*(volatile unsigned int *)0x5100001C)
#define rTCNTO1 (*(volatile unsigned int *)0x51000020)
#define rTCNTB2 (*(volatile unsigned int *)0x51000024)
#define rTCMPB2 (*(volatile unsigned int *)0x51000028)
#define rTCNTO2 (*(volatile unsigned int *)0x5100002C)
#define rTCNTB3 (*(volatile unsigned int *)0x51000030)
#define rTCMPB3 (*(volatile unsigned int *)0x51000034)
#define rTCNTO3 (*(volatile unsigned int *)0x51000038)
#define rTCNTB4 (*(volatile unsigned int *)0x5100003C)
#define rTCNTO4 (*(volatile unsigned int *)0x51000040)

/*************************************************

这里做个说明,作者自己用define定义了各个定时器寄存器的地址,这个在arm相关的头文件中已经定义好了

至于地址的值,根据需要查看说明书,上面有详细的说明,看说明书真的很重要

*************************************************/

void timer0_config()
{
/*
                Timer0
prescalerrTCFG0 的 0~7 bit决定 
                Prescaler=119
 
*/
                rTCFG0=119        
/*
                Timer0
divider valueTCFG1的 0~3 bit决定,设置为3表示divider value = 1/16
                rTCFG1
的第20~23bit用于决定Timer计数超出后所采取的响应,我们使用了中断模式(20~23bit全部为0), 
                
即计数超出后产生中断 
*/
                rTCFG1=3;
        
                rTCNTB0=26406;
                rTCMPB0=0;
}
由于我们的PCLK50700000Hz, 根据Timer input clock Frequency的计算公式我们如下计算Timer0的时钟输入频率: 

prescaler value = 119
divider value = 1/16
PCLK= 50700000
Timer input clock Frequency =50700000/ (119+1)/(1/16)=26406

    
也就是说通过设置prescalerdivider value之后,Timer0的工作频率为26406,也就是说一秒内Timer0会进行26406次递减和比较操作,假设我们现在是要让Timer01秒产生一次中断的话,我们应该设置Counter=26406Camparer=0,既: 

rTCNTB0=26406;
rTCMPB0=0;

如果我们要让Timer00.5秒产生一次中断,则我们应该设置Counter=26406/2Camparer=0,既: 

rTCNTB0=13203;
rTCMPB0=0;

如果我们要让Timer00.25秒产生一次中断,则我们应该设置Counter=26406/4Camparer=0,既: 

rTCNTB0=6601;
rTCMPB0=0;

初始化完Timer后我们要开启它。 

void timer0_start()
{
/*
               Update TCNTB0 & TCMPB0
               rTCON
寄存器的第1位是刷新Timer0COUNT BUFFER REGISTERTCNTB)和 
                COMPARE BUFFER REGISTER
TCMPB),由于是第一次加载CounterComparer 
                
所以我们需要手动刷新它们 
*/
               rTCON|=1<<1;
/*
               
rTCON0位为1,开启Timer0
               
rTCON1位置为0,停止刷新TCNTB0 和 TCMPB0
               
rTCON3位为1,设置Counter的加载模式为自动加载(auto reload),这样每当 
               Timer
计数超出之后(此时TCNTB的值等于TCMPB的值),Timer会自动把原来我们给 
               
定的Counter重新加载到TCNTB 
*/
        rTCON=0x09;        
}

    
要使你的Timer能够正常的工作,除了调用timer0_config()timer0_start()之外,我们还必须设置Timer的中断服务例程并取消对Timer的中断的屏蔽.这些操作可以参考<<s3c2410 中断异常处理>>一文.

********************************************************************

以上便是TIMER驱动的初始化,下面就是写中断了,在写中断的时候要注册一个中断号,顺便也想起我们的驱动文件在mknod的时候的需要一个设备号,申请方式有两种(静态/动态);静态的就是直接指定,如果个号没有还没有使用,那么就注册成功,但是如果已经使用,那么就会失败。查询是个麻烦的过程,所以采用动态申请是一个比较好的选择,下面分享一段代码,个人觉得比较实用:

ret=request_irq(IRQ_TIMER0,&timer_irq,SA_INTERRUPT,DEVICE_NAME,(void*)&timer_flag);                          //申请中断号

if(ret){

      printk("IRQ_TIMER0=%d is already in use!/n", IRQ_TIMER0);

      return ret;

}

dev_t dev = MKDEV(major, 0);      //将主设备号和次设备号定义到一个dev_t 数据类型的结构体之中

if (major)

     result = register_chrdev_region(dev, 1, "timer");

//静态注册一个设备,设备号先前指定好,并得到一个设备名,cat /proc/device来查看信息

else{

     result = alloc_chrdev_region(&dev, 0, 1, "timer");

//如果主设备号被占用,则由系统提供一个主设备号给设备驱动程

    major = MAJOR(dev);

//得到主设备号

}

if (result < 0){

    return result;

}

if (major == 0)

     major = result;    //如果静态分配失败。把动态非配的设备号给设备驱动程序

其中像major和result等变量是程序中定义的,major表示主设备号,result就是函数返回结果,表示注册结果。

看说明书时的小摘抄,因为翻页太麻烦了,现在看看还是挺好的