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

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

实验任务一

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

点击查看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

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

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

答:
结果
image
①loop s1的机器码为E2F2。F2的八位二进制形式为11110010,补码为10001110,为-14。即位移量为14。
image
从当前位置向前直到0771:000D(这条也要跳过)
用当前标号位置的偏移地址减去跳过的字节数
②loop s2的机器码为E2F0。F0的八位二进制形式为11110000,补码为10010000,为-16。即位移量为16。
image
从当前位置向前直到0771:0029(这条也要跳过)
用当前标号位置的偏移地址减去跳过的字节数

实验任务二

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

点击查看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) = ?
② 对源程序进行汇编、链接,得到可执行程序task2.exe。使用debug调试,观察、验证调试结果与理论分析结果是否一致。

答:
①(ax)= offset s1,(bx)= offset s2,(cx)= code段段地址
image
image
可以看到AX中的值就是POP AX指令的偏移地址,同理BX也正确。
CX是0772,也就是当前显示的段地址,即code段段地址。

实验任务三

已知逻辑段定义如下:

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, 实现题目要求。

正确编写后,预期测试结果如下:
image

附*:int 21h中的2号子功能说明如下:

; 功能:输出单个字符
mov ah, 2
mov dl, ×× ; ××是待输出的字符,或其ASCⅡ码值
int 21h

代码:

点击查看task3.asm
DATAS segment
	x db 99, 72, 85, 63, 89, 97, 55
	len equ $- x
DATAS ends

STACKS SEGMENT
    db 16 dup(0)
STACKS ENDS

CODES SEGMENT
    ASSUME CS:CODES,DS:DATAS,SS:STACKS
START:
    mov ax,DATAS
    mov ds,ax
    mov ax,0B800H
    mov es,ax
    mov dl,10
    mov bl,7
    mov bh,0
    mov si,0
    mov di,0
    mov cx,len
  s:mov ah,0
    mov al,ds:[si]
    div dl
    call c1
    call c2
    inc si
    loop s
    MOV AH,4CH
    INT 21H

 c1:add al,48
 	mov bl,7
 	mov es:[di],al
 	mov es:[di+1],bl
 	add ah,48
 	mov es:[di+2],ah
 	mov es:[di+3],bl
 	mov bl,32
    ret
 c2:mov es:[di+4],bl
 	mov es:[di+5],bh
 	add di,6
    ret
CODES ENDS
    END START

结果:
image
结果非常好

实验任务四

针对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,使得在屏幕最上方以黑底绿字显示字符串,在屏幕最下方以黑底红色显示字符串

正确编写后,预期测试结果如下:
image

代码:

点击查看task4.asm
DATAS SEGMENT
    k db 'try'
    len equ $ - k
DATAS ENDS

STACKS SEGMENT
    db 16 dup(0)
STACKS ENDS

CODES SEGMENT
    ASSUME CS:CODES,DS:DATAS,SS:STACKS
START:
    mov ax,DATAS
    mov ds,ax
    mov bl,2
    mov bh,0
    call p
    mov bl,4
    mov bh,240
    call p
    MOV AH,4CH
    INT 21H
  p:mov cx,len
  	mov si,0
  	mov ax,0B800H
  	add al,bh
  	mov es,ax
  	mov bp,0
  s:mov dl,ds:[bp]
    mov es:[si],dl
    mov es:[si+1],bl
    inc bp
    add si,2
    loop s
    ret
CODES ENDS
    END START

结果:
image

实验任务五

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

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

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

注意:

  1. 80×25彩色字符模式显示缓冲区结构,参见教材「实验9 根据材料编程」里的说明。
  2. 编写程序实现时,将data段的学号换成自己的学号。
点击查看代码
DATAS SEGMENT
    stu_no db '201983290495'
    len = $ - stu_no
DATAS ENDS

STACKS SEGMENT
    db 16 dup(0)
STACKS ENDS

CODES SEGMENT
    ASSUME CS:CODES,DS:DATAS,SS:STACKS
START:
    mov ax,DATAS
    mov ds,ax
    mov ax,0B800H
    mov es,ax
    mov cx,1920
    mov di,0
  s:call c1
    loop s
    mov cx,34
 s1:call c2
    loop s1
    mov si,0
    call m
    mov cx,34
 s2:call c2
    loop s2
    MOV AH,4CH
    INT 21H
 c1:mov byte ptr es:[di],32
 	mov byte ptr es:[di+1],16
 	add di,2
    ret
 c2:mov byte ptr es:[di],45
 	mov byte ptr es:[di+1],23
 	add di,2
 	ret
  m:mov cx,len
  p:mov dl,ds:[si]
    mov es:[di],dl
 	mov byte ptr es:[di+1],23
 	inc si
 	add di,2
 	loop p
 	ret
CODES ENDS
    END START

结果:
image

总结

  1. 基本学会了各种对内存空间的赋值操作。
  2. 掌握了如何在DOS的输出缓冲区里赋值以实现打印的操作,同时对打印的内容的格式控制也有了进一步了解。
  3. 基本掌握了call和ref的使用,能够灵活使用他们来实现编写函数的功能,并实现所需要的操作。
posted @ 2021-11-24 00:22  dove_whispers  阅读(81)  评论(2编辑  收藏  举报