实验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