80X86经常使用汇编指令集(不断补充完善中)

学习汇编语言,最关键的就在于汇编指令集的掌握以及计算机工做方式的理解,如下是80X86汇编过程当中常常用到的一些汇编指令。html

从功能分类上来讲,一共可分为算法

1、   数据传送指令:MOVXCHGLEALDSLESPUSHPOPPUSHFPOPFCBWCWDCWDE编程

2、   算术指令:ADDADCINCSUBSBBDECCMPMULDIVDAADASAAAAAS数据结构

3、   逻辑指令:ANDORXORNOTTESTSHLSALSHRSARRCLRCRROLROR并发

4、   控制转移指令:JMPJccJCXZLOOPLOOPZLOOPNZLOOPNECALLRETINTpost

5、   串操做指令:MOVSLODSSTOSCMPSSCAS学习

6、   标志处理指令:CLCSTCCLDSTD测试

7、   32CPU新增指令(后续补充并完善)编码

除上述的一些指令外,还有许多3280X86CPU新增指令,这些指令有时会简化程序设计,不过因为我也是刚刚学习汇编,这些都是从书上看到的,因此不少还不是十分了解,我写这些的目的仅仅是想让本身能更好的去记住这些指令的做用和用法,同事也但愿和我同样刚入门的朋友可以多了解一些,并无其余目的,全部的示例也并无通过实际的代码测试,因此但愿各位朋友,无论你喜欢不喜欢,反对不反对,请文明发言,谢谢!spa

------------------------------------------------数据传送指令开始-------------------------------------------------------

一、     MOV(传送)

指令写法:MOV  targetsource

功能描述:将源操做数source的值复制到target中去,source值不变

注意事项:1target不能是CS(代码段寄存器),个人理解是代码段不可写,只可读,因此相应这地方也不能对CS执行复制操做。2targetsource不能同时为内存数、段寄存器(CS\DS\ES\SS\FS\GS3)不能将当即数传送给段寄存器4targetsource必须类型匹配,好比,要么都是字节,要么都是字或者都是双字等。4)因为当即数没有明确的类型,因此将当即数传送到target时,系统会自动将当即数零扩展到与target数的位数相同,再进行传送。有时,须要用BYTE PTR WORD PTR DWORD PTR明确指出当即数的位数

写法示例:MOV  dl,01H;MOV  eax,[bp]; eax =ss:[bp] 双字传送。

二、     XCHG(交换)

指令写法:XCHG object1object2

功能描述:交换object1object2的值

注意事项:1)不能直接交换两个内存数的值 2)类型必须匹配3)两个操做数任何一个都不能是段寄存器【看来段寄存器的写入的限制很是的严格,MOV指令也不能对段寄存器进行写入】,4)必须是通用寄存器(axbxcxdxsidi或内存数

写法示例:XCHG  ax[bx][si]; XCHG ax,bx;

三、     LEA(装入有效地址)

指令写法:LEZ reg16mem

功能描述:将有效地址MEM的值装入到16位的通用寄存器中。

写法示例:假定bx=5678HEAX=1,EDX=2

                              Lea si,2[bx]                    ;si=567AH

                              Lea di,2[eax][edx]       ;di=5

注意,这里装入的是有效地址,并非实际的内存中的数值,若是要想取内存中该地址对应的数值,还须要加上段地址才行,而段地址有可能保存在DS中,也有可能保存在SS或者CS中哦:>不知道个人理解可正确。。。。

四、     LDS\LES\LGS\LSS注意,与LEA不一样的是,这里是装入的值,而不是有效地址

这几个指令,名称不一样,做用差很少。

写法:LDS reg16mem32  

功能描述:reg16等于mem32的低字,而DS对应于mem32的高字(当为LES时,这里就是ES对应于mem32的高字)

用来给一个段寄存器和一个16位通用寄存器同时复制。

注意事项:第一个操做数必须是16位通用寄存器

 

在接着往下说以前,先熟悉下堆栈的概念。堆栈,位于内存的堆栈段中,是内存的一部分,具备“先进后出”的特色,堆栈只有一个入口,即当前栈顶,当堆栈为空时,栈顶和栈底指向同一内存地址,在WINDOWS中,能够把堆栈理解成一个倒着的啤酒瓶,上面的地址大,下面的地址小,当从瓶口往啤酒瓶塞啤酒时(进栈),栈顶就会往瓶口下移动,也就是往低地址方向移动,同理,出栈时,正好相反,把啤酒给倒出来,栈顶向高地址方向移动。这就是所谓的堆栈 ,哼哼,很Easy吧。

在汇编语言中,堆栈操做的最小单位是字,也就是说,只能以字或双字为单位,同时,SSSP指向栈顶(SS为堆栈段寄存器,SP为堆栈指针,两者一相加,就构成了堆栈栈顶的内存地址)

五、     PUSH(进栈)

写法:PUSH reg1632/seg/mem1632/imm

功能描述:将通用寄存器/段寄存器/内存数/当即数的值压入栈中,即:

SP=SP-2 SS:[SP]=16位数值(当将32位数值压入栈中时,SP=SP-4SS:[SP]=32为数值)

六、     POP(出栈)

写法:POP reg1632/seg/mem1632【不能出栈到CS中】

功能描述:将堆栈口的1632)位数据推出到通用寄存器/段寄存器/内存中,即:

