/****************************************************************************** Copyright (c) 1994, 1995, 1996 Xerox Corporation. All rights reserved. Portions of this code were written by Stephen White, aka ghond. Use and copying of this software and preparation of derivative works based upon this software are permitted. Any distribution of this software or derivative works must comply with all applicable United States export control laws. This software is made available AS IS, and Xerox Corporation makes no warranty about the software, its performance or its conformity to any specification. Any person obtaining a copy of this software is requested to send their name and post office or electronic mail address to: Pavel Curtis Xerox PARC 3333 Coyote Hill Rd. Palo Alto, CA 94304 Pavel@Xerox.Com *****************************************************************************/ /* This module provides IP host name lookup with timeouts. Because * longjmps out of name lookups corrupt some UNIX name lookup modules, this * module uses a subprocess to do the name lookup. On any failure, the * subprocess is restarted. */ #include "options.h" #if NETWORK_PROTOCOL == NP_TCP /* Skip almost entire file otherwise... */ #include "my-signal.h" #include "my-stdlib.h" #include "my-unistd.h" #include "my-inet.h" /* inet_addr() */ #include "my-in.h" /* struct sockaddr_in, INADDR_ANY, htons(), * htonl(), ntohl(), struct in_addr */ #include <netdb.h> /* struct hostent, gethostbyaddr() */ #include "my-socket.h" /* AF_INET */ #include "my-wait.h" #include "my-string.h" #include <errno.h> #include "config.h" #include "log.h" #include "server.h" #include "storage.h" #include "timers.h" /****************************************************************************** * Utilities *****************************************************************************/ static pid_t spawn_pipe(void (*child_proc)(int to_parent, int from_parent), int *to_child, int *from_child) { int pipe_to_child[2], pipe_from_child[2]; pid_t pid; if (pipe(pipe_to_child) < 0) { log_perror("SPAWNING: Couldn't create first pipe"); } else if (pipe(pipe_from_child) < 0) { log_perror("SPAWNING: Couldn't create second pipe"); close(pipe_to_child[0]); close(pipe_to_child[1]); } else if ((pid = fork()) < 0) { log_perror("SPAWNING: Couldn't fork middleman"); close(pipe_to_child[0]); close(pipe_to_child[1]); close(pipe_from_child[0]); close(pipe_from_child[1]); } else if (pid != 0) { /* parent */ int status; close(pipe_to_child[0]); close(pipe_from_child[1]); *to_child = pipe_to_child[1]; *from_child = pipe_from_child[0]; /* Cast to (void *) to avoid warnings on systems that misdeclare the * argument. */ wait((void *) &status); /* wait for middleman to die */ if (status != 0) { errlog("SPAWNING: Middleman died with status %d!\n", status); close(pipe_to_child[1]); close(pipe_from_child[0]); } else if (read(*from_child, &pid, sizeof(pid)) != sizeof(pid)) { errlog("SPAWNING: Bad read() for pid\n"); close(pipe_to_child[1]); close(pipe_from_child[0]); } else { return pid; } } else { /* middleman */ close(pipe_to_child[1]); close(pipe_from_child[0]); if ((pid = fork()) < 0) { log_perror("SPAWNING: Couldn't fork child"); exit(1); } else if (pid != 0) { /* still the middleman */ write(pipe_from_child[1], &pid, sizeof(pid)); exit(0); } else { /* finally, the child */ (*child_proc)(pipe_from_child[1], pipe_to_child[0]); exit(0); } } return 0; } static void ensure_buffer(char **buffer, int *buflen, int len) { if (len > *buflen) { if (*buffer) myfree(*buffer, M_STRING); *buflen = len; *buffer = mymalloc(len, M_STRING); } } static int robust_read(int fd, void *buffer, int len) { int count; do { count = read(fd, buffer, len); } while (count == -1 && errno == EINTR); return count; } /****************************************************************************** * Data structures and types used by more than one process. *****************************************************************************/ struct request { enum { REQ_NAME_FROM_ADDR, REQ_ADDR_FROM_NAME } kind; unsigned timeout; union { unsigned length; struct sockaddr_in address; } u; }; /****************************************************************************** * Code that runs in the lookup process. *****************************************************************************/ static void timeout_proc(Timer_ID id, Timer_Data data) { _exit(1); } static void lookup(int to_intermediary, int from_intermediary) { struct request req; static char *buffer = 0; static int buflen = 0; Timer_ID id; struct hostent *e; set_server_cmdline("(MOO name-lookup slave)"); /* Read requests and do them. Before each, we set a timer. If it expires, we exit (in timeout_proc, above). The intermediary will restart us in that event. */ for (;;) { if (robust_read(from_intermediary, &req, sizeof(req)) != sizeof(req)) _exit(1); if (req.kind == REQ_ADDR_FROM_NAME) { ensure_buffer(&buffer, &buflen, req.u.length + 1); if (robust_read(from_intermediary, buffer, req.u.length) != req.u.length) _exit(1); buffer[req.u.length] = 0; id = set_timer(req.timeout, timeout_proc, 0); /* This cast is to work around systems like NeXT that declare * gethostbyname() to take a non-const string pointer. */ e = gethostbyname((void *) buffer); cancel_timer(id); if (e && e->h_length == sizeof(unsigned32)) write(to_intermediary, e->h_addr_list[0], e->h_length); else { unsigned32 addr; /* This cast is for the same reason as the one above... */ addr = inet_addr((void *) buffer); write(to_intermediary, &addr, sizeof(addr)); } } else { const char *host_name; int length; id = set_timer(req.timeout, timeout_proc, 0); e = gethostbyaddr((void *) &req.u.address.sin_addr, sizeof(req.u.address.sin_addr), AF_INET); cancel_timer(id); host_name = e ? e->h_name : ""; length = strlen(host_name); write(to_intermediary, &length, sizeof(length)); write(to_intermediary, host_name, length); } } } /****************************************************************************** * Code that runs in the intermediary process. *****************************************************************************/ static int to_lookup, from_lookup; static pid_t lookup_pid; static void restart_lookup(void) { if (lookup_pid) { kill(lookup_pid, SIGKILL); close(to_lookup); close(from_lookup); oklog("NAME_LOOKUP: Killing old lookup process ...\n"); } lookup_pid = spawn_pipe(lookup, &to_lookup, &from_lookup); if (lookup_pid) oklog("NAME_LOOKUP: Started new lookup process\n"); else errlog("NAME_LOOKUP: Can't spawn lookup process; " "will try again later...\n"); } static void intermediary(int to_server, int from_server) { struct request req; static char *buffer = 0; static int buflen = 0; int len; unsigned32 addr; set_server_cmdline("(MOO name-lookup master)"); signal(SIGPIPE, SIG_IGN); restart_lookup(); for (;;) { if (robust_read(from_server, &req, sizeof(req)) != sizeof(req)) _exit(1); if (req.kind == REQ_ADDR_FROM_NAME) { ensure_buffer(&buffer, &buflen, req.u.length); if (robust_read(from_server, buffer, req.u.length) != req.u.length) _exit(1); } if (!lookup_pid) /* Restart lookup if it's died */ restart_lookup(); if (lookup_pid) { /* Only try to deal with lookup if alive */ write(to_lookup, &req, sizeof(req)); if (req.kind == REQ_ADDR_FROM_NAME) { write(to_lookup, buffer, req.u.length); if (robust_read(from_lookup, &addr, sizeof(addr)) != sizeof(addr)) { restart_lookup(); addr = 0; } write(to_server, &addr, sizeof(addr)); } else { if (robust_read(from_lookup, &len, sizeof(len)) != sizeof(len)) { restart_lookup(); len = 0; } else { ensure_buffer(&buffer, &buflen, len); if (len > 0 && robust_read(from_lookup, buffer, len) != len) { restart_lookup(); len = 0; } } write(to_server, &len, sizeof(len)); if (len > 0) write(to_server, buffer, len); } } else { /* Lookup dead and wouldn't restart ... */ int failure = 0; write(to_server, &failure, sizeof(failure)); } } } /****************************************************************************** * Code that runs in the server process. *****************************************************************************/ static int to_intermediary, from_intermediary; static int dead_intermediary = 0; int initialize_name_lookup(void) { return spawn_pipe(intermediary, &to_intermediary, &from_intermediary); } static void abandon_intermediary(const char *prefix) { errlog("LOOKUP_NAME: %s; presumed dead...\n", prefix); dead_intermediary = 1; close(to_intermediary); close(from_intermediary); } const char * lookup_name_from_addr(struct sockaddr_in *addr, unsigned timeout) { struct request req; static char *buffer = 0; static int buflen = 0; int len; if (!dead_intermediary) { req.kind = REQ_NAME_FROM_ADDR; req.timeout = timeout; req.u.address = *addr; if (write(to_intermediary, &req, sizeof(req)) != sizeof(req)) abandon_intermediary("LOOKUP_NAME: Write to intermediary failed"); else if (robust_read(from_intermediary, &len, sizeof(len)) != sizeof(len)) abandon_intermediary("LOOKUP_NAME: Read from intermediary failed"); else if (len != 0) { ensure_buffer(&buffer, &buflen, len + 1); if (robust_read(from_intermediary, buffer, len) != len) abandon_intermediary("LOOKUP_NAME: " "Data-read from intermediary failed"); else { buffer[len] = '\0'; return buffer; } } } /* Either the intermediary is presumed dead, or else it failed to produce * a name; in either case, we must fall back on a the default, dotted- * decimal notation. */ { static char decimal[20]; unsigned32 a = ntohl(addr->sin_addr.s_addr); sprintf(decimal, "%u.%u.%u.%u", (unsigned) (a >> 24) & 0xff, (unsigned) (a >> 16) & 0xff, (unsigned) (a >> 8) & 0xff, (unsigned) a & 0xff); return decimal; } } unsigned32 lookup_addr_from_name(const char *name, unsigned timeout) { struct request req; unsigned32 addr = 0; if (dead_intermediary) { /* Numeric addresses should always work... */ addr = inet_addr((void *) name); } else { req.kind = REQ_ADDR_FROM_NAME; req.timeout = timeout; req.u.length = strlen(name); if (write(to_intermediary, &req, sizeof(req)) != sizeof(req) || write(to_intermediary, name, req.u.length) != req.u.length) abandon_intermediary("LOOKUP_ADDR: Write to intermediary failed"); else if (robust_read(from_intermediary, &addr, sizeof(addr)) != sizeof(addr)) abandon_intermediary("LOOKUP_ADDR: Read from intermediary failed"); } return addr == 0xffffffff ? 0 : addr; } #endif /* NETWORK_PROTOCOL == NP_TCP */ char rcsid_name_lookup[] = "$Id: name_lookup.c,v 2.2 1996/02/08 06:59:04 pavel Exp $"; /* $Log: name_lookup.c,v $ * Revision 2.2 1996/02/08 06:59:04 pavel * Renamed err/logf() to errlog/oklog(). Updated copyright notice for 1996. * Release 1.8.0beta1. * * Revision 2.1 1995/12/11 08:11:45 pavel * Added missing #include of "my-stdlib.h". Release 1.8.0alpha2. * * Revision 2.0 1995/11/30 04:28:09 pavel * New baseline version, corresponding to release 1.8.0alpha1. * * Revision 1.1 1995/11/30 04:27:56 pavel * Initial revision */