⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 汇编002.txt

📁 会变语言实现的一些程序
💻 TXT
📖 第 1 页 / 共 2 页
字号:
在接下来的四篇里将介绍80386的汇编指令及用法,并和8086的指令进行比较。
   80386的指令集包含了8086/8088,80186,80286的指令集,可以分为几个大类:数据传送指令,算术运算/逻辑运算指令,移位指令,控制转移指令,串操作指令,高级语言支持的指令,条件字节设置指令,位操作指令,处理器控制指令和保护方式指令。高级语言支持指令始于80186,保护方式指令始于80286,条件字节设置指令和位操作指令是80386新增的。
  本篇主要介绍数据传送指令,数据传送指令可以分为:通用数据传送,累加器专用传送,地址传送,标志传送,分别介绍如下:
  A.数值传送指令MOV,MOVZX,MOVSX,XCHG,PUSH,PUSHA,PUSHAD,POPA,POPAD,
  a.MOV,指令和8086相似,不过它支持32位操作。
  b.MOVZX,零扩展传送,格式--MOVZX DST,SRC,表示将源操作送给目的操作数,目的操作数空出的部分用0填补。
  c.MOVSX,符号扩展传送,格式--MOVSX DST,SRC,表示将源操作送给目的操作数,目的操作数空出的部分用SRC的符号位来填补,举个简单的例子来演示:
  MOV DL,90H;
  MOVSX AX,DL;AX=FF90H
  MOVZX AX,DL;AX=0090H 
  MOVSX ESI,DL;ESI=FFFFFF90H
  MOVZX ESI,DL;ESI=00000090H
  事实上在8086中也有两条指令CBW,CWD可以对操作数进行扩展。MOVSX可以对有符号数进行扩展,MOVZX可以对无符号数进行扩展,看看CBW,CWD的用法:
   CBW将字节数据扩展成字,符号位扩展到AH中
   CWD将字数据扩展成双字,符号位放到DX中
   MOV AL,70H;
   CBW;//AX=0070
   CWD;//DX=0000,AX=0070
  d.XCHG,功能和8080相同,不过它支持8位,16位,32位操作,下面的语句均是合法的。
   XCHG AH,AL
   XCHG AX,AL
   XCHG ESI,EDI
   XCHG ESI,[EBX+EDI+1000H]
  e.PUSH,和8086不同的是,它支持立即数入栈,8位入栈,当然还有32位入栈,下面的语句均是合法的。
   PUSH AL
   PUSH BH
   PUSH 100H
   PUSH EAX
   PUSH EBX
   PUSH DWORD PTR [EAX]
  f.POP,功能和用法和8086一样。
  g.PUSHA,将8个通用寄存器全部进栈,进栈顺序为:AX,CX,DX,BX,SP,BP,SI,DI,然后SP指针寄存减16,不过SP入栈的内容是PUSHA指令执行前的内容。
  h.POPA,8个通用寄存器全部出栈,堆栈指针寄存器不是堆栈中弹出的内容,而是加16而得到的,虽然这样得到的值和从堆栈中弹出来的内容一样,但物理意义不一样。
  i.PUSHAD,将8个32位通用寄存器全部入栈,入栈顺序EAX,ECX,EDX,EBX,ESP,EBP,ESI,EDI,ESP的内容是执行指令PUSHAD之前的内容
  j.POPAD,8个32位通寄存器全部出栈,ESP的内容参见h
  
  B.地址传送指令LEA,LDS,LES,LFS,LGS,LSS
  a.LEA,取有效地址,功能,用法与8086相同,不过它支持32位操作。规则:目的操作必须是16位或32位通用寄存器,当目的操作数是16位时,那么只装入有效地址的低16位。事实上LEA指令相当于伪指令OFFSET,看例子:
  MOV EAX,12345678H
  MOV EBX,56784321H
  LEA ECX,[EAX+EBX];ECX=99999999H
  b.LDS,装入指针,功能,用法与8086相同,不过它支持32位操作。格式:LDS REG,OPRD。规则,目的寄存器必须是16位或32位的通用寄存器,OPRD必须是内存单元,不可以是立即数。如果目的寄存器是16位,那么源操作数OPRD含32位指针;如果目的寄存器是32位,那么源操作数有48位指针。该指令将目的操作数OPRD所指向的内存单存的4个或6个连续字节的内容送给助记符指令中指定的DS段寄存器和指令中目的寄存器。比如:
  LDS EAX,[1000H];这表明将偏移地址为1000,1001H这两个字节单元的内容送给段寄存器DS,将偏移地址1002,1003,1004,1005四个字节单元的内容送往EAX。
  LDS AX,[1000H];这表明将偏移地址为1000,1001H这两个字节单元的内容送给段寄存器DS,将偏移地址1002,1003H两个字节单元的内容送往EAX。
  c.LES,同LDS,不过段寄存器是ES。
  d.LFS,同LDS,不过段寄存器是FS。
  e.LGS,同LDS,不过段寄存器是GS。
  h.LSS,同LDS,不过段寄存器是SS。
  
  C.标志传送指令LAHF,SAHF,PUSHF,PUSHFD,POPF,POPFD
  a.LAHF,将标志寄存器的低8位送至AH中,包括SF,ZF,ZF,PF,CF。
  b.SAHF,与i的过程恰好相反
  c.PUSHF,将标志寄存器的EFLAGS低16位内容入栈,和8086相同
  d.PUSHFD,将标志寄存器EFLAGS的内容入栈
  e.POPF,将栈顶的一个字弹出,并将它送到标志寄存器EFLAGS的低16位
  f.POPFD,将栈顶的两个字弹出,并将它送到标志寄存器EFLAGS
  
  D.累加器传送指令IN,OUT,XLAT
  a.IN,和8086相同,但可以输入一个双字节,同样如果端口的范围位于00H-FFH,可以直接用,如果超出这个范围,则先要将端口号送至DX,下面的语句是合法的:
  IN AL,20H;从20H端口读入一个字节
  IN AX,20H;从20H端口读入一个字
  MOV DX,0378H
  IN EAX,DX;从20H端口读两个字节
  b.OUT,和8086相同,但可以输出一个双字节,同样如果端口的范围位于00H-FFH,可以直接用,如果超出这个范围,则先要将端口号送至DX,下面的语句是合法的:
  OUT 20H,AL;从20H端口输出一个字节
  IN 20H,AX;从20H端口输出一个字
  MOV DX,0378H
  IN EAX,DX;从20H端口输出两个字
  c.XLAT,查表指令,功能和用法与8086相同,不过基址寄存器用的是EBX,来看看XLAT的实现过程:XLAT以BX作为基址寄存器,以AL作为变址寄存进器对指定的缓冲区进行查表,将AL指定位置的内容送往AL,比如说我们在MS-DOS方式写一个小程序:
  C:\>Debug
  -A100
     MOV BX,0120
     SUB AL,AL
     MOV DL,AL
     MOV AH,2
     INT 21
     MOV AH,4C
     INT 21
     INT 20
