2000-01-14 04:45:46 +00:00
|
|
|
/*
|
2003-06-05 08:52:47 +00:00
|
|
|
* Pseudo-implementation of RFC2553 name / address resolution functions
|
2000-01-14 04:45:46 +00:00
|
|
|
*
|
|
|
|
* But these functions are not implemented correctly. The minimum subset
|
2003-06-11 13:56:41 +00:00
|
|
|
* is implemented for ssh use only. For example, this routine assumes
|
2000-01-14 04:45:46 +00:00
|
|
|
* that ai_family is AF_INET. Don't use it for another purpose.
|
|
|
|
*/
|
|
|
|
|
2003-06-05 09:37:30 +00:00
|
|
|
#include "includes.h"
|
|
|
|
|
2003-06-11 13:56:41 +00:00
|
|
|
RCSID("$Id: fake-rfc2553.c,v 1.3 2003/06/11 13:56:41 dtucker Exp $");
|
2003-06-05 08:52:47 +00:00
|
|
|
|
|
|
|
#ifndef HAVE_GETNAMEINFO
|
|
|
|
int getnameinfo(const struct sockaddr *sa, size_t salen, char *host,
|
|
|
|
size_t hostlen, char *serv, size_t servlen, int flags)
|
|
|
|
{
|
|
|
|
struct sockaddr_in *sin = (struct sockaddr_in *)sa;
|
|
|
|
struct hostent *hp;
|
|
|
|
char tmpserv[16];
|
|
|
|
|
|
|
|
if (serv != NULL) {
|
|
|
|
snprintf(tmpserv, sizeof(tmpserv), "%d", ntohs(sin->sin_port));
|
|
|
|
if (strlcpy(serv, tmpserv, servlen) >= servlen)
|
|
|
|
return (EAI_MEMORY);
|
|
|
|
}
|
2000-01-14 04:45:46 +00:00
|
|
|
|
2003-06-05 08:52:47 +00:00
|
|
|
if (host != NULL) {
|
|
|
|
if (flags & NI_NUMERICHOST) {
|
|
|
|
if (strlcpy(host, inet_ntoa(sin->sin_addr),
|
|
|
|
hostlen) >= hostlen)
|
|
|
|
return (EAI_MEMORY);
|
|
|
|
else
|
|
|
|
return (0);
|
|
|
|
} else {
|
|
|
|
hp = gethostbyaddr((char *)&sin->sin_addr,
|
|
|
|
sizeof(struct in_addr), AF_INET);
|
|
|
|
if (hp == NULL)
|
|
|
|
return (EAI_NODATA);
|
|
|
|
|
|
|
|
if (strlcpy(host, hp->h_name, hostlen) >= hostlen)
|
|
|
|
return (EAI_MEMORY);
|
|
|
|
else
|
|
|
|
return (0);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return (0);
|
|
|
|
}
|
|
|
|
#endif /* !HAVE_GETNAMEINFO */
|
2001-02-09 01:55:35 +00:00
|
|
|
|
2000-01-14 04:45:46 +00:00
|
|
|
#ifndef HAVE_GAI_STRERROR
|
2003-05-18 14:13:38 +00:00
|
|
|
char *
|
|
|
|
gai_strerror(int err)
|
2000-01-14 04:45:46 +00:00
|
|
|
{
|
2003-05-18 14:13:38 +00:00
|
|
|
switch (err) {
|
|
|
|
case EAI_NODATA:
|
|
|
|
return ("no address associated with name");
|
|
|
|
case EAI_MEMORY:
|
|
|
|
return ("memory allocation failure.");
|
2003-06-04 23:48:32 +00:00
|
|
|
case EAI_NONAME:
|
|
|
|
return ("nodename nor servname provided, or not known");
|
2003-05-18 14:13:38 +00:00
|
|
|
default:
|
|
|
|
return ("unknown/invalid error.");
|
2000-05-31 01:20:11 +00:00
|
|
|
}
|
2000-01-14 04:45:46 +00:00
|
|
|
}
|
|
|
|
#endif /* !HAVE_GAI_STRERROR */
|
|
|
|
|
|
|
|
#ifndef HAVE_FREEADDRINFO
|
2003-05-18 14:13:38 +00:00
|
|
|
void
|
|
|
|
freeaddrinfo(struct addrinfo *ai)
|
2000-01-14 04:45:46 +00:00
|
|
|
{
|
2000-05-31 01:20:11 +00:00
|
|
|
struct addrinfo *next;
|
|
|
|
|
2003-05-19 00:39:37 +00:00
|
|
|
for(; ai != NULL;) {
|
|
|
|
next = ai->ai_next;
|
2000-05-31 01:20:11 +00:00
|
|
|
free(ai);
|
2003-05-18 14:13:38 +00:00
|
|
|
ai = next;
|
|
|
|
}
|
2000-01-14 04:45:46 +00:00
|
|
|
}
|
|
|
|
#endif /* !HAVE_FREEADDRINFO */
|
|
|
|
|
|
|
|
#ifndef HAVE_GETADDRINFO
|
2003-05-18 14:13:38 +00:00
|
|
|
static struct
|
|
|
|
addrinfo *malloc_ai(int port, u_long addr, const struct addrinfo *hints)
|
2000-01-14 04:45:46 +00:00
|
|
|
{
|
2000-05-31 01:20:11 +00:00
|
|
|
struct addrinfo *ai;
|
|
|
|
|
2003-06-05 00:04:12 +00:00
|
|
|
ai = malloc(sizeof(*ai) + sizeof(struct sockaddr_in));
|
|
|
|
if (ai == NULL)
|
|
|
|
return (NULL);
|
2000-05-31 01:20:11 +00:00
|
|
|
|
2003-05-18 14:13:38 +00:00
|
|
|
memset(ai, '\0', sizeof(*ai) + sizeof(struct sockaddr_in));
|
2000-05-31 01:20:11 +00:00
|
|
|
|
|
|
|
ai->ai_addr = (struct sockaddr *)(ai + 1);
|
|
|
|
/* XXX -- ssh doesn't use sa_len */
|
|
|
|
ai->ai_addrlen = sizeof(struct sockaddr_in);
|
|
|
|
ai->ai_addr->sa_family = ai->ai_family = AF_INET;
|
2000-01-14 04:45:46 +00:00
|
|
|
|
2000-05-31 01:20:11 +00:00
|
|
|
((struct sockaddr_in *)(ai)->ai_addr)->sin_port = port;
|
|
|
|
((struct sockaddr_in *)(ai)->ai_addr)->sin_addr.s_addr = addr;
|
|
|
|
|
2003-05-14 11:48:51 +00:00
|
|
|
/* XXX: the following is not generally correct, but does what we want */
|
|
|
|
if (hints->ai_socktype)
|
|
|
|
ai->ai_socktype = hints->ai_socktype;
|
|
|
|
else
|
|
|
|
ai->ai_socktype = SOCK_STREAM;
|
|
|
|
|
|
|
|
if (hints->ai_protocol)
|
|
|
|
ai->ai_protocol = hints->ai_protocol;
|
|
|
|
|
2003-05-18 14:13:38 +00:00
|
|
|
return (ai);
|
2000-01-14 04:45:46 +00:00
|
|
|
}
|
|
|
|
|
2003-05-18 14:13:38 +00:00
|
|
|
int
|
|
|
|
getaddrinfo(const char *hostname, const char *servname,
|
|
|
|
const struct addrinfo *hints, struct addrinfo **res)
|
2000-01-14 04:45:46 +00:00
|
|
|
{
|
2000-05-31 01:20:11 +00:00
|
|
|
struct hostent *hp;
|
2003-02-05 23:50:42 +00:00
|
|
|
struct servent *sp;
|
2000-07-08 01:50:37 +00:00
|
|
|
struct in_addr in;
|
2003-02-05 23:50:42 +00:00
|
|
|
int i;
|
|
|
|
long int port;
|
2003-03-24 02:35:58 +00:00
|
|
|
u_long addr;
|
2000-05-31 01:20:11 +00:00
|
|
|
|
2003-02-05 23:50:42 +00:00
|
|
|
port = 0;
|
|
|
|
if (servname != NULL) {
|
|
|
|
char *cp;
|
|
|
|
|
|
|
|
port = strtol(servname, &cp, 10);
|
|
|
|
if (port > 0 && port <= 65535 && *cp == '\0')
|
|
|
|
port = htons(port);
|
|
|
|
else if ((sp = getservbyname(servname, NULL)) != NULL)
|
|
|
|
port = sp->s_port;
|
|
|
|
else
|
|
|
|
port = 0;
|
|
|
|
}
|
2000-05-31 01:20:11 +00:00
|
|
|
|
|
|
|
if (hints && hints->ai_flags & AI_PASSIVE) {
|
2003-03-24 02:35:58 +00:00
|
|
|
addr = htonl(0x00000000);
|
|
|
|
if (hostname && inet_aton(hostname, &in) != 0)
|
|
|
|
addr = in.s_addr;
|
2003-05-18 14:13:38 +00:00
|
|
|
*res = malloc_ai(port, addr, hints);
|
2003-06-05 00:04:12 +00:00
|
|
|
if (*res == NULL)
|
|
|
|
return (EAI_MEMORY);
|
2003-05-18 14:13:38 +00:00
|
|
|
return (0);
|
2000-05-31 01:20:11 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (!hostname) {
|
2003-05-18 14:13:38 +00:00
|
|
|
*res = malloc_ai(port, htonl(0x7f000001), hints);
|
2003-06-05 00:04:12 +00:00
|
|
|
if (*res == NULL)
|
|
|
|
return (EAI_MEMORY);
|
2003-05-18 14:13:38 +00:00
|
|
|
return (0);
|
2000-05-31 01:20:11 +00:00
|
|
|
}
|
|
|
|
|
2000-07-11 02:15:54 +00:00
|
|
|
if (inet_aton(hostname, &in)) {
|
2003-05-18 14:13:38 +00:00
|
|
|
*res = malloc_ai(port, in.s_addr, hints);
|
2003-06-05 00:04:12 +00:00
|
|
|
if (*res == NULL)
|
|
|
|
return (EAI_MEMORY);
|
2003-05-18 14:13:38 +00:00
|
|
|
return (0);
|
2000-05-31 01:20:11 +00:00
|
|
|
}
|
|
|
|
|
2003-06-04 23:48:32 +00:00
|
|
|
/* Don't try DNS if AI_NUMERICHOST is set */
|
|
|
|
if (hints && hints->ai_flags & AI_NUMERICHOST)
|
|
|
|
return (EAI_NONAME);
|
|
|
|
|
2000-05-31 01:20:11 +00:00
|
|
|
hp = gethostbyname(hostname);
|
|
|
|
if (hp && hp->h_name && hp->h_name[0] && hp->h_addr_list[0]) {
|
2003-05-18 14:13:38 +00:00
|
|
|
struct addrinfo *cur, *prev;
|
|
|
|
|
2003-06-05 00:04:12 +00:00
|
|
|
cur = prev = *res = NULL;
|
2000-05-31 01:20:11 +00:00
|
|
|
for (i = 0; hp->h_addr_list[i]; i++) {
|
2003-05-18 14:13:38 +00:00
|
|
|
struct in_addr *in = (struct in_addr *)hp->h_addr_list[i];
|
|
|
|
|
|
|
|
cur = malloc_ai(port, in->s_addr, hints);
|
2003-06-05 00:04:12 +00:00
|
|
|
if (cur == NULL) {
|
|
|
|
if (*res != NULL)
|
|
|
|
freeaddrinfo(*res);
|
|
|
|
return (EAI_MEMORY);
|
|
|
|
}
|
2000-05-31 01:20:11 +00:00
|
|
|
if (prev)
|
|
|
|
prev->ai_next = cur;
|
|
|
|
else
|
|
|
|
*res = cur;
|
|
|
|
|
|
|
|
prev = cur;
|
|
|
|
}
|
2003-05-18 14:13:38 +00:00
|
|
|
return (0);
|
2000-05-31 01:20:11 +00:00
|
|
|
}
|
|
|
|
|
2003-05-18 14:13:38 +00:00
|
|
|
return (EAI_NODATA);
|
2000-01-14 04:45:46 +00:00
|
|
|
}
|
|
|
|
#endif /* !HAVE_GETADDRINFO */
|