实验4 8086标志寄存器及中断

实验任务1

验证性实验:有些汇编指令会影响到标志寄存器中的一个或多个状态标志位。 在debug环境中,分别实践、观察: ① add指令对标志寄存器中的零标志位ZF(Zero Flag)、进位标志位CF(Carry Flag)是否有影响? ② inc指令对标志寄存器中的零标志位ZF(Zero Flag)、进位标志位CF(Carry Flag)是否有影响?

 

 

add 零标志位ZF由0变1

add 进位标志位CF由0变1

 inc零标志位ZF由0变1

inc 进位标志位CF保持0不变

assume cs:code, ds:data
data segment
x dw 1020h, 2240h, 9522h, 5060h, 3359h, 6652h, 2530h, 7031h
y dw 3210h, 5510h, 6066h, 5121h, 8801h, 6210h, 7119h, 3912h
data ends
code segment
start:
mov ax, data
mov ds, ax
mov si, offset x
mov di, offset y
call add128
mov ah, 4ch
int 21h
add128:
push ax
push cx
push si
push di
sub ax, ax    ;进位清零
mov cx, 8
s: mov ax, [si]
adc ax, [di]
mov [si], ax
inc si
inc si
inc di
inc di
loop s
pop di
pop si
pop cx
pop ax
ret
code ends
end start

 

assume cs:code, ds:data
data segment
str db 80 dup(?)
data ends
code segment
start:
mov ax, data
mov ds, ax
mov si, 0
s1:
mov ah, 1
int 21h
mov [si], al
cmp al, '#'   ;相同zf=1,不同zf=0
je next    ;zf=1就跳转
inc si
jmp s1
next:
mov ah, 2
mov dl, 0ah
int 21h
mov cx, si
mov si, 0
s2: mov ah, 2
mov dl, [si]
int 21h
inc si
loop s2
mov ah, 4ch
int 21h
code ends
end start

 替换成代码后

 虽然答案不变,但是最好不要,因为add可能会改变进位,从而影响程序答案

执行前:

 执行后:

实验任务2

assume cs:code, ds:data
data segment
str db 80 dup(?)
data ends
code segment
start:
mov ax, data
mov ds, ax
mov si, 0
s1:            ;11行
mov ah, 1
int 21h
mov [si], al
cmp al, '#'
je next
inc si
jmp s1         ;18行
next:
mov ah, 2      ;20行
mov dl, 0ah    ;换行符
int 21h        ;22行 
mov cx, si     ;24行
mov si, 0
s2: mov ah, 2
mov dl, [si]
int 21h
inc si
loop s2         ;30行
mov ah, 4ch
int 21h
code ends
end start

① 汇编指令代码line11-18,实现的功能是?

存入输入的字符,直到#停止

② 汇编指令代码line20-22,实现的功能是?

输出换行符并结束程序

③ 汇编指令代码line24-30,实现的功能是?

输出刚刚输入的字符

实验任务3

data segment
x dw 91, 792, 8536, 65521, 2021
len equ $ - x
data ends

编写8086汇编源程序task3.asm,在屏幕上以十进制形式输出data段中这一组连续的数据,数据和数据 之间以空格间隔。 要求: 编写子程序printNumber 功能:以十进制形式输出一个任意位数的整数(整数范围0 ~ 65535) 入口参数:寄存器ax(待输出的数据 --> ax) 出口参数:无 编写子程序printSpace 功能:打印一个空格 入口参数:无 出口参数:无 在主体代码中,综合应用寻址方式和循环,调用printNumber和printSpace, 实现题目要求。

 

 

assume cs:code,ds:data,ss:stack
data segment
x dw 91, 792, 8536, 65521, 2021
len equ $ - x
data ends
stack segment
db 16 dup(0)
stack ends
code segment
start:
mov ax,data
mov ds,ax
mov ax,stack
mov ss,ax
mov sp,16
mov si,0
mov ax,ds:[si]
call printNumber
call printSpace
inc si
inc si
mov ax,ds:[si]
call printNumber
call printSpace
inc si
inc si
mov ax,ds:[si]
call printNumber
call printSpace
inc si
inc si
mov ax,ds:[si]
call printNumber
call printSpace
inc si
inc si
mov ax,ds:[si]
call printNumber
call printSpace
 mov ax,4c00h
 int 21h

printNumber:
mov bx,10    ;除数10
mov bp,0  ;计数
mov cx,4   ;最多循环4次
s1:
cmp ax, 0
je ppp
div bx         ;被除数DXAX 除10  AX保存商,DX保存余数
or dx, 30h  ;数字转字符
push dx   ;余数放在栈里
inc bp
mov dx,0
loop s1
ppp:
mov cx,bp  ;循环次数
s2:
pop dx
mov ah,2
int 21ht
loop s2
ret

