Assembler для начинающих


Возврат программы в DOS с сохранением ее резидентности



Возврат программы в DOS с сохранением ее резидентности


    Первый способ написания и загрузки постоянной функции в DOS состоит
    в том, чтобы, возвращая управление DOS, программа оставалась в
    памяти резидентной. Такую функцию существляет прерывание INT 27H.
 
      Обычно для выхода в DOS используется прерывание INT 20H, либо
    программа производит переход по адресу 0 программного префикса, как
    мы делали в программах типа .EXE. В результате управление
    возвращается DOS. Операционная система освобождает память,
    предоставленную этой программе. Следующую программу, которая
    загружается после прерывания INT 20H, DOS помещает в ту же область
    памяти, которая использовалась для предыдущей.
 


      Выход в DOS через прерывание INT 27H отличается от
    рассмотренного. Управление возвращается в DOS точно так же, как и в
    случае прерывания INT 20H, но часть памяти, занимаемая программой,
    не возвращается для дальнейшего использования. В регистре DX
    указывает на адрес первой свободной ячейки после той области
    памяти, котрую вы хотите зарезервировать. DOS резервирует эту
    область памяти, как часть системы. Это означает, что ваша программа
    становится частью DOS. Такую программу можно удалить из памяти
    только перезагрузив DOS и начав все сначала.
 
      Если выход в PC DOS осуществляется при помощи прерывания INT
    27H, то в регистре CS должен находиться адрес программного
    префикса. Легче всего это сделать, если писать использующую INT 21H
    программу как .COM программу. Написать программу типа .EXE,
    оставляющую при выходе содержимое регистров CS и DX корректным,
    довольно трудно. Поскольку создание программ типа .COM было
    рассмотрено в гл.5, будем считать, что все наши остающиеся
    резидентными программы имеют тип .COM.
 
      Рассматриваемый для прерывания DOS INT 27H пример довольно
    сложен. Он иллюстрирует не только использование INT 27H, но и
    способы замены существующей BIOS другой версией. В этом примере мы
    даже применим несколько трюков с таймером для увеличения скорости
    обработки.
 
      Пример представлен на Фиг. 10.1. Приведенная здесь программа
    предназначена для обслуживания буфера печати. Обычно при выдаче на
    печать символа программа обращается к прерыванию INT 17H - драйверу
    печати BIOS. Эта функция выдает символ на принтер после
    проверки ошибок и ожидания готовности принтера. Как правило, при
    этом обеспечивается достаточная производительность. Но допустим,
    что вы пишете несколько программ и хотите вывести их на принтер.
    Если вы попытаетесь сделать это, то не сможете обратиться к системе
    до тех пор, пока принтер не закончит работу. Чтобы
    продолжить редактирование или ассемблирование другой части
    программы, вам придется ждать завершения печати.
 
