[ZLG-MCU] 我曾经搜集的关于delay的**

[复制链接]
6885|12
 楼主| 坚持梦想 发表于 2008-4-3 17:26 | 显示全部楼层 |阅读模式
以下是delay&nbsp;函数:<br />&nbsp;&nbsp;void&nbsp;delay(unsigned&nbsp;int&nbsp;n)&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;//&nbsp;软件延迟函数<br />&nbsp;&nbsp;{<br />&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;for(;n&gt0;n--);&nbsp;&nbsp;&nbsp;&nbsp;<br />&nbsp;&nbsp;}<br />请问它的延时长度是多少,比如&nbsp;delay(200000),得到的延时是多长?<br />&nbsp;&nbsp;&nbsp;&nbsp;因为访问外设有时序问题,所以需要精确定时。<br />&nbsp;&nbsp;&nbsp;&nbsp;或者是否有其它更好的方法?<br />&nbsp;&nbsp;&nbsp;&nbsp;先谢谢了。。。<br />
zlgarm 发表于 2008-4-3 17:32 | 显示全部楼层

精确定时要用定时器。

精确定时要用定时器。<br /><br />(zlgarm_zsg)
 楼主| 坚持梦想 发表于 2008-4-3 21:42 | 显示全部楼层

其实也不需要很精确

其实大概满足就可以了,所以觉得用delay函数方便些。。
zlgmcuCHTZ 发表于 2008-4-4 21:03 | 显示全部楼层

那用软仿看下了

  
 楼主| 坚持梦想 发表于 2008-4-7 16:33 | 显示全部楼层

如何软仿

能否说得具体一些?比如:什么软件平台仿真比较好用,是否能够提示一下测试语句?谢谢!
 楼主| 坚持梦想 发表于 2008-4-7 16:47 | 显示全部楼层

能否通过这种方式计算延时长度

“n--”不是一个指令周期吗(同时也是一个机器周期,对吗?),这样是否能通过这种方式计算延时长度?<br />一个机器周期是不是与设置的时钟频率有关?要如何计算呢?
computer00 发表于 2008-4-7 17:01 | 显示全部楼层

不对。你要看过生成的汇编代码后才知道最终的指令怎样

你可以写个简单的测试程序,然后拿示波器测。例如延时后翻转IO。
jdwang 发表于 2008-4-7 18:59 | 显示全部楼层

用秒表测延时时间

用汇编程序编可以比较准确地算出延时时间。用C语言的话,如不需要太准可以编一个时间长一些的延时程序,用秒表掐一下就可以了。
eagle109 发表于 2008-4-7 23:27 | 显示全部楼层

用示波器测

用示波器测,注意IO口固有延时
zlgmcu 发表于 2008-4-8 08:40 | 显示全部楼层

看来关心这个的还比较热闹

我常用的延时办法是:<br /><br />void&nbsp;&nbsp;timeDelay(unsigned&nbsp;long&nbsp;&nbsp;ulValue)<br />{<br />&nbsp;&nbsp;&nbsp;&nbsp;do<br />&nbsp;&nbsp;&nbsp;&nbsp;{<br />&nbsp;&nbsp;&nbsp;&nbsp;}&nbsp;while&nbsp;(&nbsp;--ulValue&nbsp;&nbsp;!=&nbsp;&nbsp;0&nbsp;);<br />}<br /><br />或者简单写作<br /><br />void&nbsp;&nbsp;timeDelay(unsigned&nbsp;long&nbsp;&nbsp;ulValue)<br />{<br />&nbsp;&nbsp;&nbsp;&nbsp;while&nbsp;(&nbsp;--ulValue&nbsp;&nbsp;!=&nbsp;&nbsp;0&nbsp;);<br />}<br /><br />假设系统时钟为6MHz,则延迟20ms(近似)的写法是:<br /><br />timeDelay(20&nbsp;*&nbsp;(6000000UL&nbsp;/&nbsp;4000))<br /><br /><br />(zlgmcu_wdx)
兔巴哥 发表于 2008-4-8 10:17 | 显示全部楼层

