微機原理與系統設計筆記4 | 彙編語言程式設計與其他指令

来源:https://www.cnblogs.com/Roboduster/archive/2023/02/16/17126073.html
-Advertisement-
Play Games

打算整理彙編語言與介面微機這方面的學習記錄。本部分介紹彙編語言程式設計以及一些跟程式設計密切相關的指令類。 參考資料 西電《微機原理與系統設計》周佳社 西交《微機原理與介面技術》 課本《彙編語言與介面技術》王讓定 小甲魚《彙編語言》 1. 彙編程式結構/框架 段定義偽指令 程式不同的信息要定義在不同 ...



  • 打算整理彙編語言與介面微機這方面的學習記錄。本部分介紹彙編語言程式設計以及一些跟程式設計密切相關的指令類。

  • 參考資料

    • 西電《微機原理與系統設計》周佳社
    • 西交《微機原理與介面技術》
    • 課本《彙編語言與介面技術》王讓定
    • 小甲魚《彙編語言》

1. 彙編程式結構/框架

  • 段定義偽指令

    程式不同的信息要定義在不同的段中,該指令就

    SEGNAME SEGMENT [定位類型][組合類型]['類別']
    	;段實體
    SEGNAME ENDS
    
    • 這裡要註意首尾一致,也就是SEGNAME要一樣。
    • 有SEGMENT就要有ENDS,像C語言大括弧一樣。
    • 定義了一個段名SEGNAME,就具備了段地址屬性。

    定位類型是告訴彙編器這個邏輯段起始地址的要求,取值有:

    1

    常用是PARA 節型。

    組合類型是多模塊程式設計中告訴鏈接器器,不同模塊中同段名同組合類型的邏輯段如何鏈接。

    課程只要求單模塊編程,所以組合類型不瞭解也罷。

    老師這裡提了STACK型(說明本段是堆棧段),例子:

    STACK SEGMENT STACK
          DB 256 DUP(?)
    STACK ENDS	
    

    註意這裡第一行的第二個STACK聲明為組合類型後,就不需要在代碼段設置SS和SP了,如果不寫這個組合類型聲明,代碼段還是需要聲明。

    另一個組合類型是NONE類型,預設型。其他的暫且不提。

    類別,如果要寫,需要加上單引號說明,只告訴程式員這個段是乾什麼的,類似於段前註釋,沒有其他用處。

2. 8086彙編程式完整結構

;定義堆棧
STACK SEGMENT STACK
      DB 256 DUP(?)
      ;定義堆棧段,找16可以整除的,SP = 棧底+1
TOP LABEL WORD 
;TOP為變數名;TOP 不占用記憶體位置,獲取SP的地址;LABEL偽指令只有存儲單元地址、類型屬性但是不占記憶體位置
STACK ENDS

;定義數據段(可能有多個)
DATA SEGMENT 
;實體

DATA ENDS
X_BYTE LABLE BYTE;這樣就可以直接使用34H這個位元組型變數X_BYTE,而不用對X_WORD進行類型轉換。
X_WORD DW 1234H
;定義代碼段
CODE SEGMENT 
ASSUME CS:CODE,DS:DATA,ES:DATA;段定址偽指令,告訴彙編程式段名和段的關係
ASSUME SS:STACK
;這種關係是承諾關係,並沒有向這些寄存器賦值,還需要再賦值

START:MOV AX,DATA
      MOV DS,AX;數據初始化,不能直接DATA給DS
	  MOV ES,AX
  /*  MOV AX,STACK
	  MOV SS,AX 
	  MOV SP,OFFSET TOP*/
	  ;相當於STACK組合類型的作用
/*    MOV AH,4CH
      INT 21H */   
      ;DOS系統的4CH對應的中斷子程式,此處是結束用戶程式返回操作系統
CODE ENDS;代碼段結束
END START(CS:IP);整個程式結束
;初始化CS,ip,放置這段代碼。其實是交給OS來做了,用戶無權。

3. 其他偽指令

