Доклад: Системное программирование
1. По номеру источника прерывания путем умножения на 4 определяет смещение в таблице векторов прерываний.
2. Помещает первые два байта по вычисленному адресу в регистр ip.
3. Помещает вторые два байта по вычисленному адресу в регистр cs.
4. Передает управление по адресу, определяемому парой cs:ip.
Далее выполняется сама программа обработки прерывания. Она, в свою очередь, также может быть прервана, например, поступлением запроса от более приоритетного источника. В этом случае этапы 1 и 2 будут повторены для вновь поступившего запроса.
Набор действий по реализации этапа 3 заключается в восстановлении контекста прерванной программы. Так же, как и на этапе 1, на данном последнем этапе есть действия, выполняемые микропроцессором автоматически, и действия, задеваемые программистом. Основная задача на этапе 3 – привести стек в состояние, в котором он был сразу после передачи управления данной процедуре. Для этого программист указывает необходимые действия по восстановлению регистров и очистке стека. Этот участок кода необходимо защитить от возможности искажения содержимого регистров (в результате появления аппаратного прерывания) с помощью команды cli. Последние команды в процедуре обработки прерывания – sti и iret, при обработке которых микропроцессор выполняет следующие действия:
1) sti – разрешить аппаратные прерывания по входу INTR;
2) iret – извлечь последовательно три слова из стека и поместить их, соответственно, в регистры ip, cs, flags.
В результате этапа 3 управление возвращается очередной команде прерванной программы, которая должна была выполнится, если бы прерывания не было.
Аппаратные прерывания могут быть инициированы программно командой микропроцессора int n, где n – номер аппаратного прерывания в соответствие с таблицей векторов прерываний. При этом микропроцессор также сбрасывает флаг IF, но не вырабатывает сигнал INTA.
Написание собственного прерывания
Имеется несколько причин для написания собственного прерывания. Во-первых, большинство из готовых прерываний, обеспечиваемых операционной системе, ничто иное, как обычные процедуры, доступные для всех программ, поэтому можно добавить что-то свое. Например, многие программы могут использовать процедуру, выводящую строки на экран вертикально. Вместо того, чтобы включать ее в каждую программу в качестве процедуры можно установить ее как прерывание, написав программу, которая останется резидентной в памяти после завершения. Тогда, например, можно использовать int 80h вместо WRITE_VERTICALLY (вызов прерывания несколько медленней, чем вызов процедуры).
Второй причиной написания прерывания может быть использование какого-либо отдельного аппаратного прерывания. Это прерывание автоматически вызывается при возникновении определенных условий. В некоторых случаях BIOS инициализирует вектор этого прерывания так, что он указывает на процедуру, которая вообще ничего не делает (она содержит только iret). Можно написать свою процедуру и изменить вектор прерываний, чтобы он указывал на нее. Тогда при возникновении аппаратного прерывания будет выполняться нами написанная процедура. Одна из таких процедур это прерывание времени суток, которое автоматически вызывается 18,2 раза в секунду. Обычно это прерывание только обновляет показания часов, но можно изменить его код как угодно программисту.
И, наконец, возникает иногда желание написать прерывание, которое полностью заменит одну из процедур ОС, приспособленное к программным нуждам.
Рассмотрим пример разработки прерывания в общем случае.
Функция 25h прерывания 21h устанавливает вектор прерывания на указанный адрес cs:ip. Чтобы установить вектор, указывающий на одну из ваших процедур, нужно поместить сегмент процедуры в ds, а смещение в dx. Затем нужно поместить номер прерывания в al и вызвать функцию. Любая процедура прерывания должна завершаться не обычной инструкцией RET, а IRET (IRET выталкивает из стека три слова, включая регистр флагов, в то время как RET помещает на стек только два, если вы попытаетесь тестировать такую процедуру как обычную процедуру, но кончающуюся IRET, то Вы исчерпаете стек.). Отметим, что функция 25H автоматически запрещает аппаратные прерывания в процессе изменения вектора, поэтому не существует опасности, что посреди дороги произойдет аппаратное прерывание, использующее данный вектор.
Когда программа завершается, должны быть восстановлены оригинальные вектора прерываний. В противном случае последующая программа может вызвать данное прерывание и передать управление на то место в памяти, в котором Вашей процедуры уже нет. Функция 35 прерывания 21H возвращает текущее значение вектора прерывания, помещая значение сегмента в ES, а смещение в BX. Перед установкой своего прерывания получите текущее значение вектора, используя эту функцию, сохраните эти значения, и затем восстановите их с помощью функции 25H (как выше) перед завершением своей программы.
Например:
;---в сегменте данных:
KEEP_CS DW 0 ;хранит сегмент заменяемого прерывания
KEEP_IP DW 0 ;хранит смещение прерывания
;---в начале программы
MOV AH,25H ;функция получения вектора
MOV AL,1CH ;номер вектора
INT 21H ;теперь сегмент в ES, смещение в BX
MOV KEEP_IP,BX ;запоминаем смещение
MOV KEEP_CS,ES ;запоминаем сегмент
; ---в конце программы
CLI
PUSH DS ;DS будет разрушен