/*- * Copyright (c) 1994, Garrett Wollman * * 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 * documentation and/or other materials provided with the distribution. * * THIS SOFTWARE IS PROVIDED BY THE 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 REGENTS 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. */ #if defined(LIBC_SCCS) && !defined(lint) static char sccsid[] = "@(#)$Id: gethostnamadr.c,v 1.1.1.1 2003/11/19 01:51:27 kyu3 Exp $"; static char rcsid[] = "$Id: gethostnamadr.c,v 1.1.1.1 2003/11/19 01:51:27 kyu3 Exp $"; #endif /* LIBC_SCCS and not lint */ #include <sys/param.h> #include <sys/socket.h> #include <netinet/in.h> #include <arpa/inet.h> #include <netdb.h> #include <stdio.h> #include <ctype.h> #include <errno.h> #include <paths.h> #include <string.h> #include <arpa/nameser.h> /* XXX hack for _res */ #include <resolv.h> /* XXX hack for _res */ #include "Socklib_internals.h" enum service_type { SERVICE_NONE = 0, SERVICE_BIND, SERVICE_HOSTS, SERVICE_NIS }; #define SERVICE_MAX SERVICE_NIS static struct { const char *name; enum service_type type; } service_names[] = { { "hosts", SERVICE_HOSTS }, { _PATH_HOSTS, SERVICE_HOSTS }, { "hosttable", SERVICE_HOSTS }, { "htable", SERVICE_HOSTS }, { "bind", SERVICE_BIND }, { "dns", SERVICE_BIND }, { "domain", SERVICE_BIND }, { "yp", SERVICE_NIS }, { "yellowpages", SERVICE_NIS }, { "nis", SERVICE_NIS }, { 0, SERVICE_NONE } }; static enum service_type service_order[SERVICE_MAX + 1]; static int service_done = 0; static enum service_type get_service_name(const char *name) { int i; for(i = 0; service_names[i].type != SERVICE_NONE; i++) { if(!strcasecmp(name, service_names[i].name)) { return service_names[i].type; } } return SERVICE_NONE; } static void init_services() { char *cp, *p, buf[BUFSIZ]; register int cc = 0; FILE *fd; if ((fd = (FILE *)fopen(_PATH_HOSTCONF, "r")) == NULL) { /* make some assumptions */ service_order[0] = SERVICE_HOSTS; service_order[1] = SERVICE_BIND; service_order[2] = SERVICE_NONE; } else { while (fgets(buf, BUFSIZ, fd) != NULL && cc < SERVICE_MAX) { if(buf[0] == '#') continue; p = buf; while ((cp = strsep(&p, "\n \t,:;")) != NULL && *cp == '\0') ; if (cp == NULL) continue; do { if (isalpha(cp[0])) { service_order[cc] = get_service_name(cp); if(service_order[cc] != SERVICE_NONE) cc++; } while ((cp = strsep(&p, "\n \t,:;")) != NULL && *cp == '\0') ; } while(cp != NULL && cc < SERVICE_MAX); } service_order[cc] = SERVICE_NONE; fclose(fd); } service_done = 1; } struct hostent * gethostbyname(const char *name) { struct hostent *hp; if (_res.options & RES_USE_INET6) { /* XXX */ hp = gethostbyname2(name, AF_INET6); /* XXX */ if (hp) /* XXX */ return (hp); /* XXX */ } /* XXX */ return (gethostbyname2(name, AF_INET)); } struct hostent * gethostbyname2(const char *name, int type) { struct hostent *hp = 0; int nserv = 0; if (!service_done) init_services(); while (!hp) { switch (service_order[nserv]) { case SERVICE_NONE: return NULL; case SERVICE_HOSTS: hp = _gethostbyhtname(name, type); break; case SERVICE_BIND: hp = _gethostbydnsname(name, type); break; case SERVICE_NIS: hp = _gethostbynisname(name, type); break; } nserv++; } return hp; } struct hostent * gethostbyaddr(const char *addr, socklen_t len, int type) { struct hostent *hp = 0; int nserv = 0; if (!service_done) init_services(); while (!hp) { switch (service_order[nserv]) { case SERVICE_NONE: return 0; case SERVICE_HOSTS: hp = _gethostbyhtaddr(addr, len, type); break; case SERVICE_BIND: hp = _gethostbydnsaddr(addr, len, type); break; case SERVICE_NIS: hp = _gethostbynisaddr(addr, len, type); break; } nserv++; } return hp; } #ifdef _THREAD_SAFE struct hostent_data; /* * Temporary function (not thread safe) */ int gethostbyaddr_r(const char *addr, int len, int type, struct hostent *result, struct hostent_data *buffer) { struct hostent *hp; int ret; if ((hp = gethostbyaddr(addr, len, type)) == NULL) { ret = -1; } else { memcpy(result, hp, sizeof(struct hostent)); ret = 0; } return(ret); } #endif void sethostent(int stayopen) { _sethosthtent(stayopen); _sethostdnsent(stayopen); } void endhostent() { _endhosthtent(); _endhostdnsent(); }