实验3 转移指令跳转原理及其简单应用编程

实验任务1

给出程序task1.asm源码

assume cs:code, ds:data
data segment
x db 1, 9, 3
len1 equ $ - x ; 符号常量, $指下一个数据项的偏移地址,这个示例中,是3
y dw 1, 9, 3
len2 equ $ - y ; 符号常量, $指下一个数据项的偏移地址,这个示例中,是9
data ends
code segment
start:
mov ax, data
mov ds, ax
mov si, offset x ; 取符号x对应的偏移地址0 -> si
mov cx, len1 ; 从符号x开始的连续字节数据项个数 -> cx
mov ah, 2
s1:mov dl, [si]
or dl, 30h
int 21h
mov dl, ' '
int 21h ; 输出空格
inc si
loop s1
mov ah, 2
mov dl, 0ah
int 21h ; 换行
mov si, offset y ; 取符号y对应的偏移地址3 -> si
mov cx, len2/2 ; 从符号y开始的连续字数据项个数 -> cx
mov ah, 2
s2:mov dx, [si]
or dl, 30h
int 21h
mov dl, ' '
int 21h ; 输出空格
add si, 2
loop s2
mov ah, 4ch
int 21h
code ends
end start

 

运行截图:

① line27, 汇编指令 loop s1 跳转时,是根据位移量跳转的。通过debug反汇编,查看其机器码,分析其跳转的位移量是多少?(位移量数值以十进制数值回答)从CPU的角度,说明 是如何计算得到跳转后标号s1其后指令的偏移地址的。

位移量是:14(Loop指令结束地址为:001B,s1指令开始地址为000D,001B-000D=14)

分析:or dl, 30h命令占三个字节,inc指令占一个字节,其他指令各占两个字节,加起来一共14字节。

 

 ② line44,汇编指令 loop s2 跳转时,是根据位移量跳转的。通过debug反汇编,查看其机 器码,分析其跳转的位移量是多少?(位移量数值以十进制数值回答)从CPU的角度,说明 是如何计算得到跳转后标号s2其后指令的偏移地址的。

位移量是:16(Loop指令结束地址为:0039,s1指令开始地址为0029,001B-000D=16)

分析:or dl, 30h命令占三个字节,其他指令各占两个字节,加起来一共16字节。

 

实验任务2

task2.asm源码:

assume cs:code, ds:data
data segment
    dw 200h, 0h, 230h, 0h
data ends
stack segment
    db 16 dup(0)
stack ends
code segment
start:
    mov ax, data
    mov ds, ax
    mov word ptr ds:[0], offset s1
    mov word ptr ds:[2], offset s2
    mov ds:[4], cs
    mov ax, stack
    mov ss, ax
    mov sp, 16
    call word ptr ds:[0]
s1: pop ax
    call dword ptr ds:[2]
s2: pop bx
    pop cx
    mov ah, 4ch
    int 21h
code ends
end start

 

① 根据call指令的跳转原理,先从理论上分析,程序执行到退出(line31)之前,寄存器(ax) = ? 寄存器 (bx) = ? 寄存器(cx) = ?

  答:理论上ax中的数据应该是s1:pop ax这条指令的 IP。因为 call word ptr ds:[0] 将 s1:pop ax指令的IP压入了栈中。

    bx中应该是s2:pop bx这条指令的IP,cx中应该是s2:pop bx这条指令的CS。

    因为 call dword ptr ds:[2] 指令将 s2:pop bx的CS,IP先后压入栈中。

 

② 对源程序进行汇编、链接,得到可执行程序task2.exe。使用debug调试,观察、验证调试结果与理论 分析结果是否一致。

 

 与猜想结果一致。

 

实验任务三

给出程序源码task3.asm

assume cs:code, ds:data
data segment
    x db 99, 72, 85, 63, 89, 97, 55
    len equ $- x
data ends

code segment
start: 
    mov ax, data
    mov ds, ax
    mov si, 0
    mov bl, 10
    mov cx, len

s:  mov ah, 0
    mov al, [si]
    call printNumber
    call printSpace
    inc si
    loop s

mov ah, 4ch
int 21h


printNumber:
    div bl
    mov dl, al     ;al为商,存在dl中
    mov bh, ah   ;ah为余数,存在bh中

    or dl,30H  ;输出十位dl
    mov ah,2
    int 21h

    mov dl, bh  ;输出个位bh
    or dl,30H
    mov ah,2
    int 21h

    ret

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

code ends
end start

运行测试截图:

 

 实验任务四:

 代码:

assume cs:code, ds:data

data segment
    str db 'try'
    len equ $ - str
data ends
stack segment
    db 16 dup(0)
stack ends

code segment
start:
    mov ax,stack
    mov ss,ax
    mov sp,16
    mov ax, data
    mov ds, ax
    mov ax, 0b800h      ;显存的内存地址为b8000h~bFFFFh
    mov es, ax
    mov di, 0
    mov cx, len
    mov si, offset str
s1:
    mov bl, 2      ;设定颜色为绿色
    mov bh, 0     ;设置行
    call printStr
    loop s1

    mov cx, len
    mov si, offset str
    mov bh, 24     ;设置行
    mov al, 0a0h       ;每行的字节数为160(8每行0列)
    mov ah, 0
    mul bh
    mov di, ax       ;设置位置
s2:
    mov bl, 4      ;设定颜色为红色
    call printStr
    loop s2

    mov ah, 4ch
    int 21h

printStr:
    mov ah, bl        ;ah存颜色
    mov al, [si]        ;al存显示的内容
    mov es:[di], ax  ;es:[di]是指定输出位置
    add di, 2
    inc si
    ret

code ends
end start

运行截图:

 

 

实验任务5

代码:

assume cs:code, ds:data

data segment
stu_no db '201983290435'
len = $ - stu_no
data ends

stack segment
    dw 0,0,0,0,0,0,0,0,0      ;定义一个段,用来做栈段,容量为16个字节   
stack ends

code segment
start:
mov ax,stack
mov ss,ax
mov sp,16

mov ax, data
mov ds, ax
mov ax, 0b800h
mov es, ax

;第一层循环
    mov di, 0
    mov cx, 24
s1:
    push cx   ;将外层循环的cx压栈
    mov cx, 80  ;将cx设置成内层循环的次数
s2:
    mov ax, 1720h
    mov es:[di], ax
    add di, 2
loop s2
    mov di, 0
    pop cx ;从栈顶将cx的值进行恢复
    mov ax, es
    add ax, 0ah
    mov es, ax
loop s1

;第二层循环
    mov di, 0
    mov cx, 30
s3:
    mov ax, 172dh
    mov es:[di], ax
    add di, 2
loop s3

;第三层循环
    mov di, 003ch       ;设置学号起始位置
    mov si, offset stu_no
    mov cx, len
s4:
    mov ah, 017h        ;ah存颜色
    mov al, [si]        ;al存显示的内容
    mov es:[di], ax  ;es:[di]是指定输出位置
    inc si
    add di, 2
loop s4

;第四层循环
    mov cx, 38
s5:
    mov ax, 172dh
    mov es:[di], ax
    add di, 2
loop s5

    mov ah, 4ch
    int 21h

code ends
end start

 

截图:

 

 

实验总结:

or dl,30H
mov ah,2
int 21h

的作用是将dl寄存器中的数转化为相应的ASCLL码

 

posted @ 2021-11-29 09:00  南斋  阅读(60)  评论(3编辑  收藏  举报