mpm_common.c revision 8ff84ed7af4420c7cfbafdbd2863cb3272bae256
0922cbe8300e97215564748d449824f458196335Lennart Poettering/* ====================================================================
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering * The Apache Software License, Version 1.1
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering *
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering * Copyright (c) 2000-2001 The Apache Software Foundation. All rights
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering * reserved.
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering *
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering * Redistribution and use in source and binary forms, with or without
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering * modification, are permitted provided that the following conditions
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering * are met:
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering *
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering * 1. Redistributions of source code must retain the above copyright
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering * notice, this list of conditions and the following disclaimer.
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering *
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering * 2. Redistributions in binary form must reproduce the above copyright
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering * notice, this list of conditions and the following disclaimer in
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering * the documentation and/or other materials provided with the
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering * distribution.
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering *
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering * 3. The end-user documentation included with the redistribution,
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering * if any, must include the following acknowledgment:
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering * "This product includes software developed by the
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering * Apache Software Foundation (http://www.apache.org/)."
ef7963b561adec267dbc7dbdfc0d29105f2ca872Lennart Poettering * Alternately, this acknowledgment may appear in the software itself,
ef7963b561adec267dbc7dbdfc0d29105f2ca872Lennart Poettering * if and wherever such third-party acknowledgments normally appear.
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering *
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering * 4. The names "Apache" and "Apache Software Foundation" must
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering * not be used to endorse or promote products derived from this
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering * software without prior written permission. For written
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering * permission, please contact apache@apache.org.
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering *
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering * 5. Products derived from this software may not be called "Apache",
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering * nor may "Apache" appear in their name, without prior written
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering * permission of the Apache Software Foundation.
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering *
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering * THIS SOFTWARE IS PROVIDED ``AS IS'' AND ANY EXPRESSED OR IMPLIED
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering * WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering * DISCLAIMED. IN NO EVENT SHALL THE APACHE SOFTWARE FOUNDATION OR
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering * ITS CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering * USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering * OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering * SUCH DAMAGE.
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering * ====================================================================
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering *
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering * This software consists of voluntary contributions made by many
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering * individuals on behalf of the Apache Software Foundation. For more
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering * information on the Apache Software Foundation, please see
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering * <http://www.apache.org/>.
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering *
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering * Portions of this software are based upon public domain software
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering * originally written at the National Center for Supercomputing Applications,
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering * University of Illinois, Urbana-Champaign.
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering */
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering/* The purpose of this file is to store the code that MOST mpm's will need
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering * this does not mean a function only goes into this file if every MPM needs
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering * it. It means that if a function is needed by more than one MPM, and
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering * future maintenance would be served by making the code common, then the
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering * function belongs here.
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering *
d4205751d4643c272059a3728045929dd0e5e800Lennart Poettering * This is going in src/main because it is not platform specific, it is
d4205751d4643c272059a3728045929dd0e5e800Lennart Poettering * specific to multi-process servers, but NOT to Unix. Which is why it
d4205751d4643c272059a3728045929dd0e5e800Lennart Poettering * does not belong in src/os/unix
d4205751d4643c272059a3728045929dd0e5e800Lennart Poettering */
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering
d4205751d4643c272059a3728045929dd0e5e800Lennart Poettering#include "apr.h"
d4205751d4643c272059a3728045929dd0e5e800Lennart Poettering#include "apr_thread_proc.h"
d4205751d4643c272059a3728045929dd0e5e800Lennart Poettering#include "apr_signal.h"
d4205751d4643c272059a3728045929dd0e5e800Lennart Poettering
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering#include "httpd.h"
d4205751d4643c272059a3728045929dd0e5e800Lennart Poettering#include "http_config.h"
d4205751d4643c272059a3728045929dd0e5e800Lennart Poettering#include "http_log.h"
d4205751d4643c272059a3728045929dd0e5e800Lennart Poettering#include "http_main.h"
d4205751d4643c272059a3728045929dd0e5e800Lennart Poettering#include "mpm.h"
d4205751d4643c272059a3728045929dd0e5e800Lennart Poettering#include "mpm_common.h"
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering#include "ap_mpm.h"
d4205751d4643c272059a3728045929dd0e5e800Lennart Poettering#include "ap_listen.h"
d4205751d4643c272059a3728045929dd0e5e800Lennart Poettering
d4205751d4643c272059a3728045929dd0e5e800Lennart Poettering#ifdef HAVE_PWD_H
d4205751d4643c272059a3728045929dd0e5e800Lennart Poettering#include <pwd.h>
d4205751d4643c272059a3728045929dd0e5e800Lennart Poettering#endif
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering#ifdef HAVE_GRP_H
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering#include <grp.h>
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering#endif
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering#ifdef AP_MPM_NEEDS_RECLAIM_CHILD_PROCESSES
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poetteringvoid ap_reclaim_child_processes(int terminate)
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering{
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering int i;
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering long int waittime = 1024 * 16; /* in usecs */
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering apr_status_t waitret;
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering int tries;
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering int not_dead_yet;
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering int max_daemons;
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering ap_mpm_query(AP_MPMQ_MAX_DAEMONS, &max_daemons);
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering MPM_SYNC_CHILD_TABLE();
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering for (tries = terminate ? 4 : 1; tries <= 9; ++tries) {
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering /* don't want to hold up progress any more than
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering * necessary, but we need to allow children a few moments to exit.
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering * Set delay with an exponential backoff.
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering */
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering waittime = waittime * 4;
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering apr_sleep(waittime);
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering /* now see who is done */
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering not_dead_yet = 0;
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering for (i = 0; i < max_daemons; ++i) {
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering pid_t pid = MPM_CHILD_PID(i);
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering apr_proc_t proc;
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering if (pid == 0)
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering continue;
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering
5190bbb240ab3360f7d37714d1c877858aef8c12Thomas Hindoe Paaboel Andersen proc.pid = pid;
d4205751d4643c272059a3728045929dd0e5e800Lennart Poettering waitret = apr_proc_wait(&proc, APR_NOWAIT);
d4205751d4643c272059a3728045929dd0e5e800Lennart Poettering if (waitret != APR_CHILD_NOTDONE) {
d4205751d4643c272059a3728045929dd0e5e800Lennart Poettering MPM_NOTE_CHILD_KILLED(i);
d4205751d4643c272059a3728045929dd0e5e800Lennart Poettering continue;
d4205751d4643c272059a3728045929dd0e5e800Lennart Poettering }
d4205751d4643c272059a3728045929dd0e5e800Lennart Poettering ++not_dead_yet;
772c552f763a125d17d814c4e5a35bbba49fc27aNis Martensen switch (tries) {
d4205751d4643c272059a3728045929dd0e5e800Lennart Poettering case 1: /* 16ms */
d4205751d4643c272059a3728045929dd0e5e800Lennart Poettering case 2: /* 82ms */
d4205751d4643c272059a3728045929dd0e5e800Lennart Poettering case 3: /* 344ms */
d4205751d4643c272059a3728045929dd0e5e800Lennart Poettering case 4: /* 16ms */
d4205751d4643c272059a3728045929dd0e5e800Lennart Poettering break;
d4205751d4643c272059a3728045929dd0e5e800Lennart Poettering case 5: /* 82ms */
772c552f763a125d17d814c4e5a35bbba49fc27aNis Martensen case 6: /* 344ms */
d4205751d4643c272059a3728045929dd0e5e800Lennart Poettering case 7: /* 1.4sec */
d4205751d4643c272059a3728045929dd0e5e800Lennart Poettering /* ok, now it's being annoying */
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering ap_log_error(APLOG_MARK, APLOG_NOERRNO|APLOG_WARNING,
d4205751d4643c272059a3728045929dd0e5e800Lennart Poettering 0, ap_server_conf,
d4205751d4643c272059a3728045929dd0e5e800Lennart Poettering "child process %ld still did not exit, sending a SIGTERM",
d4205751d4643c272059a3728045929dd0e5e800Lennart Poettering (long)pid);
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering kill(pid, SIGTERM);
d4205751d4643c272059a3728045929dd0e5e800Lennart Poettering break;
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering case 8: /* 6 sec */
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering /* die child scum */
d4205751d4643c272059a3728045929dd0e5e800Lennart Poettering ap_log_error(APLOG_MARK, APLOG_NOERRNO|APLOG_ERR,
d4205751d4643c272059a3728045929dd0e5e800Lennart Poettering 0, ap_server_conf,
d4205751d4643c272059a3728045929dd0e5e800Lennart Poettering "child process %ld still did not exit, sending a SIGKILL",
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering (long)pid);
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering#ifndef BEOS
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering kill(pid, SIGKILL);
d4205751d4643c272059a3728045929dd0e5e800Lennart Poettering#else
772c552f763a125d17d814c4e5a35bbba49fc27aNis Martensen /* sending a SIGKILL kills the entire team on BeOS, and as
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering * httpd thread is part of that team it removes any chance
772c552f763a125d17d814c4e5a35bbba49fc27aNis Martensen * of ever doing a restart. To counter this I'm changing to
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering * use a kinder, gentler way of killing a specific thread
ba9904e9ce0628cce3bbd8106f6dc6914e418edaLennart Poettering * that is just as effective.
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering */
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering kill_thread(pid);
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering#endif
d4205751d4643c272059a3728045929dd0e5e800Lennart Poettering break;
d4205751d4643c272059a3728045929dd0e5e800Lennart Poettering case 9: /* 14 sec */
d4205751d4643c272059a3728045929dd0e5e800Lennart Poettering /* gave it our best shot, but alas... If this really
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering * is a child we are trying to kill and it really hasn't
d4205751d4643c272059a3728045929dd0e5e800Lennart Poettering * exited, we will likely fail to bind to the port
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering * after the restart.
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering */
d4205751d4643c272059a3728045929dd0e5e800Lennart Poettering ap_log_error(APLOG_MARK, APLOG_NOERRNO|APLOG_ERR,
d4205751d4643c272059a3728045929dd0e5e800Lennart Poettering 0, ap_server_conf,
d4205751d4643c272059a3728045929dd0e5e800Lennart Poettering "could not make child process %ld exit, "
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering "attempting to continue anyway", (long)pid);
d4205751d4643c272059a3728045929dd0e5e800Lennart Poettering break;
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering }
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering }
d4205751d4643c272059a3728045929dd0e5e800Lennart Poettering#if APR_HAS_OTHER_CHILD
d4205751d4643c272059a3728045929dd0e5e800Lennart Poettering apr_proc_other_child_check();
d4205751d4643c272059a3728045929dd0e5e800Lennart Poettering#endif
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering if (!not_dead_yet) {
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering /* nothing left to wait for */
d4205751d4643c272059a3728045929dd0e5e800Lennart Poettering break;
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering }
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering }
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering}
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering#endif /* NEED_RECLAIM_CHILD_PROCESSES */
d4205751d4643c272059a3728045929dd0e5e800Lennart Poettering
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering/* number of calls to wait_or_timeout between writable probes */
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering#ifndef INTERVAL_OF_WRITABLE_PROBES
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering#define INTERVAL_OF_WRITABLE_PROBES 10
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering#endif
d4205751d4643c272059a3728045929dd0e5e800Lennart Poetteringstatic int wait_or_timeout_counter;
d4205751d4643c272059a3728045929dd0e5e800Lennart Poettering
d4205751d4643c272059a3728045929dd0e5e800Lennart Poetteringvoid ap_wait_or_timeout(apr_wait_t *status, apr_proc_t *ret, apr_pool_t *p)
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering{
d4205751d4643c272059a3728045929dd0e5e800Lennart Poettering apr_status_t rv;
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering
d4205751d4643c272059a3728045929dd0e5e800Lennart Poettering ++wait_or_timeout_counter;
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering if (wait_or_timeout_counter == INTERVAL_OF_WRITABLE_PROBES) {
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering wait_or_timeout_counter = 0;
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering }
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering rv = apr_proc_wait_all_procs(ret, status, APR_NOWAIT, p);
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering if (APR_STATUS_IS_EINTR(rv)) {
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering ret->pid = -1;
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering return;
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering }
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering if (APR_STATUS_IS_CHILD_DONE(rv)) {
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering return;
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering }
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering#ifdef NEED_WAITPID
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering if ((ret = reap_children(status)) > 0) {
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering return;
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering }
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering#endif
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering apr_sleep(SCOREBOARD_MAINTENANCE_INTERVAL);
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering ret->pid = -1;
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering return;
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering}
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poetteringvoid ap_process_child_status(apr_proc_t *pid, apr_wait_t status)
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering{
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering int signum = WTERMSIG(status);
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering const char *sigdesc = apr_signal_get_description(signum);
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering /* Child died... if it died due to a fatal error,
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering * we should simply bail out.
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering */
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering if ((WIFEXITED(status)) &&
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering WEXITSTATUS(status) == APEXIT_CHILDFATAL) {
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering ap_log_error(APLOG_MARK, APLOG_ALERT|APLOG_NOERRNO, 0, ap_server_conf,
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering "Child %ld returned a Fatal error..." APR_EOL_STR
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering "Apache is exiting!",
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering (long)pid->pid);
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering exit(APEXIT_CHILDFATAL);
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering }
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering if (WIFSIGNALED(status)) {
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering switch (signum) {
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering case SIGTERM:
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering case SIGHUP:
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering case SIGWINCH:
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering case SIGKILL:
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering break;
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering default:
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering#ifdef WCOREDUMP
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering if (WCOREDUMP(status)) {
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering ap_log_error(APLOG_MARK, APLOG_NOERRNO|APLOG_NOTICE,
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering 0, ap_server_conf,
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering "child pid %ld exit signal %s (%d), "
5190bbb240ab3360f7d37714d1c877858aef8c12Thomas Hindoe Paaboel Andersen "possible coredump in %s",
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering (long)pid->pid, sigdesc, signum,
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering ap_coredump_dir);
5190bbb240ab3360f7d37714d1c877858aef8c12Thomas Hindoe Paaboel Andersen }
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering else
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering#endif
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering {
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering ap_log_error(APLOG_MARK, APLOG_NOERRNO|APLOG_NOTICE,
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering 0, ap_server_conf,
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering "child pid %ld exit signal %s (%d)",
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering (long)pid->pid, sigdesc, signum);
5d6a86d7a034a1fb3d6e3f1b58e2c13739270894Lennart Poettering }
20ad4cfd8e5592f634f20468798cbc1055ab9fafZbigniew Jędrzejewski-Szmek }
20ad4cfd8e5592f634f20468798cbc1055ab9fafZbigniew Jędrzejewski-Szmek }
20ad4cfd8e5592f634f20468798cbc1055ab9fafZbigniew Jędrzejewski-Szmek}
20ad4cfd8e5592f634f20468798cbc1055ab9fafZbigniew Jędrzejewski-Szmek
20ad4cfd8e5592f634f20468798cbc1055ab9fafZbigniew Jędrzejewski-Szmek#if defined(TCP_NODELAY) && !defined(MPE) && !defined(TPF)
20ad4cfd8e5592f634f20468798cbc1055ab9fafZbigniew Jędrzejewski-Szmekvoid ap_sock_disable_nagle(apr_socket_t *s)
20ad4cfd8e5592f634f20468798cbc1055ab9fafZbigniew Jędrzejewski-Szmek{
20ad4cfd8e5592f634f20468798cbc1055ab9fafZbigniew Jędrzejewski-Szmek /* The Nagle algorithm says that we should delay sending partial
20ad4cfd8e5592f634f20468798cbc1055ab9fafZbigniew Jędrzejewski-Szmek * packets in hopes of getting more data. We don't want to do
20ad4cfd8e5592f634f20468798cbc1055ab9fafZbigniew Jędrzejewski-Szmek * this; we are not telnet. There are bad interactions between
20ad4cfd8e5592f634f20468798cbc1055ab9fafZbigniew Jędrzejewski-Szmek * persistent connections and Nagle's algorithm that have very severe
20ad4cfd8e5592f634f20468798cbc1055ab9fafZbigniew Jędrzejewski-Szmek * performance penalties. (Failing to disable Nagle is not much of a
20ad4cfd8e5592f634f20468798cbc1055ab9fafZbigniew Jędrzejewski-Szmek * problem with simple HTTP.)
9444b1f20e311f073864d81e913bd4f32fe95cfdLennart Poettering *
9444b1f20e311f073864d81e913bd4f32fe95cfdLennart Poettering * In spite of these problems, failure here is not a shooting offense.
9444b1f20e311f073864d81e913bd4f32fe95cfdLennart Poettering */
9444b1f20e311f073864d81e913bd4f32fe95cfdLennart Poettering apr_status_t status = apr_setsocketopt(s, APR_TCP_NODELAY, 1);
9444b1f20e311f073864d81e913bd4f32fe95cfdLennart Poettering
9444b1f20e311f073864d81e913bd4f32fe95cfdLennart Poettering if (status != APR_SUCCESS) {
9444b1f20e311f073864d81e913bd4f32fe95cfdLennart Poettering ap_log_error(APLOG_MARK, APLOG_WARNING, status, ap_server_conf,
9444b1f20e311f073864d81e913bd4f32fe95cfdLennart Poettering "setsockopt: (TCP_NODELAY)");
9444b1f20e311f073864d81e913bd4f32fe95cfdLennart Poettering }
9444b1f20e311f073864d81e913bd4f32fe95cfdLennart Poettering}
9444b1f20e311f073864d81e913bd4f32fe95cfdLennart Poettering#endif
9444b1f20e311f073864d81e913bd4f32fe95cfdLennart Poettering
9444b1f20e311f073864d81e913bd4f32fe95cfdLennart PoetteringAP_DECLARE(uid_t) ap_uname2id(const char *name)
9444b1f20e311f073864d81e913bd4f32fe95cfdLennart Poettering{
9444b1f20e311f073864d81e913bd4f32fe95cfdLennart Poettering struct passwd *ent;
9444b1f20e311f073864d81e913bd4f32fe95cfdLennart Poettering
9444b1f20e311f073864d81e913bd4f32fe95cfdLennart Poettering if (name[0] == '#')
9444b1f20e311f073864d81e913bd4f32fe95cfdLennart Poettering return (atoi(&name[1]));
if (!(ent = getpwnam(name))) {
ap_log_error(APLOG_MARK, APLOG_STARTUP | APLOG_NOERRNO, 0, NULL, "%s: bad user name %s", ap_server_argv0, name);
exit(1);
}
return (ent->pw_uid);
}
AP_DECLARE(gid_t) ap_gname2id(const char *name)
{
struct group *ent;
if (name[0] == '#')
return (atoi(&name[1]));
if (!(ent = getgrnam(name))) {
ap_log_error(APLOG_MARK, APLOG_STARTUP | APLOG_NOERRNO, 0, NULL, "%s: bad group name %s", ap_server_argv0, name); exit(1);
}
return (ent->gr_gid);
}
#ifndef HAVE_INITGROUPS
int initgroups(const char *name, gid_t basegid)
{
#if defined(QNX) || defined(MPE) || defined(BEOS) || defined(_OSD_POSIX) || defined(TPF) || defined(__TANDEM) || defined(OS2) || defined(WIN32)
/* QNX, MPE and BeOS do not appear to support supplementary groups. */
return 0;
#else /* ndef QNX */
gid_t groups[NGROUPS_MAX];
struct group *g;
int index = 0;
setgrent();
groups[index++] = basegid;
while (index < NGROUPS_MAX && ((g = getgrent()) != NULL))
if (g->gr_gid != basegid) {
char **names;
for (names = g->gr_mem; *names != NULL; ++names)
if (!strcmp(*names, name))
groups[index++] = g->gr_gid;
}
endgrent();
return setgroups(index, groups);
#endif /* def QNX */
}
#endif /* def NEED_INITGROUPS */
#ifdef AP_MPM_USES_POD
AP_DECLARE(apr_status_t) ap_mpm_pod_open(apr_pool_t *p, ap_pod_t **pod)
{
apr_status_t rv;
*pod = apr_palloc(p, sizeof(**pod));
rv = apr_file_pipe_create(&((*pod)->pod_in), &((*pod)->pod_out), p);
apr_file_pipe_timeout_set((*pod)->pod_out, 0);
(*pod)->p = p;
return rv;
}
AP_DECLARE(apr_status_t) ap_mpm_pod_check(ap_pod_t *pod)
{
char c;
apr_size_t len = 1;
apr_status_t rv;
rv = apr_file_read(pod->pod_out, &c, &len);
if ((rv == APR_SUCCESS) && (len == 1)) {
return APR_SUCCESS;
}
if (rv != APR_SUCCESS) {
return rv;
}
return AP_NORESTART;
}
AP_DECLARE(apr_status_t) ap_mpm_pod_close(ap_pod_t *pod)
{
apr_status_t rv;
rv = apr_file_close(pod->pod_out);
if (rv != APR_SUCCESS) {
return rv;
}
rv = apr_file_close(pod->pod_in);
if (rv != APR_SUCCESS) {
return rv;
}
return rv;
}
AP_DECLARE(apr_status_t) ap_mpm_pod_signal(ap_pod_t *pod)
{
apr_socket_t *sock;
apr_sockaddr_t *sa;
apr_status_t rv;
char char_of_death = '!';
apr_size_t one = 1;
do {
rv = apr_file_write(pod->pod_in, &char_of_death, &one);
} while (APR_STATUS_IS_EINTR(rv));
if (rv != APR_SUCCESS) {
ap_log_error(APLOG_MARK, APLOG_WARNING, rv, ap_server_conf,
"write pipe_of_death");
return rv;
}
apr_sockaddr_info_get(&sa, "127.0.0.1", APR_UNSPEC, ap_listeners->bind_addr->port, 0, pod->p);
rv = apr_socket_create(&sock, sa->family, SOCK_STREAM, pod->p);
if (rv != APR_SUCCESS) {
ap_log_error(APLOG_MARK, APLOG_WARNING, rv, ap_server_conf,
"get socket to connect to listener");
return rv;
}
rv = apr_connect(sock, sa);
if (rv != APR_SUCCESS) {
ap_log_error(APLOG_MARK, APLOG_WARNING, rv, ap_server_conf,
"connect to listener");
return rv;
}
apr_socket_close(sock);
return APR_SUCCESS;
}
AP_DECLARE(void) ap_mpm_pod_killpg(ap_pod_t *pod, int num)
{
int i;
for (i = 0; i < num; i++) {
ap_mpm_pod_signal(pod);
}
}
#endif