recv_v4.lc 4.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475
  1. #include "trace.h"## 1 ##src/traceroute/recv_v4.c##
  2. /*## 2 ##src/traceroute/recv_v4.c##
  3. * Return: -3 on timeout## 3 ##src/traceroute/recv_v4.c##
  4. * -2 on ICMP time exceeded in transit (caller keeps going)## 4 ##src/traceroute/recv_v4.c##
  5. * -1 on ICMP port unreachable (caller is done)## 5 ##src/traceroute/recv_v4.c##
  6. * >= 0 return value is some other ICMP unreachable code## 6 ##src/traceroute/recv_v4.c##
  7. */## 7 ##src/traceroute/recv_v4.c##
  8. int## 8 ##src/traceroute/recv_v4.c##
  9. recv_v4(int seq, struct timeval *tv)## 9 ##src/traceroute/recv_v4.c##
  10. {## 10 ##src/traceroute/recv_v4.c##
  11. int hlen1, hlen2, icmplen;## 11 ##src/traceroute/recv_v4.c##
  12. socklen_t len;## 12 ##src/traceroute/recv_v4.c##
  13. ssize_t n;## 13 ##src/traceroute/recv_v4.c##
  14. struct ip *ip, *hip;## 14 ##src/traceroute/recv_v4.c##
  15. struct icmp *icmp;## 15 ##src/traceroute/recv_v4.c##
  16. struct udphdr *udp;## 16 ##src/traceroute/recv_v4.c##
  17. alarm(3);## 17 ##src/traceroute/recv_v4.c##
  18. for (;;) {## 18 ##src/traceroute/recv_v4.c##
  19. len = pr->salen;## 19 ##src/traceroute/recv_v4.c##
  20. n = recvfrom(recvfd, recvbuf, sizeof(recvbuf), 0, pr->sarecv, &len);## 20 ##src/traceroute/recv_v4.c##
  21. if (n < 0) {## 21 ##src/traceroute/recv_v4.c##
  22. if (errno == EINTR)## 22 ##src/traceroute/recv_v4.c##
  23. return (-3); /* alarm expired */## 23 ##src/traceroute/recv_v4.c##
  24. else## 24 ##src/traceroute/recv_v4.c##
  25. err_sys("recvfrom error");## 25 ##src/traceroute/recv_v4.c##
  26. }## 26 ##src/traceroute/recv_v4.c##
  27. Gettimeofday(tv, NULL); /* get time of packet arrival */## 27 ##src/traceroute/recv_v4.c##
  28. ip = (struct ip *) recvbuf; /* start of IP header */## 28 ##src/traceroute/recv_v4.c##
  29. hlen1 = ip->ip_hl << 2; /* length of IP header */## 29 ##src/traceroute/recv_v4.c##
  30. icmp = (struct icmp *) (recvbuf + hlen1); /* start of ICMP header */## 30 ##src/traceroute/recv_v4.c##
  31. if ((icmplen = n - hlen1) < 8)## 31 ##src/traceroute/recv_v4.c##
  32. err_quit("icmplen (%d) < 8", icmplen);## 32 ##src/traceroute/recv_v4.c##
  33. if (icmp->icmp_type == ICMP_TIMXCEED &&## 33 ##src/traceroute/recv_v4.c##
  34. icmp->icmp_code == ICMP_TIMXCEED_INTRANS) {## 34 ##src/traceroute/recv_v4.c##
  35. if (icmplen < 8 + 20 + 8)## 35 ##src/traceroute/recv_v4.c##
  36. err_quit("icmplen (%d) < 8 + 20 + 8", icmplen);## 36 ##src/traceroute/recv_v4.c##
  37. hip = (struct ip *) (recvbuf + hlen1 + 8);## 37 ##src/traceroute/recv_v4.c##
  38. hlen2 = hip->ip_hl << 2;## 38 ##src/traceroute/recv_v4.c##
  39. udp = (struct udphdr *) (recvbuf + hlen1 + 8 + hlen2);## 39 ##src/traceroute/recv_v4.c##
  40. if (hip->ip_p == IPPROTO_UDP &&## 40 ##src/traceroute/recv_v4.c##
  41. udp->uh_sport == htons(sport) &&## 41 ##src/traceroute/recv_v4.c##
  42. udp->uh_dport == htons(dport + seq))## 42 ##src/traceroute/recv_v4.c##
  43. return (-2); /* we hit an intermediate router */## 43 ##src/traceroute/recv_v4.c##
  44. } else if (icmp->icmp_type == ICMP_UNREACH) {## 44 ##src/traceroute/recv_v4.c##
  45. if (icmplen < 8 + 20 + 8)## 45 ##src/traceroute/recv_v4.c##
  46. err_quit("icmplen (%d) < 8 + 20 + 8", icmplen);## 46 ##src/traceroute/recv_v4.c##
  47. hip = (struct ip *) (recvbuf + hlen1 + 8);## 47 ##src/traceroute/recv_v4.c##
  48. hlen2 = hip->ip_hl << 2;## 48 ##src/traceroute/recv_v4.c##
  49. udp = (struct udphdr *) (recvbuf + hlen1 + 8 + hlen2);## 49 ##src/traceroute/recv_v4.c##
  50. if (hip->ip_p == IPPROTO_UDP &&## 50 ##src/traceroute/recv_v4.c##
  51. udp->uh_sport == htons(sport) &&## 51 ##src/traceroute/recv_v4.c##
  52. udp->uh_dport == htons(dport + seq)) {## 52 ##src/traceroute/recv_v4.c##
  53. if (icmp->icmp_code == ICMP_UNREACH_PORT)## 53 ##src/traceroute/recv_v4.c##
  54. return (-1); /* have reached destination */## 54 ##src/traceroute/recv_v4.c##
  55. else## 55 ##src/traceroute/recv_v4.c##
  56. return (icmp->icmp_code); /* 0, 1, 2, ... */## 56 ##src/traceroute/recv_v4.c##
  57. }## 57 ##src/traceroute/recv_v4.c##
  58. } else if (verbose) {## 58 ##src/traceroute/recv_v4.c##
  59. printf(" (from %s: type = %d, code = %d)\n",## 59 ##src/traceroute/recv_v4.c##
  60. Sock_ntop_host(pr->sarecv, pr->salen),## 60 ##src/traceroute/recv_v4.c##
  61. icmp->icmp_type, icmp->icmp_code);## 61 ##src/traceroute/recv_v4.c##
  62. }## 62 ##src/traceroute/recv_v4.c##
  63. /* Some other ICMP error, recvfrom() again */## 63 ##src/traceroute/recv_v4.c##
  64. }## 64 ##src/traceroute/recv_v4.c##
  65. }## 65 ##src/traceroute/recv_v4.c##