寄存器/段寄存器/内存= SS:[SP]   SP=SP+2(当将32位数值出栈时,SP=SP+4(注意,不能出栈给当即数哦,常量不可变嘛)

七、     PUSHAPUSHADPOPAPOPAD

做用:将全部16/32位通用寄存器进栈/出栈

如:PUSHA ;AXCXDXBX、原SPBPSIDI依次进栈。POPA出栈顺序正好相反,但要注意的是,弹出到SP的值被丢弃,SP经过增长16位来恢复(固然嘛,否则栈顶地址就被修改了,就会出息不对齐的状况,就有可能乱套了)

POPAD PUSHAD同样,只不过是32位的罢了。

八、     PUSHFPUSHFDPOPFPOPFD

功能描述:标志寄存器FLAGSEFLAGS)进栈或出栈

如:PUSHF FLAGS进栈  POPF 栈顶字出栈到FLAGS

总结下,POP PUSH一般能够用来交换两个寄存器的值,也能够用来保护寄存器的值,以下:

交换axcx的值:push axpush cxpop ax pop cx

保护寄存器:push axpush cx….中间有不少执行的代码…pop cx;pop ax;

九、LAHF\SAHF(标志寄存器传送指令)

写法:lahf;

做用:AH=FLAGS的低8位

写法:sahf;

做用:FLAGS的低8位=AH

十、符号扩展和零扩展指令

CBW;AL符号扩展为AX

CWD;AX符号扩展为32位数DX:AX

CWDE;AX符号扩展为EAX;

CDQ:EAX符号扩展为64位数EDX:EAX

MOVSX(符号扩展指令的通常形式)

写法:MOVSX reg16\32,reg8\reg16\mem8\mem16

做用:用来将8位符号扩展到16位,或者16位符号扩展到32位

MOVZX(零扩展指令)

写法:MOVZX reg16\32,reg8\reg16\mem8\mem16

零扩展,就是高位补0进行扩展。一般用在将数据复制到一个不一样的寄存器中,如AL零扩展为EBX。相同寄存器的零扩展,可使用MOV 高位, 0来实现。

十一、BSWAP(字节交换)

写法:bswap reg32

做用:将reg32的第0与第3个字节,第1与第2个字节进行交换。

示例:设EAX=12345678h

执行bswap eax;后,eax=78563412H

十二、XLAT(换码)

写法:XLAT;

做用:AL=DS:[bx+AL]

将DS:BX所指内存中的由AL指定位移处的一个字节赋值给AL。(貌似这是一个方便偷懒的指令哦。。),原来它的主要用途是查表。注意能够给它提供操做数,用来指定使用哪一个段地址,如:

XLAT ES:table;使用ES来做为段地址,table不起做用。

XLAT table ;使用table所在段对应的段寄存器做为段地址。

------------------------------------------------ 数据传送指令结束 -------------------------------------------------------

----------------------------------算术指令开始-----------------------------------------------

1三、ADD(加法)

写法:ADD reg/mem reg/mem/imm

做用:将后面的操做数加到前面的操做数中

注意:两个操做数必须类型匹配,而且不能同时是内存操做数

ADC (带进位加法)

写法:ADC reg/mem, reg/mem/imm ;

做用:dest=dest+src+cf

当CF=0时 ADD与ADC的做用是相同的。

示例:实现64位数EDX:EAX与ECX:EBX的加法:

Add EAX,EBX;

ADC EDX,ECX;

1四、INC(自加一)

写法:INC reg/mem;

做用:dest=dest+1;

1五、XADD(交换加)

