8086CPU的flag寄存器(16位)各标志位如下(这是32位EFLAG的低十六位图,但是32位与16位是一样的,只不过32位多了16位且高16位没有使用到):
标志位寄存器中保存的是当前指令运算的信息状态,比如进位信息保存在CF标志位(注意:是运算结果的标志,比如add、sub、mul、div、and、or等,而mov、push等不会改变标志位)。
**需要强调一点:**PF标志位,是奇偶标志位,但是这个奇偶指的是最低八位是奇数还是偶数,和高位没有关系。即使是16位寄存器/16位内存/32位寄存器/32位内存…也只由低八位的1的奇偶个数决定。(2017年7月31日14:29:36注)
其中CF为进位标志(Carry Flag),OF为溢出标志(Overflow Flag)。标志位只占以为以为着一个标志只能有0(假)或1(真)两种结果。而不同标志位在Debug中显示的真与假其显示方式是不同的,具体如下表:
如图所示为一部分:
我们通常认为溢出(上溢)就是因为进位时当前存储格式(1B、2B、4B等)的位数(8bit、16bit、32bit)不够而引起的。比如8位寄存器:11111111B+1B=100000000B超过了八位的1被认为是溢出寄存器(放不下),当然也是进位进上去的1。但是CF标志位的进位标志与OF标志位的溢出标志是不同的概念。
CF针对无符号数(将寄存器中的操作数都看作是无符号数)
OF针对有符号数(将寄存器中的操作数都看作是有符号数)
有符号无符号指的是最高位是否是符号位,即是以补码的形式看待还是以原码的形式看待。
CF范围:0~255/0X00~0XFF(8位)、0~65535/0X0000~0XFFFF(16位)、…
OF范围:-128~127/0X80~0XEF(8位)、-32768~32767/0X8000~0XEFFF(16位)、…
我们再画张图来说明CF与OF的数据范围的区别:
区别:
其监测的结果数据范围不同:
以8位寄存器作为操作种数据来源为例,如果一个运算的结果最终超过[-128,127]无论是大于127还是小于-128就被认为是溢出,OF被置为1,如果结果在[-128,127]就认为没溢出OF被置为0,。
而对于OF:如果计算的结果超过[0,255]的范围,就有进位,CF就被置为1,如果结果再[-128,127]范围内,就是没有进位CF被置为0。
共同点:
我们经常会将溢出与进位这两个概念混为一谈,就是因为这两个的共同点都是“存不下了”。什么叫存不下了?(依然以8位寄存器为例)。我们先举个例子(寄存器宽度为8位):
(1)、8H+8H:
对于signed:(8)+(8)=16,没超过[-128,127]的范围,OF为0
对于unsigned:(8)+(8)=16,没超过[0,255]的范围,CF为0
(2)、80H+81H:
对于signed:(-128)+(-127)=-255,超过[-128,127]的范围,OF为1
对于unsigned:(128)+(129)=257,超过[0,255]的范围,CF为1
(3)、FCH+05H:
对于signed:(-4)+(5)=1,没有超过[-128,127]的范围,OF为0
对于unsigned:(252)+(5)=257,超过[0,255]的范围,CF为1
(4)、7FH+2H:
对于signed:(127)+(2)=129,超过[-128,127]的范围,OF为1
对于unsigned:(127)+(2)=129,没超过[0,255]的范围,CF为0
我们看到超过表示范围的原因都是值太大,当前位数表示不了,但是对于同样位数的寄存器,有符号与无符号就不一样了。同样的位数,OF除了符号位外能表示的值只有七位是数据,而CF认为有八位是数据位。但是我们知道这只是其共同点,不代表这两个有关系,上例也表示,OF、CF的0与1是没有直接联系的。只要记住有符号与无符号的数据范围,并且记住OF是对于有符号数,CF对于无符号数即可。