| 1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283 |
- #include "trace.h"
- extern int gotalarm;
- /*
- * Return: -3 on timeout
- * -2 on ICMP time exceeded in transit (caller keeps going)
- * -1 on ICMP port unreachable (caller is done)
- * >= 0 return value is some other ICMP unreachable code
- */
- int
- recv_v6(int seq, struct timeval *tv)
- {
- #ifdef IPV6
- int hlen2, icmp6len, ret;
- ssize_t n;
- socklen_t len;
- struct ip6_hdr *hip6;
- struct icmp6_hdr *icmp6;
- struct udphdr *udp;
- gotalarm = 0;
- alarm(3);
- for ( ; ; ) {
- if (gotalarm)
- return(-3); /* alarm expired */
- len = pr->salen;
- n = recvfrom(recvfd, recvbuf, sizeof(recvbuf), 0, pr->sarecv, &len);
- if (n < 0) {
- if (errno == EINTR)
- continue;
- else
- err_sys("recvfrom error");
- }
- icmp6 = (struct icmp6_hdr *) recvbuf; /* ICMP header */
- if ( ( icmp6len = n ) < 8)
- continue; /* not enough to look at ICMP header */
-
- if (icmp6->icmp6_type == ICMP6_TIME_EXCEEDED &&
- icmp6->icmp6_code == ICMP6_TIME_EXCEED_TRANSIT) {
- if (icmp6len < 8 + sizeof(struct ip6_hdr) + 4)
- continue; /* not enough data to look at inner header */
- hip6 = (struct ip6_hdr *) (recvbuf + 8);
- hlen2 = sizeof(struct ip6_hdr);
- udp = (struct udphdr *) (recvbuf + 8 + hlen2);
- if (hip6->ip6_nxt == IPPROTO_UDP &&
- udp->uh_sport == htons(sport) &&
- udp->uh_dport == htons(dport + seq))
- ret = -2; /* we hit an intermediate router */
- break;
- } else if (icmp6->icmp6_type == ICMP6_DST_UNREACH) {
- if (icmp6len < 8 + sizeof(struct ip6_hdr) + 4)
- continue; /* not enough data to look at inner header */
- hip6 = (struct ip6_hdr *) (recvbuf + 8);
- hlen2 = sizeof(struct ip6_hdr);
- udp = (struct udphdr *) (recvbuf + 8 + hlen2);
- if (hip6->ip6_nxt == IPPROTO_UDP &&
- udp->uh_sport == htons(sport) &&
- udp->uh_dport == htons(dport + seq)) {
- if (icmp6->icmp6_code == ICMP6_DST_UNREACH_NOPORT)
- ret = -1; /* have reached destination */
- else
- ret = icmp6->icmp6_code; /* 0, 1, 2, ... */
- break;
- }
- } else if (verbose) {
- printf(" (from %s: type = %d, code = %d)\n",
- Sock_ntop_host(pr->sarecv, pr->salen),
- icmp6->icmp6_type, icmp6->icmp6_code);
- }
- /* Some other ICMP error, recvfrom() again */
- }
- alarm(0); /* don't leave alarm running */
- Gettimeofday(tv, NULL); /* get time of packet arrival */
- return(ret);
- #endif
- }
|