写法:XADD reg/mem, reg

做用:先将两个数交换,然将两者之和送给第一个数

1六、SUB(减法)

写法:SUB reg/mem, reg/mem/imm;

做用:dest=dest-src;

SBB(带借位减法)

写法:SBB reg/mem, reg/mem/imm

做用:dest=dest-src-cf;

注意:两个操做数必须类型匹配,且不能同时是内存数

1七、DEC(自减1)

写法:DEC reg/mem;

做用:dest=dest-1;

1八、CMP(比较)

写法:CMP reg/mem, reg/mem/imm

做用:dest-src

注意:这里并不将结果存入dest中,而仅仅是执行相减的运算,达到依据运算结果去影响EFLAG标志位的效果

1九、NEG(求补)

写法:NEG reg/mem

做用:求补就是求相反数,即:dest=0-dest;

20、CMPXCHG(比较交换)

写法:CMPXCHG reg/mem, reg;

做用:AL/AX/EAX-oprd1,若是等于0,则oprd1=oprd2,不然,AL/AX/EAX=oprd1;

即:比较AL/AX/EAX与第一个操做数,若是相等,则置ZF=1,并复制第二个操做数给第一个操做数;不然,置ZF=0,并复制第一个操做数给AL/AX/EAX。

说明:CMPXCHG主要为实现原子操做提供支持

CMPXCHG8B(8字节比较交换指令)

写法:CMPXCHG8B MEM64;

功能:将EDX:EAX中的64位数与内存的64位数进行比较,若是相等,则置ZF=1,并存储ECX:EBX到mem64指定的内存地址;不然,置ZF=0,并设置EDX:EAX为mem64的8字节内容

2一、MUL(无符号乘法)

写法:MUL reg/mem;

做用:当操做数为8位时,AX=AL*src;

当操做数为16位时,DX:AX=AX*src;

当操做数为32位时,EDX:EAX=EAX*src;

2二、IMUL(带符号位乘法)

写法:IMUL reg/mem;(做用同上)

IMUL reg16,reg16/mem16,imm16;

IMUL reg32,reg32/mem32,imm32;

IMUL reg16,imm16/reg16/imm16;

IMUL reg32,reg32/mem32/imm32;

注意:没有两个操做数均为8位的多操做数乘法。

对于同一个二进制数,采用MUL和IMUL执行的结果可能不一样,设AL=0FF,BL=1,分别执行下面的指令,会获得不一样的结果:

Mul bl; AX=0FFH(255);

Imul bl; AX=0FFFFH(-1)(高一半为低一半的扩展)

2三、DIV(无符号除法 )/IDIV(带符号数除法)

写法:DIV reg/mem;/IDIC reg/mem

做用:若是操做数是8位,AX%SRC,结果商在AL、余数在AH中;

若是操做数是16位,DX:AX%SRC,结果商在AX,余数在DX中;

若是操做数是32位,EDX:EAX%SRC,结果商在EAX,余数在EDX中;

注意:不能直接实现8位数除8位数、16位数除16位数、32除32,若须要这样,则必须先把除数符号扩展或零扩展到1六、3二、64位,而后用除法指令。

对于IDIV,余数和被除数符号相同,如:-5 IDIV 2 = 商 -2,余数:-1;

在下列状况下,会使CPU产生中断:一:除数为0 ;二:因为商太大,致使EAX\AX或AL不能容纳,从而产生了溢出。

-----------------BCD码调整指令(十进制调整指令)待补充------------------------------------------------

2四、关于BCD码:BCD码就是一种十进制数的二进制编码表示,分为压缩BCD码和非压缩BCD码,压缩BCD码用4个二进制位表示一个十进制位,即用0000B~1001B表示十进制0~9,如0110 0100 0010 1001B表示6429

用8位二进制来表示一个十进制叫非压缩BCD码,其中,低四位与压缩BCD码相同,高四位无心义。

压缩BCD码调整指令包括DAA(加法的压缩BCD码调整)和DAS(减法的压缩BCD码调整)

写法:

DAA;

做用:调整AL中的和为压缩BCD码。

功能:使用DAA指令时,一般先执行ADD/ADC指令,将两个压缩BCD码相加,结果存放在AL中,而后使用该指令将AL调整为压缩BCD码格式。

DAA的调整算法:

IF(AL低4位>9 或 AF=1)

THEN

AL=AL+6;

AF=1;

ENDIF

IF( AL高4位>9或CF=1)

THEN

AL=AL+60H;

