Конфликты IRQ
Хорошо, когда после сборки или плановой модернизации компьютер с первого раза
запускается и работает устойчиво и без глюков. Гораздо хуже, если возникают
неожиданные проблемы - спонтанные перезагрузки и зависания, сбои программ,
неработоспособность или "невидимость" устройств и т.п. Первая причина, которая
обычно приходит в голову в таком случае - конфликт прерываний. А хорошо ли мы
знаем природу этого явления, достаточно ли подготовлены r борьбе с ним?
Что такое IRQ
Прерывания - это базовый механизм реакции системы на возникающие события.
Аппаратные прерывания, называемые обычно IRQ (Interrupt ReQuest) - это
физические сигналы, с помощью которых контроллер устройства информирует
процессор о необходимости обработать некоторый запрос. Условно схема обработки
прерывания выглядит следующим образом:
- процессор получает сигнал прерывания и его номер;
- по специальной таблице отыскивается адрес программы, ответственной за
обработку прерывания с данным номером - обработчика прерывания;
- процессор приостанавливает текущую работу и переключается на выполнение
обработчика (в общем случае это некоторый драйвер);
- драйвер получает доступ к устройству и проверяет причину возникновения
прерывания;
- запускаются запрошенные действия - инициализация, конфигурирование
устройства, обмен данными и др.
- драйвер завершает работу, и процессор возвращается к прерванной
задаче.
Очевидно, что для корректной работы механизма прерываний необходимо
выполнение двух условий: во-первых, сигнал запроса должен доходить до
процессора, и, во-вторых, драйвер-обработчик должен правильно реагировать на
этот сигнал. В случае конфликта не соблюдается второе условие: сигнал прерывания
приходит, но реакция на него оказывается неправильной, в результате чего мы
имеем (в лучшем случае) неработоспособное устройство.
Конфликт
Можно сказать, что конфликт - это ситуация, при которой несколько объектов
одновременно пытаются получить доступ к ресурсу, который предназначен только для
одного из них. Конфликт прерываний возникает в том случае, если несколько
устройств используют одну и ту же линию прерывания для посылки сигнала запроса,
и нет механизма, позволяющего обрабатывать конкурирующие запросы. Если драйвер,
получая управление, работает не с тем устройством, которое послало запрос, то
либо происходит сбой, либо одно из устройств попросту не работает.
Возникает вопрос: могут ли несколько устройств использовать одну и ту же
линию прерывания, или это в принципе невозможно? Ведь если драйвер сможет
определить, от кого именно пришел запрос, то он будет реагировать на сигналы
только "своего" устройства, игнорируя все остальные. Но это должно быть
каким-либо образом заранее оговорено, иначе конфликт неизбежен.
Локальная шина PCI была спроектирована с учетом совместного использования
прерываний. Каждое устройство PCI должно корректно работать на одной линии
прерывания с другими PCI-устройствами. Это сделано следующим образом: факт
наличия сигнала на линии прерывания определяется не по фронту, т.е. изменению
уровня напряжения, а по самому факту наличия определенного напряжения. Изменять
напряжение в линии может сразу несколько устройств, становясь как бы в очередь
на обслуживание.
Таким образом, совместное использование одного IRQ несколькими
PCI-устройствами не является конфликтом по определению. Однако иногда проблемы
все-таки возникают. Во-первых, не все устройства PCI корректно работают на одной
линии прерывания с другими. Во-вторых, иногда драйверы имеют ошибки, из-за
которых они не могут правильно определять источник сигнала, мешая другим
драйверам. В-третьих, далеко не все устройства работают на шине PCI; например,
ISA-устройства, к которым относятся, например, контроллеры COM/LPT-портов,
делить прерывания с другими не умеют. Чтобы четко представлять себе, как можно
избежать конфликтов или устранить их, нужно разобраться в механизме управления
IRQ.
Организация аппаратных прерываний в персональном компьютере
Как вы знаете, персональные компьютеры начались с IBM PC. Его архитектура
предусматривала восемь линий аппаратных прерываний (IRQ), которыми управлял
специальный контроллер. Каждой из них назначался номер, который определял
приоритет прерывания и адрес его обработчика (так называемый вектор прерывания).
Новая архитектура, IBM PC AT, предусматривала еще восемь линий прерываний, для
которых использовался второй контроллер, подключаемый к одной из линии
прерывания первого контроллера. К сожалению, данная архитектура стала последней
после того, как фирма IBM потеряла возможность управлять развитием созданной ей
платформы, поэтому все современные компьютеры по-прежнему имеют только
шестнадцать прерываний, одно из которых используется вторым контроллером.
У компьютера IBM PC AT была только одна шина, по которой устройства могли
общаться с процессором и памятью - ISA. Большинство линий прерываний были
закреплены за стандартными ISA-устройствами, оставшиеся были зарезервированы на
будущее. Когда это будущее наступило, выяснилось, что новой универсальной шине
PCI досталось всего четыре свободных прерывания. Поэтому и был придуман хитрый
механизм совместного использования прерываний (IRQ Sharing) и динамического
переопределения номеров (IRQ Steering или Mapping).
Суть механизма управления прерываниями PCI-устройств в следующем. В общем
случае существует четыре физических линии PCI-прерываний, называемых PIRQ0,
PIRQ1, PIRQ2 и PIRQ3. Они подключены к контроллеру прерываний. Каждое
PCI-устройство со своей стороны как бы имеет четыре разъема, называемые INT A,
INT B, INT C и INT D. Подключать линии к разъемам можно в любом порядке.
Например, для первого PCI-слота можно сделать такую разводку: PIRQ0 - INT A,
PIRQ1 - INT B, PIRQ2 - INT C, PIRQ3 - INT D. А для второго - по-другому: PIRQ0 -
INT B, PIRQ1 - INT C, PIRQ2 - INT D, PIRQ3 - INT A. Обычно устройство требует
только одну линию прерывания, подключенную к INT A. Будучи установленным в
первый слот, устройство использует линию PIRQ0, а во втором слоте на том же
контакте будет линия PIRQ1. Тем самым устройства в разных слотах будут
использовать разные физические линии прерываний. Аппаратный конфликт между ними
будет исключен.
Шина AGP, являясь по сути специализированной модификацией PCI, тоже
использует одну из линий PIRQ - обычно PIRQ0.
Для современных систем четырех линий оказывается недостаточно, поэтому в
новых чипсетах часто применяются восемь линий PIRQ, которые точно так же в
разных комбинациях подключаются к слотам PCI и встроенным в плату устройствам.
Линии PIRQ подключаются к контроллеру прерываний. Им, как и другим линиям,
назначаются логические IRQ-номера. Если на одной физической линии находятся
несколько устройств (а это допустимо), то все они будут иметь один и тот же
номер IRQ. Если устройства находятся на разных физических линиях, они все равно
могут получить одинаковые номера IRQ. Нормальные драйверы позволят им свободно
работать без потери производительности, так как шина PCI все равно может
захватываться только одним устройством. Главное - распознать, от какого
устройства пришел сигнал.
Номера линиям PIRQ назначаются автоматически благодаря пресловутому механизму
Plug&Play. Но ведь есть и ISA-устройства, поддерживающие Plug&Play. Они
тоже имеют возможность автоматически получить номер IRQ. Но их линия прерывания
принадлежит им монопольно, и если такой же номер получит одна из линий PIRQ,
возникнет неразрешимый конфликт.
Итак, мы выяснили, что устройства PCI должны быть лишены проблем с
конфликтами IRQ. Если они, конечно, правильно работают, а так бывает не всегда.
К тому же драйверы должны поддерживать механизм совместного использования
прерываний. Устройства ISA не умеют делиться линиями прерываний и потому
являются провокаторами конфликтов. Следовательно, задача устранения конфликтов
сводится к правильному распределению номеров (источник проблем - ISA-устройства
и "кривые" драйверы) или к разведению по разным физическим линиям ("кривые"
PCI-контроллеры).
Давайте рассмотрим, каким образом в системе происходит распределение номеров,
и как мы можем повлиять на этот процесс.
Карта прерываний
Как я уже говорил, большинство номеров IRQ уже заняты стандартными
устройствами, точнее, назначены их линиям прерываний. Пройдемся по порядку:
0 - системный таймер (номер всегда занят); 1 - клавиатура (номер всегда
занят); 2 - второй контроллер прерываний (всегда занят); 3 - порт COM2
(может быть отключен, а номер - освобожден); 4 - порт COM1 (может быть
отключен, а номер - освобожден); 5 - порт LPT2 (обычно номер свободен); 6
- контроллер гибких дисков (может быть отключен, а номер - освобожден); 7 -
порт LPT1 (если не в режиме EPP или ECP, то номер свободен); 8 - часы
реального времени (всегда занят); 9 - свободен; 10 - свободен; 11 -
свободен; 12 - мышь PS/2 (может быть свободен, если нет такой мыши); 13 -
сопроцессор (всегда занят); 14 и 15 - контроллер жестких дисков (может быть
отключен, а номер - освобожден).
В типичной системе свободны номера 5, 7, 9-11, то есть пять из пятнадцати.
Кроме того, можно смело отключить COM2 и LPT1-порты, увеличив число свободных
номеров до семи. Свободны - не значит, что не заняты, просто между ними возможна
свободная перетасовка.
В любой системе имеется три стандартных PCI-устройства - ACPI-,
USB-контроллеры и видеокарта, каждое из которых займет по одному номеру. Сложное
устройство (например, звуковая карта) может потребовать несколько линий - INT A,
INT B и т.д. для своих компонентов, которые между собой не будут конфликтовать
(как-никак разные физические линии), а вот с другими устройствами - запросто.
Узнать, как в данный момент распределены номера прерываний, можно несколькими
способами. В самом начале загрузки компьютера появляется текстовая таблица
конфигурации. Сразу после нее идет перечень PCI-устройств с указанием
назначенного им номера IRQ. Другой способ работает в Windows 9x. В панели
управления есть иконка "Система", в вызываемом апплете - закладка "Устройства.
Выбираем свойства устройства "Компьютер", и там будут перечислены все устройства
с указанием их IRQ (см. скриншот).
В Windows 2000 у нас нет доступа к управлению прерываниями, поэтому для
просмотра списка IRQ нужно воспользоваться стандартной информационной утилитой
(Панель управления/Администрирование/Управление компьютером/Сведения о
системе/Ресурсы аппаратуры).
Распределение номеров IRQ средствами BIOS
В системе номера IRQ распределяются между физическими линиями дважды. Первый
раз это делает системный BIOS при начальной загрузке системы. Каждому
Plug&Play-устройству (все PCI, современные ISA, интегрированные устройства),
а точнее, его линии прерывания, назначается один номер из десяти возможных. Если
номеров не хватает, несколько линий получают один общий. Если это линии PIRQ, то
ничего страшного - при наличии нормальных драйверов и поддержки со стороны
операционной системы (об этом см. ниже) все будет работать. А если один номер
получают несколько ISA-устройств или PCI- и ISA-устройства, то конфликт просто
неизбежен, и тогда нужно вмешиваться в процесс распределения.
Прежде всего, нужно отключить все неиспользуемые ISA-устройства (в системах
без слотов ISA они тоже присутствуют) - порты COM1, COM2 и дисковод. Также можно
отключить режимы EPP и ECP порта LPT, тогда прерывание IRQ7 станет доступно.
В BIOS Setup нам понадобится раздел "PCI/PNP Configuration". Есть два базовых
способа повлиять на распределения номеров IRQ: заблокировать конкретный номер и
напрямую назначить номер линии PIRQ.
Первый способ доступен для всех BIOS: найдите список пунктов "IRQ x
used by:" (в новых BIOS скрывается в подменю "IRQ Resources"). Тем прерываниям,
которые должны быть назначены исключительно ISA-устройствам, нужно поставить
"Legacy ISA". Тем самым при раздаче номеров PCI-устройствам данные прерывания
будут пропущены. Поступать так следует в том случае, если какое-либо
ISA-устройство упорно становится на одно прерывание с PCI-устройством, из-за
чего оба не работают. Тогда мы находим номер этого IRQ и блокируем его в BIOS
Setup. PCI-устройство переходит на новый номер IRQ, а ISA-устройство остается.
Конфликт разрешен.
Второй, более удобный способ управления номерами IRQ - прямое
назначение. В том же подменю BIOS Setup могут быть пункты вида "Slot X use IRQ"
(другие названия: "PIRQx use IRQ", "PCI Slot x priority", "INT Pin x IRQ"). С их
помощью каждой из четырех линии PIRQ можно назначить конкретный номер. Кстати, в
новых AwardBIOS 6.00 можно наблюдать, какие именно устройства (включая
встроенные) используют ту или иную линию.
Распределение номеров IRQ средствами Windows
Второй раз номера прерываний распределяются операционной системой. Как
показали проведенные мной эксперименты, Windows'98 начинает вмешиваться в
произведенные BIOS'ом действия только в крайних случаях. При наличии нормального
BIOS описанные здесь приемы не понадобятся.
Следует заметить, что для правильной работы механизмов совместного
использования IRQ и динамического распределения необходимо, чтобы Windows
распознала чипсет материнской платы и загрузила IRQ Miniport. Чем более свежая
версия у Windows, тем больше чипсетов поддерживает ее собственный минипорт
(PCIIMP.PCI). Однако всегда лучше перестраховаться и установить самые свежие
драйверы чипсета.
В Windows 98 управление системой распределения IRQ осуществляется с помощью
стандартного менеджера устройств. В списке системных устройств нужно найти шину
PCI. В ее свойствах есть особая закладка (см. скриншот).
Если все настроено правильно, там будет упомянут минипорт ("успешно
загружен"), а управление шиной PCI (Steering) будет включено. Таким образом,
Windows'98 имеет средства для управления распределением номеров прерываний между
физическими линиями. Но поскольку и BIOS чаще всего с этим хорошо справляется,
этот механизм не задействуется.
Но иногда он просто необходим. Как я уже говорил, PCI-устройства не должны
конфликтовать в случае, если они используют одно и то же логическое прерывание.
Другое дело - ISA-устройства, к которым относятся также и COM- и LPT-порты. Если
устройство не-Plug&Play, BIOS может его и не заметить, отдав занятое им
прерывание PCI-устройству. Тогда нужно прерывание зарезервировать. Это делается
в диспетчере устройств Windows'98: выбираем устройство "Компьютер", вызываем его
свойства, переключаемся на вторую закладку. Дальше все понятно.
Кроме резервирования, можно непосредственно задать номер прерывания для
устройства. Для этого нужно в его свойствах найти закладку "Ресурсы", отключить
автоматическую настройку и попытаться изменить назначенный номер прерывания. К
сожалению, это работает далеко не всегда.
Windows 2000 - система особая. Если у вас современный компьютер, то он
наверняка поддерживает интерфейс конфигурирования ACPI. Windows 2000 в таком
случае вообще проигнорирует действия BIOS и "повесит" все PCI-устройства на одно
логическое прерывание. В общем случае это будет отлично работать (когда нет
ISA), но иногда случаются проблемы. Чтобы получить возможность изменять номера
прерываний, нужно либо поменять HAL-ядро, либо переустановить Windows 2000 с
отключенным в BIOS ACPI. Замена ядра производится так: в диспетчере устройств
выбирайте "Компьютер"/"Компьютер с ACPI", меняйте драйвер на "Стандартный
компьютер", перезагружайтесь. Если это не поможет, придется переустановить
Windows 2000 заново.
Надеюсь, приведенная выше информация поможет вам в борьбе с глюками "железа".
И помните: большинство возникающих проблем связано с низким уровнем компьютерной
грамотности хозяина компьютера. Поэтому нужно всегда стремиться к
самообразованию, тогда и проблем будет поменьше, а те, что все-таки возникнут -
не будут казаться неразрешимыми.
|