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

一、实验目的

1. 理解和掌握转移指令的跳转原理

2. 掌握使用call和ret指令实现子程序的方法,理解和掌握其参数传递方式

3. 理解和掌握80×25彩色字符模式显示原理

4. 综合应用寻址方式和汇编指令完成简单应用编程

 

二、实验准备

复习教材9-10章:

转移指令的跳转原理

汇编指令jmp, loop, jcxz, call, ret, retf的用法

 

三、实验内容

1. 实验任务1

 1 assume cs:code, ds:data
 2 data segment
 3 x db 1, 9, 3
 4 len1 equ $ - x ; 符号常量, $指下一个数据项的偏移地址,这个示例中,是3
 5 y dw 1, 9, 3
 6 len2 equ $ - y ; 符号常量, $指下一个数据项的偏移地址,这个示例中,是9
 7 data ends
 8 code segment
 9 start:
10 mov ax, data
11 mov ds, ax
12 mov si, offset x ; 取符号x对应的偏移地址0 -> si
13 mov cx, len1 ; 从符号x开始的连续字节数据项个数 3-> cx
14 mov ah, 2
15 s1:mov dl, [si]
16 or dl, 30h
17 int 21h
18 mov dl, ' '
19 int 21h ; 输出空格
20 inc si
21 loop s1
22 mov ah, 2
23 mov dl, 0ah
24 int 21h ; 换行
25 mov si, offset y ; 取符号y对应的偏移地址3 -> si
26 mov cx, len2/2 ; 从符号y开始的连续字数据项个数 -> cx
27 mov ah, 2
28 s2:mov dx, [si]
29 or dl, 30h
30 int 21h
31 mov dl, ' '
32 int 21h ; 输出空格
33 add si, 2
34 loop s2
35 mov ah, 4ch
36 int 21h
37 code ends
38 end start

 

 

 

 

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

 从001B跳到000D,位移量是-14(十进制)

 F2是-14的补码

 loop指令为循环指令,所有的循环指令都是短转移,对IP的修改范围为-128~127(8位),在对应的机器码中包含转移的位移,而不是目的地址。则转移的8位位移=标号处的地址-loop指令后的第一个字节的地址,即001B-000D,这8位位移由编译程序在编译时算出。

 

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

 从0039跳到0029,位移量是-16(十进制)

    F0是-16的补码

loop指令的机器码格式为E2 偏移量(8位、补码表示)

 loop指令为循环指令,所有的循环指令都是短转移,对IP的修改范围为-128~127(8位),在对应的机器码中包含转移的位移,而不是目的地址。则转移的8位位移=标号处的地址-loop指令后的第一个字节的地址,即0039-0029,这8位位移由编译程序在编译时算出,偏移量用补码表示。

 

2. 实验任务2

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) = ?

      执行call word ptr ds:[0],ds:[0]为标号s1的地址:0021,将IP:0021压入栈中。执行pop ax后,将0021取出赋给ax。

  执行call dword ptr ds:[2],ds:[2]为标号s2的地址:0026,将CS:076C,IP:0026压入栈中,执行pop bx,将0026取出赋给bx,执行pop cx,将076C取出赋给cx

  因此从理论上分析ax=0021,bx=0026,cx=076C

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

见上

 

 

 

 3. 实验任务3

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 cx,7
    mov si,0

s:    mov ah,0
    mov al,ds:[si]
    mov bl,10
    div bl
    mov ds:[len+si],al            ;将数字的两位分开存放
    mov ds:[len+1+si],ah

    mov ah,2             ;int 21h的2号子功能
    mov dl,ds:[len+si]
    add dl,30h                 ;将数字由十进制转为ASCII码
    int 21h                    ;打印

    mov ah,2
    mov dl,ds:[len+1+si]
    add dl,30h
    int 21h

    mov ah,2
    mov dl," "
    int 21h

    inc si
    loop s

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

测试结果:

 

 

4. 实验任务4

编写8086汇编源程序task4.asm,在屏幕上以指定颜色、指定行,在屏幕上输出字符串。 要求:

   编写子程序printStr

   功能:在指定行、以指定颜色,在屏幕上显示字符串

   入口参数

    字符串首字符地址 --> ds:si(其中,字符串所在段的段地址—> ds, 字符串起始地址的偏 移地址—> si)

    字符串长度 --> cx

    字符串颜色 --> bl

    指定行 --> bh (取值:0 ~24)

  出口参数:无 ;

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

未采用子程序:

assume cs:code,ds:data
data segment
    str db 'try'
    len equ $ - str
data ends

code segment
start:
        mov ax,data
    mov ds,ax
    mov ax,0b800h
    mov es,ax
    mov cx,len
    mov si,0
    mov bx,0000h    ;第一行

s1:        mov ah, 02h
    mov al,ds:[si]
    mov es:[bx],ax
    add bx,2
    inc si
    loop s1    

    mov bx,0F00h    ;最后一行
    mov cx,len
    mov si,0

s2:     mov al,ds:[si]
    mov ah,04h
    mov es:[bx],ax
    add bx,2
    inc si
    loop s2 
 
      mov ax,4c00h
       int 21h
code ends
end

 

 

 

 

 

 使用子程序

assume cs:code,ds:data
data segment
    str db 'try'
    len equ $ - str
data ends

code segment
start:
        mov ax,data
    mov ds,ax
    mov ax,0b800h
    mov es,ax

    mov si, offset PrintStr
                mov bx,0000h    ;第一行
    mov ah, 02h
    call si

    mov si, offset PrintStr
     mov bx,0F00h    ;最后一行
    mov ah,04h
    call si

    mov ax,4c00h
       int 21h
    
PrintStr:
    mov cx,len
    mov si,0

s:    mov al,ds:[si]
    mov es:[bx],ax
    add bx,2
    inc si
    loop s
    ret
code ends
end start

 

 

 

 

 

 

 

5.实验任务5

针对8086CPU,针对8086CPU,已知逻辑段定义如下: 在80×25彩色字符模式下,在屏幕最后一行正中间显示学号。要求输出窗口蓝底,学号和两侧折线,以 白色前景色显示。

注*:

1. 80×25彩色字符模式显示缓冲区结构,参见教材「实验9 根据材料编程」里的说明。

2. 编写程序实现时,将data段的学号换成自己的学号。

assume cs:code,ds:data
data segment
    str db '201983290287'
    len equ $ - str;12个字符
data ends

code segment
start:
        mov ax,data
    mov ds,ax
    mov ax,0b800h
    mov es,ax
    mov cx,0F9Fh
    mov bx,0

s1:    mov ah,17h;蓝底白字
    mov al,' ';蓝屏
    mov es:[bx],ax
    add bx,2
    loop s1

    mov bx,0F00h
    mov cx,34;一行有80个字符,占160个字节 40-12/2=34,实现居中
s2:    mov al,'-'
    mov es:[bx],ax
    add bx,2
    loop s2

    mov cx,len
    mov si,0
s3:    mov al,[si]
    inc si
    mov es:[bx],ax
    add bx,2
    loop s3

    mov cx,34

s4:    mov al,'-'
    mov es:[bx],ax
    add bx,2
    loop s4

      mov ax,4c00h
       int 21h
code ends
end

 

 

 

 

 

 

 五、实验总结

使用子程序更加简洁

posted @ 2021-11-29 09:33  whisperii  阅读(75)  评论(3编辑  收藏  举报