/c/c_socket/unix_network_programming_v1_3e/sock/loopudp.c

https://gitlab.com/peter__barnes/work · C · 210 lines · 163 code · 31 blank · 16 comment · 63 complexity · 4d88a8c84c4693f729650893e883fbb9 MD5 · raw file

  1. /*
  2. * Copyright (c) 1993 W. Richard Stevens. All rights reserved.
  3. * Permission to use or modify this software and its documentation only for
  4. * educational purposes and without fee is hereby granted, provided that
  5. * the above copyright notice appear in all copies. The author makes no
  6. * representations about the suitability of this software for any purpose.
  7. * It is provided "as is" without express or implied warranty.
  8. */
  9. #include "sock.h"
  10. /* Copy everything from stdin to "sockfd",
  11. * and everything from "sockfd" to stdout. */
  12. void
  13. loop_udp(int sockfd)
  14. {
  15. int maxfdp1, nread, ntowrite, stdineof,
  16. clilen, servlen, flags;
  17. fd_set rset;
  18. struct sockaddr_in cliaddr; /* for UDP server */
  19. struct sockaddr_in servaddr; /* for UDP client */
  20. #ifdef HAVE_MSGHDR_MSG_CONTROL
  21. struct iovec iov[1];
  22. struct msghdr msg;
  23. #ifdef IP_RECVDSTADDR /* 4.3BSD Reno and later */
  24. static struct cmsghdr *cmptr = NULL; /* malloc'ed */
  25. struct in_addr dstinaddr; /* for UDP server */
  26. #define CONTROLLEN (sizeof(struct cmsghdr) + sizeof(struct in_addr))
  27. #endif /* IP_RECVDSTADDR */
  28. #endif /* HAVE_MSGHDR_MSG_CONTROL */
  29. if (pauseinit)
  30. sleep_us(pauseinit*1000); /* intended for server */
  31. flags = 0;
  32. stdineof = 0;
  33. FD_ZERO(&rset);
  34. maxfdp1 = sockfd + 1; /* check descriptors [0..sockfd] */
  35. /* If UDP client issues connect(), recv() and write() are used.
  36. Server is harder since cannot issue connect(). We use recvfrom()
  37. or recvmsg(), depending on OS. */
  38. for ( ; ; ) {
  39. if (stdineof == 0)
  40. FD_SET(STDIN_FILENO, &rset);
  41. FD_SET(sockfd, &rset);
  42. if (select(maxfdp1, &rset, NULL, NULL, NULL) < 0)
  43. err_sys("select error");
  44. if (FD_ISSET(STDIN_FILENO, &rset)) { /* data to read on stdin */
  45. if ( (nread = read(STDIN_FILENO, rbuf, readlen)) < 0)
  46. err_sys("read error from stdin");
  47. else if (nread == 0) { /* EOF on stdin */
  48. if (halfclose) {
  49. if (shutdown(sockfd, SHUT_WR) < 0)
  50. err_sys("shutdown() error");
  51. FD_CLR(STDIN_FILENO, &rset);
  52. stdineof = 1; /* don't read stdin anymore */
  53. continue; /* back to select() */
  54. }
  55. break; /* default: stdin EOF -> done */
  56. }
  57. if (crlf) {
  58. ntowrite = crlf_add(wbuf, writelen, rbuf, nread);
  59. if (connectudp) {
  60. if (write(sockfd, wbuf, ntowrite) != ntowrite)
  61. err_sys("write error");
  62. } else {
  63. if (sendto(sockfd, wbuf, ntowrite, 0,
  64. (struct sockaddr *) &servaddr, sizeof(servaddr))
  65. != ntowrite)
  66. err_sys("sendto error");
  67. }
  68. } else {
  69. if (connectudp) {
  70. if (write(sockfd, rbuf, nread) != nread)
  71. err_sys("write error");
  72. } else {
  73. if (sendto(sockfd, rbuf, nread, 0,
  74. (struct sockaddr *) &servaddr, sizeof(servaddr))
  75. != nread)
  76. err_sys("sendto error");
  77. }
  78. }
  79. }
  80. if (FD_ISSET(sockfd, &rset)) { /* data to read from socket */
  81. if (server) {
  82. clilen = sizeof(cliaddr);
  83. #ifndef HAVE_MSGHDR_MSG_CONTROL /* vanilla BSD sockets */
  84. nread = recvfrom(sockfd, rbuf, readlen, 0,
  85. (struct sockaddr *) &cliaddr, &clilen);
  86. #else /* 4.3BSD Reno and later; use recvmsg() to get at MSG_TRUNC flag */
  87. /* Also lets us get at control information (destination address) */
  88. iov[0].iov_base = rbuf;
  89. iov[0].iov_len = readlen;
  90. msg.msg_iov = iov;
  91. msg.msg_iovlen = 1;
  92. msg.msg_name = (caddr_t) &cliaddr;
  93. msg.msg_namelen = clilen;
  94. #ifdef IP_RECVDSTADDR
  95. if (cmptr == NULL && (cmptr = malloc(CONTROLLEN)) == NULL)
  96. err_sys("malloc error for control buffer");
  97. msg.msg_control = (caddr_t) cmptr; /* for dest address */
  98. msg.msg_controllen = CONTROLLEN;
  99. #else
  100. msg.msg_control = (caddr_t) 0; /* no ancillary data */
  101. msg.msg_controllen = 0;
  102. #endif /* IP_RECVDSTADDR */
  103. msg.msg_flags = 0; /* flags returned here */
  104. nread = recvmsg(sockfd, &msg, 0);
  105. #endif /* HAVE_MSGHDR_MSG_CONTROL */
  106. if (nread < 0)
  107. err_sys("datagram receive error");
  108. if (verbose) {
  109. printf("from %s", INET_NTOA(cliaddr.sin_addr));
  110. #ifdef HAVE_MSGHDR_MSG_CONTROL
  111. #ifdef IP_RECVDSTADDR
  112. if (recvdstaddr) {
  113. if (cmptr->cmsg_len != CONTROLLEN)
  114. err_quit("control length (%d) != %d",
  115. cmptr->cmsg_len, CONTROLLEN);
  116. if (cmptr->cmsg_level != IPPROTO_IP)
  117. err_quit("control level != IPPROTO_IP");
  118. if (cmptr->cmsg_type != IP_RECVDSTADDR)
  119. err_quit("control type != IP_RECVDSTADDR");
  120. memcpy(&dstinaddr, CMSG_DATA(cmptr),
  121. sizeof(struct in_addr));
  122. bzero(cmptr, CONTROLLEN);
  123. printf(", to %s", INET_NTOA(dstinaddr));
  124. }
  125. #endif /* IP_RECVDSTADDR */
  126. #endif /* HAVE_MSGHDR_MSG_CONTROL */
  127. printf(": ");
  128. fflush(stdout);
  129. }
  130. #ifdef MSG_TRUNC
  131. if (msg.msg_flags & MSG_TRUNC)
  132. printf("(datagram truncated)\n");
  133. #endif
  134. } else if (connectudp) {
  135. /* msgpeek = 0 or MSG_PEEK */
  136. flags = msgpeek;
  137. oncemore:
  138. if ( (nread = recv(sockfd, rbuf, readlen, flags)) < 0)
  139. err_sys("recv error");
  140. else if (nread == 0) {
  141. if (verbose)
  142. fprintf(stderr, "connection closed by peer\n");
  143. break; /* EOF, terminate */
  144. }
  145. } else {
  146. /* Must use recvfrom() for unconnected UDP client */
  147. servlen = sizeof(servaddr);
  148. nread = recvfrom(sockfd, rbuf, readlen, 0,
  149. (struct sockaddr *) &servaddr, &servlen);
  150. if (nread < 0)
  151. err_sys("datagram recvfrom() error");
  152. if (verbose) {
  153. printf("from %s", INET_NTOA(servaddr.sin_addr));
  154. printf(": ");
  155. fflush(stdout);
  156. }
  157. }
  158. if (crlf) {
  159. ntowrite = crlf_strip(wbuf, writelen, rbuf, nread);
  160. if (writen(STDOUT_FILENO, wbuf, ntowrite) != ntowrite)
  161. err_sys("writen error to stdout");
  162. } else {
  163. if (writen(STDOUT_FILENO, rbuf, nread) != nread)
  164. err_sys("writen error to stdout");
  165. }
  166. if (flags != 0) {
  167. flags = 0; /* no infinite loop */
  168. goto oncemore; /* read the message again */
  169. }
  170. }
  171. }
  172. if (pauseclose) {
  173. if (verbose)
  174. fprintf(stderr, "pausing before close\n");
  175. sleep_us(pauseclose*1000);
  176. }
  177. if (close(sockfd) < 0)
  178. err_sys("close error"); /* since SO_LINGER may be set */
  179. }