1. 标志寄存器的作用
(1)用来存储相关指令的某些执行结果;
(2)用来为 CPU 执行相关指令提供行为依据;
(3)用来控制 CPU 的相关工作方式;
2. 8086 CPU 标志寄存器的结构
- flag 的 1,3,5,12,13,14,15 位在 8086 CPU 中没有使用,不具有任何含义。
2. ZF 标志
- 存储于标志寄存器的第 6 位,即 零标志位。
记录相关指令执行后,其结果是否为 0。如果为 0 zf=1,不为 0 zf=0 - 在 8086 CPU 中,运算指令的执行影响标志寄存器(比如 add sub mul div inc or and 等),传送指令的执行对标志寄存器没有影响(比如 mov push pop等)。
3. PF 标志
*存储于标志寄存器的第 2 位,即 奇偶标志位。
记录相关指令执行后,其结果的所有 bit 位中 1 的个数是否为偶数,如果 1 的个数为偶数, pf=1, 如果 1 的个数为奇数, pf=0
4. SF 标志
*存储于标志寄存器的第 7 位,其结果是否为 负。
记录相关指令执行后,如果结果为 负,sf = 1, 非负 sf = 0
5. CF 标志
*存储于标志寄存器的第 0 位,即 进位标志位。
add 有进位 cf = 1 无进位 cf = 0
sub 有借位 cf = 1 无借位 cf = 0
6. OF 标志
*存储于标志寄存器的第 11 位,即 溢出标志位。
- 对于 CF OF 所表示的进位与溢出,是分别对 无符号数 和 有符号数 运算而言的。它们之间没有 任何关系。
7. adc 指令
指令格式: adc 操作对象1, 操作对象2
功能: 操作对象1 = 操作对象1 + 操作对象2 + CF
8. sbb 指令
指令格式: sbb 操作对象1, 操作对象2
功能: 操作对象1 = 操作对象1 - 操作对象2 - CF
9. cmp 指令
指令格式: cmp 操作对象1, 操作对象2
功能:计算 操作对象1 - 操作对象2,但并不保存结果,仅仅根据计算结果对标志寄存器进行设置。
- 如果 sf = 1 of = 0
of = 0 结果没有溢出, 逻辑上真正结果的正负=实际结果的正负
sf = 1 ,(ah) < (bh) - 如果 sf = 1 of = 1
of = 1 结果有溢出, 逻辑上真正结果的正负 != 实际结果的正负
sf = 1 ,(ah) > (bh) - 如果 sf = 0 of = 0
of = 0 结果没有溢出, 逻辑上真正结果的正负=实际结果的正负
sf = 0 ,(ah) > (bh) - 如果 sf = 0 of = 1
of = 1 结果有溢出, 逻辑上真正结果的正负 != 实际结果的正负
sf = 0 ,(ah) < (bh)
10. 检测比较结果条件转移指令
assume cs:codesg,ds:datasg
datasg segment
db 8,11,8,1,8,5,63,38
datasg ends
codesg segment
start: mov ax,datasg
mov ds,ax
mov si,0 ;ds:si 指向第一个字节
mov bx,0
mov ax,0
mov dx,0
mov cx,8
s: cmp byte ptr [si],8 ;和8进行比较
je equal8 ;如果相等 执行 equal8
ja above8
jb below8
equal8: inc ax ;如果相等 计数+1
jmp short next
below8: inc dx ;如果小于8 计数+1
jmp short next
above8: inc bx ;如果大于8 计数+1
jmp short next
next: inc si
loop s
mov ax,4c00h
int 21h
codesg ends
end start
11. DF 标志
*存储于标志寄存器的第 10 位,即 方向标志位。
- 在 串处理指令中,控制每次操作后 si di 的增减
df=0 每次操作后 si di 递增
df=1 每次操作后 si di 递减
指令格式: movsb
功能:执行 movsb 指令相当于进行下面几步操作。
(1)((es)16 + (di)) = ((ds)16 + (si))
(2)如果 df=0,(si)=(si)+1 (di)=(di)+1
如果 df=1,(si)=(si)-1 (di)=(di)-1
movsw
功能:
(1)((es)16 + (di)) = ((ds)16 + (si))
(2)如果 df=0,(si)=(si)+2 (di)=(di)+2
如果 df=1,(si)=(si)-2 (di)=(di)-2
rep movsb
功能:
s:movsb
loop s
根据cx的值,重复执行后面的串传送指令,实现 cx 个字符的传送
8086 提供了 cld std 两条指令对 df 进行设置
cld df=0
std df=1
assume cs:codesg,ds:datasg
datasg segment
db 'Welcome to masm!'
db 16 dup (0)
datasg ends
codesg segment
start: mov ax,datasg
mov ds,ax
mov si,0 ;ds:si 指向 data:0
mov es,ax
mov di,10H ;ss:di 指向 data:0010
mov cx,16
cld
rep movsb
mov ax,4c00h
int 21h
codesg ends
end start
12. pushf && popf
pushf 的功能是将标志寄存器的值压栈
popf 是从栈中弹出数据,送入标志寄存器中。
pushf popf 是为直接访问标志寄存器提供了一种方法
13. 以 0 结尾的字符串重的小写字母转变成大写字母
assume cs:codesg,ds:datasg
datasg segment
db 'Beginner‘s All-purpose Symbolic Instruction Code.',0
datasg ends
codesg segment
start: mov ax,datasg
mov ds,ax
mov si,0 ;ds:si 指向 data:0
call letterc
mov ax,4c00h
int 21h
letterc: push si
comp: cmp byte ptr [si],0 ;和0进行比较
je exit
cmp byte ptr [si],'a'
jb nextchar
cmp byte ptr [si],'z'
ja nextchar
and byte ptr [si],11011111b
nextchar: inc si
jmp short comp
exit: pop si
ret
codesg ends
end start