8051系列单片机软件精确延时研究(二)

发布者:Blissful567最新更新时间:2024-07-23 来源: cnblogs关键字:8051系列  单片机  软件精确延时 手机看文章 扫描二维码
随时随地手机看文章

  由前篇可知,在DelayX10us()函数中用for循环延时会产生10个机器周期的固定误差,其中X传值、调用函数、子函数返回共5个机器周期,这是只要调用带参数子函数都有的、固定不变的;for循环判断x>0并跳转产生额外的5个机器周期的误差。


改进

  根据《在单片机KeilC开发环境中设计精确的延时函数》中提到的内容,可将for循环改为while(--x),以消除for循环产生的额外5个机器周期的误差。


  注意:应使用while(--x),这样对应生成的汇编语句才是DJNZ。如果使用while(x--),将额外产生几个指令,导致此延时函数不准。


  更改后的程序如下:


//非精确延时10*X us,固定误差5us

//@12.000MHz 12T

void DelayX10us(unsigned char x)    

{

    unsigned char i;

    do

    {

        _nop_();

        i=3;

        while(--i);

    } while (--x);

}


反汇编分析


如前,采用level8的优化等级,反汇编后的代码如下:

  

计算一下延时时间:

x固定延时循环延时总计
15(1+1+2*3+2)*115
105(1+1+2*3+2)*10105
1005(1+1+2*3+2)*1001005







  

可见,误差被缩减到5us了。


官方毫秒级延时分析

  对于上述改进后的DelayX10us函数,X最大值为255,所以其最大延时为255*10=2550us=2.55ms。如果要获得更长的延时怎么办呢?在这里需注意,不能单纯将X改为unsigned int以获得更长延时,因为8051是8位单片机,对于16位的int类型,需要分成高8位、低8位运算,在'while(--x);'这句将不只需要2个机器周期。所以我们重新定义一个毫秒级的延时函数。STC官方的延时1ms程序如下:


//@12.000MHz, STC官方版本

void Delay1ms()  //此处没有赋值那个机器周期

{

    unsigned char i, j;


    i = 2;

    j = 239;    //请注意j赋值的位置

    do

    {

        while (--j);

    } while (--i);

}


  按之前的办法计算延时周期数t=函数调用LCALL+i j赋值+循环+返回RET= 2 + 1+1+(239*2+2)*2 +2= 966[在此感谢群友Smiles指出漏加的2个赋值周期]。与预期的1000个机器周期相差较大,为什么呢?


  此处需注意变量j赋值的位置是在循环外,当外层循环执行到第二次(i=1)时,j不会赋值为239,故不能用239*2来计算。这里利用了一个溢出的小技巧。简单分析过程如下:


  i=2,j=239赋值(2个机器周期)后,进入内层循环执行'while(--j);'共239次,j=0,跳出。由之前的分析可知,'while(--j);'的汇编代码为DJNZ指令,为2周期指令,周期数t1=2+239*2。


  然后执行到外层循环'while(--i);',2个机器周期,i=1,DJNZ指令跳转到内层循环继续执行,周期数t2=2。


  再次进入内层循环后,j=0,DJNZ命令为先执行寄存器减1,再判断是否为0,所以执行--j后,j溢出为0xFF(十进制255),不等于0,语句'while(--j);'继续执行,直到j再次自减到0,跳出。周期数t3=256*2。


  再次到外层循环,'while(--i);',i=0,跳转到函数末尾准备返回。周期数t4=2。


  函数调用和返回周期数t5=2+2。此处调用函数没有参数传递,所以没有赋值那1个周期。


  总的周期数t=t1+t2+t3+t4+t5=2+239*2+2+256*2+2+4=1000。正好!


毫秒级延时函数改造

  现在我们将官方只能延时1ms函数改造为可以延时多个ms,按之前的方法,给延时代码套一个do..while循环。


  do


  {


     i = 2; j = 239;


     do { while (--j); }


    while (--i); }


   while (--x);


