【汇编语言】

汇编语言 第二版  王爽

【Chapter 01 基础知识】

学习汇编的主要目的:通过用汇编语言进行编程而深入地理解计算机底层的基本工作机理,达到可以随心所欲地控制计算机的目的.
1.1 机器语言
1.2 汇编语言的产生
1.3 汇编语言的组成
       (1) 汇编指令:机器码的助记符,有对应的机器码。
       (2) 伪指令:没有对应的机器码,由编译器执行,计算机并执行。
       (3) 其他符号:如+、-、*、/ 等,由编译器识别,没有对应的机器码。
1.4 存储器
1.5 指令和数据
补充内容:数进制

二进制(Binary):    0,1
八进制(Octal):      0,1,2,3,4,5,6,7
十进制(Decimal):  0,1,2,3,4,5,6,7,8,9
十六进制(Hex):     0,1,2,3,4,5,6,7,8,9,A,B,C,D,E,F
1.6 存储单元
      微机存储器的容量最小单位为一个字节。通常用B来标识。

1TB=1024GB   / 1GB=1024MB   / 1MB=1024KB   / 1KB=1024B   / 1Byte=8bit   
1TB=2^10GB=2^20MB=2^30KB=2^40Byte
位(bit 音译为比特)、字节(Byte)、字/单字(Word)、双字(DoubleWord)、四字
1.7 CPU对存储器的读写
       CPU和其他芯片的导线,通常称为总线。总线从逻辑上分为:地址总线、控制总线、数据总线。
       CPU与外部器件(标准的说法是芯片)进行下面3类信息的交互。
       1)存储单元的地址(地址信息)
       2)器件的选择,读或写的命令(控制信息)
       3)读或写的数据(数据信息)
        电子计算机能处理、传输的信息都是电信号,电信号需导线传送。
       CPU从3号单元中读取数据的过程如下图:


       (1) CPU通过地址线将地址信息 3 发出。
       (2) CPU通过控制线发出内存读命令,选中存储器芯片,并通知它,将要从中读取数据。
       (3) 存储器将3号单元中的数据8 通过数据线送入CPU。
       思考:CPU是如何发出读或写命令,又是如何通知存储器的?如何命令计算机(或微处理器)进行数据的读写呢?要让一个计算机或微处理器工作,应向它输入能够驱动它进行工作的电平信息(机器码)。
       写操作与读操作的步骤相似。如向3号单元写入数据26。
        (1) CPU通过地址线将地址信息3发出。
        (2) CPU通过控制线发出内存写命令,选中存储器芯片,并通知它,要向其中写入数据。
        (3) CPU通过数据线将数据26送入内存的3号单元中。
对于8086CPU,下面的机器码,能够完成从3号单元读数据。
   机器码: 1010 0001 0000 0011 0000 0000 ;CPU接收这条机器码后将完成上面所述的读写工作。
   对应的汇编指令:MOV AX,[3]
   含义: 传送3号单元的内容入AX (从3号单元读取数据送入寄存器AX)
1.8 地址总线
       CPU是通过地址总线来指定存储器单元的。地址总线能传送多少个不同的信息,CPU就可以对多少个存储单元进行寻址。(N 根地址线,CPU寻址的内存单元数为:2的N次方)
1.9 数据总线
       8根数据总线一次可传送一个8位二进制数据(一个字节),16根数据总线一次可传送两个字节。
8086有16根数据线,8088有8根数据线。那8086可一次传送数据89D8H,而8088需两次传送89D8H。
(89D8H=1000,1001,1101,1000B)
8080     数据总线    08根     地址总线     16根
8088     数据总线    08根     地址总线     20根
8086     数据总线    16根     地址总线     20根 可定位:2的20次方(1MB) 个内存单元
80286   数据总线    16根     地址总线     24根
80386   数据总线    32根     地址总线     32根 可定位:2的32次方(4GB) 个内存单元
1.10 控制总线
       有多少根控制总线,就意味着CPU提供了对外部器件的多少种控制。
       读信号输出----由CPU向外传送读信号,CPU向该控制线上输出低电平表示要读取数据。
        写信号输出----负责传送写信号
