readable_v6.lc 6.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899
  1. /* include readable_v61 */
  2. #include "icmpd.h"## 1 ##src/icmpd/readable_v6.c##
  3. #include <netinet/in_systm.h>## 2 ##src/icmpd/readable_v6.c##
  4. #include <netinet/ip.h>## 3 ##src/icmpd/readable_v6.c##
  5. #include <netinet/ip_icmp.h>## 4 ##src/icmpd/readable_v6.c##
  6. #include <netinet/udp.h>## 5 ##src/icmpd/readable_v6.c##
  7. #ifdef IPV6## 6 ##src/icmpd/readable_v6.c##
  8. #include <netinet/ip6.h>## 7 ##src/icmpd/readable_v6.c##
  9. #include <netinet/icmp6.h>## 8 ##src/icmpd/readable_v6.c##
  10. #endif## 9 ##src/icmpd/readable_v6.c##
  11. int## 10 ##src/icmpd/readable_v6.c##
  12. readable_v6(void)## 11 ##src/icmpd/readable_v6.c##
  13. {## 12 ##src/icmpd/readable_v6.c##
  14. #ifdef IPV6## 13 ##src/icmpd/readable_v6.c##
  15. int i, hlen1, hlen2, icmp6len, sport;## 14 ##src/icmpd/readable_v6.c##
  16. char buf[MAXLINE];## 15 ##src/icmpd/readable_v6.c##
  17. char srcstr[INET6_ADDRSTRLEN], dststr[INET6_ADDRSTRLEN];## 16 ##src/icmpd/readable_v6.c##
  18. ssize_t n;## 17 ##src/icmpd/readable_v6.c##
  19. socklen_t len;## 18 ##src/icmpd/readable_v6.c##
  20. struct ip6_hdr *ip6, *hip6;## 19 ##src/icmpd/readable_v6.c##
  21. struct icmp6_hdr *icmp6;## 20 ##src/icmpd/readable_v6.c##
  22. struct udphdr *udp;## 21 ##src/icmpd/readable_v6.c##
  23. struct sockaddr_in6 from, dest;## 22 ##src/icmpd/readable_v6.c##
  24. struct icmpd_err icmpd_err;## 23 ##src/icmpd/readable_v6.c##
  25. len = sizeof(from);## 24 ##src/icmpd/readable_v6.c##
  26. n = Recvfrom(fd6, buf, MAXLINE, 0, (SA *) &from, &len);## 25 ##src/icmpd/readable_v6.c##
  27. printf("%d bytes ICMPv6 from %s:", n, Sock_ntop_host((SA *) &from, len));## 26 ##src/icmpd/readable_v6.c##
  28. ip6 = (struct ip6_hdr *) buf; /* start of IPv6 header */## 27 ##src/icmpd/readable_v6.c##
  29. hlen1 = sizeof(struct ip6_hdr);## 28 ##src/icmpd/readable_v6.c##
  30. if (ip6->ip6_nxt != IPPROTO_ICMPV6)## 29 ##src/icmpd/readable_v6.c##
  31. err_quit("next header not IPPROTO_ICMPV6");## 30 ##src/icmpd/readable_v6.c##
  32. icmp6 = (struct icmp6_hdr *) (buf + hlen1);## 31 ##src/icmpd/readable_v6.c##
  33. if ((icmp6len = n - hlen1) < 8)## 32 ##src/icmpd/readable_v6.c##
  34. err_quit("icmp6len (%d) < 8", icmp6len);## 33 ##src/icmpd/readable_v6.c##
  35. printf(" type = %d, code = %d\n", icmp6->icmp6_type, icmp6->icmp6_code);## 34 ##src/icmpd/readable_v6.c##
  36. /* end readable_v61 */
  37. /* include readable_v62 */
  38. if (icmp6->icmp6_type == ICMP6_DST_UNREACH ||## 35 ##src/icmpd/readable_v6.c##
  39. icmp6->icmp6_type == ICMP6_PACKET_TOO_BIG ||## 36 ##src/icmpd/readable_v6.c##
  40. icmp6->icmp6_type == ICMP6_TIME_EXCEEDED) {## 37 ##src/icmpd/readable_v6.c##
  41. if (icmp6len < 8 + 40 + 8)## 38 ##src/icmpd/readable_v6.c##
  42. err_quit("icmp6len (%d) < 8 + 40 + 8", icmp6len);## 39 ##src/icmpd/readable_v6.c##
  43. hip6 = (struct ip6_hdr *) (buf + hlen1 + 8);## 40 ##src/icmpd/readable_v6.c##
  44. hlen2 = sizeof(struct ip6_hdr);## 41 ##src/icmpd/readable_v6.c##
  45. printf("\tsrcip = %s, dstip = %s, next hdr = %d\n",## 42 ##src/icmpd/readable_v6.c##
  46. Inet_ntop(AF_INET6, &hip6->ip6_src, srcstr, sizeof(srcstr)),## 43 ##src/icmpd/readable_v6.c##
  47. Inet_ntop(AF_INET6, &hip6->ip6_dst, dststr, sizeof(dststr)),## 44 ##src/icmpd/readable_v6.c##
  48. hip6->ip6_nxt);## 45 ##src/icmpd/readable_v6.c##
  49. if (hip6->ip6_nxt == IPPROTO_UDP) {## 46 ##src/icmpd/readable_v6.c##
  50. udp = (struct udphdr *) (buf + hlen1 + 8 + hlen2);## 47 ##src/icmpd/readable_v6.c##
  51. sport = udp->uh_sport;## 48 ##src/icmpd/readable_v6.c##
  52. /* 4find client's Unix domain socket, send headers */## 49 ##src/icmpd/readable_v6.c##
  53. for (i = 0; i <= maxi; i++) {## 50 ##src/icmpd/readable_v6.c##
  54. if (client[i].connfd >= 0 &&## 51 ##src/icmpd/readable_v6.c##
  55. client[i].family == AF_INET6 &&## 52 ##src/icmpd/readable_v6.c##
  56. client[i].lport == sport) {## 53 ##src/icmpd/readable_v6.c##
  57. bzero(&dest, sizeof(dest));## 54 ##src/icmpd/readable_v6.c##
  58. dest.sin6_family = AF_INET6;## 55 ##src/icmpd/readable_v6.c##
  59. #ifdef HAVE_SOCKADDR_SA_LEN## 56 ##src/icmpd/readable_v6.c##
  60. dest.sin6_len = sizeof(dest);## 57 ##src/icmpd/readable_v6.c##
  61. #endif## 58 ##src/icmpd/readable_v6.c##
  62. memcpy(&dest.sin6_addr, &hip6->ip6_dst,## 59 ##src/icmpd/readable_v6.c##
  63. sizeof(struct in6_addr));## 60 ##src/icmpd/readable_v6.c##
  64. dest.sin6_port = udp->uh_dport;## 61 ##src/icmpd/readable_v6.c##
  65. icmpd_err.icmpd_type = icmp6->icmp6_type;## 62 ##src/icmpd/readable_v6.c##
  66. icmpd_err.icmpd_code = icmp6->icmp6_code;## 63 ##src/icmpd/readable_v6.c##
  67. icmpd_err.icmpd_len = sizeof(struct sockaddr_in6);## 64 ##src/icmpd/readable_v6.c##
  68. memcpy(&icmpd_err.icmpd_dest, &dest, sizeof(dest));## 65 ##src/icmpd/readable_v6.c##
  69. /* 4convert type & code to reasonable errno value */## 66 ##src/icmpd/readable_v6.c##
  70. icmpd_err.icmpd_errno = EHOSTUNREACH; /* default */## 67 ##src/icmpd/readable_v6.c##
  71. if (icmp6->icmp6_type == ICMP6_DST_UNREACH) {## 68 ##src/icmpd/readable_v6.c##
  72. if (icmp6->icmp6_code == ICMP_UNREACH_PORT)## 69 ##src/icmpd/readable_v6.c##
  73. icmpd_err.icmpd_errno = ECONNREFUSED;## 70 ##src/icmpd/readable_v6.c##
  74. else if (icmp6->icmp6_code == ICMP_UNREACH_NEEDFRAG)## 71 ##src/icmpd/readable_v6.c##
  75. icmpd_err.icmpd_errno = EMSGSIZE;## 72 ##src/icmpd/readable_v6.c##
  76. }## 73 ##src/icmpd/readable_v6.c##
  77. Write(client[i].connfd, &icmpd_err, sizeof(icmpd_err));## 74 ##src/icmpd/readable_v6.c##
  78. }## 75 ##src/icmpd/readable_v6.c##
  79. }## 76 ##src/icmpd/readable_v6.c##
  80. }## 77 ##src/icmpd/readable_v6.c##
  81. }## 78 ##src/icmpd/readable_v6.c##
  82. return (--nready);## 79 ##src/icmpd/readable_v6.c##
  83. #endif## 80 ##src/icmpd/readable_v6.c##
  84. }## 81 ##src/icmpd/readable_v6.c##
  85. /* end readable_v62 */