一起学mini2440裸机开发(五)--定时器0的基础实验

时间:2022-04-20 23:31:07

     本篇文章接上一篇关于定时器原理:http://blog.csdn.net/mybelief321/article/details/8916775

实验前的准备 

      既然是关于定时器的实验,肯定要用到系统时钟,所以一定要保证系统时钟设置好,在这里需要的PCLK为50MHz。第二节分析MDK自带的S3C2440.c可以知道,默认的是不初始化系统时钟(是否选择初始化可以通过修改S3C2440.s中的CLOCK_SETUP来选择)。那么在这里首先修改一下S3C2440.s,对时钟进行初始化。

     在这里只需要设置一处即可将 CLOCK_SETUP     EQU     0修改为CLOCK_SETUP     EQU     1,这样系统在启动时就会对咱们的芯片进行初始化,初始化之后的系统时钟值为:FCLK=300MHz,HCLK=100MHz,PCLK=50MHz

实验1

     实验实现的功能:使用定时器0的定时功能,是LED每秒钟闪烁一次。   

    因为在启动代码阶段,已经对系统时钟进行初始化,PCLK=50MHz,定时器的输入频率由PCLK分频得到,如图1所示展示了从晶振输入频率到定时器工作频率产生的整体过程。

    一起学mini2440裸机开发(五)--定时器0的基础实验

      下图是我的定时器工程的文件布局图:

      一起学mini2440裸机开发(五)--定时器0的基础实验

    从上图可以看出,这个工程需要咱们自己编写的文件一共有5个:main.c  timer.c   timer.h    led.c   led.h

    实验代码:

    main.c文件内容:

/*
*  使用定时器0的定时功能,使LED灯每秒钟闪烁一次
*/

#include<s3c2440.h>
#include"timer.h"
#include"led.h"

int main()
{   
     int flag=0;
     Led_Init();   //对LED初始化为全灭
    Timer0_Init();  //定时器0初始化,打开定时器0,定时器0开始进行减1计数。
     while(1)
    {
        if(SRCPND&(1<<10))     //当TCNT0中的计数值减为0时,定时器0中断标志会置位
        {                                     //因此,在程序中可以通过不断的检测该位是否置位来判断1s定时
             flag=!flag;                     //是否到达。定时器0中断标志位位于SRCPND寄存器的第10位
             SRCPND|=(1<<10);     //清除定时器0中断标志位
         }
         if(1==flag)                      //判断falg是否为1,这里使用一个小技巧,使用if(1==flag)。也可以

                                              //使用if(flag==1)。
        {
            Led1_On();
        }  
        else
       {
           Led1_Off();
        }
       }
}

 

    timer.c文件内容

 

            #include<s3c2440.h>   //s3c2440.h对S3C2440芯片的一些地址的宏定义
            #include"timer.h"

 /***************************************************************
* 函数名称:void Timer0_Init(void)
* 参数说明:无
* 全局变量:无
* 返 回 值:无
* 功    能:对于50MHz的PCLK,经过分频获得62.5KHz的定时器0
*     的输入时钟。
***************************************************************/

void Timer0_Init(void)
{
 TCFG0&=~(0xff);          //设置第1级分频系数,分频系数为99
 TCFG0|=99;

 TCFG1&=~(0xf);            //设置第2级分频系数,分频系数为8
 TCFG1|=0x02;               //62.5KHz=50MHz/(99+1)/8

 TCNTB0=62500;            //1s中断一次。经过上述分频器得到定时器0的输入时钟频率为62.5kHz,即定时
                                       //器每秒钟计数62500次。因此,初始化时,定时器0计数值初始值为62500,在这
                                       //里我们可以看出TCMPBn没有设置,因为咱们用它的默认值0,所以就不需要设置

 TCON|=(1<<1);             //开启手动更新位,即当定时器开启后,TCMPB0和TCNTB0中的值会加载到寄存

                                      //器TCMP0和TCNT0中
 TCON=0x09;                //关闭手动更新位,设置自动加载位,同时开启定时器,这样,TCNT0进行减1计

                                     //数,当TCNT0中的计数值减到0时,TCNTB0、TCMPB0中的数据分别会
                                     //自动加载到TCNT0、TCMP0中并进行新一轮的减1计数
}

   timer.h文件内容:

      

#ifndef __TIMER_H__
#define __TIMER_H__

/***************************************************************
* 函数名称:void Timer0_Init(void)
* 参数说明:无
* 全局变量:无
* 返 回 值:无
* 功    能:对于50MHz的PCLK,经过分频获得62.5KHz的定时器0
*     的输入时钟。
***************************************************************/

void Timer0_Init(void);

#endif

   led.c文件内容:

/*
*  在我的mini2440开发板上,4个led灯对应的GPIO引脚如下
*   LED1----GPB5
*   LED2----GPB6
* LED3----GPB7
*   LED4----GPB8
*   当GPIO引脚输出低电平时,对应的led灯亮,输出高电平时,对应的led灯灭。
*/

#include<s3c2440.h> 
#include"led.h"

/***************************************************************
* 函数名称:void Led_Init(void)
* 参数说明:无
* 全局变量:无
* 返 回 值:无
* 功    能:led初始化函数,使4个led初始化为灭
*     的输入时钟。
***************************************************************/

