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

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

 

test1

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
line31~line34的4条inc指令,能否替换成如下代码?你的结论的依据/理由是什么?
add si, 2
add di, 2

不可以,在执行128位数运算时会用到进位寄存器,在执行add过程中可能对其产生修改,所以只能使用inc或者用栈保存标志寄存器的记录。

 

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

 

 

 test2

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,实现的功能是?
在接收到#号之前,持续从键盘中输入字符,并存入到ds:si中,直到出现#,跳转到next代码段
 
② 汇编指令代码line20-22,实现的功能是?
输出换行符,换行符的十六进制ASCII码值就是0AH
 
③ 汇编指令代码line24-30,实现的功能是?
实现的功能是将数据段ds中长度为si的字符串输出。

 

 

test3

assume cs:code,ds:data
data segment
    x dw 91, 792, 8536, 65521, 2021
    len equ $ - x
data ends
stack segment
    db 64 dup(0)
    top equ $+1
stack ends
code segment
start:
    mov ax,data
    mov ds,ax
    mov ax,stack
    mov ss,ax
    mov sp,top
    ;一次处理两个字节
    mov cx,len/2
    mov bx,0
s1:  
    mov ax,ds:[bx]
    push bx
    push cx
    call printNumber
    call printSpace
    pop cx
    pop bx
    add bx,2
    loop s1

    mov ah,4ch
    int 21h

printNumber:
    ;不断除以10,得到对应位的数字
    ;ax保存被除数
    mov bx,10
    mov cx,0
s2:
    mov dx,0
    div bx
    push dx
    inc cx
    cmp ax,0
    jne s2

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

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

code ends
end start 

 

 

test4

编写8086汇编源程序task4.asm,将data段中字符串里的小写字符转换成大写。

1 data segment 
2     str db "assembly language, it's not difficult but tedious" 
3     len equ $ - str 
4 data ends
  编写子程序strupr
    功能:将包含任意字符的字符串中的小写字母变成大写
    入口参数
      (ds:si ) 字符串首地址的段地址和偏移地址分别送至ds和si
      (cx) 字符串的长度
    出口参数:无
  在主体代码中,设置入口参数,调用strupr, 实现题目要求。
assume cs:code,ds:data
data segment
    str db "assembly language, it's not difficult but tedious"
    len = ($ - str)
data ends
stack segment
    db 128 dup(0)
    top equ $ + 1
stack ends
code segment
start:
    mov ax,data
    mov ds,ax
    mov ax,stack
    mov ss,ax
    mov sp,top
    mov si,0
    mov cx,len
    call strupr
    mov ah,4ch
    int 21h
strupr:
;大小写转换的方式为ascii码值减去32
s:
;判断是否为小写字母,不是的话直接跳转到s1
    cmp byte ptr ds:[si],61h
    jb s1
    cmp byte ptr ds:[si],7ah
    ja s1
    mov dl,ds:[si]
;小写转换为大写
    and dl,11011111B
    mov ds:[si],dl
s1: 
    inc si
    loop s
    ret
code ends
end start

 

 

test5

 

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
    mov dl, 70
    int 10h

    cmp al, '7'
    je s1
    mov ah, 9
    mov dx, offset str2
    int 21h

    jmp over

s1: mov ah, 9
    mov dx, offset str1
    int 21h
over:  
    mov ah, 4ch
    int 21h
code ends
end start

 

 

 task5的功能为判断输入字符是否为7,是则输出yes,不是则输出no

使用jmp与je判断语句实现if分支语句

 

 

test6

通过此项实现任务,你对中断、软中断实现机制的理解
 
设置中断例程
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
;将ds:si的内容搬到es:di
    cld
    rep movsb

    ; set IVT(Interrupt Vector Table)
    mov ax, 0
    mov es, ax
;设置中断表中的高位字为中断处理程序段地址,低字为中断处理程序偏移地址
;+2即加两个字节,即处理低位字
;*4即为4个字节->两个字符->中断表每项长2字符
    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

调用中断

assume cs:code

code segment
start:
    int 42

    mov ah, 4ch
    int 21h
code ends
end start

 

 

中断是指计算机运行过程中,出现某些意外情况需主机干预时,机器能自动停止正在运行的程序并转入处理新情况的程序,处理完毕后又返回原被暂停的程序继续运行。

软中断是一条CPU指令,用以自陷一个中断。发生中断时,中断源发出中断信号,CPU保存当前的执行环境,将寄存器内容压入栈中,查询中断表项,

得到中断例程的CS:IP,转而去处理中断。处理完成后取出栈中寄存器内容,恢复先前执行环境,继续完成未完成的任务。

 

 

posted @ 2021-12-10 17:05  mayoi  阅读(43)  评论(1编辑  收藏  举报