搜索
bottom↓
回复: 6

一个定时函数的疑问

[复制链接]

出0入0汤圆

发表于 2013-10-17 22:21:23 | 显示全部楼层 |阅读模式
这是一个IIC电调上的一个延时程序,小弟有一个疑问不接。这个延时的原理是,在当前计数器值 CountMilliseconds 的基础上加上要延时的w得到新的计数值w,将新的w和 CountMilliseconds相减,当定时器计数器 CountMilliseconds 增加到w时,他们的差值为0,。当 CountMilliseconds 在增加1时,差值应该为FFFF,此时 CheckDelay(unsigned int t)函数将返回1,表明定时到。这也是为什么SetDelay(unsigned int t)函数要减一。但是我小弟疑问的是当通过SetDelay(unsigned int t)设定了新的值,CountMilliseconds发生了溢出怎么办,也就是它回到了零,这个时候CheckDelay(unsigned int t)检测时岂不要多跑一个周期?

volatile unsigned int CountMilliseconds = 0; //毫秒计数
volatile unsigned char Timer0Overflow;       //T/C0 溢出

enum {
  STOP             = 0,
  CK               = 1,
  CK8              = 2,
  CK64             = 3,
  CK256            = 4,
  CK1024           = 5,
  T0_FALLING_EDGE  = 6,
  T0_RISING_EDGE   = 7
};


SIGNAL(SIG_OVERFLOW0)
{
static unsigned char cnt;
Timer0Overflow++;
if(!cnt--)
  {
   cnt = 3;
   CountMilliseconds += 1;
   if(I2C_Timeout) I2C_Timeout--;
   //if(PPM_Timeout) PPM_Timeout--;
   if(SIO_Timeout) SIO_Timeout--;
  }
}


void Timer0_Init(void)
{
TCCR0  = TIMER_TEILER;  //8分频
// TCNT0 = -TIMER_RELOAD_VALUE;  // reload
TIM0_START;
TIMER2_INT_ENABLE;
}


unsigned int SetDelay(unsigned int t)
{
  return(CountMilliseconds + t - 1);                                             
}

char CheckDelay (unsigned int t)//
{
  return(((t - CountMilliseconds) & 0x8000) >> 8);
}

void Delay_ms(unsigned int w)
{
unsigned int akt;
akt = SetDelay(w);
while (!CheckDelay(akt));
}

阿莫论坛20周年了!感谢大家的支持与爱护!!

知道什么是神吗?其实神本来也是人,只不过神做了人做不到的事情 所以才成了神。 (头文字D, 杜汶泽)

出0入0汤圆

发表于 2013-10-18 08:26:45 | 显示全部楼层
无符号加减没你说的这个问题

出0入0汤圆

 楼主| 发表于 2013-10-18 08:56:22 | 显示全部楼层
我后来想了想,发现确实很巧妙。如果 SetDelay函数和CheckDelay函数之间CountMilliseconds会溢出的话,说明新计算的延时值也会也会溢出。比如当前CountMilliseconds值是65530,要延时的值为11,那么新的延时计数值为65540,由于为unsigned int 型,溢出后就是0x04;这个时候0x04减去一个65530更大的数时得到的是补码(不知道是不是反正最高位为0);只有当CountMilliseconds值经过10次计数后也为0x04,再减一为0x05,此时0x04-0x05=ffff,CheckDelay函数返回1,定时到。

出0入0汤圆

 楼主| 发表于 2013-10-18 09:03:12 | 显示全部楼层
usingavr 发表于 2013-10-18 08:26
无符号加减没你说的这个问题

好像这个定时值不能大于32768,比如要定时的值为32780,此时CountMilliseconds值为2,新定时值为32781,这个时候(((t - CountMilliseconds) & 0x8000) >> 8)将返回1。不过不太可能延时32秒,不知道我分析的对不对。

出0入0汤圆

发表于 2013-10-18 09:08:22 | 显示全部楼层
YZY0707 发表于 2013-10-18 09:03
好像这个定时值不能大于32768,比如要定时的值为32780,此时CountMilliseconds值为2,新定时值为32781, ...

对的,不能大于  unsigned int 最大值的一半

出0入0汤圆

 楼主| 发表于 2013-10-18 15:29:00 | 显示全部楼层
usingavr 发表于 2013-10-18 09:08
对的,不能大于  unsigned int 最大值的一半

谢啦!这个方法确实很巧妙。

出0入0汤圆

发表于 2013-10-18 17:33:29 | 显示全部楼层
稍后细看。
回帖提示: 反政府言论将被立即封锁ID 在按“提交”前,请自问一下:我这样表达会给举报吗,会给自己惹麻烦吗? 另外:尽量不要使用Mark、顶等没有意义的回复。不得大量使用大字体和彩色字。【本论坛不允许直接上传手机拍摄图片,浪费大家下载带宽和论坛服务器空间,请压缩后(图片小于1兆)才上传。压缩方法可以在微信里面发给自己(不要勾选“原图),然后下载,就能得到压缩后的图片。注意:要连续压缩2次才能满足要求!!】。另外,手机版只能上传图片,要上传附件需要切换到电脑版(不需要使用电脑,手机上切换到电脑版就行,页面底部)。
您需要登录后才可以回帖 登录 | 注册

本版积分规则

手机版|Archiver|amobbs.com 阿莫电子技术论坛 ( 粤ICP备2022115958号, 版权所有:东莞阿莫电子贸易商行 创办于2004年 (公安交互式论坛备案:44190002001997 ) )

GMT+8, 2024-7-24 00:28

© Since 2004 www.amobbs.com, 原www.ourdev.cn, 原www.ouravr.com

快速回复 返回顶部 返回列表