《汇编语言》- 读书笔记 - 第13章-int 指令

news2024/11/15 15:34:42

《汇编语言》- 读书笔记 - 第13章-int 指令

  • 13.1 int 指令
  • 13.2 编写供应用程序调用的中断例程
    • 中断例程:求一 word 型数据的平方
      • 主程序
      • 中断处理程序
      • 执行效果
    • 中断例程:将一个全是字母,以0结尾的字符串,转化为大写
      • 主程序
      • 中断处理程序
      • 执行效果
  • 13.3 对 int、iret 和栈的深入理解
      • 主程序
      • 中断处理程序
      • 分析
      • 执行效果
  • 检测点 13.1
  • 13.4 BIOS 和 DOS 所提供的中断例程
  • 13.5 BIOS 和 DOS 中断例程的安装过程
    • BIOS中断例程的安装:
    • DOS中断例程的安装:
  • 13.6 BIOS 中断例程应用
    • 功能 02H 设置光标位置
    • 功能 09H 在光标位置显示字符
  • 13.7 DOS 中断例程应用(INT 21H)
    • 功能 4CH:返回码方式的终止进程
    • 功能 09H:输出一个字符串到标准输出设备上
  • 实验 13 编写、应用中断例程
  • 参考资料

13.1 int 指令

int n 指令的功能是:引发 n 号中断过程,从而执行 n号中断处理程序
系统将一些具有一定功能的子程序,以中断处理程序的方式提供给应用程序调用。
我们在编程的时候,可以用 int 指令调用这些子程序。
当然,也可以自己编写些中断处理程序供别人使用。
以后,我们可以将中断处理程序简称为中断例程

13.2 编写供应用程序调用的中断例程

中断例程:求一 word 型数据的平方

问题一编写、安装中断 7ch 的中断例程。
功能求一 word 型数据的平方。
参数(ax)=要计算的数据。
返回dxax 中存放结果的高 16 位和低 16 位。
应用举例求 2*3456^2 = 16C8000h = 23,887,872

主程序

assume cs:code
code segment
 start:	mov ax,3456		; (ax)=3456
		int 7ch			; 调用中断 7ch 的中断例程,计算ax中的数据的平方
		; dx:ax 存放结果,将结果乘以2
		add ax,ax		; 先加低位
		adc dx,dx		; 再加高位(这里用了进位加法)
						; 23887872
		mov ax,4c00h
		int 21h
code ends
end start

中断处理程序

assume cs:code
code segment
 start:	; ====================== 安装 ====================	
 		; ------------------- 复制数据 -------------------
		mov ax,cs
		mov ds,ax
		mov si,offset sqr	;设置 ds:si 指向源地址
		
		mov ax,0
		mov es,ax
		mov di,200h			;设置 es:di 指向目的地址

		mov cx,offset sqrend-offset sqr	;设置 cx为传输长度
		cld					;设置传输方向为正。movsb中si,di递增
		rep movsb			;重复复制数据次数由 cx 控制
		; ------------------- 复制数据 -------------------
		
		; ----------------- 设置中断向量 -----------------
		mov ax,0
		mov es,ax						; 段地址 0
		mov word ptr es:[7ch*4],200h	; 设置【中断处理程序】的:偏移地址
		mov word ptr es:[7ch*4+2],0		; 设置【中断处理程序】的:段地址
		; ----------------- 设置中断向量 -----------------
	 	; ====================== 安装 ====================	
	 	
		mov ax, 4c00H		; 退出
		int 21H
		
; =======================================================	
; --------------------- 中断处理程序 --------------------
; ax 的平方
; -------------------------------------------------------
; 参数:ax			要计算的数据。
; 返回:dx、ax 		中存放结果的高 16 位和低 16 位。
; -------------------------------------------------------
   sqr:	mul ax			; ax * ax 结果 dx=16位,ax=16位
		iret 			; 返回原程序。 pop IP, pop CS, popf
		
sqrend:	nop
; --------------------- 中断处理程序 --------------------
; =======================================================
code ends
end start
  • 安装功能跟之前都是一样的,只是改下值就行了。
  • 中断处理程序也很简单就是一个乘法就返回了。

