汇编语言的指令格式是重中之重,不然你编写出来就会偏离轨道!
汇编语言指令格式
要用汇编语言编写正确的程序,一定要了解汇编语言的指令格式。汇编语言的指令格式如下:
[标号:]操作码(空格)[操作数1] [,操作数2] [,操作数3] [;注释]
下面是一条汇编语言指令:
MAIN:MOV P3,#FFH ;该指令的功能是将数据11111111(即FFH)送到P3端口
① 标号。用来标注指令的地址,一般由 1~8 个字母和数字组合而成,但必须以字母开头,以冒号“:”结束。上面的汇编语言指令中,标号是“MAIN:”。
② 操作码。用来规定指令的操作功能,一般由 2~5 个字母(英文单词或单词的缩写)组成。上面的汇编语言指令中,操作码是“MOV”。
③ 操作数。它是操作码的操作对象,操作数通过空格与操作码分开,在一条指令中可以有0~3个操作数,操作数之间用逗号“,”隔开。上面的汇编语言指令中,操作数是“P3, #0FFH”。
④ 注释。它是编程者对指令的说明,通常用来描述程序的功能。在编写一些大型的、复杂的程序时,对程序进行注释,有利于调试程序,使别人容易读懂程序,注释以分号“;”开始。
在汇编语言指令中,一定要有操作码,而“[ ]”中的内容根据情况可有可无。要注意的是,指令中的“:”、“,”、“;”为半角符号,应在输入法为英文状态时输入。
8086汇编(16位汇编)学习笔记03.汇编指令
原文地址:
- 数据传送指令
- 算数运算类指令
- 位操作类指令
- 串操作类指令
- 控制转移类指令
- 处理器控制类指令
**传送类指令不影响标志位,**除了标志位传送指令外。
不存在存储器向存储器的传送指令。
两个操作数类型要一致.操作宽度
不允许段寄存器向段寄存器传送
立即数到内存需要注意: 需要指明立即数要占多大内存 byte 表示字节 word 表示 字,寄存器不需要是因为他自带宽度属性
源操作数和目标数的长度必须一致,要么都是byte,要么都是word,个别指令除外,8位寄存器和16位寄存器相互赋值,会被系统转化成16位
段寄存器到内存需要指明宽度
段偏移的不同写法,推荐使用 [bp + si + 12 ]
把两个地方的数据进行互换。不存在存储器向存储器的传送指令。
1.寄存器与寄存器之间可以对换数据
2.寄存器与存储器之间可以对换数据
3.不能在存储器与存储器之间对换数据
•作用:将BX 指定的缓冲区中、AL指定的位移处的一个字节取出赋给AL 。即 : al <– ds:[ bx + al ]
•该指令无操作数。
•用途:键盘的扫描码,需要转为ASCII 码,可以将扫描码做成表,扫描码作下标可以查到对应的ASCII 码
类似于 al = bx[al];
将标志寄存器的内容压入堆栈,同时栈顶指针SP减2
等同于 sub sp , 2 申请内存
寄存器SP的值就是当前栈顶
这里查看数据内存是 -d ss:fffo 是因为 sp寄存器的段基址 是 ss,sp是段偏移
8位寄存器,8位内存和还有立即数是不可以入栈的
将栈顶单元内容送到标志寄存器,同时栈顶指针SP 加 2
等同于 mov reg, [sp] 把把数据传给变量
add sp , 2 释放内存
将所有寄存器的值入栈或者出栈
一般用于函数,为了不影响其他函数的环境,自己先把寄存器环境入栈,自己操作完再出栈
标志寄存器传送指令用来传送标志寄存器FLAGS的内容,方便进行对各个标志位的直接操作
•有2对4条指令
–低8位传送:LAHF和SAHF
–16位传送:PUSHF和POPF
LAHF;AH←FLAGS的低字节
SAHF;FLAGS的低字节←AH
82 = 1000 0000 上面标志位图 可知 SF = 1 即 SF 状态标志位的值是 1 NG
55 = 0101 0101 上面标志位图 可知 使 0 , 2 ,3, 6 号的标志位全部设为 1
PUSHF
•SP←SP-2
•SS:[SP]←FLAGS
•PUSHF指令将标志寄存器的内容压入堆栈,同时栈顶指针SP减2
POPF
•FLAGS←SS:[SP]
•SP←SP+2
•POPF指令将栈顶字单元内容送标志寄存器,同时栈顶指针SP加2
32字节传送
pushfd:将eflags压栈
popfd :将栈顶32字节出栈到eflags中
8272 转成 小尾 是 72 82
可以看出 if 位 和 sf 位是 1
将值改成 FF 即 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 可以看出,所有标志位都变成了1
•地址传送指令将存储器单元的逻辑地址送至指定的寄存器
•注意不是获取存储器单元的内容
作用:将存储器操作数的有效地址传送至指定的16位寄存器中
格式: LEA reg16 ,[EA]
举例
LEA bx ,[bx + si] //相当于bx = bx+si。
操作
可以看出 lea 跟 mov 的区别 是 lea 是 将 偏移值 给 寄存器,mov是获取 地址里面的值 给 寄存器
LDS r16,mem
LES r16, mem
r16←mem,
LES指令将主存中mem指定的字送至r16,并将mem的下一字送ES寄存器
•8086通过输入输出指令与外设进行数据交换;呈现给程序员的外设是端口(Port)即I/O地址
•8086用于寻址外设端口的地址线为16条,端口最多为216=65536(64K)个,端口号为0000H~FFFFH
•每个端口用于传送一个字节的外设数据
•8086的端口有64K个,无需分段,设计有两种寻址方式
•对大于FFH的端口只能采用间接寻址方式
•将外设数据传送给CPU内的AL/AX
IN AL,i8; 字节输入:AL←I/O端口(i8直接寻址)
IN AL,DX; 字节输入: AL←I/O端口(DX间接寻址)
IN AX,i8 ; 字输入 : AX←I/O端口(i8直接寻址)
IN AX,DX; 字输入: AX←I/O端口(DX间接寻址)
•将CPU内的AL/AX数据传送给外设
OUT i8,AL; 字节输出:I/O端口←AL(i8直接寻址)
OUT DX,AL; 字节输出:I/O端口←AL(DX间接寻址)
OUT i8,AX; 字输出:I/O端口←AX(i8直接寻址)
OUT DX,AX; 字输出:I/O端口←AX(DX间接寻址)
1.四则运算是计算机经常进行的一种操作。算术运算指令实现二进制(和十进制)数据的四则运算。
2.请注意算术运算类指令对标志的影响
掌握:ADD/ADC/INC、SUB/SBB/DEC/ NEG/CMP
熟悉:MUL/IMUL、DIV/IDIV
理解:CBW/CWD、DAA/DAS、 AAA/ AAS/AAM/AAD
ADD reg,imm/reg/mem;reg←reg+imm/reg/mem //把结果存入寄存器
ADD mem,imm/reg; mem←mem+imm/reg //把结果存入内存,内存不可以与内存相加
跟上面区别是会加上上一次加法的进位值 ,常用于大数运算
ADC reg,imm/reg/mem; reg←reg+imm/reg/mem+CF
ADC mem,imm/reg; mem←mem+imm/reg+CF
SUB reg,imm/reg/mem; reg←reg-imm/reg/mem //把结果存入寄存器
SUB mem,imm/reg; mem←mem-imm/reg //把结果存入内存,内存不可以与内存相减
跟上面区别是会减去上一次减法的借位值
SUB reg,imm/reg/mem;reg←reg-imm/reg/mem
SUB mem,imm/reg; mem←mem-imm/reg
编写汇编指令计算减法:
70782DDB-76543218FA23FBC3
MOV AX,2DDB
0DAB:0100
0DAB:0103 SUB AX,3218
LV BX,7078
IODAB:0106 MOV
10DAB:0109 SBB BX,7654
0DAB:010D
-T
AX-2DDB BX-0000 CX-0000 DX-0000 SP-FFFE BP-0000 SI-0000 DI-000
SS-0DAB CS-0DAB IP-0103 NV UP EI NG NZ NA PO NC
DS-ODAB
ES-ODAB
AX,3218
0DAB:0103 2D1832
SUB
AX-FBC3 BX-0000 CX-0000 DX-0000 SP-FFFE BP-0000 SI-0000 DI-000
DS-ODAB ES-ODAB SS-0DAB CS-0DAB IP-0106 NV UP EI NG NZ NA PEI
CY
MOV
BX,7078
10DAB:0106 BB70
产生借位
AXFBC3 BX-7078 CX-000
DX-000 SP-FFE BP-0000 SI-0000 DI-0000
CS-ODAB IP-0109 NV UP EI NG NA PE CY
SS-ODAB
DS-ODAB ES-ODAB
BX,7654
0DAB:0109 81DB5476
SBB
7078 -7654
-1(借位)
CX-0000 DX-000 SP-FFFE BP-0000 SI-0000 DI-00000
FA23
AXFBC3
BX:
CS-ODAB IP-010D NV UP EI NG NZ NA PO CY
DS-ODAB
ES-ODAB SS-ODAB
AX
POP
58
IODAB:010D
DEC -1,不影响CF位。
100
-R
AX
A.
0DAB:0100 DEC AX
0DAB:0101
DEC AX
0DAB:0102
DEC AX
0DAB:0103 DEC AX
10DAB:0104
-T
100 SI-000 DI-0000
00FF
AX:
BX-000 CX-0000 DX-0000 SPFFE BP-0000
DS-0DAB ES-0DAB SS-0DAB CS-0DAB IP-0101 NV UP EI PL NZ AC PE NC
48
10DAB:0101
DEC
AX
BX-0000 CX-000 DX-0000 SP-FFFE BP-0000 SI-0000 DI-00000
AX-00FE
ES-0DAB SS-ODAB CS-0DAB IP-0102 NV UP EI PL NZ NA PO NC
DS-ODAB
DEC
0DAB:0102 48
AX
IBX-000 CX-0000 DX-0000 SP-FFFE BP-0000 SI-0000 DI-0000
AX-00FD
ES-0DAB SS-ODAB CS-0DAB IP-0103 NV UP EI PL NZ NA PO NC
OS-ODAB
0DAB:0103 48
AX
DEC
-T
AX-00FC
BX-0000 CX-0000 DX-000 SP-FFFE BP-0000 SI-00000 DI-0000
DS-ODAB ES-ODAB SS-0DAB CS-0DAB IP-0104 NV UP EI PL NZ NA PE NC
求补指令NEG(negative)
•NEG指令对操作数执行求补运算:用零减去操作数,然后结果返回操作数
•求补运算也可以表达成:将操作数按位取反后加1
•neg ax ; 如果ax = 0,则CF标志位 = 0;若ax != 0, 则CF = 1
NEG reg/mem; reg/mem← 0-reg/mem
一个数 + 这个数补码 = 0
-A
LODAB:0100 MOV AX,100
0DAB:0103 NEG AX
0DAB:0105 NEG AX
0DAB:0107
AX-0100 BX-0000 CX-000 DX-0000 SP-FFFE BP-0000 SI-00000 DI-0000
DS-0DAB ES-0DAB SS-0DAB CS-0DAB IP-0103 NV UP EI NG NZ
NZ NA PO NC
AX
0DAB:0103 F7D8
NEG
AX-FF00 BX-0000 CX-0000 DX-0000 SP-FFFE BP-0000 SI-00000 DI-0000
DS-0DAB ES-0DAB SS-0DAB CS-0DAB IP-0105 NV UP EI NG NG NA PE CY
0DAB:0105 F7D8
AX
NEG
AX-0100 BX-0000 CX-0000 DX-0000 SP-FFFE BP-0000 SI-0000 DI-0000
IDS-0DAB ES-0DAB SS-0DAB CS-0DAB IP-0107 NV UP EI PL NZ NA PE CY
JS
0179
0DAB:01077870
0DAB:0100
AX,0
MOV
0DAB:0103 NEG AX
(0DAB:0105
NEG AX
ODAB:0107
AX-000 BX-0000 CX-0000 DX-00000 SP-FFFE BP-0000 SI-0000 DI-000
NG NZ
IDS-ODAB ES-0DAB SS-0DAB CS-0DAB IP-0103 NV UP EI NO
PONC
NA
0DAB:0103 F7D8
NEG
AX
-T
AX-0000
BX-0000
0 DX-000 SP-FFFE BP-0000 SI-0000 DI-000
CX-0000 DX:
NC
CS-ODAB
SS-ODAB
DS-ODAB
PE
IP-0105 NV UP EI PL ZR NA
ES-ODAB
0DAB:0105 F7D8
NEG
AX
-T
AX-0000
BX-0000 CX-0000 DX-0000 SP-FFFE BP-0000 SI-0000 DI-0ARA
DS-ODAB ES-0DAB SS-0DAB CS-0DAB IP-0107 NV UP EI PL ZR NA PEINC
JS
0179
0DAB:01077870
作业
1 选择指令实现下列要求,OPN1 ← OPN2的含义:把操作数OPN2的值传送给OPN1。
1)、SI ← BX mov si,bx
2)、AL ← BH mov al,bh
3)、AX ← 13AH mov ax,013a
4)、SI ← 1000H mov si,1000
5)、BX ← CL mov bx,cl
6)、DS ← AX mov DS,AX
7)、AL ← 12H mov AL,12
8)、CX ← 0CDH mov CX,00CD (假设0CDH为无符号数)2 PUSHA指令能把哪些寄存器压栈?它们进栈的顺序是什么?可用什么指令把这些寄存器依次弹出堆栈?
3 PUSHAD指令完成什么操作?它与什么操作的作用相反?
4 在16位/32位CPU中,标志位寄存器的内容进栈和出栈的指令是什么?
5 写出下列指令序列中每条指令的执行结果,请在Debug环境下验证之,并注意各标志位的变化情况。
MOV BX, 23ABH BX = 23AB
ADD BL, 0ACH BX = 2357 ZF,AF,OF状态位改变
MOV AX, 23F5H ax = 23f5 无状态位改变
ADD BH, AL bx = 1857 PF,AF,OF 状态位改变
SBB BX, AX bx = F461 PF 和 SF 状态位改变
ADC AX, 12H ax = 2408 CF 和 SF 状态位改变
SUB BH, -9 bx = fd61 CF,AF,SF状态位改变
MOV
BX, 23AB
0DAB:0100
ADD
0DAB:0103
BL, OAC
MOV
0DAB:0106
AX, 23F5
BH, AL
ADD
0DAB:0109
10DAB:010B
SBB
BX, AX
ADC
AX, 12
0DAB:010D
0DAB:0110
SUB
BH, -9
0DAB:0113
AX-0000 BX-23AB CX-000 DX-0000 SP-FFFFE BP-0000 SI-0000 DI-000
DS-0DAB ES-0DAB SS-0DAB CS-0DAB IP-0103 NV UP EI NG NZ NA PO NC
0DAB:0103 80C3AC
BL, AC
ADD
-000 SI-000 DI-0000
AX-0000 BX-2357 CX-0000 DX-000 SP-FFFE BP-0000 SI
DS-ODAB ES-0DAB SS-0DAB CS-0DAB IP-0106 OV UP EI PL NZ AC PO CY
0DAB:0106 B8F523
MOV
AX,23F5
AX-23F5 BX-2357 CX-000 DX-0000 SP-FFFFE BP-0000 SI-0000 DI-0000
DS-ODAB ES-0DAB SS-0DAB CS-0DAB IP-0109 OV UP EI PL NZ AC PO CY
BH, AL
0DAB:0109 00C7
ADD
SP-FFE BP-0000 SI-000 DI-0000
AX-23F5 BX-1857 CX-000 DX-000
DS-ODAB ES-ODAB SS-ODAB CS-ODAB
IP-010B NV UP EI PL NZ NA PE CY
BX, AX
ODAB:010B 19C3
SBB
-T
I CX-000 DX-0000 SP-FFFFE BP-000 SI-0000 DI-0000
AX23F5 BX-F461 CX
DS-ODAB ES-ODAB SS-ODAB CS-ODAB IP
IP010D NV UP EI NG NZ NA PO CY
ODAB:010D83D012
AX,+12
ADC
-T
AX-2408 BX-F461 CX-0000 DX-0000 SP-FFFE BP-0000 SI-0000 DI-0000
SS-0DAB CS-ODAB IP-0110 NV UP EI PL NZ NA PO NC
DS-ODAB ES-ODAB
BH, F7
ODAB:011080EFF7
SUB
-T
AX-2408 BX-FD61 CX-000 DX-0000 SI
00 SPFFE BP-000 SI-0000 DI-000
IP-0113 NV UP EI NG NZ AC PO CY
DS-ODAB ES-ODAB
SS-ODAB CS-ODAB
6 假定DS=1123H,SS=1400H,BX=0200H,BP=1050H,DI=0400H,SI=0500H,
LIST的偏移量为250H,试确定下面各指令访问内存单元的地址。
1)、MOV AL, [1234H] 12464 = 1123 * 10 + 1234
3)、MOV [DI], AL 11630 = 1123 * 10 + 400
5)、MOV AL, [BP+DI] 15450 = 1400 * 10 + (1050 + 400)
7)、MOV EDX, [BP] 15050 = 1400*10 + 1050
9)、MOV CL, LIST[BX+SI] 12180 = 1123 * 10 + 500 + 250 +200
12)、MOV AL, [SI-0100H] 11630 = 1123 * 10 + 500 -100
7 已知SI=1200H,DS:[1205H]=1234H,试说明指令MOV BX,5[SI]和LEA BX,5[SI]的区别,执行后BX的内容分别是多少?
MOV BX,5[SI] 是将 DS最为段基址 , si+ 5 = 1025 作为段偏移 处的值 1234 给 BX ,此时 BX 值 是 1234
LEA BX,5[SI] 是将 si+ 5 = 1025 的值 给 BX ,此时 BX 的值位 1025
8 用一条指令实现把BX和SI之和传送给CX。
9 试比较下列三组指令的功能。
1)、LDS SI, [DI]
将 si[di] 处的一个字传给 si ,并将下一字传入 DS
SPFFFE BP-0000 SI-000 DI-0000
AX-0000 BX-000 CX-0000 DX-0000
UP EI NG NZ
NZ NA PO NC
ES-ODAB SS-0DAB CS-ODAB IP-0100 NV UP E
DS-ODAB
RET
10DAB:0100 C3
DS:DI 11 22 33 44
-E
J-
SP-FFFE BP-000 SI-000 DI-0000
BX-0000 CX-000 DX-0000
AX-0000
SS-0DAB CS-0DAB IP-0100 NV UP EI NG NZ NA PO NC
ES-ODAB
DS-ODAB
RET
ODAB:0100 C3
-A
ODAB:0100 LDS SI ,[DI]
0DAB:0102
-T
SI-2211 DI-0000
AX-000 BX-0000 CX-0000 DX-0000 SP-FFFE BP-0000
DS-4433 ES-0DAB SS-0DAB CS-0DAB IP-0102 NV UP EI NG NZ NA PO NC
[BX+SI],AL
0DAB:0102 000
ADD
2)、MOV SI, [DI]
3)、MOV SI, [DI+2] ? 感觉应该是: MOV DS, [DI+2]
BX-0000 CX-0000 DX-0000 SP-FFFE BP-0000 SI-0000 DI-0000
AX-0000
DS-ODAB
SS-ODAB
ES-ODAB
CS-ODAB IP-0100 NV UP EI NZ NA PO NC
DAB:0100 C3
RET
-E DS:DI 11 22 33 44 55
0DAB:0100
SI,LDIL
MOV
ODAB:0102
SI,[DI+2]
MOV
0DAB:0105
SP-FFFE BP-000 SI-2211 DI-000
AX-0000
BX-0000
CX-0000 DX-000
CS-ODAB
SS-ODAB
, IP-0102 NV UP EI NZ NA PO NC
ES-ODAB
DS-ODAB
SI,[DI+02]
MOV
8B7502
ODAB:0102
-T
AX-0000
BX-0000
FE BP-000 SI-4433 DI-000
CX-0000 DX-000 SPFFFE E
NV UP EI NG NZ NA PO NC
ES-ODAB
SS0DAB CS-0DAB IP-0105
DS-ODAB
[BX+SI],AL
ADD
10DAB:0105 000
1 的指令 相当于 2,3两组指令的组合
-E DS:SI 11 22 33 44 55 66
AX-000 BX-00000 CX-0000 DX-0000 SP-FFFFE BP-0000 SI-0000 DI-0000
DS-ODAB ES-0DAB SS-0DAB CS-0DAB IP-0100 NV UP EI NG NG NA PO NC
RET
0DAB:0100 C3
-A
30 MOV SI, [DI]
ODAB:0100
2 MOV DS,WORD PTR [DI+2]
0DAB:0102
0DAB:0105
AX-000 BX-0000 CX
CX-00000 DX-000
00 SI-2211 DI-000
000 SPFFFE BP0000 ST
NV UP EI NG NZ NA PO NC
DS-0DAB ES-0DAB SS-0DAB CS-0DAB IP-0102
DS,WORD PTR [DI+02]
DS:0002-4433
0DAB:0102 8E5D02
MOV
AX-000 BX-0000 CX-00000 DX-0000 SP-FFE BP-0000 SI-2211
DI-0000
DS-4433
LES-0DAB SS-0DAB CS-0DAB IP-0105 NV UP EI NG NZ NA PO NC
[BX+SI],AL
DS:221100
ADD
0DAB:0105 000
10 标志位CF有着重要的作用,请简述之,并至少写出三种把CF置0的方法。
1 8位寄存器 与一个数相加 结果不产生进位
2 将 ah 寄存器的最低位值 改为 0 ,执行 SAHF 指令将AH寄存器内容送FLAGS的低字节
3 将 ax 的最低位 置 0 ,在通过 PUSHF 将 ax的值入栈,在通过 POPF 出栈
- 使用今天学的指令,实现三目运算。
Reg == 0 ? 0 : -1
Reg == 1 ? 1 : 0
Reg == 8 ? 9 : 8
Reg == 6 ? 8 : 9
12 分析 test2.exe
TEST2.zip 输出5个星号(粉色)
调试方法: 把 exe 文件 放在 挂载文件目录之下 , debug 文件名即可
首先查看寄存器数据和标志位 此时栈顶 地址 为 E140
只有AX值发生改变 指令是 mov ax,[bx+06] 所以推断出 DF40 + 6 处内存的值位为 FEF0
下一步 CALL 0E16:0000,改变了栈顶
继续执行 MOV AX,0E04 将 ax值 赋值成 0e04 跟 ss 的一样
MOV SS,AX 将 ax值 赋值给 ss 申请栈空间
MOV SP,0000 将sp的值 置0
MOV AX,0E14 将 ax 的值 改成 0e14
MOV DS,AX 将 ax 的值 给 ds
MOV AX,0E16 将 ax 的 值 改成 0e16 跟CS一样
PUSH AX 将 ax 的值 入栈
MOV AX,0026 将 ax 的值 改成 0026
PUSH AX 将 ax 的值 入栈
MOV AX,0E14 再将 ax 的值 改成 0e14 更 DS同值
PUSH AX 将 ax 的值 入栈
MOV AX,0000 再将 ax 的值 改成 0000
PUSH AX 将 ax 的值 入栈
retf 指令是⽤栈中的数据,修改CS和IP的内容,从⽽实现了远转移(就是修改了CS和IP)
ADD [DI],CL
MOV AH,09
ADD [BX],CL
MOV AL,2A
ADD [BX+DI],DL
MOV BH,00
ADD [BP+DI],DL
MOV BL,DA
ADD [DI],DL
MOV CX,0005
ADD [BX+SI],BL
本文作者及来源:Renderbus瑞云渲染农场https://www.renderbus.com
文章为作者独立观点不代本网立场,未经允许不得转载。