|
Packit |
1034e0 |
/*
|
|
Packit |
1034e0 |
* tracepath.c
|
|
Packit |
1034e0 |
*
|
|
Packit |
1034e0 |
* This program is free software; you can redistribute it and/or
|
|
Packit |
1034e0 |
* modify it under the terms of the GNU General Public License
|
|
Packit |
1034e0 |
* as published by the Free Software Foundation; either version
|
|
Packit |
1034e0 |
* 2 of the License, or (at your option) any later version.
|
|
Packit |
1034e0 |
*
|
|
Packit |
1034e0 |
* Authors: Alexey Kuznetsov, <kuznet@ms2.inr.ac.ru>
|
|
Packit |
1034e0 |
*/
|
|
Packit |
1034e0 |
|
|
Packit |
1034e0 |
#include <stdio.h>
|
|
Packit |
1034e0 |
#include <stdlib.h>
|
|
Packit |
1034e0 |
#include <unistd.h>
|
|
Packit |
1034e0 |
#include <sys/socket.h>
|
|
Packit |
1034e0 |
#include <netinet/in.h>
|
|
Packit |
1034e0 |
#include <netinet/icmp6.h>
|
|
Packit |
1034e0 |
|
|
Packit |
1034e0 |
#include <linux/types.h>
|
|
Packit |
1034e0 |
#include <linux/errqueue.h>
|
|
Packit |
1034e0 |
#include <errno.h>
|
|
Packit |
1034e0 |
#include <string.h>
|
|
Packit |
1034e0 |
#include <netdb.h>
|
|
Packit |
1034e0 |
#include <limits.h>
|
|
Packit |
1034e0 |
#include <resolv.h>
|
|
Packit |
1034e0 |
#include <sys/time.h>
|
|
Packit |
1034e0 |
#include <sys/uio.h>
|
|
Packit |
1034e0 |
#include <arpa/inet.h>
|
|
Packit |
1034e0 |
|
|
Packit |
1034e0 |
#ifdef USE_IDN
|
|
Packit |
1034e0 |
#include <locale.h>
|
|
Packit |
1034e0 |
#define getnameinfo_flags NI_IDN
|
|
Packit |
1034e0 |
#else
|
|
Packit |
1034e0 |
#define getnameinfo_flags 0
|
|
Packit |
1034e0 |
#endif
|
|
Packit |
1034e0 |
|
|
Packit |
1034e0 |
#ifndef SOL_IPV6
|
|
Packit |
1034e0 |
#define SOL_IPV6 IPPROTO_IPV6
|
|
Packit |
1034e0 |
#endif
|
|
Packit |
1034e0 |
|
|
Packit |
1034e0 |
#ifndef IP_PMTUDISC_DO
|
|
Packit |
1034e0 |
#define IP_PMTUDISC_DO 3
|
|
Packit |
1034e0 |
#endif
|
|
Packit |
1034e0 |
#ifndef IPV6_PMTUDISC_DO
|
|
Packit |
1034e0 |
#define IPV6_PMTUDISC_DO 3
|
|
Packit |
1034e0 |
#endif
|
|
Packit |
1034e0 |
|
|
Packit |
1034e0 |
#define MAX_HOPS_LIMIT 255
|
|
Packit |
1034e0 |
#define MAX_HOPS_DEFAULT 30
|
|
Packit |
1034e0 |
|
|
Packit |
1034e0 |
struct hhistory
|
|
Packit |
1034e0 |
{
|
|
Packit |
1034e0 |
int hops;
|
|
Packit |
1034e0 |
struct timeval sendtime;
|
|
Packit |
1034e0 |
};
|
|
Packit |
1034e0 |
|
|
Packit |
1034e0 |
struct hhistory his[64];
|
|
Packit |
1034e0 |
int hisptr;
|
|
Packit |
1034e0 |
|
|
Packit |
1034e0 |
struct sockaddr_storage target;
|
|
Packit |
1034e0 |
socklen_t targetlen;
|
|
Packit |
1034e0 |
__u16 base_port;
|
|
Packit |
1034e0 |
int max_hops = MAX_HOPS_DEFAULT;
|
|
Packit |
1034e0 |
|
|
Packit |
1034e0 |
int overhead;
|
|
Packit |
1034e0 |
int mtu;
|
|
Packit |
1034e0 |
void *pktbuf;
|
|
Packit |
1034e0 |
int hops_to = -1;
|
|
Packit |
1034e0 |
int hops_from = -1;
|
|
Packit |
1034e0 |
int no_resolve = 0;
|
|
Packit |
1034e0 |
int show_both = 0;
|
|
Packit |
1034e0 |
int mapped;
|
|
Packit |
1034e0 |
|
|
Packit |
1034e0 |
#define HOST_COLUMN_SIZE 52
|
|
Packit |
1034e0 |
|
|
Packit |
1034e0 |
struct probehdr
|
|
Packit |
1034e0 |
{
|
|
Packit |
1034e0 |
__u32 ttl;
|
|
Packit |
1034e0 |
struct timeval tv;
|
|
Packit |
1034e0 |
};
|
|
Packit |
1034e0 |
|
|
Packit |
1034e0 |
void data_wait(int fd)
|
|
Packit |
1034e0 |
{
|
|
Packit |
1034e0 |
fd_set fds;
|
|
Packit |
1034e0 |
struct timeval tv;
|
|
Packit |
1034e0 |
FD_ZERO(&fds);
|
|
Packit |
1034e0 |
FD_SET(fd, &fds);
|
|
Packit |
1034e0 |
tv.tv_sec = 1;
|
|
Packit |
1034e0 |
tv.tv_usec = 0;
|
|
Packit |
1034e0 |
select(fd+1, &fds, NULL, NULL, &tv;;
|
|
Packit |
1034e0 |
}
|
|
Packit |
1034e0 |
|
|
Packit |
1034e0 |
void print_host(const char *a, const char *b, int both)
|
|
Packit |
1034e0 |
{
|
|
Packit |
1034e0 |
int plen;
|
|
Packit |
1034e0 |
plen = printf("%s", a);
|
|
Packit |
1034e0 |
if (both)
|
|
Packit |
1034e0 |
plen += printf(" (%s)", b);
|
|
Packit |
1034e0 |
if (plen >= HOST_COLUMN_SIZE)
|
|
Packit |
1034e0 |
plen = HOST_COLUMN_SIZE - 1;
|
|
Packit |
1034e0 |
printf("%*s", HOST_COLUMN_SIZE - plen, "");
|
|
Packit |
1034e0 |
}
|
|
Packit |
1034e0 |
|
|
Packit |
1034e0 |
int recverr(int fd, struct addrinfo *ai, int ttl)
|
|
Packit |
1034e0 |
{
|
|
Packit |
1034e0 |
int res;
|
|
Packit |
1034e0 |
struct probehdr rcvbuf;
|
|
Packit |
1034e0 |
char cbuf[512];
|
|
Packit |
1034e0 |
struct iovec iov;
|
|
Packit |
1034e0 |
struct msghdr msg;
|
|
Packit |
1034e0 |
struct cmsghdr *cmsg;
|
|
Packit |
1034e0 |
struct sock_extended_err *e;
|
|
Packit |
1034e0 |
struct sockaddr_storage addr;
|
|
Packit |
1034e0 |
struct timeval tv;
|
|
Packit |
1034e0 |
struct timeval *rettv;
|
|
Packit |
1034e0 |
int slot = 0;
|
|
Packit |
1034e0 |
int rethops;
|
|
Packit |
1034e0 |
int sndhops;
|
|
Packit |
1034e0 |
int progress = -1;
|
|
Packit |
1034e0 |
int broken_router;
|
|
Packit |
1034e0 |
char hnamebuf[NI_MAXHOST] = "";
|
|
Packit |
1034e0 |
|
|
Packit |
1034e0 |
restart:
|
|
Packit |
1034e0 |
memset(&rcvbuf, -1, sizeof(rcvbuf));
|
|
Packit |
1034e0 |
iov.iov_base = &rcvbuf;
|
|
Packit |
1034e0 |
iov.iov_len = sizeof(rcvbuf);
|
|
Packit |
1034e0 |
msg.msg_name = (__u8*)&addr;
|
|
Packit |
1034e0 |
msg.msg_namelen = sizeof(addr);
|
|
Packit |
1034e0 |
msg.msg_iov = &iov;
|
|
Packit |
1034e0 |
msg.msg_iovlen = 1;
|
|
Packit |
1034e0 |
msg.msg_flags = 0;
|
|
Packit |
1034e0 |
msg.msg_control = cbuf;
|
|
Packit |
1034e0 |
msg.msg_controllen = sizeof(cbuf);
|
|
Packit |
1034e0 |
|
|
Packit |
1034e0 |
gettimeofday(&tv, NULL);
|
|
Packit |
1034e0 |
res = recvmsg(fd, &msg, MSG_ERRQUEUE);
|
|
Packit |
1034e0 |
if (res < 0) {
|
|
Packit |
1034e0 |
if (errno == EAGAIN)
|
|
Packit |
1034e0 |
return progress;
|
|
Packit |
1034e0 |
goto restart;
|
|
Packit |
1034e0 |
}
|
|
Packit |
1034e0 |
|
|
Packit |
1034e0 |
progress = mtu;
|
|
Packit |
1034e0 |
|
|
Packit |
1034e0 |
rethops = -1;
|
|
Packit |
1034e0 |
sndhops = -1;
|
|
Packit |
1034e0 |
e = NULL;
|
|
Packit |
1034e0 |
rettv = NULL;
|
|
Packit |
1034e0 |
|
|
Packit |
1034e0 |
slot = -base_port;
|
|
Packit |
1034e0 |
switch (ai->ai_family) {
|
|
Packit |
1034e0 |
case AF_INET6:
|
|
Packit |
1034e0 |
slot += ntohs(((struct sockaddr_in6 *)&addr)->sin6_port);
|
|
Packit |
1034e0 |
break;
|
|
Packit |
1034e0 |
case AF_INET:
|
|
Packit |
1034e0 |
slot += ntohs(((struct sockaddr_in *)&addr)->sin_port);
|
|
Packit |
1034e0 |
break;
|
|
Packit |
1034e0 |
}
|
|
Packit |
1034e0 |
|
|
Packit |
1034e0 |
if (slot >= 0 && slot < 63 && his[slot].hops) {
|
|
Packit |
1034e0 |
sndhops = his[slot].hops;
|
|
Packit |
1034e0 |
rettv = &his[slot].sendtime;
|
|
Packit |
1034e0 |
his[slot].hops = 0;
|
|
Packit |
1034e0 |
}
|
|
Packit |
1034e0 |
broken_router = 0;
|
|
Packit |
1034e0 |
if (res == sizeof(rcvbuf)) {
|
|
Packit |
1034e0 |
if (rcvbuf.ttl == 0 || rcvbuf.tv.tv_sec == 0)
|
|
Packit |
1034e0 |
broken_router = 1;
|
|
Packit |
1034e0 |
else {
|
|
Packit |
1034e0 |
sndhops = rcvbuf.ttl;
|
|
Packit |
1034e0 |
rettv = &rcvbuf.tv;
|
|
Packit |
1034e0 |
}
|
|
Packit |
1034e0 |
}
|
|
Packit |
1034e0 |
|
|
Packit |
1034e0 |
for (cmsg = CMSG_FIRSTHDR(&msg;; cmsg; cmsg = CMSG_NXTHDR(&msg, cmsg)) {
|
|
Packit |
1034e0 |
switch (cmsg->cmsg_level) {
|
|
Packit |
1034e0 |
case SOL_IPV6:
|
|
Packit |
1034e0 |
switch(cmsg->cmsg_type) {
|
|
Packit |
1034e0 |
case IPV6_RECVERR:
|
|
Packit |
1034e0 |
e = (struct sock_extended_err *)CMSG_DATA(cmsg);
|
|
Packit |
1034e0 |
break;
|
|
Packit |
1034e0 |
case IPV6_HOPLIMIT:
|
|
Packit |
1034e0 |
#ifdef IPV6_2292HOPLIMIT
|
|
Packit |
1034e0 |
case IPV6_2292HOPLIMIT:
|
|
Packit |
1034e0 |
#endif
|
|
Packit |
1034e0 |
memcpy(&rethops, CMSG_DATA(cmsg), sizeof(rethops));
|
|
Packit |
1034e0 |
break;
|
|
Packit |
1034e0 |
default:
|
|
Packit |
1034e0 |
printf("cmsg6:%d\n ", cmsg->cmsg_type);
|
|
Packit |
1034e0 |
}
|
|
Packit |
1034e0 |
break;
|
|
Packit |
1034e0 |
case SOL_IP:
|
|
Packit |
1034e0 |
switch(cmsg->cmsg_type) {
|
|
Packit |
1034e0 |
case IP_RECVERR:
|
|
Packit |
1034e0 |
e = (struct sock_extended_err *)CMSG_DATA(cmsg);
|
|
Packit |
1034e0 |
break;
|
|
Packit |
1034e0 |
case IP_TTL:
|
|
Packit |
1034e0 |
rethops = *(__u8*)CMSG_DATA(cmsg);
|
|
Packit |
1034e0 |
break;
|
|
Packit |
1034e0 |
default:
|
|
Packit |
1034e0 |
printf("cmsg4:%d\n ", cmsg->cmsg_type);
|
|
Packit |
1034e0 |
}
|
|
Packit |
1034e0 |
}
|
|
Packit |
1034e0 |
}
|
|
Packit |
1034e0 |
if (e == NULL) {
|
|
Packit |
1034e0 |
printf("no info\n");
|
|
Packit |
1034e0 |
return 0;
|
|
Packit |
1034e0 |
}
|
|
Packit |
1034e0 |
if (e->ee_origin == SO_EE_ORIGIN_LOCAL)
|
|
Packit |
1034e0 |
printf("%2d?: %-32s ", ttl, "[LOCALHOST]");
|
|
Packit |
1034e0 |
else if (e->ee_origin == SO_EE_ORIGIN_ICMP6 ||
|
|
Packit |
1034e0 |
e->ee_origin == SO_EE_ORIGIN_ICMP) {
|
|
Packit |
1034e0 |
char abuf[NI_MAXHOST];
|
|
Packit |
1034e0 |
struct sockaddr *sa = (struct sockaddr *)(e + 1);
|
|
Packit |
1034e0 |
socklen_t salen;
|
|
Packit |
1034e0 |
|
|
Packit |
1034e0 |
if (sndhops>0)
|
|
Packit |
1034e0 |
printf("%2d: ", sndhops);
|
|
Packit |
1034e0 |
else
|
|
Packit |
1034e0 |
printf("%2d?: ", ttl);
|
|
Packit |
1034e0 |
|
|
Packit |
1034e0 |
switch (sa->sa_family) {
|
|
Packit |
1034e0 |
case AF_INET6:
|
|
Packit |
1034e0 |
salen = sizeof(struct sockaddr_in6);
|
|
Packit |
1034e0 |
break;
|
|
Packit |
1034e0 |
case AF_INET:
|
|
Packit |
1034e0 |
salen = sizeof(struct sockaddr_in);
|
|
Packit |
1034e0 |
break;
|
|
Packit |
1034e0 |
default:
|
|
Packit |
1034e0 |
salen = 0;
|
|
Packit |
1034e0 |
}
|
|
Packit |
1034e0 |
|
|
Packit |
1034e0 |
if (no_resolve || show_both) {
|
|
Packit |
1034e0 |
if (getnameinfo(sa, salen,
|
|
Packit |
1034e0 |
abuf, sizeof(abuf), NULL, 0,
|
|
Packit |
1034e0 |
NI_NUMERICHOST))
|
|
Packit |
1034e0 |
strcpy(abuf, "???");
|
|
Packit |
1034e0 |
} else
|
|
Packit |
1034e0 |
abuf[0] = 0;
|
|
Packit |
1034e0 |
|
|
Packit |
1034e0 |
if (!no_resolve || show_both) {
|
|
Packit |
1034e0 |
fflush(stdout);
|
|
Packit |
1034e0 |
if (getnameinfo(sa, salen, hnamebuf, sizeof hnamebuf, NULL, 0, getnameinfo_flags))
|
|
Packit |
1034e0 |
strcpy(hnamebuf, "???");
|
|
Packit |
1034e0 |
} else
|
|
Packit |
1034e0 |
hnamebuf[0] = 0;
|
|
Packit |
1034e0 |
|
|
Packit |
1034e0 |
if (no_resolve)
|
|
Packit |
1034e0 |
print_host(abuf, hnamebuf, show_both);
|
|
Packit |
1034e0 |
else
|
|
Packit |
1034e0 |
print_host(hnamebuf, abuf, show_both);
|
|
Packit |
1034e0 |
}
|
|
Packit |
1034e0 |
|
|
Packit |
1034e0 |
if (rettv) {
|
|
Packit |
1034e0 |
int diff = (tv.tv_sec-rettv->tv_sec)*1000000+(tv.tv_usec-rettv->tv_usec);
|
|
Packit |
1034e0 |
printf("%3d.%03dms ", diff/1000, diff%1000);
|
|
Packit |
1034e0 |
if (broken_router)
|
|
Packit |
1034e0 |
printf("(This broken router returned corrupted payload) ");
|
|
Packit |
1034e0 |
}
|
|
Packit |
1034e0 |
|
|
Packit |
1034e0 |
if (rethops<=64)
|
|
Packit |
1034e0 |
rethops = 65-rethops;
|
|
Packit |
1034e0 |
else if (rethops<=128)
|
|
Packit |
1034e0 |
rethops = 129-rethops;
|
|
Packit |
1034e0 |
else
|
|
Packit |
1034e0 |
rethops = 256-rethops;
|
|
Packit |
1034e0 |
|
|
Packit |
1034e0 |
switch (e->ee_errno) {
|
|
Packit |
1034e0 |
case ETIMEDOUT:
|
|
Packit |
1034e0 |
printf("\n");
|
|
Packit |
1034e0 |
break;
|
|
Packit |
1034e0 |
case EMSGSIZE:
|
|
Packit |
1034e0 |
printf("pmtu %d\n", e->ee_info);
|
|
Packit |
1034e0 |
mtu = e->ee_info;
|
|
Packit |
1034e0 |
progress = mtu;
|
|
Packit |
1034e0 |
break;
|
|
Packit |
1034e0 |
case ECONNREFUSED:
|
|
Packit |
1034e0 |
printf("reached\n");
|
|
Packit |
1034e0 |
hops_to = sndhops<0 ? ttl : sndhops;
|
|
Packit |
1034e0 |
hops_from = rethops;
|
|
Packit |
1034e0 |
return 0;
|
|
Packit |
1034e0 |
case EPROTO:
|
|
Packit |
1034e0 |
printf("!P\n");
|
|
Packit |
1034e0 |
return 0;
|
|
Packit |
1034e0 |
case EHOSTUNREACH:
|
|
Packit |
1034e0 |
if ((e->ee_origin == SO_EE_ORIGIN_ICMP &&
|
|
Packit |
1034e0 |
e->ee_type == 11 &&
|
|
Packit |
1034e0 |
e->ee_code == 0) ||
|
|
Packit |
1034e0 |
(e->ee_origin == SO_EE_ORIGIN_ICMP6 &&
|
|
Packit |
1034e0 |
e->ee_type == 3 &&
|
|
Packit |
1034e0 |
e->ee_code == 0)) {
|
|
Packit |
1034e0 |
if (rethops>=0) {
|
|
Packit |
1034e0 |
if (sndhops>=0 && rethops != sndhops)
|
|
Packit |
1034e0 |
printf("asymm %2d ", rethops);
|
|
Packit |
1034e0 |
else if (sndhops<0 && rethops != ttl)
|
|
Packit |
1034e0 |
printf("asymm %2d ", rethops);
|
|
Packit |
1034e0 |
}
|
|
Packit |
1034e0 |
printf("\n");
|
|
Packit |
1034e0 |
break;
|
|
Packit |
1034e0 |
}
|
|
Packit |
1034e0 |
printf("!H\n");
|
|
Packit |
1034e0 |
return 0;
|
|
Packit |
1034e0 |
case ENETUNREACH:
|
|
Packit |
1034e0 |
printf("!N\n");
|
|
Packit |
1034e0 |
return 0;
|
|
Packit |
1034e0 |
case EACCES:
|
|
Packit |
1034e0 |
printf("!A\n");
|
|
Packit |
1034e0 |
return 0;
|
|
Packit |
1034e0 |
default:
|
|
Packit |
1034e0 |
printf("\n");
|
|
Packit |
1034e0 |
errno = e->ee_errno;
|
|
Packit |
1034e0 |
perror("NET ERROR");
|
|
Packit |
1034e0 |
return 0;
|
|
Packit |
1034e0 |
}
|
|
Packit |
1034e0 |
goto restart;
|
|
Packit |
1034e0 |
}
|
|
Packit |
1034e0 |
|
|
Packit |
1034e0 |
int probe_ttl(int fd, struct addrinfo *ai, int ttl)
|
|
Packit |
1034e0 |
{
|
|
Packit |
1034e0 |
int i;
|
|
Packit |
1034e0 |
struct probehdr *hdr = pktbuf;
|
|
Packit |
1034e0 |
|
|
Packit |
1034e0 |
memset(pktbuf, 0, mtu);
|
|
Packit |
1034e0 |
restart:
|
|
Packit |
1034e0 |
for (i=0; i<10; i++) {
|
|
Packit |
1034e0 |
int res;
|
|
Packit |
1034e0 |
|
|
Packit |
1034e0 |
hdr->ttl = ttl;
|
|
Packit |
1034e0 |
switch (ai->ai_family) {
|
|
Packit |
1034e0 |
case AF_INET6:
|
|
Packit |
1034e0 |
((struct sockaddr_in6 *)&target)->sin6_port = htons(base_port + hisptr);
|
|
Packit |
1034e0 |
break;
|
|
Packit |
1034e0 |
case AF_INET:
|
|
Packit |
1034e0 |
((struct sockaddr_in *)&target)->sin_port = htons(base_port + hisptr);
|
|
Packit |
1034e0 |
break;
|
|
Packit |
1034e0 |
}
|
|
Packit |
1034e0 |
gettimeofday(&hdr->tv, NULL);
|
|
Packit |
1034e0 |
his[hisptr].hops = ttl;
|
|
Packit |
1034e0 |
his[hisptr].sendtime = hdr->tv;
|
|
Packit |
1034e0 |
if (sendto(fd, pktbuf, mtu-overhead, 0, (struct sockaddr *)&target, targetlen) > 0)
|
|
Packit |
1034e0 |
break;
|
|
Packit |
1034e0 |
res = recverr(fd, ai, ttl);
|
|
Packit |
1034e0 |
his[hisptr].hops = 0;
|
|
Packit |
1034e0 |
if (res==0)
|
|
Packit |
1034e0 |
return 0;
|
|
Packit |
1034e0 |
if (res > 0)
|
|
Packit |
1034e0 |
goto restart;
|
|
Packit |
1034e0 |
}
|
|
Packit |
1034e0 |
hisptr = (hisptr + 1) & 63;
|
|
Packit |
1034e0 |
|
|
Packit |
1034e0 |
if (i<10) {
|
|
Packit |
1034e0 |
data_wait(fd);
|
|
Packit |
1034e0 |
if (recv(fd, pktbuf, mtu, MSG_DONTWAIT) > 0) {
|
|
Packit |
1034e0 |
printf("%2d?: reply received 8)\n", ttl);
|
|
Packit |
1034e0 |
return 0;
|
|
Packit |
1034e0 |
}
|
|
Packit |
1034e0 |
return recverr(fd, ai, ttl);
|
|
Packit |
1034e0 |
}
|
|
Packit |
1034e0 |
|
|
Packit |
1034e0 |
printf("%2d: send failed\n", ttl);
|
|
Packit |
1034e0 |
return 0;
|
|
Packit |
1034e0 |
}
|
|
Packit |
1034e0 |
|
|
Packit |
1034e0 |
static void usage(void) __attribute((noreturn));
|
|
Packit |
1034e0 |
|
|
Packit |
1034e0 |
static void usage(void)
|
|
Packit |
1034e0 |
{
|
|
Packit |
1034e0 |
fprintf(stderr, "Usage: tracepath [-4] [-6] [-n] [-b] [-l <len>] [-p port] <destination>\n");
|
|
Packit |
1034e0 |
exit(-1);
|
|
Packit |
1034e0 |
}
|
|
Packit |
1034e0 |
|
|
Packit |
1034e0 |
|
|
Packit |
1034e0 |
int main(int argc, char **argv)
|
|
Packit |
1034e0 |
{
|
|
Packit |
1034e0 |
struct addrinfo hints = {
|
|
Packit |
1034e0 |
.ai_family = AF_UNSPEC,
|
|
Packit |
1034e0 |
.ai_socktype = SOCK_DGRAM,
|
|
Packit |
1034e0 |
.ai_protocol = IPPROTO_UDP,
|
|
Packit |
1034e0 |
#ifdef USE_IDN
|
|
Packit |
1034e0 |
.ai_flags = AI_IDN | AI_CANONNAME,
|
|
Packit |
1034e0 |
#endif
|
|
Packit |
1034e0 |
};
|
|
Packit |
1034e0 |
struct addrinfo *ai, *result;
|
|
Packit |
1034e0 |
int ch;
|
|
Packit |
1034e0 |
int status;
|
|
Packit |
1034e0 |
int fd;
|
|
Packit |
1034e0 |
int on;
|
|
Packit |
1034e0 |
int ttl;
|
|
Packit |
1034e0 |
char *p;
|
|
Packit |
1034e0 |
char pbuf[NI_MAXSERV];
|
|
Packit |
1034e0 |
|
|
Packit |
1034e0 |
#ifdef USE_IDN
|
|
Packit |
1034e0 |
setlocale(LC_ALL, "");
|
|
Packit |
1034e0 |
#endif
|
|
Packit |
1034e0 |
|
|
Packit |
1034e0 |
/* Support being called using `tracepath4` or `tracepath6` symlinks */
|
|
Packit |
1034e0 |
if (argv[0][strlen(argv[0])-1] == '4')
|
|
Packit |
1034e0 |
hints.ai_family = AF_INET;
|
|
Packit |
1034e0 |
else if (argv[0][strlen(argv[0])-1] == '6')
|
|
Packit |
1034e0 |
hints.ai_family = AF_INET6;
|
|
Packit |
1034e0 |
|
|
Packit |
1034e0 |
while ((ch = getopt(argc, argv, "46nbh?l:m:p:")) != EOF) {
|
|
Packit |
1034e0 |
switch(ch) {
|
|
Packit |
1034e0 |
case '4':
|
|
Packit |
1034e0 |
if (hints.ai_family != AF_UNSPEC) {
|
|
Packit |
1034e0 |
fprintf(stderr, "tracepath: Only one -4 or -6 option may be specified\n");
|
|
Packit |
1034e0 |
exit(2);
|
|
Packit |
1034e0 |
}
|
|
Packit |
1034e0 |
hints.ai_family = AF_INET;
|
|
Packit |
1034e0 |
break;
|
|
Packit |
1034e0 |
case '6':
|
|
Packit |
1034e0 |
if (hints.ai_family != AF_UNSPEC) {
|
|
Packit |
1034e0 |
fprintf(stderr, "tracepath: Only one -4 or -6 option may be specified\n");
|
|
Packit |
1034e0 |
exit(2);
|
|
Packit |
1034e0 |
}
|
|
Packit |
1034e0 |
hints.ai_family = AF_INET6;
|
|
Packit |
1034e0 |
break;
|
|
Packit |
1034e0 |
case 'n':
|
|
Packit |
1034e0 |
no_resolve = 1;
|
|
Packit |
1034e0 |
break;
|
|
Packit |
1034e0 |
case 'b':
|
|
Packit |
1034e0 |
show_both = 1;
|
|
Packit |
1034e0 |
break;
|
|
Packit |
1034e0 |
case 'l':
|
|
Packit |
1034e0 |
if ((mtu = atoi(optarg)) <= overhead) {
|
|
Packit |
1034e0 |
fprintf(stderr, "Error: pktlen must be > %d and <= %d.\n",
|
|
Packit |
1034e0 |
overhead, INT_MAX);
|
|
Packit |
1034e0 |
exit(1);
|
|
Packit |
1034e0 |
}
|
|
Packit |
1034e0 |
break;
|
|
Packit |
1034e0 |
case 'm':
|
|
Packit |
1034e0 |
max_hops = atoi(optarg);
|
|
Packit |
1034e0 |
if (max_hops < 0 || max_hops > MAX_HOPS_LIMIT) {
|
|
Packit |
1034e0 |
fprintf(stderr,
|
|
Packit |
1034e0 |
"Error: max hops must be 0 .. %d (inclusive).\n",
|
|
Packit |
1034e0 |
MAX_HOPS_LIMIT);
|
|
Packit |
1034e0 |
}
|
|
Packit |
1034e0 |
break;
|
|
Packit |
1034e0 |
case 'p':
|
|
Packit |
1034e0 |
base_port = atoi(optarg);
|
|
Packit |
1034e0 |
break;
|
|
Packit |
1034e0 |
default:
|
|
Packit |
1034e0 |
usage();
|
|
Packit |
1034e0 |
}
|
|
Packit |
1034e0 |
}
|
|
Packit |
1034e0 |
|
|
Packit |
1034e0 |
argc -= optind;
|
|
Packit |
1034e0 |
argv += optind;
|
|
Packit |
1034e0 |
|
|
Packit |
1034e0 |
if (argc != 1)
|
|
Packit |
1034e0 |
usage();
|
|
Packit |
1034e0 |
|
|
Packit |
1034e0 |
/* Backward compatiblity */
|
|
Packit |
1034e0 |
if (!base_port) {
|
|
Packit |
1034e0 |
p = strchr(argv[0], '/');
|
|
Packit |
1034e0 |
if (p) {
|
|
Packit |
1034e0 |
*p = 0;
|
|
Packit |
1034e0 |
base_port = atoi(p+1);
|
|
Packit |
1034e0 |
} else
|
|
Packit |
1034e0 |
base_port = 44444;
|
|
Packit |
1034e0 |
}
|
|
Packit |
1034e0 |
sprintf(pbuf, "%u", base_port);
|
|
Packit |
1034e0 |
|
|
Packit |
1034e0 |
status = getaddrinfo(argv[0], pbuf, &hints, &result);
|
|
Packit |
1034e0 |
if (status) {
|
|
Packit |
1034e0 |
fprintf(stderr, "tracepath: %s: %s\n", argv[0], gai_strerror(status));
|
|
Packit |
1034e0 |
exit(1);
|
|
Packit |
1034e0 |
}
|
|
Packit |
1034e0 |
|
|
Packit |
1034e0 |
fd = -1;
|
|
Packit |
1034e0 |
for (ai = result; ai; ai = ai->ai_next) {
|
|
Packit |
1034e0 |
if (ai->ai_family != AF_INET6 &&
|
|
Packit |
1034e0 |
ai->ai_family != AF_INET)
|
|
Packit |
1034e0 |
continue;
|
|
Packit |
1034e0 |
fd = socket(ai->ai_family, ai->ai_socktype, ai->ai_protocol);
|
|
Packit |
1034e0 |
if (fd < 0)
|
|
Packit |
1034e0 |
continue;
|
|
Packit |
1034e0 |
memcpy(&target, ai->ai_addr, sizeof(*ai->ai_addr));
|
|
Packit |
1034e0 |
targetlen = ai->ai_addrlen;
|
|
Packit |
1034e0 |
break;
|
|
Packit |
1034e0 |
}
|
|
Packit |
1034e0 |
if (fd < 0) {
|
|
Packit |
1034e0 |
perror("socket/connect");
|
|
Packit |
1034e0 |
exit(1);
|
|
Packit |
1034e0 |
}
|
|
Packit |
1034e0 |
|
|
Packit |
1034e0 |
switch (ai->ai_family) {
|
|
Packit |
1034e0 |
case AF_INET6:
|
|
Packit |
1034e0 |
overhead = 48;
|
|
Packit |
1034e0 |
if (!mtu)
|
|
Packit |
1034e0 |
mtu = 128000;
|
|
Packit |
1034e0 |
if (mtu <= overhead)
|
|
Packit |
1034e0 |
goto pktlen_error;
|
|
Packit |
1034e0 |
|
|
Packit |
1034e0 |
on = IPV6_PMTUDISC_DO;
|
|
Packit |
1034e0 |
if (setsockopt(fd, SOL_IPV6, IPV6_MTU_DISCOVER, &on, sizeof(on)) &&
|
|
Packit |
1034e0 |
(on = IPV6_PMTUDISC_DO,
|
|
Packit |
1034e0 |
setsockopt(fd, SOL_IPV6, IPV6_MTU_DISCOVER, &on, sizeof(on)))) {
|
|
Packit |
1034e0 |
perror("IPV6_MTU_DISCOVER");
|
|
Packit |
1034e0 |
exit(1);
|
|
Packit |
1034e0 |
}
|
|
Packit |
1034e0 |
on = 1;
|
|
Packit |
1034e0 |
if (setsockopt(fd, SOL_IPV6, IPV6_RECVERR, &on, sizeof(on))) {
|
|
Packit |
1034e0 |
perror("IPV6_RECVERR");
|
|
Packit |
1034e0 |
exit(1);
|
|
Packit |
1034e0 |
}
|
|
Packit |
1034e0 |
if (
|
|
Packit |
1034e0 |
#ifdef IPV6_RECVHOPLIMIT
|
|
Packit |
1034e0 |
setsockopt(fd, SOL_IPV6, IPV6_HOPLIMIT, &on, sizeof(on)) &&
|
|
Packit |
1034e0 |
setsockopt(fd, SOL_IPV6, IPV6_2292HOPLIMIT, &on, sizeof(on))
|
|
Packit |
1034e0 |
#else
|
|
Packit |
1034e0 |
setsockopt(fd, SOL_IPV6, IPV6_HOPLIMIT, &on, sizeof(on))
|
|
Packit |
1034e0 |
#endif
|
|
Packit |
1034e0 |
) {
|
|
Packit |
1034e0 |
perror("IPV6_HOPLIMIT");
|
|
Packit |
1034e0 |
exit(1);
|
|
Packit |
1034e0 |
}
|
|
Packit |
1034e0 |
if (!IN6_IS_ADDR_V4MAPPED(&(((struct sockaddr_in6 *)&target)->sin6_addr)))
|
|
Packit |
1034e0 |
break;
|
|
Packit |
1034e0 |
mapped = 1;
|
|
Packit |
1034e0 |
/*FALLTHROUGH*/
|
|
Packit |
1034e0 |
case AF_INET:
|
|
Packit |
1034e0 |
overhead = 28;
|
|
Packit |
1034e0 |
if (!mtu)
|
|
Packit |
1034e0 |
mtu = 65535;
|
|
Packit |
1034e0 |
if (mtu <= overhead)
|
|
Packit |
1034e0 |
goto pktlen_error;
|
|
Packit |
1034e0 |
|
|
Packit |
1034e0 |
on = IP_PMTUDISC_DO;
|
|
Packit |
1034e0 |
if (setsockopt(fd, SOL_IP, IP_MTU_DISCOVER, &on, sizeof(on))) {
|
|
Packit |
1034e0 |
perror("IP_MTU_DISCOVER");
|
|
Packit |
1034e0 |
exit(1);
|
|
Packit |
1034e0 |
}
|
|
Packit |
1034e0 |
on = 1;
|
|
Packit |
1034e0 |
if (setsockopt(fd, SOL_IP, IP_RECVERR, &on, sizeof(on))) {
|
|
Packit |
1034e0 |
perror("IP_RECVERR");
|
|
Packit |
1034e0 |
exit(1);
|
|
Packit |
1034e0 |
}
|
|
Packit |
1034e0 |
if (setsockopt(fd, SOL_IP, IP_RECVTTL, &on, sizeof(on))) {
|
|
Packit |
1034e0 |
perror("IP_RECVTTL");
|
|
Packit |
1034e0 |
exit(1);
|
|
Packit |
1034e0 |
}
|
|
Packit |
1034e0 |
}
|
|
Packit |
1034e0 |
|
|
Packit |
1034e0 |
pktbuf = malloc(mtu);
|
|
Packit |
1034e0 |
if (!pktbuf) {
|
|
Packit |
1034e0 |
perror("malloc");
|
|
Packit |
1034e0 |
exit(1);
|
|
Packit |
1034e0 |
}
|
|
Packit |
1034e0 |
|
|
Packit |
1034e0 |
for (ttl = 1; ttl <= max_hops; ttl++) {
|
|
Packit |
1034e0 |
int res;
|
|
Packit |
1034e0 |
int i;
|
|
Packit |
1034e0 |
|
|
Packit |
1034e0 |
on = ttl;
|
|
Packit |
1034e0 |
switch (ai->ai_family) {
|
|
Packit |
1034e0 |
case AF_INET6:
|
|
Packit |
1034e0 |
if (setsockopt(fd, SOL_IPV6, IPV6_UNICAST_HOPS, &on, sizeof(on))) {
|
|
Packit |
1034e0 |
perror("IPV6_UNICAST_HOPS");
|
|
Packit |
1034e0 |
exit(1);
|
|
Packit |
1034e0 |
}
|
|
Packit |
1034e0 |
if (!mapped)
|
|
Packit |
1034e0 |
break;
|
|
Packit |
1034e0 |
/*FALLTHROUGH*/
|
|
Packit |
1034e0 |
case AF_INET:
|
|
Packit |
1034e0 |
if (setsockopt(fd, SOL_IP, IP_TTL, &on, sizeof(on))) {
|
|
Packit |
1034e0 |
perror("IP_TTL");
|
|
Packit |
1034e0 |
exit(1);
|
|
Packit |
1034e0 |
}
|
|
Packit |
1034e0 |
}
|
|
Packit |
1034e0 |
|
|
Packit |
1034e0 |
restart:
|
|
Packit |
1034e0 |
for (i=0; i<3; i++) {
|
|
Packit |
1034e0 |
int old_mtu;
|
|
Packit |
1034e0 |
|
|
Packit |
1034e0 |
old_mtu = mtu;
|
|
Packit |
1034e0 |
res = probe_ttl(fd, ai, ttl);
|
|
Packit |
1034e0 |
if (mtu != old_mtu)
|
|
Packit |
1034e0 |
goto restart;
|
|
Packit |
1034e0 |
if (res == 0)
|
|
Packit |
1034e0 |
goto done;
|
|
Packit |
1034e0 |
if (res > 0)
|
|
Packit |
1034e0 |
break;
|
|
Packit |
1034e0 |
}
|
|
Packit |
1034e0 |
|
|
Packit |
1034e0 |
if (res < 0)
|
|
Packit |
1034e0 |
printf("%2d: no reply\n", ttl);
|
|
Packit |
1034e0 |
}
|
|
Packit |
1034e0 |
printf(" Too many hops: pmtu %d\n", mtu);
|
|
Packit |
1034e0 |
|
|
Packit |
1034e0 |
done:
|
|
Packit |
1034e0 |
freeaddrinfo(result);
|
|
Packit |
1034e0 |
|
|
Packit |
1034e0 |
printf(" Resume: pmtu %d ", mtu);
|
|
Packit |
1034e0 |
if (hops_to>=0)
|
|
Packit |
1034e0 |
printf("hops %d ", hops_to);
|
|
Packit |
1034e0 |
if (hops_from>=0)
|
|
Packit |
1034e0 |
printf("back %d ", hops_from);
|
|
Packit |
1034e0 |
printf("\n");
|
|
Packit |
1034e0 |
exit(0);
|
|
Packit |
1034e0 |
|
|
Packit |
1034e0 |
pktlen_error:
|
|
Packit |
1034e0 |
fprintf(stderr, "Error: pktlen must be > %d and <= %d\n",
|
|
Packit |
1034e0 |
overhead, INT_MAX);
|
|
Packit |
1034e0 |
exit(1);
|
|
Packit |
1034e0 |
}
|