Синхронизацию осуществляет головной модуль. Когда процесс выполняет системный вызов
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
M_IOCARGS,
копирует в него параметры команды и направляет сообщение вниз по потоку. После этого процесс опять переходит в состояние сна.Когда модуль получает сообщение M_IOCARGS
M_COPYIN
и M_IOCARGS
. Такая ситуация может возникнуть, например, если один из параметров являлся указателем на структуру данных. Для получения копии структуры модулю потребуется дополнительная итерация.