A
                 Microsoft (R) Macro Assembler Version 5.00              4/2/89 16:06:27
             Фиг. 10.1 Буфер для печати                          Page     1-1
 
 
                                      PAGE        ,132
                                           TITLE      Фиг. 10.1 Буфер для печати
              0000                        ABS0 SEGMENT AT 0
              0020                              ORG  4*8H
              0020      ????????          TIMER_INT  DD   ?    ; Аппратное прерывание от таймера
              005C                              ORG  4*17H
              005C      ????????          PRINTER_INT      DD   ?    ; Прерывание к BIOS для печати
              0408                              ORG  408H
              0408      ????              PRINTER_BASE     DW   ?    ; Базовый адрес адаптера принтера
              040A                        ABS0 ENDS
 
              0000                        CODE SEGMENT
              0100                              ORG  100H
                                           ASSUME  CS:CODE,DS:CODE,ES:CODE
              0100      EB 09 90                JMP  START
 
              0103      ????????          PRINT_VECTOR     DD   ?    ; Место для хранения исходного вектора 17h
              0107      ????????          TIMER_VECTOR     DD   ?    ; Место для хранения исходного вектора 9h
 
              010B                        START:
              010B      2B C0                   SUB  AX,AX            ; Установка регистра ES на сегмент ABS0
              010D      8E C0                   MOV  ES,AX
                                           ASSUME  ES:ABS0
              010F      26: A1 005C R                 MOV  AX,WORD PTR PRINTER_INT
              0113      26: 8B 1E 005E R        MOV  BX,WORD PTR PRINTER_INT+2
              0118      26: 8B 0E 0020 R        MOV  CX,WORD PTR TIMER_INT
              011D      26: 8B 16 0022 R        MOV  DX,WORD PTR TIMER_INT+2
              0122      A3 0103 R               MOV  WORD PTR PRINT_VECTOR,AX
              0125      89 1E 0105 R                  MOV  WORD PTR PRINT_VECTOR+2,BX
              0129      89 0E 0107 R                  MOV  WORD PTR TIMER_VECTOR,CX
              012D      89 16 0109 R                  MOV  WORD PTR TIMER_VECTOR+2,DX
 
                                     ;-----  Во время занесения векторов прерываний прерывания запрещены
 
              0131      FA                      CLI
 
                         Фиг. 10.1 Буфер печати (начало)
              0132      26: C7 06 005C R 0162         MOV  WORD PTR PRINTER_INT,offset PRINT_HANDLER
                   R
              0139      26: 8C 0E 005E R        MOV  WORD PTR PRINTER_INT+2,CS
              013E      26: C7 06 0020 R 0196         MOV  WORD PTR TIMER_INT,offset TIMER_HANDLER
                   R
              0145      26: 8C 0E 0022 R        MOV  WORD PTR TIMER_INT+2,CS
              014A      B0 36                   MOV  AL,00110110b
              014C      E6 43                   OUT  43H,AL
              014E      B0 00                   MOV  AL,0 ; Увеличение скорости работы таймера в 256 раз
              0150      E6 40                   OUT  40H,AL
              0152      B0 01                   MOV  AL,1
              0154      E6 40                   OUT  40H,AL
              0156      FB                      STI
              0157      8D 16 28FE R                  LEA  DX,BUFFER_END    ; Занесение адреса конца программы
              015B      CD 27                   INT  27H        ; Выход с сохранением программы в памяти
 
              015D      00                TIMER_COUNT      DB   0
              015E      01EE R                  BUFFER_HEAD      DW   BUFFER_START
              0160      01EE R                  BUFFER_TAIL      DW   BUFFER_START
 
                                     ;-----  Эта подпрограмма управляет вызовом прерывания 17h
 
              0162                        PRINT_HANDLER    PROC FAR
                                           ASSUME  CS:CODE,DS:nothing,ES:nothing
              0162      0A E4                   OR   AH,AH
              0164      74 05                   JZ   BUFFER_CHARACTER ; Проверка на функцию вывода символа
              0166      2E: FF 2E 0103 R        JMP  PRINT_VECTOR           ; Переход на стандартный обработчик
                                                                   ;  прерывания 17h
              016B                        BUFFER_CHARACTER:
              016B      FB                      STI
              016C      53                      PUSH BX
              016D      51                      PUSH CX
              016E      56                      PUSH SI
              016F      2B C9                   SUB  CX,CX            ; Счетчик отсчетов таймера
              0171                        PRINT_LOOP:
              0171      2E: 8B 1E 0160 R        MOV  BX,BUFFER_TAIL ; Выборка адреса конца буфера
              0176      8B F3                   MOV  SI,BX
              0178      E8 01E2 R               CALL ADVANCE_POINTER ; Перемещение указателя на следующий байт
              017B      2E: 3B 1E 015E R        CMP  BX,BUFFER_HEAD ; Проверка на наличие места в буфере
              0180      74 0E                   JE   BUFFER_FULL      ; Нет места,ожидается пока оно появится
              0182      2E: 88 04               MOV  CS:[SI],AL ; Вывод символа в буфер
              0185      2E: 89 1E 0160 R        MOV  BUFFER_TAIL,BX ; Занесение нового адреса конца буфера
              018A      B4 00                   MOV  AH,0       ; Код возврата из прерывания 17h
              018C                        PRINT_RETURN:
              018C      5E                      POP  SI
              018D      59                      POP  CX
              018E      5B                      POP  BX
              018F      CF                      IRET
              0190                        BUFFER_FULL:
              0190      E2 DF                   LOOP PRINT_LOOP ; Повторить цикл проверки занятости буфера
              0192      B4 01                   MOV  AH,1       ; Буфер занят слишком долго,ошибка
              0194      EB F6                   JMP  PRINT_RETURN
              0196                        PRINT_HANDLER    ENDP
 
                         Фиг. 10.1 Буфер печати (продолжение)
                                     ;-----  Эта программа вызывает 4660 раз в секунду
 
              0196                        TIMER_HANDLER    PROC FAR
                                           ASSUME  CS:CODE,DS:nothing,ES:nothing
              0196      50                      PUSH AX
              0197      53                      PUSH BX
              0198      2E: 8B 1E 015E R        MOV  BX,BUFFER_HEAD
              019D      2E: 3B 1E 0160 R        CMP  BX,BUFFER_TAIL ; Есть ли что-нибудь в буфере?
              01A2      75 14                   JNZ  TEST_READY ; Переход,если буфер не пуст
 
                                     ;-----  Эта подпрограмма управляет таймером в скоростном режиме
 
              01A4                        TIMER_RETURN:
              01A4      5B                      POP  BX
              01A5      2E: FE 06 015D R        INC  TIMER_COUNT      ; Увеличение счетчика делителя таймера
              01AA      75 06                   JNZ  SKIP_NORMAL
              01AC      58                      POP  AX         ; Это выполняется один раз на 256 прерываний
              01AD      2E: FF 2E 0107 R        JMP  TIMER_VECTOR     ; Переход на стандартную программу обработки
                                                             ;  прерывания от таймера
              01B2                        SKIP_NORMAL:
              01B2      B0 20                   MOV  AL,20H
              01B4      E6 20                   OUT  20H,AL     ; Конец прерывания
              01B6      58                      POP  AX
              01B7      CF                      IRET
 
                                     ;-----  Символ в буфере,производится попытка напечатать его
 
              01B8                        TEST_READY:
              01B8      52                      PUSH DX
              01B9      1E                      PUSH DS
              01BA      2B D2                   SUB  DX,DX
              01BC      8E DA                   MOV  DS,DX            ; Установка регистра DS на сегмент ABS0
                                           ASSUME  DS:ABS0
              01BE      8B 16 0408 R                  MOV  DX,PRINTER_BASE
              01C2      42                      INC  DX         ; Установка на порт состояния
              01C3      EC                      IN   AL,DX
              01C4      A8 80                   TEST AL,80H     ; Проверка готовности принтера
              01C6      74 16                   JZ   NO_PRINT
              01C8      4A                      DEC  DX         ; Установка на порт данных
              01C9      2E: 8A 07               MOV  AL,CS:[BX] ; Выбрка выводимого символа
              01CC      E8 01E2 R               CALL ADVANCE_POINTER
              01CF      2E: 89 1E 015E R        MOV  BUFFER_HEAD,BX
              01D4      EE                      OUT  DX,AL            ; Вывод символа в порт принтера
              01D5      83 C2 02                ADD  DX,2       ; Установка на порт управления
              01D8      B0 0D                   MOV  AL,0DH
              01DA      EE                      OUT  DX,AL            ; Передача символа из порта в принтер
              01DB      B0 0C                   MOV  AL,0CH
              01DD      EE                      OUT  DX,AL
              01DE                        NO_PRINT:
              01DE      1F                      POP  DS
              01DF      5A                      POP  DX
              01E0      EB C2                   JMP  TIMER_RETURN     ; Возврат через подпрограмму управления
              01E2                        TIMER_HANDLER    ENDP       ;  таймером
 
              01E2                        ADVANCE_POINTER PROC   NEAR
              01E2      43                      INC  BX         ; Сдвиг указателя
 
                         Фиг. 10.1 Буфер печати (продолжение)
              01E3      81 FB 28FE R                  CMP  BX,offset BUFFER_END
              01E7      75 04                   JNE  ADVANCE_RETURN  ; Проверка на конец циклического буфера
              01E9      8D 1E 01EE R                  LEA  BX,BUFFER_START ; Установка указателя на начало буфера
              01ED                        ADVANCE_RETURN:
              01ED      C3                      RET
              01EE                        ADVANCE_POINTER ENDP
 
              01EE                        BUFFER_START     LABEL      BYTE
              01EE      2710[                   DB   10000 DUP (?)
                       ??
                                ]
 
              28FE                        BUFFER_END LABEL      BYTE
              28FE                        CODE ENDS
 
                                           END
            Фиг. 10.1 Буфер печати (продолжение)
 
      Приведенная в примере программа может облегчить решение задачи.
    Конечно, это не обойдется вам даром. Программа отводит под буфер
    печати некоторую область памяти, котрая будет постоянно за ним
    закреплена. DOS изымает эту область из общего объема памяти,
    предоставляемой пользователю. Например, если в системе 96K байт
    памяти, а 10 кбайт отводится под буфер печати, то пользоваться
    Макроассемблером уже не удастся. Для макроассемблера требуется 96
    кбайт, а после создания буфера печати останется лишь 86 кбайт.
    Поэтому, прежде чем организовать буферизацию печати, убедитесь, что
    в системе останется еще достаточный объем памяти.
 
      Буферизация печати осуществляется примерно так. Стандартная
    команда PRINT (INT 17H) заменяется процедурой, которая помещает
    символы в буфер вместо того, чтобы посылать их на принтер. Эта
    часть программы и называется буферизацией печати. Отдельная часть
    программы, называемая выводом на печать, извлекает символы из
    буфера печати и пересылает их на принтер.
 
      Основным моментом в данном примере является замена прерывания
    INT 17H базовой системы ввода-вывода. Почти все прикладные
    программы для вывода на печать используют именно это прерывание, а
    это означает, что теперь все обычные операции печати будут
    приводить к пересылке символов в подпрограмму буферизации печати, а
    не на принтер. В частности, в нашем примере, мы можем
    листинг ассемблирования вывести на принтер, нажав клавиши
    Ctrl-PrtSc, служащие для пересылки символов с экрана на печать.
 
      Когда мы выводим листинг ассемблирования с программой
    буферизации печати в памяти, символы поступают в буфер в памяти, а
    не на принтер. Буферизация очень незначительно
    увеличивает время просмотра. Когда файл выведен на экран (и в буфер
    печати), управление возвращается DOS. Вы можете прекратить
    пересылку символов на принтер, снова нажав клавиши Ctrl-PrtSc.
    Листинговый файл находится в буфере, и DOS готова продолжить
    выполнение других заданий, например, редактирование или
    ассемблирование.
      Затем начинает выполняться вторая часть программы. Эта
    процедура извлекает символы из буфера и пересылает их на принтер.
    Она управляется прерыванием от таймера. При каждом прерывании от
    таймера процедура вывода на печать также получает управление. Если
    в буфере имеется символ, и если устройство печати находится в
    состоянии "готово", то подпрограмма пересылает этот символ на
    принтер. Таким образом, символы извлекаются из буфера и
    пересылаются на принтер со скоростью работы этого устройства.
    Поскольку программа вывода на печать работает в фоновом режиме,
    одновременно могут выполняться другие задания, например,
    редактирование или ассемблирование.
 
      Обратимся к программе, представленной на Фиг. 10.1, и
    рассмотрим, как взаимодействуют ее компоненты. Во-первых, в ней
    описан сегмент ABS0, содержащий вектор прерываний, с которым
    программа имеет дело. Приведенная в примере программа заменяет как
    прерывание вывода на печать INT 17H, так и прерывание от таймера
    INT 8. Заметим также, что в сегменте ABS0 определяется адрес
    PRINTER_BASE. В этой ячейке находится базовый адрес для устройства
    печати 0. В данном примере предполагается, что все операции печати
    производятся на системном устройстве печати.
 
      Сегмент CODE - это та секция программы, которая остается
    резидентной. При помощи команды ORG 100H мы составили эту программу
    как файл типа .COM. Это означает, что для создания из выходного
    файла редактора связей файла типа .COM, необходимо выполнить
    описанную в гл.5 последовательность действий. Для хранения исходных
    значений вектора печати и вектора таймера в программе используются
    области памяти PRINT_VECTOR и TIMER_VECTOR. Хотя программа заменяет
    значения этих векторов, при выводе на печать в ней должны быть
    известны их исходные значения.
 
      Первая часть сегмента CODE, начиная с метки START, является
    инициализирующей частью программы. В ней считываются исходные
    значения векторов прерываний и сохраняются в области данных
    сегмента CODE. В процедуре инициализации векторы прерываний в
    нижних адресах памяти заменяются новыми, используемыми в процедурах
    буферизации и вывода на печать. Обратите внимание на команду CLI,
    которая блокирует прерывания перед выполнением этой операции.
    Поскольку программа изменяет прерывание таймера, она не может
    допустить обработку его шага в этот момент времени. Если бы
    прерывание от таймера произошло в тот момент, когда программа
    изменила только одно из двух слов вектора прерываний от таймера, то
    микропроцессор продолжил бы выполнение с непредсказуемого адреса
    памяти. Разумнее запретить прерывания, чем допустить возможность
    перехода по неизвестному адресу.
 
      Прежде чем разблокировать прерывания, программа изменяет
    текущее значение счетчика таймера. Обычно прерывания от таймера
    происходят примерно 18 раз в секунду. Устройство печати может
    печатать по 80 символов в секунду. Если бы процедура вывода на
    печать выдавала по одному символу при каждом прерывании от таймера,
    то максимальная скорость печати составила бы 18 символов в секунду.
    Если ускорить таймер, прерывания от таймера будут происходить чаще.
    Это позволит программе выдавать на печать все 80 символов в
    секунду. В приведенном примере в таймер загружается значение
    счетчика 256, оно в 256 раз меньше стандартного значения.
    Компенсируется это увеличение скорости при помощи процедуры
    TIMER_HANDLER.
 
      Процедура инициализации возвращает управление в DOS при помощи
    прерывания INT 27H. Перед выходом из процедуры в регистр DX
    загружается указатель на байт, сразу следующий за последнм байтом
    всей программы. Заметим, что все процедуры и буфер печати мы
    расположили в пределах этой области памяти. В соответствии с
    правилами действия прерывания INT 27H DOS не затронет эту
    область.
 
      Приведенная программа зря расходует часть памяти.
    Инициализирующая ее часть выполняется только один раз, поэтому нет
    смысла оставлять ее в памяти. Можно оптимизировать программу
    поместив часть кода от команды START до INT 27H после метки
    BUFFER_END. В этом случае при прерывании INT 27H инициализирующая
    часть программы оказалась бы за пределами защищаемой области
    памяти, и следующая загружаемая DOS программа перекрыла бы
    процедуру инициализации. Экономия около 90 байт из более чем 10000
    байт в нашем примере не впечетляет, но она вполне доступна в случае
    необходимости.
 
      Далее следует процедура PRINT_HANDLER. Эта подпрограмма
    вместо базовой системы ввода-вывода осуществляет управление
    принтером при каждом обращении программ к прерыванию INT 17H для
    вывода данных на печать. Первые три команды управляют перехватом
    управления у BIOS. Наша процедура работает только тогда, когда
    должен быть напечатан символ (AH = 0). При любом другом коде
    функции работу выполняет BIOS, поэтому программа производит
    проверку, не равен ли регистр AH нулю. Если нет, то производится
    косвенный переход с использованием сохраненного значения исходного
    вектора печати. В результате управление передается процедуре
    входящей в BIOS, которая выполняет требуемую функцию. Сказанное
    означает, что в нашей процедуре обработки прерывания достаточно
    написать только поддержку сделанных изменений.
 
      Относительно рассмотренного способа управления печатью следует
    сделать два замечания. Во-первых, передача дальше всех функций
    печати кроме случая AH = 0 - не блестящая идея. Если какая-либо
    программа инициализирует принтер (AH = 2) во время работы механизма
    буферизации, то BIOS берет управление на себя и выдает на принтер
    команду RESET. Эта команда обрывает ту строку, которая в это время
    выводится на печать, что в большинстве случаев приводит к потере
    одного или нескольких символов. Если вы хотите сделать эту
    программу более защищенной от ошибок, то вам придется рассмотреть
    вопрос об управлении всеми функциями печати.
 
      Второе, на что следует обратить внимание - это использование
    сохраненного вектора прерываний печати. Можно было бы обратиться к
    листингу BIOS, приведенному в техническом справочнике, и найти
    начальный адрес процедуры печати. Затем включить этот адрес
    непосредственно в код программы так же, как это делается для других
    абсолютных адресов. Однако в результате программа оказалась бы
    жестко к этому адресу в системе BIOS. Если фирма IBM изменит
    процедуры BIOS и, таким образом, - адрес процедуры печати, то
    рассмотренная программа не сможет больше работать. Конечно, если
    пишите эту программу для своей собственной машины, а покупать новую
    или продавать свою программу не собираетесь, то указанных проблем
    не возникнет. Однако в общем случае надо избегать использования
    абсолютных адресов, если есть выбор. В приведенном примере
    процедура инициализации легко может использовать вектор
    прерываний печати для определения адреса процедуры печати
    BIOS в ПЗУ.
 
      В оставшейся части процедуры PRINT_HANDLER символ помещается в
    буфер печати. Перед тем, как поместить символ программа проверяет,
    есть ли в буфере место. Если буфер полон, программа ждет, пока
    освободится место. Это ожидание не вызовет проблем, поскольку и
    стандартная процедура BIOS ждет, чтобы принтер был готов принять
    символ. Из соображений безопасности в регистре CX накапливается
    число проходов по ветви "занято". Если это число становится равным
    64K, а буфер по-прежнему полон, то это может означать какой-то
    сбой. В этом случае процедура PRINT_HANDLER так же, как и BIOS,
    выдает сообщение о превышении допустимого времени ожидания.
 
      В приведенном примере процедура печати использует также
    внутреннюю процедуру ADVANCE_POINTER. Эта несложная процедура
    делает буфер печати циклическим. Если указатель сдвигается за
    пределы буфера, подпрограмма переносит его на начало буфера. Она
    аналогична процедуре BIOS для буфера клавиатуры. Только в данном
    случае в буфер помещается 10000 символов, а не 16.
 
      Интересно рассмотреть работу процедуры TIMER_HANDLER из
    приведенного примера. Инициализирующая процедура связывает эту
    подпрограмму с аппаратным прерыванием от таймера, поэтому на каждом
    цикле таймера она получает управление. Помимо пересылки кодов на
    принтер, эта процедура должна обеспечивать, через компенсацию
    ускоения таймера, выполнение его обычных функций, таких как
    ведение времени дня.
 
      Сначала процедура работы с таймером проверяет, имеются ли
    предназначенные для вывода на печать символы. Нет смысла пытаться
    переслать символы на принтер, если пересылать нечего. Если в буфере
    нет символов, процедура проходит на метку TIMER_RETURN. Этот
    фрагмент процедуры обслуживает ускорение таймера.
 
      Метка TIMER_RETURN указывает часть программы, обеспечивающую
    нормальное функционирование таймера. При каждом прерывании от
    таймера значение байта TIMER_COUNT увеличивается на единицу. Если
    этот байт не нулевой, то процедура выходит из прерывания после
    выдачи сигнала о завершении прерывания на контроллер прерываний.
    Если этот байт равен нулю, то выход из программы осуществляется
    посредством косвенного перехода по сохраненному вектору прерывания
    от таймера TIMER_VECTOR. При этом управление передается процедуре
    BIOS для определения текущего времени и выключения дисковода.
    Дублировать эти операции в нашей программе не требуется. Переход в
    BIOS происходит только один из 256 раз выполнения подпрограммы
    работы с таймером. Но поскольку скорость таймера была увеличена в
    256 раз, процедура реакции на прерывание от таймера базовой системы
    ввода-вывода по-прежнему будет получать управление 18,2 раза в
    секунду. Это означает, что текущее время будет поддерживаться
    правильно, и мотор дисковода будет выключен вовремя. Именно поэтому
    и было выбрано ускорение таймера в 256 раз, хотя и ускорения в 5
    раз было бы достаточно, чтобы обеспечить работу устройства печати с
    максимальной скоростью.
 
      Ускорение таймера в 256 раз было выбрано потому, что это было
    просто сделать. Однако если брать в расчет производительность, то
    лучше было бы ускорить работу таймера в 5 раз, поскольку на
    обработку каждого прерывания от таймера тратится по меньшей мере 10
    микросекунд, и даже больше, если в буфере печати есть символы.
    Время, затраченное на обработку прерываний, идет в ущерб выполнению
    системой других заданий, например ассемблирования. При такой
    частоте прерываний от таймера, становится заметным замедление
    работы. Для оптимизации производительности следует ускорять таймер
    менее, чем в 256 раз.
 
      Что же происходит в процедуре работы с таймером, когда в буфере
    есть символы, предназначенные для печати? Программа считывает порт
    состояния, чтобы определить, готов ли принтер к приему символа.
    Поскольку в процедуре используется базовый адрес из области данных
    BIOS, то наша подпрограмма будет работать и с автономным адаптером
    устройства печати, и с портом адаптера монохромного дисплея. Если
    устройство печати не готово, процедура возвращает управление на
    метку TIMER_RETURN, где в случае необходимости поддерживаются
    стандартные функции таймера. Процедура вывода на печать не ждет,
    когда устройство печати освободится, если оно занято. Мы знаем, что
    прерывание от таймера очень скоро повторится, тогда мы и повторим
    попытку вывода. Ожидание готовности устройства печати здесь
    связывало бы бы всю систему. Результат был бы таким же, как и в
    случае отсутствия буферизации печати.
 
      Если принтер готов, программа извлекает символ из буфера и
    передает его на принтер. И в данном случае программа вновь не
    делает всего, что следовало бы. Подпрограмма, входящая в BIOS,
    делает проверку на ситуацию ошибки при передаче каждого символа. То
    же самое следовало бы делать и в нашей процедуре. Но что же
    произойдет в случае сбоя? Если процедура вывода обнаружила ошибку,
    то как она сможет сообщить программе, что это произошло во время
    печати? В некоторых случаях к этому моменту программа передававшая
    даные для печати уже завершила свою работу. Наилучший выход может
    состоять в проверке ошибок при каждой пересылке символа на принтер
    процедурой работы с таймером. При обнаружении ошибки процедура
    PRINT_HANDLER должна выдать сообщение об ошибке, что далее все
    программы будут производить вывод на печать через прерывание INT
    17H. Возможно, это не идеальный вариант, но, вероятно, лучший.
 
      Прежде чем закончить рассмотрение примера, следует обратить
    внимание еще на одну проблему. Существуют и другие процедуры,
    изменяющие частоту прерываний от таймера. BASICA - расширенная
    версия интерпретатора Бейсика, для ускорения таймера используется
    прием, во многом аналогичный приведенному. При вызове программы
    BASICA после установки буферизованной печати, процедура
    TIMER_HANDLER получает прерывания уже не с той частотой, которая
    предполагается. Поскольку процедура TIMER_HANDLER ограничивает
    передачу управления прерыванием от таймера процедуре BIOS, текущее
    время замедлится в 256 раз. BASICA осуществляет также инициализацию
    устройства печати, что, как мы уже видели, мешает выводу на печать.
    Это означает, что программа буферизации печати будет работать не
    для всех приложений. Однако она иллюстрирует использование
    прерывания INT 27H для создания постоянной системной функции.
    Приведенный пример иллюстрирует также метод переопределения
    векторов BIOS для подцепления новой функции к уже имеющимся
    программам.




Содержание раздела