Simplify fake getname/addrinfo() functions, possibly fixing freeing a NULL pointer.

This commit is contained in:
Guus Sliepen 2003-08-16 12:10:28 +00:00
parent dbfd6f284e
commit 03995ca52e
2 changed files with 50 additions and 85 deletions

View file

@ -20,11 +20,11 @@ char *gai_strerror(int ecode)
{ {
switch (ecode) { switch (ecode) {
case EAI_NODATA: case EAI_NODATA:
return "no address associated with hostname."; return "No address associated with hostname";
case EAI_MEMORY: case EAI_MEMORY:
return "memory allocation failure."; return "Memory allocation failure";
default: default:
return "unknown error."; return "Unknown error";
} }
} }
#endif /* !HAVE_GAI_STRERROR */ #endif /* !HAVE_GAI_STRERROR */
@ -34,91 +34,66 @@ void freeaddrinfo(struct addrinfo *ai)
{ {
struct addrinfo *next; struct addrinfo *next;
do { while(ai) {
next = ai->ai_next; next = ai->ai_next;
free(ai); free(ai);
} while (NULL != (ai = next)); ai = next;
}
} }
#endif /* !HAVE_FREEADDRINFO */ #endif /* !HAVE_FREEADDRINFO */
#ifndef HAVE_GETADDRINFO #ifndef HAVE_GETADDRINFO
static struct addrinfo *malloc_ai(int port, uint32_t addr) static struct addrinfo *malloc_ai(uint16_t port, uint32_t addr)
{ {
struct addrinfo *ai; struct addrinfo *ai;
ai = malloc(sizeof(struct addrinfo) + sizeof(struct sockaddr_in)); ai = xmalloc_and_zero(sizeof(struct addrinfo) + sizeof(struct sockaddr_in));
if (ai == NULL)
return(NULL);
memset(ai, 0, sizeof(struct addrinfo) + sizeof(struct sockaddr_in));
ai->ai_addr = (struct sockaddr *)(ai + 1); ai->ai_addr = (struct sockaddr *)(ai + 1);
/* XXX -- ssh doesn't use sa_len */
ai->ai_addrlen = sizeof(struct sockaddr_in); ai->ai_addrlen = sizeof(struct sockaddr_in);
ai->ai_addr->sa_family = ai->ai_family = AF_INET; ai->ai_addr->sa_family = ai->ai_family = AF_INET;
((struct sockaddr_in *)(ai)->ai_addr)->sin_port = port; ((struct sockaddr_in *)(ai)->ai_addr)->sin_port = port;
((struct sockaddr_in *)(ai)->ai_addr)->sin_addr.s_addr = addr; ((struct sockaddr_in *)(ai)->ai_addr)->sin_addr.s_addr = addr;
return(ai); return ai;
} }
int getaddrinfo(const char *hostname, const char *servname, int getaddrinfo(const char *hostname, const char *servname, const struct addrinfo *hints, struct addrinfo **res)
const struct addrinfo *hints, struct addrinfo **res)
{ {
struct addrinfo *cur, *prev = NULL; struct addrinfo *prev = NULL;
struct hostent *hp; struct hostent *hp;
struct in_addr in; struct in_addr in = {0};
int i, port; int i;
uint16_t port = 0;
if (servname) if (servname)
port = htons(atoi(servname)); port = htons(atoi(servname));
else
port = 0;
if (hints && hints->ai_flags & AI_PASSIVE) { if (hints && hints->ai_flags & AI_PASSIVE) {
if (NULL != (*res = malloc_ai(port, htonl(0x00000000)))) *res = malloc_ai(port, htonl(0x00000000));
return 0; return 0;
else
return EAI_MEMORY;
} }
if (!hostname) { if (!hostname) {
if (NULL != (*res = malloc_ai(port, htonl(0x7f000001)))) *res = malloc_ai(port, htonl(0x7f000001));
return 0; return 0;
else
return EAI_MEMORY;
} }
#ifdef HAVE_INET_ATON
if (inet_aton(hostname, &in)) {
if (NULL != (*res = malloc_ai(port, in.s_addr)))
return 0;
else
return EAI_MEMORY;
}
#endif
hp = gethostbyname(hostname); hp = gethostbyname(hostname);
if (hp && hp->h_name && hp->h_name[0] && hp->h_addr_list[0]) {
for (i = 0; hp->h_addr_list[i]; i++) {
cur = malloc_ai(port, ((struct in_addr *)hp->h_addr_list[i])->s_addr);
if (cur == NULL) {
if (*res)
freeaddrinfo(*res);
return EAI_MEMORY;
}
if (prev)
prev->ai_next = cur;
else
*res = cur;
prev = cur;
}
return 0;
}
if(!hp || !hp->h_addr_list[0])
return EAI_NODATA; return EAI_NODATA;
for (i = 0; hp->h_addr_list[i]; i++) {
*res = malloc_ai(port, ((struct in_addr *)hp->h_addr_list[i])->s_addr);
if(prev)
prev->ai_next = *res;
prev = *res;
}
return 0;
} }
#endif /* !HAVE_GETADDRINFO */ #endif /* !HAVE_GETADDRINFO */

View file

@ -16,41 +16,31 @@
#ifndef HAVE_GETNAMEINFO #ifndef HAVE_GETNAMEINFO
int getnameinfo(const struct sockaddr *sa, size_t salen, char *host, int getnameinfo(const struct sockaddr *sa, size_t salen, char *host, size_t hostlen, char *serv, size_t servlen, int flags)
size_t hostlen, char *serv, size_t servlen, int flags)
{ {
struct sockaddr_in *sin = (struct sockaddr_in *)sa; struct sockaddr_in *sin = (struct sockaddr_in *)sa;
struct hostent *hp; struct hostent *hp;
char tmpserv[16];
if (serv) { if(serv)
snprintf(tmpserv, sizeof(tmpserv), "%d", ntohs(sin->sin_port)); snprintf(serv, sizeof(tmpserv), "%d", ntohs(sin->sin_port));
if (strlen(tmpserv) >= servlen)
return EAI_MEMORY; if(!host)
else return 0;
strcpy(serv, tmpserv);
if(flags & NI_NUMERICHOST) {
strncpy(host, inet_ntoa(sin->sin_addr), sizeof(host));
return 0;
} }
if (host) { hp = gethostbyaddr((char *)&sin->sin_addr, sizeof(struct in_addr), AF_INET);
if (flags & NI_NUMERICHOST) {
if (strlen(inet_ntoa(sin->sin_addr)) >= hostlen)
return EAI_MEMORY;
strcpy(host, inet_ntoa(sin->sin_addr)); if(!hp || !hp->h_name)
return 0;
} else {
hp = gethostbyaddr((char *)&sin->sin_addr,
sizeof(struct in_addr), AF_INET);
if (hp == NULL)
return EAI_NODATA; return EAI_NODATA;
if (strlen(hp->h_name) >= hostlen) if(strlen(hp->h_name) >= hostlen)
return EAI_MEMORY; return EAI_MEMORY;
strcpy(host, hp->h_name); strncpy(host, hp->h_name, hostlen);
return 0;
}
}
return 0; return 0;
} }
#endif /* !HAVE_GETNAMEINFO */ #endif /* !HAVE_GETNAMEINFO */