
- •Оглавление
- •Общие сведения
- •Среда выполнения программ
- •Представление информации
- •Двоичные числа без знака
- •Двоичные числа со знаком
- •Символьная информация
- •Двоично-десятичные числа
- •Адресация памяти
- •Физическая память
- •Сегментация
- •Регистры микропроцессора
- •Регистры общего назначения
- •Сегментные регистры
- •Регистр флагов
- •15 0
- •Указатель инструкции
- •Выполнение инструкций
- •Адресация операндов
- •Регистровые операнды
- •Операнды в памяти
- •Непосредственные операнды
- •Адреса переходов
- •Формат кода команды
- •Префиксы
- •Код операции
- •Байт ModRegR/m
- •7 6 5 4 3 2 1 0
- •Отклонение
- •Непосредственный операнд
- •Инициализация процессора
- •Прерывания
- •Источники и причины прерываний
- •Внешние прерывания
- •Немаскируемые прерывания
- •Маскируемые прерывания
- •Исключения
- •Прерывание по ошибке деления
- •Прерывание по точке останова
- •Прерывание по переполнению
- •Прерывание по трассировке
- •Программные прерывания
- •Обработка прерываний
- •Действия микропроцессора по обработке прерывания
- •Приоритет прерываний
- •Идентификация источников прерываний
- •Система команд
- •Машинные инструкции
- •Aaa– коррекция сложения двоично-десятичных распакованных чисел
- •Замечания по программированию
- •Aad– коррекция деления двоично-десятичных распакованных чисел
- •Замечания по программированию
- •Aam– коррекция умножения двоично-десятичных распакованных чисел
- •Замечания по программированию
- •Aas– коррекция вычитания двоично-десятичных распакованных чисел
- •Add– двоичное сложение без переноса
- •Замечания по программированию
- •Adc– двоичное сложение с переносом
- •Замечания по программированию
- •And– логическое и
- •Call– вызов процедуры
- •Cbw– преобразование байта в слово
- •Cmps– сравнение строк
- •Замечания по программированию
- •Cwd– преобразование слово в двойное слово
- •Daa– коррекция сложения двоично-десятичных упакованных чисел
- •Das– коррекция вычитания двоично-десятичных упакованных чисел
- •Dec– декремент
- •Замечания по программированию
- •Div– деление беззнаковое
- •Esc– команда сопроцессора
- •Замечания по программированию
- •Hlt– останов процессора
- •Idiv– деление знаковое
- •Imul– умножение знаковое
- •In– ввод информации из порта ввода-вывода
- •Inc– инкремент
- •Замечания по программированию
- •Jmp– безусловный переход
- •Lahf– загрузка флагов состояния в регистрAh
- •Lds– загрузка дальнего указателя для сегмента данных
- •Lea– загрузка эффективного адреса
- •Замечания по программированию
- •Les– загрузка дальнего указателя для дополнительного сегмента данных
- •Lods– загрузка строки
- •Замечания по программированию
- •Loop/looPcc– цикл по счётчику в регистреCx
- •Mov– пересылка
- •Movs– пересылка строк
- •Замечания по программированию
- •Mul– умножение беззнаковое
- •Neg– смена знака
- •Out– вывод информации из порта ввода-вывода
- •Pop– извлечение информации из стека
- •Замечания по программированию
- •Popf– загрузка регистра флагов из стека
- •Push– занесение информации в стек
- •Замечания по программированию
- •Pushf– запись регистра флагов в стек
- •Rcl– циклический сдвиг влево через флажок переноса
- •Замечания по программированию
- •Rcr– циклический сдвиг вправо через флажок переноса
- •Замечания по программированию
- •Ret– возврат из процедуры
- •Замечания по программированию
- •Rol– циклический сдвиг влево
- •Замечания по программированию
- •Ror– циклический сдвиг вправо
- •Замечания по программированию
- •Sahf– загрузкаAHв регистр флагов
- •Sal – арифметический сдвиг влево
- •Замечания по программированию
- •Sar– арифметический сдвиг вправо
- •Замечания по программированию
- •Sbb– двоичное вычитание с заёмом
- •Замечания по программированию
- •Scas– сканирование строки
- •Замечания по программированию
- •Shl – логический сдвиг влево
- •Замечания по программированию
- •Shr– логический сдвиг вправо
- •0 ─►│ Операнд ├──►│cf│
- •Замечания по программированию
- •Замечания по программированию
- •Sub– двоичное вычитание
- •Замечания по программированию
- •Test– логическое сравнение
- •Wait– ожидание сопроцессора
- •Замечания по программированию
- •Xchg– обмен информацией
- •Xlatb– табличная перекодировка
- •Xor– исключающее или
- •Префиксы машинных инструкций
- •Ds:/es:/ss:/cs:– префиксы замены сегмента
- •Lock– блокировка шины
- •Rep/rePcc– повторение строковой операции
- •Приложение 1. Коды операций
- •Приложение 2. Время выполнения команд
Идентификация источников прерываний
К сожалению, архитектура процессоров семейства 80x86 не позволяет однозначно идентифицировать источник прерывания по его вектору. Например, прерывание по вектору 0 может произойти по одной из следующих причин:
– по обнаружению ошибки во время выполнения инструкций деления DIVиIDIV(“штатная” причина для прерывания по этому вектору);
– по инструкции INT 0;
– при поступлении сигнала внешнего маскируемого прерывания, сопровождаемого передачей нулевого номера вектора прерывания5.
В связи с отсутствием однозначной идентификации при возникновении прерывания его обработчик должен определить конкретную причину возникновения прерывания. Способ идентификации определяется программистом, однако в любом случае следует обратить внимание на следующие моменты.
Используемый в персональных компьютерах контроллер прерываний на базе микросхем типа 8259 (эмулируемый современными чипсетами) фиксирует факт обслуживания прерывания по каждой из своих линий запросов прерываний. Обработчик прерывания должен проверить состояние соответствующего внутреннего регистра контроллера прерываний, чтобы определить, послужил ли причиной прерывания внешний запрос или какое-нибудь событие внутри программы. Естественно, что если данному вектору прерывания не соответствует какая-либо линия запроса внешнего прерывания (IRQ), то и произойти это прерывание по внешнему запросу не может.
При прерывании по ошибке деления (вектор 0) сохранённый в стеке адрес будет указывать на команду деления, вызвавшую прерывание, в то время как при выполнении инструкции INT 0– на следующую за ней инструкцию, которая также может оказаться командой деления. Необходимо проанализировать код команды, на которую указывает сохранённый в стеке адрес инструкции – если это не команда деления, то причиной прерывания является инструкцияINT 0. Если же сохранён адрес команды деления, то требуется “ручная” проверка её параметров. Если они окажутся корректными, то прерывание возникло из-за инструкцииINT 0, непосредственно предшествующей команде деления. Если же параметры некорректны, причиной могла послужить и предшествовавшая инструкцияINT 0, и инструкция деления. Совершенно однозначная идентификация в такой ситуации невозможна: хотя последние два байта, предшествующие команде деления, могут совпасть с кодом командыINT 0(CD00), нельзя поручиться, что эти байты не являются частью многобайтового кода какой-нибудь другой команды.
Чтобы определить, произошло ли прерывание по вектору 1 в результате трассировки выполнения программы с помощью флажка TFили из-за выполнения инструкцииINT 1, нужно проверить состояние флажкаTFв содержимом регистраFLAGS, сохранённом в стеке в процессе прерывания. Если этот флажок там сброшен, прерывание было вызвано инструкциейINT 1. Если же флажок был установлен, то имело место прерывание трассировки, что, однако, не исключает вероятности того, что одновременно была выполнена и инструкцияINT 1. Однозначная идентификация в такой ситуации невозможна по той же причине, что и для вектора 0.
Прерывание по вектору 2 может произойти либо при поступлении внешнего сигнала немаскируемого прерывания NMI, либо при выполнении двухбайтовой командыINT 2. Однозначная идентификация возможна только в том случае, если аппаратура вычислительной машины обеспечивает программно-доступную фиксацию сведений о возникшем немаскируемом прерывании подобно тому, как контроллер прерываний 8259 при соответствующем программировании фиксирует информацию о маскируемых запросах прерываний по линиямIRQ.
Прерывания по векторам 3 и 4 могут быть программно идентифицированы однозначно: они происходят либо из-за выполнения однобайтовых инструкций INT3иINTO(кодыCCиCEсоответственно), либо из-за двухбайтовых инструкцийINT 3иINT 4(кодыCD03 иCD04 соответственно).
Таким образом, в общем случае абсолютно точно идентифицировать источник прерывания по векторам 0–2 невозможно. Как уже упоминалось, фирма Intelзарезервировала векторы 0–31 для “внутренних нужд” микропроцессора; часть из них задействована в современных образцах процессоров. Поэтому настоятельно не рекомендуется использовать инструкцииINTс номерами 0–31, а также назначать этим векторам маскируемые аппаратные прерывания.