Рабочий поток выполняет эту функцию, и, следовательно, эта функция выполняется в контексте процесса. По умолчанию при этом вес прерывания разрешены и никакие захваченные блокировки не удерживаются. Ели это необходимо, то функция может переходить в состояние ожидания. Следует заметить, что несмотря на то, что обработчики отложенных действий и выполняются в контексте процесса, эти обработчики не могут переходить в пространство пользователя, так как у потоков пространства ядра нет адресного пространства пользователя. Ядро может обращаться в пространство пользователя, только когда оно выполняется от имени пользовательского процесса, который имеет адресное пространство пользователя, отображенное на память, как, например, в случае выполнения системного вызова.
Блокировки между очередями отложенных действий и другими частями ядра осуществляются также, как и в случае любого другого кода, работающего в контексте процесса. Это позволяет сделать написание обработчиков отложенных действий достаточно простым. В следующих двух главах это раскрывается более детально.
Теперь, когда отложенное действие создано, его нужно запланировать на выполнение. Для того чтобы поставить обработчик данного действия в очередь на выполнение потоками
schedule_work(&work);
Действие планируется на выполнение немедленно и будет выполнено, как только рабочий поток events
, работающий на данном процессоре, перейдет в состояние выполнения.
Иногда необходимо, чтобы действие было выполнено не немедленно, а с некоторой задержкой. В этом случае работа может быть запланирована на выполнение в некоторый момент времени в будущем. Для этого используется следующая функция.
schedule_delayed_work(&work, delay);
В этом случае действие, представленное структурой work_struct
, с адресом &work
, не будет выполнено, пока не пройдет хотя бы заданное в параметре delay
количество импульсов таймера. О том, как использовать импульсы таймера для измерения времени, рассказывается в главе 10, "Таймеры и управление временем".
Действия, поставленные в очередь, выполняются, когда рабочий поток возвращается к выполнению. Иногда нужно гарантировать, что, перед тем как двигаться дальше, заданный пакет отложенных действий завершен. Это особенно важно для загружаемых модулей, которые, вероятно, должны вызывать эту функцию, перед выгрузкой. В других местах также может быть необходимо гарантировать, что нет ожидающих на выполнение действий, для предотвращения состояния конкуренции.
Для этого есть следующая функция, которая позволяет ждать, пока очередь действий
void flush_scheduled_work(void);
Данная функция ожидает, пока все действия в очереди действий
Заметим, что эта функция не отменяет никаких отложенных действий с задержками. Любые действия, которые запланированы на выполнение с помощью функции schedule_delayed_work
и задержки которых еще не закончены, — не очищаются с помощью функций flush_scheduled_work
. Для отмены отложенных действий с задержками следует использовать функцию
int cancel_delayed_work(struct work_struct *work);
Эта функция отменяет отложенное действие, которое связано с данной структурой work_struct
, если оно запланировано.
Если для поставленных целей недостаточно очереди отложенных действий, которая используется по умолчанию, то можно создать новую очередь действий и соответствующие рабочие потоки. Так как при этом создается по одному потоку на каждый процессор, то новые очереди действий необходимо создавать, только если необходима большая производительность за счет выделенного набора потоков.
Новая очередь действий и связанные с ней рабочие потоки создаются с помощью простого вызова функции.
struct workqueue_struct *create_workqueue(const char *name);
Параметр name используется для того, чтобы присваивать имена потокам ядра. Например, очередь events
, которая используется по умолчанию, создается с помощью следующего вызова.
struct workqueue_struct *keventd_wq = create_workqueue("events");
При этом также создаются все рабочие потоки (по одному на каждый процессор), которые подготавливаются к выполнению работы.