Упражнения

Пример 1.Привести пример блокировки дискового прерывания.

 

Решение.Для блокировки дискового прерывания необходимо маскировать 6-й разряд регистра маски IMR контроллера прерываний, соответствующий линии IRQ6. После выполнения операций, требующих маскирования дискового прерывания, следует обязательно очистить регистр IMR, иначе обращения к дискам будет невозможно.

 

MOV AL, 01000000b; маскируем 6-й бит регистра маски

OUT 21h, AL ; передаем код в регистр маски прерываний

... ; выполнение операций, требующих маскирования прерывания

MOV AL, 0 ; код очистки регистра маски прерываний

OUT 21h, AL ; передаем код в регистр маски прерываний

 

Пример 2.Привести пример замены векторы прерывания с помощью функций MS-DOS.

 

Решение.В практике разработки системных программ часто возникают необходимость дополнения существующего или создания собственного обработчика прерывания. Перед установкой нового вектора прерывания следует получить текущее значение вектора, сохранить его, а затем восстановить перед завершением программы.

Функция 35h прерывания 21h возвращает текущее значение вектора прерывания, помещая адрес сегмента в ES, а смещения в BX. Функция 25h прерывания 21h позволяет установить вектор прерывания на определенный адрес, указанный регистровой парой DS:DX. Заметим также, что обе функции автоматически запрещают аппаратные прерывания на время изменения вектора, поэтому не существует опасности, что может произойти аппаратное прерывание, использующее данный вектор.

В приведенном примере изменяется вектор прерывания клавиатуры 09h. Код завершения подпрограммы обработки прерывания перед инструкцией IRET необходим только для аппаратного прерывания. Для обработчиков, являющихся расширениями существующих прерываний, код завершения не нужен.

 

; сегмент данных

OldInterruptSegment DW ? ; адрес сегмента заменяемого вектора прерывания

OldInterruptOffset DW ? ; адрес смещения заменяемого вектора прерывания

 

; в начале программы

MOV AH,35h ; номер функции для получения вектора

MOV AL,09h ; номер запрашиваемого вектора прерывания

INT 21h ; выполнение функции получения адреса вектора

MOV OldInterruptOffset,BX ; сохранение адреса смещения вектора

MOV OldInterruptSegment,ES ; сохранение адреса сегмента вектора

 

; установка нового прерывания

PUSH DS ; сохранение в стеке содержимого DS

MOV DX,OFFSET Routine ; помещение в DX смещения процедуры обработки

MOV AX,SEG Routine ; помещение в AX сегмента процедуры обработки

MOV DS,AX ; передаем в DS адрес сегмента

MOV AH,25h ; номер функции для установки вектора

MOV AL,09h ; номер изменяемого вектора

INT 21h ; изменение вектора прерывания

POP DS ; восстановление из стека прежнего содержимого DS

 

; новый обработчик прерывания 9h

Routine PROC FAR ; объявление подпрограммы обработчика как дальней

PUSHA ; сохранение регистров в стеке

... ; выполнение кода подпрограммы обработки прерывания

POPA ; восстановление регистров

MOV AL,20h ; завершение обработки аппаратного прерывания

OUT 20h,AL ; передаем код в регистр обслуживания прерывания

IRET ; возврат из процедуры обработки прерывания

Routine ENDP ; завершение подпрограммы

 

; восстановление прежнего прерывания

PUSH DS ; сохранение в стеке регистра DS

MOV DX,OldInterruptOffset ; помещение в DX смещение прежнего вектора

MOV AX,OldInterruptSegment ; помещение в AX сегмента прежнего вектора

MOV DS,AX ; передаем адрес сегмента в регистр DS

MOV AH,25h ; номер функции для установки вектора

MOV AL,09h ; номер изменяемого вектора

INT 21h ; восстановление прежнего вектора

POP DS ; восстановление из стека регистра DS

 

 

Пример 3.Используя ассемблер TASM, разработать резидентную программу: общую структуру, загрузчик и резидентную часть. Функциональность резидентной части может быть произвольной. Использовать режим работы ассемблера IDEAL. В качестве прерывания использовать программное прерывание от системного таймера 1Ch. Предусмотреть возможность завершения работы резидентной программы по клавишной комбинации «Ctrl+x».

 

Решение.Изменение строки таблицы векторов прерываний или иначе подмена прерывания дает возможность создания резидентных программTSR(Terminate and Stay Resident, завершить и оставить резидентным). Такие программы, постоянно расположены в памяти и выполняют работу в фоновом режиме. Примером таких программ могут быть, например, драйвера клавиатуры, дисплея, различных периферийных устройств.

