xref: /minix3/external/bsd/bind/dist/lib/bind9/getaddresses.c (revision 00b67f09dd46474d133c95011a48590a8e8f94c7)
1*00b67f09SDavid van Moolenbroek /*	$NetBSD: getaddresses.c,v 1.6 2015/07/08 17:28:58 christos Exp $	*/
2*00b67f09SDavid van Moolenbroek 
3*00b67f09SDavid van Moolenbroek /*
4*00b67f09SDavid van Moolenbroek  * Copyright (C) 2004, 2005, 2007, 2014, 2015  Internet Systems Consortium, Inc. ("ISC")
5*00b67f09SDavid van Moolenbroek  * Copyright (C) 2001, 2002  Internet Software Consortium.
6*00b67f09SDavid van Moolenbroek  *
7*00b67f09SDavid van Moolenbroek  * Permission to use, copy, modify, and/or distribute this software for any
8*00b67f09SDavid van Moolenbroek  * purpose with or without fee is hereby granted, provided that the above
9*00b67f09SDavid van Moolenbroek  * copyright notice and this permission notice appear in all copies.
10*00b67f09SDavid van Moolenbroek  *
11*00b67f09SDavid van Moolenbroek  * THE SOFTWARE IS PROVIDED "AS IS" AND ISC DISCLAIMS ALL WARRANTIES WITH
12*00b67f09SDavid van Moolenbroek  * REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY
13*00b67f09SDavid van Moolenbroek  * AND FITNESS.  IN NO EVENT SHALL ISC BE LIABLE FOR ANY SPECIAL, DIRECT,
14*00b67f09SDavid van Moolenbroek  * INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM
15*00b67f09SDavid van Moolenbroek  * LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE
16*00b67f09SDavid van Moolenbroek  * OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR
17*00b67f09SDavid van Moolenbroek  * PERFORMANCE OF THIS SOFTWARE.
18*00b67f09SDavid van Moolenbroek  */
19*00b67f09SDavid van Moolenbroek 
20*00b67f09SDavid van Moolenbroek /* Id: getaddresses.c,v 1.22 2007/06/19 23:47:16 tbox Exp  */
21*00b67f09SDavid van Moolenbroek 
22*00b67f09SDavid van Moolenbroek /*! \file */
23*00b67f09SDavid van Moolenbroek 
24*00b67f09SDavid van Moolenbroek #include <config.h>
25*00b67f09SDavid van Moolenbroek #include <string.h>
26*00b67f09SDavid van Moolenbroek 
27*00b67f09SDavid van Moolenbroek #include <isc/net.h>
28*00b67f09SDavid van Moolenbroek #include <isc/netaddr.h>
29*00b67f09SDavid van Moolenbroek #include <isc/netdb.h>
30*00b67f09SDavid van Moolenbroek #include <isc/netscope.h>
31*00b67f09SDavid van Moolenbroek #include <isc/result.h>
32*00b67f09SDavid van Moolenbroek #include <isc/sockaddr.h>
33*00b67f09SDavid van Moolenbroek #include <isc/util.h>
34*00b67f09SDavid van Moolenbroek 
35*00b67f09SDavid van Moolenbroek #include <bind9/getaddresses.h>
36*00b67f09SDavid van Moolenbroek 
37*00b67f09SDavid van Moolenbroek #ifdef HAVE_ADDRINFO
38*00b67f09SDavid van Moolenbroek #ifdef HAVE_GETADDRINFO
39*00b67f09SDavid van Moolenbroek #ifdef HAVE_GAISTRERROR
40*00b67f09SDavid van Moolenbroek #define USE_GETADDRINFO
41*00b67f09SDavid van Moolenbroek #endif
42*00b67f09SDavid van Moolenbroek #endif
43*00b67f09SDavid van Moolenbroek #endif
44*00b67f09SDavid van Moolenbroek 
45*00b67f09SDavid van Moolenbroek #ifndef USE_GETADDRINFO
46*00b67f09SDavid van Moolenbroek #ifndef ISC_PLATFORM_NONSTDHERRNO
47*00b67f09SDavid van Moolenbroek extern int h_errno;
48*00b67f09SDavid van Moolenbroek #endif
49*00b67f09SDavid van Moolenbroek #endif
50*00b67f09SDavid van Moolenbroek 
51*00b67f09SDavid van Moolenbroek isc_result_t
bind9_getaddresses(const char * hostname,in_port_t port,isc_sockaddr_t * addrs,int addrsize,int * addrcount)52*00b67f09SDavid van Moolenbroek bind9_getaddresses(const char *hostname, in_port_t port,
53*00b67f09SDavid van Moolenbroek 		   isc_sockaddr_t *addrs, int addrsize, int *addrcount)
54*00b67f09SDavid van Moolenbroek {
55*00b67f09SDavid van Moolenbroek 	struct in_addr in4;
56*00b67f09SDavid van Moolenbroek 	struct in6_addr in6;
57*00b67f09SDavid van Moolenbroek 	isc_boolean_t have_ipv4, have_ipv6;
58*00b67f09SDavid van Moolenbroek 	int i;
59*00b67f09SDavid van Moolenbroek 
60*00b67f09SDavid van Moolenbroek #ifdef USE_GETADDRINFO
61*00b67f09SDavid van Moolenbroek 	struct addrinfo *ai = NULL, *tmpai, hints;
62*00b67f09SDavid van Moolenbroek 	int result;
63*00b67f09SDavid van Moolenbroek #else
64*00b67f09SDavid van Moolenbroek 	struct hostent *he;
65*00b67f09SDavid van Moolenbroek #endif
66*00b67f09SDavid van Moolenbroek 
67*00b67f09SDavid van Moolenbroek 	REQUIRE(hostname != NULL);
68*00b67f09SDavid van Moolenbroek 	REQUIRE(addrs != NULL);
69*00b67f09SDavid van Moolenbroek 	REQUIRE(addrcount != NULL);
70*00b67f09SDavid van Moolenbroek 	REQUIRE(addrsize > 0);
71*00b67f09SDavid van Moolenbroek 
72*00b67f09SDavid van Moolenbroek 	have_ipv4 = ISC_TF((isc_net_probeipv4() == ISC_R_SUCCESS));
73*00b67f09SDavid van Moolenbroek 	have_ipv6 = ISC_TF((isc_net_probeipv6() == ISC_R_SUCCESS));
74*00b67f09SDavid van Moolenbroek 
75*00b67f09SDavid van Moolenbroek 	/*
76*00b67f09SDavid van Moolenbroek 	 * Try IPv4, then IPv6.  In order to handle the extended format
77*00b67f09SDavid van Moolenbroek 	 * for IPv6 scoped addresses (address%scope_ID), we'll use a local
78*00b67f09SDavid van Moolenbroek 	 * working buffer of 128 bytes.  The length is an ad-hoc value, but
79*00b67f09SDavid van Moolenbroek 	 * should be enough for this purpose; the buffer can contain a string
80*00b67f09SDavid van Moolenbroek 	 * of at least 80 bytes for scope_ID in addition to any IPv6 numeric
81*00b67f09SDavid van Moolenbroek 	 * addresses (up to 46 bytes), the delimiter character and the
82*00b67f09SDavid van Moolenbroek 	 * terminating NULL character.
83*00b67f09SDavid van Moolenbroek 	 */
84*00b67f09SDavid van Moolenbroek 	if (inet_pton(AF_INET, hostname, &in4) == 1) {
85*00b67f09SDavid van Moolenbroek 		if (have_ipv4)
86*00b67f09SDavid van Moolenbroek 			isc_sockaddr_fromin(&addrs[0], &in4, port);
87*00b67f09SDavid van Moolenbroek 		else
88*00b67f09SDavid van Moolenbroek 			isc_sockaddr_v6fromin(&addrs[0], &in4, port);
89*00b67f09SDavid van Moolenbroek 		*addrcount = 1;
90*00b67f09SDavid van Moolenbroek 		return (ISC_R_SUCCESS);
91*00b67f09SDavid van Moolenbroek 	} else if (strlen(hostname) <= 127U) {
92*00b67f09SDavid van Moolenbroek 		char tmpbuf[128], *d;
93*00b67f09SDavid van Moolenbroek 		isc_uint32_t zone = 0;
94*00b67f09SDavid van Moolenbroek 
95*00b67f09SDavid van Moolenbroek 		strcpy(tmpbuf, hostname);
96*00b67f09SDavid van Moolenbroek 		d = strchr(tmpbuf, '%');
97*00b67f09SDavid van Moolenbroek 		if (d != NULL)
98*00b67f09SDavid van Moolenbroek 			*d = '\0';
99*00b67f09SDavid van Moolenbroek 
100*00b67f09SDavid van Moolenbroek 		if (inet_pton(AF_INET6, tmpbuf, &in6) == 1) {
101*00b67f09SDavid van Moolenbroek 			isc_netaddr_t na;
102*00b67f09SDavid van Moolenbroek 
103*00b67f09SDavid van Moolenbroek 			if (!have_ipv6)
104*00b67f09SDavid van Moolenbroek 				return (ISC_R_FAMILYNOSUPPORT);
105*00b67f09SDavid van Moolenbroek 
106*00b67f09SDavid van Moolenbroek 			if (d != NULL) {
107*00b67f09SDavid van Moolenbroek #ifdef ISC_PLATFORM_HAVESCOPEID
108*00b67f09SDavid van Moolenbroek 				isc_result_t iresult;
109*00b67f09SDavid van Moolenbroek 
110*00b67f09SDavid van Moolenbroek 				iresult = isc_netscope_pton(AF_INET6, d + 1,
111*00b67f09SDavid van Moolenbroek 							    &in6, &zone);
112*00b67f09SDavid van Moolenbroek 
113*00b67f09SDavid van Moolenbroek 				if (iresult != ISC_R_SUCCESS)
114*00b67f09SDavid van Moolenbroek 					return (iresult);
115*00b67f09SDavid van Moolenbroek #else
116*00b67f09SDavid van Moolenbroek 				/*
117*00b67f09SDavid van Moolenbroek 				 * The extended format is specified while the
118*00b67f09SDavid van Moolenbroek 				 * system does not provide the ability to use
119*00b67f09SDavid van Moolenbroek 				 * it.	Throw an explicit error instead of
120*00b67f09SDavid van Moolenbroek 				 * ignoring the specified value.
121*00b67f09SDavid van Moolenbroek 				 */
122*00b67f09SDavid van Moolenbroek 				return (ISC_R_BADADDRESSFORM);
123*00b67f09SDavid van Moolenbroek #endif
124*00b67f09SDavid van Moolenbroek 			}
125*00b67f09SDavid van Moolenbroek 
126*00b67f09SDavid van Moolenbroek 			isc_netaddr_fromin6(&na, &in6);
127*00b67f09SDavid van Moolenbroek 			isc_netaddr_setzone(&na, zone);
128*00b67f09SDavid van Moolenbroek 			isc_sockaddr_fromnetaddr(&addrs[0],
129*00b67f09SDavid van Moolenbroek 						 (const isc_netaddr_t *)&na,
130*00b67f09SDavid van Moolenbroek 						 port);
131*00b67f09SDavid van Moolenbroek 
132*00b67f09SDavid van Moolenbroek 			*addrcount = 1;
133*00b67f09SDavid van Moolenbroek 			return (ISC_R_SUCCESS);
134*00b67f09SDavid van Moolenbroek 		}
135*00b67f09SDavid van Moolenbroek 	}
136*00b67f09SDavid van Moolenbroek #ifdef USE_GETADDRINFO
137*00b67f09SDavid van Moolenbroek 	memset(&hints, 0, sizeof(hints));
138*00b67f09SDavid van Moolenbroek 	if (!have_ipv6)
139*00b67f09SDavid van Moolenbroek 		hints.ai_family = PF_INET;
140*00b67f09SDavid van Moolenbroek 	else if (!have_ipv4)
141*00b67f09SDavid van Moolenbroek 		hints.ai_family = PF_INET6;
142*00b67f09SDavid van Moolenbroek 	else {
143*00b67f09SDavid van Moolenbroek 		hints.ai_family = PF_UNSPEC;
144*00b67f09SDavid van Moolenbroek #ifdef AI_ADDRCONFIG
145*00b67f09SDavid van Moolenbroek 		hints.ai_flags = AI_ADDRCONFIG;
146*00b67f09SDavid van Moolenbroek #endif
147*00b67f09SDavid van Moolenbroek 	}
148*00b67f09SDavid van Moolenbroek 	hints.ai_socktype = SOCK_STREAM;
149*00b67f09SDavid van Moolenbroek #ifdef AI_ADDRCONFIG
150*00b67f09SDavid van Moolenbroek  again:
151*00b67f09SDavid van Moolenbroek #endif
152*00b67f09SDavid van Moolenbroek 	result = getaddrinfo(hostname, NULL, &hints, &ai);
153*00b67f09SDavid van Moolenbroek 	switch (result) {
154*00b67f09SDavid van Moolenbroek 	case 0:
155*00b67f09SDavid van Moolenbroek 		break;
156*00b67f09SDavid van Moolenbroek 	case EAI_NONAME:
157*00b67f09SDavid van Moolenbroek #if defined(EAI_NODATA) && (EAI_NODATA != EAI_NONAME)
158*00b67f09SDavid van Moolenbroek 	case EAI_NODATA:
159*00b67f09SDavid van Moolenbroek #endif
160*00b67f09SDavid van Moolenbroek 		return (ISC_R_NOTFOUND);
161*00b67f09SDavid van Moolenbroek #ifdef AI_ADDRCONFIG
162*00b67f09SDavid van Moolenbroek 	case EAI_BADFLAGS:
163*00b67f09SDavid van Moolenbroek 		if ((hints.ai_flags & AI_ADDRCONFIG) != 0) {
164*00b67f09SDavid van Moolenbroek 			hints.ai_flags &= ~AI_ADDRCONFIG;
165*00b67f09SDavid van Moolenbroek 			goto again;
166*00b67f09SDavid van Moolenbroek 		}
167*00b67f09SDavid van Moolenbroek #endif
168*00b67f09SDavid van Moolenbroek 	default:
169*00b67f09SDavid van Moolenbroek 		return (ISC_R_FAILURE);
170*00b67f09SDavid van Moolenbroek 	}
171*00b67f09SDavid van Moolenbroek 	for (tmpai = ai, i = 0;
172*00b67f09SDavid van Moolenbroek 	     tmpai != NULL && i < addrsize;
173*00b67f09SDavid van Moolenbroek 	     tmpai = tmpai->ai_next)
174*00b67f09SDavid van Moolenbroek 	{
175*00b67f09SDavid van Moolenbroek 		if (tmpai->ai_family != AF_INET &&
176*00b67f09SDavid van Moolenbroek 		    tmpai->ai_family != AF_INET6)
177*00b67f09SDavid van Moolenbroek 			continue;
178*00b67f09SDavid van Moolenbroek 		if (tmpai->ai_family == AF_INET) {
179*00b67f09SDavid van Moolenbroek 			struct sockaddr_in *sin;
180*00b67f09SDavid van Moolenbroek 			sin = (struct sockaddr_in *)tmpai->ai_addr;
181*00b67f09SDavid van Moolenbroek 			isc_sockaddr_fromin(&addrs[i], &sin->sin_addr, port);
182*00b67f09SDavid van Moolenbroek 		} else {
183*00b67f09SDavid van Moolenbroek 			struct sockaddr_in6 *sin6;
184*00b67f09SDavid van Moolenbroek 			sin6 = (struct sockaddr_in6 *)tmpai->ai_addr;
185*00b67f09SDavid van Moolenbroek 			isc_sockaddr_fromin6(&addrs[i], &sin6->sin6_addr,
186*00b67f09SDavid van Moolenbroek 					     port);
187*00b67f09SDavid van Moolenbroek 		}
188*00b67f09SDavid van Moolenbroek 		i++;
189*00b67f09SDavid van Moolenbroek 
190*00b67f09SDavid van Moolenbroek 	}
191*00b67f09SDavid van Moolenbroek 	freeaddrinfo(ai);
192*00b67f09SDavid van Moolenbroek 	*addrcount = i;
193*00b67f09SDavid van Moolenbroek #else
194*00b67f09SDavid van Moolenbroek 	he = gethostbyname(hostname);
195*00b67f09SDavid van Moolenbroek 	if (he == NULL) {
196*00b67f09SDavid van Moolenbroek 		switch (h_errno) {
197*00b67f09SDavid van Moolenbroek 		case HOST_NOT_FOUND:
198*00b67f09SDavid van Moolenbroek #ifdef NO_DATA
199*00b67f09SDavid van Moolenbroek 		case NO_DATA:
200*00b67f09SDavid van Moolenbroek #endif
201*00b67f09SDavid van Moolenbroek #if defined(NO_ADDRESS) && (!defined(NO_DATA) || (NO_DATA != NO_ADDRESS))
202*00b67f09SDavid van Moolenbroek 		case NO_ADDRESS:
203*00b67f09SDavid van Moolenbroek #endif
204*00b67f09SDavid van Moolenbroek 			return (ISC_R_NOTFOUND);
205*00b67f09SDavid van Moolenbroek 		default:
206*00b67f09SDavid van Moolenbroek 			return (ISC_R_FAILURE);
207*00b67f09SDavid van Moolenbroek 		}
208*00b67f09SDavid van Moolenbroek 	}
209*00b67f09SDavid van Moolenbroek 	if (he->h_addrtype != AF_INET && he->h_addrtype != AF_INET6)
210*00b67f09SDavid van Moolenbroek 		return (ISC_R_NOTFOUND);
211*00b67f09SDavid van Moolenbroek 	for (i = 0; i < addrsize; i++) {
212*00b67f09SDavid van Moolenbroek 		if (he->h_addrtype == AF_INET) {
213*00b67f09SDavid van Moolenbroek 			struct in_addr *inp;
214*00b67f09SDavid van Moolenbroek 			inp = (struct in_addr *)(he->h_addr_list[i]);
215*00b67f09SDavid van Moolenbroek 			if (inp == NULL)
216*00b67f09SDavid van Moolenbroek 				break;
217*00b67f09SDavid van Moolenbroek 			isc_sockaddr_fromin(&addrs[i], inp, port);
218*00b67f09SDavid van Moolenbroek 		} else {
219*00b67f09SDavid van Moolenbroek 			struct in6_addr *in6p;
220*00b67f09SDavid van Moolenbroek 			in6p = (struct in6_addr *)(he->h_addr_list[i]);
221*00b67f09SDavid van Moolenbroek 			if (in6p == NULL)
222*00b67f09SDavid van Moolenbroek 				break;
223*00b67f09SDavid van Moolenbroek 			isc_sockaddr_fromin6(&addrs[i], in6p, port);
224*00b67f09SDavid van Moolenbroek 		}
225*00b67f09SDavid van Moolenbroek 	}
226*00b67f09SDavid van Moolenbroek 	*addrcount = i;
227*00b67f09SDavid van Moolenbroek #endif
228*00b67f09SDavid van Moolenbroek 	if (*addrcount == 0)
229*00b67f09SDavid van Moolenbroek 		return (ISC_R_NOTFOUND);
230*00b67f09SDavid van Moolenbroek 	else
231*00b67f09SDavid van Moolenbroek 		return (ISC_R_SUCCESS);
232*00b67f09SDavid van Moolenbroek }
233