CF=1;

ENDIF

说明:CF反映压缩BCD码相加的进位。

DAS;

做用:调整AL中的差为压缩BCD码。

功能:使用DAS指令时,一般先执行SUB/SBB指令,将两个压缩BCD码相减,结果存放在AL中,而后使用该指令将AL调整为压缩BCD码格式。

DAS的调整算法:

IF(AL低4位>9 或 AF=1)

THEN

AL=AL-6;

AF=1;

ENDIF

IF( AL高4位>9或CF=1)

THEN

AL=AL-60H;

CF=1;

ENDIF

说明:CF反映压缩BCD码相减的借位。

特别注意,若是使用DAA或DAS指令,则参加加法或减法运算的操做数应该是压缩BCD码,若是将任意两个二进制数相加或相减,而后调整,则得不到正确的结果。

关键是调整的规则,其中AF标志位就是专门为BCD码调整设计的,当低四位有向高四位进位或借位时,值为1。而CF就是最高位有进位或者借位时,为1.

非压缩BCD码调整指令,包括AAA,AAS,AAM,AAD。

写法:AAA ;

做用:调整AL中的和为非压缩BCD码;调整后,AL高4位等于0,AH=AH+产生的CF

功能:使用AAA指令时,一般先执行ADD/ADC指令,以AL为目的操做数,将两个非压缩BCD码(与高位无关)相加,而后使用AAA将AL调整为非压缩BCD码格式,且高4位等于0,同时,将调整产生的进位加到AH中。

AAA调整算法:

IF(AL低4位>9 或者 AF=1)

THEN

AL=AL+6;

AH=AH+1;

AF=1;

CF=1;

ELSE

AF=0;CF=0;

ENDIF

AL=AL AND OFH;;AL高4位清0

写法:AAS ;

做用:调整AL中的差为非压缩BCD码;调整后,AL高4位等于0,AH=AH-产生的CF

功能:使用AAS指令时,一般先执行SUB/SBB指令,以AL为目的操做数,将两个非压缩BCD码(与高位无关)相减,而后使用AAS将AL调整为非压缩BCD码格式,且高4位等于0,同时,将调整产生的借位从AH中减去。

AAA调整算法:

IF(AL低4位>9 或者 AF=1)

THEN

AL=AL-6;

AH=AH-1;

AF=1;

CF=1;

ELSE

AF=0;CF=0;

ENDIF

AL=AL AND OFH;;AL高4位清0

写法:AAM;

做用:AH=AX DIV 10, AL=AX MOD 10;

功能:使用AAM时,一般先执行MUL/IMUL指令,将两个一字节非压缩BCD码(高四位必须为0)相乘,结果存入AX.而后使用AAM指令将AX(AH=0)调整为两字节压缩BUC码格式。

写法:AAD;

做用:AL=AH*10+AL,AH=0;

功能:使用AAD时,一般先执行该指令,将AX中的两字节非压缩BCD码(AH与AL的高4位必须为0)调整为相应的二进制表示,而后使用DIV/IDIV指令,除以一个一字节的非压缩BCD码(高四位必须为0),可获得非压缩BCD码的除法结果。

特别注意,参加非压缩BCD码乘法或除法的操做数高4位必须为0。

-----------------------------算术指令结束-----------------------------------------------------------------------------

-----------------------------------------位操做指令开始-----------------------------------------------------

2五、AND\OR\XOR\NOT\TEST

写法:

AND reg/mem,reg/mem/imm;

OR reg/mem,reg/mem/imm;

XOR reg/mem,reg/mem/imm;

NOT reg/mem;

TEST reg/mem,reg/mem/imm;

做用:AND\TEST\OR\XOR,两个操做数必须类型匹配,并且不能同时是内存操做数

XOR一般用来将寄存器清0,如 XOR AX,AX;

TEST与AND的关系相似于CMP与SUB。TEST的典型用法是检查某位是否为1,如:

TEST DX,109H;

若 DX的第0,3,8位至少有一位为1,则 ZF=0,不然ZF=1;

2六、移位指令

SHL(逻辑左移)

写法:SHL REG\mem,1\CL ;

做用:将dest的各个二进制位向左移动1(CL)位,并将DEST的最高位移出到CF,最低位移入0

SAL(算术左移)

写法:SAL REG\mem,1\CL ;

做用:将dest的各个二进制位向左移动1(CL)位,并将DEST的最高位移出到CF,最低位移入0(同SHL)。

SHR(逻辑右移)