执行效果

在这里插入图片描述

中断例程:将一个全是字母,以0结尾的字符串,转化为大写

问题二编写、安装中断 7ch 的中断例程。
功能将一个全是字母,以0结尾的字符串,转化为大写。
参数ds:si指向字符串的首地址。
应用举例data 段中的字符串转化为大写

主程序

assume cs:code
data segment
	db 'conversation',0
data ends

code segment
 start:	mov ax,data
		mov ds,ax
		mov si,0
		int 7ch		
						
		mov ax,4c00h
		int 21h
code ends
end start

中断处理程序

assume cs:code
code segment
 start:	; ====================== 安装 ====================	
 		; ------------------- 复制数据 -------------------
		mov ax,cs
		mov ds,ax
		mov si,offset capital	;设置 ds:si 指向源地址
		
		mov ax,0
		mov es,ax
		mov di,200h				;设置 es:di 指向目的地址

		mov cx,offset capitalend-offset capital	;设置 cx为传输长度
		cld					;设置传输方向为正。movsb中si,di递增
		rep movsb			;重复复制数据次数由 cx 控制
		; ---------------- 安装(复制数据) ----------------
		
		; ----------------- 设置中断向量 -----------------
		mov ax,0
		mov es,ax						; 段地址 0
		mov word ptr es:[7ch*4],200h	; 设置【中断处理程序】的:偏移地址
		mov word ptr es:[7ch*4+2],0		; 设置【中断处理程序】的:段地址
		; ----------------- 设置中断向量 -----------------
	 	; ====================== 安装 ====================
	 
		mov ax, 4c00H		; 退出
		int 21H
		
; =======================================================
; --------------------- 中断处理程序 --------------------
; 将一个全是字母,以0结尾的字符串,转化为大写。
; -------------------------------------------------------
; 参数:ds:si	指向字符串的首地址。
; -------------------------------------------------------
   capital:	push cx			; 备份寄存器
			push si
	change:	mov cl,[si]		; 取字符
			mov ch,0		; 高位不参与计算,置 0 
			jcxz ok			; 如果取到 0 跳到 ok 循环结束
			and byte ptr [si],11011111b	; 否则:字符转大写
			inc si			; si++ 指向下一字符
			jmp short change; 跳到开头继续循环
		ok:	pop si			; 还原寄存器
			pop cx
			iret 			; 返回原程序。 pop IP, pop CS, popf
		
capitalend:	nop
; --------------------- 中断处理程序 --------------------
; =======================================================
code ends
end start

没什么新知识点,都是之前学过的:安装、遍历字符串、字符转大写。

执行效果

在这里插入图片描述

13.3 对 int、iret 和栈的深入理解

