历史上的今天

今天是:2024年11月04日(星期一)

正在发生

2021年11月04日 | 一起学mini2440裸机开发(五)--定时器0的基础实验

发布者:梦幻之光 来源: eefocus关键字:mini2440  裸机开发  定时器0 手机看文章 扫描二维码
随时随地手机看文章

实验前的准备 


既然是关于定时器的实验,肯定要用到系统时钟,所以一定要保证系统时钟设置好,在这里需要的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所示展示了从晶振输入频率到定时器工作频率产生的整体过程。

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

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


实验代码:


main.c文件内容:


/*

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

*/


#include

#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芯片的一些地址的宏定义

            #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 

#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引脚。

在实验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,仿真就可以了。

关键字:mini2440  裸机开发  定时器0 引用地址:一起学mini2440裸机开发(五)--定时器0的基础实验

上一篇:2440+dm9000A裸机以太网通讯
下一篇:mini2440也即S3C2440地址分配问题

推荐阅读

        集微网专访(文/小如)2016年,一场史无前例的技术营销将AI产业推到了世人面前。时至今日,那场来自围棋盘上的机器人围剿依旧让人们津津乐道。自2016年起,AI便成了妇孺皆知的名词,加之资本、政策的青睐与热捧,AI进入到狂热阶段,随之而来的是成批的AI初创企业。        那么,处在目光焦点下的AI公司,有多少是虚火,又有多少...
来自美国哥伦比亚大学创意机器实验室的 Hod Lipson,研发出自我建模的仿生群体 “粒子机器人”,这种机器人未来将可能拥有创造力和自主思考能力。“一个真正的粒子机器人可能会由成千上百个粒子组成,即便其中 20% 的粒子坏掉,也仍然可以继续运作。”Hod Lipson 强调,“我们认为它带来的好处将会远远超过它的风险。”11 月 2 日,在腾讯 WE...
Zigbee 是什么?如果您的家里面有使用一款智能家居技术,那么这款技术就很有可能就是用的 zigbee 协议。Zigbee 是一种无线通信协议,是基于 IEEE820.15.4 标准的低功耗局域网协议。是根据国际标准规定的,zigbee 技术是一种短距离、低功耗的无线通信技术,该技术支持恒温器到冰箱灯泡等各种智能家居设备。Zigbee 是由 zigbee 联盟设定的 ,该联...
对于近期的磷酸铁锂电池回潮,和后面磷酸锰铁锂的升级方案,还是需要有一个清醒的认识——在高镍在前端工艺调整和不断进步以后,围绕安全和能量密度的平衡可能会有一个转折。 从安全角度来看,复合集流体是近年来安全技术的一个重要突破,宁德时代的布局可以说是国内的一个风向标。 相对而言,国外的电池标的,也会在各种屏障下有一定的发展。 按照目前的...

史海拾趣

问答坊 | AI 解惑

数码管时钟

给大家共享下咯…

查看全部问答∨

用过研华PCL816卡的看过来

第一次调理板卡,遇到点问题,急寻解决方法,谢谢! …

查看全部问答∨

单片机c51频率计

有没有c51频率计的代码呀,呵呵最好是用中断2的。谢谢啦…

查看全部问答∨

TTL转232,232转USB,遇到点困惑

我的连线方式是:板子+TTL转RS232+RS232转USB+PC。 “+”代表连接,这样转换会有什么问题么,为什么收不到数据呢。 直接将TTL转232接到PC机上就能收到,并且RS232转USB的线肯定是好的,希望懂得朋友指点迷津,谢谢了…

查看全部问答∨

wince5.0安装问题??急!!!

为何总提示要在.NET Frame 1.1下安装,可我的系统(winxp)已经装了 microsoft visual studio 2005 和 microsoft .net Framework SDK V2.0   ???…

查看全部问答∨

VS2005环境下EDB数据库编程的问题

请问下 用到EDB数据库编程时 在“VS2005|项目|属性|属性配置|C/C++|预处理器”是怎样设置的。 我的做法是在后面加了“EDB”,但出现好多错误: error C2065: \'CEPROPSPEC\' : undeclared identifier ………

查看全部问答∨

请问2440下的串口如何改变序号啊?

我用的是2440+ce5.0,是优龙的BSP,后来发现原来用的是UART1作为调试串口,而UART0用做功能串口。现在我把bootloader里面和BSP下面的相应的UART0和UART1的地址定义交换了,发现UART0是可以做调试串口了,但是UART1却无法收发数据。在platform.reg下 ...…

查看全部问答∨

新手请问2440一定要买JLINK吗

我想从头自学嵌入式,就在网上买了套MINI2440开发板,那个店主说调试要用到并口,我的电脑没有并口,一定要买那个JLINK吗,要180元,好贵。。。…

查看全部问答∨

关于nand ecc?

常见ecc 源码中: static const u_char nand_ecc_precalc_table[] =   {     0x00, 0x55, 0x56, 0x03, 0x59, 0x0c, 0x0f, 0x5a, 0x5a, 0x0f, 0x0c, 0x59, 0x03, 0x56, 0x55, 0x00,     0x65, 0x30, 0x33, 0x66, 0x3c, 0x69, 0x6a, 0x ...…

查看全部问答∨

AD采样频率与EV的T1CMPR T1PR的关系

你们好,各位DSP高手。我现在在利用2812的AD功能进行模数转换,看见TI的示例程序中使用了EV定时器。程序如下: // Configure EVA // Assumes EVA Clock is already enabled in InitSysCtrl();   EvaRegs.T1CMPR = 0x0380;     ...…

查看全部问答∨
小广播
设计资源 培训 开发板 精华推荐

最新单片机文章
何立民专栏 单片机及嵌入式宝典

北京航空航天大学教授,20余年来致力于单片机与嵌入式系统推广工作。

 
EEWorld订阅号

 
EEWorld服务号

 
汽车开发圈

电子工程世界版权所有 京ICP证060456号 京ICP备10001474号-1 电信业务审批[2006]字第258号函 京公网安备 11010802033920号 Copyright © 2005-2024 EEWORLD.com.cn, Inc. All rights reserved