1.1~1.10 小结:
      在存储器中指令和数据没有任何区别,都是二进制信息。
       存储单元从零开始顺序编号。
       地址总线的宽度决定了CPU的寻址能力。
       数据总线的宽度决定了CPU与其他器件数据传送时的一次数据传递量。
       控制总线的宽度决定了CPU对系统中其他器件的控制能力。
       我们在基于一个计算机硬件系统编程时,必须知道这个系统中的内存地址空间分配情况。当我们想知道在某类存储器中读写数据的时候,需知道它的第一个单元的地址和最后一个单元的地址。

1.11 内存地址空间
       一个CPU的地址总线宽度为10,则寻址为1024个内存单元。
1.12 主板
1.13 接口卡
1.14 各类存储器芯片




1.15 内存地址空间
    最终运行程序的是CPU,我们用汇编编程的时候,必须要从CPU的角度考虑问题,对CPU来讲,系统中的所有存储器中的存储单元都处于一个统一的逻辑存储器中,它的容量受CPU寻址能力的限制。这个逻辑存储器即是我们所说的内存地址空间。


地址0~7FFFH的32KB空间为主随机存储器的地址空间;

地址8000H~9FFFH的8KB空间为显存地址空间;

地址A000H~FFFFH的24KB空间为各个ROM的地址空间。



A:

(1) 13
(2) 1024 , 0, 1023
(3) 8192, 1024
(4) 2^ 30, 2^ 20,2^ 10 (^ : 表示乘方)
(5) 64,1, 16, 4
(6) 1,1,2,2,4
(7) 512, 256
(8) 二进制

 

 

 

 

 

 

 

 

 

 

==========================================

 

 

===========================================

HelloWorld 源代码

DATAS SEGMENT   
    msg db 'hello World'  
DATAS ENDS   
STACKS SEGMENT   
STACKS ENDS   
CODES SEGMENT   
    ASSUME CS:CODES,DS:DATAS,SS:STACKS   
START:   
    MOV AX,DATAS ;数据段初始   
    MOV DS,AX    ;数据段初始   
             
    mov bx,0B800H;定义显存地址   
    mov es,bx ;   
    mov cx,11 ;loop 次数   
    mov si,0 ;源变址寄存器   
    mov bx,0 ;ax bx cx dx   
    mov dx,00011100b ;存放字的属性   
    ;XRGBXRGB ;闪 底色RGB 高亮 前景RGB   
       
    ;mov al,ds:[0]   
    ;mov es:[0],al   
    ;mov es:[1],01000010B   
       
    s:   
        mov al,ds:[si] ;ds:si的数据放到AL   
        mov es:[bx],al ;把AL上的数据 放到 显存上   
        mov es:[bx+1],dx ;设置字的属性(属性放在dx上)   
        inc si ;相当于add si,1   
        add bx,2   
        loop s   
       
    MOV AH,4CH ;中断   
    INT 21H   
      
CODES ENDS   
END START   

====================================================================
汇编指令大全

1. 通用数据传送指令.
        MOV    传送字或字节.
        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 )
    2. 输入输出端口传送指令.
        IN      I/O端口输入. ( 语法: IN 累加器, {端口号│DX} )
        OUT    I/O端口输出. ( 语法: OUT {端口号│DX},累加器 )
          输入输出端口由立即方式指定时, 其范围是 0-255; 由寄存器 DX 指定时,
          其范围是 0-65535.
    3. 目的地址传送指令.
        LEA    装入有效地址.
          例: 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.
    4. 标志传送指令.
        LAHF    标志寄存器传送,把标志装入AH.
        SAHF    标志寄存器传送,把AH内容装入标志寄存器.
        PUSHF  标志入栈.
        POPF    标志出栈.
        PUSHD  32位标志入栈.
        POPD    32位标志出栈.