Для создания резидентной программы необходимо переопределить вектор прерывания. Новый обработчик прерывания либо полностью заменяет стандартный, либо только дополняет его. Во втором случае обработчику управление может быть передано до выполнения стандартной подпрограммы обработки прерывания, либо после.

Прерывание по таймеру имеет наивысший приоритет IRQ0, поэтому оно будет вытеснять любые другие прерывания до тех пор, пока не будут заблокированы все прерывания командой CLI. Микросхема таймера получает сигналы от часов с частотой 1.19 МГц. Прерывание по таймеру возникает, когда внутренний счетчик микросхемы таймера обнулится. Первоначально счетчик устанавливается в 65535 и при получении очередного импульса его значение уменьшается на 1. Поэтому прерывания происходят с частотой 18.2Гц.

Микросхема контроллера прерываний в ответ на запрос генерирует прерывание INT 8, которое обслуживает программа BIOS. После обновления времени суток и выполнения ряда других служебных действий, программа BIOS выполняет команду INT 1Ch, программное прерывание, называемое прерыванием пользователя по таймеру.

По умолчанию подпрограмма обработки прерывания INT 1Сh содержит единственную команду возврата из прерывания IRET. Программы, которые должны выполняться периодически, могут установить свою собственную подпрограмму обработки прерывания INT 1Сh. После ее выполнения управление вновь будет передано процедуре обработки времени суток INT 8. Схема расширения прерывания таймера показана на рис. 43.

 

 

 


Рис. 43. Расширение прерывания таймера.

 

 

Определим теперь составные части TSR. Прежде всего, необходимо выделить подпрограмму установки и резидентную часть. Резидентная часть находится перед установщиком, т.к. ее адрес должен быть известен заранее. В свою очередь установщик состоит из загрузчика и программы выгрузки. Загрузчик должен сохранить старый вектор, установить новый вектор прерывания и завершить программу, оставив ее резидентной (рис. 44).

Резидентная часть обработчика определяет функциональность TSR. Она может быть произвольной, поэтому мы ее приводить не будем и оставим читателю для самостоятельной работы. Обработчики прерываний должны быть дальними процедурами, которые сохраняют в стеке состояние всех регистров перед началом выполнения и восстанавливают их по окончанию работы.

 

 
 

 

 


Рис. 44. Алгоритм подпрограммы установки.

 

 
 

 

 


Рис. 45. Алгоритм обработчика прерывания 1Ch.

Если обработчик прерывания полностью заменяет стандартный обработчик аппаратного прерывания, то перед возвратом он должен сообщить контроллеру прерываний об окончании обработки, чтобы разрешить аппаратные прерывания нижних уровней. Если обработчик только дополняет старый обработчик прерывания, то он должен содержать его вызов. Для определенности выберем вариант полной замены существующего обработчика.

Алгоритм работы нового обработчика прерывания от таймера приведен на рис. 45.

Каждой клавише соответствует определенный числовой код, так называемый скан-код. Этот код посылает контроллер клавиатуры 8048 микросхеме интерфейса с периферией 8255. Скан-код однобайтовый, младшие 7 бит которого представляют идентификационный номер, присвоенный каждой клавише, а старший бит определяет состояние клавиши (1 – клавиша нажата, 0 – отпущена).

Завершение работы резидентной части должно происходить по клавишной комбинации «Ctrl+x». В нашем примере скан-код для левой нажатой клавиши «Ctrl» равен 1Dh, а для отпущенной клавиши – 9Dh. Для клавиши «x» (не символа) соответствующие коды равны 2Dh и 0Ah. Прерывание клавиатуры 9h преобразует скан-коды в ASCII коды и устанавливает статус клавиш переключателей. Получить скан-код можно, обратившись к микросхеме 8255 по адресу 60h.

Следовательно, отследить клавишную комбинацию «Ctrl+x» можно таким образом. Зафиксировать факт нажатия левой клавиши «Ctrl», например, с помощью специальной переменной (флага). Флаг устанавливается в 1 при получении скан-кода 1Dh и сбрасывается в 0 при коде 9Dh. Затем следует периодически проверять код нажатой клавиши. При скан-коде 2Dh и установленном флаге имеет место требуемая клавишная комбинация.

Возможный вариант реализации программы TSR приведен ниже.

 

IDEAL

 

MODEL MEDIUM

 

STACK 100h

 

DATASEG

; Флаг состояния левой клавиши Ctrl

LeftCtrlON DB 0

; Область сохранения старого вектора прерывания

OldTimerInterruptOffset DW ?

OldTimerInterruptSegment DW ?

ENDS

 

CODESEG

 

; Основной модуль программы

PROC TimerTest

; Инициализировать регистр DS

MOV AX,DGROUP

MOV DS,AX

