【汇编 - 功能】中断安装中断实现
前言
在8086汇编中,“中断安装中断”可以理解为通过一个自定义中断来统一管理其他中断服务程序的安装/卸载,实现中断向量的动态配置。以下是一个实现这一功能的代码,通过int 62h作为“中断安装中断”,支持注册新中断、卸载已有中断。
代码实现
; 常量定义
VECTOR_TABLE_SEG equ 0 ; 中断向量表段地址(固定为0段)
MAX_INT_NUM equ 0FFh ; 最大中断号(0-255)
code segment
assume cs:code, ds:data
; ------------------------------
; 中断安装中断服务程序(入口:AH=功能号)
; 功能:
; AH=00h:安装中断(AL=中断号,DS:SI=中断服务程序地址)
; AH=01h:卸载中断(AL=中断号,恢复为默认处理程序)
; AH=02h:查询中断(AL=中断号,返回DS:SI=当前中断服务程序地址)
; ------------------------------
install_int proc far
push ax
push bx
push cx
push dx
push si
push di
push es
push ds ; 保存所有寄存器
push bp
mov bp, sp
; 检查中断号合法性(0-255)
cmp al, MAX_INT_NUM
ja int_end ; 中断号超出范围,不处理
; 根据功能号跳转
cmp ah, 00h
je install ; 安装中断
cmp ah, 01h
je uninstall ; 卸载中断
cmp ah, 02h
je query ; 查询中断
jmp int_end
; ------------------------------
; 功能00h:安装中断
; 入口:AL=中断号,DS:SI=中断服务程序(段:偏移)
; 操作:将DS:SI写入中断向量表对应位置
; ------------------------------
install:
; 计算中断向量在表中的地址:中断号×4(每个向量占4字节:偏移2字节+段2字节)
mov bl, al
mov bh, 0
shl bx, 2 ; BX = 中断号×4(向量表偏移)
; 指向中断向量表(0段)
mov ax, VECTOR_TABLE_SEG
mov es, ax
; 关中断,保证写入向量的原子性
cli
; 保存原中断向量(用于卸载时恢复)
mov ax, es:[bx] ; 原偏移
mov [old_vector + bx], ax
mov ax, es:[bx + 2] ; 原段
mov [old_vector + bx + 2], ax
; 写入新中断向量(DS:SI)
mov ax, si ; 新偏移
mov es:[bx], ax
mov ax, ds ; 新段
mov es:[bx + 2], ax
sti ; 开中断
jmp int_end
; ------------------------------
; 功能01h:卸载中断
; 入口:AL=中断号
; 操作:恢复中断向量表中该中断号的原始值
; ------------------------------
uninstall:
mov bl, al
mov bh, 0
shl bx, 2 ; BX=中断号×4
mov ax, VECTOR_TABLE_SEG
mov es, ax
cli ; 关中断
; 恢复原向量(从保存的old_vector中读取)
mov ax, [old_vector + bx] ; 原偏移
mov es:[bx], ax
mov ax, [old_vector + bx + 2] ; 原段
mov es:[bx + 2], ax
sti ; 开中断
jmp int_end
; ------------------------------
; 功能02h:查询中断
; 入口:AL=中断号;返回:DS:SI=当前中断服务程序地址
; ------------------------------
query:
mov bl, al
mov bh, 0
shl bx, 2 ; BX=中断号×4
mov ax, VECTOR_TABLE_SEG
mov es, ax
; 读取当前向量(偏移→SI,段→DS)
mov si, es:[bx] ; 当前偏移
mov dx, es:[bx + 2] ; 当前段
mov ds, dx ; DS=当前段
jmp int_end
; ------------------------------
; 中断结束:恢复寄存器并返回
; ------------------------------
int_end:
pop bp
pop ds
pop es
pop di
pop si
pop dx
pop cx
pop bx
pop ax
iret ; 中断返回
install_int endp
; ------------------------------
; 初始化:安装"中断安装中断"自身(int 62h)
; ------------------------------
init proc
cli ; 关中断
mov ax, VECTOR_TABLE_SEG
mov es, ax
mov di, 62h * 4 ; int 62h的向量地址:62h×4=0000:0188
; 保存int 62h的原向量(用于程序退出时恢复)
mov ax, es:[di]
mov [orig_62h_off], ax
mov ax, es:[di + 2]
mov [orig_62h_seg], ax
; 写入自定义中断服务程序地址
mov ax, offset install_int
stosw ; 偏移
mov ax, cs
stosw ; 段
sti ; 开中断
ret
init endp
; ------------------------------
; 退出:恢复int 62h的原始向量
; ------------------------------
exit proc
cli
mov ax, VECTOR_TABLE_SEG
mov es, ax
mov di, 62h * 4
mov ax, [orig_62h_off]
stosw
mov ax, [orig_62h_seg]
stosw
sti
; 调用DOS退出
mov ah, 4Ch
int 21h
ret
exit proc
; ------------------------------
; 测试用的示例中断服务程序(如键盘中断处理)
; ------------------------------
test_int proc far
push ax
mov ah, 02h
mov dl, 'A' ; 触发时显示'A'
int 21h
pop ax
iret
test_int endp
; ------------------------------
; 主程序:测试中断安装功能
; ------------------------------
main:
mov ax, data
mov ds, ax ; 初始化数据段
call init ; 安装int 62h(中断安装中断)
; 测试1:通过int 62h安装test_int到int 63h
mov ah, 00h ; 功能:安装
mov al, 63h ; 目标中断号63h
mov si, offset test_int ; 中断服务程序偏移
; DS已指向data,此处需将DS改为test_int所在段(即CS)
push cs
pop ds
int 62h ; 调用中断安装中断
; 恢复DS为data段
mov ax, data
mov ds, ax
; 测试2:调用int 63h,验证是否安装成功
int 63h ; 应显示'A'
; 测试3:通过int 62h查询int 63h的当前服务程序
mov ah, 02h ; 功能:查询
mov al, 63h ; 中断号63h
int 62h ; 返回DS:SI=test_int地址(可自行验证)
; 测试4:通过int 62h卸载int 63h
mov ah, 01h ; 功能:卸载
mov al, 63h ; 中断号63h
int 62h
; 退出程序(恢复环境)
call exit
code ends
; 数据段:保存原始中断向量
data segment
orig_62h_off dw 0 ; int 62h原始偏移
orig_62h_seg dw 0 ; int 62h原始段
; 保存各中断的原始向量(每个中断占4字节,共256个中断)
old_vector dw MAX_INT_NUM * 2 dup(0) ; 256×4字节=1024字节
data ends
end main代码说明
- 核心功能设计:自定义int 62h作为“中断安装中断”,提供三类操作:a.安装中断(AH=00h):将指定中断号与服务程序绑定,保存原始向量以便后续恢复。b.卸载中断(AH=01h):将中断号的向量恢复为安装前的原始值。c.查询中断(AH=02h):获取指定中断号当前绑定的服务程序地址。
- 中断向量表操作:中断向量表位于0段,每个中断号对应4字节(偏移2字节+段2字节)。通过 中断号×4 计算向量地址,操作时需关中断( cli )保证原子性,避免写入过程被其他中断打断。
- 自安装与恢复:程序初始化时,将自身的 install_int 服务程序注册到int 62h,并保存int 62h的原始向量;程序退出时恢复原始向量,避免破坏系统环境。
- 测试逻辑:主程序通过int 62h安装一个测试中断(int 63h),调用该中断验证效果,再查询、卸载,完整演示中断安装中断的使用流程。