zoukankan      html  css  js  c++  java
  • 32位汇编知识总结

    重要知识

    • 汇编读写
    ;WriteString	 ReadString
    
    ;WriteString
    ; Writes a null-terminated string to standard
    ; output. Input parameter: EDX points to the string.
    ;ReadString 
    ; Reads a string of up to 128 characters from the console
    ; past the end of line, and places the characters in a buffer.
    ; Strips off CR/LF from the input buffer.
    ; Receives: EDX points to the input buffer,
    ;           ECX contains the maximum string length
    ; Returns:  EAX = size of the input string.
    
    ;WriteInt	 ReadInt
    
    ;WriteInt 
    ;Receives: EAX = the integer 输出eax
    ;ReadInt
    ;Returns:  If CF=0, the integer is valid, and EAX = binary value.
    ;If CF=1, the integer is invalid and EAX = 0.
    
    
    ;WriteHex  	 ReadHex
    
    ;WriteHex
    ; Writes an unsigned 32-bit hexadecimal number to standard output.
    ; Input parameters: EAX = the number to write.
    ;ReadHex 
    ;Reads a 32-bit hexadecimal integer from standard input
    ;Returns: EAX = binary integer value
    
    ;Crlf
    ; Writes a carriage return / linefeed(写入回车或换行符)
    
    
    ;WriteChar
    ; Write a character to standard output
    ; Recevies: AL = character
    
    
    • 变量的分类(汇编中用大小区分变量)
    
    ;DB / byte 定义字节. 8位(一个字节数据占一个字节单元,读完一个,偏移量加1)
    ;DW / word 定义Word.通常在x86 32位系统上为2个字节(一个字节数据占2个字节单元,读完一个,偏移量加2)
    ;DD / Dword 定义双字.典型的x86 32位系统通常为4个字节(一个双字节数据占4个字节单元,读完一个,偏移量加4)
    
    • 跳转指令
    ; jmp :无条件转移指令
    
    ; jnc :表示不进位时跳转
    ; jnz :表示非0跳转
    
    ;有符号数比较
    ; jg :greater than
    ; jz :zero equal to
    ; jl :less than
    
    ;无符号比较
    ;ja :above var1>var2
    ;jz,je     var1=var2
    ;jb :below var1<var2
    
    ;xchg 交换两个变量的值
    ;交换指令XCHG是两个寄存器,寄存器和内存变量之间内容的交换指令,两个操作数的数据类型要相同,可以是一个字节,也可以是一个字,也可以是双字
    
    • 操作数与四则运算
    opcode oprand1 [,operand2]
    ;1.oprand1称为目的操作数
    ;  oprand2称为原操作数 
    ;2.目的操作数不能为常数
    ;3.两个操作数的类型必须一样(大小相同)
    ;4.指令如果涉及到两个操作数,两个操作数不能都是内存变量,必须有一个是寄存器
    
    ;四则运算
    add oprand1,operand2 eflag;无进位
    addc oprand1,operand2 CF;有进位
    add byte ptr [m1],20;将m1当作byte长度看待
    add word ptr [m1],20;将m1当作word长度看待
    add dword ptr [m1],20;将m1当作dword长度看待
    
    sub oprand1,oprand2 eflag
    
    mul oprand1 ;结果必须由64位保存,结果储存在edx,eax中(高位在edx中,低位在eax中)
    ;字节操作数 (AX)
    ;字操作数 (DX,AX)
    ;双字操作数 (EDX,EAX)
    ;mul通过操作数来辨别乘法进行的位数(如果是立即数无法辨别大小)
    
    
    imul oprand1 ;有进位,存在edx,eax
    ;被乘数和积在edx,eax组成的64位中
    
    div  oprand ;被除数在edx,eax组成的64位中
    ;结果:商在eax中,余数在edx中(如果被除数是32位记得将edx赋值为0)
    
    
    
    • 函数实现
    F1 PROC
    	ret
    F1 ENDP
    
    ;规定函数返回值放在寄存器内
    
    ;函数内用栈保存寄存器信息
    isPrime PROC
    	push ebx
    	push edx
    	push esp
    	
    	
    	pop esp
    	pop edx
    	pop ebx
    	ret
    isPrime ENDP
    
    ;习惯规定如果返回值32位放的下的话默认放在eax中
    
    • 寄存器的使用
    ;32位寄存器高16位无法直接使用,对应的低16位AX,BX,CX,DX,SI,DI,BP,SP可视情况使用
    ;以上16位寄存器还可以细分:
    ;AX:AH AL
    ;BX:BH BL
    ;CX:CH CL
    ;DX:DH DL
    
    
    ;EAX是"累加器"(accumulator), 它是很多加法乘法指令的缺省寄存器。
    
    ;EBX是"基地址"(base)寄存器, 在内存寻址时存放基地址。
    
    ;ECX是计数器(counter), 是重复(REP)前缀指令和LOOP指令的内定计数器。
    
    ;EDX则总是被用来放整数除法产生的余数。
    
    ;ESI/EDI分别叫做"源/目标索引寄存器"(source/destination index),因为在很多字符串操作指令中, DS:ESI指向源串,而ES:EDI指向目标串
    
    ;EBP是"基址指针"(BASE POINTER), 它最经常被用作高级语言函数调用的"框架指针"(frame pointer). 
    
    ;ESP专门用作堆栈指针,被形象地称为栈顶指针,堆栈的顶部是地址小的区域,压入堆栈的数据越多,ESP也就越来越小。在32位平台上,ESP每次减少4字节
    
    ;EIP储存下一条cpu执行的指令地址
    
    
    
    ;EFLAG取值含义
    ;IP: EI/DI 中断标志位
    ;中断控制在硬件中完成(接口和CPU中的中断控制指令集)
    
    • X进制常数表示
    ;进制前缀:
    ;二进制 0B		0B101010
    ;八进制 0O	 	0O767676
    ;十六进制 0x		0xA7B2
    
    ;进制后缀:
    ;二进制 B
    ;八进制 O
    ;十进制 D
    ;十六进制 H
    
    • 寻址方式
    ;立即寻址(目的操作数是常量)
    ;直接寻址(数据在变量内 eg. val)
    ;寄存器寻址(数据在寄存器内)
    ;寄存器间接寻址 (不能出现[100],应该用变量定义,再使用变量名;寻址寄存器用32位) 
    mov eax,offset val
    mov ebx,[eax]
    ;寄存器相对寻址(在寄存器间接寻址的基础上对地址加减偏移量)
    ;基址—变址寻址
    ;基址—变址相对寻址
    
    • 指令
    ;逻辑运算指令
    
    ;逻辑非指令 NOT OPR
    
    ;逻辑与指令 AND DST,SRC
    
    ;逻辑或指令 OR DST,SRC
    
    ;异或指令 XOR DST,SRC
    
    ;测试指令 TEST OPR1,OPR2
    
    ;移位指令
    
    ;逻辑左移 SHL OPR,CNT   (SHL = Shift Left)
    ;逻辑右移 SHR OPR,CNT   (SHR = Shift Right)
    
    ;CNT(存在CL 八位寄存器中)   
    
    ;串处理指令
    ;B:BYTE  W:WORD  D:DOUBLE WORD
    ;MOVS(MOVSB,MOVSW,MOVSD)
    ;源串地址放在ESI中,源串长度放在ECX中(以单位偏移量为单位长度),目的串地址放在EDI中
    ;DF标志位 DF=0时表示ESI/EDI储存串的首地址,DF=1时表示ESI/EDI储存串的尾地址
    
    ;STOS  初始化串
    ;指令对应寄存器(存放初始化单元)
    ;STOSB  AL
    ;STOSW  AX
    ;STOSD  EAX
    
    ;LODS	从串地址中取出一个子节/字/双字放在AL/AX/EAX中
    ;CMPS	比较两个串的大小(较小串长度放在ECX中)
    ;SCAS	扫描字符串,在字符串中找与AL/AX/EAX中数据相同或不同的子串(比较最大次数放在ECX中)
     
    
    ;设置方向标志位  CTD STD
    ;CLD指令功能:(STD功能相反)
    ;将标志寄存器Flag的方向标志位DF清零。
    ;在字串操作中使变址寄存器SI或DI的地址指针自动增加,字串处理由前往后。
    
    ;串重复前缀
    ;REP REPE/REPZ REPNE/REPNZ
    
    ;堆栈操作
    
    ;进栈指令
    ;push pushad
    ;pop popad
    ;pushad/popad 可以将寄存器在一行内一次性入栈/出栈
    ;PUSHAD指令压入32位寄存器,其入栈顺序是:EAX,ECX,EDX,EBX,ESP,EBP,ESI,EDI(执行后ESP- 32)
    ;POPAD指令按照与上面相反的顺序依次弹出寄存器的值,顺序为EDI,ESI,EBP,ESP,EBX,EDX,ECX,EAX(执行后ESP+32)
    
    ret
    ;ret指令的内部操作是:栈顶字单元出栈,其值赋给IP寄存器。即实现了一个程序的转移,将栈顶字单元保存的偏移地址作为下一条指令的偏移地址。
    
    ;函数执行完时  ret 12
    ;栈顶pop 赋值给EIP  ESP地址加12
    
    
    
    • 小知识点
    ;分号之后的部分为注释
    
    call DumpRegs ;显示寄存器信息D
    
    ;伪指令有2个特点:
    ;(1)由于是伪“指令”,因而它只存在于汇编语言中。高级语言中不叫指令,叫语句;
    ;(2)由于是“伪”指令,也即“假”指令,因而不是可执行指令,不会产生机器代码,不会占用ROM空间,只用于汇编过程中为汇编程序提供汇编信息。
    
    mov ebx,var1
    cmp ebx,var2 ;var1-var2=>eflag 相减后会改变eflag,执行该语句后var1,var2不变
    
    ;负数用的是补码表示方式
    
    ;小端编址方式: 00404000 -> 18 FC FF FF
         
    ;数组用法: 
    ;arr dd 10 DUP(?)
    ;arr dd 1,2,34,3,5,6,8 直接为数组赋值
    ;arrlen = ($-arr) / 4 可以用这种方式动态获得数组大小(注意必须紧连着arr的定义,4代表的是偏移量)
    ;一个内存地址对应一个字节(Byte): 0xFF 或者说 8-bit:11110000b。
    
    ;Excellent db 'Excellent',0 ;''中的字符串不含结束字符0,需要手动添加
    
    lea edx,msg ;==mov edx,offset msg
    ;lea = load effective address 
    
    inc eax ;自增
    dec eax ;自减
    ;操作数可以是变量,不影响CF标志
    again:
    	add v,1 ;inc v 使用inc不会导致eflag改变,inc不会造成进位
    	jnc again ;jnc 表示不进位时跳转
    
    ;相同的mov指令但是选用的寄存器不同机器码也可能不一样
    
    ;外设相对于cpu可以看作寄存器
    
    ;带符号扩展
    ;MOVSX reg1, reg2
    ;MOVSX reg, mem
    ;如果拓展不用用到高位,高位自动清零
    ;目的操作数的位数要比源操作数大
    ;源操作数可以是8位或者16位,而目的操作数必须是16位或者32位
    
    ;循环
    mov ecx,n ;n为常数
    Addr:
    	;循环体
    	loop Addr ;ecx-- 并判断ecx与0是否相等,如果不相等,继续循环;如果相等,跳出循环 
    	
    

    汇编程序模板

    • 模板

      TITLE Program Template ;Template.asm
      
      INCLUDE Irvine32.inc
      .data
      ;此处定义全局变量
      .code
      ;代码段
      main PROC
      
      	exit
      main ENDP
      
      F1 PROC
      
      	ret
      F1 ENDP
      
      END main
      
      ;传参模板(保护现场)
      push x
      push y
      f1 proc
      push ebp
      mov ebp,esp
      sub esp,4
      pushad
      
      popad
      mov eax,[ebp-4]
      add esp,4
      pop ebp
      ret 8
      f1 endp
      ;用栈保存寄存器状态,防止修改寄存器值影响main函数执行
      
    • 基础计算

      TITLE Add and Subtract
      INCLUDE Irvine32.inc
      .data
      val1 DWORD 10000h ; h代表16进制,10000h是十进制2^16
      val2 DWORD 40000h
      val3 DWORD 20000h
      ;finalVal DWORD
      .data? ;其中的数据段也可以定义在.data中
      finalVal DWORD ? ;定义未初始化数据
      temp DWORD 2 DUP(?) ;dup在汇编中是一条伪指令,用来重复初始化数据 此处就是将两个DWORD的空间都定为?
      .code
      main PROC
      	mov eax,val1 
      	add eax,val2
      	sub eax,val3
      	mov finalVal,eax ;将寄存器eax中保存的运算结果保存在数据段finalVal中
      	call DumpRegs ;显示寄存器信息(registers是寄存器的意思)
      	exit
      main ENDP
      END main
      
    • 整数输入输出与比较

      INCLUDE Irvine32.inc
      
      .data
      var1 dd ?
      var2 dd ? ;dd是指doubleword (4个子节)
      
      .code
      
      main PROC
      	
      	call ReadInt ;读入Interger,存入eax
      	mov var1,eax
      	
      	call ReadInt
      	mov var2,eax
      	
      	cmp var1,eax ;比较var1和eax的大小
      	jl L1		 ;跳转到L1
      	mov eax,var1
      L1:	call WriteInt ;输出eax中存放的数
      
      	exit
      main ENDP
      
      END main
      
    • 条件判断与分支

      INCLUDE Irvine32.inc
      
      .data
      mark DD ?
      Excellent db 'Excellent',0 ;''中的字符串不含结束字符0,需要手动添加
      Good db 'Good',0
      Pass db 'Pass',0
      Failed db 'Failed',0
      
      .code
      main PROC
      	call ReadInt
      	mov mark,eax
      	cmp eax,90
      	Jb L1
      	mov edx,offset Excellent
      	jmp final
      
      L1:	cmp eax,80
      	jb L2
      	mov edx,offset Good
      	jmp final
      
      L2:	cmp eax,60
      	jb L3
      	mov edx,offset Pass
      	jmp final
      
      L3:	mov edx,offset Failed
      
      final:	call WriteString ;WriteString的使用与EDX有关
      	exit
      main ENDP
      
      END main
      
      
    • 输入10个整数比较大小

      INCLUDE Irvine32.inc
      .data
      arr dd 10 dup(?)
      .code
      main PROC
      	mov ebx,0
      again1:	cmp ebx,10
      	jae final1
      	call ReadInt
      	mov [arr + 4*ebx],eax
      	add ebx,1
      	jmp again1
      	
      final1:
      	
      	mov ebx,1
      	mov eax,[arr] 
      again2:	cmp ebx,10
      	jae final2
      	cmp eax,[arr + 4*ebx]
      	jae L1
      	mov eax,[arr + 4*ebx]
      L1:	
      	add ebx,1
      	jmp again2
      	
      final2:
      	call WriteInt
      	exit
      main ENDP
      
      END main
      
    • 判断32位以内自然数是否为素数

      
      INCLUDE Irvine32.inc
      ;正整数判断是否是素数
      .data
      val1 dd ?
      val2 dd 2
      YES db 'Yes',0
      NO db 'No',0
      .code
      main PROC
      	call ReadInt 
      	cmp eax,1
      	jz result1
      	mov val1,eax
      	mov edx,0
      	div val2
      	mov ecx,eax
      	mov edx,0
      
      again:	mov edx,0
      	mov eax,val1
      	cmp ecx,1
      	jz result2
      	div ecx
      	cmp edx,0
      	jz result1
      	;sub ecx,1
      	loop again
      	  
      	
      result1: mov edx,offset	NO
      	call WriteString
      	jmp over
      result2: mov edx,offset	YES
      	call WriteString
      over:	
      	exit
      main ENDP
      
      END main
      
    • 输入整数输出不带加号整数

      INCLUDE Irvine32.inc
      .data
      val1 dd ?
      val2 dd 10
      msg db '0',0
      .code
      main PROC
              call ReadInt
              mov val1,eax
              mov ecx,0
      
      again:  mov edx,0
              mov eax,val1
              div val2
              mov val1,eax
              mov eax,edx
              add al,'0'
              push eax
              add ecx,1
              mov ebx,val1
              cmp ebx,0
              jz over
              jmp again
      
      over:
      again1: 
              pop eax
              call WriteChar
              loop again1
              
          exit
      main ENDP
      
      END main
      
    • ret使用案例

      INCLUDE Irvine32.inc
      .data
      msg1 db 'Hello world',0
      msg2 db 'Goodbye',0
      .code
      main PROC
              push next
              jmp f1
      next:   call crlf
              lea edx,msg2
              call writeString
              exit
          
      main ENDP
      
      f1 PROC
          lea edx,msg1
          call writeString
          ret
      f1 ENDP
      END main
      
      
      
    • 进制转换

      INCLUDE Irvine32.inc
      .data
      val1 dd ?
      val2 dd ?
      arr db '0123456789ABCDEF'
      .code
      main PROC
              call ReadInt
              mov val1,eax
              call ReadInt
              mov val2,eax
              call change
      
              exit
      main ENDP
      
      change PROC
              mov ecx,0
              
      again:  mov edx,0
              mov eax,val1
              div val2
              mov val1,eax
              mov al,[arr+edx]
              push eax
              add ecx,1
              mov ebx,val1
              cmp ebx,0
              jz over
              jmp again
      over:
      again1: 
              pop eax
              call WriteChar
              loop again1
              ret
      change ENDP
      
      END main
      
    • 选择排序(升序)

      INCLUDE Irvine32.inc
        .data
        arr dd 1,100,-9,-8,-100,2000,435,875,-2321,0
        arrLen = ($ - arr) / 4
        tempLen dd arrLen
        msg1 db "Before sort : ",0
        msg2 db "After sort : ",0
        .code
        main PROC
            
            mov edx,offset msg1
            call WriteString
            call crlf
        
            mov esi,0
        again1:
            cmp esi,arrLen
            jz next1
            mov eax,arr[4*esi]
            call WriteInt
            mov al,' '
            call WriteChar
            inc esi
            jmp again1
            
        next1:
            call crlf
            mov esi,1
            mov ecx,tempLen
        again2:  
            cmp esi,arrLen
            jz next2
            call findMaxIndex
            dec ecx
            ;call WriteInt
            ;call crlf
            mov ebx,[arr+4*ecx]
            mov edx,[arr+4*eax]
            mov [arr+4*ecx],edx
            mov [arr+4*eax],ebx
            ;mov eax,ecx
            ;call WriteInt
            ;call crlf
            inc esi
            jmp again2
        
        next2:
        
            mov edx,offset msg2
            call WriteString
            call crlf
            mov esi,0
        again3:
            cmp esi,arrLen
            jz next3
            mov eax,arr[4*esi]
            call WriteInt
            mov al,' '
            call WriteChar
            inc esi
            jmp again3
        
        next3:
            exit
        main ENDP
        
        findMaxIndex PROC
            push esi
            
             
            mov esi,0
            mov eax,0
        again:
            inc esi
            cmp esi,ecx
            jz over
            mov ebx,[arr+4*eax]
            cmp ebx,[arr+4*esi]
            jge next
            mov eax,esi
        next:
            jmp again
        
        over:
            
            pop esi
            ret
        findMaxIndex ENDP
        
        END main
      
    • 求最大公约数

       INCLUDE Irvine32.inc
       .data
       m dd ?
       n dd ?
       r dd ?
       msg db '最大公约数是:',0
       .code
       main PROC
           call ReadInt
           mov m,eax
           call ReadInt
           mov n,eax
      
           cmp m,eax   ;保证 m >= n  
           jge next
           xchg m,eax
           xchg n,eax  
       next:
      
           call gcd
      
           mov eax,m
           mov edx,offset msg
           call WriteString
           call WriteInt 
           exit
       main ENDP
       gcd PROC
           again:
           mov edx,0
           mov eax,m
           mov ebx,n
           div ebx
           mov m,ebx
           mov n,edx
           cmp edx,0
           jz next1
      
           jmp again
       next1:
           ret
       gcd ENDP
       END main
      
    • 求完美数

      INCLUDE Irvine32.inc
      .data
      k dd ?
      count dd 0
      .code
      main PROC
          call ReadInt
          mov k,eax
          mov esi,0
          mov ebx,2
      again:
          cmp esi,k
          jz next
          call findNum
          mov esi,count
          inc ebx
          jmp again
      next:
          exit
      main ENDP
      findNum PROC
          push eax
          push esi
          mov esi,1
          mov ecx,0
      again1:
          cmp esi,ebx
          jz over
          mov eax,ebx
          mov edx,0
          div esi
          cmp edx,0
          jnz next1
          add ecx,esi
      next1:
          inc esi
          jmp again1
      over:
          cmp ecx,ebx
          jnz next2
          mov eax,ebx
          call WriteInt
          call crlf
          inc count
      next2:
      s
          pop esi
          pop eax
          ret
      findNum ENDP
      END main
      
      
    • 正整数的素数分解

      INCLUDE Irvine32.inc
      .data
      n dd ?
      temp dd ?
      divNum dd 10
      .code
      main PROC
          call ReadInt
          mov temp,eax
          mov n,eax
          mov ecx,0
      again3:
          cmp eax,0
          jz next4
          mov edx,0
          div divNum
          push edx    
          inc ecx
          jmp again3
      next4:
      
      again4:
          cmp ecx,0
          jz next5
          pop eax
          add al,'0'
          call WriteChar
          dec ecx
          jmp again4
      
      next5:
      
          mov al,'='
          call WriteChar
          mov esi,2
      
      again:
          mov ecx,0
          call findPrime
          cmp ecx,0
          jz next2
          mov eax,esi
          add al,'0'
          call WriteChar
          mov al,'^'
          call WriteChar
          mov eax,ecx
          add al,'0'
          call Writechar
          mov eax,temp
          cmp eax,1
          jz next3 
          mov al,'*'
          call WriteChar
      next3:
      next2:
          cmp esi,n
          jz next 
          inc esi
          jmp again
      next:
          exit
      main ENDP
      
      findPrime PROC
      
      again1:
          mov edx,0
          mov eax,temp
          div esi
          cmp edx,0
          jnz next1
          inc ecx
          mov temp,eax
      
          jmp again1
      next1:
          ret
      findPrime ENDP
      
      END main
      
    • byte ptr [arr]测试

      INCLUDE Irvine32.inc
      .data
      arr1 dd 1,5,3,6,234,2,56
      arr2 db 123,4,44,111,67,34,7,88
      arr3 dw 113,3455,67,234,777,432,1,3,5
      .code
      main proc
          mov eax,dword ptr [arr1] ;使用时源操作数和目的操作数类型需要对等
          call WriteInt
          mov al,byte ptr [arr1]
          call WriteInt
          mov ax,word ptr [arr1]
          call WriteInt
          exit
      main endp
      
      end main
      
  • 相关阅读:
    解决firefox的button按钮文字不能垂直居中
    郁闷呢
    我的生活走入正轨
    空闲的日子写写日记
    今天可以写心事
    终于可以写字了
    有地方可以发表自己的心事了。
    Shopify:产品详情页购买按钮下方支持的支付方式图标如何修改?
    Wordpress报错:Allowed memory size of 134217728 bytes exhausted
    安装Xshell报错:由于找不到MSVCR110.dll,无法继续执行代码。重新安装程序可能会解决此问题
  • 原文地址:https://www.cnblogs.com/beyondzones/p/15011475.html
Copyright © 2011-2022 走看看