Глава 10
Вверх
CSEG segment
Start:
;Переходим на метку инициализации. Нам нужно будет перехватить прерывание 21h,
; ==== Обработчик 21h-ого прерывания ====
;Ниже идет, собственно, обработчик 21h прерывания (он будет резидентный).
Int_21h_proc proc
;Если нет, прейдем на оригинальный обработчик прерывания 21h.
Ok_09:
push cs ;Адрес строки должен быть в DS:DX
;Выводим нашу строку (My_string) вместо той, которую должна вывести
pop dx ;Восстановим использованные регистры
;Переменная для хранения оригинального адреса обработчика 21h
;Строка, которая будет выводится
int_21h_proc endp
; ==== Конец обработчика 21h-ого прерывания ====
;Со следующей метки нашей программы уже не будет в памяти
Init:
mov ah,35h ;AH содержит номер функции (это понятно)
mov word ptr Int_21h_vect,bx
;Итак, адрес сохранили. Теперь перехватываем прерывание:
;Все! Теперь, если какая-либо программа вызовет 21h, то вначале компьютер
mov dx,offset Init
CSEG ends
Назад |
Начало урока |
Вверх |
Вперед
10.4 Первый обработчик прерывания
;-------------------------------------
; Prog10.4 Обработчик прерывания
;-------------------------------------
assume cs:CSEG, ds:CSEG, es:CSEG, ss:CSEG
org 100h
;а также оставить программу резидентной в памяти
jmp Init
;После того, как программа выйдет, процедура Int_21h_proc останется в памяти
;и будет контролировать функцию 09 прерывания 21h.
cmp ah,9 ;Проверим: это функция 09h?
je Ok_09 ;Если так, то на метку Ok_09
;Всё. На метку Ok_09 программа уже не вернется.
jmp dword ptr cs:[Int_21h_vect]
push ds ;Сохраним регистры
push dx
pop ds
;программа, вызвавшая 21h-ое прерывание
mov dx,offset My_string
pushf ;Так надо! Позже рассмотрим
call dword ptr cs:[Int_21h_vect]
pop ds
iret ;Продолжим работу (выйдем из прерывания)
;Программа, выводящая строку, считает, что на экран было выведено ее
;сообщение. Но на самом деле это не так!
Int_21h_vect dd ?
My_string db 'Моя строка!$'
;(это нерезидентная часть). Она затрется сразу после выхода (после
;вызова прерывания 27h).
;Установим наш обработчик (Int_21h_proc) (адрес нашего обработчика,
;если быть точнее) на 21h прерывание. Это позволяет сделать функция
;25h прерывания 21h.
;Но прежде нам нужно запомнить оригинальный адрес этого прерывания.
;Для этого используется функция 35h прерывания 21h:
mov al,21h ;AL указывает номер прерывания, адрес (или вектор)
;которого нужно получить
int 21h ;Теперь в ES:BX адрес (вектор) 21h прерывания
;(ES - сегмент, BX - смещение)
mov word ptr Int_21h_vect+2,es ;! Обратите внимание на форму записи
mov ax,2521h
mov dx,offset Int_21h_proc ;DX должен указывать на наш
;обработчик (т.е. Int_21h_proc)
int 21h
;попадет на наш обработчик (Int_21h_proc).
;Что осталось? Завершить программу, оставив ее резидентной в памяти
;(чтобы никто не затер наш обработчик, иначе компьютер зависнет).
int 27h
;Прерывание 27h выходит в DOS (как 20h), при этом оставив нашу программу
;резидентной. DX должен указывать на последний байт, остающийся в памяти
;(это как раз метка Init). Т.е. в памяти остается от 0000h до адреса,
;по которому находится метка Init.
end Start
Содержание