-E120 'ABCDEFGHIJKLLMMDDKDJDK'
=G100
   屏幕上会显示A,如果AL=3,那么屏幕会显示D

   以上所有的指令均不影响EFLAGS的各标志位。

  算术运算指令,逻辑运算指令,移位指令
   AA.算术运算指令
   A.加减法运算ADD,ADC,INC,SUB,SBB,DEC,CMP,NEG
   a.ADD,和8086功能,用法相同,不过支持32位操作,下面的语句都是合法的。
   ADD  ESI,EDI
   ADD  EAX,DWORD  PTR  [1000H]
   b.ADC,带进位的加法指令,即OPRDS+OPRDD+CF,其中OPRDS代表源操作数,OPRDD代表目的操作,CF代表进位标志位,功能和用法与8086相同,支持32位操作。
   c.SUB,和8086相同,支持32位操作。
   d.SBB,带进位的减法指令,即OPRDD-OPRDS-CF,其中OPRDS代表源操作数,OPRDD代表目的操作数,CF代表进位标志位,功能和用法与8086相同,支持32位操作。
   e.DEC,减1操作,功能和用法与8086相同,支持32位操作。
   f.CMP,比较操作,功能和用法与8086相同,支持32位操作。
   g.NEG,求补操作,功能和用法与8086相同,支持32位操作。
   h.INC  加1操作,功能和用法与8086相同,支持32位操作。
   
   B.乘除法指令MUL,DIV,IMUL,IDIV
   a.MUL,无符号数乘法指令,和8086功能用法一样,即指令中只给出一个操作,被乘数已默认,如果指令给出的操作数是32位的话,被乘数默认为EAX,那么乘积将存放在EDX:EAX中,其中EDX存放高32位,EAX存放低32位,如果此时EDX=0,即高32位为0的话,那么OF=0,CF=0,否则被置1。如果指令给出的操数作是16位的话,被乘数默认为AX那么乘积将放在DX:AX中,其中DX中将存放高16位,AX中存放低16位。如果指令给出的操作数是8位的话,被乘数默认为AL,那么乘积将放在AX,AH中存放高8位,AL中存放低8位。
  b.DIV,无符号数的除法指令,和8086一样,指令给出一个操作数,被除数已默认。如果指令中给出的操作数为32,那么被除数将是EDX:EAX, 最终的商将存放在EAX, 余数将存放在EDX中。如果指令给出操作数为16位,那么被除数为EAX,最终得到的商放在AX,余数放在EAX的高16位。如果指令中给出的操作数为8位,那么被除数是16位,最终得到的商将放在AL中,余数放在AH中。
  c.IMUL,有符号数的乘法指令,除了具有8086的用法外,有新的形式:
    c1.IMUL DST,SRC;将源操作数SRC与目的操作DST相乘,并将结果送往DST。
    c2.IMUL DST,SRC1,SRC2;将源操作数SRC1与源操作数SRC2相乘,并将结果送往DST。
    使用这种形式必须遵守的规则,形式c1指令中目的操作数必须是16位或32位通寄存器,源操作数的长度必须与目的操作的长度一样(8位立即数除外,即00H-FFH或80H-7FH),源操作数可以是通用寄存器,也可以是存储单元或立即数。形式c2指令中的源操作数SRC1可以是通用寄存器也可以是存储单元,源操作数SRC2必须是立即数,DST必须是16位或32位通用寄存器。呵呵,对于这些规则无需去问为什么,这是硬件的特性决定的,如果一定要问为什么,那只能问INTEL公司的硬件工程师了:)。同时,有一点要注意的是:这两种形式的指令,目的寄存器的长度与源操作数长度一样(8位立即数除外),这样的话,该指令事实上对有符号数和无符号数是一样的,因为乘积的低位部分均存储在目的寄存器中,而高位部分在这两种形式的指令中不予以存储。
  d.IDIV,有符号数的除法指令,用法和8086相同,不过支持32位操作。   

  C.符号扩展指令CBW,CWD,CWDE,CDQ

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -