实验3 转移指令跳转原理及其简单应用编程
实验任务1
给出程序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
运行截图:
① line27, 汇编指令 loop s1 跳转时,是根据位移量跳转的。通过debug反汇编,查看其机器码,分析其跳转的位移量是多少?(位移量数值以十进制数值回答)从CPU的角度,说明 是如何计算得到跳转后标号s1其后指令的偏移地址的。
位移量是:14(Loop指令结束地址为:001B,s1指令开始地址为000D,001B-000D=14)
分析:or dl, 30h命令占三个字节,inc指令占一个字节,其他指令各占两个字节,加起来一共14字节。
② line44,汇编指令 loop s2 跳转时,是根据位移量跳转的。通过debug反汇编,查看其机 器码,分析其跳转的位移量是多少?(位移量数值以十进制数值回答)从CPU的角度,说明 是如何计算得到跳转后标号s2其后指令的偏移地址的。
位移量是:16(Loop指令结束地址为:0039,s1指令开始地址为0029,001B-000D=16)
分析:or dl, 30h命令占三个字节,其他指令各占两个字节,加起来一共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
① 根据call指令的跳转原理,先从理论上分析,程序执行到退出(line31)之前,寄存器(ax) = ? 寄存器 (bx) = ? 寄存器(cx) = ?
答:理论上ax中的数据应该是s1:pop ax这条指令的 IP。因为 call word ptr ds:[0] 将 s1:pop ax指令的IP压入了栈中。
bx中应该是s2:pop bx这条指令的IP,cx中应该是s2:pop bx这条指令的CS。
因为 call dword ptr ds:[2] 指令将 s2:pop bx的CS,IP先后压入栈中。
② 对源程序进行汇编、链接,得到可执行程序task2.exe。使用debug调试,观察、验证调试结果与理论 分析结果是否一致。
与猜想结果一致。
实验任务三
给出程序源码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 si, 0 mov bl, 10 mov cx, len s: mov ah, 0 mov al, [si] call printNumber call printSpace inc si loop s mov ah, 4ch int 21h printNumber: div bl mov dl, al ;al为商,存在dl中 mov bh, ah ;ah为余数,存在bh中 or dl,30H ;输出十位dl mov ah,2 int 21h mov dl, bh ;输出个位bh or dl,30H mov ah,2 int 21h ret printSpace: mov ah, 2 mov dl, ' ' int 21h ret code ends end start
运行测试截图:
实验任务四:
代码:
assume cs:code, ds:data data segment str db 'try' len equ $ - str data ends stack segment db 16 dup(0) stack ends code segment start: mov ax,stack mov ss,ax mov sp,16 mov ax, data mov ds, ax mov ax, 0b800h ;显存的内存地址为b8000h~bFFFFh mov es, ax mov di, 0 mov cx, len mov si, offset str s1: mov bl, 2 ;设定颜色为绿色 mov bh, 0 ;设置行 call printStr loop s1 mov cx, len mov si, offset str mov bh, 24 ;设置行 mov al, 0a0h ;每行的字节数为160(8每行0列) mov ah, 0 mul bh mov di, ax ;设置位置 s2: mov bl, 4 ;设定颜色为红色 call printStr loop s2 mov ah, 4ch int 21h printStr: mov ah, bl ;ah存颜色 mov al, [si] ;al存显示的内容 mov es:[di], ax ;es:[di]是指定输出位置 add di, 2 inc si ret code ends end start
运行截图:
实验任务5
代码:
assume cs:code, ds:data data segment stu_no db '201983290435' len = $ - stu_no data ends stack segment dw 0,0,0,0,0,0,0,0,0 ;定义一个段,用来做栈段,容量为16个字节 stack ends code segment start: mov ax,stack mov ss,ax mov sp,16 mov ax, data mov ds, ax mov ax, 0b800h mov es, ax ;第一层循环 mov di, 0 mov cx, 24 s1: push cx ;将外层循环的cx压栈 mov cx, 80 ;将cx设置成内层循环的次数 s2: mov ax, 1720h mov es:[di], ax add di, 2 loop s2 mov di, 0 pop cx ;从栈顶将cx的值进行恢复 mov ax, es add ax, 0ah mov es, ax loop s1 ;第二层循环 mov di, 0 mov cx, 30 s3: mov ax, 172dh mov es:[di], ax add di, 2 loop s3 ;第三层循环 mov di, 003ch ;设置学号起始位置 mov si, offset stu_no mov cx, len s4: mov ah, 017h ;ah存颜色 mov al, [si] ;al存显示的内容 mov es:[di], ax ;es:[di]是指定输出位置 inc si add di, 2 loop s4 ;第四层循环 mov cx, 38 s5: mov ax, 172dh mov es:[di], ax add di, 2 loop s5 mov ah, 4ch int 21h code ends end start
截图:
实验总结:
or dl,30H mov ah,2 int 21h
的作用是将dl寄存器中的数转化为相应的ASCLL码