57 Setsockopt(sockfd, SOL_SOCKET, SO_REUSEADDR, on, sizeof(on));
58 wild = Malloc(salen);
59 memcpy(wild, sa, salen); /* копирует семейство и порт */
60 sock_set_wild(wild, salen);
61 Bind(sockfd, wild, salen);
62 printf("bound %s\n", Sock_ntop(wild, salen));
63 if ((pid = Fork) == 0) { /* дочерний процесс */
64 mydg_echo(sockfd, wild, salen);
65 exit(0); /* никогда не выполняется */
66 }
67 exit(0);
68 }
69 void
70 mydg_echo(int sockfd, SA *myaddr, socklen_t salen)
71 {
72 int n;
73 char mesg[MAXLINE];
74 socklen_t len;
75 struct sockaddr *cli;
76 cli = Malloc(salen);
77 for (;;) {
78 len = salen;
79 n = Recvfrom(sockfd, mesg, MAXLINE, 0, cli, len);
80 printf("child %d, datagram from %s",
81 getpid, Sock_ntop(cli, len));
82 printf(", to %s\n", Sock_ntop(myaddr, salen));
83 Sendto(sockfd, mesg, n, 0, cli, len),
84 }
85 }
Глава 24
24.1. Да, разница есть. В первом примере два байта отсылаются с единственным срочным указателем, который указывает на байт, следующий за
b
. Во втором же примере (вызываются две функции) сначала отсылается символ
a
с указателем срочности, который указывает на следующий за ним байт, а за этим сегментом следует еще один TCP-сегмент, содержащий символ
b
с другим указателем срочности, указывающим на следующий за ним байт.
24.2. В листинге Д.10 приведена версия программы с использованием функции
poll
.
Листинг Д.10. Версия программы из листинга 24.4, использующая функцию poll вместо функции select
//oob/tcprecv03p.c
1 #include "unp.h"
2 int
3 main(int argc, char **argv)
4 {
5 int listenfd, connfd, n, justreadoob = 0;
6 char buff[100];
7 struct pollfd pollfd[1];
8 if (argc == 2)
9 listenfd = Tcp_listen(NULL, argv[1], NULL);
10 else if (argc == 3)
11 listenfd = Tcp_listen(argv[1], argv[2], NULL);
12 else
13 err_quit("usage: tcprecv03p [ host ] port#");
14 connfd = Accept(listenfd, NULL, NULL);
15 pollfd[0].fd = connfd;
16 pollfd[0].events = POLLRDNORM;
17 for (;;) {
18 if (justreadoob == 0)
19 pollfd[0].events |= POLLRDBAND;
20 Poll(pollfd, 1, INFTIM);
21 if (pollfd[0].revents POLLRDBAND) {
22 n = Recv(connfd, buff, sizeof(buff) - 1, MSG_OOB);
23 buff[n] = 0; /* завершающий нуль */
24 printf("read %d OOB byte: %s\n", n, buff);
25 justreadoob = 1;
26 pollfd[0].events = ~POLLRDBAND; /* отключение бита */
27 }
28 if (pollfd[0].revents POLLRDNORM) {
29 if ((n = Read(connfd, buff, sizeof(buff) - 1)) == 0) {
30 printf("received EOF\n");
31 exit(0);
32 }
33 buff[n] = 0; /* завершающий нуль */
34 printf("read %d bytes %s\n", n, buff);
35 justreadoob = 0;
36 }
37 }
38 }
Глава 25
25.1. Нет, такая модификация приведет к ошибке. Проблема состоит в том, что
nqueue
уменьшается до того, как завершается обработка элемента массива
dg[iget]
, что позволяет обработчику сигналов считывать новую дейтаграмму в данный элемент массива.
Глава 26