if (pending) {
struct softirq_action *h = softirq_vec;
softirq_pending(cpu) = 0;
do {
if (pending & 1)
h->action(h);
h++;
pending >>= 1;
} while (pending);
}
Этот фрагмент кода является сердцем обработчика отложенных прерываний. Он проверяет и выполняет все ожидающие отложенные прерывания.
• Присваивает локальной переменной pending значение, возвращаемое макросом softirq_pending()
n, то отложенное прерывание с этим номером ожидает на выполнение.• Когда значение битовой маски отложенных прерываний сохранено, оригинальная битовая маска очищается[38]
.• Переменной h
softirq_vec.• Если первый бит маски, которая хранится в переменной pending
h->action(h).• Указатель h
softirq_vec.• Осуществляется логический сдвиг битовой маски, хранящейся в переменной pending
• Указатель h
• Последовательное повторение производится до тех пор, пока битовая маска не станет равной нулю. В этот момент больше нет ожидающих отложенных прерываний, и наша работа выполнена. Заметим, что такой проверки достаточно для того, чтобы гарантировать, что указатель h
softirq_vec, так как битовая маска pending имеет 32 бит и цикл не может выполниться больше 32 раз.Использование отложенных прерываний
Отложенные прерывания зарезервированы для наиболее важных и критичных ко времени выполнения обработчиков нижних половин в системе. Сейчас только две подсистемы — подсистема SCSI и сетевая подсистема — напрямую используют механизм softirq. В дополнение к этому, таймеры ядра и тасклеты построены на базе отложенных прерываний. Если есть желание добавить новое отложенное прерывание, то стоит себя спросить, почему будет недостаточно использования тасклетов. Тасклеты могут создаваться динамически, а также их легче использовать в связи с более простыми требованиями к блокировкам. Кроме того, их производительность все еще остается очень хорошей. Тем не менее для задач, критичных ко времени выполнения, которые способны сами обеспечивать эффективные блокировки, использование механизма softirq — будет правильным решением.
Отложенные прерывания должны объявляться на этапе компиляции с помощью соответствующего перечисления (enum
. Ядро использует указанный в перечислении индекс, который начинается с нуля, как значение относительного приоритета отложенных прерываний. Отложенные прерывания с меньшим номером выполняются раньше отложенных прерываний с большим номером.Создание нового отложенного прерывания состоит в добавлении новой записи в этот перечень (enum
HI_SOFTIRQ — имеет наибольший приоритет, a TASKLET_SOFTIRQ — наименьший. Новая запись, скорее всего, должна быть где-то ниже записей для сетевых устройств и выше записи для TASKLET_SOFTIRQ. В табл. 7.2 показан список всех типов отложенных прерываний.Таблица 7.2
. Список отложенных прерываний| Отложенное прерывание | Приоритет | Описание |
|---|---|---|
HI_SOFTIRQ | 0 | Высокоприоритетные тасклеты |
TIMER_SOFTIRQ | 1 | Обработчик нижних половин таймеров |
NET_TX_SOFTIRQ | 2 | Отправка сетевых пакетов |
NET_RX_SOFTIRQ | 3 | Прием сетевых пакетов |
SCSI_SOFTIRQ | 4 | Обработчик нижних половин подсистемы SCSI |
TASKLET_SOFTIRQ | 5 | Тасклеты |
Далее во время выполнения должен быть зарегистрирован обработчик отложенного прерывания с помощью вызова open_softirq()
data. Для сетевой подсистемы это делается, например, следующим образом.open_softirq(NET_TX_SOFTIRQ, net_tx_action, NULL);
open_softirq(NET_RX_SOFTIRQ, net_rx_action, NULL);