UNIX: разработка сетевых приложений
5.7. Нормальное завершение
На этом этапе соединение установлено, и все, что бы мы ни вводили на стороне клиента, отражается обратно.
linux % <b>tcpcli01 127.0.0.1</b> <i>эту строку мы показывали раньше</i><b>hello, world</b> <i>наш ввод</i>hello, world <i>отраженная сервером строка</i><b>good bye</b>good bye<b>^D</b> <i>Ctrl+D - наш завершающий символ для обозначения конца файла</i>Мы вводим две строки, каждая из них отражается, затем мы вводим символ конца файла (EOF)
, который завершает работу клиента. Если мы сразу же выполним команду<b>Ctrl+D</b>, то увидим следующее:netstatlinux % <b>netstat -а | grep 9877</b>tcp 0 0 *:9877 *:*tcp 0 0 local host:42758 localhost:9877Клиентская часть соединения (локальный порт 42 758) входит в состояние TIME_WAIT (см. раздел 2.6), и прослушивающий сервер все еще ждет подключения другого клиента. (В этот раз мы передаем вывод
программеnetstat, чтобы вывести только строки с заранее известным портом нашего сервера. Но при этом также удаляется строка заголовка.)grepПеречислим этапы нормального завершения работы нашего клиента и сервера.
1. Когда мы набираем символ EOF, функция
возвращает пустой указатель, и функцияfgetsвозвращает управление (см. листинг 5.4).str_cli2. Когда функция
возвращает управление клиентской функцииstr_cli(см. листинг 5.3), последняя завершает работу, вызывая функциюmain.exit3. При завершении процесса выполняется закрытие всех открытых дескрипторов, так что клиентский сокет закрывается ядром. При этом серверу посылается сегмент FIN, на который TCP сервера отвечает сегментом ACK. Это первая половина последовательности завершения работы соединения TCP. На этом этапе сокет сервера находится в состоянии CLOSE_WAIT, а клиентский сокет — в состоянии FIN_WAIT_2 (см. рис. 2.4 и 2.5).
4. Когда TCP сервера получает сегмент FIN, дочерний процесс сервера находится в состоянии ожидания в вызове функции
(см. листинг 5.2), а затем функцияreadвозвращает нуль. Это заставляет функциюreadвернуть управление функцииstr_echoдочернего процесса сервера.main5. Дочерний процесс сервера завершается с помощью вызова функции
(см. листинг 5.1).exit6. Все открытые дескрипторы в дочернем процессе сервера закрываются. Закрытие присоединенного сокета дочерним процессом вызывает отправку двух последних сегментов завершения соединения TCP: FIN от сервера клиенту и ACK от клиента (см. рис. 2.5). На этом этапе соединение полностью завершается. Клиентский сокет входит в состояние TIME_WAIT.
7. Другая часть завершения процесса относится к сигналу
. Он отправляется родительскому процессу, когда завершается дочерний процесс. Это происходит и в нашем примере, но мы не перехватываем данный сигнал в коде, и по умолчанию он игнорируется. Дочерний процесс входит в состояние зомби (zombie). Мы можем проверить это с помощью командыSIGCHLD.pslinux % <b>ps -t pts/6 -o pid,ppid,tty,stat,args,wchan</b>PID PPID TT STAT COMMAND WCHAN22038 22036 pts/6 S -bash read_chan17870 22038 pts/6 S ./tcpserv01 wait_for_connect19315 17870 pts/6 Z [tcpserv01 <defu do_exitТеперь дочерний процесс находится в состоянии
(зомби).ZПроцессы-зомби нужно своевременно удалять, а это требует работы с сигналами Unix. Поэтому в следующем разделе мы сделаем обзор управления сигналами, а затем продолжим рассмотрение нашего примера.
5.8. Обработка сигналов POSIX
Сигнал — это уведомление процесса о том, что произошло некое событие. Иногда сигналы называют программными прерываниями (software interrupts). Подразумевается, что процесс не знает заранее о том, когда придет сигнал.
Сигналы могут посылаться в следующих направлениях:
■ одним процессом другому процессу (или самому себе);
■ ядром процессу.
Сигнал
, упомянутый в конце предыдущего раздела, ядро посылает родительскому процессу при завершении дочернего.SIGCHLDДля каждого сигнала существует определенное действие (action или disposition — характер). Действие, соответствующее сигналу, задается с помощью вызова функции
(ее описание следует далее) и может быть выбрано тремя способами:sigaction1. Мы можем предоставить функцию, которая вызывается при перехвате определенного сигнала. Эта функция называется обработчиком сигнала (signal handler), а действие называется перехватыванием сигнала (catching). Сигналы
иSIGKILLперехватить нельзя. Наша функция вызывается с одним целочисленным аргументом, который является номером сигнала, и ничего не возвращает. Следовательно, прототип этой функции имеет вид:SIGSTOPvoid handler(int <i>signo</i>);Для большинства сигналов вызов функции
и задание функции, вызываемой при получении сигнала, — это все, что требуется для обработки сигнала. Но дальше вы увидите, что для перехватывания некоторых сигналов, в частностиsigaction,SIGIOиSIGPOLL, требуются дополнительные действия со стороны процесса.SIGURG2. Мы можем игнорировать сигнал, если действие задать как
. СигналыSIG_IGNиSIGKILLне могут быть проигнорированы.SIGSTOP3. Мы можем установить действие для сигнала по умолчанию, задав его как
. Действие сигнала по умолчанию обычно заключается в завершении процесса по получении сигнала, а некоторые сигналы генерируют копию области памяти процесса в его текущем каталоге (так называемый дамп — core dump). Есть несколько сигналов, для которых действием по умолчанию является игнорирование. Например,SIG_DFLиSIGCHLD(посылается по получении внеполосных данных, см. главу 24) — это два сигнала, игнорируемых по умолчанию, с которыми мы встретимся в тексте.SIGURGФункция signal
Согласно POSIX, чтобы определить действие для сигнала, нужно вызвать функцию
. Однако это достаточно сложно, поскольку один аргумент этой функции — это структура, для которой необходимо выделение памяти и заполнение. Поэтому проще задать действие сигнала с помощью функцииsigaction. Первый ее аргумент — это имя сигнала, а второй — либо указатель на функцию, либо одна из константsignalиSIG_IGN. Но функцияSIG_DFLсуществовала еще до появления POSIX.1, и ее различные реализации имеют разную семантику сигналов с целью обеспечения обратной совместимости. В то же время POSIX четко диктует семантику при вызове функцииsignal. Это обеспечивает простой интерфейс с соблюдением семантики POSIX. Мы включили эту функцию в нашу собственную библиотеку вместе функциямиsigactionи функциями-обертками, которые мы используем для построения всех наших программ. Она представлена в листинге 5.5. Функция-оберткаerr_<i>XXX</i>здесь не показана, потому что ее вид не зависит от того, какую именно функциюSignalона должна вызывать.signal