Синхронизацию осуществляет головной модуль. Когда процесс выполняет системный вызов M_IOCTL
, содержащее команду и ее параметры, и отправляет его вниз по потоку. Если какой- либо модуль вниз по потоку может выполнить указанную команду, в ответ он направляет подтверждение в виде сообщения M_IOCACK
. Если ни один из модулей и сам драйвер не смогли обработать команду, драйвер направляет вверх по потоку сообщение M_IOCNAK
. При получении одного из этих сообщений головной модуль пробуждает процесс и передает ему результаты выполнения команды.
При обработке сообщения промежуточным модулем или драйвером возникает проблема передачи данных. Как правило, команда
Эта схема неприменима для подсистемы STREAMS. Обработка сообщений модулем или драйвером выполняется в системном контексте и не имеет отношения к адресному пространству текущего процесса. Поэтому модуль не имеет возможности копировать параметры команды и возвращать результаты обработки, используя адресное пространство задачи.
Для преодоления этой проблемы в подсистеме STREAMS предлагаются два подхода.
Первый из них основан на использовании специальной команды I_STR
. При этом в качестве параметра передается указатель на структуру strioctl
:
ioctl(fd, I_STR, (struct strioctl*)arg);
struct strioctl {
int ic_cmd;
int ic_timout;
int ic_len;
char* ic_dp;
}
где ic_cmd
— фактическая команда,
ic_timeout
— число секунд, которое головной модуль будет ожидать подтверждения запроса, после он вернет процессу ошибку тайм-аута ETIME
,
ic_len
— размер блока параметров команды,
ic_dp
— указатель на блок параметров.
Если головной модуль не может обработать команду, он формирует сообщение M_IOCTL
и копирует в него команду (ic_cmd
) и блок параметров (ic_len
, ic_dp
). После этого сообщение направляется вниз по потоку. Когда модуль получает сообщение, оно содержит все необходимые данные для обработки команды. Если команда предполагает передачу информации процессу, модуль записывает необходимые данные в то же сообщение, изменяет его тип на M_IOCACK
и отправляет его вверх по потоку. В свою очередь головной модуль получает сообщение и производит передачу параметров процессу.
Другой подход получил название прозрачных команд M_IOCTL
и копирует в него параметры вызова — command
и arg
. Обычно параметр arg
является указателем на блок параметров, размер и содержимое которого известны только модулю (или драйверу), отвечающему за обработку данной команды. Поэтому головной модуль просто копирует этот указатель, не интерпретируя его и тем более не копируя в сообщение сам блок параметров. Сообщение передается вниз по потоку.
Когда модуль получает сообщение, в ответ он отправляет сообщение M_COPYIN
, содержащее размер и расположение данных[65], необходимых для выполнения команды. Головной модуль пробуждает процесс, вызвавший M_IOCARGS,
копирует в него параметры команды и направляет сообщение вниз по потоку. После этого процесс опять переходит в состояние сна.
Когда модуль получает сообщение M_IOCARGS
, он интерпретирует содержащиеся в нем параметры и выполняет команду. В некоторых случаях для получения всех параметров, необходимых для выполнения команды, может потребоваться дополнительный обмен сообщениями M_COPYIN
и M_IOCARGS
. Такая ситуация может возникнуть, например, если один из параметров являлся указателем на структуру данных. Для получения копии структуры модулю потребуется дополнительная итерация.
Вильям Л Саймон , Вильям Саймон , Наталья Владимировна Макеева , Нора Робертс , Юрий Викторович Щербатых
Зарубежная компьютерная, околокомпьютерная литература / ОС и Сети, интернет / Короткие любовные романы / Психология / Прочая справочная литература / Образование и наука / Книги по IT / Словари и Энциклопедии