汇编语言第16章实验

assume cs:codesg
codesg segment

start:	
			mov bx,cs
			mov ds,bx
			mov si,offset setscreen
			mov bx,0
			mov es,bx
			mov di,200h		;将中断例程安装到0:0200h处
			
			mov cx,offset setscreenEnd - offset setscreen
			cld
			rep movsb
		
			cli					;设置IF=0  防止设置中断向量时 发生中断
			mov word ptr es:[7ch*4],200h
			mov word ptr es:[7ch*4+2],0	;设置新的中断向量
			sti
        
			mov ax,4c00h
			int 21h
org 200h	;指定后面的程序放到从偏移地址为200h处,子程序被安装到到0:200h处时 子程序的偏移地址起始地址为200h
			;而没有被安装时 刚编译完加载程序的地址为 cs:xxx  xxx为 setscreen标号的所在偏移地址  不一定为200h
			;把这段(表现为二进制数据形式的)指令复制到另外一块内存区域,每条指令在内存中的位置发生巨大变化,
			;原来的直接定址表就不适用了,它提供的偏移地址和现在的子程序位置不对应了,程序的跳转就会失控。
			
setscreen: 
			jmp short set
			table dw sub1,sub2,sub3,sub4

	set:	push bx
			cmp ah,3	;判断功能号是否大于3
			ja sret
			mov bl,ah
			mov bh,0
			add bx,bx	;根据ah功能号计算对应子程序在table中的偏移
			
			call word ptr table[bx]
			
	sret:  	pop bx
			iret		;注意这里时iret

;功能清屏
sub1:		
			push cx
			push es
			push bx
			mov cx,2000
			mov bx,0b800h
			mov es,bx
			mov bx,0
	sub1s:	mov byte ptr es:[bx],' '
			add bx,2
			loop sub1s
			pop bx
			pop es
			pop cx
			ret
;设置前景色  低3位为前景色,al存放颜色
sub2:
			push cx
			push bx
			push es
			mov cx,2000
			mov bx,0b800h
			mov es,bx
			mov bx,1
	sub2s:	and byte ptr es:[bx],11111000b
			or byte ptr es:[bx],al
			add bx,2
			loop sub2s
			pop es
			pop bx
			pop cx
			ret

;设置背景色  第4 5 6位为背景色  al存放颜色  0000 0111
sub3:
			push cx
			push bx
			push es
			mov cx,2000
			mov bx,0b800h
			mov es,bx
			mov bx,1
	sub3s:	and byte ptr es:[bx],10001111b
			shl al,1
			shl al,1
			shl al,1
			shl al,1
			or byte ptr es:[bx],al   ;al需要左移4位
			add bx,2
			loop sub3s
			pop es
			pop bx
			pop cx
			ret
			
;向上滚动一行  即将后面的数据依次向前复制 movsb
sub4:		
			push cx
			push bx
			push es
			push ds
			push di
			push si
			
			mov bx,0b800h
			mov es,bx
			mov ds,bx
			mov si,160		;ds:si 源地址
			mov di,0		;es:di 目的地址
			mov cx,24		;复制24行 每行160字节
			cld				;cld  DF=0  std DF=1 最后一个字母表示要设置的标志寄存器  cli  sti
	sub4s:  push cx
			mov cx,160		;每160个字节一个小循环
			rep movsb
			pop cx
			loop sub4s
			
			mov cx,80
			mov bx,0
	sub4s1: mov byte ptr es:[bx],' ';最后一行 单独处理
			add bx,2
			loop sub4s1
			
			pop si
			pop di
			pop ds
			pop es
			pop bx
			pop cx
			ret
setscreenEnd: 
			nop
codesg ends
end start

测试程序

assume cs:code
code segment
begin:  
        mov ah,2		;测试2号程序
        mov al,2    ;   闪烁 背景R G B 高亮 前景R G B
        int 7CH
        
        mov ax,4c00h
        int 21H
        
code ends
end begin
### 汇编语言王爽第四版第十实验内容及指导 #### 实验目标 通过编写并调试简单的汇编程序,理解子程序调用机制以及堆栈的工作原理。掌握 `CALL` 和 `RET` 指令的作用及其对寄存器的影响。 #### 实验环境设置 为了确保能够顺利运行和测试代码,在开始之前需配置好开发环境: - 使用 MASM 或 TASM 编译工具; - 准备 DOSBox 或其他兼容 MS-DOS 的虚拟机来模拟真实的硬件平台; #### 关键知识点解析 当执行 `call s` 指令时会先保存当前指令指针(IP),即将其压入堆栈中以便稍后返回继续执行原位置后的代码[^1]。之后跳转至指定标签处执行新流程。而当遇到 `ret` 返回指令时,则自动弹出先前存储于栈内的地址值重新加载回 IP 寄存器从而实现函数间的来回切换功能[^2]。 #### 示例代码分析 下面给出一段用于演示上述理论概念的小例子,并附带详细说明每一步骤的具体作用: ```assembly assume cs:code, ss:stack ; 定义段描述符 stack segment ; 堆栈段定义 db 16 dup(0) ; 初始化16字节大小的空间作为工作区 stack ends ; code segment ; 代码段开始标记 start: mov ax, stack ; 将堆栈段基址送入AX暂存器内 mov ss, ax ; 设置SS为新的堆栈段选择子 mov sp, 16 ; SP指向最顶端位置准备接收后续数据项 mov ax, 1000h ; AX=CS初始设定值 push ax ; CS的值被推送到堆栈顶部 mov ax, 0 ; 清零AX用来表示IP应置位成0 push ax ; 把清空后的AX也一并加入其中代表起始偏移量 retf ; 远程返回操作码触发远距离跳跃动作 ; 此刻CPU将会依次取出两个双字分别赋予IP/CS完成跨段转移 code ends ; 结束整个源文件编辑过程 end start ; 明确指出入口点位于哪里 ``` 这段程序展示了如何利用堆栈管理控制流转向不同部分的方法。特别值得注意的是最后那条远程返回命令(`retf`)它不仅改变了下一条待处理语句的位置还同时更改了正在使用的代码段编号实现了真正的模块化编程风格。 #### 调试技巧提示 建议读者尝试借助 DEBUG 工具逐步单步追踪以上实例中的各项变动状况,尤其是关注各个通用目的寄存器的状态改变趋势。这有助于加深对于底层架构运作细节的理解程度[^3]。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值