Литмир - Электронная Библиотека
Содержание  
A
A

Далее дочерний процесс может применить

exec
для вызова любой программы, которая читает стандартный ввод. В данном случае мы используем команду
od
. Команда
od
будет ждать, когда данные станут ей доступны, как если бы она ждала ввода с терминала пользователя. В действительности без специального программного кода, позволяющего непосредственно выяснить разницу, она не будет знать, что ввод приходит из канала, а не с терминала.

Родительский процесс начинает с закрытия конца чтения канала,

file_pipes[0]
, потому что он никогда не будет читать из канала. Затем он пишет данные в канал. Когда все данные записаны, родительский процесс закрывает конец записи в канал и завершается. Поскольку теперь нет файловых дескрипторов, открытых для записи в канал, программа
od
сможет считать три байта, записанных в канал, но последующие операции чтения далее будут возвращать 0 байтов, указывая на конец файла. Когда
read
вернет 0, программа
od
завершится. Это аналогично выполнению команды
od
, введенной с терминала, и последующему нажатию комбинации клавиш <Ctrl>+<D> для отправки признака конца файла команде
od
.

На рис. 13.3 показан результат вызова

pipe
, на рис. 13.4 — результат вызова
fork
, а на рис. 13.5 представлена программа, когда она готова к передаче данных.

Основы программирования в Linux - image041.jpg

Рис. 13.3

Основы программирования в Linux - image042.jpg

Рис. 13.4

Основы программирования в Linux - image043.jpg

Рис. 13.5

Именованные каналы: FIFO

До сих пор вы могли передавать данные только между связанными программами, т.е. программами, которые стартовали из общего процесса-предка. Часто это очень неудобно, хотелось бы, чтобы и у несвязанных процессов была возможность обмениваться данными.

Вы можете сделать это с помощью каналов FIFO, часто называемых именованными каналами. Именованный канал — это файл специального типа (помните, что в ОС Linux все, что угодно, — файл!), существующий в виде имени в файловой системе, но ведущий себя как неименованные каналы, которые вы уже встречали.

Вы можете создавать именованные каналы из командной строки и внутри программы. С давних времен программой создания их в командной строке была команда

mknod
:

$ <b>mknod <i>имя_файла</i> p</b>

Однако команды

mknod
нет в списке команд X/Open, поэтому она включена не во все UNIX-подобные системы. Предпочтительнее применять в командной строке

$ <b>mkfifo <i>имя_файла</i></b>

Примечание

У некоторых более старых версий UNIX была только команда

mknod
. В стандарте X/Open issue 4 Version 2 есть вызов функции
mknod
, но не программа командной строки. ОС Linux, как всегда настроенная дружелюбно, предлагает оба варианта:
mknod
и
mkfifo
.

Внутри программы можете применять два разных вызова:

<b>#include &lt;sys/types.h&gt;</b>

<b>#include &lt;sys/stat.h&gt;</b>

<b>int mkfifo(const char *filename, mode_t mode);</b>

<b>int mknod(const char* filename, mode_t mode | S_IFIFO, (dev_t)0);</b>

Помимо команды

mknod
вы можете использовать функцию
mknod
для создания файлов специальных типов. Единственный переносимый вариант применения этой функции, создающий именованный канал, — использование значения 0 типа
dev_t
и объединений с помощью операции or режима доступа к файлу и
S_IFIFO
. В примерах мы будем применять более простую функцию
mkfifo
.

Итак, выполните упражнение 13.9.

Упражнение 13.9. Создание именованного канала

Далее приведен исходный текст примера fifo1.c.

#include &lt;unistd.h&gt;

#include &lt;stdlib.h&gt;

#include &lt;stdio.h&gt;

#include &lt;sys/types.h&gt;

#include &lt;sys/stat.h&gt;

int main() {

 int res = mkfifo(&quot;/tmp/my_fifo&quot;, 0777);

 if (res == 0) printf (&quot;FIFO created\n&quot;);

 exit(EXIT_SUCCESS);

}

Вы можете создать канал и заглянуть в него:

$ <b>./fifo1</b>

FIFO created

$ <b>ls -lF /tmp/my_fifo</b>

prwxr-xr-x 1 rick users 0 2007-06-16 17:18 /tmp/my_fifo|

Обратите внимание на то, что первый символ вывода —

р
, обозначающий канал. Символ
|
в конце добавлен опцией
-F
команды
ls
и тоже обозначает канал.

Как это работает

Программа применяет функцию

mkfifo
для создания специального файла. Несмотря на то, что запрашиваете режим
0777
, он заменяется пользовательской маской (
umask
), устанавливаемой (в данном случае
022
) точно так же, как при создании обычного файла, поэтому у результирующего файла режим
755
. Если ваша
umask
установлена иначе, например, ее значение
0002
, вы увидите другие права доступа у созданного файла.

Удалить FIFO можно как традиционный файл с помощью команды

rm
или внутри программы посредством системного вызова
unlink
.

Доступ к FIFO

У именованных каналов есть одно очень полезное свойство: поскольку они появляются в файловой системе, их можно применять в командах на месте обычного имени файла. Прежде чем вы продолжите программирование с использованием созданного вами файла FIFO, давайте исследуем поведение такого файла с помощью обычных команд для работы с файлом (упражнение 13.10).

235
{"b":"285844","o":1}