Скачайте в формате документа WORD

Программирование - интерфейс RS-232

ПРОГРАММИРОВАНИЕ ПОСЛЕДОВАТЕЛЬНОГО ПОРТА


Последовательный порт в системе MS-DOS обеспечиваета входа во

внешний мир. Основной задачей последовательного порта является

направление и получение данных по шине в видеа поток битов. (В

противоположность параллельному порту, в котором внутренний байт

передается целиком). Вы можете использовать последовательный порт

для подключения к системе "мыши", направления данных на принтер

или для установления автоматической телефонной связи c использо-

ванием модема. Хотя системы MS-DOS не нуждаются для работы в пос-

ледовательном порте, эти порты стали стандартной периферией сис-

темы.

Последовательный порт в системах MS-DOS способен поддерживать

стандарт асинхронной передача данных RS-232C. Хотя даже посредс-

твом ROM-BIOS, стандартной части всех систем MS-DOS, MS -DOS сама

по себеа включаета некоторую поддержкуа программирования портов

RS-232C (например, прерывание номер 14h), эта поддержка не отве-

чает требованиям высокоскоростной связи. Если Вы хотите включить

в свою прикладную программуа эффективные возможностиа последова-

тельной связи, Вы должны осуществлять доступ к последовательному

порту на аппаратном ровне. В этой главе показано, как это дела-

ется.


Основы асинхронной последовательной связи


Говоря о передаче данных, мы интересуемся передачейа байтов

данных от одного устройства к другому, например, от персонального

компьютера к модему или к последовательномуа принтеру. Если мы

имеема восемь линий между двумя стройствами, то мы можем назна-

чить каждой линии бит и послать сразу один байт данных. Это будет

параллельная передача. Таким образом работает параллельный порт

персонального компьютера, кроме того, в дополнение к восьми лини-

яма данных имеются другие сигнальные линии, оказывающие помощь в

передаче данных.

С другой стороны, если мы имеем одну линию для передачи сиг-

налов, то необходимо посылать каждый байт данных последовательно,

по одному биту. Более того, мы может посылать данные синхронно,

таким образом, что каждый байт посылается ва ранееа определенное

время (скажем, один байт каждые х секунд), или асинхронно со ско-

ростью, которую предварительно определять необязательно.

Последовательная связь дешевле, чема параллельная, так как

требует меньше линий передачи данных - минимум две для двусторон-

ней связи. Кроме того режим асинхронной передачи оказывает значи-

тельно меньшее воздействие на аппаратуру ввиду того, что не тре-

буется дополнительное специальноеа оборудование для поддержки

синхронизации между передатчиком и приемником.

Таким образом, асинхронная последовательная связь является

предпочтительным решением ввиду низкой стоимости иа простоты ис-

пользуемых аппаратных средств. Конечно, в этом режиме передачи мы

должны преобразовывать каждый байт данных в серию битов и казы-

вать приемнику начало и конец каждого байта. На рисунке 8-1 про-

иллюстрирована концепция асинхронной последовательной связи.

Предположим, что мы меем преобразовывать каждый байт в поток

единиц и нулей, то есть биты, которые могут быть переданы через

среду связи (например, телефонную линию). В самом деле, нивер-

сальный асинхронный приемопередатчик (UART), как мы видим в сле-

дующем разделе, выполняет точно такую же функцию. Обычно, в то

время как линия находится в режиме ожидания, для демонстрации то-

го, что линия в порядке, по ней передается единица, обозначая не-

занятость линии. С другой стороны, когда линия находится в состо-



┌ ─ ─ ─ ─ ─ ─ ─ ─ ─ ─ ─ ─ ─ ─ ─ ─ ─ ─ ─ ─ ─ ─ ─ ─ ─ ─ ─ ─а ┐

Восстанавливается один байт

│ └а 7а 6а 5а 4а 3а 2а 2а 1 ┘└──┐

┌───Принимается один символ────┐└─┘└─┘└─┘└─┘└─┘└─┘└─┘└─┘ ┌─┐|

|A||B||7||6||5||4||3||1||0||C|а ^а ^а ^а ^а ^а ^а ^а ^ ||

--> └─┘└─┘└─┘└─┘└─┘└─┘└─┘└─┘└─┘└─┘ ┌─────┬─────────┬─────┐а |─┘└┐

┌──────────────────────────────────|а Dа | | |а | |

|│ от модем а└─────┴─────────┴─────┘а |───|

|а UART на приемном конце скорость в бодах | |

|└ ─ ─ ─ ─ ─ ─ ─ ─ ─ ─ ─ ─ ─ ─ ─ ─ ─ ─ ─ ─ ─ ─ ─ ─ ─ ─ ─ ─ ─| |

| |───|

|а ┌───────┐ | |

└──| |──┐ |───|

└───────┘а │ Принимающий

│ Телефонная линия компьютер

┌┼┐┌─┐

└─┘ <-- └┼┘

│ ┌───────┐

└───────| модем |───|

┌────────┐ └───────┘ |

│ │ |

а PC │ |

┌──┴────────┴──┐ |

│ │ |

└┐┌────────────┘ v

||а ┌ ─ ─ ─ ─ ─ ─ ─ ─ ─ ─ ─ ─ ─ ─ ─ ─ ─ ─ ─ ─ ─ ─ ─ ─ ─ ─ ──┐

||а │ ┌──────── Байт───────────┐

|| 7а 6а 5а 4а 3а 2а 2а 1 |

|└──┘а └─┘└─┘└─┘└─┘└─┘└─┘└─┘└─┘┌───Посылаемый один символ──┐

└───┐ |A||B||7||6||5||4||3||1||0||C|

┌─────┬─────────┬─────┐ └─┘└─┘└─┘└─┘└─┘└─┘└─┘└─┘└─┘└─┘

|а | | |а Dа |──────────────────────────────┘

└─────┴─────────┴─────┘ к модему --> |

|а скорость в бодах UART в последовательном адаптере PC

└ ─ ─ ─ ─ ─ ─ ─ ─ ─── ─ ─ ─ ─ ─ ─ ─ ─ ─ ─ ─ ─ ─ ─ ─ ─ ─ ┘

A-стоп; B-четность; C-начало; D-сдвиговый регистр

янии логического нуля, говорится, что он стоита ва режиме

выдерживания интервалов. Таким образом, логические единица и ноль

рассматриваются, соответственно, как MARK и SPACE.


Рис.8-1. Асинхронная последовательная связь.


В асинхронной связи изменение словия состояния линии с MARK

на SPACE означает начало символа (смотри рисунок 8-2). Это назы-

вается стартовым битом. За стартовым битом следует комбинация би-

тов, представляющая символ, и затем бит контроля четности. Нако-

нец, линия переходит ва состояниеа ожидания MARK, которая

представляет собой стоповый бит и означает конец текущего симво-

ла. Число битов, используемых для представления символа, называ-

ется длиной слова и обычно бывает равно семи или восьми. Конт-

рольный бит используется для выполнения элементарной проверки на

наличие ошибки.


<--------- Направление передачи

Линия Линия возвращается в

свободн свободное состояние

A

B ────┐а ┌ ─ ┬ ─ ┬ ─ ┬ ─ ┬ ─ ┬ ─ ┬ ─ ┬ ─ ┬ ─ ┬───┬───────┐а ┌ ─ ─

│ 0 1 2 3 4 5 6 7 │ │

C ─ ─ └──┘ ─ ┴ ─ ┴ ─ ┴ ─ ┴ ─ ┴ ─ ┴ ─ ┴ ─ ┴ ─ ┴ ─ ┴ ─ ─ ─ └──┘ ─ ─

^ └──────── 7 или 8 бит данных ───┘ ^ ^ ^

Стартовый бит четности Начало другого

бит │ символа

бит стоповый

Время ----->


