Simplify fake getname/addrinfo() functions, possibly fixing freeing a NULL pointer.
This commit is contained in:
parent
dbfd6f284e
commit
03995ca52e
2 changed files with 50 additions and 85 deletions
|
@ -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;
|
|
||||||
else
|
|
||||||
return EAI_MEMORY;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!hostname) {
|
|
||||||
if (NULL != (*res = malloc_ai(port, htonl(0x7f000001))))
|
|
||||||
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);
|
|
||||||
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;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
return EAI_NODATA;
|
if (!hostname) {
|
||||||
|
*res = malloc_ai(port, htonl(0x7f000001));
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
hp = gethostbyname(hostname);
|
||||||
|
|
||||||
|
if(!hp || !hp->h_addr_list[0])
|
||||||
|
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 */
|
||||||
|
|
|
@ -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;
|
return EAI_NODATA;
|
||||||
} else {
|
|
||||||
hp = gethostbyaddr((char *)&sin->sin_addr,
|
|
||||||
sizeof(struct in_addr), AF_INET);
|
|
||||||
if (hp == NULL)
|
|
||||||
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 */
|
||||||
|
|
Loading…
Reference in a new issue