二、算术运算指令
───────────────────────────────────────
          ADD    加法.
        ADC    带进位加法.
        INC    加 1.
        AAA    加法的ASCII码调整.
        DAA    加法的十进制调整.
        SUB    减法.
        SBB    带借位减法.
        DEC    减 1.
        NEC    求反(以 0 减之).
        CMP    比较.(两操作数作减法,仅修改标志位,不回送结果).
        AAS    减法的ASCII码调整.
        DAS    减法的十进制调整.
        MUL    无符号乘法.
        IMUL    整数乘法.
          以上两条,结果回送AH和AL(字节运算),或DX和AX(字运算),
        AAM    乘法的ASCII码调整.
        DIV    无符号除法.
        IDIV    整数除法.
          以上两条,结果回送:
              商回送AL,余数回送AH, (字节运算);
          或  商回送AX,余数回送DX, (字运算).
        AAD    除法的ASCII码调整.
        CBW    字节转换为字. (把AL中字节的符号扩展到AH中去)
        CWD    字转换为双字. (把AX中的字的符号扩展到DX中去)
        CWDE    字转换为双字. (把AX中的字符号扩展到EAX中去)
        CDQ    双字扩展.    (把EAX中的字的符号扩展到EDX中去)

三、逻辑运算指令
───────────────────────────────────────
          AND    与运算.
        OR      或运算.
        XOR    异或运算.
        NOT    取反.
        TEST    测试.(两操作数作与运算,仅修改标志位,不回送结果).
        SHL    逻辑左移.
        SAL    算术左移.(=SHL)
        SHR    逻辑右移.
        SAR    算术右移.(=SHR)
        ROL    循环左移.
        ROR    循环右移.
        RCL    通过进位的循环左移.
        RCR    通过进位的循环右移.
          以上八种移位指令,其移位次数可达255次.
              移位一次时, 可直接用操作码.  如 SHL AX,1.
              移位>1次时, 则由寄存器CL给出移位次数.
                如  MOV CL,04
                    SHL AX,CL

四、串指令
───────────────────────────────────────
             DS:SI  源串段寄存器  :源串变址.
            ES:DI  目标串段寄存器:目标串变址.
            CX      重复次数计数器.
            AL/AX  扫描值.
            D标志  0表示重复操作中SI和DI应自动增量; 1表示应自动减量.
            Z标志  用来控制扫描或比较操作的结束.
        MOVS    串传送.
            ( MOVSB  传送字符.    MOVSW  传送字.    MOVSD  传送双字. )
        CMPS    串比较.
            ( CMPSB  比较字符.    CMPSW  比较字. )
        SCAS    串扫描.
            把AL或AX的内容与目标串作比较,比较结果反映在标志位.
        LODS    装入串.
            把源串中的元素(字或字节)逐一装入AL或AX中.
            ( LODSB  传送字符.    LODSW  传送字.    LODSD  传送双字. )
        STOS    保存串.
            是LODS的逆过程.
        REP            当CX/ECX<>0时重复.
        REPE/REPZ      当ZF=1或比较结果相等,且CX/ECX<>0时重复.
        REPNE/REPNZ    当ZF=0或比较结果不相等,且CX/ECX<>0时重复.
        REPC          当CF=1且CX/ECX<>0时重复.
        REPNC          当CF=0且CX/ECX<>0时重复.

五、程序转移指令
───────────────────────────────────────
     1>无条件转移指令 (长转移)
        JMP    无条件转移指令
        CALL    过程调用
        RET/RETF过程返回.
    2>条件转移指令 (短转移,-128到+127的距离内)
        ( 当且仅当(SF XOR OF)=1时,OP1<OP2 )
        JA/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" 时转移.
    3>循环控制指令(短转移)
        LOOP            CX不为零时循环.
        LOOPE/LOOPZ    CX不为零且标志Z=1时循环.
        LOOPNE/LOOPNZ  CX不为零且标志Z=0时循环.
        JCXZ            CX为零时转移.
        JECXZ          ECX为零时转移.
    4>中断指令
        INT    中断指令
        INTO    溢出中断
        IRET    中断返回
    5>处理器控制指令
        HLT    处理器暂停, 直到出现中断或复位信号才继续.
        WAIT    当芯片引线TEST为高电平时使CPU进入等待状态.
        ESC    转换到外处理器.
        LOCK    封锁总线.
        NOP    空操作.
        STC    置进位标志位.
        CLC    清进位标志位.
        CMC    进位标志取反.
        STD    置方向标志位.
        CLD    清方向标志位.
        STI    置中断允许位.
        CLI    清中断允许位.

