小程式 預定義樣式 自用
實驗3 多個段的彙編源程式編寫與除錯
實驗任務1
使用任意文字編輯器,錄入彙編源程式task1.asm。
assume cs:code, ds:data data segment db 'Nuist' db 5 dup(2) ;db 2,3,4,5,6 data ends code segment start: mov ax, data mov ds, ax mov ax, 0b800H mov es, ax mov cx, 5 mov si, 0 mov di, 0f00h s: mov al, [si] and al, 0dfh ;大寫 mov es:[di], al mov al, [5+si] mov es:[di+1], al inc si add di, 2 loop s mov ah, 4ch int 21h code ends end start
閱讀源程式,從理論上分析原始碼的功能,尤其是line15-25,迴圈實現的功能是什麼,逐行理解每條指令的功能。
1、使用masm、link對task1.asm進行彙編、連結,得到可執行檔案task1.exe,執行並觀察結果。
觀察到螢幕左下角打印出“NUIST”五個綠色字母
2、使用debug工具對程式進行除錯,執行到程式返回前,即line27之前,觀察結果。
先反彙編找到line27所在位置
再使用g命令
觀察到螢幕左下角打印出“NUIST”五個綠色字母
3、修改line4裡5個位元組單元的值,重新彙編、連結、執行,觀察結果。
db 5 dup(2) --> 改成: db 2,3,4,5,6
修改後執行結果:
觀察到字母的顏色發生了變化
聯想之前實驗二可知這裡的數字表示每個字元的顏色
實驗任務2
已知資料段data中定義位元組資料如下:
data segments
db 23, 50, 66, 71, 35
data ends
編寫程式,在螢幕上以十進位制整數形式列印輸出這5個兩位數。
1、編寫程式如下:
assume cs:code,ds:data data SEGMENT db 23,50,66,71,35 data ENDS code SEGMENT start: mov ax,data mov ds,ax mov si,0 mov cx,5 s: mov ax,0 mov al,[si] mov bl,10 div bl inc bx add ax,3030h mov dx,ax mov ah,2 int 21h mov dl,dh int 21h inc si loop s mov ah,4ch int 21h code ENDS END start
2、執行結果:
實驗任務3
教材「實驗5 編寫、除錯具有多個段的程式」(1)
assume cs:code, ds:data, ss:stack
data segment
dw 0123h, 0456h, 0789h, 0abch, 0defh, 0fedh, 0cbah, 0987h
data ends
stack segment
dw 0, 0, 0, 0, 0, 0, 0, 0
stack ends
code segment
start: mov ax,stack
mov ss, ax
mov sp,16
mov ax, data
mov ds, ax
push ds:[0]
push ds:[2]
pop ds:[2]
pop ds:[0]
mov ax,4c00h
int 21h
code ends
end start
1、CPU執行程式,程式返回前,data段中的資料不變
即data段中資料仍為:0123h, 0456h, 0789h, 0abch, 0defh, 0fedh, 0cbah, 0987h
程式執行前:
程式返回前:
觀察到data段資料不變
2、程式返回前,CS=076CH, SS=076BH, DS=076AH
3、設程式載入後,code段的段地址為X,則data段的段地址為X-2,stack段的段地址為X-1。
實驗任務4
教材「實驗5 編寫、除錯具有多個段的程式」(2)
assume cs:code, ds:data, ss:stack
code segment
start: mov ax,stack
mov ss, ax
mov sp,16
mov ax, data
mov ds, ax
push ds:[0]
push ds:[2]
pop ds:[2]
pop ds:[0]
mov ax,4c00h
int 21h
code ends
data segment
dw 0123h, 0456h
data ends
stack segment
dw 0,0
stack ends
end start
1、CPU執行程式,程式返回前,data中資料不變,即0123h, 0456h。
2、CS=076CH,SS=076BH,DS=076AH
反彙編
執行到程式返回前
3、設程式載入後,code段的段地址為X,則data段的段地址為X-2,stack段的段地址為X-1。
4、對於如下定義的段:
name segment
...
name ends
如果段中的資料佔N個位元組,則程式載入後,該段實際佔有的空間為⌈N/16⌉*16位元組。
實驗任務5
教材「實驗5 編寫、除錯具有多個段的程式」(3)
assume cs:code, ds:data, ss:stack
code segment
start: mov ax,stack
mov ss, ax
mov sp,16
mov ax, data
mov ds, ax
push ds:[0]
push ds:[2]
pop ds:[2]
pop ds:[0]
mov ax,4c00h
int 21h
code ends
data segment
dw 0123h, 0456h
data ends
stack segment
dw 0,0
stack ends
end start
1、CPU執行程式,程式返回前,data中資料不變,即0123h, 0456h。
2、CS=076AH,SS=076EH,DS=076DH
反彙編
執行到程式返回前
3、設程式載入後,code段的段地址為X,則data段的段地址為X+3,stack段的段地址為X+4。
實驗任務6
如果將1、2、3題中的最後一條偽指令“end start”改為“end”(也就是說,不指明程式的入口),則哪個程式仍然可以正確執行?請說明原因。
答:只有第3題的程式(task5)仍可以正確執行。
前兩題程式開始定義的資料段,而第3題定義的是程式碼段。若不指定程式入口,CPU將從第一個單元開始執行,故只有第3題能正確執行。
實驗任務7
教材「實驗5 編寫、除錯具有多個段的程式」(5)
assume cs:code
a segment
db 1,2,3,4,5,6,7,8
a ends
b segment
db 1,2,3,4,5,6,7,8
b ends
c segment
db 8 dup(0)
c ends
code segment
start:
mov ax,a
mov ds,ax
mov ax,c
mov es,ax
mov si,0
mov cx,8
s: mov al,[si]
add al,[si+10H]
mov es:[si],al
inc si
loop s
mov ah,4ch
int 21h
code ends
end start
除錯執行結果:
相加前
相加後
計算結果正確
實驗任務8
教材「實驗5 編寫、除錯具有多個段的程式」(6)
assume cs:code
a segment
dw 1,2,3,4,5,6,7,8,9,0ah,0bh,0ch,0dh,0eh,0fh,0ffh
a ends
b segment
dw 8 dup(0)
b ends
code segment
start:
mov ax,a
mov ds,ax
mov ax,b
mov ss,ax
mov sp,10H
mov bx,0
mov cx,8
s: push [bx]
add bx,2
loop s
mov ah,4ch
int 21h
code ends
end start
除錯執行結果:
成功逆序儲存至b段