readable_v4.lc 5.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990
  1. /* include readable_v41 */
  2. #include "icmpd.h"## 1 ##src/icmpd/readable_v4.c##
  3. #include <netinet/in_systm.h>## 2 ##src/icmpd/readable_v4.c##
  4. #include <netinet/ip.h>## 3 ##src/icmpd/readable_v4.c##
  5. #include <netinet/ip_icmp.h>## 4 ##src/icmpd/readable_v4.c##
  6. #include <netinet/udp.h>## 5 ##src/icmpd/readable_v4.c##
  7. int## 6 ##src/icmpd/readable_v4.c##
  8. readable_v4(void)## 7 ##src/icmpd/readable_v4.c##
  9. {## 8 ##src/icmpd/readable_v4.c##
  10. int i, hlen1, hlen2, icmplen, sport;## 9 ##src/icmpd/readable_v4.c##
  11. char buf[MAXLINE];## 10 ##src/icmpd/readable_v4.c##
  12. char srcstr[INET_ADDRSTRLEN], dststr[INET_ADDRSTRLEN];## 11 ##src/icmpd/readable_v4.c##
  13. ssize_t n;## 12 ##src/icmpd/readable_v4.c##
  14. socklen_t len;## 13 ##src/icmpd/readable_v4.c##
  15. struct ip *ip, *hip;## 14 ##src/icmpd/readable_v4.c##
  16. struct icmp *icmp;## 15 ##src/icmpd/readable_v4.c##
  17. struct udphdr *udp;## 16 ##src/icmpd/readable_v4.c##
  18. struct sockaddr_in from, dest;## 17 ##src/icmpd/readable_v4.c##
  19. struct icmpd_err icmpd_err;## 18 ##src/icmpd/readable_v4.c##
  20. len = sizeof(from);## 19 ##src/icmpd/readable_v4.c##
  21. n = Recvfrom(fd4, buf, MAXLINE, 0, (SA *) &from, &len);## 20 ##src/icmpd/readable_v4.c##
  22. printf("%d bytes ICMPv4 from %s:", n, Sock_ntop_host((SA *) &from, len));## 21 ##src/icmpd/readable_v4.c##
  23. ip = (struct ip *) buf; /* start of IP header */## 22 ##src/icmpd/readable_v4.c##
  24. hlen1 = ip->ip_hl << 2; /* length of IP header */## 23 ##src/icmpd/readable_v4.c##
  25. icmp = (struct icmp *) (buf + hlen1); /* start of ICMP header */## 24 ##src/icmpd/readable_v4.c##
  26. if ((icmplen = n - hlen1) < 8)## 25 ##src/icmpd/readable_v4.c##
  27. err_quit("icmplen (%d) < 8", icmplen);## 26 ##src/icmpd/readable_v4.c##
  28. printf(" type = %d, code = %d\n", icmp->icmp_type, icmp->icmp_code);## 27 ##src/icmpd/readable_v4.c##
  29. /* end readable_v41 */
  30. /* include readable_v42 */
  31. if (icmp->icmp_type == ICMP_UNREACH ||## 28 ##src/icmpd/readable_v4.c##
  32. icmp->icmp_type == ICMP_TIMXCEED ||## 29 ##src/icmpd/readable_v4.c##
  33. icmp->icmp_type == ICMP_SOURCEQUENCH) {## 30 ##src/icmpd/readable_v4.c##
  34. if (icmplen < 8 + 20 + 8)## 31 ##src/icmpd/readable_v4.c##
  35. err_quit("icmplen (%d) < 8 + 20 + 8", icmplen);## 32 ##src/icmpd/readable_v4.c##
  36. hip = (struct ip *) (buf + hlen1 + 8);## 33 ##src/icmpd/readable_v4.c##
  37. hlen2 = hip->ip_hl << 2;## 34 ##src/icmpd/readable_v4.c##
  38. printf("\tsrcip = %s, dstip = %s, proto = %d\n",## 35 ##src/icmpd/readable_v4.c##
  39. Inet_ntop(AF_INET, &hip->ip_src, srcstr, sizeof(srcstr)),## 36 ##src/icmpd/readable_v4.c##
  40. Inet_ntop(AF_INET, &hip->ip_dst, dststr, sizeof(dststr)),## 37 ##src/icmpd/readable_v4.c##
  41. hip->ip_p);## 38 ##src/icmpd/readable_v4.c##
  42. if (hip->ip_p == IPPROTO_UDP) {## 39 ##src/icmpd/readable_v4.c##
  43. udp = (struct udphdr *) (buf + hlen1 + 8 + hlen2);## 40 ##src/icmpd/readable_v4.c##
  44. sport = udp->uh_sport;## 41 ##src/icmpd/readable_v4.c##
  45. /* 4find client's Unix domain socket, send headers */## 42 ##src/icmpd/readable_v4.c##
  46. for (i = 0; i <= maxi; i++) {## 43 ##src/icmpd/readable_v4.c##
  47. if (client[i].connfd >= 0 &&## 44 ##src/icmpd/readable_v4.c##
  48. client[i].family == AF_INET &&## 45 ##src/icmpd/readable_v4.c##
  49. client[i].lport == sport) {## 46 ##src/icmpd/readable_v4.c##
  50. bzero(&dest, sizeof(dest));## 47 ##src/icmpd/readable_v4.c##
  51. dest.sin_family = AF_INET;## 48 ##src/icmpd/readable_v4.c##
  52. #ifdef HAVE_SOCKADDR_SA_LEN## 49 ##src/icmpd/readable_v4.c##
  53. dest.sin_len = sizeof(dest);## 50 ##src/icmpd/readable_v4.c##
  54. #endif## 51 ##src/icmpd/readable_v4.c##
  55. memcpy(&dest.sin_addr, &hip->ip_dst,## 52 ##src/icmpd/readable_v4.c##
  56. sizeof(struct in_addr));## 53 ##src/icmpd/readable_v4.c##
  57. dest.sin_port = udp->uh_dport;## 54 ##src/icmpd/readable_v4.c##
  58. icmpd_err.icmpd_type = icmp->icmp_type;## 55 ##src/icmpd/readable_v4.c##
  59. icmpd_err.icmpd_code = icmp->icmp_code;## 56 ##src/icmpd/readable_v4.c##
  60. icmpd_err.icmpd_len = sizeof(struct sockaddr_in);## 57 ##src/icmpd/readable_v4.c##
  61. memcpy(&icmpd_err.icmpd_dest, &dest, sizeof(dest));## 58 ##src/icmpd/readable_v4.c##
  62. /* 4convert type & code to reasonable errno value */## 59 ##src/icmpd/readable_v4.c##
  63. icmpd_err.icmpd_errno = EHOSTUNREACH; /* default */## 60 ##src/icmpd/readable_v4.c##
  64. if (icmp->icmp_type == ICMP_UNREACH) {## 61 ##src/icmpd/readable_v4.c##
  65. if (icmp->icmp_code == ICMP_UNREACH_PORT)## 62 ##src/icmpd/readable_v4.c##
  66. icmpd_err.icmpd_errno = ECONNREFUSED;## 63 ##src/icmpd/readable_v4.c##
  67. else if (icmp->icmp_code == ICMP_UNREACH_NEEDFRAG)## 64 ##src/icmpd/readable_v4.c##
  68. icmpd_err.icmpd_errno = EMSGSIZE;## 65 ##src/icmpd/readable_v4.c##
  69. }## 66 ##src/icmpd/readable_v4.c##
  70. Write(client[i].connfd, &icmpd_err, sizeof(icmpd_err));## 67 ##src/icmpd/readable_v4.c##
  71. }## 68 ##src/icmpd/readable_v4.c##
  72. }## 69 ##src/icmpd/readable_v4.c##
  73. }## 70 ##src/icmpd/readable_v4.c##
  74. }## 71 ##src/icmpd/readable_v4.c##
  75. return (--nready);## 72 ##src/icmpd/readable_v4.c##
  76. }## 73 ##src/icmpd/readable_v4.c##
  77. /* end readable_v42 */