A-длительность 1 бита; B-MARK или 1; C-SPACE или 0


Рис.8-2. Представление в асинхронной последовательной

связи формата одиночного символа


Как передатчика (илиа приемник) знают о длительности каждого

бита?а Действительно, и передатчик, и приемник должны знать его

длительность или детектирование битов будет невозможно. Длитель-

ность каждого бита определяется генераторамиа тактовыха импульсов

приемника и передатчика. Отметим, однако, что генераторы в прием-

нике и передатчике должны иметь одну и ту же частоту, но не тре-

буется, чтобы они были синхронизированы. Выбор частоты генератора

зависит от скорости передачи в бодах, которая означает число из-

менений состояния линии каждую секунду. Номинально тактовая час-

тота "16-кратная скорость передачи в бодах" означает, что линия

проверяется достаточно часто для надежного распознавания старто-

вого бита.

Существует одно обычное состояние линии, которое иногда ис-

пользуется для привлечения внимания приемника. Нормальным состоя-

ниема линииа является MARK (или 1) и начало символа определяется

переходом SPACE (0). Если линия находится в состоянии SPACE в те-

чение периода времени большем, чем время, которое она затратила

бы на получение всех битов символа, тогда мы говорим, что насту-

пило состояниеа BREAK. Ва кодах ASCII отсутствует представление

BREAK - это означает, что линия "умерла"а н непродолжительный

промежуток времени, который составляет BREAK.


Контроль по четности и обнаружение ошибок


Ранее мы поминали, что бит контроля четности полезен для об-

наружения ошибок. Например, если выбрана проверка на четность,

этот бит устанавливается таким образом, что общее число единиц в

текущема словеа является четным (такая же логика используется для

проверки на нечетность). В приемнике четность вычисляется заново

и сравнивается с битом контроля четности. Если они не равны, то

приемник сообщает, что имеет место ошибка четности. Главный не-

достаток обнаружения ошибки посредством проверки на четность зак-

лючается в том, что можно только обнаружить ошибки, которые вли-

яют на один единственный бит. Например, битовая комбинация 0100

1 0 (ASCII A), переданная восемью битами с проверкой н чет-

ность, может измениться (скажем,из-за шума в линии) на 0100 00

(ASCII G), однако приемника не обнаружит ошибку, так как провер-

ка на четность выполняется.


Связь с использованием стандарта RS-232C.


Ранее мы упоминали о передаче по телефонной линии единиц и

нулей. Несмотря на то, что в персональном компьютере мы пред-

ставляем единицы и нули посредством ровней напряжения, сигналы,

передаваемые по телефонной линии обычно являются тонами различной

частоты. стройство, которое находится между аппаратурой персо-

нального компьютера и передающей линией и делает возможной пере-

дачу данных, называется модемом (модулятор/демодулятор). Модем

может преобразовывать информацию ва представление "напряжение/нет

напряжения" цифровых схем и обратно, так же аналоговые сигналы

(например, тоны), предназначенные для передачи по телефонной ли-

нии. Стандарты, такие как RS-232C (выдвинутый Ассоциацией элект-

ронной промышленности, EIA), описывают метод обмена информацией

междуа модемома (или, ва соответствии с терминологией ассоциации

EIA, "аппаратура передачи данных, DCE") иа связной аппаратурой

