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
    
posted @ 2021-07-14 16:14  NoahQ  阅读(759)  评论(0编辑  收藏  举报