kingway00 发表于 2014-12-15 22:57:16

atmega8汇编延时程序CPC指令作用令人疑惑

本帖最后由 kingway00 于 2014-12-15 22:56 编辑

我在ICCAVR上用C语言写了一个简单的延时程序
void delay(unsigned int cnt)
{
    while(--cnt);
}
然后查看它的汇编代码如下:
_delay:
   MOVW R24,R16
   SBIW R24,1
   MOVW R16,R24
   CPI        R16,0
   CPC        R16,R17
   BRNE _delay
   RET

汇编代码意图很明显,就是利用R16,R17寄存器存储cnt参数。

我很奇怪为什么红字部分这样写就能实现R16,R17寄存器都等于0的判断(即延时结束)
我实验了一下,在R16=R17=0x01的时候,理论上CPC R16,R17这条指令应该等于0
但实际上并非如此,而是只有R16,R17都等于0的时候,CPC指令才等于0
我就奇怪了,不知道哪位知道汇编的解释下?

alias 发表于 2014-12-16 01:01:55

>> 在R16=R17=0x01的时候,理论上CPC R16,R17这条指令应该等于0

在R16=R17=0x01的时候,CPI R16,0 指令执行後,Z=0 因 R16=0x01。理论上CPC R16,R17这条指令应该等于0,但 CPC R16,R17 指令执行时如不等时即清Z,但相等时却会不置位Z而保恃Z的早前状态,这是CPC指令的特点。

所以当 R17:R16 都是零时,CPI R16,0 会令Z=1, 而 CPC R16,R17 则保恃Z不变,所以 BRNE 失效。

kingway00 发表于 2014-12-16 09:41:49

alias 发表于 2014-12-16 01:01
>> 在R16=R17=0x01的时候,理论上CPC R16,R17这条指令应该等于0

在R16=R17=0x01的时候,CPI R16,0 指令执 ...

alias大神,你再次给了我一次犀利的点拨,谢谢{:lol:}

alias 发表于 2014-12-16 15:12:58

kingway00 发表于 2014-12-16 09:41
alias大神,你再次给了我一次犀利的点拨,谢谢

CPC 指令Z 的置位/清除也可以理解为:

Z= ( Z 之前状态 ) AND ( 比较下 Z 的结果)

kingway00 发表于 2014-12-16 15:44:43

alias 发表于 2014-12-16 15:12
CPC 指令Z 的置位/清除也可以理解为:

Z= ( Z 之前状态 ) AND ( 比较下 Z 的结果) ...

恩,确实是,我后来看了下AVR STUDIO的帮助文档,CPC指令Z标志位的计算是这样的

Z: R7 • R6 • R5 • R4 • R3 • R2 • R1 • R0 • Z                   Previous value remains unchanged when the result is zero; cleared otherwise.

就是你的意思{:lol:}

alias 发表于 2014-12-16 16:41:12

还有你楼主位红色的 CPC R16,R17 指令我会写成:

   CPC R17,R16

运行结果会是一样,但这般写对我来说会更合理。

so_so_so_so 发表于 2015-3-5 13:08:23

收益了               
页: [1]
查看完整版本: atmega8汇编延时程序CPC指令作用令人疑惑