上面的完整程式結構中已經提了很多偽指令,簡單提一提其他偽指令。

  • EQU 和 = 偽指令

    VAR_NAME EQU/= Exp;給表達式賦一個名稱
    ;用EQU定義時,這個VAR_NAME只能EQU定義一次,用=定義則可以再次定義
    
  • ORG偽指令

    ORG 表達式
    

    為後續指令指定段內偏移地址,可以人為使字型數據對齊。

    舉個例子:操作系統中的boot.asm中對於 org07c00h是這樣註釋的:“告訴編譯器程式載入到7c00h處”:

    org 07c00h
    ;告訴編譯器程式載入到7c00h處
    mov ax,cs
    mov ds,ax
    mov es,ax
    
    call screen
    jmp $
    
    screen:mov ax,bootmsg
    mov bp,ax
    mov cx,16
    mov ax,01301h
    mov bx,000ch
    mov dl,0
    int 10h
    ret
    
    bootmsg:db'Hello OS world!' 
    times 510-($-$$) db 0 
    dw 0xaa55
    

想直接聽後面的了,不想聽前面的指令系統了,折磨。

4. 子程式設計及其調用返回指令

這部分是3.11和4.5合在一起講,前者是子程式調用返回指令,後者是子程式程式設計。

子程式/過程就是彙編語言里的函數,將一些功能性、重覆性的代碼放到子程式中調用,可以讓自己的代碼更加清晰。我們需要在主幹代碼調用子程式,在子程式結束時返回

4.1 子程式指令

  • 子程式定義:

    以過程形式定義:

    son_name1 PROC [類型] 
    	;主體
    	RET;子程式返回,段間子程式可以用RETF
    son_name1 ENDP
    

    上面代碼中的[類型]:

    • 段內子程式:NEAR,可以預設
    • 段間子程式:FAR,不能預設

    以標號形式定義(比較方便):

    LABEL:
    	;主體
    	RET
    
  • 子程式調用與返回

    • 段內子程式調用與返回

      CODE SEGMENT
           ASSUME CS:CODE
      START:
      
      	 ……
      	CALL SUB1;段內直接定址
      	; call near sub1的near可以預設
      	; ……CS不發生改變
      	
      	MOV AH,4CH
          INT 21H	
      	
      SUB1 PROC
      	RET
      SUB1 ENDP
      CODE ENDS 
      END START
      
      • 執行call時,IP放入堆棧(SP-2)字單元;
      • IP+DISP(到跳轉標號的相對位移)得到新的IP;
      • RET 執行後,棧頂彈出放回IP,SP+2;
    • 段間子程式調用與返回

      CODE1 SEGMENT
           ASSUME CS:CODE1
      START:
      
      	 ……
      	CALL FAR PTR SUB1;段內直接定址
      	 ;……CS:IP(CS和IP都要保存)
      	
      	MOV AH,4CH
          INT 21H	
      CODE1 ENDS
      
      CODE2 SEGMENT
      	ASSUME CS:CODE2
      SUB1 PROC FAR
      	……
      	RETF;也可以直接寫為RET
      SUB1 ENDP; 子程式/過程結束指令
      CODE2 ENDS
      
      • 執行call時,cs先入棧(sp-2),ip再入棧(sp-2),
      • RET 時,完成CS IP出棧。
      • 這裡註意一個事情,在子程式內使用了堆棧push pop後,需要在最後複原至調用初,才能將子程式正確返回。

4.2 信息的保護和恢復

也就是主程式和子程式會復用一些寄存器和存儲器單元,調用子程式時需要保存主程式的這些東西,這個過程在主程式或子程式中都可以進行,通常在子程式中進行。

4.3 主程式和子程式的參數傳遞

寄存器、存儲器、堆棧都能傳遞參數。

至於怎麼傳遞,其實就是一個程式設計者自己規定的過程,比如主程式將參數放到AX,子程式從AX來拿。在堆棧參數傳遞參數方式時,需要用到BP指針。

4.4 子程式說明文件

作為學生,這部分當然不寫,瞭解一下吧。

  1. 子程式名
  2. 子程式功能
  3. 入口參數與傳遞方式
  4. 出口參數與傳遞方式
  5. 子程式用到的寄存器

4.5 嵌套、遞歸、可再用

  • 嵌套子程式設計時註意堆棧溢出
  • 設計自上而下,調試自下而上,保證調用的程式無問題
  • 遞歸子程式:自己調用自己,需要考慮出口條件
  • 可再用是一種提醒,感覺子程式可再用有點複雜,要在一次調用還未結束時又調用該子程式,要註意正確返回。