六、伪指令
───────────────────────────────────────
          DW      定义字(2字节).
        PROC    定义过程.
        ENDP    过程结束.
        SEGMENT 定义段.
        ASSUME  建立段寄存器寻址.
        ENDS    段结束.
        END    程序结束.

一、状态寄存器
PSW(Program Flag)程序状态字寄存器,是一个16位寄存器,由条件码标志(flag)和控制标志构成,如下所示:
15 14 13 12 11 10 9 8 7 6 5 4 3 2 1 0
        OF DF IF TF SF ZF   AF   PF   CF

条件码:
①OF(Overflow Flag)溢出标志。溢出时为1,否则置0。
②SF(Sign Flag)符号标志。结果为负时置1,否则置0.
③ZF(Zero Flag)零标志,运算结果为0时ZF位置1,否则置0.
④CF(Carry Flag)进位标志,进位时置1,否则置0.
⑤AF(Auxiliary carry Flag)辅助进位标志,记录运算时第3位(半个字节)产生的进位置。有进位时1,否则置0.
⑥PF(Parity Flag)奇偶标志。结果操作数中1的个数为偶数时置1,否则置0.
控制标志位:
⑦DF(Direction Flag)方向标志,在串处理指令中控制信息的方向。
⑧IF(Interrupt Flag)中断标志。
⑨TF(Trap Flag)陷井标志。
二、 直接标志转移(8位寻址)
指令格式 机器码 测试条件 如...则转移  
  指令格式 机器码 测试条件 如...则转移
JC  72 C=1 有进位 JNS 79 S=0 正号
JNC  73 C=0 无进位 JO 70 O=1 有溢出
JZ/JE  74 Z=1 零/等于 JNO 71 O=0 无溢出
JNZ/JNE  75 Z=0 不为零/不等于 JP/JPE 7A P=1 奇偶位为偶
JS 78 S=1 负号 JNP/IPO 7B P=0 奇偶位为奇
三、间接标志转移(8位寻址)
指令格式 机器码 测试格式 如...则转移
JA/JNBE(比较无符号数) 77 C或Z=0 >  高于/不低于或等于
JAE/JNB(比较无符号数) 73 C=0 >=  高于或等于/不低于
JB/JNAE(比较无符号数) 72 C=1 <  低于/不高于或等于
JBE/JNA(比较无符号数) 76 C或Z=1 <=  低于或等于/不高于
JG/JNLE(比较带符号数) 7F (S异或O)或Z=0 >  大于/不小于或等于
JGE/JNL(比较带符号数) 7D S异或O=0 >=  大于或等于/不小于
JL/JNGE(比较带符号数) 7C S异或O=1 <  小于/不大于或等于
JLE/JNG(比较带符号数) 7E (S异或O)或Z=1 <=  小于或等于/不大于
四、无条件转移指令(fisheep译)
操作码  伪码指令 含义
EB  cb JMP rel8 相对短跳转(8位),使rel8处的代码位下一条指令
E9  cw JMP rel16  相对跳转(16位),使rel16处的代码位下一条指令
FF  /4 JMP r/m16 绝对跳转(16位),下一指令地址在r/m16中给出
FF  /4 JMP r/m32 绝对跳转(32位),下一指令地址在r/m32中给出
EA  cb JMP ptr16:16 远距离绝对跳转, 下一指令地址在操作数中
EA  cb JMP ptr16:32  远距离绝对跳转, 下一指令地址在操作数中
FF  /5 JMP m16:16 远距离绝对跳转, 下一指令地址在内存m16:16中
FF  /5 JMP m16:32  远距离绝对跳转, 下一指令地址在内存m16:32中
五、16位/32位寻址方式(fisheep译)

操作码 伪码指令 跳转含义  跳转类型 跳转的条件(标志位)
0F 87  cw/cd JA rel16/32 大于 near (CF=0 and ZF=0)
0F 83  cw/cd JAE rel16/32 大于等于 near (CF=0)
0F 82  cw/cd JB rel16/32 小于 near (CF=1)
0F 86  cw/cd JBE rel16/32 小于等于 near (CF=1 or ZF=1)
0F 82  cw/cd JC rel16/32 进位 near (CF=1)
0F 84  cw/cd JE rel16/32 等于 near (ZF=1)
0F 84  cw/cd JZ rel16/32 为0 near (ZF=1)
0F 8F  cw/cd JG rel16/32 大于 near (ZF=0 and SF=OF)
0F 8D  cw/cd JGE rel16/32  大于等于 near (SF=OF)
0F 8C  cw/cd JL rel16/32 小于  near (SF<>OF)
0F 8E  cw/cd JLE rel16/32 小于等于 near (ZF=1 or SF<>OF)
0F 86  cw/cd JNA rel16/32  不大于 near (CF=1 or ZF=1)
0F 82  cw/cd JNAE rel16/32 不大于等于 near (CF=1)
0F 83  cw/cd JNB rel16/32 不小于 near (CF=0)
0F 87  cw/cd JNBE rel16/32 不小于等于 near (CF=0 and ZF=0)
0F 83  cw/cd JNC rel16/32  不进位 near (CF=0)
0F 85  cw/cd JNE rel16/32 不等于 near (ZF=0)
0F 8E  cw/cd JNG rel16/32 不大于 near (ZF=1 or SF<>OF)
0F 8C  cw/cd JNGE rel16/32  不大于等于 near (SF<>OF)
0F 8D  cw/cd JNL rel16/32 不小于 near (SF=OF)
0F 8F  cw/cd JNLE rel16/32  不小于等于 near (ZF=0 and SF=OF)
0F 81  cw/cd JNO rel16/32 未溢出 near (OF=0)
0F 8B  cw/cd JNP rel16/32  不是偶数 near (PF=0)
0F 89  cw/cd JNS rel16/32  非负数 near (SF=0)
0F 85  cw/cd JNZ rel16/32 非零(不等于) near (ZF=0)
0F 80  cw/cd JO rel16/32  溢出 near (OF=1)
0F 8A  cw/cd JP rel16/32  偶数 near (PF=1)
0F 8A  cw/cd JPE rel16/32 偶数 near (PF=1)
0F 8B  cw/cd JPO rel16/32  奇数 near (PF=0)
0F 88  cw/cd JS rel16/32 负数 near (SF=1)
0F 84  cw/cd JZ rel16/32  为零(等于) near (ZF=1)

注:一些指令操作数的含义说明:
  rel8      表示 8 位相对地址
  rel16    表示 16 位相对地址
  rel16/32  表示 16或32 位相对地址
  r/m16    表示16位寄存器
  r/m32    表示32位寄存器

浮点指令  对下面的指令先做一些说明:
st(i):代表浮点寄存器,所说的出栈、入栈操作都是对st(i)的影响
src,dst,dest,op等都是指指令的操作数,src表示源操作数,dst/dest表示目的操作数
mem8,mem16,mem32,mem64,mem80等表示是内存操作数,后面的数值表示该操作数的内存位数(8位为一字节)
x <- y 表示将y的值放入x,例st(0) <- st(0) - st(1)表示将st(0)-st(1)的值放入浮点寄存器st(0)
1.  数据传递和对常量的操作指令
指令格式
指令含义
执行的操作

FLD src  装入实数到st(0)  st(0) <- src (mem32/mem64/mem80)
FILD src  装入整数到st(0)  st(0) <- src (mem16/mem32/mem64)
FBLD src  装入BCD数到st(0)  st(0) <- src (mem80)
 
FLDZ  将0.0装入st(0)  st(0) <- 0.0
FLD1  将1.0装入st(0)  st(0) <- 1.0
FLDPI  将pi装入st(0)  st(0) <- ?(ie, pi)
FLDL2T  将log2(10)装入st(0)  st(0) <- log2(10)

FLDL2E  将log2(e)装入st(0)  st(0) <- log2(e)
FLDLG2  将log10(2)装入st(0)  st(0) <- log10(2)
FLDLN2  将loge(2)装入st(0)  st(0) <- loge(2)
 
FST dest  保存实数st(0)到dest  dest <- st(0) (mem32/mem64)
FSTP dest    dest <- st(0) (mem32/mem64/mem80);然后再执行一次出栈操作
FIST dest  将st(0)以整数保存到dest  dest <- st(0) (mem32/mem64)
FISTP dest  dest <- st(0) (mem16/mem32/mem64);然后再执行一次出栈操作
FBST dest  将st(0)以BCD保存到dest  dest <- st(0) (mem80)
FBSTP dest      dest<- st(0) (mem80);然后再执行一次出栈操作
2.比较指令
指令格式  指令含义  执行的操作
FCOM  实数比较  将标志位设置为 st(0) - st(1) 的结果标志位
FCOM op  实数比较  将标志位设置为 st(0) - op (mem32/mem64)的结果标志位

FICOM op  和整数比较  将Flags值设置为st(0)-op 的结果op (mem16/mem32)
FICOMP op  和整数比较  将st(0)和op比较 op(mem16/mem32)后;再执行一次出栈操作
 
FTST  零检测  将st(0)和0.0比较
FUCOM st(i)      比较st(0) 和st(i)                  [486]
FUCOMP st(i)          比较st(0) 和st(i),并且执行一次出栈操作
FUCOMPP st(i)        比较st(0) 和st(i),并且执行两次出栈操作
FXAM      Examine: Eyeball st(0) (set condition codes)
3.运算指令
指令格式  指令含义  执行的操作
加法
FADD  加实数  st(0) <-st(0) + st(1)
FADD src    st(0) <-st(0) + src (mem32/mem64)
FADD st(i),st    st(i) <- st(i) + st(0)
FADDP st(i),st      st(i) <- st(i) + st(0);然后执行一次出栈操作
FIADD src  加上一个整数  st(0) <-st(0) + src (mem16/mem32)
减法
FSUB  减去一个实数  st(0) <- st(0) - st(1)
FSUB src    st(0) <-st(0) - src (reg/mem)
FSUB st(i),st    st(i) <-st(i) - st(0)
FSUBP st(i),st    st(i) <-st(i) - st(0),然后执行一次出栈操作
FSUBR st(i),st  用一个实数来减  st(0) <- st(i) - st(0)
FSUBRP st(i),st    st(0) <- st(i) - st(0),然后执行一次出栈操作
FISUB src  减去一个整数  st(0) <- st(0) - src (mem16/mem32)
FISUBR src  用一个整数来减  st(0) <- src - st(0) (mem16/mem32)
乘法

FMUL  乘上一个实数  st(0) <- st(0) * st(1)
FMUL st(i)    st(0) <- st(0) * st(i)
FMUL st(i),st    st(i) <- st(0) * st(i)
FMULP st(i),st    st(i) <- st(0) * st(i),然后执行一次出栈操作
FIMUL src  乘上一个整数  st(0) <- st(0) * src (mem16/mem32)
除法
FDIV  除以一个实数  st(0) <-st(0) /st(1)
FDIV st(i)    st(0) <- st(0) /t(i)
FDIV st(i),st    st(i) <-st(0) /st(i)
FDIVP st(i),st    st(i) <-st(0) /st(i),然后执行一次出栈操作
FIDIV src  除以一个整数  st(0) <- st(0) /src (mem16/mem32)
FDIVR st(i),st  用实数除  st(0) <- st(i) /st(0)
FDIVRP st(i),st    FDIVRP st(i),st
FIDIVR src  用整数除  st(0) <- src /st(0) (mem16/mem32)
 
FSQRT  平方根  st(0) <- sqrt st(0)
 
FSCALE  2的st(0)次方  st(0) <- 2 ^ st(0)
FXTRACT  Extract exponent:  st(0) <-exponent of st(0); and gets pushed
st(0) <-significand of st(0)
 
FPREM  取余数  st(0) <-st(0) MOD st(1)
FPREM1  取余数(IEEE),同FPREM,但是使用IEEE标准[486]
      
