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

实验任务1

程序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

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


位移量为14。001B-000D=000E=14。
转移指令机器码为E2F2。F2=(11110010)补=(10001110)原=-14,CPU在执行转移指令后IP先+2然后减去14。

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


位移量为16。0039-0029=0010=16
转移指令机器码为E2F0。F2=(11110000)补=(10010000)原=-16,CPU在执行转移指令后IP先+2然后减去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

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

call先把ip或者cs:ip压栈然后跳。
ax为s1的段地址。cx:bx为s2的偏移地址。

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

实验任务3

给出程序源码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 cx, len
    mov byte ptr ds:[len], 10
    mov si, 0
s:
    mov al, ds:[si]
    mov ah, 0
    call printNumber
    call printSpace
    inc si
    loop s

    mov ah, 4ch
    int 21h

printNumber:
    div byte ptr ds:[len]
    ;AH:AL余:数商
    
    mov bx, ax

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

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

    ret

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

运行测试截图

实验任务4

给出程序源码task4.asm

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 ax, 0b800h
    mov es, ax
    mov si, 0
    mov di, 0
    mov cx, len
    mov bl, 02h
    call printStr


    mov si, 0
    mov di, 160 * 24
    mov cx, len
    mov bl, 04h
    call printStr
    
    mov ax, 4c00h
    int 21h

printStr:
s:
    mov al, [si]
    mov es:[di], al
    inc di
    mov es:[di], bl
    inc di
    inc si
    loop s
    ret
code ends
end start

运行测试截图

实验任务5

给出程序源码task5.asm

assume cs:code, ds:data

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

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

    mov ax, 0b800h
    mov es, ax
    mov di, 0
    mov bl, 17h
    mov cx, 80 * 25

    s1:
    mov al, ' '
    mov es:[di], al
    inc di
    mov es:[di], bl
    inc di
    loop s1

    mov di, 160 * 24
    mov cx, 34
s2:
    mov al, '-'
    mov es:[di], al
    inc di
    mov es:[di], bl
    inc di
    loop s2

    mov di, 160 * 24 + 34 * 2
    mov cx, 12
    mov si, 0
s3:
    mov al, [si]
    mov es:[di], al
    inc di
    mov es:[di], bl
    inc di
    inc si
    loop s3

    mov di, 160 * 24 + 46 * 2 
    mov cx, 34
s4:
    mov al, '-'
    mov es:[di], al
    inc di
    mov es:[di], bl
    inc di
    loop s4

    mov ax, 4c00h
    int 21h
code ends
end start

运行测试截图

实验总结

  1. 从0b800开始是显存,大小为80x25,每个单元两个字节,高位字节放数据低位设置显示格式。
  2. 同一个寄存器可能有很多用途。可以保存很多不同指令的数据。
posted @ 2021-11-28 14:40  nxdxml  阅读(88)  评论(4)    收藏  举报