這一節偏實踐,紙上論道不可取。用到再說吧,個人彙編編程經驗感覺子程式這部分不難,這是要保護信息恢覆信息。可再入不大明白(那就儘量不設計)。

這裡有幾個子程式設計常式,我感覺還不錯。例3.32、例3.33、例3.34。

5. 中斷調用與返回

3.12,結合了一部分第8章中斷系統的知識。

說實話,"中斷" 這個詞通過操作系統OS這個課已經比較瞭解了,這裡從微機角度溫習一下。

但感覺這部分講的有點亂。

5.1 中斷

  • 中斷:是CPU執行程式時,由於某個事件的發生,暫停當前正在執行的程式,去執行處理這個事件的中斷服務程式,執行完畢後再返回原程式繼續執行。

  • 中斷分為外部可屏蔽中斷、外部不可屏蔽、內部中斷(軟體中斷)。具體如下圖:

    2

  • 為了識別中斷形式並採取對應服務,CPU都編排了中斷類型號,這個具體CPU要查手冊。

    8086中:

    • 除法錯:n=0

    • 單步中斷:n=1

    • 斷點中斷:n=3

      int 3指令可以臨時替換原有的指令,叫做設置斷點,程式執行到int 3時,會進入類型3的中斷服務程式,可以在中斷服務程式中讀原執行環節,來觀察哪裡有問題,之後再返回原程式,繼續執行。

    • int n

      更泛指,比如DOS系統功能調用INT 21H

  • 後續會學上圖中的中斷處理硬體,8259A。

5.2 中斷向量

中斷向量其實是中斷服務程式的指針,存放中斷服務程式的入口地址。

3

系統將中斷向量組織成中斷向量表(換算表),放置在存儲器最低地址0000H開始的1024個單元。每個中斷向量占4B:

  • 入口地址的偏移地址在低2位元組,段基址在高2位元組。所以調用時也是對應交給IP和CS。

  • 換算關係是:中斷類型號*4就是該中斷向量在中斷向量表中的開始地址

  • 中斷向量表在OS初始化時就設置了。在自己寫操作系統時,可以設置這個表。

5.3 中斷響應過程

A. 內部中斷

  • 中斷調用指令: INT n

  • CPU執行以下操作:

    1. 保護現場:PSW IP CS 入棧

    2. 清除 IF TF 標誌(TF清0說明中斷期間CPU不會再中斷)

    3. 從中斷向量表中取出中斷向量給CS IP賦值。

      如果是外部硬體中斷,不知道這個查表要用的n,CPU需要從外部硬體中獲取中斷類型號。

    4. 轉到中斷服務程式執行。

  • 中斷服務最後一句,中斷返回:IRET,將IP CS PSW出棧。返回原程式。

B. 外部中斷

再來說說外部中斷的中斷號獲取:

4

  • 簡單先說一下上圖的情況,是在說8086的中斷引腳情況

  • NMI不可屏蔽中斷就不提了。

  • INTR引腳:

    • 受IF中斷標誌位控制,當IF=1時,CPU執行完當前指令後,就會響應硬體中斷;
    • 下麵就是硬體中斷的響應過程
      1. 保護現場:PSW IP CS 入棧
      2. 清除 IF TF 標誌(此時IF=0)
      3. CPU從外部中斷控制器8259中獲取中斷號;
        • 首先需要CPU告訴8259,它現在響應了中斷;即INTA引腳發出負脈衝給8259.
        • 其次CPU要問要響應哪個硬體中斷,即再次發出INTA負脈衝給8259(第二次)
        • 8259收到後將內部寄存器存儲的N值送到數據線。N不固定。
      4. 根據拿到的N,從中斷向量表中取出中斷向量給CS IP賦值。
    • IRET中斷返回。
  • 只有一個8259則只能控制8個硬體,可以採用8259級聯的方式來增加控制硬體數。

    這裡可以算控制硬體個數,但是不難。

5.4 中斷程式結構/框架

中斷很像子程式,但是有所不同,因為中斷隨時可能發生,自己寫的主程式無法預料何時發生中斷,所以中斷程式內部開始時應當有保護現場操作,返回時有恢復現場操作

