2003-05-21 21:28:21 +08:00
|
|
|
|
2004-09-28 16:34:51 +08:00
|
|
|
/*
|
2004-09-30 00:00:49 +08:00
|
|
|
* Copyright (C) Igor Sysoev
|
2004-09-28 16:34:51 +08:00
|
|
|
*/
|
|
|
|
|
|
|
|
|
2003-05-21 21:28:21 +08:00
|
|
|
#include <ngx_config.h>
|
|
|
|
#include <ngx_core.h>
|
|
|
|
|
|
|
|
|
2004-06-30 23:30:41 +08:00
|
|
|
int ngx_ncpu;
|
2003-05-21 21:28:21 +08:00
|
|
|
int ngx_max_sockets;
|
|
|
|
int ngx_inherited_nonblocking;
|
|
|
|
|
|
|
|
|
2004-06-16 01:47:16 +08:00
|
|
|
#if (NGX_POSIX_IO)
|
|
|
|
|
|
|
|
ngx_os_io_t ngx_os_io = {
|
|
|
|
ngx_unix_recv,
|
|
|
|
ngx_readv_chain,
|
|
|
|
NULL,
|
|
|
|
ngx_writev_chain,
|
|
|
|
0
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
int ngx_os_init(ngx_log_t *log)
|
|
|
|
{
|
|
|
|
return ngx_posix_init(log);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
2003-12-15 04:10:27 +08:00
|
|
|
void ngx_signal_handler(int signo);
|
2003-07-04 00:30:22 +08:00
|
|
|
|
|
|
|
|
2003-12-15 04:10:27 +08:00
|
|
|
typedef struct {
|
|
|
|
int signo;
|
|
|
|
char *signame;
|
|
|
|
void (*handler)(int signo);
|
|
|
|
} ngx_signal_t;
|
2003-05-21 21:28:21 +08:00
|
|
|
|
2003-11-20 00:26:41 +08:00
|
|
|
|
2003-12-15 04:10:27 +08:00
|
|
|
ngx_signal_t signals[] = {
|
2004-01-06 04:55:48 +08:00
|
|
|
{ ngx_signal_value(NGX_RECONFIGURE_SIGNAL),
|
|
|
|
"SIG" ngx_value(NGX_RECONFIGURE_SIGNAL),
|
2003-12-15 04:10:27 +08:00
|
|
|
ngx_signal_handler },
|
2003-07-04 00:30:22 +08:00
|
|
|
|
2004-01-06 04:55:48 +08:00
|
|
|
{ ngx_signal_value(NGX_REOPEN_SIGNAL),
|
|
|
|
"SIG" ngx_value(NGX_REOPEN_SIGNAL),
|
2003-12-15 04:10:27 +08:00
|
|
|
ngx_signal_handler },
|
2003-07-04 00:30:22 +08:00
|
|
|
|
2004-01-14 00:43:23 +08:00
|
|
|
{ ngx_signal_value(NGX_NOACCEPT_SIGNAL),
|
|
|
|
"SIG" ngx_value(NGX_NOACCEPT_SIGNAL),
|
2004-01-06 04:55:48 +08:00
|
|
|
ngx_signal_handler },
|
|
|
|
|
|
|
|
{ ngx_signal_value(NGX_TERMINATE_SIGNAL),
|
|
|
|
"SIG" ngx_value(NGX_TERMINATE_SIGNAL),
|
2003-12-15 04:10:27 +08:00
|
|
|
ngx_signal_handler },
|
|
|
|
|
|
|
|
{ ngx_signal_value(NGX_SHUTDOWN_SIGNAL),
|
|
|
|
"SIG" ngx_value(NGX_SHUTDOWN_SIGNAL),
|
2004-01-06 04:55:48 +08:00
|
|
|
ngx_signal_handler },
|
|
|
|
|
|
|
|
{ ngx_signal_value(NGX_CHANGEBIN_SIGNAL),
|
|
|
|
"SIG" ngx_value(NGX_CHANGEBIN_SIGNAL),
|
2003-12-15 04:10:27 +08:00
|
|
|
ngx_signal_handler },
|
|
|
|
|
2004-01-29 04:38:54 +08:00
|
|
|
{ SIGALRM, "SIGALRM", ngx_signal_handler },
|
|
|
|
|
2004-01-14 05:33:59 +08:00
|
|
|
{ SIGINT, "SIGINT", ngx_signal_handler },
|
|
|
|
|
2004-06-15 15:55:11 +08:00
|
|
|
{ SIGIO, "SIGIO", ngx_signal_handler },
|
|
|
|
|
2004-01-09 05:02:06 +08:00
|
|
|
{ SIGCHLD, "SIGCHLD", ngx_signal_handler },
|
2003-12-15 04:10:27 +08:00
|
|
|
|
2004-01-09 05:02:06 +08:00
|
|
|
{ SIGPIPE, "SIGPIPE, SIG_IGN", SIG_IGN },
|
2003-12-15 04:10:27 +08:00
|
|
|
|
2004-01-09 05:02:06 +08:00
|
|
|
{ 0, NULL, NULL }
|
2003-12-15 04:10:27 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
int ngx_posix_init(ngx_log_t *log)
|
|
|
|
{
|
|
|
|
ngx_signal_t *sig;
|
|
|
|
struct rlimit rlmt;
|
|
|
|
struct sigaction sa;
|
|
|
|
|
2004-06-07 03:49:18 +08:00
|
|
|
ngx_pagesize = getpagesize();
|
|
|
|
|
2004-06-30 23:30:41 +08:00
|
|
|
if (ngx_ncpu == 0) {
|
|
|
|
ngx_ncpu = 1;
|
|
|
|
}
|
|
|
|
|
2003-12-15 04:10:27 +08:00
|
|
|
for (sig = signals; sig->signo != 0; sig++) {
|
|
|
|
ngx_memzero(&sa, sizeof(struct sigaction));
|
|
|
|
sa.sa_handler = sig->handler;
|
|
|
|
sigemptyset(&sa.sa_mask);
|
|
|
|
if (sigaction(sig->signo, &sa, NULL) == -1) {
|
|
|
|
ngx_log_error(NGX_LOG_EMERG, log, ngx_errno,
|
|
|
|
"sigaction(%s) failed", sig->signame);
|
|
|
|
return NGX_ERROR;
|
|
|
|
}
|
|
|
|
}
|
2003-05-21 21:28:21 +08:00
|
|
|
|
|
|
|
if (getrlimit(RLIMIT_NOFILE, &rlmt) == -1) {
|
|
|
|
ngx_log_error(NGX_LOG_ALERT, log, errno,
|
|
|
|
"getrlimit(RLIMIT_NOFILE) failed)");
|
|
|
|
return NGX_ERROR;
|
|
|
|
}
|
|
|
|
|
|
|
|
ngx_log_error(NGX_LOG_INFO, log, 0,
|
2003-11-26 04:44:56 +08:00
|
|
|
"getrlimit(RLIMIT_NOFILE): " RLIM_T_FMT ":" RLIM_T_FMT,
|
2003-05-21 21:28:21 +08:00
|
|
|
rlmt.rlim_cur, rlmt.rlim_max);
|
|
|
|
|
|
|
|
ngx_max_sockets = rlmt.rlim_cur;
|
|
|
|
|
|
|
|
#if (HAVE_INHERITED_NONBLOCK)
|
|
|
|
ngx_inherited_nonblocking = 1;
|
|
|
|
#else
|
|
|
|
ngx_inherited_nonblocking = 0;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
return NGX_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2003-12-15 04:10:27 +08:00
|
|
|
void ngx_signal_handler(int signo)
|
|
|
|
{
|
2004-01-09 05:02:06 +08:00
|
|
|
char *action;
|
2004-01-06 04:55:48 +08:00
|
|
|
struct timeval tv;
|
2004-01-14 05:33:59 +08:00
|
|
|
ngx_int_t ignore;
|
2004-01-07 00:49:34 +08:00
|
|
|
ngx_err_t err;
|
2004-01-06 04:55:48 +08:00
|
|
|
ngx_signal_t *sig;
|
2003-12-15 04:10:27 +08:00
|
|
|
|
2004-01-14 05:33:59 +08:00
|
|
|
ignore = 0;
|
2004-01-14 00:43:23 +08:00
|
|
|
|
2004-01-07 00:49:34 +08:00
|
|
|
err = ngx_errno;
|
|
|
|
|
2003-12-15 04:10:27 +08:00
|
|
|
for (sig = signals; sig->signo != 0; sig++) {
|
|
|
|
if (sig->signo == signo) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2004-01-06 04:55:48 +08:00
|
|
|
ngx_gettimeofday(&tv);
|
|
|
|
ngx_time_update(tv.tv_sec);
|
2003-12-19 16:15:11 +08:00
|
|
|
|
2004-01-09 05:02:06 +08:00
|
|
|
action = "";
|
2003-12-19 16:15:11 +08:00
|
|
|
|
2004-01-09 05:02:06 +08:00
|
|
|
switch (ngx_process) {
|
2003-12-15 04:10:27 +08:00
|
|
|
|
2004-01-09 05:02:06 +08:00
|
|
|
case NGX_PROCESS_MASTER:
|
2004-01-22 14:47:28 +08:00
|
|
|
case NGX_PROCESS_SINGLE:
|
2004-01-09 05:02:06 +08:00
|
|
|
switch (signo) {
|
2003-12-15 04:10:27 +08:00
|
|
|
|
2004-01-09 05:02:06 +08:00
|
|
|
case ngx_signal_value(NGX_SHUTDOWN_SIGNAL):
|
|
|
|
ngx_quit = 1;
|
2004-01-23 17:26:18 +08:00
|
|
|
action = ", shutting down";
|
2004-01-09 05:02:06 +08:00
|
|
|
break;
|
2003-12-15 04:10:27 +08:00
|
|
|
|
2004-01-09 05:02:06 +08:00
|
|
|
case ngx_signal_value(NGX_TERMINATE_SIGNAL):
|
2004-01-14 00:43:23 +08:00
|
|
|
case SIGINT:
|
2004-01-09 05:02:06 +08:00
|
|
|
ngx_terminate = 1;
|
|
|
|
action = ", exiting";
|
|
|
|
break;
|
2003-12-15 04:10:27 +08:00
|
|
|
|
2004-01-14 00:43:23 +08:00
|
|
|
case ngx_signal_value(NGX_NOACCEPT_SIGNAL):
|
|
|
|
ngx_noaccept = 1;
|
|
|
|
action = ", stop the accepting connections";
|
2004-01-09 05:02:06 +08:00
|
|
|
break;
|
|
|
|
|
|
|
|
case ngx_signal_value(NGX_RECONFIGURE_SIGNAL):
|
|
|
|
ngx_reconfigure = 1;
|
|
|
|
action = ", reconfiguring";
|
|
|
|
break;
|
|
|
|
|
|
|
|
case ngx_signal_value(NGX_REOPEN_SIGNAL):
|
2004-02-05 04:30:08 +08:00
|
|
|
ngx_reopen = 1;
|
|
|
|
action = ", reopen logs";
|
|
|
|
break;
|
2004-01-09 05:02:06 +08:00
|
|
|
|
|
|
|
case ngx_signal_value(NGX_CHANGEBIN_SIGNAL):
|
2004-02-06 00:58:36 +08:00
|
|
|
if (getppid() > 1 || ngx_new_binary > 0) {
|
|
|
|
|
2004-01-14 05:33:59 +08:00
|
|
|
/*
|
|
|
|
* Ignore the signal in the new binary if its parent is
|
|
|
|
* not the init process, i.e. the old binary's process
|
|
|
|
* is still running. Or ingore the signal in the old binary's
|
|
|
|
* process if the new binary's process is already running.
|
|
|
|
*/
|
|
|
|
|
|
|
|
action = ", ignoring";
|
|
|
|
ignore = 1;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2004-01-09 05:02:06 +08:00
|
|
|
ngx_change_binary = 1;
|
|
|
|
action = ", changing binary";
|
|
|
|
break;
|
|
|
|
|
2004-01-29 04:38:54 +08:00
|
|
|
case SIGALRM:
|
|
|
|
if (!ngx_terminate) {
|
|
|
|
ngx_timer = 1;
|
2004-06-15 15:55:11 +08:00
|
|
|
action = ", shutting down old worker processes";
|
2004-01-29 04:38:54 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
break;
|
|
|
|
|
2004-06-15 15:55:11 +08:00
|
|
|
case SIGIO:
|
|
|
|
ngx_sigio = 1;
|
|
|
|
break;
|
|
|
|
|
2004-01-09 05:02:06 +08:00
|
|
|
case SIGCHLD:
|
|
|
|
ngx_reap = 1;
|
|
|
|
break;
|
|
|
|
}
|
2003-07-04 00:30:22 +08:00
|
|
|
|
2004-01-06 04:55:48 +08:00
|
|
|
break;
|
2003-07-04 00:30:22 +08:00
|
|
|
|
2004-01-09 05:02:06 +08:00
|
|
|
case NGX_PROCESS_WORKER:
|
|
|
|
switch (signo) {
|
|
|
|
|
|
|
|
case ngx_signal_value(NGX_SHUTDOWN_SIGNAL):
|
|
|
|
ngx_quit = 1;
|
2004-01-23 17:26:18 +08:00
|
|
|
action = ", shutting down";
|
2004-01-09 05:02:06 +08:00
|
|
|
break;
|
|
|
|
|
|
|
|
case ngx_signal_value(NGX_TERMINATE_SIGNAL):
|
2004-01-14 00:43:23 +08:00
|
|
|
case SIGINT:
|
2004-01-09 05:02:06 +08:00
|
|
|
ngx_terminate = 1;
|
|
|
|
action = ", exiting";
|
|
|
|
break;
|
|
|
|
|
|
|
|
case ngx_signal_value(NGX_REOPEN_SIGNAL):
|
|
|
|
ngx_reopen = 1;
|
|
|
|
action = ", reopen logs";
|
|
|
|
break;
|
|
|
|
|
|
|
|
case ngx_signal_value(NGX_RECONFIGURE_SIGNAL):
|
2004-01-14 00:43:23 +08:00
|
|
|
case ngx_signal_value(NGX_NOACCEPT_SIGNAL):
|
2004-01-09 05:02:06 +08:00
|
|
|
case ngx_signal_value(NGX_CHANGEBIN_SIGNAL):
|
2004-06-15 15:55:11 +08:00
|
|
|
case SIGIO:
|
2004-01-09 05:02:06 +08:00
|
|
|
action = ", ignoring";
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2004-01-06 04:55:48 +08:00
|
|
|
break;
|
|
|
|
}
|
2004-01-07 00:49:34 +08:00
|
|
|
|
2004-01-09 05:02:06 +08:00
|
|
|
ngx_log_error(NGX_LOG_INFO, ngx_cycle->log, 0,
|
|
|
|
"signal %d (%s) received%s", signo, sig->signame, action);
|
|
|
|
|
2004-01-14 05:33:59 +08:00
|
|
|
if (ignore) {
|
|
|
|
ngx_log_error(NGX_LOG_CRIT, ngx_cycle->log, 0,
|
|
|
|
"the changing binary signal is ignored: "
|
|
|
|
"you should shutdown or terminate "
|
|
|
|
"before either old or new binary's process");
|
|
|
|
}
|
|
|
|
|
2004-01-09 05:02:06 +08:00
|
|
|
if (signo == SIGCHLD) {
|
|
|
|
ngx_process_get_status();
|
|
|
|
}
|
|
|
|
|
2004-01-07 00:49:34 +08:00
|
|
|
ngx_set_errno(err);
|
2003-07-04 00:30:22 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2003-05-21 21:28:21 +08:00
|
|
|
int ngx_posix_post_conf_init(ngx_log_t *log)
|
|
|
|
{
|
|
|
|
ngx_fd_t pp[2];
|
|
|
|
|
|
|
|
if (pipe(pp) == -1) {
|
|
|
|
ngx_log_error(NGX_LOG_EMERG, log, ngx_errno, "pipe() failed");
|
|
|
|
return NGX_ERROR;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (dup2(pp[1], STDERR_FILENO) == -1) {
|
|
|
|
ngx_log_error(NGX_LOG_EMERG, log, errno, "dup2(STDERR) failed");
|
|
|
|
return NGX_ERROR;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (pp[1] > STDERR_FILENO) {
|
|
|
|
if (close(pp[1]) == -1) {
|
|
|
|
ngx_log_error(NGX_LOG_EMERG, log, errno, "close() failed");
|
|
|
|
return NGX_ERROR;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return NGX_OK;
|
|
|
|
}
|