nasm例子
objdump反汇编
gasm与nasm比较
8086汇编指令集
8086汇编指令集
一、数据传输指令
它们在存贮器和寄存器、寄存器和输入输出端口之间传送数据.
通用数据传送指令.
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15MOV 传送字或字节.
MOVSX 先符号扩展,再传送.
MOVZX 先零扩展,再传送.
PUSH 把字压入堆栈.
POP 把字弹出堆栈.
PUSHA 把AX,CX,DX,BX,SP,BP,SI,DI依次压入堆栈.
POPA 把DI,SI,BP,SP,BX,DX,CX,AX依次弹出堆栈.
PUSHAD 把EAX,ECX,EDX,EBX,ESP,EBP,ESI,EDI依次压入堆栈.
POPAD 把EDI,ESI,EBP,ESP,EBX,EDX,ECX,EAX依次弹出堆栈.
BSWAP 交换32位寄存器里字节的顺序
XCHG 交换字或字节.(至少有一个操作数为寄存器,段寄存器不可作为操作数)
CMPXCHG 比较并交换操作数.(第二个操作数必须为累加器AL/AX/EAX)
XADD 先交换再累加.(结果在第一个操作数里)
XLAT 字节查表转换.----BX指向一张256字节的表的起点,AL为表的索引值(0-255,即0-FFH);
返回AL为查表结果.([BX+AL]->AL)输入输出端口传送指令.
1
2IN I/O端口输入. ( 语法: IN 累加器, {端口号│DX} )
OUT I/O端口输出. ( 语法: OUT {端口号│DX},累加器 )输入输出端口由立即方式指定时, 其范围是 0-255; 由寄存器 DX 指定时,其范围是 0-65535.目的地址传送指令.
1
2
3
4
5
6LEA 装入有效地址.例: LEA DX,string ;把偏移地址存到DX.
LDS 传送目标指针,把指针内容装入DS.例: LDS SI,string ;把段地址:偏移地址存到DS:SI.
LES 传送目标指针,把指针内容装入ES.例: LES DI,string ;把段地址:偏移地址存到ES:DI.
LFS 传送目标指针,把指针内容装入FS.例: LFS DI,string ;把段地址:偏移地址存到FS:DI.
LGS 传送目标指针,把指针内容装入GS.例: LGS DI,string ;把段地址:偏移地址存到GS:DI.
LSS 传送目标指针,把指针内容装入SS.例: LSS DI,string ;把段地址:偏移地址存到SS:DI.标志传送指令.
1
2
3
4
5
6LAHF 标志寄存器传送,把标志装入AH.
SAHF 标志寄存器传送,把AH内容装入标志寄存器.
PUSHF 标志入栈.
POPF 标志出栈.
PUSHD 32位标志入栈.
POPD 32位标志出栈.
二、算术运算指令
1 | ADD 加法. |
三、逻辑运算指令
1 | AND 与运算. |
四、串指令
DS:SI 源串段寄存器 :源串变址.
ES:DI 目标串段寄存器:目标串变址.
CX 重复次数计数器.
AL/AX 扫描值.
D标志 0表示重复操作中SI和DI应自动增量; 1表示应自动减量.
Z标志 用来控制扫描或比较操作的结束.
1 | MOVS 串传送.( MOVSB 传送字符. MOVSW 传送字. MOVSD 传送双字. ) |
五、程序转移指令
无条件转移指令 (长转移)
1
2
3JMP 无条件转移指令
CALL 过程调用
RET/RETF 过程返回.条件转移指令 (短转移,-128到+127的距离内)( 当且仅当(SF XOR OF)=1时,OP1<OP2 )
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20JA/JNBE 不小于或不等于时转移.
JAE/JNB 大于或等于转移.
JB/JNAE 小于转移.
JBE/JNA 小于或等于转移.
以上四条,测试无符号整数运算的结果(标志C和Z).
JG/JNLE 大于转移.
JGE/JNL 大于或等于转移.
JL/JNGE 小于转移.
JLE/JNG 小于或等于转移.
以上四条,测试带符号整数运算的结果(标志S,O和Z).
JE/JZ 等于转移.
JNE/JNZ 不等于时转移.
JC 有进位时转移.
JNC 无进位时转移.
JNO 不溢出时转移.
JNP/JPO 奇偶性为奇数时转移.
JNS 符号位为 "0" 时转移.
JO 溢出转移.
JP/JPE 奇偶性为偶数时转移.
JS 符号位为 "1" 时转移.循环控制指令(短转移)
1
2
3
4
5LOOP CX不为零时循环.
LOOPE/LOOPZ CX不为零且标志Z=1时循环.
LOOPNE/LOOPNZ CX不为零且标志Z=0时循环.
JCXZ CX为零时转移.
JECXZ ECX为零时转移.中断指令
1
2
3INT 中断指令
INTO 溢出中断
IRET 中断返回处理器控制指令
1
2
3
4
5
6
7
8
9
10
11
12HLT 处理器暂停, 直到出现中断或复位信号才继续.
WAIT 当芯片引线TEST为高电平时使CPU进入等待状态.
ESC 转换到外处理器.
LOCK 封锁总线.
NOP 空操作.
STC 置进位标志位.
CLC 清进位标志位.
CMC 进位标志取反.
STD 置方向标志位.
CLD 清方向标志位.
STI 置中断允许位.
CLI 清中断允许位.
六、伪指令
1 | DW 定义字(2字节). |
七、处理机控制指令:标志处理指令
1 | CLC 进位位置0指令 |
AT&T汇编指令enter, leave, call, ret
assembly/nasm生成程序
assembly/汇编intel与att
汇编intel与att
汇编有两种语法标准, 分别是intel, att语法
intel语法对应, intel x86_x64体系, x86系列CPU上
att语法对应, gcc objdump 默认格式, 多种CPU上可用(x86, power, VAX)
编译器有
- gasm: .s文件
- nasm: .asm文件
- masm: .asm文件
- fasm: .asm文件
intel att 区别
https://blog.csdn.net/goodcrony/article/details/92794938 ATT与Intel汇编代码格式
指令名
Intel, 内存操作数前用前缀byte prt
, word ptr
, dword ptr
表大小
ATT, 指令都有字符后缀b, w, l
, 表内存操作数的大小
Intel是
mov al, byte ptr foo
AT&T是movb $foo, %al
movb
传送字节 (1 B = 8 bit)movw
传送字 (2 B = 16 bit)movl
传送双字 (4 B = 32 bit)
因为在许多机器上, 32位数都称为长字(long word)
指令顺序
Intel右->左
ATT左->右
Intel是
mov dst, src
ATT是mov src, dst
寄存器名字, Intel不用前缀%
Intel是
esp
ATT是%esp
寄存器中值 = 地址
Intel是
dword ptr [ebp+8]
ATT是8(%ebp)
立即数, Intel不用前缀$
Intel十六, 二进制后缀h
和b
ATT十六进制立即数前缀0x
十进制数
Intel是123
ATT是$123
十六进制数
Intel是123h
ATT是$0x123
ATT直接跳转, (跳转目标: 指令内一部分编码)
jmp Label_1
ATT间接跳转, (跳转目标: 寄存器R, 存储器M中的值[代码地址])
‘*’后为操作数
jmp *%eax
寄存器%eax中的值 = 跳转目标jmp *(%eax)
%eax中的值 = 存储器M地址, 存储器M中的值 = 跳转目标
远跳转和远调用
ATT是
ljmp/lcall section,section,offset
Intel是jmp/call far section:offset
远返回
ATT是
lret $stack-adjust
Intel是ret far stack-adjust
Intel可多代码段程序
ATT不用多代码段程序, UNIX要求所有代码一个段seg
gasm
已有c文件
1 | #include <math.h> |
对应gcc, 通过c文件, 生成s文件
1 | dassein@pad:~/nasm_pjt$ gcc -E area_2019_sep_24.c > area_2019_sep_24.i |
z直接通过c生成可执行文件
1 | gcc -c area_2019_sep_24.c -o area_2019_sep_24 |
生成s文件
1 | .file "area_2019_sep_24.c" |
nasm
用c2nasm.bash反汇编nasm已有c文件1
./c2nasm.bash area_2019_sep_24.c
自动生成area_2019_sep_24.asm
1 | default rel |
附录: c2nasm.bash
先把objconv程序放到/bin中, 供cmd使用obconv
1 | #!/bin/bash |