Blame src/daemon/gnuserv.c

Packit Service 407539
/* -*-C-*-
Packit Service 407539
 * Server code for handling requests from clients and forwarding them
Packit Service 407539
 * on to the GNU Emacs process.
Packit Service 407539
 *
Packit Service 407539
 * This file is part of GNU Emacs.
Packit Service 407539
 *
Packit Service 407539
 * Copying is permitted under those conditions described by the GNU
Packit Service 407539
 * General Public License.
Packit Service 407539
 *
Packit Service 407539
 * Copyright (C) 1989 Free Software Foundation, Inc.
Packit Service 407539
 *
Packit Service 407539
 * Author: Andy Norman (ange@hplb.hpl.hp.com), based on 'etc/server.c'
Packit Service 407539
 * from the 18.52 GNU Emacs distribution.
Packit Service 407539
 *
Packit Service 407539
 * Please mail bugs and suggestions to the author at the above address.
Packit Service 407539
 */
Packit Service 407539
Packit Service 407539
/* HISTORY
Packit Service 407539
 * 11-Nov-1990                bristor@simba
Packit Service 407539
 *    Added EOT stuff.
Packit Service 407539
 */
Packit Service 407539
Packit Service 407539
/*
Packit Service 407539
 * This file incorporates new features added by Bob Weiner <weiner@mot.com>,
Packit Service 407539
 * Darrell Kindred <dkindred@cmu.edu> and Arup Mukherjee <arup@cmu.edu>.
Packit Service 407539
 * Please see the note at the end of the README file for details.
Packit Service 407539
 *
Packit Service 407539
 * (If gnuserv came bundled with your emacs, the README file is probably
Packit Service 407539
 * ../etc/gnuserv.README relative to the directory containing this file)
Packit Service 407539
 */
Packit Service 407539
Packit Service 407539
#include <config.h>
Packit Service 407539
Packit Service 407539
#include <glib/gi18n-lib.h>
Packit Service 407539
#include <glibtop.h>
Packit Service 407539
#include <glibtop/open.h>
Packit Service 407539
#include <glibtop/close.h>
Packit Service 407539
#include <glibtop/command.h>
Packit Service 407539
#include <glibtop/parameter.h>
Packit Service 407539
Packit Service 407539
#include "server_config.h"
Packit Service 407539
Packit Service 407539
#include <glibtop/gnuserv.h>
Packit Service 407539
Packit Service 407539
#include <errno.h>
Packit Service 407539
#include <locale.h>
Packit Service 407539
Packit Service 407539
#include "daemon.h"
Packit Service 407539
Packit Service 407539
#ifdef AIX
Packit Service 407539
#include <sys/select.h>
Packit Service 407539
#endif
Packit Service 407539
Packit Service 407539
#ifdef NEED_DECLARATION_PROGRAM_INVOCATION_NAME
Packit Service 407539
extern char *program_invocation_name, *program_invocation_short_name;
Packit Service 407539
#endif
Packit Service 407539
Packit Service 407539
#ifndef HAVE_PROGRAM_INVOCATION_SHORT_NAME
Packit Service 407539
char *program_invocation_short_name;
Packit Service 407539
#endif
Packit Service 407539
Packit Service 407539
#ifndef HAVE_PROGRAM_INVOCATION_NAME
Packit Service 407539
char *program_invocation_name;
Packit Service 407539
#endif
Packit Service 407539
Packit Service 407539
Packit Service 407539
#if !defined(INTERNET_DOMAIN_SOCKETS)
Packit Service 407539
#error "Internet Domain sockets are required"
Packit Service 407539
#endif
Packit Service 407539
Packit Service 407539
#ifdef AUTH_MAGIC_COOKIE
Packit Service 407539
#include <X11/X.h>
Packit Service 407539
#include <X11/Xauth.h>
Packit Service 407539
Packit Service 407539
static Xauth *server_xauth = NULL;
Packit Service 407539
Packit Service 407539
#endif /* AUTH_MAGIC_COOKIE */
Packit Service 407539
Packit Service 407539
gboolean enable_debug = FALSE;
Packit Service 407539
gboolean verbose_output = FALSE;
Packit Service 407539
static gboolean no_daemon = FALSE;
Packit Service 407539
static gboolean invoked_from_inetd = FALSE;
Packit Service 407539
static int changed_uid = 0;
Packit Service 407539
Packit Service 407539
void
Packit Service 407539
syslog_message (int priority, const char *format, ...)
Packit Service 407539
{
Packit Service 407539
    va_list ap;
Packit Service 407539
    char buffer [BUFSIZ];
Packit Service 407539
Packit Service 407539
    va_start (ap, format);
Packit Service 407539
    vsnprintf (buffer, BUFSIZ-1, format, ap);
Packit Service 407539
    va_end (ap);
Packit Service 407539
Packit Service 407539
    syslog (priority, "%s", buffer);
Packit Service 407539
}
Packit Service 407539
Packit Service 407539
void
Packit Service 407539
syslog_io_message (int priority, const char *format, ...)
Packit Service 407539
{
Packit Service 407539
    va_list ap;
Packit Service 407539
    char buffer [BUFSIZ];
Packit Service 407539
    char buffer2 [BUFSIZ];
Packit Service 407539
Packit Service 407539
    va_start (ap, format);
Packit Service 407539
    vsnprintf (buffer, BUFSIZ-1, format, ap);
Packit Service 407539
    va_end (ap);
Packit Service 407539
Packit Service 407539
    snprintf (buffer2, BUFSIZ-1, "%s: %s", buffer, g_strerror (errno));
Packit Service 407539
    syslog (priority, "%s", buffer2);
Packit Service 407539
}
Packit Service 407539
Packit Service 407539
/*
Packit Service 407539
 * timed_read - Read with timeout.
Packit Service 407539
 */