void Led_Init(void)
{
 GPBCON&=~((3<<10)|(3<<12)|(3<<14)|(3<<16));  //注意这里设置寄存器的方式,采用先与后或的方式
 GPBCON|=((1<<10)|(1<<12)|(1<<14)|(1<<16));    //设置GPB5、6、7、8为输出功能

 GPBUP&=~((1<<5)|(1<<6)|(1<<7)|(1<<8));      //上拉电阻使能。上拉电阻的作用是:当GPIO引脚处于

                                                                          //第三态(既不是输出高电平,也不是输出低电平,而是呈

                                                                         //高阻态,即相当于没接芯片)时,它的电平状态由上拉
                                                                           //电阻、下拉电阻确定

 GPBDAT|=((1<<5)|(1<<6)|(1<<7)|(1<<8));      //初始化这4个引脚输出高电平,即4个led灯全灭。
}

    led.h文件内容:

#ifndef __LED_H__
#define __LED_H__

#define Led1_On() {GPBDAT&=(~(1<<5));}       //在程序调用函数时,调用函数需要保存函数的返回地址,
#define Led1_Off() {GPBDAT|=(1<<5);}           //然后从函数返回是需要将返回地址赋值给PC,这些都会
#define Led2_On() {GPBDAT&=(~(1<<6));}    //使程序执行速度变慢。为了改善这种情况,对于这种
#define Led2_Off() {GPBDAT|=(1<<6);}         //代码量很小的程序段,可以使用宏的形式实现。
#define Led3_On() {GPBDAT&=(~(1<<7));}
#define Led3_Off() {GPBDAT|=(1<<7);}
#define Led4_On() {GPBDAT&=(~(1<<8));}
#define Led4_Off() {GPBDAT|=(1<<8);}
/***************************************************************
* 函数名称:void Led_Init(void)
* 参数说明:无
* 全局变量:无
* 返 回 值:无
* 功    能:led初始化函数,使4个led初始化为灭
*     的输入时钟。
***************************************************************/

void Led_Init(void);

#endif

 

      以上实验1的代码我已经上传到csdn,如有需要可自行下载:http://download.csdn.net/detail/mybelief321/5371577,mske编译成功后,就可以仿真看到结果了。

     上述实验1的已经说完了,再强调一下,我没有再编写代码另外设置系统时钟,而是用MDK自带的S3C2440.s来初始化的,只需要修改一处地方,本文章开头所示。

      还有一处需要说明一下,那就是关于main函数里的中断。虽然定时器0中断标志还没有讲,但是我们可以先了解以下三点:

      ①SRCPND寄存器中的每一位代表一种类型的中断标志,当该位置1时,说明相应的中断发生了。

      ②定时器0中断标志位于SRCPND寄存器的第10位,当定时器0中的计数值减到0时,会触发定时器0中断标志,即SRCPND寄存器第10位会置1.

     ③清除定时器0中断标志的方法是:想SRCPND寄存器的第10位写入1即可。

 

实验2

      实验1主要是针对定时器的原理进行的实验,下面的实验是为了展示定时器0的脉冲宽度调制功能。

    从下图可以看出,当TCMP0=TCNT0时,TOUT0引脚电平会发生翻转;当TCNT0中计数值减为0时,TOUT0引脚电平再次发生翻转。因此,可以利用TOUT0引脚电平的两次翻转进行脉冲宽度调制,即PWM。查询S3C2440数据手册可以得到,TOUT0引脚对应的是GPB0引脚。

      一起学mini2440裸机开发(五)--定时器0的基础实验

       在实验1的基础上,修改timer.c中定时器0初始化函数。修改后的定时器0初始化函数如下:    

void  Timer0_Init(void)
{
 GPBCON&=~(3<<0);
 GPBCON|=(1<<1);

 TCFG0&=~(0xff);    //设置第1级分频系数,分频系数为99
 TCFG0|=99;

 TCFG1&=~(0xf);      //设置第2级分频系数,分频系数为8
 TCFG1|=0x02;        //62.5KHz=50MHz/(99+1)/8
                                //此时定时器0的工作频率为62500Hz

 TCNTB0=62;                  //由于此时定时器0的工作频率为62500Hz,即每秒种可以计数62500次,而其初始值为62,所以产生的方波
                                       //的频率为62500/62=1008Hz。又因为TCMP0=TCNT0/2,所以产生的方波高电平持续时间和低电平持续时间
 TCMPB0=TCNTB0/2;   //各占一半

 TCON|=(1<<1); //开启手动更新位,即当定时器开启后,TCMPB0和TCNTB0中的值会加载到寄存器TCMP0和TCNT0中
 TCON=0x0d;     //关闭手动更新位,设置自动加载位,同时开启定时器,设置当TCMP0=TCNT0时,TOUT0引脚电平发生翻转
}

      程序总体流程:

       ①开启定时器0后,TCNTB0、TCMPB0中的值分别装入TCNT0和TCMP0中,然后,TCNT0从初值62开始减1计数;

       ②当TCNT0=TCMP0=31时,TOUT0引脚电平发生翻转;

       ③TCNT0继续减1计数,当TCNT0减为0时,TOUT0引脚电平再次发生翻转,此时恰好产生一个方波;

       ④TCNTB0、TCMPB0中的值被自动装入TCNT0、TCMP0中,TCNT0继续从初值62开始减1计数。

 

     上述程序,占空比为50%,如果将 TCMPB0=TCNTB0/2 改为 TCMPB0=TCNTB0/4,那么输出方波的占空比发生了变化,变为75%,这就是所谓的PWM功能。

     由于咱们的开发板的GPB0连接着蜂鸣器,所以你可以通过调节占空比来改变蜂鸣器的频率,将上述初始化函数修改后,直接make,仿真就可以了。