写法:SHR REG\mem,1\CL ;

做用:将dest的各个二进制位向左移动1(CL)位,并将DEST的最低位移出到CF,最高位移入0

SAR(算术右移)

写法:SAR REG\mem,1\CL ;

做用:将dest的各个二进制位向左移动1(CL)位,并将DEST的最低位移出到CF,最高位不变

SHLD(双精度左移)

写法:SHLD REG16/REG32/MEM16/MEM32, REG16/REG32, IMM8/CL;(类型须匹配)

做用:将OPRD1的各二进制左移,并将oprd1的最高位移到CF,oprd2的最高位移到oprd1的最低位,可是,oprd2的值不变

SHRD(双精度右移)

写法与做用与双精度左移相似。注意移动方向为右移。

以上位移指令对标志位的影响:

若移位后符号位发生了变化,则OF=1,不然OF=0;CF为最后移入位;按通常规则影响ZF与SF。然而,若移位次数为0,则不影响标志位;若移位次数大于1,则OF无定义

2七、循环移位指令

ROL(循环左移)

写法:ROL REG\MEM, 1\CL;或 ROL REG/MEM,IMM8;(类型可不匹配)

做用:将DEST的各二进制位向左移动,并将最高位移出到CF,并同时移入最低位

ROR(循环右移)

写法:ROR REG\MEM, 1\CL;或 ROR REG/MEM,IMM8;(类型可不匹配)

做用:将DEST的各二进制位向右移动,并将最低位移出到CF,并同时移入最高位

RCL(带进位循环左移)

写法:RCL REG\MEM, 1\CL;或 RCL REG/MEM,IMM8;(类型可不匹配)

做用:将DEST的各二进制位向左移动,并将最高位移出到CF,原CF移入最低位

RCR(带进位循环右移)

写法:RCR REG\MEM, 1\CL;或 RCR REG/MEM,IMM8;(类型可不匹配)

做用:将DEST的各二进制位向右移动,并将最低位移出到CF,原CF移入最高位

2八、位测试指令

BT(位测试)

写法:BT REG16/MEM16,REG16/IMM8;或BT REG32/MEM32,REG32/IMM8;

做用:CF=DEST的第index位,dest不变。

BTS(位测试并置位)

写法:BTS REG16/MEM16,REG16/IMM8;或BTS REG32/MEM32,REG32/IMM8;

做用:CF=DEST的第index位,dest的第index位=1;

BTR(位测试并复位)

写法:BTR REG16/MEM16,REG16/IMM8;或BTR REG32/MEM32,REG32/IMM8;

做用:CF=DEST的第index位,dest的第index位=0;

BTC(位测试并复位)

写法:BTC REG16/MEM16,REG16/IMM8;或BTC REG32/MEM32,REG32/IMM8;

做用:CF=DEST的第index位,dest的第index位取反;

说明:若dest为寄存器,则以index除以16(dest为reg16)或32(dest为reg32)的余数做为测试位。固然,index最好不要超出操做数的位数。

若dest为内存操做数,则不管其类型为字或双字,测试位为相对于起始地址的位移,例如,设BX=50,X为字类型的变量,则执行指令BT X,BX;后,CF=X+6单元的第2位,由于50%8=6余2.

BTS、BTC、BTR指令可用于并发程序设计。

2九、位扫描指令

BSF(前向位扫描)

写法:BSF reg16/reg32, reg16/reg32/mem16/mem32;(类型须匹配)

做用:dest=src中值为1的最低位编号(从低位向高位搜索)

BSR(后向位扫描)

写法:BSR reg16/reg32, reg16/reg32/mem16/mem32;(类型须匹配)

做用:dest=src中值为1的最高位编号(从高位向低位搜索)

说明:BSF和BSR搜索SRC操做数中首次出现1的位置,BSF从低位向高位搜索,BSR反之。若找到一个1,则置ZF=0,并存储位编号到DEST操做数中。若SRC=0,即没有1出现,则置ZF=1,且dest的值不肯定。

好比,有以下二进制数0111 1111 1010 0100

执行bsf后,位编号为2,执行bsr后,位编号为14.

30、条件置位指令

通用写法:SETcc reg8/mem8

做用:若条件cc成立,则dest=1,不然,dest=0;

SETcc有不少种命令形式,这里的cc只是一个描述符,具体的参见下面的三个表,其中,E(Equal)表示相等,G(Greatet)表示带符号大于,L(Less)表示带符号小于,A(Above)表示无符号大于,B(Below)表示无符号小于。