printSpace:
mov dl,' '
mov ah,02;
int 21h
ret

code ends
end start

实验任务4

编写8086汇编源程序task4.asm,将data段中字符串里的小写字符转换成大写。 要求: 编写子程序strupr 功能:将包含任意字符的字符串中的小写字母变成大写 入口参数 (ds:si ) 字符串首地址的段地址和偏移地址分别送至ds和si (cx) 字符串的长度 出口参数:无 在主体代码中,设置入口参数,调用strupr, 实现题目要求。

data segment
str db "assembly language, it's not difficult but tedious"
len equ $ - str
data ends

assume cs:code,ds:data
data segment
str db "assembly language, it's not difficult but tedious"
len equ $ - str
data ends
code segment
start:
mov ax,data
mov ds,ax
mov si,offset str
call strupr
mov ax, 4c00h    
int 21h
strupr:
mov cx,len
s:
mov al,ds:[si]
cmp al,20h    ;空格
je aa
cmp al,27h    ;单引号
je bb
cmp al,2ch    ;逗号
je cc
and al,11011111b
mov ds:[si],al
aa:
bb:
cc:
inc si
loop s
ret

code ends
end start

实验任务5

assume cs:code, ds:data
data segment
str1 db "yes", '$'
str2 db "no", '$'
data ends
code segment
start:
mov ax, data
mov ds, ax
mov ah, 1
int 21h ; 从键盘输入字符
mov ah, 2
mov bh, 0
mov dh, 24 ; 设置光标位置在第24行
mov dl, 70 ; 设置光标位置在第70列
int 10h ; 设置光标位置
cmp al, '7'
je s1
mov ah, 9
mov dx, offset str2
int 21h ; 显示标号str2处的字符串
jmp over
s1: mov ah, 9
mov dx, offset str1
int 21h ; 显示标号str2处的字符串
over:
mov ah, 4ch
int 21h
code ends
end start

运行程序,输入7,观察结果。输入其他字符,观察结果。结合运行结果和注释,理解代码实现的功能。

输入7则输出yes,输入其他数字则输出no

DOS系统功能调用int 21h的1号子功能

功能:从键盘上输入单个字符

入口参数:(ah) = 1

出口参数: (al)存放输入字符的ASCⅡ码

mov ah, 1
int 21h ; (al) <-- 输入字符的ascⅡ码

DOS系统功能调用int 21h的9号子功能

功能:显示字符串

入口参数:(ah) = 9,(ds:dx) = 字符串的首地址的段地址和偏移地址

出口参数: 无

其它要求:字符串必须以$结束

mov ah, 9
mov ds, ×× ; ××是待输出字符串所在段的段地址
mov dx, ×× ; ××是待输出字符串第一个字符的偏移地址
int 21h

BIOS中断例程int 10h的2号子功能

功能:设置光标位置

入口参数:(ah) = 2, (bh) = 页号(默认取0), (dh) = 行号, (dl) = 列号

出口参数:无

mov ah, 2
mov bh, ×× ; ××是页号
mov dh, ××
mov dl, ×× ; ××是列号
int 10h

实验任务6

实验任务1、2、3、5中使用了不少系统提供的中断例程。本实验任务中,要求自行实现一个42号软中断 例程,使得通过 int 42 或 int 2ah 软中断调用,实现在屏幕最下方中间以黑底绿字打印"welcome to 2049!"。

assume cs:code
code segment
start:
; 42 interrupt routine install code
mov ax, cs
mov ds, ax
mov si, offset int42 ; set ds:si
mov ax, 0
mov es, ax
mov di, 200h ; set es:di
mov cx, offset int42_end - offset int42
cld
rep movsb
; set IVT(Interrupt Vector Table)
mov ax, 0
mov es, ax
mov word ptr es:[42*4], 200h
mov word ptr es:[42*4+2], 0
mov ah, 4ch
int 21h
int42:
jmp short int42_start
str db "welcome to 2049!"
len equ $ - str
; display string "welcome to 2049!"
int42_start:
mov ax, cs
mov ds, ax
mov si, 202h
mov ax, 0b800h
mov es, ax
mov di, 24*160 + 32*2
mov cx, len
s: mov al, [si]
mov es:[di], al
mov byte ptr es:[di+1], 2
inc si
add di, 2
loop s
iret
int42_end:
nop
code ends
end start
ssume cs:code
code segment
start:
int 42 ; 调用自己实现的42号软中断
mov ah, 4ch
int 21h
code ends
end start

断标志码为42的中断程序

中断:CPU会中断当前正在运行的任务,转而去执行中断程序

软中断:CPU内部产生中断信号,不会直接中断CPU

posted @ 2021-12-10 14:16  木回  阅读(144)  评论(2编辑  收藏  举报