Blame tracepath.c

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