假如从keil里看了c言语的反汇编代码然后依据晶振和指令核算延时的时刻这样尽管十分的精确可是适当的费事并且简略搞错,我这儿介绍一个最简略的办法.能够验证你的延时函数
这儿用一个例程具体介绍一下。
进程参阅如下:
在编译器下树立一个新项目,也能够运用已有项目。此进程中需求留意,单片机晶振的挑选,由于for循环里指令的履行时刻和晶振有直接关系,本例中晶振运用11.0592M。
编写一段关于延时的函数,首要运用for循环,代码如下:
void delay_ms(unsigned int ms)
{
unsigned int i;
unsigned char j;
for(i=0;i
{
for(j=0;j<200;j++);
for(j=0;j<102;j++);
}
}
其间ms是输入参数,假如输入1,便是要求程序延时1ms。
j变量是调整程序运转的时刻参数。调整j的数值,使1次循环的时刻在1ms。
将此程序编译经过,然后运用软件仿真,调整时刻。
下面这个sec便是程序运转到现在的这一行所用的时刻。
两次时刻差便是延时函数运用的时刻,假如与1ms相差比较多,用户能够调整j参数的值,使延时时刻尽量挨近1ms。如增大j的值for(j=0;j<105;j++);
此办法得出延时函数,在晶振不同的情况下,延时时刻会禁绝。软件调试成果,这个程序的延时时刻为:1.01779ms,一般的单片机体系中都能够使用。
下面来说说汇编的传统核算办法:
指令周期、机器周期与时钟周期
指令周期:CPU履行一条指令所需求的时刻称为指令周期,它是以机器周期为单位的,指令不同,所需的机器周期也不同。
时钟周期:也称为振动周期,一个时钟周期 =晶振的倒数。
MCS-51单片机的一个机器周期=6个状况周期=12个时钟周期。
MCS-单片机的指令有单字节、双字节和三字节的,它们的指令周期不尽相同,一个单周期指令包括一个机器周期,即12个时钟周期,所以一条单周期指令被履行所占时刻为12*(1/12000000)=1us。
了解了上面这些咱们来看一个比如
;============延时1秒子程序========================
DELAY_1S: ;延时子程序,12M晶振延时1.002035秒
MOV R4,#10
L3: MOV R2 ,#200 ;1指令周期
L1: MOV R3 ,#249 ;1指令周期
L2: DJNZ R3 ,L2 ;2指令周期
DJNZ R2 ,L1 ;2指令周期
DJNZ R4 ,L3 ;2指令周期
RET ;2指令周期
;循环体延时时刻: [(249*2+1+2)*200+1+2]*10*12/12000000=1.002030s
;加上ACALL指令和第一条mov指令以及最终一条RET指令算出来整个函数的时刻为1.002035s
;================================================
一般选用的是11.0592MHZ的晶振:
[(249*2+1+2)*200+1+2]*10*12/11059200=1.08727213541666666…S
汇编延时子程序的延时核算问题
关于程序
DELAY: MOV R0,#00H
DELAY1: MOV R1,#0B3H
DJNZ R1,$
DJNZ R0,DELAY1
查指令表可知 MOV一个机器周期,DJNZ 指令需用两个机器周期,而一个机器周期时刻长度为12/11.0592MHz,所以该段程序履行时刻为:
((0B3×2+1+2)×256+1)×12÷11059200=100.2789mS
第一层:DJNZ R1,$:履行了B3H次,一次两个周期,所认为0B3×2;
第二层:MOV R1,#0B3H为一个周期,DJNZ R0,DELAY1为两个周期,这样循环一次便是0B3×2+1+2个周期;第二层的履行次数本来是255次,但由于赋首值为0,而DJNZ是先减1,再比较的,所以就应该是256次。
这样的话,整个循环履行完应该是(0B3×2+1+2)×256+1次。再加上开端赋值这一句,便是((0B3×2+1+2)×256+1)了
还阐明一下:
nop指令或许_nop_(); 函数占一个机器周期,在stc单片机的12T形式下一个机器周期是一个振动周期的12分频,假如你的晶振是12MHZ,那你的一个机器周期便是1微秒.一个nop指令的履行时刻也便是1US
当在6T形式(下载的时分可挑选形式)下12M晶振的时分,一个nop便是0.5US了.