/* $KAME: getaddrinfo.c,v 1.14 2001/01/06 09:41:15 jinmei Exp $ */
/*
* Copyright (C) 1995, 1996, 1997, and 1998 WIDE Project.
* All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met:
* 1. Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* 2. Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in the
* 3. Neither the name of the project nor the names of its contributors
* may be used to endorse or promote products derived from this software
* without specific prior written permission.
*
* THIS SOFTWARE IS PROVIDED BY THE PROJECT AND CONTRIBUTORS ``AS IS'' AND
* ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
* ARE DISCLAIMED. IN NO EVENT SHALL THE PROJECT OR CONTRIBUTORS BE LIABLE
* FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
* DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
* OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
* HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
* LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
* OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
* SUCH DAMAGE.
*/
/*! \file
* Issues to be discussed:
*\li Thread safe-ness must be checked.
*\li Return values. There are nonstandard return values defined and used
* in the source code. This is because RFC2553 is silent about which error
* code must be returned for which situation.
*\li IPv4 classful (shortened) form. RFC2553 is silent about it. XNET 5.2
* says to use inet_aton() to convert IPv4 numeric to binary (allows
* classful form as a result).
* current code - disallow classful form for IPv4 (due to use of inet_pton).
*\li freeaddrinfo(NULL). RFC2553 is silent about it. XNET 5.2 says it is
* invalid.
* current code - SEGV on freeaddrinfo(NULL)
* Note:
*\li We use getipnodebyname() just for thread-safeness. There's no intent
* to let it do PF_UNSPEC (actually we never pass PF_UNSPEC to
* getipnodebyname().
*\li The code filters out AFs that are not supported by the kernel,
* when globbing NULL hostname (to loopback, or wildcard). Is it the right
* thing to do? What is the relationship with post-RFC2553 AI_ADDRCONFIG
* in ai_flags?
*\li (post-2553) semantics of AI_ADDRCONFIG itself is too vague.
* (1) what should we do against numeric hostname (2) what should we do
* against NULL hostname (3) what is AI_ADDRCONFIG itself. AF not ready?
* non-loopback address configured? global address configured?
* \par Additional Issue:
* To avoid search order issue, we have a big amount of code duplicate
* from gethnamaddr.c and some other places. The issues that there's no
* lower layer function to lookup "IPv4 or IPv6" record. Calling
* gethostbyname2 from getaddrinfo will end up in wrong search order, as
* follows:
* \li The code makes use of following calls when asked to resolver with
* ai_family = PF_UNSPEC:
*\code getipnodebyname(host, AF_INET6);
* getipnodebyname(host, AF_INET);
*\endcode
* \li This will result in the following queries if the node is configure to
*\code
* lookup DNS for IPv6 address
* lookup DNS for IPv4 address
*\endcode
* which may not meet people's requirement.
* \li The right thing to happen is to have underlying layer which does
* PF_UNSPEC lookup (lookup both) and return chain of addrinfos.
* This would result in a bit of code duplicate with _dns_ghbyname() and
* friends.
*/
#include "port_before.h"
#include <netdb.h>
#include <resolv.h>
#include <string.h>
#include <stdlib.h>
#include <stddef.h>
#include <ctype.h>
#include <unistd.h>
#include <stdio.h>
#include <errno.h>
#include <stdarg.h>
#include <irs.h>
#include <isc/assertions.h>
#include "port_after.h"
#include "irs_data.h"
#define SUCCESS 0
#define ANY 0
#define NO 0
static const char in_addrany[] = { 0, 0, 0, 0 };
static const char in6_addrany[] = {
0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0
};
static const char in6_loopback[] = {
0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 1
};
static const struct afd {
int a_af;
int a_addrlen;
int a_socklen;
int a_off;
const char *a_addrany;
const char *a_loopback;
int a_scoped;
} afdl [] = {
sizeof(struct sockaddr_in6),
sizeof(struct sockaddr_in),
in_addrany, in_loopback, 0},
};
struct explore {
int e_af;
int e_socktype;
int e_protocol;
const char *e_protostr;
int e_wild;
};
#if 0
#endif
{ -1, 0, 0, NULL, 0 },
};
static int str_isnumber __P((const char *));
const char *, struct addrinfo **));
struct addrinfo **));
const char *, struct addrinfo **));
const char *, struct addrinfo **));
const char *, struct addrinfo **));
struct addrinfo *, const char *));
const struct afd *, const char *));
static int addrconfig __P((int));
const struct addrinfo *));
const char *));
#if 0
static const char *ai_errlist[] = {
"Success",
"Address family for hostname not supported", /*%< EAI_ADDRFAMILY */
"Temporary failure in name resolution", /*%< EAI_AGAIN */
"Invalid value for ai_flags", /*%< EAI_BADFLAGS */
"Non-recoverable failure in name resolution", /*%< EAI_FAIL */
"ai_family not supported", /*%< EAI_FAMILY */
"Memory allocation failure", /*%< EAI_MEMORY */
"No address associated with hostname", /*%< EAI_NODATA */
"hostname nor servname provided, or not known", /*%< EAI_NONAME */
"servname not supported for ai_socktype", /*%< EAI_SERVICE */
"ai_socktype not supported", /*%< EAI_SOCKTYPE */
"System error returned in errno", /*%< EAI_SYSTEM */
"Invalid value for hints", /*%< EAI_BADHINTS */
"Resolved protocol is unknown", /*%< EAI_PROTOCOL */
"Unknown error", /*%< EAI_MAX */
};
#endif
/* XXX macros that make external reference is BAD. */
do { \
/* external reference: pai, error, and label free */ \
error = EAI_MEMORY; \
goto free; \
} \
} while (/*CONSTCOND*/0)
do { \
/* external reference: error and label free */ \
if (error != 0) \
goto free; \
} while (/*CONSTCOND*/0)
do { \
/* external reference: pai, error and label free */ \
if (error != 0) \
goto free; \
} while (/*CONSTCOND*/0)
#ifndef SOLARIS2
do { \
/* external reference: error, and label bad */ \
goto bad; \
/*NOTREACHED*/ \
} while (/*CONSTCOND*/0)
#else
do { \
/* external reference: error, and label bad */ \
goto bad; \
} while (/*CONSTCOND*/0)
#endif
#define MATCH_FAMILY(x, y, w) \
#define MATCH(x, y, w) \
#if 0 /*%< bind8 has its own version */
char *
int ecode;
{
return ai_errlist[ecode];
}
#endif
void
{
do {
if (ai->ai_canonname)
/* no need to free(ai->ai_addr) */
} while (ai);
}
static int
str_isnumber(p)
const char *p;
{
char *ep;
if (*p == '\0')
return NO;
errno = 0;
return YES;
else
return NO;
}
int
{
int error = 0;
/*
* clear _ai_pad to preserve binary
* compatibility with previously compiled 64-bit
* applications in a pre-SUSv3 environment by
* guaranteeing the upper 32-bits are empty.
*/
#endif
pai->ai_addrlen = 0;
return EAI_NONAME;
if (hints) {
/* error check for hints */
case PF_UNSPEC:
case PF_INET:
case PF_INET6:
break;
default:
}
/*
* We need to clear _ai_pad to preserve binary
* compatibility. See prior comment.
*/
#endif
/*
* are meaningful combination.
*/
continue;
continue;
continue;
}
}
}
}
/*
* post-2553: AI_ALL and AI_V4MAPPED are effective only against
* AF_INET6 query. They needs to be ignored if specified in other
* occassions.
*/
case AI_V4MAPPED:
case AI_ALL | AI_V4MAPPED:
break;
case AI_ALL:
#if 1
/* illegal */
#else
break;
#endif
}
/*
* check for special cases. (1) numeric servname is disallowed if
* for raw and other inet{,6} sockets.
*/
#ifdef PF_INET6
#endif
) {
#ifdef PF_INET6
#else
#endif
}
if (error)
}
/* NULL hostname, or numeric hostname */
continue;
continue;
continue;
/*
*/
continue;
/*
* filter out AFs that are not supported by the kernel
* XXX errno?
*/
continue;
} else
if (error)
goto free;
}
/*
* XXX
* If numreic representation of AF1 can be interpreted as FQDN
* representation of AF2, we need to think again about the code below.
*/
goto good;
/*
* hostname as alphabetical name.
* We'll make sure that
* - if returning addrinfo list is empty, return non-zero error
* value (already known one or EAI_NONAME).
* - otherwise,
* + if we haven't had any errors, return 0 (i.e. success).
* + if we've had an error, free the list and return the error.
* without any assumption on the behavior of explore_fqdn().
*/
/* first, try to query DNS for all possible address families. */
if (error) {
goto free;
}
goto free;
}
/*
* we would like to prefer AF_INET6 than AF_INET, so we'll make an
* outer loop by AFs.
*/
continue;
WILD_SOCKTYPE(ex))) {
continue;
}
WILD_PROTOCOL(ex))) {
continue;
}
#ifdef AI_ADDRCONFIG
/*
* If AI_ADDRCONFIG is specified, check if we are
* expected to return the address family or not.
*/
continue;
#endif
/*
*/
continue;
goto free;
}
}
good:
return(SUCCESS);
} else {
/*
* All the process succeeded, but we've had an empty list.
* This can happen if the given hints do not match our
* candidates.
*/
error = EAI_NONAME;
}
free:
bad:
return(error);
}
/*%
* FQDN hostname, DNS lookup
*/
static int
const char *hostname;
const char *servname;
{
int error = 0;
const char *cp;
/*
*/
return(0);
return(0);
#if 0 /*%< XXX (notyet) */
}
#endif
if (!net_data->ho_stayopen) {
}
int e = h_errno;
switch(e) {
case NETDB_INTERNAL:
error = EAI_SYSTEM;
break;
case TRY_AGAIN:
break;
case NO_RECOVERY:
break;
case HOST_NOT_FOUND:
case NO_DATA:
error = EAI_NONAME;
break;
default:
case NETDB_SUCCESS: /*%< should be impossible... */
error = EAI_NONAME;
break;
}
goto free;
}
/* canonname should already be filled. */
}
return(0);
free:
if (result)
return error;
}
static int
{
int error;
error = 0;
continue;
error = EAI_MEMORY;
goto fail;
}
}
return 0;
fail:
return error;
}
/*%
* hostname == NULL.
* passive socket -> anyaddr (0.0.0.0 or ::)
* non-passive socket -> localhost (127.0.0.1 or ::1)
*/
static int
const char *servname;
{
int error;
return 0;
/* xxx meaningless?
* GET_CANONNAME(cur->ai_next, "anyaddr");
*/
} else {
/* xxx meaningless?
* GET_CANONNAME(cur->ai_next, "localhost");
*/
}
return 0;
free:
return error;
}
/*%
* numeric hostname
*/
static int
const char *hostname;
const char *servname;
{
int error;
return 0;
case AF_INET:
} else
}
break;
#endif
default:
} else
}
break;
}
return 0;
free:
bad:
return error;
}
/*%
* numeric hostname with scope
*/
static int
const char *hostname;
const char *servname;
{
#ifndef SCOPE_DELIMITER
#else
int error;
struct sockaddr_in6 *sin6;
return 0;
/*
* Handle special case of <scoped_address><delimiter><scope id>
*/
return EAI_MEMORY;
/* terminate at the delimiter */
if (error == 0) {
continue;
return(EAI_NONAME); /*%< XXX: is return OK? */
}
#ifdef HAVE_SIN6_SCOPE_ID
#endif
}
}
return error;
#endif
}
static int
const char *str;
{
return EAI_MEMORY;
}
return 0;
}
static struct addrinfo *
const char *addr;
{
char *p;
return NULL;
#ifdef HAVE_SA_LEN
#endif
return ai;
}
/* XXX need to malloc() the same way we do from other functions! */
static struct addrinfo *
{
size_t l;
return NULL;
if (pai->ai_canonname) {
return NULL;
}
} else {
/* just to make sure */
}
return ai;
}
static int
/* get_port does not touch first argument. when matchonly == 1. */
/* LINTED const cast */
}
static int
const char *proto;
int port;
int allownumeric;
return 0;
case AF_INET:
#ifdef AF_INET6
case AF_INET6:
#endif
break;
default:
return 0;
}
switch (ai->ai_socktype) {
case SOCK_RAW:
return EAI_SERVICE;
case SOCK_DGRAM:
case SOCK_STREAM:
allownumeric = 1;
break;
case ANY:
case AF_INET:
#ifdef AF_INET6
case AF_INET6:
#endif
allownumeric = 1;
break;
default:
allownumeric = 0;
break;
}
break;
default:
return EAI_SOCKTYPE;
}
if (str_isnumber(servname)) {
if (!allownumeric)
return EAI_SERVICE;
return EAI_SERVICE;
} else {
switch (ai->ai_socktype) {
case SOCK_DGRAM:
proto = "udp";
break;
case SOCK_STREAM:
proto = "tcp";
break;
default:
break;
}
return EAI_SERVICE;
}
if (!matchonly) {
case AF_INET:
((struct sockaddr_in *)(void *)
break;
case AF_INET6:
((struct sockaddr_in6 *)(void *)
break;
}
}
return 0;
}
static const struct afd *
int af;
{
return NULL;
return afd;
}
return NULL;
}
/*%
* post-2553: AI_ADDRCONFIG check. if we use getipnodeby* as backend, backend
* will take care of it.
* the semantics of AI_ADDRCONFIG is not defined well. we are not sure
* if the code is right or not.
*/
static int
int af;
{
int s;
/* XXX errno */
if (s < 0) {
return 0;
} else
close(s);
return 1;
}
/* convert a string to a scope identifier. XXX: IPv6 specific */
static int
{
char *ep;
/* empty scopeid portion is invalid */
if (*scope == '\0')
return (0);
#ifdef USE_IFNAMELINKID
/*
* Using interface names as link indices can be allowed
* only when we can assume a one-to-one mappings between
* links and interfaces. See comments in getnameinfo.c.
*/
if (scopeid == 0)
goto trynumeric;
return (1);
}
#endif
/* still unclear about literal, allow numeric only - placeholder */
goto trynumeric;
if (IN6_IS_ADDR_MC_ORGLOCAL(a6))
goto trynumeric;
else
goto trynumeric; /*%< global */
/* try to convert to a numeric id as a last resort */
errno = 0;
return (1);
} else
return (0);
}
struct addrinfo *
{
return(NULL);
return(NULL);
#if 0 /*%< the trick seems too much */
- sizeof(struct in_addr);
}
continue;
#endif /* 0 */
/* GET_PORT(cur->ai_next, servname); */
/*
* RFC2553 says that ai_canonname will be set only for
* the first element. we do it for all the elements,
* just for convenience.
*/
}
continue;
free:
/* continue, without tht pointer CUR advanced. */
}
}
struct addrinfo *
const char *cp;
{
return(NULL);
}
static struct net_data *
init()
{
goto error;
return (NULL);
}
}
return (net_data);
}