ICode9

精准搜索请尝试: 精确搜索
首页 > 编程语言> 文章详细

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

2021-12-01 23:32:25  阅读:123  来源: 互联网

标签:di 编程 mov 指令 跳转 ax data ds


1. 实验任务1

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

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

运行截图:

offset的作用是取得标号的偏移地址,通过offset x可以取得x的起始地址,也就是0.

len1 equ相当于定义一个常量,常量名为equ,对应偏移地址为$-x,即x后面一个数据项的偏移量,本题为3.

 

 回答问题:

① line27, 汇编指令 loop s1 跳转时,是根据位移量跳转的。通过debug反汇编,查看其机器码,分析其跳转的位移量是多少?

(位移量数值以十进制数值回答)从CPU的角度,说明是如何计算得到跳转后标号s1其后指令的偏移地址的。

答:使用u命令反汇编,机器码为E2F2,F2转换成原码后对应的十进制数值为-14,故位移量为-14.

可以发现CPU首先取得目前正在执行的指令LOOP的地址加上本条指令占用的地址,即001B,减去位移量14,

001B-E= 000D,跳转至IP为000D处。

 

② line44,汇编指令 loop s2 跳转时,是根据位移量跳转的。通过debug反汇编,查看其机器码,分析其跳转的位移量是多少?

(位移量数值以十进制数值回答)从CPU的角度,说明 是如何计算得到跳转后标号s2其后指令的偏移地址的。 

 

答:机器码为E2F0, F0对应的十进制值为-16,故跳转的位移量为-16。CPU跳转过程如下:

CPU首先取得目前正在执行的指令LOOP的地址0037指令,加上本条指令占用的地址2,变为0039,

然后减去标号的地址进行跳转,0039+(-0010)=0029,即跳转至IP为0029处。

 

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

反汇编截图见上面两小题中。

 

2.实验任务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的偏移地址,bx寄存器存放的是s2的偏移地址;cx寄存器存放的是cs的段地址。        执行源程序line24 call word ptr ds:[0]时,下一条指令s1:pop ax的IP值入栈,接着跳转执行s1:pop ax, 栈顶的IP值(即s1的偏移地址)赋给ax。        执行源程序line27 call dword ptr ds:[2]时,代码段段地址cs和下一条指令的IP值相继入栈,接着跳转 到执行s2段的代码pop bx以及pop cx,即将栈顶的IP(S2偏移地址)出栈赋给bx,再将段地址cs出栈赋给cx。   ② 对源程序进行汇编、链接,得到可执行程序task2.exe。使用debug调试,观察、验证调试结果与理论 分析结果是否一致。

         由上面几张图可知,执行完call word ptr ds:[0]的IP值0021,在执行完s1:pop ax后被赋给了ax寄存器。

执行完call dword ptr ds:[2]的CS=076C和IP=0026入栈,执行s2段后,bx = 0026,cx=076C。

        调试结果与理论分析结果一致。

 

3. 实验任务3         编写8086汇编源程序task3.asm,在屏幕上以十进制形式输出data段中这一组连续的数据,数据和数据 之间以空格间隔。 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 bx,0
s:
   mov al,[di]
   mov ah,0
   call printNumber
   call printSpace
   inc di
   loop s

   mov ax,4c00h
   int 21h

printNumber:
   mov bl,10
   div bl

   mov dl,al
   or dl,30h
   mov bh,ah
   mov ah,2
   int 21h

   mov dl,bh
   or dl,30h
   int 21h
   ret

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

 

4. 实验任务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 cx,len
   mov si,0
   mov bl,2h
   mov bh,0
   call printStr

   mov cx,len 
   mov si,0
   mov bl,4h
   mov bh,24
   call printStr

   mov ah,4ch
   int 21h

printStr:
   mov ax,0b800h
   mov es,ax
   mov ax,00a0h
   mul bh
   mov di,ax

s:
   mov al,[si]
   mov es:[di],al
   inc si
   inc di
   mov es:[di],bl
   inc di
   loop s
   ret

code ends
end start
截图:

 

5. 实验任务5

 task5.asm代码

assume cs:code,ds:data

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

code segment
start:
   mov ax,data
   mov ds,ax
   
   mov bl,16
   call backGround
   
   mov bh,24
   mov bl,23
   mov si,0
   call font

   mov ah,4ch
   int 21h

backGround:
    mov ax,0b800h
    mov es,ax
    mov cx,25*80
    mov al,' '
    mov di,0
    s:
    mov es:[di],al
    inc di
    mov es:[di],bl
    inc di
    loop s
    ret

font:
    mov ax,0b800h
    mov es,ax
    mov ax,00a0h
    mul bh
    mov di,ax
    
    mov cx,(80-len)/2
    s1:
        mov al,'-'
        mov es:[di],al
        inc di
        mov es:[di],bl
        inc di
        loop s1

    mov cx,len

    s2:
        mov al,[si]
        mov es:[di],al
        inc si
        inc di
        mov es:[di],bl
        inc di
        loop s2
     mov cx,(80-len)/2

    s3:
        mov al,'-' 
        mov es:[di],al
        inc di
        mov es:[di],bl
        inc di
        loop s3
        ret 

code ends
end start

截图:

 

 

 

标签:di,编程,mov,指令,跳转,ax,data,ds
来源: https://www.cnblogs.com/freddy2002/p/15606581.html

本站声明: 1. iCode9 技术分享网(下文简称本站)提供的所有内容,仅供技术学习、探讨和分享;
2. 关于本站的所有留言、评论、转载及引用,纯属内容发起人的个人观点,与本站观点和立场无关;
3. 关于本站的所有言论和文字,纯属内容发起人的个人观点,与本站观点和立场无关;
4. 本站文章均是网友提供,不完全保证技术分享内容的完整性、准确性、时效性、风险性和版权归属;如您发现该文章侵犯了您的权益,可联系我们第一时间进行删除;
5. 本站为非盈利性的个人网站,所有内容不会用来进行牟利,也不会利用任何形式的广告来间接获益,纯粹是为了广大技术爱好者提供技术内容和技术思想的分享性交流网站。

专注分享技术,共同学习,共同进步。侵权联系[81616952@qq.com]

Copyright (C)ICode9.com, All Rights Reserved.

ICode9版权所有