INT_NAME:
	PUSH AX ;保護現場操作
	;......
	PUSH SI
	
	; 主體
	; ...
	POP SI ;恢復現場操作
	POP AX
	IRET ;中斷返回

5.5 系統調用

系統調用分為兩類

  • BIOS功能調用,這部分主要是操作硬體。
  • DOS功能調用,講道理這是操作系統提供的服務,偏向軟體。自己寫的OS可以自行設計功能調用。8086彙編主要面向DOS操作系統(這點是過時的地方...)。

A. DOS功能調用

DOS系統功能調用的方法一般可分為以下幾步:

  1. 設置所要調用功能的入口參數。
  2. 在AH寄存器中存入所要調用功能的功能號。
  3. INT 21H指令自動轉入中斷子程式入口。
  4. 相應中斷子程式運行完畢,可按規定取得出口參數。

常用的DOS功能調用有 INT 21H,AH里放參數決定具體服務(如果需要參數),比如放入01H就是等待用戶從鍵盤輸入一個字元,回顯,按鍵的ASIIC碼放到AL中。

補充一個小知識:

  • ASCII數字如何得到數字:

    AND AL,0FH;消除高位即可
    

這裡舉了一個鍵盤按鍵並回顯的常式,跳轉表法,挺巧妙的,老師講了兩個跳轉表法的實現方式,第一個是在課本4.25,第二個就是將跳轉表定義到了代碼段。

