main.c revision 6465fefe3bbbb6444e1515403e4922546f3e4861
b44cd8821087f2afebf85fec5b588f5720a9415cTom Gundersen/*-*- Mode: C; c-basic-offset: 8; indent-tabs-mode: nil -*-*/
b44cd8821087f2afebf85fec5b588f5720a9415cTom Gundersen This file is part of systemd.
b44cd8821087f2afebf85fec5b588f5720a9415cTom Gundersen Copyright 2010 Lennart Poettering
b44cd8821087f2afebf85fec5b588f5720a9415cTom Gundersen systemd is free software; you can redistribute it and/or modify it
b44cd8821087f2afebf85fec5b588f5720a9415cTom Gundersen under the terms of the GNU Lesser General Public License as published by
b44cd8821087f2afebf85fec5b588f5720a9415cTom Gundersen the Free Software Foundation; either version 2.1 of the License, or
b44cd8821087f2afebf85fec5b588f5720a9415cTom Gundersen (at your option) any later version.
b44cd8821087f2afebf85fec5b588f5720a9415cTom Gundersen systemd is distributed in the hope that it will be useful, but
b44cd8821087f2afebf85fec5b588f5720a9415cTom Gundersen WITHOUT ANY WARRANTY; without even the implied warranty of
b44cd8821087f2afebf85fec5b588f5720a9415cTom Gundersen MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
b44cd8821087f2afebf85fec5b588f5720a9415cTom Gundersen Lesser General Public License for more details.
b44cd8821087f2afebf85fec5b588f5720a9415cTom Gundersen You should have received a copy of the GNU Lesser General Public License
b44cd8821087f2afebf85fec5b588f5720a9415cTom Gundersen along with systemd; If not, see <http://www.gnu.org/licenses/>.
4afd3348c7506dd1d36305b7bcb9feb8952b9d6bLennart Poetteringstatic ManagerRunningAs arg_running_as = _MANAGER_RUNNING_AS_INVALID;
be077570f779664ed87b50f60608df9fbe258821Tom Gundersenstatic bool arg_dump_core = true;
be077570f779664ed87b50f60608df9fbe258821Tom Gundersenstatic bool arg_crash_shell = false;
be077570f779664ed87b50f60608df9fbe258821Tom Gundersenstatic bool arg_confirm_spawn = false;
be077570f779664ed87b50f60608df9fbe258821Tom Gundersenstatic ShowStatus arg_show_status = _SHOW_STATUS_UNSET;
be077570f779664ed87b50f60608df9fbe258821Tom Gundersenstatic bool arg_switched_root = false;
2dead8129f7b6fe644e17e1dc1739bebacfe1364Tom Gundersenstatic ExecOutput arg_default_std_output = EXEC_OUTPUT_JOURNAL;
2dead8129f7b6fe644e17e1dc1739bebacfe1364Tom Gundersenstatic ExecOutput arg_default_std_error = EXEC_OUTPUT_INHERIT;
2dead8129f7b6fe644e17e1dc1739bebacfe1364Tom Gundersenstatic usec_t arg_default_restart_usec = DEFAULT_RESTART_USEC;
2dead8129f7b6fe644e17e1dc1739bebacfe1364Tom Gundersenstatic usec_t arg_default_timeout_start_usec = DEFAULT_TIMEOUT_USEC;
2dead8129f7b6fe644e17e1dc1739bebacfe1364Tom Gundersenstatic usec_t arg_default_timeout_stop_usec = DEFAULT_TIMEOUT_USEC;
2dead8129f7b6fe644e17e1dc1739bebacfe1364Tom Gundersenstatic usec_t arg_default_start_limit_interval = DEFAULT_START_LIMIT_INTERVAL;
2dead8129f7b6fe644e17e1dc1739bebacfe1364Tom Gundersenstatic unsigned arg_default_start_limit_burst = DEFAULT_START_LIMIT_BURST;
2dead8129f7b6fe644e17e1dc1739bebacfe1364Tom Gundersenstatic usec_t arg_shutdown_watchdog = 10 * USEC_PER_MINUTE;
87322b3aee0dc649ff1ae7a403dcc9d7305baba2Tom Gundersenstatic struct rlimit *arg_default_rlimit[_RLIMIT_MAX] = {};
87322b3aee0dc649ff1ae7a403dcc9d7305baba2Tom Gundersenstatic uint64_t arg_capability_bounding_set_drop = 0;
87322b3aee0dc649ff1ae7a403dcc9d7305baba2Tom Gundersenstatic nsec_t arg_timer_slack_nsec = NSEC_INFINITY;
87322b3aee0dc649ff1ae7a403dcc9d7305baba2Tom Gundersenstatic usec_t arg_default_timer_accuracy_usec = 1 * USEC_PER_MINUTE;
be077570f779664ed87b50f60608df9fbe258821Tom Gundersenstatic bool arg_default_cpu_accounting = false;
be077570f779664ed87b50f60608df9fbe258821Tom Gundersenstatic bool arg_default_blockio_accounting = false;
be077570f779664ed87b50f60608df9fbe258821Tom Gundersenstatic bool arg_default_memory_accounting = false;
be077570f779664ed87b50f60608df9fbe258821Tom Gundersenstatic void pager_open_if_enabled(void) {
20af7091de0cdf92bf299addfc3f96c3ef805bd8Tom Gundersen /* Pass this on immediately, if this is not PID 1 */
be077570f779664ed87b50f60608df9fbe258821Tom Gundersen log_emergency("Caught <%s>, not dumping core.", signal_to_string(sig));
be077570f779664ed87b50f60608df9fbe258821Tom Gundersen /* We want to wait for the core process, hence let's enable SIGCHLD */
be077570f779664ed87b50f60608df9fbe258821Tom Gundersen log_emergency_errno(errno, "Caught <%s>, cannot fork for core dump: %m", signal_to_string(sig));
be077570f779664ed87b50f60608df9fbe258821Tom Gundersen else if (pid == 0) {
be077570f779664ed87b50f60608df9fbe258821Tom Gundersen /* Enable default signal handler for core dump */
be077570f779664ed87b50f60608df9fbe258821Tom Gundersen /* Don't limit the core dump size */
be077570f779664ed87b50f60608df9fbe258821Tom Gundersen /* Just to be sure... */
2dead8129f7b6fe644e17e1dc1739bebacfe1364Tom Gundersen /* Raise the signal again */
22805d9207d5242681e5667ee304572e4abf9b94Beniamino Galvani kill(pid, sig); /* raise() would kill the parent */
2dead8129f7b6fe644e17e1dc1739bebacfe1364Tom Gundersen assert_not_reached("We shouldn't be here...");
2dead8129f7b6fe644e17e1dc1739bebacfe1364Tom Gundersen /* Order things nicely. */
2dead8129f7b6fe644e17e1dc1739bebacfe1364Tom Gundersen log_emergency_errno(r, "Caught <%s>, waitpid() failed: %m", signal_to_string(sig));
bd57b45029ff25067704c9538e79f31e71c10045Tom Gundersen log_emergency("Caught <%s>, core dump failed (child "PID_FMT", code=%s, status=%i/%s).",
87322b3aee0dc649ff1ae7a403dcc9d7305baba2Tom Gundersen ? exit_status_to_string(status.si_status, EXIT_STATUS_FULL)
87322b3aee0dc649ff1ae7a403dcc9d7305baba2Tom Gundersen log_emergency("Caught <%s>, dumped core as pid "PID_FMT".", signal_to_string(sig), pid);
2dead8129f7b6fe644e17e1dc1739bebacfe1364Tom Gundersen .sa_flags = SA_NOCLDSTOP|SA_NOCLDWAIT|SA_RESTART,
b826ab586c9e0a9c0d438a75c28cf3a8ab485929Tom Gundersen /* Let the kernel reap children for us */
0cb3c286883b694fc52a18a3b559ff98931641f3Tom Gundersen assert_se(sigaction(SIGCHLD, &sa, NULL) == 0);
87322b3aee0dc649ff1ae7a403dcc9d7305baba2Tom Gundersen log_emergency_errno(errno, "Failed to fork off crash shell: %m");
87322b3aee0dc649ff1ae7a403dcc9d7305baba2Tom Gundersen else if (pid == 0) {
87322b3aee0dc649ff1ae7a403dcc9d7305baba2Tom Gundersen log_emergency_errno(errno, "execle() failed: %m");
87322b3aee0dc649ff1ae7a403dcc9d7305baba2Tom Gundersen log_info("Successfully spawned crash shell as PID "PID_FMT".", pid);
b826ab586c9e0a9c0d438a75c28cf3a8ab485929Tom Gundersenstatic void install_crash_handler(void) {
87322b3aee0dc649ff1ae7a403dcc9d7305baba2Tom Gundersen .sa_flags = SA_NODEFER, /* So that we can raise the signal again from the signal handler */
87322b3aee0dc649ff1ae7a403dcc9d7305baba2Tom Gundersen /* We ignore the return value here, since, we don't mind if we
87322b3aee0dc649ff1ae7a403dcc9d7305baba2Tom Gundersen * cannot set up a crash handler */
87322b3aee0dc649ff1ae7a403dcc9d7305baba2Tom Gundersen r = sigaction_many(&sa, SIGNALS_CRASH_HANDLER, -1);
b826ab586c9e0a9c0d438a75c28cf3a8ab485929Tom Gundersen log_debug_errno(r, "I had trouble setting up the crash handler, ignoring: %m");
87322b3aee0dc649ff1ae7a403dcc9d7305baba2Tom Gundersenstatic int console_setup(void) {
87322b3aee0dc649ff1ae7a403dcc9d7305baba2Tom Gundersen tty_fd = open_terminal("/dev/console", O_WRONLY|O_NOCTTY|O_CLOEXEC);
87322b3aee0dc649ff1ae7a403dcc9d7305baba2Tom Gundersen return log_error_errno(tty_fd, "Failed to open /dev/console: %m");
4afd3348c7506dd1d36305b7bcb9feb8952b9d6bLennart Poettering /* We don't want to force text mode. plymouth may be showing
6f08fb7b34b4eebf664de83c8bc7b9941658aeb4Tom Gundersen * pictures already from initrd. */
b44cd8821087f2afebf85fec5b588f5720a9415cTom Gundersen return log_error_errno(r, "Failed to reset /dev/console: %m");
6f08fb7b34b4eebf664de83c8bc7b9941658aeb4Tom Gundersenstatic int set_default_unit(const char *u) {
arg_default_unit = c;
static const char * const rlmap[] = {
if (!in_initrd())
if (in_initrd())
arg_dump_core = r;
arg_crash_shell = r;
arg_crash_chvt = r;
arg_confirm_spawn = r;
char **env;
if (env)
const char *filename, \
unsigned line, \
const char *section, \
unsigned section_line, \
const char *lvalue, \
int ltype, \
const char *rvalue, \
void *data, \
void *userdata) { \
static int config_parse_cpu_affinity2(
const char *unit,
const char *filename,
unsigned line,
const char *section,
unsigned section_line,
const char *lvalue,
int ltype,
const char *rvalue,
void *data,
void *userdata) {
size_t l;
unsigned ncpus = 0;
unsigned cpu;
return log_oom();
free(t);
return log_oom();
CPU_FREE(c);
return -EBADMSG;
CPU_FREE(c);
static int config_parse_show_status(
const char* unit,
const char *filename,
unsigned line,
const char *section,
unsigned section_line,
const char *lvalue,
int ltype,
const char *rvalue,
void *data,
void *userdata) {
static void strv_free_free(char ***l) {
strv_free(*i);
free(l);
static void free_join_controllers(void) {
const char *filename,
unsigned line,
const char *section,
unsigned section_line,
const char *lvalue,
int ltype,
const char *rvalue,
void *data,
void *userdata) {
return log_oom();
free(s);
strv_uniq(l);
strv_free(l);
if (!arg_join_controllers) {
if (!arg_join_controllers) {
strv_free(l);
return log_oom();
arg_join_controllers[0] = l;
strv_free(l);
return log_oom();
for (a = arg_join_controllers; *a; a++) {
if (strv_overlap(*a, l)) {
if (strv_extend_strv(&l, *a) < 0) {
strv_free(l);
strv_free_free(t);
return log_oom();
c = strv_copy(*a);
strv_free(l);
strv_free_free(t);
return log_oom();
t[n++] = strv_uniq(l);
arg_join_controllers = t;
static int parse_config_file(void) {
{ "Manager", "CapabilityBoundingSet", config_parse_bounding_set, 0, &arg_capability_bounding_set_drop },
#ifdef HAVE_SECCOMP
{ "Manager", "DefaultLimitSIGPENDING", config_parse_limit, 0, &arg_default_rlimit[RLIMIT_SIGPENDING] },
conf_dirs_nulstr = arg_running_as == MANAGER_SYSTEM ? CONF_DIRS_NULSTR("systemd/system.conf") : CONF_DIRS_NULSTR("systemd/user.conf");
opterr = 0;
case ARG_LOG_LEVEL:
case ARG_LOG_TARGET:
case ARG_LOG_COLOR:
if (optarg) {
log_show_color(true);
case ARG_LOG_LOCATION:
if (optarg) {
log_show_location(true);
case ARG_DEFAULT_STD_OUTPUT:
case ARG_DEFAULT_STD_ERROR:
case ARG_UNIT:
case ARG_SYSTEM:
case ARG_USER:
case ARG_TEST:
if (arg_no_pager < 0)
arg_no_pager = true;
case ARG_NO_PAGER:
arg_no_pager = true;
case ARG_VERSION:
case ARG_DUMP_CORE:
arg_dump_core = r;
case ARG_CRASH_SHELL:
arg_crash_shell = r;
case ARG_CONFIRM_SPAWN:
arg_confirm_spawn = r;
case ARG_SHOW_STATUS:
if (optarg) {
case ARG_DESERIALIZE: {
int fd;
FILE *f;
if (r < 0 || fd < 0) {
return r < 0 ? r : -EINVAL;
if (arg_serialization)
arg_serialization = f;
case ARG_SWITCHED_ROOT:
arg_switched_root = true;
if (arg_no_pager < 0)
arg_no_pager = true;
return -EINVAL;
return -EINVAL;
static int help(void) {
static int version(void) {
assert(m);
r = manager_open_serialization(m, &f);
goto fail;
m->n_reloading ++;
bus_manager_send_reloading(m, true);
if (!fds) {
r = -ENOMEM;
goto fail;
goto fail;
goto fail;
goto fail;
goto fail;
*_f = f;
fail:
fclose(f);
if (!rl)
return log_oom();
static void test_mtab(void) {
static const char ok[] =
if (r == -ENOENT)
"Please make sure to replace this file by a symlink to avoid incorrect or misleading mount(8) output.");
static void test_usr(void) {
log_warning("/usr appears to be on its own filesystem and is not already mounted. This is not a supported setup. "
"Consult http://freedesktop.org/wiki/Software/systemd/separate-usr-is-broken for more information.");
static int initialize_join_controllers(void) {
if (!arg_join_controllers)
return -ENOMEM;
return -ENOMEM;
#ifdef HAVE_SECCOMP
Iterator i;
void *id;
if (!seccomp)
return log_oom();
if (r == -EEXIST)
goto finish;
goto finish;
static int status_welcome(void) {
NULL);
if (r == -ENOENT) {
NULL);
if (r < 0 && r != -ENOENT)
static int write_container_id(void) {
if (isempty(c))
bool reexecute = false;
bool skip_setup = false;
bool loaded_policy = false;
bool arm_reboot_watchdog = false;
bool queue_default_job = false;
bool empty_etc = false;
#ifdef HAVE_SYSV_COMPAT
skip_setup = true;
skip_setup = false;
umask(0);
log_open();
if (in_initrd())
if (!skip_setup) {
goto finish;
} else if (ima_setup() < 0) {
goto finish;
goto finish;
goto finish;
if (!skip_setup) {
if (clock_is_localtime() > 0) {
int min;
} else if (!in_initrd()) {
log_open();
log_open();
goto finish;
r = initialize_join_controllers();
goto finish;
if (!skip_setup)
kmod_setup();
goto finish;
if (parse_config_file() < 0) {
goto finish;
goto finish;
geteuid() == 0) {
goto finish;
sd_booted() <= 0) {
goto finish;
running_in_chroot() > 0) {
goto finish;
skip_setup = true;
goto finish;
goto finish;
goto finish;
goto finish;
goto finish;
log_close();
goto finish;
if (arg_serialization)
setsid();
log_open();
goto finish;
if (virtualization)
if (in_initrd())
* /etc/machine-id as flag file. This allows container
if (empty_etc)
_cleanup_free_ char *t;
if (arg_show_status > 0)
test_mtab();
test_usr();
goto finish;
goto finish;
if (arg_syscall_archs) {
goto finish;
if (empty_etc) {
r = unit_file_preset_all(UNIT_FILE_SYSTEM, false, NULL, UNIT_FILE_PRESET_ENABLE_ONLY, false, NULL, 0);
goto finish;
if (arg_serialization) {
if (queue_default_job) {
goto finish;
goto finish;
goto finish;
if (r == -EPERM) {
log_debug("Default target could not be isolated, starting instead: %s", bus_error_message(&error, r));
goto finish;
goto finish;
goto finish;
r = manager_loop(m);
goto finish;
switch (m->exit_code) {
case MANAGER_EXIT:
goto finish;
case MANAGER_RELOAD:
r = manager_reload(m);
case MANAGER_REEXECUTE:
goto finish;
reexecute = true;
goto finish;
case MANAGER_SWITCH_ROOT:
if (!switch_root_init)
goto finish;
reexecute = true;
goto finish;
case MANAGER_REBOOT:
case MANAGER_POWEROFF:
case MANAGER_HALT:
case MANAGER_KEXEC: {
goto finish;
pager_close();
m = manager_free(m);
if (reexecute) {
const char **args;
unsigned i, args_size;
watchdog_close(true);
if (switch_root_dir) {
if (!switch_root_init) {
if (switch_root_dir)
if (switch_root_dir)
clearenv();
if (arg_serialization) {
if (fds) {
if (switch_root_init) {
if (arg_serialization) {
if (fds) {
#ifdef HAVE_VALGRIND_VALGRIND_H
if (shutdown_verb) {
switch (log_get_target()) {
case LOG_TARGET_KMSG:
case LOG_TARGET_CONSOLE:
if (log_get_show_color())
if (log_get_show_location())
watchdog_close(r < 0);
watchdog_close(true);
if (error_message)
freeze();
return retval;