10 struct icmpd_err icmpd_err;
11 struct sockaddr_un sun;
12 Sock_bind_wild(sockfd, pservaddr->sa_family);
13 icmpfd = Socket(AF_LOCAL, SOCK_STREAM, 0);
14 sun.sun_family = AF_LOCAL;
15 strcpy(sun.sun_path, ICMPD_PATH);
16 Connect(icmpfd, (SA*)&sun, sizeof(sun));
17 Write_fd(icmpfd, "1", 1, sockfd);
18 n = Read(icmpfd, recvline, 1);
19 if (n != 1 || recvline[0] != '1')
20 err_quit("error creating icmp socket, n = %d, char = %c",
21 n, recvline[0]);
22 FD_ZERO(&rset);
23 maxfdp1 = max(sockfd, icmpfd) + 1;
2-3
Аргументы функции те же, что и во всех ее предыдущих версиях.
12
Вызываем функцию sock_bind_wild
для связывания при помощи функции bind
универсального IP-адреса и динамически назначаемого порта с UDP-сокетом. Таким образом копия сокета, который пересылается демону, оказывается связана с портом, поскольку демону необходимо знать этот порт.
Демон также может произвести подобное связывание, если локальный порт не был связан с сокетом, который был передан демону, но это работает не во всех системах. В реализациях SVR4, таких как Solaris 2.5, сокеты не являются частью ядра, и когда один процесс связывает (bind) порт с совместно используемым сокетом, другой процесс при попытке использовать копию этого сокета получает ошибки. Простейшее решение — потребовать, чтобы приложение связывало локальный порт прежде, чем передавать сокет демону.
13-16
Мы создаем сокет семейства AF_INET
и подключаемся к известному имени сервера при помощи вызова connect
.
17-21
Вызываем функцию write_fd
, приведенную в листинге 15.11 для отправки копии UDP-сокета демону. Мы также посылаем одиночный байт данных — символ "1"
, поскольку некоторые реализации не передают дескриптор без данных. Демон посылает обратно одиночный байт данных, состоящий из символа "1"
, для обозначения успешного выполнения. Любой другой ответ означает ошибку.
22-23
Инициализируем набор дескрипторов и вычисляем первый аргумент для функции select
(максимальный из двух дескрипторов, увеличенный на единицу).
Вторая половина нашего клиента приведена в листинге 28.22. Это цикл, который считывает данные из стандартного ввода, посылает строку серверу, считывает ответ сервера и записывает ответ в стандартный вывод.
Листинг 28.22. Вторая часть приложения dg_cli
//icmpd/dgcli01.c
24 while (Fgets(sendline, MAXLINE, fp) != NULL) {
25 Sendto(sockfd, sendline, strlen(sendline), 0, pservaddr, servlen);
26 tv.tv_sec = 5;
27 tv.tv_usec = 0;
28 FD_SET(sockfd, &rset);
29 FD_SET(icmpfd, &rset);
30 if ((n = Select(maxfdp1, &rset, NULL, NULL, &tv)) == 0) {
31 fprintf(stderr, "socket timeout\n");
32 continue;
33 }
34 if (FD_ISSET(sockfd, &rset)) {
35 n = Recvfrom(sockfd, recvline, MAXLINE, 0, NULL, NULL);
36 recvline[n] = 0; /* завершающий нуль */
37 Fputs(recvline, stdout);
38 }
39 if (FD_ISSET(icmpfd, &rset)) {
40 if ((n = Read(icmpfd, &icmpd_err, sizeof(icmpd_err))) == 0)
41 err_quit("ICMP daemon terminated");
42 else if (n != sizeof(icmpd_err))