问题用 7ch 中断例程完成 loop 指令的功能 (模拟 loop s
参数bx 存 s 到 se 之间的位移
cx 循环次数
返回
应用举例在屏幕中间显示80个“!”

主程序

assume cs:code
code segment
 start:	mov ax,0b800h
		mov es,ax
		mov di,160*12				; 从第16向第一个字符开始(正好一行80个字符)
		
		mov bx,offset s-offset se	; 参数:se到s的【位移】= -9
		mov cx,80					; 参数:循环80; ES: 当前偏移量 000Eh = 23 + (-9) = 14
	s:	mov byte ptr es:[di],'!'	; 
		add di,2
		int 7ch						; 实现 loop s 效果
   se:	nop							; 当前偏移量 0017h = 23
   
		mov ax,4c00h
		int 21h
code ends
end start

在这里插入图片描述

  1. sse 共 9 个字节。把它存进 bx,传给中断处理程序,
    在中断处理程序中我们就是利用它来算出 s 偏移量,来实现循环的。
  2. 从上面的截图可以看出 mov byte ptr es:[di],'!' 这一句被反编译成了两句。
    2.1. 先是 ES: 看上去是设置默认数据段的意思 。
    2.2. 因为第二句中 MOV BYIE PIR [DI],21 中没有写段地址,只写了 [DI]

中断处理程序

assume cs:code
code segment
 start:	; ---------------- 安装(复制数据) ----------------
		mov ax,cs
		mov ds,ax
		mov si,offset lp		;设置 ds:si 指向源地址
		
		mov ax,0
		mov es,ax
		mov di,200h				;设置 es:di 指向目的地址

		mov cx,offset lpend-offset lp	;设置 cx为传输长度
		cld					;设置传输方向为正。movsb中si,di递增
		rep movsb			;重复复制数据次数由 cx 控制
		; ---------------- 安装(复制数据) ----------------
		
		; ----------------- 设置中断向量 -----------------
		mov ax,0
		mov es,ax						; 段地址 0
		mov word ptr es:[7ch*4],200h	; 设置【中断处理程序】的:偏移地址
		mov word ptr es:[7ch*4+2],0		; 设置【中断处理程序】的:段地址
		; ----------------- 设置中断向量 -----------------
	 
		mov ax, 4c00H		; 退出
		int 21H				; 触发33号中断
		
; =======================================================
; --------------------- 中断处理程序 --------------------
; 模拟 loop s 效果
; -------------------------------------------------------
; 参数:bx		下一句指令到 s 之间的位移
; 参数:cx-1	循环次数
; -------------------------------------------------------
	lp:	push bp			; 备份寄存器
		mov bp,sp		; 取栈顶所指的内存位置
		dec cx			; 计数 cx = cx-1
		jcxz lpret		; cx 为 0 就跳出循环
		add [bp+2],bx	; 分析:1
		
 lpret:	pop bp			; 还原寄存器
		iret 			; 分析:2 【重点】
		
 lpend:	nop
; --------------------- 中断处理程序 --------------------
; =======================================================
code ends
end start

分析

  • 分析1. 上面我们从 sp 拿到了栈顶内存地址存到 bp
    这里我们就直接修改这段内存中的偏移量了。当前栈中:[bp, ip, cs, flags]
    注意:这时的 bp 是栈顶,跟栈中备份的那个bp值可不是同一个东西了
    所以修改时使用的是 [bp+2] 指向中断前备份的 IP
  • 分析2. iret 是用来返回原程序,继续执行的。它会从栈中恢复中断前保存的寄存器状态,还原了CS:IP 就能实现继续执行中断前的下一条指令了。
pop IP
pop CS
popf

这里是本节的重点了:欺骗善良无知的iret

  1. 在中断触发前,IP 指向了标号 se 处。(在主程序中我已经计算出了ses之间的偏移)
  2. 中断过程中,会备份当前寄存器:pushf, push cs, push ip
  3. 中断处理程序中:我们在 add [bp+2],bx ; 分析:1 这句修改了栈中备份的IP的值。
    所以iret恢复后其实指向了标号 s,这也就实现了跳回去重新执行的效果。

执行效果

在这里插入图片描述

检测点 13.1

《汇编语言》- 读书笔记 - 检测点 13.1

13.4 BIOS 和 DOS 所提供的中断例程

  1. BIOS(基本输入输出系统)存储在系统板ROM中,包含以下内容:
    • 硬件系统检测初始化程序
    • 外部中断内部中断的中断例程
    • 用于对硬件设备进行I/O操作的中断例程
    • 其他与硬件系统相关的中断例程
  2. DOS操作系统同样提供了中断例程,这些例程是操作系统程序员提供的编程接口
  3. BIOS和DOS提供的中断例程包括许多子程序,它们实现了常见的编程功能。
  4. 程序员可以通过int指令直接调用:BIOSDOS中断例程来完成特定任务。
  5. 硬件设备相关DOS中断例程通常会进一步调用BIOS的中断例程以执行底层硬件操作。

13.5 BIOS 和 DOS 中断例程的安装过程

在8086架构的早期个人计算机系统中,BIOS和DOS中断例程的安装过程并非由用户程序直接执行安装,而是内置于系统的初始化过程中。以下是一个简化的概述:

BIOS中断例程的安装:

  1. 开机后,CPU 一加电,自动从FFFF:0000H地址处开始执行ROM中的代码(即BIOS)。
    (这个位置存放着一条跳转指令,它会跳转到ROM中的实际BIOS启动代码区域。)
  2. BIOS首先进行硬件自检(POST, Power-On Self-Test),检测和初始化硬件设备。
  3. BIOS建立中断向量表,将自身的中断处理程序入口地址填入到中断向量表对应的内存位置。
    (只需要有登记入口地址,因为中断处理程序本身已固化在BIOS的ROM中。)

DOS中断例程的安装:

  1. 硬件自检和初始化完成后,调用 int 19h 进行操作系统的引导。
    1.1. 在BIOSint 19h 中断做出响应时,它首先从硬盘的第一个扇区(即主引导记录MBR)读取512字节的数据到内存特定位置(如0x7C00)。
    1.2. MBR中的前446字节通常是机器码构成的操作系统引导程序(Boot Loader),该引导程序接着会被CPU执行。
    1.3. 引导程序的任务是找到并加载实际的操作系统,例如MS-DOS或更现代的操作系统的核心文件,并将控制权转交给操作系统。
  2. 当DOS启动时,它会接管控制权,并且根据需要重定向某些中断向量,以便调用自己的中断例程来处理特定的服务请求。
  3. DOS通常会保留一部分中断向量用于调用原始的BIOS服务,同时提供额外的中断服务例程以支持文件操作、磁盘读写等功能。

总之,在8086环境下,BIOS和DOS中断例程的安装是指在系统启动阶段由固件操作系统自动完成中断向量表的填充和设置,而不是程序员手动安装的过程。

《汇编语言》- 读书笔记 - 检测点 13.2.

13.6 BIOS 中断例程应用

功能 02H 设置光标位置

INT 10H显示服务 Video Service
功能02H设置光标位置
参数AH=调用 02H 号子程序
BH=显示页码
DH=行(Y坐标)
DL=列(X坐标)
返回
assume cs:code
code segment
 start:	mov ah,2	; 设置光标位置
		mov bh,0	; 显示页码,第0页
		mov dh,5	; dh 中放行号
		mov dl,12	; dl 中放列号
		int 10h
	ok:	mov ax,4c00h
		int 21h
code ends
end start

在这里插入图片描述

功能 09H 在光标位置显示字符

INT 10H显示服务 Video Service
功能09H在当前光标处按指定属性显示字符
参数AH=功能号 09H
AL=字符内容
BH=显示页码
BL=颜色属性
CX=字符重复个数
返回
assume cs:code
code segment
 start:	mov ah,2			; 设置光标位置
		mov bh,0			; 显示页码,第0页
		mov dh,5			; dh 中放行号
		mov dl,12			; dl 中放列号
		int 10h
		
		mov ah,9			; 在光标位置显示字符
		mov al,'a'			; 字符内容
		mov bl,11001010b	; 颜色属性
		mov bh,0			; 显示页码
		mov cx,20			; 字符重复个数
		int 10h
		
	ok:	mov ax,4c00h
		int 21h
code ends
end start

在这里插入图片描述

13.7 DOS 中断例程应用(INT 21H)

在8086架构下,int 21hDOS操作系统为应用程序提供的一种系统API
通过调用int 21h并设置不同的AH寄存器值,程序员可以请求执行多种操作,如:文件操作、设备I/O、内存管理等。
这些中断服务例程相当于DOS系统的函数库,使得应用程序能够与操作系统进行交互和访问底层资源。

功能 4CH:返回码方式的终止进程

INT 21HDOS 中断例程
功能 4CH终止程序的执行,并可返回一个代码
参数AH=功能号 4CH
AL=返回值
返回
mov ah,4ch		; 调用 4ch 号功能
mov al,00h		; 返回值
int 21h

功能 09H:输出一个字符串到标准输出设备上

INT 21HDOS 中断例程
功能09H输出一个字符串到标准输出设备上。待显示的字符串以$作为其结束标志
参数AH=功能号09H
DS:DX=待输出字符的地址
返回
assume cs:code
data segment
	db 'Welcome to masm','$'
data ends

code segment
 start:	mov ah,2			; 设置光标位置
		mov bh,0			;0页显示
		mov dh,5			; dh 中放行号
		mov dl,12			; dl 中放列号
		int 10h
		
		mov ax,data
		mov ds,ax
		mov dx,0			;ds:dx指向字符串的首地址 data:0
		mov ah,9
		int 21h
		
	ok:	mov ax,4c00h
		int 21h
code ends
end start

在这里插入图片描述

实验 13 编写、应用中断例程

《汇编语言》- 读书笔记 - 第13章-实验13编写、应用中断例程

参考资料

DOS 中断参考手册

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1483284.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

中央处理器CPU中的技术

1 知识加油站 1.1 cpu 指令的执行过程 取指:cpu 获取 程序计数器 中存放的指令地址。读取内存中此地址对应指令并存入指令寄存器译码:指令译码器,解析指令运行:算数逻辑单元计算回写:将执行结果写入对应位置 2. cpu…

如何使用ShellSweep检测特定目录中潜在的webshell文件

关于ShellSweep ShellSweep是一款功能强大的webshell检测工具,该工具使用了PowerShell、Python和Lua语言进行开发,可以帮助广大研究人员在特定目录中检测潜在的webshell文件。 ShellSweep由多个脚本模块组成,能够通过计算文件内容的熵来评估…

xsslabs第四关

测试 "onclick"alert(1) 这与第三关的代码是一样的&#xff0c;但是每一关考的点是不一样的所以我们看一下源代码 <!DOCTYPE html><!--STATUS OK--><html> <head> <meta http-equiv"content-type" content"text/html;ch…

C++string类讲解

大家好鸭 见字如面&#xff0c;已经有好久没有写文章了&#xff0c;这段时间忙着学习&#xff0c;也忙着玩&#xff0c;所以停更了一段时间 今天让我们来谈一谈关于C中的string类 什么是string类&#xff1f; 在c语言中我们操作字符串往往采用指针&#xff0c;这样的访问方式并…

不看后悔的腾讯云优惠券领取入口指南,2024最新代金券

腾讯云代金券领取渠道有哪些&#xff1f;腾讯云官网可以领取、官方媒体账号可以领取代金券、完成任务可以领取代金券&#xff0c;大家也可以在腾讯云百科蹲守代金券&#xff0c;因为腾讯云代金券领取渠道比较分散&#xff0c;腾讯云百科txybk.com专注汇总优惠代金券领取页面&am…

二级医院云HIS系统,云HIS源码,支持分院HIS,集团HIS

云HIS具有可扩展、易共享、易协同、低成本、体验号、更便捷、易维护的优势&#xff0c;重新定义了数字化医院信息系统&#xff0c;实现数字化医院信息系统的转型升级。云 HIS 系统功能完善&#xff0c;涵盖临床各业务部门&#xff0c;采集、抽提、汇总、存贮、展现所有的临床诊…

FL Studio 21.2.3.3586 for Mac中文版新功能介绍及2024年最新更新日志

如果你正计划学习音乐制作&#xff0c;一款强大且易学的音乐制作软件是必不可少的。由于很多小伙伴对音乐制作软件没有实际体验过&#xff0c;到底选择哪一款软件最合适成为当下最纠结的问题。 这里为大家推荐一款功能强大且适合新手小伙伴的音乐编曲软件—FL Studio 21.2.3.35…

[物联网] OneNet 多协议TCP透传

[物联网] OneNet 多协议TCP透传 STM32物联网–ONENET云平台的多协议接入产品创建 : https://blog.csdn.net/qq_44942724/article/details/134492924 Onenet tcp 透传 : https://blog.csdn.net/flyme2010/article/details/107086001 tcp服务端测试工具 : http://tcp.xnkiot.com/…

解决导入项目后在idea中不显示的问题

问题&#xff1a; 今天下午重新打开寒假之前负责的项目&#xff0c;发现打不开了&#xff0c; 从master拉取最新代码到我的分支&#xff0c;发现我的分支上显示就是这样子&#xff0c;无论怎么更新代码都不行。 原因&#xff1a; 在上一次上传代码的时候&#xff0c;我把我分…

YOLOv9改进|使用AKConv改进RepNCSPELAN4

专栏介绍&#xff1a;YOLOv9改进系列 | 包含深度学习最新创新&#xff0c;主力高效涨点&#xff01;&#xff01;&#xff01; 一、改进点介绍 AKConv是一种具有任意数量的参数和任意采样形状的可变卷积核&#xff0c;对不规则特征有更好的提取效果。 RepNCSPELAN4是YOLOv9中的…

JAVA的学习日记

JAVA的学习日记&#xff08;2024.3.1&#xff09;&#xff08;b站韩顺平老师课程学习笔记版&#xff09; ps:捡起忘光光的Java语言 Sublime //1. public是公有&#xff0c;class是类 //2. public class Hello表示Hello是一个类&#xff0c;是一个public公有的类 //3. Hello{…

【框架】MyBatis 框架重点解析

MyBatis 框架重点解析 1. MyBatis 执行流程 会话工厂生产的 SqlSession 对象提供了对数据库执行SQL命令所需的所有方法&#xff0c;包括但不限于以下功能&#xff1a; 数据库操作&#xff1a;SqlSession可以执行查询&#xff08;select&#xff09;、插入&#xff08;insert&a…

桥接模式(Bridge Pattern) C++

上一节&#xff1a;适配器模式&#xff08;Adapter Pattern&#xff09; C 文章目录 0.理论1.组件2.使用场景 1.实践 0.理论 桥接模式&#xff08;Bridge Pattern&#xff09;是一种结构型设计模式&#xff0c;它的核心思想是将抽象部分与其实现部分分离&#xff0c;使它们可…

liunx安装jdk、redis、nginx

jdk安装 下载jdk,解压。 sudo tar -zxvf /usr/local/jdk-8u321-linux-x64.tar.gz -C /usr/local/ 在/etc/profile文件中的&#xff0c;我们只需要编辑一下&#xff0c;在文件的最后加上java变量的有关配置&#xff08;其他内容不要动&#xff09;。 export JAVA_HOME/usr/l…

操作系统系列学习——多进程图像

文章目录 前言多进程图像 前言 一个本硕双非的小菜鸡&#xff0c;备战24年秋招&#xff0c;计划学习操作系统并完成6.0S81&#xff0c;加油&#xff01; 本文总结自B站【哈工大】操作系统 李治军&#xff08;全32讲&#xff09; 老师课程讲的非常好&#xff0c;感谢 【哈工大】…

设置文字之间的间距应该如何实现

设置文字之间的间距&#xff0c;通常指的是字母之间&#xff08;字符间距&#xff09;或单词之间的间距。在CSS中&#xff0c;这可以通过letter-spacing和word-spacing属性来实现。 字符间距&#xff08;letter-spacing&#xff09; letter-spacing属性用于调整字符之间的间距…

【Kotlin】函数

1 常规函数 1.1 无参函数 fun main() {myFun() }fun myFun() {println("myFun") // 打印: myFun } 1.2 有参函数 1&#xff09;常规调用 fun main() {myFun("myFun") // 打印: myFun }fun myFun(str: String) {println(str) } 2&#xff09;形参指定默…

C++:设计包含min 函数的栈

目录 题目 代码实现 输出 题目 定义栈的数据结构&#xff0c;要求添加一个min 函数&#xff0c;能够得到栈的最小元素。 要求函数min、push 以及pop 的时间复杂度都是O(1)。 代码实现 #include <iostream>template<typename T>class stack { public:stack() {…

React富文本编辑器开发(四)

上一节我们做了块级元素的格式操作&#xff0c;这节我们来讲行内元素的相关操作。行内元素的样式一般指 粗体、斜体、代码或 删除线等 。通过前一章的内容得知&#xff0c;元素的渲染是通过渲染器来呈现的&#xff0c;块级元素通过指定 renderElement, 行内元素&#xff08;即内…

vxe-table编辑单元格动态插槽slot的使用

业务场景&#xff1a;表格中只有特定某一行的的单元格可以编辑&#xff0c;列很多&#xff0c;为每个列写个插槽要写很多重复代码&#xff0c;所以这里使用动态插槽&#xff0c;简化代码量。显示编辑图标&#xff0c;点击编辑图标隐藏。失去焦点保存调后台接口。 解决办法&…