main.c revision a70ad967b8413f0977e92d086264c1ced00d40ed
/* Copyright (C) 2002 Timo Sirainen */
#include "common.h"
#include "ioloop.h"
#include "lib-signals.h"
#include "network.h"
#include "env-util.h"
#include "fd-close-on-exec.h"
#include "auth-process.h"
#include "login-process.h"
#include "mail-process.h"
#include "ssl-init.h"
#include "log.h"
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <fcntl.h>
#include <syslog.h>
const char *process_names[PROCESS_TYPE_MAX] = {
"unknown",
"auth",
"login",
"imap",
"pop3",
"ssl-param"
};
struct hash_table *pids;
int null_fd, inetd_login_fd;
{
size_t i;
for (i = 0; i < max_len; i++) {
if (str[i] == '\0')
return TRUE;
}
return FALSE;
}
void child_process_init_env(void)
{
/* remove all environment, we don't need them */
env_clean();
/* we'll log through master process */
env_put("LOG_TO_MASTER=1");
}
{
}
static void settings_reload(void)
{
i_warning("SIGHUP received - reloading configuration");
/* restart auth and login processes */
if (!master_settings_read(configfile))
i_warning("Invalid configuration, keeping old one");
}
{
switch (status) {
case FATAL_LOGOPEN:
return "Can't open log file";
case FATAL_LOGWRITE:
return "Can't write to log file";
case FATAL_LOGERROR:
return "Internal logging error";
case FATAL_OUTOFMEM:
return "Out of memory";
case FATAL_EXEC:
return "exec() failed";
case FATAL_DEFAULT:
return NULL;
}
return NULL;
}
{
const char *process_type_name, *msg;
int status, process_type;
if (lib_signal_hup != 0) {
lib_signal_hup = 0;
}
/* get the type and remove from hash */
if (process_type == PROCESS_TYPE_IMAP ||
if (process_type == PROCESS_TYPE_SSL_PARAM)
/* write errors to syslog */
if (status != 0) {
if (process_type == PROCESS_TYPE_LOGIN)
i_error("child %s (%s) returned error %d%s",
}
} else if (WIFSIGNALED(status)) {
if (process_type == PROCESS_TYPE_LOGIN)
i_error("child %s (%s) killed with signal %d",
}
}
i_warning("waitpid() failed: %m");
}
{
const char *p;
return NULL; /* defaults to "*" or "[::]" */
if (name[0] == '[') {
/* IPv6 address */
if (p == NULL)
p++;
if (*p == '\0')
p = NULL;
else if (*p != ':')
} else {
if (p != NULL)
}
if (p != NULL) {
}
/* IPv4 any */
return ip;
}
/* IPv6 any */
return ip;
}
/* Return the first IP if there happens to be multiple. */
if (ret != 0) {
i_fatal("Can't resolve address %s: %s",
}
if (ips_count < 1)
return ip;
}
{
const char *const *proto;
int *fd;
#ifdef HAVE_SSL
#else
ssl_port = 0;
#endif
/* resolve */
/* register wanted protocols */
}
!set->ssl_disable) {
}
}
!set->ssl_disable) {
}
} else {
}
continue;
if (*fd != -1)
if (port == 0)
else {
if (*fd == -1)
}
}
}
{
return TRUE;
return TRUE;
return FALSE;
}
{
return TRUE;
return TRUE;
}
return FALSE;
}
static void open_fds(void)
{
struct server_settings *server;
/* initialize fds. */
if (null_fd == -1)
/* make sure all fds between 0..3 are used. */
while (null_fd < 4) {
}
if (!IS_INETD()) {
}
}
/* close stdin and stdout. close stderr unless we're logging
i_fatal("dup2(0) failed: %m");
i_fatal("dup2(1) failed: %m");
if (!have_stderr(settings_root)) {
i_fatal("dup2(2) failed: %m");
}
}
{
else {
/* log to file or stderr */
}
}
static void main_init(void)
{
/* deny file access from everyone else except owner */
(void)umask(0077);
log_init();
ssl_init();
}
static void main_deinit(void)
{
if (lib_signal_kill != 0)
/* make sure we log if child processes died unexpectedly */
ssl_deinit();
i_error("close(null_fd) failed: %m");
log_deinit();
closelog();
}
{
if (pid < 0)
i_fatal("fork() failed: %m");
if (pid != 0)
_exit(0);
if (setsid() < 0)
i_fatal("setsid() failed: %m");
}
static void print_help(void)
{
printf("Usage: dovecot [-F] [-c <config file>]\n");
}
{
/* parse arguments */
int foreground = FALSE;
int i;
lib_init();
master_uid = geteuid();
inetd_login_fd = -1;
for (i = 1; i < argc; i++) {
/* foreground */
foreground = TRUE;
/* config file */
i++;
configfile = argv[i];
/* starting through inetd. */
inetd_login_fd = dup(0);
if (inetd_login_fd == -1)
i_fatal("dup(0) failed: %m");
foreground = TRUE;
return 0;
} else {
print_help();
}
}
/* read and verify settings before forking */
if (!master_settings_read(configfile))
open_fds();
/* we don't need any environment */
env_clean();
if (!foreground)
main_init();
main_deinit();
lib_deinit();
return 0;
}