其他的不再寫了,主要就是設置參數,設置調用號再輔以子程式實現相應的功能:

  • 02H向屏幕輸出一個字元
  • 06H 控制台輸入、輸出
  • 09H 向屏幕輸出一串字元(字元串數據變數定義時必須要有結束符 '$'

DOS與BIOS 系統調用一覽:https://blog.csdn.net/icurious/article/details/51628343

0AH 從鍵盤輸入一串字元(有等待功能,鍵盤輸入回車時結束輸入並顯示),此時必須有鍵盤緩衝區(存儲器中):

DATA  SEGMENT
……
    KEYbuf1 DB 20H;允許用戶輸入的字元個數
	        DB ?;用來統計實際輸入的字元數
	KEYbuf2 DB 20H DUP(?);存放實際輸入的字元串
    ;鍵盤輸入緩存區
DATA ENDS
CODE SEGMENT 
     ASSUME CS:CODE,DS:DATA
START:
     MOV AX,DATA
	 MOV DS,AX
	 ……
	 ;從鍵盤輸入一串字元,有回顯
	 MOV AH,0AH
	 MOV DX,OFFSET KEYbuf1
	 INT 21H
......

課後題4.39題回頭看看。

顯示系統時間:(這裡是DOS系統調用獲取,當然也可以用cmos獲取--王爽彙編實驗14,不過還沒有講到與外設交互)

;返回值:CX = year (1980-2099) DH = month DL = day AL = day of week (00h=Sunday)
data segment
data ends
 
code segment
        assume ds:data, cs:code
 
main:
        mov ax, data
        mov ds, ax
 
        mov ah, 2ah
        int 21h
 
        mov ah, 4ch
        int 21h
code ends
end main

B. BIOS 功能調用

  • 00H 設置顯示器,清屏功能

    MOV AH,00H
    MOV AL,3;彩色文本,且大小是80X25
    MOV BX,0;頁數,與顯卡數據緩存區有關
    INT 10H
    
  • 02H 設置游標位置

    MOV AH,02H
    MOV DH,行
    MOV DL,列
    MOV BL,0
    INT 10H
    

6. 字元串操作指令

對傳送指令再升級,對一組數據進行整體操作。

  1. 源字元串和目的字元串均為隱含定址

    • 如果源串在存儲器中,則存儲器的的地址在DS:SI;

      若源串在REG中,位元組:AL,字:AX

    • 目的字元串,必須在ES:DI(必須對應)

      在REG同上

  2. 如果源串目的串在存儲器中,則CPU執行時,SI、DI的地址會自動改變,這個改變受DF的控制:

    • DF=0時, +;

    • DF=1時, -;

      這裡在編程時需要向其賦值確保其值。如確保DF==0: CLD

    • ± 的程度受操作數類型大小的控制(放在頭部或尾部):若為位元組操作自動+1/-1,若為字操作自動+2/-2

  3. 在串操作指令的左邊可以增加重覆首碼,重覆次數隱含放在CX中

    • REP首碼:等效於LOOP,放在MOOSB LODSW STOSB前
    • REPZ/REPE:等效於LOOPZ,放在CMPSB,SCASB之前,(相等重覆)
    • REPNZ/REPNE:等效於LOOPNZ,放在CMPSW SCASB之前,(不相等重覆)
  • 串傳送指令

    MOVSB
    MOVSW
    
  • 串比較指令

    CMPSB
    CMPSW
    

    源字元串減去目的字元串(對應位置依次比較),設置狀態寄存器

  • 串掃描指令

    SCASB
    SCASW
    
  • 串裝入指令

    LOOSB
    LOOSW
    
  • 串存貯指令

    STOSB
    STOSW
    

6.1 字元串傳送指令

三種書寫格式:MOVSB、MOVSW,MOVS DST,SRC

  • 以MOVSB存儲器操作為例:

    1. ES:DI<-- DS:SI

      源串段地址可以更改(如使用第三種書寫格式),而目的串寄存器不可更改,見書P74頁ES段到ES段的傳送。

    2. SI和DI會自動+/-

    3. 若加上REP重覆首碼,則執行第三步,CX-1->CX,若CX≠0,則重覆執行MOVSB

      例題見P75 例3.36

      DATA SEGMENT 
           BUFFER1 DB 100 DUP(?)
           BUFFER2 DB 100 DUP(?)
      DATA ENS
      CODE SEGMENT
           ASSUME CS:CODE,DS:DATA,ES:DATA
      START:
           MOV AX,DATA
           MOV DS,AX
           MOV ES,AX
           LEA SI,BUFFER1
           MOV DI,OFFSET BUFFER2
      	 CMP SI,DI
      	 //JB DZJXFX
      	 MOV CX,100
         /*CLD ;DF=0
      	 REP MOVSB	 字元串指令*/
      	 ;下麵是數據指令完成的
      N1:	 MOV AL.[SI]
      	 MOV ES:[DI],AL
      	 INC SI
      	 INC DI
      	 LOOP N1
      	 MOV AH,4CH
      	 INT 21H
      CODE ENDS
      END STRAT
      

      源串和目的串重疊,視情況正向搬運或者逆向搬運。(有兩種情況:

      • 若源串起始地址比目的串起始地址高:SI DI增大方向搬
      • 低:SI DI 減小方向搬

6.2 字元串比較指令

CMPSB CMPSW CMPS DST,SRC

以CMPSW為例:(源減去目的,這與數據比較指令不同)

  1. SRC-DST,也即是(DS:SI)-(ES:DI),設置6個狀態寄存器,若ZF==1,說明相等。

  2. SI DI 自動+/- 2

  3. 若有重覆首碼REPZ,(CX-1)-->CX,若CX≠0且ZF=1,重覆執行CMPSW

    若首碼REPNZ,則修改上麵條件改為ZF=0即可:若CX≠0且ZF=0

常式在書P76 例3.37

MOV SI,OFFSET BUFFER1
MOV DI,OFFSET BUFFER2
MOV CX,100
CLD
REPNZ CMPSW
JZ Find
MOV ADDR,0FFFFH
JMP EXIT

Find:SUB SI,2
MOV ADDR,SI
……
MOV AH,4CH
INT 21H

6.3 字元串掃描指令

SCASB、SCASW、SCAS DST,適用於在字元串中找某個字元或篩出某個字元。

以SCASB為例:

  1. 源減去目的:AL-(ES:DI),結果設置6個標誌,主要是ZF,ZF=1,則找到
  2. DI+/-1,沒用到SI
  3. 若有重覆首碼REPNZ,(CX-1)-->CX,若CX≠0且ZF=0,重覆執行SCASB;退出迴圈體可以測試ZF==1來判斷找到。

6.4 字元串裝入指令

LODSB LODSW,意為將源字元串(存儲器中)讀出到AL/AX中。

以LODSB為例:

  1. AL<--(DS:SI)
  2. (SI+/-1)-->SI

這裡老師講了兩個電腦通信的例子,8250,emm,略去。

6.5 字元串存貯指令

STOSB STOSW STOS

以STOSB為例:

  1. AL-->(ES:DI)
  2. DI±1-->DI
  3. 可以帶REP重覆首碼

常式P79 例3.39

DATA SEGMENT 
    BUFFER1 DW 
    BUFFER2 DB	
DATA ENDS

CODE SEGMENT
    ASSUME CS:CODE,DS:DATA,ES:DATA
STRAT:
    LEA DI,BUFFER1
	MOV AX,0000H
	MOV CX,200
	CLD
    REP STOSW
    /*也可以改為
    L1:MOV ES:[DI],AX
    	INC DI
    	INC DI;字,所以減了兩次
    	LOOP L1
    */
	LEA DI,BUFFER2
	MOV AL,55H
	MOV CX,256
	CLD
	REP STOSB

例子2,P112例4.7

7.輸入輸出指令、其他指令

輸入輸出指令針對I/O操作。

7.1 輸入指令

格式:IN DST, SRC,這裡SRC是埠地址(8086的前16根地址線定址),

  • 若埠地址在0~255之間,SRC可以採用直接定址:

    IN AL,埠地址
    
  • 若埠地址>255,需採用間接定址(都可以採用間接定址)。

    MOV DX,埠地址
    IN AL,DX;觸發IO操作的地址寄存器
    ;符合語法,雖然DX沒加括弧,但是合法
    ;這裡的意思比較抽象,就是地址放在DX中,in進來的內容放在AL中
    

上述所說是位元組操作,若為字操作,則AL改為AX。

7.2 輸出指令

格式:OUT DST, SRC,操作數交換一下,其他要求與輸入完全一致。

5

  • 輸入輸出指令要使用外部匯流排,一個匯流排周期包括4個基本的時鐘周期,而輸入輸出指令包括了5個時鐘周期,也就是輸入輸出指令工作時間要多一個周期T1、T2、T3、TW(多出的等待周期)、T4

這裡老師舉了一個外設操作例子(P61最後~P62),講得很好,很整體,甚至包括了IO地址解碼電路(控制信號也要參與解碼d),但學到外設再說。

7.3 其他指令

這部分直接看書更快。

  • 標誌位處理指令,詳見筆記3的標誌位處理指令(7個)

  • 處理器控制指令 5個

    • 空操作指令 NOP,什麼也不做,占用機器3個時鐘周期,用於延時、等待。

      看一個延時子程式(雙層迴圈):

          MOV BX,3;一個值,外層迴圈的次數
      L2: MOV CX,0;65536
      L1: NOP
          LOOP
      	DEC BX
      	JNZ L2
      
    • 暫停指令 HLT,使CPU進入暫停狀態,不再向下執行。退出條件:

      • RESET信號有效直接複位,或者NMI不可屏蔽中斷。此時退出但也不會執行後續程式。

      • INTR可屏蔽中斷發生,CPU保護當前程式現場(IP指針在hlt的下一句),保護了當前程式,返回後則繼續執行後續程式。

        這個可以用於CPU跟外設的同步(用的不多)

    • WAIT等待指令,每隔5個時鐘周期對TEST引腳狀態進行測試,當TEST線低電平則退出等待,執行下一條指令。

7.4 巨集指令

巨集定義指令定義出來的自定義指令。巨集指令必須在程式最開頭定義。

巨集指令名 MACRO <形式參數>;參數多於1項,用逗號分隔
; ... ;指令或偽指令構成的巨集體
ENDM ;巨集定義指令結束,註意這裡結束不需要寫巨集名

例子:

;定義一條巨集指令,將<某通用寄存器REG><左/右移><若幹次>
SHIFT MACRO REG,DIR,N
      MOV CL,N
	  DIR REG,CL;或者:S&DIR REG,Cl
	  ; 因為 SHL,SHR,SAL,SAR 都以S開頭
      ENDM
      
;某程式體中,將AX邏輯左移3次:巨集調用如下:
SHIFT AX,HL,3

當彙編器彙編到這個巨集調用時,會將前面的巨集定義的形參替換掉,並插入此處(巨集展開)。

巨集調用和子程式都可以來放一個經常調用的功能模塊,但區別在於:

  • 巨集調用每次調用就都將語句插入(巨集展開),浪費空間,但節省調用時間(沒有CALL和RET)。
  • 子程式彙編後空間固定,所以節省空間,但浪費時間。
;定義一條巨集指令,完成游標回車換行
CRLF MACRO
     MOV AH,2
	 MOV DL,0DH
	 INT 21H
	 MOV AH,2
	 MOV DL,0AH
	 INT 21H
ENDM

一個值得研討的點:當巨集指令中有迴圈或分支,也就是有標號,主程式多次巨集調用時,會提示標號重覆:

  • 此時應當將標號聲明為 局部標號 LOCAL NEXT,彙編時自動分配不同標號。

    ;數據塊搬家
    DATAMOV MACRO DATA1,DATA2,N
            LOCAL NEXT;聲明NEXT是局部標號
            LEA SI,DATA1
    		LEA DI,DATA2
    		MOV CX,N
    NEXT:	MOV AL,[SI]
    		MOV [DI],AL
    		INC SI
    		INC DI
    		LOOP NEXT
    		ENDM
    	; ……
    	; 主程式中的巨集調用
    	DATAMOV buf1,buf2,100
    	;彙編展開;
    	; LOOP NEXT變為LOOP ??0000
    	; ......
    	DATAMOV buf3,buf4,200
    	;彙編展開
    	;LOOP NEXT變為LOOP ??0001
    

除了上面介紹的,還有重覆巨集,格式是:

; 重覆巨集
REPT <重覆次數>:
	;重覆程式體
ENDM ; 也是以此結尾

; 不定重覆巨集
IRP <形參>, < <參數1>,<參數2>,...>
	;重覆程式體
ENDM ; 也是以此結尾

;不定字元重覆巨集
IRPC <形參>,<字元串>;將字元串中的各個字元依次賦給形參重覆執行指定程式體
;重覆程式體
ENDM ; 也是以此結尾

IRPC CC,AAB
	ADD AX,CC&X
ENDM
/* 相當於
	ADD AX,AX
	ADD AX,AX
	ADD AX,BX
*/

8. 分支程式設計

  • 單分支程式;多分支程式

  • 要點1是分支條件:

    1. 某種運算產生條件
    2. 判斷條件是否成立
    3. 根據條件產生分支

    多分支程式結構與上面幾步類似,此外還可以通過跳轉表來實現多分支

  • 要點2是轉移範圍:

    • 條件轉移只能在當前IP的-128~+127之間轉移
  • 要點3是程式調試時要保證每個分支結果的正確性(因為單次調試可能只走了其中一條分支)

  • 要點4:分支程式設計要點

    1. 正確選擇分支條件,確定條件轉移指令
    2. 編寫程式時,要保證每個分支的正確性
    3. 調試時逐個分支調試

例子:書P112例4.7,兩遞增數組合併。

STACK SEGMENT STACK 'STACK'
	DW 100H UP(?)
TOP LABLE WORD
STACK ENDS

DATA SEGMENT
DAT1 DW 10
	 DB 10H,25H,67H,73H,83H,95H,0A8H,0C2H,0E6H
DAT2 DW 13
     DB 05,12H,45H,58H,65H,67H,70H,76H,88H,92H,0CDH,0DEH

DAT DW ?
	DB 200 DUP(?)
DATA ENDS

CODE SEGMENT
	ASSUME CS:CODE,DS:DATA,ES:DATA,SS:STACK

START:
	; 一些初始化
	MOV AX, DATA
	MOV DS, AX
	MOV ES, AX
	MOV AX, STACK
	MOV SS, AX
	LEA SP, TOP
	; 計算 目標數組 大小
	MOV CX, DAT1
	MOV DX, DAT2
	MOV DAT, CX
	ADD DAT, DX
	; 置位三個指針,ES:SI,DS:BX,ES:DI
	LEA SI, DAT1+2
	LEA BX, DAT2+2
	LEA DI, DAT+2
	CLD
	
L1:
	MOV AL, [BX]
	INC BX
L2:
	CMP AL,[SI]
	JB L3; AL<[SI],去L3
	; AL>=[SI],向下執行
	MOVSB
    DEC CX ;控制迴圈次數,DAT1數組是否比較完成
    JZ L4 ;結束
    JMP L2 ;無條件迴圈L2
L3:
	STOSB ;DAT2區數據傳送到DAT
	DEC DX
	JZ L5	;如果兩邊數組有剩餘
	JMP L1
L4: 
	MOV SI,BX
	DEC SI
	MOV CX,DX
L5:
	REP MOVSB
	MOV AH,4CH
	INT 21H
CODE ENDS
     END START

9. 迴圈程式設計

  • 迴圈程式組成

    1. 初始化部分
    2. 迴圈工作部分
    3. 迴圈參數調整
    4. 迴圈出口判定,返回2或者繼續到5
    5. 結果處理/出口
  • 常式,氣泡排序法程式:書P108頁例4.5

    核心代碼:

    ; 初始化部分
    MOV CH,N-1;外層
    NEXT2:
    	MOV CL,CH; 內層
    	LEA SI,BUF1; 數組首址
    	MOV BL,0; 交換標誌
    
    ; 迴圈工作
    NEXT1:
    	MOV AL,[SI]
    	CMP AL,[SI+1]
    	JAE NOCHG	;AL>=[SI+1]不交換
    ;否則交換
    	XCHG AL,[SI+1]
    	MOV [SI],AL
    	MOV BL,-1 ;有交換,說明未排完
    
    NOCHG:
    	; 調整迴圈條件並判定出口
    	INC SI;
    	DEC CL
    	JNZ NEXT1 ;未到0繼續內層交換
        ;否則繼續外層迴圈
        CMP BL,0;是否已經排好序
    	JZ EXIT ;如果排好序則退出
    	DEC CH	
    	JNZ NEXT2
    
    EXIT: MOV AH,4CH
          INT 21H
    

警惕程式挖空填寫題。


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 工欲善其事必先利其器,在使用Python開發程式之前,在電腦上搭建Python開發環境是必不可少的環節,目前Python最新穩定版本是3.11.1,且支持到2027年,如下圖所示 本文手把手帶你從0 到1搭建Python最新版3.11.1開發環境,堪稱保姆級教程,快快收藏啦~ 一、Python解釋 ...
  • 最近幾年,Java 的技術棧發展的非常快,Java作為一門十分流行的面向對象編程語言,其開發工具也是非常多的,當然因為接觸時間長短以及個人喜好,每個人都有自己的選擇。對此,我對目前市面上常見的Java開發工具做了一些簡單的整理,希望能幫助到一些小伙伴。 一、常見的Java開發工具有哪些? Eclip ...
  • 一、字元串str 大白話的意思其實就是文本類型的數據>>>:引號引起來的部分都是字元串 應用場景:姓名 地址 愛好 name = 'kevin' addr = '浦東新區' hobby = '學習' 定義字元串的四種方式 # 方式1: name = 'kevin' # 方式2: name = "ke ...
  • MQ,中文是消息隊列(MessageQueue),字面來看就是存放消息的隊列。也就是事件驅動架構中的Broker。 快速入門 1.publisher實現 public class PublisherTest { @Test public void testSendMessage() throws I ...
  • 一、安裝PHP 下載:https://windows.php.net/download 我下載的是此時的最新版8.2.3 下載後 解壓目錄 放到C:/tool下麵目錄重命名為PHP 目錄自己定 我這個tool目錄是個人習慣 你可以放到主流的C:\Program Files下也可以放到D盤E盤任何地方 ...
  • 一、部署gitlab 這裡使用的是Centos8,安裝Docker環境 ,這裡不說了,參考:https://www.cnblogs.com/wei325/p/15139701.html gitlab有ce版和ee版,ce版為免費版本;ee版為企業版本,需要收費;企業使用ce版足夠了,這裡用ce版。 ...
  • 概述 工廠方法模式(FactoryMethod),定義一個創建產品對象的工廠介面,讓工廠子類決定實例化那一個產品類。我們把被創建的對象稱為“產品”,把創建產品的對象稱為“工廠”。如果要創建的產品不多,只要一個工廠類就可以完成,這種模式叫“簡單工廠模式”,它不屬於 23 種經典設計模式,它的缺點是增加 ...
  • 常用命令 ip addr 查看ip地址 pwd 顯示當前所有路徑 top 查看進程的cpu、記憶體占用情況 ps -ef | grep -i 進程名字 查看進程運行信息 kill -9 進程pid 強制殺死進程 history 查看歷史命令 !歷史命令編號 執行該歷史命令 shutdown 關機 re ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...