FRNDINT  取整(四舍五入)  st(0) <- INT( st(0) ); depends on RC flag
 
FABS  求绝对值  st(0) <- ABS( st(0) ); removes sign
FCHS  改变符号位(求负数)  st(0) <-st(0)
 
F2XM1  计算(2 ^ x)-1  st(0) <- (2 ^ st(0)) - 1
FYL2X  计算Y * log2(X)  st(0)为Y;st(1)为X;将st(0)和st(1)变为st(0) * log2( st(1) )的值
 
FCOS  余弦函数Cos  st(0) <- COS( st(0) )
FPTAN  正切函数tan  st(0) <- TAN( st(0) )
FPATAN  反正切函数arctan  st(0) <- ATAN( st(0) )
FSIN  正弦函数sin  st(0) <- SIN( st(0) )
FSINCOS  sincos函数  st(0) <-SIN( st(0) ),并且压入st(1)
st(0) <- COS( st(0) )
      
FYL2XP1  计算Y * log2(X+1)  st(0)为Y; st(1)为X; 将st(0)和st(1)变为st(0) * log2( st(1)+1 )的值
处理器控制指令
FINIT  初始化FPU  
FSTSW AX  保存状态字的值到AX  AX<- MSW
FSTSW dest  保存状态字的值到dest  dest<-MSW (mem16)
      
FLDCW src  从src装入FPU的控制字  FPU CW <-src (mem16)
FSTCW dest  将FPU的控制字保存到dest  dest<- FPU CW
      
FCLEX  清除异常  
    
FSTENV dest  保存环境到内存地址dest处 保存状态字、控制字、标志字和异常指针的值
FLDENV src  从内存地址src处装入保存的环境  
FSAVE dest  保存FPU的状态到dest处 94字节  
FRSTOR src  从src处装入由FSAVE保存的FPU状态  
      
FINCSTP  增加FPU的栈指针值  st(6) <-st(5); st(5) <-st(4),...,st(0) <-?
FDECSTP  减少FPU的栈指针值  st(0) <-st(1); st(1) <-st(2),...,st(7) <-?
      
FFREE st(i)  标志寄存器st(i)未被使用  
    
FNOP  空操作,等同CPU的nop  st(0) <-st(0)
WAIT/FWAIT  同步FPU与CPU:停止CPU的运行,直到FPU完成当前操作码
 
FXCH  交换指令,交换st(0)和st(1)的值  st(0) <-st(1)
st(1) <- st(0)

 ======================================================
VC和VB 嵌入汇编的方法

VC的嵌入ASM的方法

#define CallBaseAddr  0x962AC4  
#define SkillAttackAddr 0x457680  
void maindlg::SkillAttack(long SkillID)  
{  
    __asm  
    {  
    pushad  
    mov  eax,DWORD ptr ds:[CallBaseAddr]  
    mov  eax, DWORD ptr ds:[eax+0x1c]  
    mov  eax, DWORD ptr ds:[eax+0x20]  
    mov  ecx,eax  
    push -1  
    push 0  
    push 0  
    mov eax, SkillID  
    push eax  
    mov eax,SkillAttackAddr  
    call eax  
    popad  
    }  
}  

VB的嵌入汇编的方法

Sub SkillAttack(SkillID As Long, GameProcId As Long)   
Dim asm As New clsASM   
Const Address = &H457680   
Const Base_Addr = &H962AC4   
With asm                            
    .Pushad                            
    .Mov_EAX_DWORD_Ptr Base_Addr      
    .Mov_EAX_DWORD_Ptr_EAX_Add &H1C    
    .Mov_EAX_DWORD_Ptr_EAX_Add &H20    
    .Mov_ECX_EAX                      
    .Push -1                          
    .Push 0                            
    .Push 0                            
    .Mov_EAX SkillID                  
    .Push_EAX                          
    .Mov_EAX Address                  
    .Call_EAX                          
    .Popad                            
    .ret                              
End With                              
asm.inject GameProcId   
asm.Run_ASM GameProcId, 0   
asm.free GameProcId, 0   
End Sub  

 
============================================================================
 
 
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值