此时延时机器周期数=1+2+(2+239*2+2+256*2+2+2)*X+2=998*X+5,倍增量不是1000,存在误差。x=1时,t=1003;x=10时,t=9985;x=100时,t=99805,已有接近200us的误差了,所以我们将括号内的机器周期改为1000,只需改j=240,即可使总周期数T=1+2+(2+240*2+2+256*2+2+2)*X+2=1000*X+5,固定误差5us,更改后的代码如下:


/**

 * 晶振12MHz,12T模式下延时1*x ms,固定误差5us。

 */

void DelayX1ms(unsigned char x)

{

    unsigned char i, j;

    do

    {

        i = 2;

        j = 240;

        do

        {

            while (--j);

        } while (--i);

    } while (--x);

}


仿照这个模式即可写出任意固定误差5us的延时程序了。


另外,对于几微秒的延时,就建议采用_nop_()延时了。同时考虑移植性,不建议在程序中直接写多个_nop_()来延时。将所有延时函数写在Delay.h和Delay.c文件中,其余程序通通调用这个库,以后要更改,比如换了STC的1T单片机,延时需要修改,也只需要改这两个文件就可以了。


最后,附上我在用的延时函数库,所有X倍延时的固定误差=5us。


#ifndef __DELAY_H__

#define __DELAY_H__


typedef unsigned char UINT8


//定义默认设置:晶振12MHz,模式12T

#define FSOC_12M_MOD_12T

/**

 * 晶振12MHz,12T模式下的延时。

 */

#ifdef FSOC_12M_MOD_12T

#define NOP()        _nop_()

#define Delay1us()    NOP()

#define Delay2us()    NOP();NOP()

#define Delay5us()    NOP();NOP();NOP();NOP();NOP()

#endif


void DelayX10us(UINT8 X);

void DelayX1ms(UINT8 X);

void DelayX10ms(UINT8 X);

void DelayX1s(UINT8 X);


#endif


/**

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

 ******    Copyright(C), 2010-2016, NULL Co.,Ltd     ******

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


 *@Tittle        :    通用延时函数

 *@Version        :    v1.1

 *@Author        :    Liy

 *@Dat            :    2016-08-10 15:03:15

 *@Desctription    :    延时函数库

 *@History        :

 *    #v1.1     2016-08-10 15:03:41

 *        1. 删除固定延时函数,仅保留X倍延时;

 *        2. 优化X倍延时函数的时间,所有X倍延时函数误差控制为5us。

 *    #v1.0    2016-08-03 16:44:18

 *        1. 完成12MHz、12T模式下常用延时函数


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

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

 */



#include 'Delay.h'


/**

 * 晶振12MHz,12T模式下延时

 */

#ifdef FSOC_12M_MOD_12T

/**

 * 晶振12MHz,12T模式下延时10*X us,固定误差5us。

 * X最大值255

 */

void DelayX10us(UINT8 X)

{

    UINT8 i;

    do

    {

        _nop_();

        i = 3;

        while (--i);

    } while (--X);

}


/**

 * 晶振12MHz,12T模式下延时1*X ms,固定误差5us。

 * X最大值255

 */

void DelayX1ms(UINT8 X)

{

    UINT8 i, j;

    do

    {

        i = 2;

        j = 240;

        do

        {

            while (--j);

        } while (--i);

    } while (--X);

}


/**

 * 晶振12MHz,12T模式下延时10*X ms,固定误差5us。

 * 周期数N=((2+(114*2+2)+(256*2+2)*19+2))*X+5

 * X最大值255

 */

void DelayX10ms(UINT8 X)

{

    UINT8 i, j;

    do

    {

        i = 20;

        j = 114;

        do

        {

            while (--j);

        } while (--i);

    } while (--X);

}


/**

 * 晶振12MHz,12T模式下延时10*X s,固定误差5us。

 * 周期数N=(1+3+(((123*2+2)*1+(256*2+2)*153) +2 ) + ((256*2+2)*256+2)*7+2)*X+5

 * X最大值255

 */

void DelayX1s(UINT8 X)

{

    UINT8 i, j, k;


    do

    {

        NOP();

        i = 8;

        j = 154;

        k = 123;

        do

        {

            do

            {

                while (--k);

            } while (--j);

        } while (--i);

    } while (--X);

}


#endif


好了,简单了解了下汇编,软件延时也基本精确了,结束。


关键字:8051系列  单片机  软件精确延时 引用地址:8051系列单片机软件精确延时研究(二)

上一篇:基于STC12系列单片机的通用红外遥控信号分析程序(一)
下一篇:8051系列单片机软件精确延时研究(一)

推荐阅读最新更新时间:2026-03-25 12:48

8051系列单片机软件精确延时研究(一)
前言   最近自学STC公司的8051系列单片机,编程中如流水灯等非精确延时多用软件延时实现,写了几个类似DelayX10us(unsigned char x)的函数方便调用,函数内部的语句多是用STC官方延时程序再自己套一个for或者do..while循环改造而成,像这样: //非精确延时10*Xus //@12.000MHz 12T模式 void DelayX10us(unsigned char x) { unsigned char i; for (; x 0; x--) { _nop_(); i = 2; while (--i); } }   由于不懂汇编,所以对代码
[单片机]
<font color='red'>8051</font><font color='red'>系列</font><font color='red'>单片机</font><font color='red'>软件</font><font color='red'>精确</font><font color='red'>延时</font>研究(一)
通过单片机软件实现精确延时
  在很多情况下,定时器/计数器经常被用作其他用途或者无法实现时,只能用软件方法延时,比如,延时超过定时器定时范围的最大值。下面我们来分析一下通过软件实现的延时。      1.较短时间的精确延时      在汇编语言里,我们可以用NOP(空操作指令)来实现,一条NOP指令占用一个机器周期,比如我们用12MHz晶振,执行一条NOP指令,它就可以延时lμs,精确度可以达到1μ。使用也比较方便,我们可以在需要延时的程序中直接加上NOP指令即可实现延时。      同样,我们可以在C51中通过使用带一NOP-()语句的函数实现,_NOP-()就相当于汇编中的NOP指令,我们可以定义一系列不同的延时函数,比如Delayl0μs   D
[单片机]
通过<font color='red'>单片机</font><font color='red'>软件</font>实现<font color='red'>精确</font><font color='red'>延时</font>
利用Keil u3调试,精确实现软件延时
用定时器延时,有时候显得有点麻烦,我们不如考虑软件精确延时,软件延时无非就是利用for或while多重循环。以前用到延时函数时,都是从网上下载别人写好的延时子程序。延时5ms,400ms,1s, ,这些延时函数的函数名中都清清楚楚地标明了延时的时间,可我一直不知道这些函数是如何编写的,确切地说,是如果根据延时时间来确定循环次数的。如果是纳秒级的延时,可以通过示波器来观察波形,或者反汇编一下,计算一下指令执行时间,但如果延时时间相对较长,示波器便无能为力了。这几天好好看了一下Keil调试,发现Keil的功能实在是太强大了。利用Keil uVersion的调试就可以写出精确的软件延时程序。以下是我的简单小结,文中所有程序都是在Xtal=
[单片机]
基于C8051F2xx系列MCU芯片实现锅炉水处理控制装置的设计
自然水中通常含有钙镁等离子,俗称硬水。在锅炉用水中需要去除水中的钙镁离子而形成软水以防止锅炉结垢。在生产中锅炉水的软化处理是一项重要的安全指标,所以,对于锅炉水处理的技术要求愈来愈高。单片机以其较高的灵活性和稳定性广泛应用在自动控制领域。本文所设计的锅炉水处理控制装置,由高低水位控制进水阀开关,选用单片机为核心,C语言编程实现循环时间电路控制。该装置已成功应用于成都富华水处理公司。 1 软水生产工艺过程及对自控系统的要求 1.1 生产工艺过程 软化水设备的工作原理是基于阳离子交换原理。水由交换柱上流下,与交换树脂中的盐离子充分接触达到把原水中的杂质、易结垢的重金属阳离子去除掉。其生产工艺大致分为下列几步:①条件满足后运行;②松
[单片机]
基于C<font color='red'>8051</font>F2xx<font color='red'>系列</font><font color='red'>MCU</font>芯片实现锅炉水处理控制装置的设计
基于8051增强型单片机的RJM8L系列超低功耗MCU介绍
对于一些采用电池供电的产品需要长达数年不换电池情况下能连续工作,系统低功耗设计就尤为重要。MCU微控制器的低功耗设计决定系统的成败关键。MCU微控制器的低功耗技术涉及到软件、系统和底层的硬件工艺等。对于应用来说,在空闲的时候,可以将其时钟关闭以节省动态功耗,或小部分电路以低速低功耗的方式运行,SRAM的读写动态功耗相当可观,因此应该尽量减少读写SRAM。 瑞纳捷针对低功耗应用推出了RJM8L151S和RJM8L003系列产品,已大量应用到各领域,如:烟雾报警器,LoRa模组,智能门锁,灯控设备,GPS定位器,安防探测器,电子烟等应用领域。 RJM8L151S和RJM8L003系列产品是基于8051增强型单片机,工作电压2.
[单片机]
基于<font color='red'>8051</font>增强型<font color='red'>单片机</font>的RJM8L<font color='red'>系列</font>超低功耗<font color='red'>MCU</font>介绍
8051内核锦锐MCU开发指南:CA51F3系列
一、锦锐MCU简介 公司官网:深圳市锦锐科技有限公司 http://www.cachip.com.cn 主要产品:   8 bit Flash单片机系列    * CA51F0系列(AD + PWM类型)    * CA51F2系列(LCD + 触摸类型)    * CA51F3系列(AD + 触摸类型)    * CA51F4系列(LCD驱动升压类型)    * CA51F5系列(PWM + 触摸类型)   收音机系列    * 手调收音芯片    * 手调数显立体声收音芯片    * PLL电调收音芯片    * PLL电调收音RDS芯片    * WB灾难预警收音芯片   DAB接收模组系列    * C912 DAB接收模
[单片机]
<font color='red'>8051</font>内核锦锐<font color='red'>MCU</font>开发指南:CA51F3<font color='red'>系列</font>
HOLTEK推出HT85F2280/70/60 8051 A/D Flash Type MCU系列
Holtek推出全新的8051 A/D Flash Type MCU的HT85F2280、HT85F2270、HT85F2260系列,全系列宽工作电压范围2.2V~5.5V,符合工业等级-40℃ ~ 85℃工作温度与高抗噪声之性能要求,是一系列混合信号高性能MCU,使用1T Pipeline架构8051 CPU,做为高速数据处理引擎,内建高速12-bit ADC及可程序增益放大器(PGA),为嵌入式系统提供一个SOC应用平台。 HT85F2280、HT85F2270、HT85F2260 8051 A/D Flash Type MCU系列Program Memory为16Kx8 ~ 64Kx8、SRAM由1280 ~ 2304
[单片机]
基于C8051F系列单片机的标签打印机接口设计
摘要:为满足与日俱增的嵌入式系统的打印需求,设计了一种以片上系统型C8051F系列单片机为主机控制器的标签打印机接口模块。详细阐述了单片机与打印机之间的串行接口设计方法和斑马公司标签打印机专用的EPL2打印描述语言的使用,在掌握EPL2语言的基础上,利用KeilC编程工具进行驱动程序的编写,并给出了相应的驱动程序,实现了C8051F020单片机通过RS 232接口对一种标签打印机的控制。 关键词:C8051F020;标签打印机;串行接口;EPL2 0 引言 随着信息化技术的高速发展,各行业对于数据打印的需求日益增加。为了能够更加直观的浏览数据,微型标签打印机在智能仪器仪表、电子收款机、计价器等系统中几乎成为标准配置。本课题来
[工业控制]
基于C<font color='red'>8051</font>F<font color='red'>系列</font><font color='red'>单片机</font>的标签打印机接口设计
小广播
最新单片机文章
何立民专栏 单片机及嵌入式宝典

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

厂商技术中心

 
EEWorld订阅号

 
EEWorld服务号

 
汽车开发圈

 
机器人开发圈

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