表一:测试单个标志位的SETcc指令

SETcc指令

描述

置1条件

SETC,SETB,SETNAE

有进位时置1

CF=1

SETNC,SETNB,SETAE

无进位时置1

CF=0

SETZ,SETE

为0(相等)时置1

ZF=1

SETNA,SETNE

非0(不等)时置1

ZF=0

SETS

为负时置1

SF=1

SETNS

为正时置1

SF=0

SET0

溢出时置1

OF=1

SETNO

不溢出时置1

OF=0

SETP,SETPE

‘1’的个数为偶数时置1

PF=1

SETNP,SETPO

‘1’的个数为奇数时置1

PF=0

表二:用于带符号数比较的SETcc指令,这些指令经常使用在CMP指令以后,以判断带符号数的大小

SETcc指令

描述

置1条件

SETG,SETNLE

大于(不小于等于)时置1

SF=OF且ZF=0

SETGE,SETNL

大于等于(不小于)时置1

SF=OF

SETL,SETNGE

小于(不大于等于)时置1

SF≠OF

SETLE,SETNG

小于等于(不大于)时置1

SF≠OF或ZF=1

表三:用于无符号数比较的SETcc指令,经常使用在CMP指令以后,用来判断无符号数的大小

SETcc指令

描述

置1条件

SETA,SETNBE

大于(不小于等于)时置1

CF=0且ZF=0

SETAE,SETNB,SETNC

大于等于(不小于)时置1

CF=0

SETB,SETNAE,SETC

小于(不大于等于)时置1

CF=1

SETBE,SETNA

小于等于(不大于)时置1

CF=1或ZF=1

-----------------------------------------位操做指令结束----------------------------------------------------

----------------------------------------------------

---------------------------控制转移指令开始--------------------------------------------------

3一、JMP(无条件转移指令)

执行代码的跳转,分为两种,一:段内转移,即要跳过去的代码地址和当前地址在同一段,这时只要修改IP(专用寄存器--指令指针)便可;二:段间转移:即要跳过去的代码地址和当前代码地址不在同一段内,须要同时修改CS和IP的值。

写法:

一、JMP label;若label与该指令位于同一代码段内,IP=label的偏移地址,不然CS:IP=label的分段地址,简单的说,就是跳到label的地址去。

二、JMP reg16/mem16;段内转移,偏移地址=reg16/[mem16]

三、JMP mem32;段间间接转移,段地址CS=mem32高字,偏移地址IP=mem32低字。

说明:当操做数是内存操做数时,若内存操做数是双字类型,则产生段间转移,若内存操做数是字类型,则产生段内间接转移。当不能肯定类型时,编译器将报错

3二、Jcc(条件转移指令)

写法:Jcc label;

做用:若条件成立,则IP=label的偏移地址,不然,CPU将忽略该条件转移,继续执行下一条指令

条件转移有如下几种形式:

表一:测试单个标志位的Jcc指令:

Jcc指令

描述

转移条件

JC,JB,JNAE

有进位时转移

CF=1

JNC,JNB,JAE

无进位时转移

CF=0

JZ,JE

为零(相等)时转移

ZF=1

JNZ,JNE

非零(不等)时转移

ZF=0

JS

为负时转移

SF=1

JNS

为正时转移

SF=0

JO

溢出时转移

OF=1

JON

不溢出时转移

OF=0

JP,JPE

‘1’的个数为偶数时转移

PF=1

JNP,JPO

‘1’的个数为奇数时转移

PF=0

表二:用于带符号数比较的Jcc指令(经常使用在CMP指令以后,以判断带符号数的大小)

Jcc指令

描述

转移条件

JG,JNLE

大于(不小于等于)时转移

SF=OF且ZF=0

JGE,JNL

大于等于(不小于)时转移

SF=OF

JL,LNGE

小于(不大于等于)时转移

SF<>OF

JLE,LNG

小于等于(不大于)时转移

SF<>OF或ZF=1

表三:用于无符号数比较的Jcc指令(经常使用在CNO指令以后,以判断无符号数的大小)

Jcc指令

描述

转移条件

JA,JNBE

大于(不小于等于)时转移

CF=0且ZF=0

JAE,JNB,JNC

大于等于(不小于)时转移

CF=0

JB,LNAE,JC

小于(不大于等于)时转移

CF=1

JBE,LNA

小于等于(不大于)时转移

ZF=1或CF=1

