mirror of
https://gitee.com/acl-dev/acl.git
synced 2024-12-04 21:09:19 +08:00
204 lines
6.0 KiB
C++
204 lines
6.0 KiB
C++
/* System libraries. */
|
||
#include "stdafx.h"
|
||
|
||
#include <stdio.h>
|
||
#include <stdlib.h>
|
||
#include <string.h>
|
||
#include <sys/stat.h>
|
||
#include <syslog.h>
|
||
#include <signal.h>
|
||
#include <unistd.h>
|
||
#include <string.h>
|
||
#include <fcntl.h>
|
||
#include <limits.h>
|
||
|
||
#include "master/master_params.h"
|
||
#include "master/master.h"
|
||
|
||
const char *var_master_version = "version1.0";
|
||
char *var_master_procname;
|
||
|
||
/* usage - show hint and terminate */
|
||
|
||
static void usage(const char *me)
|
||
{
|
||
acl_msg_fatal("usage: %s [-c root_dir] [-l log_file] [-e exit_time]"
|
||
" [-D (debug)] [-t (test)] [-v] [-k (keep_mask)]", me);
|
||
}
|
||
|
||
/* main - main program */
|
||
|
||
int main(int argc, char **argv)
|
||
{
|
||
static ACL_VSTREAM *lock_fp = NULL;
|
||
int ch;
|
||
int fd;
|
||
int n, keep_mask = 0;
|
||
char *ptr;
|
||
ACL_WATCHDOG *watchdog;
|
||
ACL_VSTRING *strbuf;
|
||
|
||
/*
|
||
* Strip and save the process name for diagnostics etc.
|
||
*/
|
||
var_master_procname = acl_mystrdup(acl_safe_basename(argv[0]));
|
||
acl_var_master_pid = getpid();
|
||
|
||
acl_var_master_conf_dir = NULL;
|
||
acl_var_master_log_file = NULL;
|
||
|
||
while ((ch = getopt(argc, argv, "vc:l:k")) > 0) {
|
||
switch (ch) {
|
||
case 'v':
|
||
acl_msg_verbose++;
|
||
break;
|
||
case 'c':
|
||
acl_var_master_conf_dir = acl_mystrdup(optarg);
|
||
break;
|
||
case 'l':
|
||
acl_var_master_log_file = acl_mystrdup(optarg);
|
||
break;
|
||
case 'k':
|
||
keep_mask = 1;
|
||
break;
|
||
default:
|
||
usage(argv[0]);
|
||
/* NOTREACHED */
|
||
}
|
||
}
|
||
|
||
/* Initialize. */
|
||
if (!keep_mask)
|
||
umask(077); /* never fails! */
|
||
|
||
/* Don't die when a process goes away unexpectedly. */
|
||
signal(SIGPIPE, SIG_IGN);
|
||
|
||
if (acl_var_master_conf_dir == NULL || acl_var_master_log_file == NULL)
|
||
usage(argv[0]);
|
||
|
||
ptr = strrchr(acl_var_master_conf_dir, '/');
|
||
if (ptr != NULL && *(ptr + 1) == '\0') /* as "/opt/master/conf/" */
|
||
*ptr = '\0';
|
||
|
||
/*
|
||
* This program takes no other arguments.
|
||
*/
|
||
if (argc > optind)
|
||
usage(argv[0]);
|
||
|
||
/*
|
||
* When running a child process, don't leak any open files that were
|
||
* leaked to us by our own (privileged) parent process. Descriptors 0-2
|
||
* are taken care of after we have initialized error logging.
|
||
*
|
||
* Some systems such as AIX have a huge per-process open file limit.
|
||
* In those cases, limit the search for potential file descriptor leaks
|
||
* to just the first couple hundred.
|
||
*
|
||
* The Debian post-installation script passes an open file descriptor
|
||
* into the master process and waits forever for someone to close it.
|
||
* Because of this we have to close descriptors > 2, and pray that
|
||
* doing so does not break things.
|
||
*/
|
||
|
||
acl_closefrom(3); /* 0: stdin; 1: stdout; 2: stderr */
|
||
|
||
/* Initialize logging and exit handler. */
|
||
/* <20>ر<EFBFBD> 0, <20><>ʹ<EFBFBD><CAB9><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>ֹ<EFBFBD><D6B9><EFBFBD>־<EFBFBD>ļ<EFBFBD><C4BC><EFBFBD><EFBFBD><EFBFBD> :) */
|
||
close(0);
|
||
acl_msg_open(acl_var_master_log_file, var_master_procname);
|
||
|
||
/*
|
||
* If started from a terminal, get rid of any tty association. This also
|
||
* means that all errors and warnings must go to the syslog daemon.
|
||
*/
|
||
for (fd = 1; fd < 3; fd++) {
|
||
(void) close(fd);
|
||
if (open("/dev/null", (int) O_RDWR, 0) != fd)
|
||
acl_msg_fatal("open /dev/null: %s", acl_last_serror());
|
||
}
|
||
|
||
/*
|
||
* Make some room for plumbing with file descriptors. XXX This breaks
|
||
* when a service listens on many ports. In order to do this right we
|
||
* must change the master-child interface so that descriptors do not
|
||
* need to have fixed numbers.
|
||
*
|
||
* In a child we need two descriptors for the flow control pipe, one
|
||
* for child->master status updates and at least one for listening.
|
||
*/
|
||
for (n = 0; n < 5; n++) {
|
||
fd = dup(1);
|
||
if (acl_close_on_exec(fd, ACL_CLOSE_ON_EXEC) < 0)
|
||
acl_msg_fatal("dup(0): %s", acl_last_serror());
|
||
if (acl_msg_verbose)
|
||
acl_msg_info("dup(0), fd = %d, n = %d", fd, n);
|
||
}
|
||
|
||
acl_master_flow_init(); /* just for prefork service -- zsx, 2012.3.24 */
|
||
acl_master_config(); /* load configure and start all services processes */
|
||
|
||
/*
|
||
* Finish initialization, last part. We must process configuration
|
||
* files after processing command-line parameters, so that we get
|
||
* consistent results when we SIGHUP the server to reload
|
||
* configuration files.
|
||
*/
|
||
acl_master_sigsetup();
|
||
|
||
/* open pid file, lock it and write the master's pid value into it */
|
||
|
||
strbuf = acl_vstring_alloc(10);
|
||
|
||
lock_fp = acl_open_lock(acl_var_master_pid_file, O_RDWR | O_CREAT,
|
||
0644, strbuf);
|
||
if (lock_fp == 0)
|
||
acl_msg_fatal("%s(%d): open lock file %s: %s", __FILE__,
|
||
__LINE__, acl_var_master_pid_file,
|
||
acl_vstring_str(strbuf));
|
||
acl_vstring_sprintf(strbuf, "%*lu\n", (int) sizeof(unsigned long) * 4,
|
||
(unsigned long) acl_var_master_pid);
|
||
acl_vstream_writen(lock_fp, acl_vstring_str(strbuf),
|
||
ACL_VSTRING_LEN(strbuf));
|
||
acl_close_on_exec(ACL_VSTREAM_FILE(lock_fp), ACL_CLOSE_ON_EXEC);
|
||
acl_vstring_free(strbuf);
|
||
|
||
acl_msg_info("daemon started -- version %s, configuration %s",
|
||
var_master_version, acl_var_master_conf_dir);
|
||
|
||
/*
|
||
* Process events. The event handler will execute the read/write/timer
|
||
* action routines. Whenever something has happened, see if we received
|
||
* any signal in the mean time. Although the master process appears to do
|
||
* multiple things at the same time, it really is all a single thread, so
|
||
* that there are no concurrency conflicts within the master process.
|
||
*/
|
||
watchdog = acl_watchdog_create(1000, (ACL_WATCHDOG_FN) 0, (char *) 0);
|
||
|
||
for (;;) {
|
||
#ifdef HAS_VOLATILE_LOCKS
|
||
if (acl_myflock(ACL_VSTREAM_FILE(lock_fp), ACL_INTERNAL_LOCK,
|
||
ACL_FLOCK_OP_EXCLUSIVE) < 0)
|
||
{
|
||
acl_msg_fatal("refresh exclusive lock: %m");
|
||
}
|
||
#endif
|
||
|
||
acl_watchdog_start(watchdog); /* same as trigger servers */
|
||
|
||
acl_event_loop(acl_var_master_global_event);
|
||
if (acl_var_master_gotsighup) {
|
||
acl_msg_info("reload configuration");
|
||
acl_var_master_gotsighup = 0; /* this first */
|
||
acl_master_refresh(); /* then this */
|
||
}
|
||
if (acl_var_master_gotsigchld) {
|
||
if (acl_msg_verbose)
|
||
acl_msg_info("got sigchld");
|
||
acl_var_master_gotsigchld = 0; /* this first */
|
||
acl_master_reap_child(); /* then this */
|
||
}
|
||
}
|
||
}
|