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

实验任务1

源代码:

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

  

① add指令对标志寄存器中的零标志位ZF(Zero Flag)、进位标志位CF(Carry Flag)是否有影响?

 

如图所示,add指令对标志寄存器中的零标志位ZF(Zero Flag)有影响,对进位标志位CF(Carry Flag)也有影响。

② inc指令对标志寄存器中的零标志位ZF(Zero Flag)、进位标志位CF(Carry Flag)是否有影响?

 

如图所示,inc指令对标志寄存器中的零标志位ZF(Zero Flag)有影响,对进位标志位CF(Carry Flag)无影响。

 

① line31~line34的4条inc指令,能否替换成如下代码?你的结论的依据/理由是什么?

答:不能,因为使用add会对进位标志位CF产生影响,从而会影响到adc指令的执行,即标号x和y处数据的相加,而inc指令不会对CF产生影响。

 

② 在debug中调试,观察数据段中做128位加之前,和,加之后,数据段的值的变化。

加之前:

 

加之后:

 

 

实验任务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:
mov ah, 1
int 21h
mov [si], al
cmp al, '#'
je next
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

  

运行程序,从键盘上输入一串字符,以#结束(比如,输入George Orwell, 1984#),观察结果。结合运 行结果,理解代码并回答问题:

 

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

答:

Line12:从键盘上输入单个字符:。

Line13:将输入字符的ASCII码存放在al寄存器中:。

Line14:将al中的数据放入str中。

Line15,16:将输入的字符与‘#’进行比较,若相同,则跳转到next处。

Line17,18:si加一,指向下一个输入的字符,直到遇到终结符#,停止循环。

 

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

答:

打印一个换行符。

 

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

答:

Line24:将输入字符串的长度赋给cx,即循环次数。

Line25:将输出指针指向字符串的第一位。

Line26-28:打印当前字符。

Line29-30:输出指针指向当前字符的下一位,进行循环,直到所有的字符都被打印完为止。

 

实验任务3

源代码:

assume cs:code,ds:data,ss:stack

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

stack segment
    dw 8 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 cx,len/2
s0:
mov ax,[si]
mov dx,0
push cx
call printNumber
call printSpace
pop cx
inc si
inc si
loop s0

mov ah,4ch
int 21h

printNumber:
mov cx,0

flag1:
mov bx,10
mov dx,0
div bx
push dx
inc cx
cmp ax,0
jne flag1

s1:
pop dx
mov ah,2
or dl,30h
int 21h
loop s1
ret

printSpace:
mov ah,2
mov dl,' '
int 21h
ret

code ends
end start

  

运行测试截图:

 

实验任务4

源代码:

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,0
mov cx,len
call strupr

mov ax,4c00h
int 21h

strupr:
s0:
mov al,[si]
cmp al,97
jb s1
cmp al,122
ja s1
and al,11011111B
mov [si],al
s1:
inc si
loop s0
ret 

code ends
end start

  

在debug中调试截图( call strupr 调用之前,数据段的值,以及,调用之后,数据段的值):

 

实验任务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’,如果是,则在屏幕第24行第70列输出‘yes’;如果不是,则在屏幕第24行第70列输出‘no’。

 

实验任务6

实验结果:

问:通过此项实现任务,你对中断、软中断实现机制的理解:

答:当CPU收到中断信息后,需要对中断信息进行处理,中断分为两种,硬中断(由硬件产生)和软中断(由软件产生),我们可以编写中断处理程序来对不同的中断信息进行处理,当CUP接收到中断信息后,可以根据中断向量表来找到相应的中断处理程序的入口地址,而处理完毕后,即可通过iret指令返回。

 

实验总结

  通过本次实验,我对第11章标志寄存器,第12章内中断部分的知识有了更加深刻的理解,在做实验的过程中,源代码中也用到了很多以前章节所学到的很多知识,代码也更加灵活,写的时候选择性也很高,怎样以最简单的方式写出程序,我想我在这一次的实验中学到了不少这部分的知识,本次实验也让我对以前知识的灵活运用有了更加深刻的理解,让我受益良多。

 

posted @ 2021-12-11 14:42  城风欧欧  阅读(78)  评论(2)    收藏  举报