我曾经搜集的关于delay的**

void&nbsp;delay2(unsigned&nbsp;char&nbsp;I)&nbsp;<br />{&nbsp;<br />while(--I);&nbsp;<br />}&nbsp;<br /><br />为最佳方法。&nbsp;<br /><br /><br />&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;分析:假设外挂12M(之后都是在这基础上讨论)&nbsp;<br /><br />&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;我编译了下,传了些参数,并看了汇编代码,观察记录了下面的数据:&nbsp;<br /><br />delay2(0):延时518us&nbsp;518-2*256=6&nbsp;<br />delay2(1):延时7us(原帖写“5us”是错的,^_^)&nbsp;<br />delay2(10):延时25us&nbsp;25-20=5&nbsp;<br />delay2(20):延时45us&nbsp;45-40=5&nbsp;<br />delay2(100):延时205us&nbsp;205-200=5&nbsp;<br />delay2(200):延时405us&nbsp;405-400=5&nbsp;<br /><br />&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;见上可得可调度为2us,而最大误差为6us。精度是很高了!&nbsp;<br /><br />&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;但这个程序的最大延时是为518us,显然不能满足实际需要,因为很多时候需要延迟比较长的时间。&nbsp;<br /><br />&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;那么,接下来讨论将t分配为两个字节,即uint型的时候,会出现什么情况。&nbsp;<br /><br />Void&nbsp;delay8(uint&nbsp;t)&nbsp;<br />{&nbsp;<br />while(--t);&nbsp;<br />}&nbsp;<br /><br />&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;我编译了下,传了些参数,并看了汇编代码,观察记录了下面的数据:&nbsp;<br /><br />delay8(0):延时524551us&nbsp;524551-8*65536=263&nbsp;<br />delay8(1):延时15us&nbsp;<br />delay8(10):延时85us&nbsp;85-80=5&nbsp;<br />delay8(100):延时806us&nbsp;806-800=6&nbsp;<br />delay8(1000):延时8009us&nbsp;8009-8000=9&nbsp;<br />delay8(10000):延时80045us&nbsp;80045-8000=45&nbsp;<br />delay8(65535):延时524542us&nbsp;524542-524280=262&nbsp;<br /><br />&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;如果把这个程序的可调度看为8us,那么最大误差为263us,但这个延时程序还是不能满足要求的,因为延时最大为524.551ms。&nbsp;<br /><br />&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;那么用ulong&nbsp;t呢?&nbsp;<br /><br />&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;一定很恐怖,不用看编译后的汇编代码了。。。&nbsp;<br /><br /><br />&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;那么如何得到比较小的可调度,可调范围大,并占用比较少得RAM呢?请看下面的程序:&nbsp;<br /><br />/*------------------------------------------------------------------&nbsp;<br />函数全称:50us&nbsp;延时&nbsp;<br />注意事项:基于1MIPS,AT89系列对应12M晶振,W77、W78系列对应3M晶振&nbsp;<br />例子提示:调用delay_50us(20),得到1ms延时&nbsp;<br />输&nbsp;入:&nbsp;<br />返&nbsp;回:无&nbsp;<br />------------------------------------------------------------------*/&nbsp;<br />void&nbsp;delay_50us(uint&nbsp;t)&nbsp;<br />{&nbsp;<br />uchar&nbsp;j;&nbsp;<br />for(;t&gt0;t--)&nbsp;<br />for(j=19;j&gt0;j--)&nbsp;<br />;&nbsp;<br />}&nbsp;<br /><br />我编译了下,传了些参数,并看了汇编代码,观察记录了下面的数据:&nbsp;<br />delay_50us(1):延时63us&nbsp;63-50=13&nbsp;<br />delay_50us(10):延时513us&nbsp;503-500=13&nbsp;<br />delay_50us(100):延时5013us&nbsp;5013-5000=13&nbsp;<br />delay_50us(1000):延时50022us&nbsp;50022-50000=22&nbsp;<br /><br />赫赫,延时50ms,误差仅仅22us,作为C语言已经是可以接受了。再说要求再精确的话,就算是<br />用汇编也得改用定时器了。&nbsp;<br /><br />/*------------------------------------------------------------------&nbsp;<br />函数全称:50ms&nbsp;延时&nbsp;<br />注意事项:基于1MIPS,AT89系列对应12M晶振,W77、W78系列对应3M晶振&nbsp;<br />例子提示:调用delay_50ms(20),得到1s延时&nbsp;<br />全局变量:无&nbsp;<br />返回:&nbsp;无&nbsp;<br />------------------------------------------------------------------*/&nbsp;<br />void&nbsp;delay_50ms(uint&nbsp;t)&nbsp;<br />{&nbsp;<br />uint&nbsp;j;&nbsp;<br />for(;t&gt0;t--)&nbsp;<br />for(j=6245;j&gt0;j--)&nbsp;<br />;&nbsp;<br />}&nbsp;<br />我编译了下,传了些参数,并看了汇编代码,观察记录了下面的数据:&nbsp;<br />delay_50ms(1):延时50&nbsp;010&nbsp;10us&nbsp;<br />delay_50ms(10):延时499&nbsp;983&nbsp;17us&nbsp;<br />delay_50ms(100):延时4&nbsp;999&nbsp;713&nbsp;287us&nbsp;<br />delay_50ms(1000):延时4&nbsp;997&nbsp;022&nbsp;2.978ms&nbsp;<br /><br />赫赫,延时50s,误差仅仅2.978ms,可以接受!&nbsp;<br /><br />上面程序没有采用long,也没采用3层以上的循环,而是将延时分拆为两个程序以提高精度。<br />应该是比较好的做法了。&nbsp;<br /><br />如果想要得到更高精度的延时,可以这么做:&nbsp;<br />void&nbsp;delay_50us(uint&nbsp;t)&nbsp;<br />{&nbsp;<br />uchar&nbsp;j;&nbsp;<br />if(t&gt255)&nbsp;<br />{&nbsp;<br />针对性给于延时补偿;&nbsp;<br />}&nbsp;<br />if(t&lt255)&nbsp;<br />{&nbsp;<br />针对性给于延时补偿;&nbsp;<br />}&nbsp;<br />for(;t&gt0;t--)&nbsp;<br />for(j=18;j&gt0;j--)&nbsp;//根据实际,将原来19改为18或者更小&nbsp;<br />;&nbsp;<br />}&nbsp;<br />void&nbsp;delay_50ms(uint&nbsp;t)&nbsp;<br />{&nbsp;<br />uint&nbsp;j;&nbsp;<br />if(t&gt…)&nbsp;<br />{&nbsp;<br />针对性给于延时补偿;&nbsp;<br />}&nbsp;<br />if(t&gt…)&nbsp;<br />{&nbsp;<br />针对性给于延时补偿;&nbsp;<br />}&nbsp;<br />if(t&gt…)&nbsp;<br />{&nbsp;<br />针对性给于延时补偿;&nbsp;<br />}&nbsp;<br />……&nbsp;<br />……&nbsp;<br />for(;t&gt0;t--)&nbsp;<br />for(j=6244;j&gt0;j--)&nbsp;//根据实际,将原来6245改为6244或者更小&nbsp;<br />;&nbsp;<br />}<br />1s延时程序<br />晶振是12M,对于MC-51系统的单片机来说,一个机器周期是1us。&nbsp;<br />…………………………&nbsp;<br />MOV&nbsp;R0,#100&nbsp;<br />LOOP1:&nbsp;MOV&nbsp;R1,#100&nbsp;<br />LOOP2:&nbsp;MOV&nbsp;R2,#48&nbsp;<br />DJNZ&nbsp;R2,$&nbsp;<br />NOP&nbsp;<br />DJNZ&nbsp;R1,LOOP2&nbsp;<br />DJNZ&nbsp;R0,LOOP1&nbsp;<br />…………………………&nbsp;<br />[(2*48+4)*100+3]*100+1=1000301(个机器周期)即1.000301秒定时。<br /><br /><br /><br />.(晶振12MHz,一个机器周期1us.)&nbsp;一.&nbsp;500ms延时子程序&nbsp;程序:&nbsp;void&nbsp;delay500ms(void)&nbsp;<br /><br />{&nbsp;unsigned&nbsp;char&nbsp;i,j,k;<br /><br />&nbsp;for(i=15;i&gt0;i--)&nbsp;for(j=202;j&gt0;<br /><br />j--)&nbsp;for(k=81;k&gt0;k--);&nbsp;}<br /><br />&nbsp;产生的汇编:&nbsp;C:0x0800&nbsp;7F0F&nbsp;MOV&nbsp;R7,#0x0F&nbsp;<br /><br />C:0x0802&nbsp;7ECA&nbsp;MOV&nbsp;R6,#0xCA&nbsp;<br /><br />C:0x0804&nbsp;7D51&nbsp;MOV&nbsp;R5,#0x51&nbsp;<br /><br />C:0x0806&nbsp;DDFE&nbsp;DJNZ&nbsp;R5,C:0806&nbsp;<br /><br />C:0x0808&nbsp;DEFA&nbsp;DJNZ&nbsp;R6,C:0804&nbsp;<br /><br />C:0x080A&nbsp;DFF6&nbsp;DJNZ&nbsp;R7,C:0802&nbsp;C:0x080C&nbsp;22&nbsp;RET&nbsp;<br /><br />计算分析:&nbsp;程序共有三层循环&nbsp;一层循环n:R5*2&nbsp;=&nbsp;81*2&nbsp;=&nbsp;162us&nbsp;DJNZ&nbsp;2us&nbsp;二层循<br />环m:R6*(n+3)&nbsp;=&nbsp;202*165&nbsp;=&nbsp;33330us&nbsp;DJNZ&nbsp;2us&nbsp;+&nbsp;R5赋值&nbsp;1us&nbsp;=&nbsp;3us&nbsp;三层循<br />环:&nbsp;R7*(m+3)&nbsp;=&nbsp;15*33333&nbsp;=&nbsp;499995us&nbsp;DJNZ&nbsp;2us&nbsp;+&nbsp;R6赋值&nbsp;1us&nbsp;=&nbsp;3us&nbsp;循环外:&nbsp;5us&nbsp;子程序调<br />用&nbsp;2us&nbsp;+&nbsp;子程序返回&nbsp;2us&nbsp;+&nbsp;R7赋值&nbsp;1us&nbsp;=&nbsp;5us&nbsp;延时总时间&nbsp;=&nbsp;三层循环&nbsp;+&nbsp;循环外&nbsp;=&nbsp;499995+5&nbsp;=<br />&nbsp;500000us&nbsp;=500ms&nbsp;计算公式:延时时间=[(2*R5+3)*R6+3]*R7+5&nbsp;二.&nbsp;200ms延时子程序&nbsp;程序:&nbsp;<br /><br />void&nbsp;delay200ms(void)&nbsp;<br /><br />{&nbsp;unsigned&nbsp;char&nbsp;i,j,k;&nbsp;<br /><br />for(i=5;i&gt0;i--)&nbsp;for(j=132;j&gt0;j--)&nbsp;<br /><br />for(k=150;k&gt0;k--);&nbsp;}&nbsp;<br /><br />产生的汇编<br /><br />&nbsp;C:0x0800&nbsp;7F05&nbsp;MOV&nbsp;R7,#0x05<br /><br />&nbsp;C:0x0802&nbsp;7E84&nbsp;MOV&nbsp;R6,#0x84<br /><br />&nbsp;C:0x0804&nbsp;7D96&nbsp;MOV&nbsp;R5,#0x96&nbsp;<br /><br />C:0x0806&nbsp;DDFE&nbsp;DJNZ&nbsp;R5,C:0806&nbsp;<br /><br />C:0x0808&nbsp;DEFA&nbsp;DJNZ&nbsp;R6,C:0804&nbsp;<br /><br />C:0x080A&nbsp;DFF6&nbsp;DJNZ&nbsp;R7,C:0802&nbsp;<br /><br />C:0x080C&nbsp;22&nbsp;RET&nbsp;<br /><br />三.&nbsp;10ms延时子程序&nbsp;程序:&nbsp;<br /><br />void&nbsp;delay10ms(void)&nbsp;<br /><br />{&nbsp;unsigned&nbsp;char&nbsp;i,j,k;<br /><br />&nbsp;for(i=5;i&gt0;i--)&nbsp;<br /><br />for(j=4;j&gt0;j--)<br /><br />&nbsp;for(k=248;k&gt0;k--);&nbsp;}&nbsp;<br /><br />产生的汇编&nbsp;C:0x0800&nbsp;7F05&nbsp;MOV&nbsp;R7,#0x05&nbsp;<br /><br />C:0x0802&nbsp;7E04&nbsp;MOV&nbsp;R6,#0x04&nbsp;<br /><br />C:0x0804&nbsp;7DF8&nbsp;MOV&nbsp;R5,#0xF8<br /><br />&nbsp;C:0x0806&nbsp;DDFE&nbsp;DJNZ&nbsp;R5,C:0806&nbsp;<br /><br />C:0x0808&nbsp;DEFA&nbsp;DJNZ&nbsp;R6,C:0804&nbsp;<br /><br />C:0x080A&nbsp;DFF6&nbsp;DJNZ&nbsp;R7,C:0802<br /><br />&nbsp;C:0x080C&nbsp;22&nbsp;RET&nbsp;<br /><br />四.&nbsp;1s延时子程序&nbsp;程序:&nbsp;<br /><br />void&nbsp;delay1s(void)&nbsp;<br /><br />{&nbsp;unsigned&nbsp;char&nbsp;h,i,j,k;<br /><br />&nbsp;for(h=5;h&gt0;h--)&nbsp;<br /><br />for(i=4;i&gt0;i--)&nbsp;<br /><br />for(j=116;j&gt0;j--)<br /><br />&nbsp;for(k=214;k&gt0;k--);&nbsp;}&nbsp;<br /><br />产生的汇编<br /><br />&nbsp;C:0x0800&nbsp;7F05&nbsp;MOV&nbsp;R7,#0x05&nbsp;<br /><br />C:0x0802&nbsp;7E04&nbsp;MOV&nbsp;R6,#0x04&nbsp;<br /><br />C:0x0804&nbsp;7D74&nbsp;MOV&nbsp;R5,#0x74&nbsp;<br /><br />C:0x0806&nbsp;7CD6&nbsp;MOV&nbsp;R4,#0xD6&nbsp;<br /><br />C:0x0808&nbsp;DCFE&nbsp;DJNZ&nbsp;R4,C:0808&nbsp;<br /><br />C:0x080A&nbsp;DDFA&nbsp;DJNZ&nbsp;R5,C:0806&nbsp;<br /><br />C:0x080C&nbsp;DEF6&nbsp;DJNZ&nbsp;R6,C:0804&nbsp;<br /><br />C:0x080E&nbsp;DFF2&nbsp;DJNZ&nbsp;R7,C:0802&nbsp;
gpfrank 发表于 2008-4-10 10:27 | 显示全部楼层

很有用的帖子

很有用的帖子
gotomy 发表于 2008-6-25 20:20 | 显示全部楼层

很有用的帖子

  
您需要登录后才可以回帖 登录 | 注册

本版积分规则

12

主题

39

帖子

0

粉丝
快速回复 在线客服 返回列表 返回顶部