персонального компьютер (или "оконечная аппаратура обработки

данных, DTE"). Модем может работать в двух режимах: полудуплекс-

нома иа дуплексном. В полудуплексном режиме модем может осущест-

влять передачу только в одном направлении в один промежутока вре-

мени, в то время как при работе ва дуплекснома режиме

осуществляется независимая двухсторонняя связь. Стандарт RS-232C

обеспечивает управление такими сигналами, как "запрос-на-передачу

(RTS)" и "открыт-для -передачи (CTS)", которые могут быть исполь-

зованы для координации процесса передачи и приема данных. Термин

"квитирование установления связи" используется для описания коор-

динации приема и передачи сигналов. Как показано на рисунке 8-3,

стандарт RS-232C соответствует кабелю и соединителям, используе-

мым для связи персонального компьютера и модема.

Несмотря на то, что мы использовали модем в качестве примера

ппаратуры передачи данных (DCE), другие устройства, такие как

"мышь" или принтер с соответствующей схемой, также могут обмени-

ваться данными с персональным компьютерома через последовательный

порт. Таким образом, в этой главе все поминания о модеме прием-

лемы в равной степени как к последовательному принтеруа така и к

последовательной "мыши".



┌ ─ ─ ─ ─ ─ ─ ─ ─ ─ ─ ─ ─ ─ ─ ─ ─а ─ ─ ─ ─ ─ ┐

Д ┌───── Номер контакта ──────┐ DCE

|а ┌─────┐ ┌─────┐а |

| 2|─────────────ПД───────────| 2 |

|а | 3|──────────────П───────────| 3 |а |

| 4|─────────────ЗП───────────| 4 |

|а | 5|────────────ОДП───────────| 5 |а |

| 6|────────────ГПД───────────| 6 | к

|а | ┌──7|───────────Общий──────────| 7──┐|а телефонной

| 8|───────────ДСПЛ───────────| 8 | линии

|а | 20|────────────ТДГ───────────|20 |а | ^

| 22|─────────────RI───────────|22 | |

|а |.| |. |а |────┐ |

|.| |. | | |

┌─|а | .| | . |а | | |

RS ---> | └─────┘ └─────┘ | |

| └ ─ ─ ─ ─ ─ ─ ─ ─ ─ ─ ─ ─ ─ ─ ─ ─а ─ ─ ─ ─ ─ ┘ | |

┌───┴────┐ | |

│ │ |--| |

│ │ | ---|

┌──┴────────┴──┐ ┌─┴───┴─┐

│ │ | |

└──────────────┘ └───────┘

PC или Д Модем ил


Д-оконечная аппаратура обработки данных;RS-кабельRS-232C;

ПД - передача данных; П-прием данных;ЗП -запрос на передачу;

ОДП - открыт для передачи; ГПД - готов к передаче данных;

ДСПЛ - детектор сигнала приемной линии; ТДГ -терминал данных

готов;а АПД - или аппаратура передачи данных


Рис.8-3. Связь RS-232C.


правление потоком с помощью XON/XOFF


В дополнение к квитированию становления связи посредством

ппаратных сигналов RTS/CTS, для достижения управления потоком с

использованием программного обеспечения применяются специальные

управляющие символы ASCIIа (Control-Q/Control-S или XON/ XOFF).

Управлять потоком необходимо ввиду того, что иногда либо передат-

чика либо приемник не могут поддерживать скорость передачи и они

должны иметь возможность информировать другую сторону о необходи-

мости остановкиа на время, требуемое для того, чтобы отставшая

сторона смогла догнать другую.

Предположим, что приемник имеет буфер для хранения поступаю-

щих символов. Как только буфер после заполнения закрывается, при-

емник может послать символ XOFFа передатчику, сигнализируя, что

передача должна быть приостановлена. Конечно, приемник должен по-

нять значение XOFF и прекратить передачу символов. Затем, когда

приемник обработает символы (скажем, запишет их в файл на диске)

и буфер освободится, тогда посылается символ XON, показывающий,

что передача может быть продолжена. Эта схема управления потоком

широко применяется ввиду ее простоты. Большинство связныха прог-

рамм предоставляют возможность дуплексной связи с правлением по-

током, основанном на применении символов XON/XOFF.


Последовательный порт с точки зрения программиста


Аппаратура последовательного порта в системах MS-DOS известна

как последовательный адаптер или асинхронный связной адаптер (да-

лее мы будем называть его последовательным адаптером). Адаптер

основан на Intel 8259 UART (универсальный асинхронный приемопере-

датчик), имеет порт RS-232C для подключения к модему и, как адап-

тер дисплея, программируется посредством набора регистров. Мик-

ропроцессора имеета доступа к регистрам через ранее определенные

дреса порта ввода/вывода.

ниверсальный асинхронный приемопередатчик Intel 8250 прав-

ляется посредством записи в набор восьмибитовых регистров и чте-

ния из них. Эти регистры доступны программисту через адреса пор-

та. Адреса портов задаются последовательно, поэтомуа достаточно

знать адреса первого порта. Он также известен как базовый адрес

последовательного адаптера. В персональном компьютере IBM PC двум

последовательным портам COM1 и COM2 присвоены базовые адреса пор-

та 3F8h и 2F8h соответственно. Так, для последовательного адапте-

ра COM1 первый регистр имеет адрес 3F8h, следующий 3F9h и так да-

лее.

В 8250а имеется семь физических регистров и они описываются в

порядке возрастания начального номера, начиная с базового адреса.

Как показано н рисунке 8-4, базовый адрес порта имеет один ре-

гистр, который делится на два, как приемный буферный регистра и

регистр хранения передачи (THR), который используется для хране-

ния одного передаваемого или принимаемого символа. Затем следует

регистр разрешения прерываний, который используется для разреше-

ния или блокировки генерации прерываний последовательныма адапте-

ром. Третий регистр, называемый регистром идентификации прерыва-

ний, содержит сообщение ниверсального асинхронного

приемопередатчик об идентичности прерывания. Затем следует ре-

гистр правления линией, используемый для становления различных

связныха параметров, таких как длина слова, количество стоповых

битов, четность и скорость передачи в бодах. Пятый регистр - это

регистр управления модемом, который используется для передачи мо-

дему сигналов, таких как DTR (терминал готов) и RTSа (запроса на

передачу). Наконец, два последних регистра, регистр состояния ли-

нии и регистр состояния модема, показывают соответственно состоя-

ние линии и модема.

Первые два регистра применяются также для установки скорости

передачи ва бодах. Скорость передачиа в бодах определяется как

16-битовый делитель тактовой частоты, используемой для последова-

тельного адаптера (1.8432 Гц в большинстве систем MS -DOS). Зна-

чение делителя вычисляется по формуле


1,843,200

делитель= -----------------------------16 Х скорость передачи в бодах


Чтобы становить скорость передачи в бодах, Вы должны проде-

лать следующее:

1. становить в 1 наиболее значимый бита регистр правления

линиейа (она называется битом защелки доступа делителя или

DLAB).

2.Загрузить младший и старший байты делителя соответственно в

приемный буфер и регистр разрешения прерываний.

3. становить DLAB в 0 для обеспечения нормальной работы ниверсального асинхронного приемопередатчика.



A Приемный буфер/регистр хранения передачи

B ┌───┬───┬───┬───┬───┬───┬───┬───┐

(COM1-3F8| 1.| | | | | | | 2.|

COM2-2F8)└───┴───┴───┴───┴───┴───┴───┴───┘


Регистр разрешения прерывания

B+1 ┌───┬───┬───┬───┬───┬───┬───┬───┐

| 0 | 0 | 0 | 0 | 3.| 4.| 5.| 6.|

└───┴───┴───┴───┴───┴───┴───┴───┘

становить бит в 1 для разрешения


Регистр идентификации прерывания

B+2 ┌───┬───┬───┬───┬───┬───┬───┬───┐

| 0 | 0 | 0 | 0 | 0 | | | 7.|

└───┴───┴───┴───┴───┴───┴───┴───┘

└─3-битовый идентификатор прерывания

110 = состояние линии

100 = приемные данные

010 = буфер передачи свободен

= состояние модема

Регистр правления линией

B+3 ┌───┬───┬───┬───┬───┬───┬───┬───┐

| 8.| | 9. |10.|а 11. |

└───┴───┴───┴───┴───┴───┴───┴───┘

└─BREAK: 1 станавливает линию в SPACE

Регистр правления модемом

B+4 ┌───┬───┬───┬───┬───┬───┬───┬───┐а a. - OUT2

| 0 | 0 | 0 |12.| a.| b.| c.| d.|а b. - OUT1

└───┴───┴───┴───┴───┴───┴───┴───┘а c. - RTS

└── 13. d. - DTR

Регистр состояния линии

B+5 ┌───┬───┬───┬───┬───┬───┬───┬───┐

| 0 |14.|15.|16.|17.|18.|19.|20.|

└───┴───┴───┴───┴───┴───┴───┴───┘


Регистр состояния модем аa. - RLSD

B+6 ┌───┬───┬───┬───┬───┬───┬───┬───┐а b. - Delta RLST

| a.| RI|DSR|CTS| b.| c.| d.| e.|а c. - Delta RI

└───┴───┴───┴───┴───┴───┴───┴───┘а d. - Delta DSR

e. - Delta CTS


A - адрес порта или регистр

B - базовый адрес; 1. - бит 7 данных;а 2. - бит 0 данных;

3. - состояние модема; 4. - состояние линии приема;

5. - регистр хранения передачи свободен; 6. - прием данных

разрешен; 7. - 0 означает, что прерывание ждет;

8. - бит доступа к защелке; 9. - четность:а =нет,

001=нечетность, 011=четность; 10. - число стоповых битов;

11. - длина слова: 10-7, 11-8; 12. - проверка обратного цикла;

13. - должен быть 1 для прерывания ввода/вывода персонального

компьютера; 14. - передача свободна; 15. - регистр хранения пере-

дачи свободен; 16. - обнаружен BREAK; 17. - ошибка кадровой

синхронизации; 18. - ошибка четности;а 19. - ошибка выхода за

границы; 20. - данные для приема готовы;


Рис.8-4. Регистры в 8250 UART.


Применяя этот подход Вы можете становить любое значение ско-

рости передачи в бодах. Обратите внимание, что максимально воз-

можной скоростью передачи является 1/16а тактовойа частоты, или

115,200а бод (для этой скорости передачи делитель равен 1). Этот

предел вытекает из того, что делитель не может быть меньше едини-

цы. Для установки скорости передачи в бодах Вы можете также ис-

пользовать прерывание BIOS 14h. Мы рассмотрим применение BIOS да-

лее в этой главе.


правляемый прерываниями последовательный ввод/вывод


Существует два общих метода ввода/вывода в любой вычислитель-

ной системе:а порядоченный и правляемый прерываниями. порядо-

ченность относится к повторяющейся проверке состояния регистра

устройства ввода/вывода для инициализации требуемой транзакции. В

упорядоченном вводе/выводе программа, запрашивающая символ ввода,

многократно считывает состояние регистра в стройстве ввода/выво-

да до тех пор, пока оно не покажет, что символ доступен для ввода

(или до тех пор, пока программа не решит, что "время закончи-

лось"). Когда состояние казывает, что имеется готовый для работы

символ, программа считывает его из соответствующего регистра ст-

ройства ввода/вывода. Сходная последовательность "ждать, до тех

пор пока не готов, затем писать" используется при выведении сим-

волов на устройство ввода/ вывода. Таким образом, дальнейшее вы-

полнение программы приостанавливается до завершения выполнения

операции ввода/вывода.

Большой проблемой для порядоченного ввода/вывода через ком-

муникационный порт является то, что при скорости передачиа выше

300а бода программе трудно что-либо сделать с получаемым символом

кроме как отображать его на экране. Рассмотрим следующий пример.

Предположим, что мы читаем символы со скоростью 300 бод и имеем

следующие связные параметры:а длина слова 7 бит, проверка на чет-

ность и один стоповый бит, который вместе со стартовым битом, до-

бавляет до 10 бит на символ. Вы ожидаете получать около 30 симво-

лова каждую секунду. После чтения символ программа имеет около

1/30 секунды для выполнения других операций. Если Вы не желаете

потерять какие-либо символы,то в это время Вы должны снова начать

упорядочение порта. Что произойдет, когда скорость возрастет до

9600 бод?а Временной интервал между символами слишком мал для вы-

ведения символа на экран дисплея, не позволяета интерпретировать

специальные символы и эмулировать терминал.

В подходе, основанном на правлении прерываниями, программа

предоставляета возможность прерыванияма стройств ввода/вывода

поступать непосредственно на центральный процессор, который про-

должает выполнять свою работу, не связываясь с стройством. Когда

устройство готово к вводу/выводу, оно сигнализирует об этом цент-

ральному процессору посредством аппаратуры. Получив этот сигнал,

центральный процессор сохраняет свое текущее состояние и вызывает

подпрограммуа обслуживания прерываний, адрес которой хранится в

таблице векторов прерываний. Эта подпрограмма выполняет операцию

ввода/вывода, затем восстанавливает состояние машины и возвраща-

ется в прерванную программу. учитывайте регистр символов, посту-

пающих в коммуникационный порт персонального компьютера. Органи-

зовав где-нибудь некоторыеа ячейки памяти (буфер), Вы можете

использовать простую подпрограмму обработки прерываний, которая

быстро считывает символ из коммуникационного порт иа сохраняет

его в следующей доступной ячейке памяти в буфере. Символы не бу-

дут теряны в процессе считывания и сохранения символа драйвером

прерываний перед поступлением следующего символа. Эта несложная

задача достаточно проста для выполнения в короткие временныеа ин-

тервалы между поступающими символами приа скорости передачи

9600 бод. Прелесть этого метода заключается в том, что время об-

работки главной программой символов, хранящихся в буфере, не име-

ет значения. Конечно, существует риск переполнения буфера, но эта

проблем может быть решена простым величением его размера. Если

этот способ не очень хорош, то для избежания переполнения буфера

можно использовать правление потоком с помощью XON/XOFF.

Из наших рассуждений должно стать очевидным, что правляемая

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

помощью XON/XOFF, предпочтительнее порядоченной связи.


Прерывания последовательного адаптера


Последовательный адаптера персонального компьютера может быть

запрограммирован для прерывания работы центрального процессора

всякий раз как только происходит одно из четырех событий (смотри

рисунок 8-5). Универсальный асинхронный приемопередатчик присваи-

вает приоритет каждому из четырех событий. В таблице 8-1 перечис-

лены четыре прерывания.



Таблица 8-1

Прерывания последовательного адаптера

───────────────┬────────────────────────────────────────────────

Приоритета |а Идентификатор прерывания

───────────────|────────────────────────────────────────────────

1 |а Состояние приемной линии (RLS)

2 |а Доступность данных для приема (RDA)

3 |а Регистр хранения передачи свободен (THRE)

4 |а Состояние модема (MS)

───────────────|────────────────────────────────────────────────


Символы ASCII

┌─┬─┬─┬─┬─┬─┬─┬─┐ ┌─────────────────────────────────┐

└─┴─┴─┴─┴─┴─┴─┴─┘ |а Регистр разрешения 1 2 3 4а |

| прерывания ┌─┬─┬─┬─┐ |

| └─┴─┴─┴─┘ |

| │ |

| |

|а Идентификация ┌─┬─┬─┬─┐ |

| прерывания └─┴─┴─┴─┘ |

Последовательный | ┌────────┘ |

адаптер └──────────────┐ |

│ ││││││││││││││││ |

└─┴┴┴┴┴┴┴┴┴┴┴┴┴┴┴┴─┘


┌─┐

Прерывание к 8259A ─┘ └──


Рис.8-5. Прерывания последовательного адаптера


Событием высшего приоритет является прерывание "состояние

приемной линии"а (RLS), которое обрабатывается чтением регистра

состояния линии. Прерывание RLS имеет место, когда происходит од-

но из следующих событий:


- Линия отключается (логический 0) на период времени больше

требуемого для получения символа.

- Символа получен до того, как был считан предыдущий (ошибка

выхода за границы).

- Ошибка при проверке на четность.

- При восстановлении символа из полученных битов не обнаружен

стоповый бит (ошибка кадровой синхронизации).


Затем следуета прерываниеа "доступность данныха для приема"

(RDA), которое возникает при готовности символ для чтения из

приемного буферного регистра. Оно может быть сброшено путем счи-

тывания символа из регистра.

Следующий приоритета имеет прерывание "регистр хранения пере-

дачи свободен" (THRE). Как следует из имени прерывания, оно имеет

место, когда регистр, хранящий символ, предназначенный для пере-

дачи (имеет тот же адрес порта, что и регистр приемного буфера),

свободен. Прерываниеа обрабатывается записью в этот регистр или

чтением из регистра идентификации прерывания. Необходима другой

метод сброса этого прерывания, так как иногда даже через прерыва-

ния универсального асинхронного приемопередатчика сообщается, что

буфер передачи пуст и, возможно, отсутствует информация для пере-

дачи.

Самый низкий приоритета имеета прерывание "состояние модема"

(MS). Оно имеет место, когда модем:


- Получает (посылает) сигнал "открыт для передачи" (CTS).

- Показываета своюа готовность становкой линии "набор данных

готов" (DSR).

- Получаета сигнал, станавливающий линию "индикатор кольца"

(RI) в логическую 1.

- Определяета сигнала переноса (тот тон, который Вы слышите,

когда вызываете номер и модем отвечает), устанавливая линию

"определение сигнала приемной линии" (RLSD) в 1.


Прерывание состояния модема может быть сброшено чтением регистра

состояния модема.

Эти прерывания могута включаться и выключаться индивидуально

установкой соответствующих битов в регистреа разрешения прерыва-

ний. В последовательно-параллельном адаптере фирмы IBM (также как

и в асинхронном адаптере фирмы IBM) бит с именем OUT2а в регистре

управления модемом должен быть становлен в 1 до того, как преры-

вания универсального асинхронного приемопередатчика могут достичь

центральный процессор. Когда имеют место прерывания, последова-

тельный адаптер собирает их согласно приоритету и направляет пре-

рывание высшего приоритет в регистр идентификации прерывания.

даптер прекращает отвечать на последующие прерывания равного или

меньшего приоритета до тех пор, пока не определит, что текущее не

обслужено подпрограммой обслуживания прерываний. Программируемый

контроллер прерываний 8259A.

Ва системаха MS-DOSа центральныйа процессор (микропроцессор

80х86) непосредственно не принимаета прерывания, поступающие от

ппаратных средств, таких как последовательный адаптер. Прерыва-

ния аппаратуры сначала обслуживаются чипом программируемого конт-

роллер прерыванийа Intel 8259A. 8259A действует как "приемщик"

центрального процессора. Контроллер 8259A, кака программируемое

устройство, принимает до восьми различных прерываний и может мас-

кировать (игнорировать) прерывания индивидуально. 8259A отвечает

н каждое незамаскированное или разрешенное прерывание и направ-

ляет его центральному процессору при словии, что никакое другое

прерывание высшего приоритета не обслуживается в настоящее время.

Как контроллер 8259A присваивает приоритеты?а Кака иа универ-

сальный асинхронный приемопередатчик имеет свой метод определения

приоритетов прерываний последовательного адаптера, 8259A обладает

своей схемой присваивания приоритетов прерываниям. Последователь-

ный адаптер является только одним из аппаратныха средств, которые

могут направлять прерывания контроллеру 8259A. Каждое устройство

жестко смонтировано или соединено проводниками с различными вход-

ными устройствами, известными как входные стройства запроса пре-

рываний (IRQ) контроллера 8259A. Поэтому, обычно говорят об IRQ,

присвоенном аппаратному прерыванию. Другой характеристикой, свя-

занной с IRQ прерывания, является номер прерывания, используемый

для обращения к отдельному прерыванию. В персональном компьютере

IBM PC этот номер равен восьми плюс IRQ. Когда имеет место преры-

вание, центральный процессор использует его номер в качестве ин-

декса в таблице, известной как таблица векторов прерываний (рас-

положен ва начале памяти), которая содержит адрес подпрограммы

обработки данного прерывания. Так как контроллер 8259A связывает

высшие приоритеты с низкими IRQ, аппаратные стройства, требующие

максимального внимания, имеют низкие IRQ. Таким образом, систем-

ный таймер имеет IRQ0, клавиатура имеет IRQ1 и так далее.

Несмотря на то, что MS-DOS 3.3 поддерживает четыре коммуни-

кационных порта, с COM1 по COM4, эта поддержка не означает ничего

кроме обладания четырьмя драйверами с этимиа именами, каждый из

которыха поддерживается безбуферно и только порядоченным вво-

дом/выводом. Так как мы интересуемся правляемыми прерываниями

последовательным вводом/выводом, детали поддержки системой MS-DOS

коммуникационных портов не относятся к данному вопросу.

В персональнома компьютереа IBMа PCа только два первых порта,

COM1 и COM2, имеют определенные номера IRQ и номер прерываний.

Другие последовательные порты, такие как COM3 и COM4 могут быть

использованы для управляемого прерываниями ввода/вывода после с-

тановки адаптерова и присваивания номеров IRQ посредством соот-

ветствующей установки перемычек. Как только станет известен номер

IRQ, программирование портов COM3 и COM4 выполняется таким же об-

разом, как и портов COM1 и COM2. Более того, до конца этой главы

мы будем рассматривать только порты COM1 и COM2.

Двум последовательным портам COM1 и COM2 присвоены соответс-

твенно IRQ4а и IRQ3, то есть номера прерываний 12 и 11 (десятич-

ные). Между прочим, номера прерываний должны быть известны, так

кака функциональные вызовы DOS (посредством программного прерыва-

ния 21h) с функциональными номерами 35h и 25h могут быть исполь-

зованы соответственно для получения и становки векторов прерыва-

ний.

Есть ещеа несколько моментов, о которых необходимо помянуть

перед началом разговора о программировании последовательныха пор-

това для правляемого прерываниямиа ввода/вывода. Микропроцессор

80х86 автоматически делает невозможными все прерывания в то время

когд он передает правление обслуживающей подпрограмме текущего

прерывания. Несмотря на то, что контроллер 8259A во время обслу-

живания прерывания задерживает последующие прерывания того же или

меньшего приоритета, прерывания старшего приоритета все еще полу-

чают подтверждение о приеме, если становлен флаг прерывания. Ес-

ли мы вновь немедленно не разрешим прерывание до начал обслужи-

вания прерывания от последовательного порта, многие жизненно

важные системные функции, передаваемые прерываниями (такиеа как

системный таймер, клавиатура и контроллер диска), обслуживаться

не будут. Важно, поэтому, используя команду STI (установить флаг

прерывания)а переключить прерывание как только обслуживающая под-

программа примет на себя правление. Это предоставита возможность

таймеру, клавиатуреа и контроллеру диска прерывать подпрограмму

обслуживания последовательного порта, позволяя функционировать

другим устройствам.

Как мы сможем сообщить контроллеру 8259A о том, что обработка

последовательного прерывания завершена?а Наша служебная подпрог-

рамма должна направить 8259A команду "конец прерывания" (EIO) пе-

ред возвращением управления центральному процессору. Несмотря на

то, что существуют способы требования EIO для различных IRQ, для

схемы приоритета, используемой в персональном компьютере, доста-

точно направить контроллеру 8259A то, что известно как "неспеци-

фический" EOI (код 20h). Название "неспецифический" вытекаета из

того, что эта команда не определяет, какое прерывание обслужива-

лось. Она просто говорит контроллеру, что обслуживание прерывания

высшего приоритета завершено. Это разрешает обслуживание прерыва-

ний того же или высшего IRQ.


Программирование контроллера 8259A


правляемый прерываниями ввод/вывод требует правильной ста-

новки контроллера 8259A. В противном случае, прерывания, генери-

руемые последовательным адаптером, никогда не будут приняты мик-

ропроцессорома 80х86. Таким образом, важно выяснить сначала то,

как мы можем программировать 8259A.

Кака вся аппаратур персонального компьютера, контроллер

8259A программируется посредством двух имена команда (регистров).

Они расположены в адресах 20h и 21h порта ввода/вывода соответс-

твенно (рисунок 8-6). Регистр с адресом 21h используется исключи-

тельно для маскирования прерываний. Прерывание маскируется (т.е.

не принимается) в том случае, если бит, соответствующий своему

IRQа (считая справа налево, причем самому правому биту присвоено

значение IRQ0), становлен в логическую единицу. Порт по адресу

20hа используется для направления команды прерывания контроллеру

8259A. Как мы отметили ранее, в системах MS-DOS это делается за-

писью 20h в этот порт.



Прерывание к CPU ───┐

|

┌ ─ ─ ─ ─ ─ ─ ─ ─ ─ ─ ─ ─ ─ ─ ─ ─ ─ ─ ─ ─ ─ ─ ─ ─┴─ ─ ─ ─ ─ ─ ┐

┌───┬───┬───┬───┬───┬───┬───┬───┐ конец

│ Порт 20hа | | | |а а| | | | | прерывания |

└───┴───┴───┴───┴───┴───┴───┴───┘ EOI=20h

│ └──── 20h |

8259A IRQ

│ ┌───┬───┬───┬───┬───┬───┬───┬───┐ 1=a. |

| 0 | 0 | 0 | 0 | 3.| 4.| 5.| 6.| 0=b.

│ └───┴───┴───┴─┬─┴─┬─┴───┴───┴───┘ |

Программируемый контроллер прерываний

└ ─ ─ ─ ─ ─ ─ ─ ─ ─ ─ ─ ─ ─ ─ | ─ | ─ ─ ─ ─ ─ ─ ─ ─ ─ ─ ─ ─ ─ ┘

| | ┐ Последовательные

| └─── COM2:│ порты

└─────── COM1:│ персонального

┘ компьютера


a. - замаскировать прерывание; b. - разрешить прерывание;


Рис.8-6. Программируемый контроллер прерываний 8259A


В системаха MS-DOSа первомуа последовательномуа портуа (COM1)

присвоен номер IRQ4 (прерывание номер 12), в то время как второй

(COM2) имеет IRQ3 (прерывание 11). Как казывалось ранее, порты

COM3 и COM4 могут обрабатываться таким же образом, предполагая,

что известны номера IRQ, присвоенные этим портам во время ста-

новки (только порты COM1 и COM2 имеют заранееа присвоенные номера

IRQ). Поэтому, контроллер 8259A может быть запрограммирован для

получения прерываний от порта COM1 чтением с порта 21h и обратной

записью содержания, логически множенного на EFh. Прерывания пор-

та COM1 могут быть замаскированы не только повторением предыдущих

действий, но логическим сложением с 10h, вместо логического мно-

жения. Таким образом, контроллер 8259A может быть запрограммиро-

вана для разрешения прерываний порта COM1 с помощью следующего

фрагмента программы


IN AL,21H ;получить текущую маску прерывания

AND AL,EFH ;разрешить IRQ4

OUT 21H,AL ;снова записать его обратно


Когда прерывания порт COM1 снова выключаются, 8259A может

быть запрограммирован следующим образом


IN AL,21H ;получить текущую маску прерывания

OR AL,10H ;запретить IRQ4

OUT 21H,AL ;снова записать его обратно


Подобное программирование стройств, выполняемое считыванием

содержание регистра с последующей записью его обратно с соотвест-

вующе измененным битом, рекомендуется ввиду того, что мы не нару-

шаем предварительной становки битов.

Кроме разрешения и запрещения приема прерываний, контроллер

8259A должен быть проинформирован о завершении обработки обычного

прерывания. Как казывалось ранее, это выполняется направлением

20h в адрес 20h порта ввода/вывода следующим образом:


MOV AL,20H ;код конца прерывания

OUT 20H,AL ;в порт 2Н контроллера 8259A


Использование средств MS-DOS для программирования

последовательного порта


Если Вы знаете назначение каждого регистра последовательного

даптера, программирование последовательного порта включает в се-

бя правильную установку регистров, разрешение прерываний и ста-

новку программы обработки прерываний. Мы же рассказали о том,

кака программировать контроллера прерываний. Теперь мы готовы к

рассмотрению остальных деталей программирования последовательного

даптера.


Драйвер, TSR или автономная программа


Существует несколько способов получения доступа к последова-

тельному адаптеру в системе MS-DOS. Вы можете правлять последо-

вательным портом через станавливаемый драйвер стройства, выпол-

няющий ввод/вывод через этот порт. В главе 6 "Устанавливаемые

драйверы устройств" рассматриваются деталиа разработки такого

драйвера. Главныма недостатком такого подхода являются издержки,

связанные с доступом к драйверу через DOS, достоинством то, что

любая программа, которая знает о Вашем драйвере, может его ис-

пользовать. Если Вы выбрали этот способ, то можете обеспечить в

драйвере возможности IOCTL таким образом, что связные параметры,

такие как скорость передачи в бодах и длина слова, могут быть с-

тановлены вызовами DOS IOCTL (номер функции DOS 44h).

Вторым подходом является становка программы TSR (завершенная

и оставленная резидентно), которая используется с помощью преры-

вания BIOS RS-232C (14h) и расширяет свою функциональность обес-

печением управляемого прерываниями ввода/вывода. Этот метод также

дает любой программе доступ к последовательному порту посредством

Вашего драйвер TSR в то время как Вы подтверждаете необходимые

установки регистра при использовании новых коммуникационных функ-

ций TSR. Механизм доступа будет таким же, как вызов функции BIOS

RS-232C, который мы вскоре опишем.

Третий метода заключается в разработке автономной программы,

которая включает в себя служебную подпрограмму обработки прерыва-

ний последовательного порта. Ва этом случае, при запуске этой

прикладной программы, Вы можете становить обработчик прерываний

последовательного порт и сбросить его после прекращения работы

программы. Этот способ создает возможности организации высокоско-

ростного (9600 бод) последовательного ввода/вывода ввиду наличия

в нем меньшего количества недостатков по сравнению с двумя други-

ми методами.

Независимо от того, какой подхода Вы выберите, правление

последовательным портом останется таким же. Далее мы уделим осо-

бое внимание деталям.


Использование BIOS для последовательной связи


Вы, вероятно, спросите, можно ли реализовать эффективный ввод

/вывод через BIOS. К сожалению, нет. BIOS не предоставляет эффек-

тивной возможности правления апоследовательныма адаптером. Для

программирования последовательного адаптер BIOS имеет функцию

RS-232C, доступную через прерывание 14h. К сожаления, эта функция

поддерживаета только порядоченный ввод/вывод, который не очень

эффективен ввиду недостатков, изложенных ранее. Тем не менее, эта

функция идеальна для становки таких параметров коммуникационного

порта, как скорость передачи в бодах, длина слова и стоповый бит,

использующий прерывание 14h BIOS.


становка коммуникационных параметров с использованием BIOS


Даже при использовании в последовательнома вводе/выводе BIOS

не столь эффективна как правляемый прерываниями подход. Полезно

посмотреть, как коммуникационные параметры (скорость передачи в

бодах, длина слова, четность и стоповые биты) могут быть станов-

лены с использованием функций BIOS RS-232C, доступных по прерыва-

нию 14h.

Прерывание 14h с нулем в АН станавливает параметры последо-

вательного порта. Номер порта должен находиться в DX. Нуль в DX

указывает на порт COM1, в то время, кака единиц казываета на

COM2. Выбранные коммуникационные параметры направляются в регистр

AL в пакованном формате, показанном на рисунке 8-7. Скорость пе-

редачи определяется 3-битовыма значением, четность - 2-битовым

значением, число стоповых битов - одним битом и длин слов -

2-битовым значением. В таблице 8-2 показаны кодированные значения

каждого коммуникационного параметра. чтите, что скорость переда-

чи драйверова череза порта COM в системе DOS 3.3 может достигать

19200 бод, в то время, как ROM-BIOS ограничивается 9600 бодами.

Для достижения скорости передачи, не казанной в таблице 8-2, Вы

можете использовать возможность программирования скорости переда-

чи ниверсального асинхронного приемопередатчика, рассмотренную

ранее.



Скорость Длина

┌передачи в ┬Чет-сть┬ a.┬ слова ┐

|а бодах | | | |

┌───┬───┬───┬───┬───┬───┬───┬───┐

| | | | | | | | |

└───┴───┴───┴───┴───┴───┴───┴───┘

Боды Четность Стоп-биты Длина слова

┌─────┬─────┐а ┌──┬──┐ ┌──┬──┐ ┌────┬────┐

| | 110 |а |00|а | |0 | 1| | 10 | 7а |

|─────|─────|а |──|b.| |──|──| |────|────|

| 001 | 150 |а |10|а | |1 | 2| | 11 | 8а |

|─────|─────|а |──|──| └──┴──┘ └────┴────┘

| 010 | 300 |а |01|c.|

|─────|─────|а |──|──|

| 011 | 600 |а |11|d.|

|─────|─────|а └──┴──┘

| 100 |1200 |

|─────|─────|

| 101 |2400 |

|─────|─────|

| 110 |9600 |

|─────|─────|

| | |

└─────┴─────|


a.- стоповые биты; b. - нет; c. - нечетность; d. - четность


Рис.8-7. Коммуникационные параметры, пакованные в одиночный

байт в формате, требуемом прерыванием 14h BIOS




Таблица 8-2

Кодированные значения коммуникационных

апараметров для прерывания 14h

──────────────┬──────────────────────┬─────────────────────────

Имя параметра | Фактическое значение | Кодированноеа значение

──────────────|──────────────────────|─────────────────────────

Скорость пе-а | 110 | 0

редачи в бодах| 150 | 1

| 300 | 2

| 600 | 3

| 1200 | 4

| 2400 | 5

| 4800 | 6

| 9600 | 7

| |

Четность | Нет | 0 или 2

| Нечетность | 1

| Четность | 3

| |

Стоповые биты | 1 | 0

| 2 | 1

| |

Длина слов | 7 | 2

| 8 | 3

──────────────┴──────────────────────┴─────────────────────────


Подготовить пакованную формуа параметров просто. Например,

если Вы используете язык высокого ровня, такой кака Си, можно

достичь результата следующим образом:


pckd_commparams = (baudrate << 5) | (parity <<3) |

(stopbits << 2) | (wordlength);


Мы использовали операторы сдвига бита и поразрядного логического

сложения язык Си. Переменные baudrate, parity, stopbitsа и

wordlengthа должны быть кодированными значениями коммуникационных

параметров из последней колонки таблицы 8-2. Как только параметры

примут такой формат, Вы можете вызывать BIOS прерыванием 14h. Ис-

пользование функции int86 компилятора Microsoftа С иллюстрируется

следующим фрагментом программы:


#include <dos.h>

#defineа BIOS_RS232а 0x14

/* номер прерывания для обслуживания BIOS */

static union REGS xr, yr;

.

xr.h.ah = 0; /* номер функции для вызова BIOS RS-232 */

xr.h.al = pckd_commparams; /* связные параметры */

xr.x.dx = port_number;а /* 0 означает COM1, 1 означает COM2 */

int86(BIOS_RS232, &xr, &yr); /* сделать вызов */

.

Компилятор Microsoft C версии 5.0 облегчает вызов подпрограмм

BIOS: функция _bios_serialcomm служит интерфейсом между Вашейа Си

программой и прерыванием BIOS 14h. Например, еслиа Вы выберете

8-битовую длинуа слова, 1 стоповый бит, отсутствие проверки на

четность и скорость передачи 300 бод, то достаточно вызвать


_bios_serialcom(_COM_INIT, COM1, (_COM_CHR8 | _COM_STOP1 |

_COM_NOPARITY | _COM_300) );


Вызов функции _bios_serialcom


status = _bios_serialcomm(service_code, port_number, data);


принимает три параметра, выраженных целым без знака и возвращает

кода состояния того же типа данных для демонстрации результата

требуемой операции. Параметр service_code используется для опре-

деления требуемойа операции и port_number принимает значение 0

(COM1) или 1 (COM2). Значение data зависит от требуемой слуги.

Подробности использования этой функции изложены в Microsoft C 5.0

Run-time Library Reference.


Получение адреса последовательного порта


Другим полезным встроенным свойством BIOS является то, что на

этапе самотестирования при включении (POST) она проверяет наличие

последовательных адаптеров COM1/COM2 (хотя MS-DOS 3.3 поддержива-

ет порты COM3 и COM4, BIOS распознает только COM1 и COM2) и, если

находит тот или другой, адрес первого регистра каждого адаптера

заносится ва область памяти, начиная со смещения нуля сегмента

14h. Так как в персональном компьютере 20-битовый физический ад-

рес равен 10h * 16-битовый сегмент + 16-битовое смещение, и если

Ваша система MS-DOS имеет один последовательный порт, назначенный

как COM1, то тогда слово в физической ячейке 400h будет содержать

3F8h (если так же присутствует COM2, следующее слово в ячейке 402

h будет содержать 2F8h). Таким образом, Вы можете получить адрес

последовательного адаптера из этой области данных BIOS н смеще-

нии 0 и сегменте 40h. Например, в Microsoft C Вы можете стано-

вить базовый адрес порта следующим образом:


#define BIOS_DATA ((short far *)(0x4L))

static short comport, /* для базового адреса порта */

port_number; /* 0 для COM1, 1 для COM2 */

.

.

.

comport = *(BIOS_DATA + port_number);


if(comport == 0) /* последовательный адаптер не становлен */

{

printf("Последовательный адаптер не установлен! ";

exit(1);

}


Так как инициализируется переменная comport, все другие ре-

гистры последовательного адаптера могут быть доступны прибавлени-

ем соответствующих смещений к базовому адресу. В языке Си Вы мо-

жете использовать директиву препроцессора #define для становки

дресов этих регистров. Например, если инициализируется comport,

Вы имеетеа возможность обратиться к регистрам последовательного

порта по их именам, определив их следующим образом:


#define IER (comport + 1) /* регистр разрешения прерывания */

#define IIR (comport + 2) /* определение прерывания */

#define LCR (comport + 3) /* регистр правления линией */

#define MCR (comport + 4) /* регистр правления модемом */

#define LSR (comport + 5) /* регистр состояния линии */

#define MSR (comport + 6) /* регистр состояния модема */


Настройка на правляемый прерываниями последовательный

ввод/вывод


После получения базового адреса порта из области данных BIOS

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

прерываний перед началом правляемого прерываниями последователь-

ного ввода/вывода. Номер прерывания и IRQ прерывания последова-

тельного порта зависит от того, какой порт используется, COM1 или

COM2. Как только Вы получили номер прерывания, Вы должны получить

дрес текущего обработчика и сохранить его. Таким образом, после

выхода из программы Вы можете восстановить первоначальноеа значе-

ние вектора прерываний. Функции MS-DOS 35h и 25h, соответственно,

получают и устанавливают обработчики для заданных номерова преры-

ваний. Для этой цели Microsoft Cа предоставляета подпрограммы

_dos_getvect и _dos_setvect.Используя язык Си Вы можете проделать

это следующим образом:


short int_number; /* номер прерывания для связного порта */

oid interrupt far s_inthndlr(void);

/* становить обработчик */

static voidа (interrupt far *old_handler)()

/* место для прежнего */

.

/* получить вектор прежнего прерывания и сохранить его */

old_handler = _dos_getvect(int_number);

/* становить новый обработчик с именем s_inthndlr

* запретить прерывания во время замены обработчика

*/

_disaple();

_dos_setvect(int_number, s_inthndlr);

_enable();


В приведенном примере мы представили обработчика кака функцию

типа interrupt, которая является новым ключевым словом, содержа-

щимся в Microsoft C 5.0. В следующем разделе продемонстрировано,

какима образома атрибут interrupt позволяет Вам писать обработчик

прерываний непосредственно в Microsoft C 5.0 (Turbo Cа 1.5а имеет

такую же возможность).

Следует обратить внимание на использование функций _disable и

_enable. Эти две функции соответствуют ассемблерным командам STI

и CLI. Таким образом, мы выключаем прерывания во время перехода

от одного обработчика последовательных прерываний к другому. С

другой стороны, прерывание, поступающее во время переключения,

может привести к тому, что может произойти сбой центрального про-

цессора, так как вектор прерывания не являлся адресом какого-либо

действующего обработчика.

После того, как обработчик прерываний займет свое место, Вы

можете становить коммуникационные параметры и разрешить последо-

вательному порту генерировать прерывания. Вы также должны разре-

шить распознавание этих прерываний контроллером 8259A. Еще раз Вы

должны запретить прерывания до тех пор, пока порт и 8259A не бу-

дута готовы. Вот кака мы можема проделать это с использованием

Microsoft C 5.0.


/* разрешает маску, зависящую от порта */

short intmask, int_enable_mask;

.

.

.

/* включает прерывания коммуникационного порта.

* станавливает 8259A

*/

_disable();

/* станавливает регистр правления модемом (порт =а MCR) */

outp(MCR, MCRALL);

/* разрешает все прерывания последовательной платы

* (порт = IER)

*/

outp(IER, IERALL);

/* считывает регистр маски прерывания 8259A и записывает его

* обратно после логического множения с _int_enable_mask

*/

intmask = inp(P8259_1) & int_enable_mask;

outp(P8259_1, intmask);

_enable();


С этой точки зрения последовательный порт начнета работать в

управляемом прерываниями режиме. Происходящие события зависят от

обработчика прерываний, который мы сейчас рассмотрим.


Обработка прерываний последовательного порта


Наш обработчик, s_inthndlr, будет вызван при генерации после-

довательным портом прерывания. Мы должны немедленно разрешить

прием системой последующих прерываний таким образом, чтобы другие

приоритетные задачи (такие как таймер) могли обрабатываться мик-

ропроцессором.

Следующий шаг - идентификация точной причины, вызвавшей гене-

рацию прерывания последовательного порта. Для получения информа-

ции Вы должны считать содержание регистра идентификацииа прерыва-

ния (IIR). Кака только определится причина прерывания, можно

выполнять его обработку, как описывалось в разделе, посвященном

универсальному асинхронному приемопередатчику.

Так как последовательный порт способен генерировать прерыва-

ние в то время, пока Вы обрабатываете другое, необходимо прове-

рять бит 0 (последний значащий бит) IIR на это словие. Если этот

бит равен нулю, то имеется другое прерывание и его следует обра-

ботать. С другой стороны, если значение бита равно единице, оче-

редные прерывания отсутствуют. В этом случае Вы должны направить

контроллеру 8259A сигнал "конец прерывания" и выйти из обработчи-

ка. Такима образом, обработчика представляета собой бесконечный

цикл, который продолжает обрабатывать последовательные прерывания

до тех пор, пока не прекратится их поступление. В Microsoft C 5.0

обработчик может быть реализован как


oid interrupt far s_inthndlr(void)

{

int c;

register int int_id, intmask;


/* прерывания разрешаются немедленно */

_enable();


while (TRUE)

{

/* чтение регистра идентификации прерываний, IRR */

int_id = inp(IRR);

if (bit0(int_id) == 1)

{

/* если бит 0 равен 1, тогда прерывания не поступают. послать

* сигнал конец прерывания программируемому контроллеру

* прерываний 8259A и затем вернуться.

*/

outp(P8259_0, END_OF_INT);

return;

}

/* если есть прерывание получения данных, разрешить прерывания

* для "свободен регистр хранения передачи"

*/

if (int_id) >= RXDATAREADY)

turnon_int(THEREINT,intmask);


/* обработать прерывание в соответствии с идентификатором.

* Следующий список составлен согласно возрастанию приоритета.

*/

switch (int_id)

{

case MDMSTATUS:а /* состояние готовности модема */

.

break;

case TXREGEMPTY:а /* послать символ */

.

break;

case RXDATAREADY: /* читать символ */

.

break;

case RLINESTATUS: /* читать состояние линии */

.

break;

/* пропустить, если идентификатор не является одним из

* перечисленных */

}

}

}


Обратите внимание, что мы воспользовались ключевыма словом

interrupt,имеющемся в Microsoft C 5.0 и позволяющем писать обра-

ботчик на языке Си. Это ключевое слово используется как специфи-

катор функции, которую Вы желаете становить в качестве обработ-

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

функции с атрибутом interrupt компилятор генерирует код для пер-

воначального помещения в стек регистров AX, CX, DX, BX, SP, BP,

SP, SI, DI, DS и ES. Затем он устанавливает регистр DS в режим

ссылки на сегмент данных казанной функции. После этой начальной

последовательности следует код функции. В заключении компилятор

использует команду IRET вместо обычной команды RET для выхода из

функции. Данный пример является типичным для использования атри-

бута interrupt. В компиляторе Turbo C также имеется это ключевое

слово, но помещение регистров в стек происходит в другом порядке.

Когда Вы пишете обработчик прерываний на языке Си, то необхо-

димо соблюдать такие же предосторожности, как и при написании об-

работчиков на языке ассемблера. Например, Вы не должны использо-

вать какую-либо библиотечную подпрограмму, вызывающую функцию DOS

(доступ к ним осуществляется посредством команды прерывания 21h).

Такими функциямиа ва языке Си являются подпрограммы ввода/вывода

файлов. С другой стороны, подпрограммы, находящиеся в категории

подпрограмм манипулирования строками, хранятся ва функции

interrupt.


Очереди обработчика прерываний


Целью обработчика прерываний последовательного порта является

скорейшее сохранение поступающих символов. Наилучшим образом это

достигается с использованием буфера. Прикладная программа может

восстанавливать из этого буфера символы со своей скоростью, не

беспокоясь о потере какого-либо символа, потому что она работает

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

обработчику прерываний через другой буфер.

Каждый из этих буферов должен вести себя кака линия проверки

регистра. Символы поступают в линию один за другим и программа,

считывающая символы, принимает первый из них и обрабатывает его,

затем принимает следующий и так далее. Буфер такого типа известен

как "первый пришел - первый вышел" (FIFO) буфер. Это также назы-

вается очередью.

На рисунке 8-8 показан концептуальная реализация очереди.

Очередь, естественно, имеет начало и конец. В реальной реализации

размер очереди, т.е. максимального числа символов, которое она

может содержать, фиксировано. добно считать ячейки памяти, свя-

занные с очередью, кругом, таким образом, что пройдя последнюю

ячейку, мы возвращаемся к первой. Это делает эффективным исполь-

зование ограниченного пространства, доступного для очереди. Такая

реализация очереди называется циклической.


┌───┬───┬───┬───┬───┬───┐

| | | | | | |

|───|───┴───┴───┴───|───|

| | | |

|───| |───|

| |Задние Первые| |

|───|───┐ ┌───|───|

| | │─┐ ┌─| | |

└───┴───┘ | | └───┴───┘

^ | | v

Вход Выход


Рис.8-8. Циклический буфер FIFO (очередь)


борка перед закрытием магазина


После того, как Ваша прикладная программа перестает нуждаться

в дальнейшем последовательном вводе/выводе, необходимо восстано-

вить порт в его обычное состояние. Восстановление включает в себя

установкуа всех битов регистра разрешения прерываний порта в ноль

и выключение всех сигналов правления модемом. Затема контроллер

8259Aа должен быть запрограммирован для прекращения приема преры-

ваний последовательного порта. В заключение, вектора последова-

тельного прерывания необходимо сбросить в начальное значение, ко-

торое было сохранено при инициализации ввода/вывода. Вот как это

реализуется в Microsoft C 5.0:


int intmask;

.

.

.

/* Запретить прерывания на время очистки */

_disable();

/* Сначала сбросить регистр разрешения прерываний порта */

outp(IER,IEROFF);

/* Выключить все биты регистра правления модемом */

outp(MCR,MCROFF);


/* Затем запретить распознавание контроллером 8259A прерываний

последовательного порта */

intmask = inp(P8259_1) | int_disable_mask;

outp(P8259_1, intmask);


/* Восстановить первоначальный вектор прерываний */

_dos_setvect(int_number, old_handler);


/* Снова разрешить прерывания */

_enable();


Пример программы


Мы описали аппаратные средства последовательного порта, каза

ли, какие действия необходимо выполнить дляпрограммирования порта

в целях организации эффективного управляемого прерываниями вво-

да/вывода. Осталось только объединить отдельные части, для того,

чтобы показать, каким образом создается завершенная коммуникацион

ная программа. Мы делаема это в листинге 8-1, который содержит

основную коммуникационную программу, написанную н Microsoftа C

версии 5.0.


Листинг 8-1. Коммуникационная программа на Microsoft C 5.0


Заключение


В этойа главе рассмотрены характеристики аппаратных средств

последовательного порта в системах MS-DOS иа представлены методы

его программирования. Также содержится небольшая коммуникационная

программа, написанная на Microsoft C 5.0, для иллюстрации реали-

зации этих методов на практике. Программирование последовательно-

го порта для правляемого прерывания ввода/вывод выполняется

следующим образом:


1. Получить базовый адрес выбранного коммуникационного порта

из области данных BIOS на сегменте 40h и со смещением 0.

2. Используя функцию MS-DOS 35h, получить адрес старой подпрограммы обслуживания прерывания для номер прерывания,

соответствующего данному адаптеру, и сохранить его.

3. Используя функцию MS-DOS 25h, становить для номера прерывания Вашу собственную подпрограмму обслуживания прерывания.

4. c использованием функции BIOS 14h становить коммуникационные параметры адаптера.

5. становить очереди приема и передачи для содержания входящих и исходящих символов.

6. Включить в регистре правления модемом необходимые сигналы

(например, DTR - терминал готов и RTS - запрос на передачу).

7. Разрешить все прерывания адаптера (установкой битов с 0 по

3 в регистре разрешения прерывания в 1).

8. Так же включить бит OUT2 в регистре управления модемом для

разрешения прерываний последовательного адаптера.

9. Запрограммировать контроллер 8259A для распознавания прерываний с IRQ этого адаптера (путем установки соответствующего бита регистра маски прерывания, доступного через адрес порта 21h, в 0).


При выполнении некоторых операций, если пользователь решает

прервать сеанс работы, должна быть вызвана подпрограмм "очист-

ки". Очистка выполняется следующим образом:


1. Выключить прерывания последовательного адаптера.

2. Сбросить биты регистра правления модемом.

3. Восстановить старую подпрограмму обработки прерывания.

4. Замаскировать прерывания для этого IRQ в 8259A.