3三、JCXZ/JECXZ(Jump if CX/ECX is zero)

写法:JCXZ label;(若CX=0,则转移到label)

JECXZ label;(若ECX=0,则转移到label)

说明:label相对位移量必须在-126~127之间

 

3四、循环指令

LOOP label;

做用:CX=CX-1;若CX<>0,则转移到label;

LOOPZ/LOOPE label;

做用:CX=CX-1;若CX<>0且 ZF=1,则转移到label;

LOOPNZ/LOOPNE label;

做用:CX=CX-1;若CX<>0且ZF=0,则转移到label;

说明:label相对位移量必须在-128~127之间

3五、过程调用和返回指令

CALL(过程调用)

写法:CALL label;

做用:若label与该指令在同一代码段,则为段内直接调用,IP进栈,IP=label的偏移地址,若是是不在同一代码段,则为段间间接调用,CS:IP进栈,CS:IP=label的分段地址

写法:CALL reg16/mem16;

做用:段内间接调用,IP进栈,IP=reg16/【mem16】

写法:CALL mem32;

做用:段间间接调用,CS:IP进栈,CS等于mem32高字,ip等于mem32低字。

该指令与JMP指令的区别就是保存了CS:IP的值,这样在调用指令结束后,能够返回回来而已。

RET(过程返回)

写法:RET; 近返回或远返回

RETN; 近返回;

RETF; 远返回

RET imm16; 近返回或远返回,并调整堆栈,SP=SP+imm16;

RETN imm16;近返回,并调整堆栈,SP=SP+imm16;

RETF imm16;远返回,并调整堆栈,SP=SP+imm16;

做用:RET/RETN/RETF:返回地址出栈,从而使调用返回,其中,远返回是POP一个双字到CS:IP,而近返回是POP一个字到IP

RET/RETN/RETF imm16:在返回后,CPU当即将imm16加到堆栈指针SP。这种机制用来在返回前将参数从栈中移除。

说明:CALL 与 RET必须配合使用,而且确保返回时栈顶正好是返回地址,否则就会出错。

3六、IN T(中断指令)

写法:INT n;(n为中断号,取值为0~255)

一般,程序内部的跳转,用JMP或CALL,而且JMP和CALL得参数是要跳转的过程的入口指令地址,而INT则是调用系统提供的中断服务程序,而且参数是中断号,而后由CPU根据中断号去计算中断服务程序的入口地址,MS DOS使用中断号21H做为系统调用,通常INT中断的步骤以下:

(1)由AH给出中断号

(2)根据相应功能的要求,设置入口参数

(3)INT 21H

(4)分析和使用出口参数

好比以下代码实现程序的退出并返回DOS:

Mov ah,4ch;-----给出中断号

Int 21h; -----开始中断

说明:除了直接以AL或AX返回出口参数外,INT 21H仍是用AL或AX做为返回码,对于功能号0~2eh,由AL返回0(表示成功)或1(表示失败);其他功能号则由CF返回0或者1,并由AX返回错误码。

---------------------------控制转移指令结束--------------------------------------------------

-----------------------------标志处理指令开始-----------------------------------------------------------

3七、标志处理指令

CLC ; CF=0

STC ; CF=1

CMC ; CF=NOT CF

CLD ; DF=0

STD ; DF=1

CLI ; IF=0(应慎用)

STI ; IF=1

-----------------------------标志处理指令结束-----------------------------------------------------------

---------------------------串操做指令开始--------------------------------------------------

到这为止,所涉及的指令都是处理一个操做数,若是要处理连续内存单元的一批数据,一般需借助于循环。而串操做指令就能够用来处理内存中的数据串,并在助记符后面加上B、W、D分别表示操做类型为字节、字或双字

3八、MOVS(串传送)

写法:

MOVSB/MOVSW/MOVSD

功能:

ES:[DI]=DS:[SI]

If(DF=0)

Then

SI=SI+size;

DI=DI+size;

Else

SI=SI-size;

DI=DI-size;

Endif

其中,size等于1(B)、2(W)、4(D).

做用:将DS:SI所指源串的一个字节/字/双字复制到ES:DI所指的内存单元,而后,若DF=0,则SI和DI增长一、二、4,不然减小一、二、4.

如今有点明白为何SI为源变址寄存器,而DI为目标变址寄存器了,而DS为数据段寄存器,ES为附加段寄存器了

 

3九、LODS(串载入)

写法:LODSB\LODSW\LODSD

功能:

AL/AX/EAX=DS:[SI];

IF (DF=0) THEN

SI=SI+size;

ELSE

SI=SI-size;

Endif

做用:将DS:SI所指源串的值复制到AL/AX/EAX中,而后,根据DF使SI增长或减少一、二、4

40、STOS(串存储)

写法:

STOSB\STOSW\STOSD

功能:

ES:[DI]=AL/AX/EAX;

IF (DF=0) THEN

DI=DI+size;

ELSE

DI=DI-size;

ENDIF

做用:将AL/AX/EAX中的值复制到ES:[DI]所指的内存单元中去,并根据DF标志位的值调整DI

4一、CMPS(串比较)

写法:CMPSB/CMPSW/CMPSD

功能:

DS:[SI]-ES:[DI];

IF (DF=0) THEN

SI=SI+size;DI=DI+size;

ELSE

SI=SI-size;DI=DI-size;

ENDIF

做用:将DS:SI所指内存值与ES:DI所指内存值进行比较,并根据比较结果设置标志位,而后,对SI和DI作相应的调整。

4二、SCAS(串扫描)

写法:SCASB/SCASW/SCASD

功能:

AL/AX/EAX-ES:[DI];

IF (CF=0) THEN

DI=DI+size;

ELSE

DI=DI-size;

ENDIF

做用:将AL/AX/EAX与ES:DI所指内存值进行比较,根据比较结果设置标志位,而后根据DF调整相应的DI的值。

说明:以上串操做的共性:

DS:SI指向源串,ES:DI指向目的串

SI和DI自动增长或减小一、二、4,关键看DF及操做类型是B\W\D

4三、重复前缀

重复前缀用来和以上几个串操做指令混合使用

REP(重复)

功能:当CX<>0时,重复执行后面的串指令,每执行一次,CX自动-1,该指令只能用在MOVS\LODS\STOS以前

REPZ/REPE(为零/等于时重复)

功能:当CX<>0且ZF=1时,重复执行后面的指令,每执行一次,CX自动-1,该指令只能用在CMPS\ACAS以前。

REPNZ/REPNE(非零/不等于时重复)

功能:CX<>0且ZF=0时,重复执行后面的指令,每执行一次,CX自动-1,该指令只能用在CMPS\ACAS以前。

说明:REPNE SCAS(B/W/D)适用于在多字节、字、双字数据结构中搜索特定值。

---------------------------串操做指令结束--------------------------------------------------

---------------------------CPU控制指令开始--------------------------------------------------

4四、NOP(无操做)

写法:NOP;

做用:该指令不作任何事情,只占用1个字节,耗费一个指令执行周期。

4五、HIT(暂停)

写法:HIT;

做用:HIT使CPU进入暂停状态,这时CPU不执行任何操做,直到系统复位或发生外部中断为止,中断使CPU继续执行后面的指令(貌似和屏保或待机的功能相似)

4六、LOCK(封锁前缀)

功能:LOCK指令用于多处理器系统,做为某些指令的前缀,可使CPU经过锁住总线等方式,抱着指令做为原子性操做,即:指令执行过程不会被打断操做。

该指令用于如下指令的前缀时,以保证原子性的对内存的“读-修改-写”操做:

1) 加法:ADD\ADC\INC\XADD

2) 减法:SUB\SBB\DEC\NEG

3) 交换:XCHG\CMPXCHG\CMPXCHG8B

4) 逻辑:AND\NOT\OR\XOR

5) 位测试:BTS\BTC\BTR

说明:其余类型指令不能加LOCK前缀,另外,XCHG老是原子性操做,不管前面有没有加LOCK前缀。LOCK前缀典型用于BTS指令,以实现多处理器环境中程序的并发执行,如:

LOCK BTS [EBX],AX

LOCK ADD [SI],AL

---------------------------CPU控制指令结束--------------------------------------------------

到这里为止,一些基本的汇编指令都已经学习完了,可是还得好好的去应用,否则还真的记不住这些指令的功能。

总结一下,通常状况下,通用寄存器能够较随便使用,段寄存器和指针寄存器用来指示位置,通常不能随便更改,另一个就是标志寄存器的各个标志位的意义也很是的重要,不少指令都是根据标志位来执行操做的。

接下来准备学习汇编的编程格式,而后就能够写一些简单的程序并本身进行调试了 ,哈哈哈哈哈。偶很是的期待。

转载于:https://www.cnblogs.com/itjin45/archive/2009/09/02/1559129.html