; Установить новый обработчик прерывания

CALL SetTimerInterrupt

; Передать управление MS DOS и оставить резидентным

INT 27h

ENDP TimerTest

 

; Новый обработчик прерывания

PROC TimerInterrupt FAR

; Сохранить регистры общего назначения

PUSHA

PUSH DS

; Инициализировать регистр DS

MOV AX,DGROUP

MOV DS,AX

; Получить скан-код

IN AL,60h

; Разрешить прерывания с более низким уровнем

MOV AL,20h

OUT 20h,AL

; Нажата клавиша Ctrl?

CMP AL,1Dh

JZ @@on

; Отпущена клавиша Ctrl?

CMP AL,9Dh

JNZ @@x

; Сбросить флаг левой клавиши Ctrl

MOV [LeftCtrlON],0

JMP SHORT @@go

; Установить флаг левой клавиши Ctrl

@@on:MOV [LeftCtrlON],1

JMP SHORT @@go

; Нажата клавиша «x»?

@@x: CMP AL,2Dh

JNZ @@go

; Нажата левая клавиша Ctrl?

CMP [LeftCtrlON],1

JNZ @@go

; Восстановить прежний обработчик прервания

CALL RestoreOldTimerInterrupt

JMP SHORT @@exit

@@go:... ; «Полезная» часть резидента

@@exit: ; Завершить обработку прерывания

POP DS

POPA

; Выход из обработчика прерывания

IRET

ENDP TimerInterrupt

 

; Установить новый вектор прерывания

PROC SetTimerInterrupt NEAR

PUSHA

PUSH ES

MOV AX,0

MOV ES,AX

; Запомнить прежний вектор обработчика прерывания

MOV AX,[ES:1Ch*4]

MOV [OldTimerInterruptOffset],AX

MOV AX,[ES:1Ch*4+2]

MOV [OldTimerInterruptSegment],AX

; Установка вектора прерывания на обработчик

CLI ; Запретить прерывания

MOV AX,offset TimerInterrupt

MOV [ES:1Ch*4],AX

MOV AX,CS

MOV [ES:1Ch*4+2],AX

STI ; Разрешить прерывания

POP ES

POPA

RET

ENDP SetTimerInterrupt

 

; Восстановить исходный вектор прерывания

PROC RestoreOldTimerInterrupt NEAR

PUSHA

; Настроить регистр ES на таблицу векторов прерываний

PUSH ES

MOV AX,0

MOV ES,AX

; Восстановить прежний вектор обработчика прерывания

CLI

MOV AX,[OldTimerInterruptOffset]

MOV [ES:1Ch*4],AX

MOV AX,[OldTimerInterruptSegment]

MOV [ES:1Ch*4+2],AX

STI

POP ES

POPA

RET

ENDP RestoreOldTimerInterrupt

 

ENDS

 

END

 

В сегменте данных объявлен флаг состояния левой клавиши Ctrl и выделена область для хранения вектора 1Ch. Кодовый сегмент включает несколько подпрограмм. Подпрограмма TimerTest получает управление первой. Она совсем проста и единственное, что выполняет – вызов процедуры SetTimerInterrupt и затем передает управление операционной системе с помощью прерывания 27h.

В свою очередь процедура SetTimerInterrupt устанавливает новый вектор прерывания 1Ch. Здесь и далее в процедуре RestoreOldTimerInterrupt используется иной способ замены вектора. Вместо функций 25h и 35h, рассмотренных в примере 2, показано вычисление вектора 1Ch и установка нового обработчика прерывания.

Резидентная часть представлена подпрограммой TimerInterrupt. Вначале анализируется комбинация «Ctrl+x» и разрешаются прерывания с более низким приоритетом. Если комбинация не выбрана, управление передается на выполнение «полезной» части резидента. Затем происходит выход из подпрограммы обработки прерывания. Если клавишная комбинация выбрана, происходит вызов процедуры RestoreOldTimerInterrupt для восстановления прежнего вектора 1Ch.

Рассмотренный пример резидентной программы имеет достаточно общий характер. Тем не менее, его можно относительно легко адаптировать для решения задач в области измерительной техники. Например, применение алгоритмов программирования плат сбора данных в рамках обработчика TimerInterrupt позволит создать драйвер измерительного устройства. В его ведение может, например, входить выполнение оцифровки сигналов, поступающих от измерительных преобразователей и их первичная обработка. Причем работать драйвер будет в фоновом режиме под операционной системой MS-DOS.

Контрольные вопросы

1. В чем назначение и каковы функции контроллера прерываний?

2. Какие существуют типы прерываний?

3. Как обрабатываются программные и аппаратные прерывания?

4. Приведите примеры прерываний BIOS и MS-DOS.