diff options
Diffstat (limited to 'lib/libc/net/getnameinfo.c')
-rw-r--r-- | lib/libc/net/getnameinfo.c | 388 |
1 files changed, 210 insertions, 178 deletions
diff --git a/lib/libc/net/getnameinfo.c b/lib/libc/net/getnameinfo.c index 1ae197c15aa..62c4b9d9287 100644 --- a/lib/libc/net/getnameinfo.c +++ b/lib/libc/net/getnameinfo.c @@ -1,7 +1,7 @@ /* * %%% copyright-cmetz-96-bsd * Copyright (c) 1996-1999, Craig Metz, All rights reserved. - * + * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions * are met: @@ -17,7 +17,7 @@ * 4. Neither the name of the author nor the names of contributors * may be used to endorse or promote products derived from this software * without specific prior written permission. - * + * * THIS SOFTWARE IS PROVIDED BY THE REGENTS 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 @@ -36,197 +36,229 @@ #include <sys/types.h> #include <sys/socket.h> - #include <netinet/in.h> #include <sys/un.h> #include <sys/utsname.h> #include <netdb.h> +#include <arpa/inet.h> #include <errno.h> #include <string.h> #include <resolv.h> -#ifndef AF_LOCAL -#define AF_LOCAL AF_UNIX -#endif /* AF_LOCAL */ - -#ifndef min +#ifndef min #define min(x,y) (((x) > (y)) ? (y) : (x)) -#endif /* min */ +#endif /* min */ -#define RETURN_ERROR(x) do { \ - rval = (x); \ - goto ret; \ - } while(0) - -static int netdb_lookup_name(int family, void *addr, int addrlen, char *name, - int namelen, int flags) +static int +netdb_lookup_name(int family, void *addr, int addrlen, char *name, + int namelen, int flags) { - struct hostent *hostent; - char *c, *c2; - int rval, i; - - if (!(hostent = gethostbyaddr(addr, addrlen, family))) { - switch(h_errno) { - case NETDB_INTERNAL: - RETURN_ERROR(EAI_SYSTEM); - case HOST_NOT_FOUND: - RETURN_ERROR(1); - case TRY_AGAIN: - RETURN_ERROR(EAI_AGAIN); - case NO_RECOVERY: - RETURN_ERROR(EAI_FAIL); - case NO_DATA: - RETURN_ERROR(1); - default: - RETURN_ERROR(EAI_FAIL); - }; - }; - - endhostent(); - - c = hostent->h_name; - if ((flags & NI_NOFQDN) && (_res.options & RES_INIT) && _res.defdname[0] && - (c2 = strstr(c + 1, _res.defdname)) && (*(--c2) == '.')) { - *c2 = 0; - i = min(c2 - c, namelen) - 1; - strncpy(name, c, i); - } else - strncpy(name, c, namelen - 1); - - rval = 0; + struct hostent *hostent; + char *c, *c2; + int i; -ret: - return rval; + if (!(hostent = gethostbyaddr(addr, addrlen, family))) { + switch (h_errno) { + case NETDB_INTERNAL: + return(EAI_SYSTEM); + case HOST_NOT_FOUND: + return(1); + case TRY_AGAIN: + return(EAI_AGAIN); + case NO_RECOVERY: + return(EAI_FAIL); + case NO_DATA: + return(1); + default: + return(EAI_FAIL); + } + } + + endhostent(); + + c = hostent->h_name; + if ((flags & NI_NOFQDN) && (_res.options & RES_INIT) && _res.defdname[0] && + (c2 = strstr(c + 1, _res.defdname)) && (*(--c2) == '.')) { + *c2 = 0; + i = min(c2 - c, namelen); + strlcpy(name, c, i); + } else + strlcpy(name, c, namelen); + return 0; } -int getnameinfo(const struct sockaddr *sa, size_t addrlen, char *host, size_t hostlen, char *serv, size_t servlen, int flags) +int +getnameinfo(const struct sockaddr *sa, size_t addrlen, char *host, + size_t hostlen, char *serv, size_t servlen, int flags) { - int rval; - int serrno = errno; - - if (!sa || (addrlen != SA_LEN(sa))) - RETURN_ERROR(EAI_FAIL); - - if (host && (hostlen > 0)) - switch(sa->sa_family) { - case AF_INET6: - if (IN6_IS_ADDR_UNSPECIFIED(&((struct sockaddr_in6 *)sa)->sin6_addr)) { - if (flags & NI_NUMERICHOST) - goto inet6_noname; - else - strncpy(host, "*", hostlen - 1); - break; - }; - - if (IN6_IS_ADDR_V4MAPPED(&((struct sockaddr_in6 *)sa)->sin6_addr)) { - struct sockaddr_in sin; - memset(&sin, 0, sizeof(struct sockaddr_in)); - sin.sin_len = sizeof(struct sockaddr_in); - sin.sin_family = AF_INET; - sin.sin_port = ((struct sockaddr_in6 *)sa)->sin6_port; - sin.sin_addr.s_addr = ((u_int32_t *)&((struct sockaddr_in6 *)sa)->sin6_addr)[3]; - if (!(rval = getnameinfo((struct sockaddr *)&sin, sizeof(struct sockaddr_in), host, hostlen, serv, servlen, flags | NI_NAMEREQD))) - goto ret; - if (rval != EAI_NONAME) - goto ret; - goto inet6_noname; - }; - - if (flags & NI_NUMERICHOST) - goto inet6_noname; - - if ((rval = netdb_lookup_name(AF_INET6, - &((struct sockaddr_in6 *)sa)->sin6_addr, sizeof(struct in6_addr), - host, hostlen, flags)) < 0) - goto ret; - - if (!rval) - break; - -inet6_noname: - if (flags & NI_NAMEREQD) - RETURN_ERROR(EAI_NONAME); - - if (!inet_ntop(AF_INET6, &((struct sockaddr_in6 *)sa)->sin6_addr, host, hostlen)) - RETURN_ERROR(EAI_NONAME); - - break; - case AF_INET: - if (flags & NI_NUMERICHOST) - goto inet_noname; - - if (!((struct sockaddr_in *)sa)->sin_addr.s_addr) { - strncpy(host, "*", hostlen - 1); - break; - }; - - if ((rval = netdb_lookup_name(AF_INET, - &((struct sockaddr_in *)sa)->sin_addr, sizeof(struct in_addr), - host, hostlen, flags)) < 0) - goto ret; - - if (!rval) - break; -inet_noname: - if (flags & NI_NAMEREQD) - RETURN_ERROR(EAI_NONAME); - - if (!inet_ntop(AF_INET, &((struct sockaddr_in *)sa)->sin_addr, host, hostlen)) - RETURN_ERROR(EAI_NONAME); - - break; - case AF_LOCAL: - if (!(flags & NI_NUMERICHOST)) { - struct utsname utsname; - - if (!uname(&utsname)) { - strncpy(host, utsname.nodename, hostlen - 1); - break; - }; - }; - - if (flags & NI_NAMEREQD) - RETURN_ERROR(EAI_NONAME); - - strncpy(host, "localhost", hostlen - 1); - break; - default: - RETURN_ERROR(EAI_FAMILY); - }; - - if (serv && (servlen > 0)) - switch(sa->sa_family) { - case AF_INET: - case AF_INET6: - if (!(flags & NI_NUMERICSERV)) { - struct servent *s; - if (s = getservbyport(((struct sockaddr_in *)sa)->sin_port, (flags & NI_DGRAM) ? "udp" : "tcp")) { - strncpy(serv, s->s_name, servlen - 1); - break; - }; - if (!((struct sockaddr_in *)sa)->sin_port) { - strncpy(serv, "*", servlen - 1); - break; - }; - }; - snprintf(serv, servlen - 1, "%d", ntohs(((struct sockaddr_in *)sa)->sin_port)); - break; - case AF_LOCAL: - strncpy(serv, ((struct sockaddr_un *)sa)->sun_path, servlen - 1); - break; - }; - - if (host && (hostlen > 0)) - host[hostlen-1] = 0; - if (serv && (servlen > 0)) - serv[servlen-1] = 0; - rval = 0; + int rval; + int saved_errno; -ret: - if (rval == 1) - rval = EAI_FAIL; + if (sa == NULL || addrlen != sa->sa_len) + return EAI_FAIL; + saved_errno = errno; + + if (host && hostlen > 0) { + switch (sa->sa_family) { + case AF_INET6: + { + struct sockaddr_in6 *sin6 = (void *)sa; + + if (IN6_IS_ADDR_UNSPECIFIED(&sin6->sin6_addr)) { + if (flags & NI_NUMERICHOST) + goto inet6_noname; + strlcpy(host, "*", hostlen); + break; + } + + if (IN6_IS_ADDR_V4MAPPED(&sin6->sin6_addr)) { + struct sockaddr_in sin; + + memset(&sin, 0, sizeof(struct sockaddr_in)); + sin.sin_len = sizeof(struct sockaddr_in); + sin.sin_family = AF_INET; + sin.sin_port = sin6->sin6_port; + sin.sin_addr.s_addr = + ((u_int32_t *)&sin6->sin6_addr)[3]; + if (!(rval = getnameinfo((struct sockaddr *)&sin, + sizeof(struct sockaddr_in), host, hostlen, + serv, servlen, flags | NI_NAMEREQD))) + goto ret; + if (rval != EAI_NONAME) + goto ret; + goto inet6_noname; + } + + if (flags & NI_NUMERICHOST) + goto inet6_noname; + if ((rval = netdb_lookup_name(AF_INET6, + &sin6->sin6_addr, sizeof(struct in6_addr), + host, hostlen, flags)) < 0) + goto ret; - errno = serrno; + if (!rval) + break; + inet6_noname: + if (flags & NI_NAMEREQD) { + rval = EAI_NONAME; + goto ret; + } + if (!inet_ntop(AF_INET6, &sin6->sin6_addr, host, hostlen)) { + rval = EAI_NONAME; + goto ret; + } + break; + } + case AF_INET: + { + struct sockaddr_in *sin = (void *)sa; - return rval; -}; + if (flags & NI_NUMERICHOST) + goto inet_noname; + + if (sin->sin_addr.s_addr == 0) { + strlcpy(host, "*", hostlen); + break; + } + + if ((rval = netdb_lookup_name(AF_INET, + &sin->sin_addr, sizeof(struct in_addr), + host, hostlen, flags)) < 0) + goto ret; + + if (!rval) + break; + inet_noname: + if (flags & NI_NAMEREQD) { + rval = EAI_NONAME; + goto ret; + } + if (!inet_ntop(AF_INET, &sin->sin_addr, host, hostlen)) { + rval = EAI_NONAME; + goto ret; + } + break; + } + case AF_LOCAL: + if (!(flags & NI_NUMERICHOST)) { + struct utsname utsname; + + if (!uname(&utsname)) { + strlcpy(host, utsname.nodename, hostlen); + break; + } + } + + if (flags & NI_NAMEREQD) { + rval = EAI_NONAME; + goto ret; + } + + strlcpy(host, "localhost", hostlen); + break; + default: + rval = EAI_FAMILY; + goto ret; + } + } + + if (serv && servlen > 0) { + switch (sa->sa_family) { + case AF_INET: + { + struct sockaddr_in *sin = (void *)sa; + struct servent *s; + + if ((flags & NI_NUMERICSERV) == 0) { + s = getservbyport(sin->sin_port, + (flags & NI_DGRAM) ? "udp" : "tcp"); + if (s) { + strlcpy(serv, s->s_name, servlen); + break; + } + if (sin->sin_port == 0) { + strlcpy(serv, "*", servlen); + break; + } + } + snprintf(serv, servlen, "%d", ntohs(sin->sin_port)); + break; + } + case AF_INET6: + { + struct sockaddr_in6 *sin6 = (void *)sa; + struct servent *s; + + if ((flags & NI_NUMERICSERV) == 0) { + + s = getservbyport(sin6->sin6_port, + (flags & NI_DGRAM) ? "udp" : "tcp"); + if (s) { + strlcpy(serv, s->s_name, servlen); + break; + } + if (sin6->sin6_port == 0) { + strlcpy(serv, "*", servlen); + break; + } + } + snprintf(serv, servlen, "%d", ntohs(sin6->sin6_port)); + break; + } + case AF_LOCAL: + { + struct sockaddr_un *sun = (void *)sa; + + strlcpy(serv, sun->sun_path, servlen); + break; + } + } + } + rval = 0; + +ret: + if (rval == 1) + rval = EAI_FAIL; + errno = saved_errno; + return (rval); +} |