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

1. 实验任务1

使用任何一款文本编辑器,录入8086汇编程序源码task1.asm。
task1.asm

assume cs:code, ds:data

data segment
    x db 1, 9, 3
    len1 equ $ - x

    y dw 1, 9, 3
    len2 equ $ - y
data ends

code segment
start:
    mov ax, data
    mov ds, ax

    mov si, offset x
    mov cx, len1
    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
    mov cx, len2/2
    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

对源程序进行汇编、链接,得到可执行程序task1.exe,运行后,结合运行结果和注释,及必要的debug
调试:

1. 理解运算符offset、伪指令equ、预定义符号$的灵活使用。
通过line5、line8,以及数据项的数据属性(字节、字、双字,等),可以方便计算出连续数据项
的个数,而无需人工计数。
注*: 符号常量len1, len2不占用数据段内存空间

equ代表给符号名一个地址值,$代表当前的地址值,len1 equ $ -x 就是把 len1定义为当前地址到x的长度值。

offset是取其后变量或标号的偏移地址。

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

 

 

 反汇编可以看到该指令对应的机器码为E2F2,LOOP的机器码为E2。

将F2转化为二进制为11110010,为补码形式,转化为原码结果=10001110,对应十进制数为-14。

LOOP S1的偏移地址为0019,转化成十进制为25,执行该指令的地址变化情况如下 25+2-14=13,转化为十六进制为000D,对应s1的偏移地址。

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

 

 

 解题过程同上,F0原码为10010000,对应十进制数为-16,地址变化情况为55+2-16=41,转化为十六进制为0029,对应s2偏移地址。

③ 附上上述分析时,在debug中进行调试观察的反汇编截图

2. 实验任务2

使用任何一款文本编辑器,录入8086汇编程序源码task2.asm。
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) = ?

ds:[0]内存为offset s1,执行 call word ptr ds:[0] ,第一步,push ip,ip值入栈;pop ax,将offset s1的值赋给ax;

同理的,将offset s2的值赋给bx,cs的值赋给cx。

因此(ax)=offset s1; (bx)=offset s2;(cx)=cs;

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

 

 

 

 3. 实验任务3

针对8086CPU,已知逻辑段定义如下:

data segment
    x db 99, 72, 85, 63, 89, 97, 55
    len equ $- x
data ends

编写8086汇编源程序task3.asm,在屏幕上以十进制形式输出data段中这一组连续的数据,数据和数据
之间以空格间隔。

要求:

编写子程序printNumber
功能:以十进制形式输出一个两位数
入口参数:寄存器ax(待输出的数据 --> ax)
出口参数:无

编写子程序printSpace

功能:打印一个空格
入口参数:无
出口参数:无
在主体代码中,综合应用寻址方式和循环,调用printNumber和printSpace, 实现题目要求。

assume ds:data, cs:code

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 cx,len;cx中存储定义的字的个数,这也是循环会执行的次数
    s:  call printNumber;打印一个两位数
    call printSpace;打印一个空格
    inc si
    loop s

    mov ax,4c00h
    int 21h

    printNumber:
    mov ah, 0
    mov al, [si]
        mov bl, 10
    div bl
    mov bl,al
    mov bh,ah

    mov ah,2
    mov dl,bl
    add dl,30h
    int 21h

    mov ah,2
    mov dl,bh
    add dl,30h
    int 21h
    ret


    printSpace:
    mov ah,2
    mov dl,' '
    int 21h
    ret
code ends
end start

运行结果如下所示:

 

4. 实验任务4

针对8086CPU,已知逻辑段定义如下:

data segment
str db 'try'
len equ $ - str
data ends

编写8086汇编源程序task4.asm,在屏幕上以指定颜色、指定行,在屏幕上输出字符串。
要求:
  编写子程序printStr
    功能:在指定行、以指定颜色,在屏幕上显示字符串
    入口参数
      字符串首字符地址 --> ds:si(其中,字符串所在段的段地址—> ds, 字符串起始地址的偏
      移地址—> si)
      字符串长度 --> cx
      字符串颜色 --> bl
      指定行 --> bh (取值:0 ~24)
    出口参数:无

  在主体代码中,两次调用printStr,使得在屏幕最上方以黑底绿字显示字符串,在屏幕最下方以黑
  底红色显示字符串

assume ds:data, cs:code

data segment
str db 'try'
len equ $ - str
data ends

code segment
start:
    mov ax,data
    mov ds,ax

    mov si,0
    mov cx,len;字符串长度为len赋值给cx
    mov bh,0
    mov bl,00000010B;对应绿色
    call printStr

    mov si,0
    mov cx,len;字符串长度为len赋值给cx
    mov bh,24
    mov bl,00000100B;对应红色
    call printStr

    mov ax,4c00h
    int 21h


    printStr:
    mov ax,0b800h;设置ax为显存段地址
    mov es,ax

    mov al,bh
    mov bh,0a0h
    mul bh
    mov bp,ax
    s:
    mov ah,ds:[si];每次取出一个字节给ah
    mov es:[bp],ah;把字母存给显存地址低八位中
    inc bp
    mov es:[bp],bl;将颜色信息存入高八位
    inc bp
    inc si
    loop s
    ret

code ends
end start

编译后运行效果如下所示:

 

5. 实验任务5

针对8086CPU,针对8086CPU,已知逻辑段定义如下:

data segment
  stu_no db '20498329042'
  len = $ - stu_no
data end

在80×25彩色字符模式下,在屏幕最后一行正中间显示学号。要求输出窗口蓝底,学号和两侧折线,以
白色前景色显示。

assume ds:data, cs:code

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

code segment
start:
    mov ax,0b800h
    mov es,ax
    mov bp,1
    mov cx,8000h

    first:
    mov byte ptr es:[bp],00011111B
    add bp,2
    loop first

    mov ax,data
    mov ds,ax;ds作数据段
    mov ax,0050h;一行的长度
    sub ax,len;减去学号的长度
    mov bh,02h;除以二得到横线的长度
    div bh
    mov bl,al;商赋给bx,后面要用
    mov bh,0

    mov ax,0b800h;显存地址存给es
    mov es,ax
    mov bp,0F00h;首字母偏移量赋给bp
    mov cx,bx

    s:
    mov byte ptr es:[bp],'-'
    inc bp
    inc bp
    loop s;打印横杠

    mov cx,len
    mov di,0

    s1:
    mov al,ds:[di]
    mov es:[bp],al
    inc bp
    inc bp
    inc di
    loop s1;打印学号

    mov cx,bx

    s2:
    mov byte ptr es:[bp],'-'
    inc bp
    inc bp
    loop  s2;打印横杠

    mov ax,4c00h
    int 21h

code ends
end start

编译后显示效果如下所示:

 

 实验总结

1、通过本次实验,主要掌握了对跳转指令的理解和应用,加深了对显存数据的掌握。

2、了解了Loop指令的机器码的含义以及如何计算跳转地址。

posted @ 2021-11-26 11:42  -千-  阅读(29)  评论(3)    收藏  举报