xref: /netbsd-src/usr.sbin/traceroute/as.c (revision 64b9d7212f20f6c15c57a22007ea107730bc2ae4)
1*64b9d721Schristos /*	$NetBSD: as.c,v 1.4 2011/05/10 01:52:49 christos Exp $	*/
2f5d592cfSatatat 
3f5d592cfSatatat /*
4f5d592cfSatatat  * Copyright (c) 2001 The NetBSD Foundation, Inc.
5f5d592cfSatatat  * All rights reserved.
6f5d592cfSatatat  *
7f5d592cfSatatat  * This code is derived from software contributed to The NetBSD Foundation
8f5d592cfSatatat  * by Andrew Brown.
9f5d592cfSatatat  *
10f5d592cfSatatat  * Redistribution and use in source and binary forms, with or without
11f5d592cfSatatat  * modification, are permitted provided that the following conditions
12f5d592cfSatatat  * are met:
13f5d592cfSatatat  * 1. Redistributions of source code must retain the above copyright
14f5d592cfSatatat  *    notice, this list of conditions and the following disclaimer.
15f5d592cfSatatat  * 2. Redistributions in binary form must reproduce the above copyright
16f5d592cfSatatat  *    notice, this list of conditions and the following disclaimer in the
17f5d592cfSatatat  *    documentation and/or other materials provided with the distribution.
18f5d592cfSatatat  *
19f5d592cfSatatat  * THIS SOFTWARE IS PROVIDED BY THE NETBSD FOUNDATION, INC. AND CONTRIBUTORS
20f5d592cfSatatat  * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
21f5d592cfSatatat  * TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
22f5d592cfSatatat  * PURPOSE ARE DISCLAIMED.  IN NO EVENT SHALL THE FOUNDATION OR CONTRIBUTORS
23f5d592cfSatatat  * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
24f5d592cfSatatat  * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
25f5d592cfSatatat  * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
26f5d592cfSatatat  * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
27f5d592cfSatatat  * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
28f5d592cfSatatat  * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
29f5d592cfSatatat  * POSSIBILITY OF SUCH DAMAGE.
30f5d592cfSatatat  */
31f5d592cfSatatat 
32f5d592cfSatatat #include <sys/cdefs.h>
33f5d592cfSatatat #include <sys/types.h>
34f5d592cfSatatat #include <sys/socket.h>
35f5d592cfSatatat #include <netinet/in.h>
36f5d592cfSatatat #include <arpa/inet.h>
37f5d592cfSatatat #include <netdb.h>
38f5d592cfSatatat #include <unistd.h>
39f5d592cfSatatat #include <string.h>
40f5d592cfSatatat #include <stdlib.h>
41f5d592cfSatatat #include <errno.h>
42f5d592cfSatatat #include <err.h>
43f5d592cfSatatat #include <stdio.h>
44f5d592cfSatatat 
45f5d592cfSatatat #include "as.h"
46f5d592cfSatatat 
47f5d592cfSatatat #define DEFAULT_AS_SERVER "whois.radb.net"
48f5d592cfSatatat #undef AS_DEBUG_FILE
49f5d592cfSatatat 
50f5d592cfSatatat struct aslookup {
51f5d592cfSatatat 	FILE *as_f;
52f5d592cfSatatat #ifdef AS_DEBUG_FILE
53f5d592cfSatatat 	FILE *as_debug;
54f5d592cfSatatat #endif /* AS_DEBUG_FILE */
55f5d592cfSatatat };
56f5d592cfSatatat 
57f5d592cfSatatat void *
as_setup(const char * server)58*64b9d721Schristos as_setup(const char *server)
59f5d592cfSatatat {
60f5d592cfSatatat 	struct aslookup *asn;
61*64b9d721Schristos 	struct addrinfo hints, *res0, *res;
62f5d592cfSatatat 	FILE *f;
63*64b9d721Schristos 	int s, error;
64f5d592cfSatatat 
65*64b9d721Schristos 	s = -1;
66*64b9d721Schristos 	if (server == NULL)
67*64b9d721Schristos 		server = getenv("RA_SERVER");
68f5d592cfSatatat 	if (server == NULL)
69f5d592cfSatatat 		server = DEFAULT_AS_SERVER;
70f5d592cfSatatat 
71*64b9d721Schristos 	memset(&hints, 0, sizeof(hints));
72*64b9d721Schristos 	hints.ai_family = PF_UNSPEC;
73*64b9d721Schristos 	hints.ai_socktype = SOCK_STREAM;
74*64b9d721Schristos 	error = getaddrinfo(server, "whois", &hints, &res0);
75*64b9d721Schristos 	if (error == EAI_SERVICE) {
76f5d592cfSatatat 		warnx("warning: whois/tcp service not found");
77*64b9d721Schristos 		error = getaddrinfo(server, "43", &hints, &res0);
78*64b9d721Schristos 	}
79f5d592cfSatatat 
80*64b9d721Schristos 	if (error != 0) {
81*64b9d721Schristos 		warnx("%s: %s", server, gai_strerror(error));
82f5d592cfSatatat 		return (NULL);
83f5d592cfSatatat 	}
84f5d592cfSatatat 
85*64b9d721Schristos 	for (res = res0; res; res = res->ai_next) {
86*64b9d721Schristos 		s = socket(res->ai_family, res->ai_socktype, res->ai_protocol);
87*64b9d721Schristos 		if (s < 0)
88*64b9d721Schristos 			continue;
89*64b9d721Schristos 		if (connect(s, res->ai_addr, res->ai_addrlen) >= 0)
90f5d592cfSatatat 			break;
91f5d592cfSatatat 		close(s);
92f5d592cfSatatat 		s = -1;
93f5d592cfSatatat 	}
94*64b9d721Schristos 	freeaddrinfo(res0);
95*64b9d721Schristos 	if (s < 0) {
96f5d592cfSatatat 		warn("connect");
97f5d592cfSatatat 		return (NULL);
98f5d592cfSatatat 	}
99f5d592cfSatatat 
100f5d592cfSatatat 	f = fdopen(s, "r+");
101f5d592cfSatatat 	(void)fprintf(f, "!!\n");
102f5d592cfSatatat 	(void)fflush(f);
103f5d592cfSatatat 
104f5d592cfSatatat 	asn = malloc(sizeof(struct aslookup));
105f5d592cfSatatat 	if (asn == NULL)
106f5d592cfSatatat 		(void)fclose(f);
107f5d592cfSatatat 	else
108f5d592cfSatatat 		asn->as_f = f;
109f5d592cfSatatat 
110f5d592cfSatatat #ifdef AS_DEBUG_FILE
111ecde3f37Swiz 	if (asn) {
112f5d592cfSatatat 		asn->as_debug = fopen(AS_DEBUG_FILE, "w");
113f5d592cfSatatat 		if (asn->as_debug) {
114f5d592cfSatatat 			(void)fprintf(asn->as_debug, ">> !!\n");
115f5d592cfSatatat 			(void)fflush(asn->as_debug);
116f5d592cfSatatat 		}
117ecde3f37Swiz 	}
118f5d592cfSatatat #endif /* AS_DEBUG_FILE */
119f5d592cfSatatat 
120f5d592cfSatatat 	return (asn);
121f5d592cfSatatat }
122f5d592cfSatatat 
123*64b9d721Schristos unsigned int
as_lookup(void * _asn,char * addr,sa_family_t family)124*64b9d721Schristos as_lookup(void *_asn, char *addr, sa_family_t family)
125f5d592cfSatatat {
126f5d592cfSatatat 	struct aslookup *asn = _asn;
127f5d592cfSatatat 	char buf[1024];
128*64b9d721Schristos 	unsigned int as;
129*64b9d721Schristos 	int rc, dlen, plen;
130f5d592cfSatatat 
131*64b9d721Schristos 	as = 0;
132*64b9d721Schristos 	rc = dlen = 0;
133*64b9d721Schristos 	plen = (family == AF_INET6) ? 128 : 32;
134*64b9d721Schristos 	(void)fprintf(asn->as_f, "!r%s/%d,l\n", addr, plen);
135f5d592cfSatatat 	(void)fflush(asn->as_f);
136f5d592cfSatatat 
137f5d592cfSatatat #ifdef AS_DEBUG_FILE
138f5d592cfSatatat 	if (asn->as_debug) {
139*64b9d721Schristos 		(void)fprintf(asn->as_debug, ">> !r%s/%d,l\n", addr, plen);
140f5d592cfSatatat 		(void)fflush(asn->as_debug);
141f5d592cfSatatat 	}
142f5d592cfSatatat #endif /* AS_DEBUG_FILE */
143f5d592cfSatatat 
144f5d592cfSatatat 	while (fgets(buf, sizeof(buf), asn->as_f) != NULL) {
145f5d592cfSatatat 		buf[sizeof(buf) - 1] = '\0';
146f5d592cfSatatat 
147f5d592cfSatatat #ifdef AS_DEBUG_FILE
148f5d592cfSatatat 		if (asn->as_debug) {
149f5d592cfSatatat 			(void)fprintf(asn->as_debug, "<< %s", buf);
150f5d592cfSatatat 			(void)fflush(asn->as_debug);
151f5d592cfSatatat 		}
152f5d592cfSatatat #endif /* AS_DEBUG_FILE */
153f5d592cfSatatat 
154f5d592cfSatatat 		if (rc == 0) {
155f5d592cfSatatat 			rc = buf[0];
156f5d592cfSatatat 			switch (rc) {
157f5d592cfSatatat 			    case 'A':
158f5d592cfSatatat 				/* A - followed by # bytes of answer */
159f5d592cfSatatat 				sscanf(buf, "A%d\n", &dlen);
160f5d592cfSatatat #ifdef AS_DEBUG_FILE
161f5d592cfSatatat 				if (asn->as_debug) {
162f5d592cfSatatat 					(void)fprintf(asn->as_debug,
163f5d592cfSatatat 					     "dlen: %d\n", dlen);
164f5d592cfSatatat 					(void)fflush(asn->as_debug);
165f5d592cfSatatat 				}
166f5d592cfSatatat #endif /* AS_DEBUG_FILE */
167f5d592cfSatatat 				break;
168f5d592cfSatatat 			    case 'C':
169f5d592cfSatatat 			    case 'D':
170f5d592cfSatatat 			    case 'E':
171f5d592cfSatatat 			    case 'F':
172f5d592cfSatatat 				/* C - no data returned */
173f5d592cfSatatat 				/* D - key not found */
174f5d592cfSatatat 				/* E - multiple copies of key */
175f5d592cfSatatat 				/* F - some other error */
176f5d592cfSatatat 				break;
177f5d592cfSatatat 			}
178f5d592cfSatatat 			if (rc == 'A')
179f5d592cfSatatat 				/* skip to next input line */
180f5d592cfSatatat 				continue;
181f5d592cfSatatat 		}
182f5d592cfSatatat 
183f5d592cfSatatat 		if (dlen == 0)
184f5d592cfSatatat 			/* out of data, next char read is end code */
185f5d592cfSatatat 			rc = buf[0];
186f5d592cfSatatat 		if (rc != 'A')
187f5d592cfSatatat 			/* either an error off the bat, or a done code */
188f5d592cfSatatat 			break;
189f5d592cfSatatat 
190f5d592cfSatatat 		/* data received, thank you */
191f5d592cfSatatat 		dlen -= strlen(buf);
192f5d592cfSatatat 
193f5d592cfSatatat 		/* origin line is the interesting bit */
194f5d592cfSatatat 		if (as == 0 && strncasecmp(buf, "origin:", 7) == 0) {
195*64b9d721Schristos 			sscanf(buf + 7, " AS%u", &as);
196f5d592cfSatatat #ifdef AS_DEBUG_FILE
197f5d592cfSatatat 			if (asn->as_debug) {
198f5d592cfSatatat 				(void)fprintf(asn->as_debug, "as: %d\n", as);
199f5d592cfSatatat 				(void)fflush(asn->as_debug);
200f5d592cfSatatat 			}
201f5d592cfSatatat #endif /* AS_DEBUG_FILE */
202f5d592cfSatatat 		}
203f5d592cfSatatat 	}
204f5d592cfSatatat 
205f5d592cfSatatat 	return (as);
206f5d592cfSatatat }
207f5d592cfSatatat 
208f5d592cfSatatat void
as_shutdown(void * _asn)209*64b9d721Schristos as_shutdown(void *_asn)
210f5d592cfSatatat {
211f5d592cfSatatat 	struct aslookup *asn = _asn;
212f5d592cfSatatat 
213f5d592cfSatatat 	(void)fprintf(asn->as_f, "!q\n");
214f5d592cfSatatat 	(void)fclose(asn->as_f);
215f5d592cfSatatat 
216f5d592cfSatatat #ifdef AS_DEBUG_FILE
217f5d592cfSatatat 	if (asn->as_debug) {
218f5d592cfSatatat 		(void)fprintf(asn->as_debug, ">> !q\n");
219f5d592cfSatatat 		(void)fclose(asn->as_debug);
220f5d592cfSatatat 	}
221f5d592cfSatatat #endif /* AS_DEBUG_FILE */
222f5d592cfSatatat 
223f5d592cfSatatat 	free(asn);
224f5d592cfSatatat }
225