Packit Service 407539
Packit Service 407539
static int
Packit Service 407539
timed_read (int fd, char *buf, int max, int timeout, int one_line)
Packit Service 407539
{
Packit Service 407539
    fd_set rmask;
Packit Service 407539
    struct timeval tv;	/* = {timeout, 0}; */
Packit Service 407539
    char c = 0;
Packit Service 407539
    int nbytes = 0;
Packit Service 407539
    int r;
Packit Service 407539
Packit Service 407539
    tv.tv_sec = timeout;
Packit Service 407539
    tv.tv_usec = 0;
Packit Service 407539
Packit Service 407539
    FD_ZERO (&rmask);
Packit Service 407539
    FD_SET (fd, &rmask);
Packit Service 407539
Packit Service 407539
    do {
Packit Service 407539
	r = select (fd + 1, &rmask, NULL, NULL, &tv;;
Packit Service 407539
Packit Service 407539
	if (r > 0) {
Packit Service 407539
	    if (read (fd, &c, 1) == 1) {
Packit Service 407539
		*buf++ = c;
Packit Service 407539
		++nbytes;
Packit Service 407539
	    } else {
Packit Service 407539
		syslog_io_message (LOG_WARNING, "read error on socket");
Packit Service 407539
		return -1;
Packit Service 407539
	    }
Packit Service 407539
	} else if (r == 0) {
Packit Service 407539
	    syslog_io_message (LOG_WARNING, "read timed out");
Packit Service 407539
	    return -1;
Packit Service 407539
	} else {
Packit Service 407539
	    syslog_io_message (LOG_WARNING, "error in select");
Packit Service 407539
	    return -1;
Packit Service 407539
	}
Packit Service 407539
    } while ((nbytes < max) && !(one_line && (c == '\n')));
Packit Service 407539
Packit Service 407539
    --buf;
Packit Service 407539
    if (one_line && *buf == '\n') {
Packit Service 407539
	*buf = 0;
Packit Service 407539
    }
Packit Service 407539
    return nbytes;
Packit Service 407539
}
Packit Service 407539
Packit Service 407539
Packit Service 407539
/*
Packit Service 407539
 * permitted -- return whether a given host is allowed to connect to the server.
Packit Service 407539
 */
Packit Service 407539
Packit Service 407539
static int
Packit Service 407539
permitted (u_long host_addr, int fd)
Packit Service 407539
{
Packit Service 407539
    int i;
Packit Service 407539
Packit Service 407539
    char auth_protocol[128];
Packit Service 407539
    char buf[1024];
Packit Service 407539
    int auth_data_len;
Packit Service 407539
Packit Service 407539
    /* Read auth protocol name */
Packit Service 407539
Packit Service 407539
    if (timed_read (fd, auth_protocol, AUTH_NAMESZ, AUTH_TIMEOUT, 1) <= 0)
Packit Service 407539
	return FALSE;
Packit Service 407539
Packit Service 407539
    if (enable_debug)
Packit Service 407539
	syslog_message (LOG_DEBUG,
Packit Service 407539
			"Client sent authenticatin protocol '%s'.",
Packit Service 407539
			auth_protocol);
Packit Service 407539
Packit Service 407539
    if (strcmp (auth_protocol, DEFAUTH_NAME) &&
Packit Service 407539
	strcmp (auth_protocol, MCOOKIE_NAME)) {
Packit Service 407539
	syslog_message (LOG_WARNING,
Packit Service 407539
			"Invalid authentication protocol "
Packit Service 407539
			"'%s' from client",
Packit Service 407539
			auth_protocol);
Packit Service 407539
	return FALSE;
Packit Service 407539
    }
Packit Service 407539
Packit Service 407539
    if (!strcmp (auth_protocol, MCOOKIE_NAME)) {
Packit Service 407539
	/*
Packit Service 407539
	 * doing magic cookie auth
Packit Service 407539
	 */
Packit Service 407539
Packit Service 407539
	if (timed_read (fd, buf, 10, AUTH_TIMEOUT, 1) <= 0)
Packit Service 407539
	    return FALSE;
Packit Service 407539
Packit Service 407539
	auth_data_len = atoi (buf);
Packit Service 407539
Packit Service 407539
	if (auth_data_len < 1 || (size_t)auth_data_len > sizeof(buf)) {
Packit Service 407539
	    syslog_message(LOG_WARNING, "Invalid data length supplied by client");
Packit Service 407539
	    return FALSE;
Packit Service 407539
	}
Packit Service 407539
Packit Service 407539
	if (timed_read (fd, buf, auth_data_len, AUTH_TIMEOUT, 0) != auth_data_len)
Packit Service 407539
	    return FALSE;
Packit Service 407539
Packit Service 407539
#ifdef AUTH_MAGIC_COOKIE
Packit Service 407539
	if (!invoked_from_inetd && server_xauth && server_xauth->data &&
Packit Service 407539
	    !memcmp (buf, server_xauth->data, auth_data_len)) {
Packit Service 407539
	    return TRUE;
Packit Service 407539
	}
Packit Service 407539
#else
Packit Service 407539
	syslog_message (LOG_WARNING,
Packit Service 407539
			"Client tried Xauth, but server is "
Packit Service 407539
			"not compiled with Xauth");
Packit Service 407539
#endif
Packit Service 407539
Packit Service 407539
	/*
Packit Service 407539
	 * auth failed, but allow this to fall through to the
Packit Service 407539
	 * GNU_SECURE protocol....
Packit Service 407539
	 */
Packit Service 407539
Packit Service 407539
	if (verbose_output) {
Packit Service 407539
	    if (changed_uid || invoked_from_inetd)
Packit Service 407539
		syslog_message (LOG_WARNING,
Packit Service 407539
				"Xauth authentication not allowed, "
Packit Service 407539
				"trying GNU_SECURE ...");
Packit Service 407539
	    else
Packit Service 407539
		syslog_message (LOG_WARNING,
Packit Service 407539
				"Xauth authentication failed, "
Packit Service 407539
				"trying GNU_SECURE auth...");
Packit Service 407539
	}
Packit Service 407539
    }
Packit Service 407539
Packit Service 407539
    /* Other auth protocols go here, and should execute only if
Packit Service 407539
     * the * auth_protocol name matches. */
Packit Service 407539
Packit Service 407539
    /* Now, try the old GNU_SECURE stuff... */
Packit Service 407539
Packit Service 407539
    if (enable_debug)
Packit Service 407539
	syslog_message (LOG_DEBUG, "Doing GNU_SECURE auth ...");
Packit Service 407539
Packit Service 407539
    /* Now check the chain for that hash key */
Packit Service 407539
    for (i = 0; i < HOST_TABLE_ENTRIES; i++) {
Packit Service 407539
	if (enable_debug)
Packit Service 407539
	    syslog_message (LOG_DEBUG, "Trying %lx - %lx",
Packit Service 407539
			    host_addr, permitted_hosts [i]);
Packit Service 407539
	if (permitted_hosts [i] == 0L)
Packit Service 407539
	    return (FALSE);
Packit Service 407539
	if (host_addr == permitted_hosts [i])
Packit Service 407539
	    return (TRUE);
Packit Service 407539
    }
Packit Service 407539
Packit Service 407539
    return (FALSE);
Packit Service 407539
}
Packit Service 407539
Packit Service 407539
Packit Service 407539
/*
Packit Service 407539
 * setup_table -- initialise the table of hosts allowed to contact the server,
Packit Service 407539
 * by reading from the file specified by the GNU_SECURE
Packit Service 407539
 * environment variable
Packit Service 407539
 * Put in the local machine, and, if a security file is specifed,
Packit Service 407539
 * add each host that is named in the file.
Packit Service 407539
 * Return the number of hosts added.
Packit Service 407539
 */
Packit Service 407539
Packit Service 407539
static int
Packit Service 407539
setup_table (void)
Packit Service 407539
{
Packit Service 407539
    char hostname [HOSTNAMSZ];
Packit Service 407539
Packit Service 407539
#ifdef AUTH_MAGIC_COOKIE
Packit Service 407539
    char screen [BUFSIZ];
Packit Service 407539
#endif
Packit Service 407539
Packit Service 407539
    long host_addr;
Packit Service 407539
    int i, hosts = 0;
Packit Service 407539
Packit Service 407539
    /* Make sure every entry is null */
Packit Service 407539
    for (i = 0; i < HOST_TABLE_ENTRIES; i++)
Packit Service 407539
	permitted_hosts [i] = 0;
Packit Service 407539
Packit Service 407539
    gethostname (hostname, HOSTNAMSZ);
Packit Service 407539
Packit Service 407539
    if ((host_addr = glibtop_internet_addr (hostname)) == -1) {
Packit Service 407539
	syslog_io_message (LOG_ERR, "Can't resolve '%s'", hostname);
Packit Service 407539
	exit (1);
Packit Service 407539
    }
Packit Service 407539
Packit Service 407539
#ifdef AUTH_MAGIC_COOKIE
Packit Service 407539
Packit Service 407539
    sprintf (screen, "%d", SERVER_PORT);
Packit Service 407539
Packit Service 407539
    server_xauth = XauGetAuthByAddr
Packit Service 407539
	(FamilyInternet,
Packit Service 407539
	 sizeof (host_addr), (char *) &host_addr,
Packit Service 407539
	 strlen (screen), screen,
Packit Service 407539
	 strlen (MCOOKIE_X_NAME), MCOOKIE_X_NAME);
Packit Service 407539
    hosts++;
Packit Service 407539
Packit Service 407539
#endif /* AUTH_MAGIC_COOKIE */
Packit Service 407539
Packit Service 407539
    /* Resolv host names from permitted_host_names []. */
Packit Service 407539
Packit Service 407539
    for (i = 0; i < HOST_TABLE_ENTRIES; i++) {
Packit Service 407539
	if (!permitted_host_names [i])
Packit Service 407539
	    continue;
Packit Service 407539
	if (enable_debug)
Packit Service 407539
	    syslog_message (LOG_DEBUG, "Resolving %s ...",
Packit Service 407539
			    permitted_host_names [i]);
Packit Service 407539
	permitted_hosts [i] =
Packit Service 407539
	    glibtop_internet_addr (permitted_host_names [i]);
Packit Service 407539
	if ((long) permitted_hosts [i] == -1) {
Packit Service 407539
	    syslog_io_message (LOG_ERR, "Can't resolve '%s'",
Packit Service 407539
			       permitted_host_names [i]);
Packit Service 407539
	    exit (1);
Packit Service 407539
	}
Packit Service 407539
    }
Packit Service 407539
Packit Service 407539
    if (enable_debug)
Packit Service 407539
	for (i = 0; i < HOST_TABLE_ENTRIES; i++)
Packit Service 407539
	    syslog_message (LOG_DEBUG, "Host %s - %lx",
Packit Service 407539
			    permitted_host_names [i],
Packit Service 407539
			    permitted_hosts [i]);
Packit Service 407539
Packit Service 407539
    hosts += HOST_TABLE_ENTRIES;
Packit Service 407539
Packit Service 407539
    return hosts;
Packit Service 407539
}				/* setup_table */
Packit Service 407539
Packit Service 407539
/*
Packit Service 407539
 * internet_init -- initialize server, returning an internet socket that can
Packit Service 407539
 * be listened on.
Packit Service 407539
 */
Packit Service 407539
Packit Service 407539
static int
Packit Service 407539
internet_init (void)
Packit Service 407539
{
Packit Service 407539
    int ls;			/* socket descriptor */
Packit Service 407539
    struct sockaddr_in server;	/* for local socket address */
Packit Service 407539
Packit Service 407539
    if (setup_table () == 0)
Packit Service 407539
	return -1;
Packit Service 407539
Packit Service 407539
    /* clear out address structure */
Packit Service 407539
    memset ((char *) &server, 0, sizeof (struct sockaddr_in));
Packit Service 407539
Packit Service 407539
    /* Set up address structure for the listen socket. */
Packit Service 407539
    server.sin_family = AF_INET;
Packit Service 407539
    server.sin_addr.s_addr = INADDR_ANY;
Packit Service 407539
Packit Service 407539
    /* We use a fixed port given in the config file. */
Packit Service 407539
    server.sin_port = htons (SERVER_PORT);
Packit Service 407539
Packit Service 407539
    if (verbose_output)
Packit Service 407539
	syslog_message (LOG_INFO, "Using port %u.", SERVER_PORT);
Packit Service 407539
Packit Service 407539
    /* Create the listen socket. */
Packit Service 407539
    if ((ls = socket (AF_INET, SOCK_STREAM, 0)) == -1) {
Packit Service 407539
	syslog_io_message (LOG_ERR, "unable to create socket");
Packit Service 407539
	exit (1);
Packit Service 407539
    }
Packit Service 407539
Packit Service 407539
    /* Bind the listen address to the socket. */
Packit Service 407539
    if (bind (ls, (struct sockaddr *) &server,
Packit Service 407539
	      sizeof (struct sockaddr_in)) == -1) {
Packit Service 407539
	syslog_io_message (LOG_ERR, "bind");
Packit Service 407539
	exit (1);
Packit Service 407539
    }
Packit Service 407539
Packit Service 407539
    /* Initiate the listen on the socket so remote users * can connect.  */
Packit Service 407539
    if (listen (ls, 20) == -1) {
Packit Service 407539
	syslog_io_message (LOG_ERR, "listen");
Packit Service 407539
	exit (1);
Packit Service 407539
    }
Packit Service 407539
Packit Service 407539
    return (ls);
Packit Service 407539
}				/* internet_init */
Packit Service 407539
Packit Service 407539
Packit Service 407539
/*
Packit Service 407539
 * handle_internet_request -- accept a request from a client and send the
Packit Service 407539
 * information to stdout (the gnu process).
Packit Service 407539
 */
Packit Service 407539
Packit Service 407539
static void
Packit Service 407539
handle_internet_request (int ls)
Packit Service 407539
{
Packit Service 407539
    int s;
Packit Service 407539
    size_t addrlen = sizeof (struct sockaddr_in);
Packit Service 407539
    struct sockaddr_in peer;	/* for peer socket address */
Packit Service 407539
    pid_t pid;
Packit Service 407539
Packit Service 407539
    memset ((char *) &peer, 0, sizeof (struct sockaddr_in));
Packit Service 407539
Packit Service 407539
    if ((s = accept (ls, (struct sockaddr *) &peer, (void *) &addrlen)) == -1) {
Packit Service 407539
	syslog_io_message (LOG_ERR, "accept");
Packit Service 407539
	exit (1);
Packit Service 407539
    }
Packit Service 407539
Packit Service 407539
    if (verbose_output)
Packit Service 407539
	syslog_message (LOG_INFO, "Connection was made from %s port %u.",
Packit Service 407539
			inet_ntoa (peer.sin_addr), ntohs (peer.sin_port));
Packit Service 407539
Packit Service 407539
    /* Check that access is allowed - if not return crud to the client */
Packit Service 407539
    if (!permitted (peer.sin_addr.s_addr, s)) {
Packit Service 407539
	close (s);
Packit Service 407539
	syslog_message (LOG_CRIT, "Refused connection from %s.",
Packit Service 407539
			inet_ntoa (peer.sin_addr));
Packit Service 407539
	return;
Packit Service 407539
    }			/* if */
Packit Service 407539
Packit Service 407539
    if (verbose_output)
Packit Service 407539
	syslog_message (LOG_INFO, "Accepted connection from %s port %u.",
Packit Service 407539
			inet_ntoa (peer.sin_addr), ntohs (peer.sin_port));
Packit Service 407539
Packit Service 407539
    pid = fork ();
Packit Service 407539
Packit Service 407539
    if (pid == -1) {
Packit Service 407539
	syslog_io_message (LOG_ERR, "fork failed");
Packit Service 407539
	exit (1);
Packit Service 407539
    }
Packit Service 407539
Packit Service 407539
    if (pid) {
Packit Service 407539
	if (verbose_output)
Packit Service 407539
	    syslog_message (LOG_INFO, "Child pid is %d.", pid);
Packit Service 407539
	return;
Packit Service 407539
    }
Packit Service 407539
Packit Service 407539
    handle_parent_connection (s);
Packit Service 407539
Packit Service 407539
    close (s);
Packit Service 407539
Packit Service 407539
    if (verbose_output)
Packit Service 407539
	syslog_message (LOG_INFO, "Closed connection to %s port %u.",
Packit Service 407539
			inet_ntoa (peer.sin_addr), ntohs (peer.sin_port));
Packit Service 407539
Packit Service 407539
    _exit (0);
Packit Service 407539
}				/* handle_internet_request */
Packit Service 407539
Packit Service 407539
static void
Packit Service 407539
handle_signal (int sig)
Packit Service 407539
{
Packit Service 407539
    if (sig == SIGCHLD)
Packit Service 407539
	return;
Packit Service 407539
Packit Service 407539
    syslog_message (LOG_ERR, "Catched signal %d.\n", sig);
Packit Service 407539
    exit (1);
Packit Service 407539
}
Packit Service 407539
Packit Service 407539
static const GOptionEntry options [] = {
Packit Service 407539
    { "debug", 'd', 0, G_OPTION_ARG_NONE, &enable_debug,
Packit Service 407539
      N_("Enable debugging"), NULL },
Packit Service 407539
    { "verbose", 'v', 0, G_OPTION_ARG_NONE, &verbose_output,
Packit Service 407539
      N_("Enable verbose output"), NULL },
Packit Service 407539
    { "no-daemon", 'f', 0, G_OPTION_ARG_NONE, &no_daemon,
Packit Service 407539
      N_("Don’t fork into background"), NULL },
Packit Service 407539
    { "inetd", 'i', 0, G_OPTION_ARG_NONE, &invoked_from_inetd,
Packit Service 407539
      N_("Invoked from inetd"), NULL },
Packit Service 407539
    { NULL }
Packit Service 407539
};
Packit Service 407539
Packit Service 407539
int
Packit Service 407539
main (int argc, char **argv)
Packit Service 407539
{
Packit Service 407539
    const unsigned method = GLIBTOP_METHOD_PIPE;
Packit Service 407539
    const unsigned long features = GLIBTOP_SYSDEPS_ALL;
Packit Service 407539
    glibtop *server = glibtop_global_server;
Packit Service 407539
    GOptionContext *goption_context;
Packit Service 407539
    GError *error = NULL;
Packit Service 407539
Packit Service 407539
    int ils = -1;		/* internet domain listen socket */
Packit Service 407539
Packit Service 407539
    setlocale (LC_ALL, "");
Packit Service 407539
Packit Service 407539
    /* On non-glibc systems, this is not set up for us.  */
Packit Service 407539
    if (!program_invocation_name) {
Packit Service 407539
	char *arg;
Packit Service 407539
Packit Service 407539
	program_invocation_name = (char *) argv[0];
Packit Service 407539
	arg = strrchr (argv[0], '/');
Packit Service 407539
	program_invocation_short_name =
Packit Service 407539
	    arg ? (arg + 1) : program_invocation_name;
Packit Service 407539
    }
Packit Service 407539
Packit Service 407539
    g_set_prgname (program_invocation_short_name);
Packit Service 407539
    goption_context = g_option_context_new (NULL);
Packit Service 407539
    g_option_context_add_main_entries (goption_context, options, NULL);
Packit Service 407539
    g_option_context_parse (goption_context, &argc, &argv, &error);
Packit Service 407539
    g_option_context_free (goption_context);
Packit Service 407539
Packit Service 407539
    if (error != NULL) {
Packit Service 407539
        g_printerr ("%s\n", error->message);
Packit Service 407539
        g_error_free (error);
Packit Service 407539
        g_printerr (_("Run “%s --help” to see a full list of "
Packit Service 407539
                    "available command line options.\n"),
Packit Service 407539
                    program_invocation_name);
Packit Service 407539
        exit(1);
Packit Service 407539
    }
Packit Service 407539
Packit Service 407539
    if (enable_debug)
Packit Service 407539
	verbose_output = 1;
Packit Service 407539
Packit Service 407539
    if (no_daemon) {
Packit Service 407539
	openlog ("libgtop-daemon", LOG_PERROR | LOG_PID, LOG_LOCAL0);
Packit Service 407539
    } else {
Packit Service 407539
	openlog ("libgtop-daemon", LOG_PID, LOG_LOCAL0);
Packit Service 407539
    }
Packit Service 407539
Packit Service 407539
    if (!no_daemon && !invoked_from_inetd) {
Packit Service 407539
	pid_t pid = fork ();
Packit Service 407539
Packit Service 407539
	if (pid == -1) {
Packit Service 407539
	    syslog_io_message (LOG_ERR, "fork failed");
Packit Service 407539
	    exit (1);
Packit Service 407539
	} else if (pid)
Packit Service 407539
	    exit (0);
Packit Service 407539
Packit Service 407539
	close (0);
Packit Service 407539
Packit Service 407539
	setsid ();
Packit Service 407539
    }
Packit Service 407539
Packit Service 407539
    glibtop_init_r (&glibtop_global_server, 0, GLIBTOP_INIT_NO_INIT);
Packit Service 407539
Packit Service 407539
    signal (SIGCHLD, handle_signal);
Packit Service 407539
Packit Service 407539
    /* If we are root, completely switch to SERVER_UID and
Packit Service 407539
     * SERVER_GID. Otherwise we completely drop any priviledges.
Packit Service 407539
     */
Packit Service 407539
Packit Service 407539
    if (enable_debug)
Packit Service 407539
	syslog_message (LOG_DEBUG, "Parent ID: (%d, %d) - (%d, %d)",
Packit Service 407539
			getuid (), geteuid (), getgid (), getegid ());
Packit Service 407539
Packit Service 407539
    if (geteuid () == 0) {
Packit Service 407539
	changed_uid = 1;
Packit Service 407539
	if (setregid (SERVER_GID, SERVER_GID)) {
Packit Service 407539
	    syslog_io_message (LOG_ERR, "setregid (SERVER_GID)");
Packit Service 407539
	    exit (1);
Packit Service 407539
	}
Packit Service 407539
	if (setreuid (SERVER_UID, SERVER_UID)) {
Packit Service 407539
	    syslog_io_message (LOG_ERR, "setreuid (SERVER_UID)");
Packit Service 407539
	    exit (1);
Packit Service 407539
	}
Packit Service 407539
    } else {
Packit Service 407539
	if (setreuid (geteuid (), geteuid ())) {
Packit Service 407539
	    syslog_io_message (LOG_ERR, "setreuid (euid)");
Packit Service 407539
	    exit (1);
Packit Service 407539
	}
Packit Service 407539
    }
Packit Service 407539
Packit Service 407539
    if (enable_debug)
Packit Service 407539
	syslog_message (LOG_DEBUG, "Parent ID: (%d, %d) - (%d, %d)",
Packit Service 407539
			getuid (), geteuid (), getgid (), getegid ());
Packit Service 407539
Packit Service 407539
    if (invoked_from_inetd) {
Packit Service 407539
	size_t addrlen = sizeof (struct sockaddr_in);
Packit Service 407539
	struct sockaddr_in peer;
Packit Service 407539
Packit Service 407539
	memset ((char *) &peer, 0, sizeof (struct sockaddr_in));
Packit Service 407539
Packit Service 407539
	if (getpeername (0, (struct sockaddr *) &peer, (void *) &addrlen)) {
Packit Service 407539
	    syslog_io_message (LOG_ERR, "getpeername");
Packit Service 407539
	    exit (1);
Packit Service 407539
	}
Packit Service 407539
Packit Service 407539
	if (verbose_output)
Packit Service 407539
	    syslog_message (LOG_INFO, "Connection was made from %s port %u.",
Packit Service 407539
			    inet_ntoa (peer.sin_addr), ntohs (peer.sin_port));
Packit Service 407539
Packit Service 407539
	/* Check that access is allowed - if not return crud to the client */
Packit Service 407539
	if (!permitted (peer.sin_addr.s_addr, 0)) {
Packit Service 407539
	    close (0);
Packit Service 407539
	    syslog_message (LOG_CRIT, "Refused connection from %s.",
Packit Service 407539
			    inet_ntoa (peer.sin_addr));
Packit Service 407539
	    exit (1);
Packit Service 407539
	}
Packit Service 407539
Packit Service 407539
	handle_parent_connection (0);
Packit Service 407539
	exit (0);
Packit Service 407539
    }
Packit Service 407539
Packit Service 407539
    /* get a internet domain socket to listen on. */
Packit Service 407539
    ils = internet_init ();
Packit Service 407539
Packit Service 407539
    if (ils <= 0) {
Packit Service 407539
	syslog_message (LOG_ERR, "Unable to get internet domain socket.");
Packit Service 407539
	exit (1);
Packit Service 407539
    }
Packit Service 407539
Packit Service 407539
    glibtop_set_parameter_l (server, GLIBTOP_PARAM_METHOD,
Packit Service 407539
			     &method, sizeof (method));
Packit Service 407539
Packit Service 407539
    server->features = features;
Packit Service 407539
Packit Service 407539
    glibtop_init_r (&server, 0, 0);
Packit Service 407539
Packit Service 407539
    while (1) {
Packit Service 407539
	fd_set rmask;
Packit Service 407539
	int status, ret;
Packit Service 407539
Packit Service 407539
	while ((ret = wait3 (&status, WNOHANG, NULL)) != 0) {
Packit Service 407539
	    if ((ret == -1) && (errno == ECHILD))
Packit Service 407539
		break;
Packit Service 407539
Packit Service 407539
	    if ((ret == -1) && ((errno == EAGAIN)))
Packit Service 407539
		continue;
Packit Service 407539
	    if (ret == 0) {
Packit Service 407539
		syslog_io_message (LOG_WARNING, "wait3");
Packit Service 407539
		continue;
Packit Service 407539
	    }
Packit Service 407539
Packit Service 407539
	    if (verbose_output)
Packit Service 407539
		syslog_message (LOG_INFO, "Child %d exited.", ret);
Packit Service 407539
	}
Packit Service 407539
Packit Service 407539
	FD_ZERO (&rmask);
Packit Service 407539
Packit Service 407539
	/* Only the child accepts connections from standard
Packit Service 407539
	 * input made by its parent. */
Packit Service 407539
Packit Service 407539
	FD_SET (ils, &rmask);
Packit Service 407539
Packit Service 407539
	if (enable_debug)
Packit Service 407539
	    syslog_message (LOG_DEBUG,
Packit Service 407539
			    "Server ready and waiting for connections.");
Packit Service 407539
Packit Service 407539
	if (select (ils+1, &rmask, (fd_set *) NULL, (fd_set *) NULL,
Packit Service 407539
		    (struct timeval *) NULL) < 0) {
Packit Service 407539
	    if (errno == EINTR)
Packit Service 407539
		continue;
Packit Service 407539
	    syslog_io_message (LOG_ERR, "select");
Packit Service 407539
	    exit (1);
Packit Service 407539
	}
Packit Service 407539
Packit Service 407539
	if (FD_ISSET (ils, &rmask))
Packit Service 407539
	    handle_internet_request (ils);
Packit Service 407539
    }
Packit Service 407539
Packit Service 407539
    return 0;
Packit Service 407539
}