UNIX: разработка сетевых приложений - Страница 51
5.7. Нормальное завершение
На этом этапе соединение установлено, и все, что бы мы ни вводили на стороне клиента, отражается обратно.
linux % tcpcli01 127.0.0.1 эту строку мы показывали раньшеhello, world наш вводhello, world отраженная сервером строкаgood byegood bye^D Ctrl+D - наш завершающий символ для обозначения конца файлаМы вводим две строки, каждая из них отражается, затем мы вводим символ конца файла (EOF)
Ctrl+Dnetstatlinux % netstat -а | grep 9877tcp 0 0 *:9877 *:*tcp 0 0 local host:42758 localhost:9877Клиентская часть соединения (локальный порт 42 758) входит в состояние TIME_WAIT (см. раздел 2.6), и прослушивающий сервер все еще ждет подключения другого клиента. (В этот раз мы передаем вывод
netstatgrepПеречислим этапы нормального завершения работы нашего клиента и сервера.
1. Когда мы набираем символ EOF, функция
fgetsstr_cli2. Когда функция
str_climainexit3. При завершении процесса выполняется закрытие всех открытых дескрипторов, так что клиентский сокет закрывается ядром. При этом серверу посылается сегмент FIN, на который TCP сервера отвечает сегментом ACK. Это первая половина последовательности завершения работы соединения TCP. На этом этапе сокет сервера находится в состоянии CLOSE_WAIT, а клиентский сокет — в состоянии FIN_WAIT_2 (см. рис. 2.4 и 2.5).
4. Когда TCP сервера получает сегмент FIN, дочерний процесс сервера находится в состоянии ожидания в вызове функции
readreadstr_echomain5. Дочерний процесс сервера завершается с помощью вызова функции
exit6. Все открытые дескрипторы в дочернем процессе сервера закрываются. Закрытие присоединенного сокета дочерним процессом вызывает отправку двух последних сегментов завершения соединения TCP: FIN от сервера клиенту и ACK от клиента (см. рис. 2.5). На этом этапе соединение полностью завершается. Клиентский сокет входит в состояние TIME_WAIT.
7. Другая часть завершения процесса относится к сигналу
SIGCHLDpslinux % ps -t pts/6 -o pid,ppid,tty,stat,args,wchanPID 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). Сигналы
SIGKILLSIGSTOPvoid handler(int signo);Для большинства сигналов вызов функции
sigactionSIGIOSIGPOLLSIGURG2. Мы можем игнорировать сигнал, если действие задать как
SIG_IGNSIGKILLSIGSTOP3. Мы можем установить действие для сигнала по умолчанию, задав его как
SIG_DFLSIGCHLDSIGURGФункция signal
Согласно POSIX, чтобы определить действие для сигнала, нужно вызвать функцию
sigactionsignalSIG_IGNSIG_DFLsignalsigactionerr_XXXSignalsignal