UNIX: разработка сетевых приложений
Часть 115 из 399 Информация о книге
22 evnts.sctp_data_io_event = 1;23 Setsockopt(sock_fd, IPPROTO_SCTP, SCTP_EVENTS, &evnts, sizeof(evnts));24 Listen(sock_fd, LISTENQ);25 for (;;) {26 len = sizeof(struct sockaddr_in);27 rd_sz = Sctp_recvmsg(sock_fd, readbuf, sizeof(readbuf),28 (SA*)&cliaddr, &len, &sri, &msg_flags);29 if (stream_increment) {30 sri.sinfo_stream++;31 if (sri.sinfo_stream >=32 sctp_get_no_strms(sock_fd, (SA*)&cliaddr, len))33 sri.sinfo_stream = 0;34 }35 Sctp_sendmsg(sock_fd, readbuf, rd_sz,36 (SA*)&cliaddr, len,37 sri.sinfo_ppid,38 sri.sinfo_flags, sri.sinfo_stream, 0, 0);39 }40 }Настройка приращения номера потокаПо умолчанию наш сервер отвечает клиенту через поток, номер которого на единицу больше номера потока, по которому было получено сообщение. Если приложению в строке вызова передается целочисленный аргумент, он интерпретируется как значение флага13-14, с помощью которого приращение номера потока можно отключить. Мы воспользуемся этим параметром командной строки, когда будем говорить о блокировании в разделе 10.5.stream_incrementСоздание сокета SCTPСоздается сокет SCTP типа «один-ко-многим».15Связывание с адресомСтруктура адреса сокета Интернета заполняется универсальным адресом (16-20) и номером заранее известного порта сервераINADDR_ANY. Связывание с универсальным адресом означает, что конечная точка SCTP будет использовать все доступные локальные адреса для всех создаваемых ассоциаций. Для многоинтерфейсных узлов это означает, что удаленная конечная точка сможет устанавливать ассоциации и передавать пакеты на любой локальный интерфейс. Выбор номера порта SCTP основывался на рис. 2.10. Обратите внимание, что ход рассуждений для сервера тот же, что и в одном из предшествовавших примеров в разделе 5.2.SERV_PORTПодписка на уведомленияСервер изменяет параметры подписки на уведомления для сокета SCTP. Сервер подписывается только на событие21-23, что позволяет ему получать структуруsctp_data_io_event. По ее содержимому сервер сможет определять номер потока полученного сообщения.sctp_sndrcvinfoРазрешение установки входящих ассоциацийСервер разрешает устанавливать входящие ассоциации, вызывая функцию24. Затем управление передается главному циклу.listenОжидание сообщенияСервер инициализирует размер структуры адреса сокета клиента, после чего блокируется в ожидании сообщения от какого-либо удаленного собеседника.26-28Увеличение номера потокаСервер проверяет состояние флага29-34и определяет, нужно ли увеличивать номер потока. Если флаг установлен (никакие аргументы в командной строке не передавались), сервер увеличивает номер потока, по которому было получено сообщение, на единицу. Если полученное число достигает предельного количества потоков (получаемого вызовомstream_increment), сервер сбрасывает номер потока в 0. Функцияsctp_get_no_strmsв листинге не приведена. Она использует параметрsctp_get_no_strms(см. раздел 7.10) для определения согласованного количества потоков.SCTP_STATUSОтправка ответаСервер отсылает сообщения, используя идентификатор протокола, флаги и номер потока (который, возможно, был увеличен), хранящиеся в структуре35-38.sriЗаметьте, что нашему серверу не нужны уведомления об установке ассоциаций, поэтому он отключает все события, которые привели бы к передаче сообщений в буфер сокета. Сервер полагается на сведения из структуры
, а обратный адрес берет из переменнойsctp_sndrcvinfo. Этого оказывается достаточно для отправки эхо-ответа собеседнику через установленную им ассоциацию.cliaddrПрограмма работает до тех пор, пока пользователь не завершит ее передачей сигнала.
10.3. Потоковый эхо-клиент SCTP типа «один-ко-многим»: функция main
В листинге 10.2 приведена функция
нашего клиента SCTP.mainЛистинг 10.2. Потоковый эхо-клиент SCTP
//sctp/sctpclient01.c1 #include "unp.h"2 int3 main(int argc, char **argv)4 {5 int sock_fd;6 struct sockaddr_in servaddr;7 struct sctp_event_subscribe evnts;8 int echo_to_all=0;9 if (argc < 2)10 err_quit("Missing host argument - use '%s host [echo]'\n", argv[0]);11 if (argc > 2) {12 printf("Echoing messages to all streams\n");13 echo_to_all = 1;14 }15 sock_fd = Socket(AF_INET, SOCK_SEQPACKET, IPPROTO_SCTP);16 bzero(&servaddr, sizeof(servaddr));17 servaddr.sin_family = AF_INET;18 servaddr.sin_addr.s_addr = htonl(INADDR_ANY);19 servaddr.sin_port = htons(SERV_PORT);20 Inet_pton(AF_INET, argv[1], &servaddr.sin_addr);21 bzero(&evnts, sizeof(evnts));22 evnts.sctp_data_io_event = 1;