main.c revision 3408ba015aee3a88c91962c028738be757779519
1a14a53cfded6e78c6e8dfb73fdff0039971d642Lennart Poettering/*-*- Mode: C; c-basic-offset: 8; indent-tabs-mode: nil -*-*/
1a14a53cfded6e78c6e8dfb73fdff0039971d642Lennart Poettering
1a14a53cfded6e78c6e8dfb73fdff0039971d642Lennart Poettering/***
1a14a53cfded6e78c6e8dfb73fdff0039971d642Lennart Poettering This file is part of systemd.
1a14a53cfded6e78c6e8dfb73fdff0039971d642Lennart Poettering
1a14a53cfded6e78c6e8dfb73fdff0039971d642Lennart Poettering Copyright 2010 Lennart Poettering
1a14a53cfded6e78c6e8dfb73fdff0039971d642Lennart Poettering
1a14a53cfded6e78c6e8dfb73fdff0039971d642Lennart Poettering systemd is free software; you can redistribute it and/or modify it
1a14a53cfded6e78c6e8dfb73fdff0039971d642Lennart Poettering under the terms of the GNU Lesser General Public License as published by
1a14a53cfded6e78c6e8dfb73fdff0039971d642Lennart Poettering the Free Software Foundation; either version 2.1 of the License, or
1a14a53cfded6e78c6e8dfb73fdff0039971d642Lennart Poettering (at your option) any later version.
1a14a53cfded6e78c6e8dfb73fdff0039971d642Lennart Poettering
1a14a53cfded6e78c6e8dfb73fdff0039971d642Lennart Poettering systemd is distributed in the hope that it will be useful, but
1a14a53cfded6e78c6e8dfb73fdff0039971d642Lennart Poettering WITHOUT ANY WARRANTY; without even the implied warranty of
1a14a53cfded6e78c6e8dfb73fdff0039971d642Lennart Poettering MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
1a14a53cfded6e78c6e8dfb73fdff0039971d642Lennart Poettering Lesser General Public License for more details.
1a14a53cfded6e78c6e8dfb73fdff0039971d642Lennart Poettering
1a14a53cfded6e78c6e8dfb73fdff0039971d642Lennart Poettering You should have received a copy of the GNU Lesser General Public License
1a14a53cfded6e78c6e8dfb73fdff0039971d642Lennart Poettering along with systemd; If not, see <http://www.gnu.org/licenses/>.
1a14a53cfded6e78c6e8dfb73fdff0039971d642Lennart Poettering***/
1a14a53cfded6e78c6e8dfb73fdff0039971d642Lennart Poettering
1a14a53cfded6e78c6e8dfb73fdff0039971d642Lennart Poettering#include <stdio.h>
1a14a53cfded6e78c6e8dfb73fdff0039971d642Lennart Poettering#include <errno.h>
1a14a53cfded6e78c6e8dfb73fdff0039971d642Lennart Poettering#include <string.h>
1a14a53cfded6e78c6e8dfb73fdff0039971d642Lennart Poettering#include <unistd.h>
1a14a53cfded6e78c6e8dfb73fdff0039971d642Lennart Poettering#include <sys/types.h>
96115cdfe0241ae9b4e7177cd3874c0a93d00b39Thomas Hindoe Paaboel Andersen#include <sys/stat.h>
1a14a53cfded6e78c6e8dfb73fdff0039971d642Lennart Poettering#include <getopt.h>
1af7211984a8dba3c5ba40fae794c4c55f5e6bd3Lennart Poettering#include <signal.h>
1af7211984a8dba3c5ba40fae794c4c55f5e6bd3Lennart Poettering#include <sys/wait.h>
1a14a53cfded6e78c6e8dfb73fdff0039971d642Lennart Poettering#include <fcntl.h>
1a14a53cfded6e78c6e8dfb73fdff0039971d642Lennart Poettering#include <sys/prctl.h>
1a14a53cfded6e78c6e8dfb73fdff0039971d642Lennart Poettering#include <sys/mount.h>
1a14a53cfded6e78c6e8dfb73fdff0039971d642Lennart Poettering
1ca208fb4f93e5869704af1812cbff7130a2fc03Zbigniew Jędrzejewski-Szmek#ifdef HAVE_VALGRIND_VALGRIND_H
1a14a53cfded6e78c6e8dfb73fdff0039971d642Lennart Poettering#include <valgrind/valgrind.h>
1a14a53cfded6e78c6e8dfb73fdff0039971d642Lennart Poettering#endif
9a5cb1371b6d8b0a04bd08665bcf9b06cb40c64cZbigniew Jędrzejewski-Szmek#ifdef HAVE_SECCOMP
e48fdd84432bbf9c2ecc339183258c7c33116032Lennart Poettering#include <seccomp.h>
e48fdd84432bbf9c2ecc339183258c7c33116032Lennart Poettering#endif
1af7211984a8dba3c5ba40fae794c4c55f5e6bd3Lennart Poettering
73b80ec2d999c45ce13f3e034704249d80829f7eLennart Poettering#include "sd-daemon.h"
1b9e5b126359a2a2ec37de1f94f046093abc74b8Lennart Poettering#include "sd-messages.h"
f9ac15442e4132f00eca5495d53c17062aae13e0Lennart Poettering#include "sd-bus.h"
1a14a53cfded6e78c6e8dfb73fdff0039971d642Lennart Poettering#include "manager.h"
1a14a53cfded6e78c6e8dfb73fdff0039971d642Lennart Poettering#include "log.h"
73b80ec2d999c45ce13f3e034704249d80829f7eLennart Poettering#include "load-fragment.h"
73b80ec2d999c45ce13f3e034704249d80829f7eLennart Poettering#include "fdset.h"
73b80ec2d999c45ce13f3e034704249d80829f7eLennart Poettering#include "special.h"
1a14a53cfded6e78c6e8dfb73fdff0039971d642Lennart Poettering#include "conf-parser.h"
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering#include "missing.h"
1a14a53cfded6e78c6e8dfb73fdff0039971d642Lennart Poettering#include "label.h"
1a14a53cfded6e78c6e8dfb73fdff0039971d642Lennart Poettering#include "build.h"
1a14a53cfded6e78c6e8dfb73fdff0039971d642Lennart Poettering#include "strv.h"
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering#include "def.h"
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering#include "virt.h"
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering#include "architecture.h"
1a14a53cfded6e78c6e8dfb73fdff0039971d642Lennart Poettering#include "watchdog.h"
1a14a53cfded6e78c6e8dfb73fdff0039971d642Lennart Poettering#include "path-util.h"
1a14a53cfded6e78c6e8dfb73fdff0039971d642Lennart Poettering#include "switch-root.h"
1a14a53cfded6e78c6e8dfb73fdff0039971d642Lennart Poettering#include "capability.h"
1a14a53cfded6e78c6e8dfb73fdff0039971d642Lennart Poettering#include "killall.h"
1a14a53cfded6e78c6e8dfb73fdff0039971d642Lennart Poettering#include "env-util.h"
1a14a53cfded6e78c6e8dfb73fdff0039971d642Lennart Poettering#include "clock-util.h"
1a14a53cfded6e78c6e8dfb73fdff0039971d642Lennart Poettering#include "fileio.h"
1a14a53cfded6e78c6e8dfb73fdff0039971d642Lennart Poettering#include "dbus-manager.h"
1a14a53cfded6e78c6e8dfb73fdff0039971d642Lennart Poettering#include "bus-error.h"
4a62c710b62a5a3c7a8a278b810b9d5b5a0c8f4fMichal Schmidt#include "bus-util.h"
4a62c710b62a5a3c7a8a278b810b9d5b5a0c8f4fMichal Schmidt
1a14a53cfded6e78c6e8dfb73fdff0039971d642Lennart Poettering#include "mount-setup.h"
1a14a53cfded6e78c6e8dfb73fdff0039971d642Lennart Poettering#include "loopback-setup.h"
1a14a53cfded6e78c6e8dfb73fdff0039971d642Lennart Poettering#include "hostname-setup.h"
1ebab691c7749779072741f71865bd0e055b7ecfLennart Poettering#include "machine-id-setup.h"
c3834f9b881f2b1a68dc7d797c134f0b66b47b57Lennart Poettering#include "selinux-setup.h"
c3834f9b881f2b1a68dc7d797c134f0b66b47b57Lennart Poettering#include "ima-setup.h"
ee530d8b73246f29781bd54a707ca75c7ef5a6cbLennart Poettering#include "smack-setup.h"
1a14a53cfded6e78c6e8dfb73fdff0039971d642Lennart Poettering#ifdef HAVE_KMOD
1a14a53cfded6e78c6e8dfb73fdff0039971d642Lennart Poettering#include "kmod-setup.h"
1a14a53cfded6e78c6e8dfb73fdff0039971d642Lennart Poettering#endif
1a14a53cfded6e78c6e8dfb73fdff0039971d642Lennart Poettering
4a62c710b62a5a3c7a8a278b810b9d5b5a0c8f4fMichal Schmidtstatic enum {
4a62c710b62a5a3c7a8a278b810b9d5b5a0c8f4fMichal Schmidt ACTION_RUN,
1a14a53cfded6e78c6e8dfb73fdff0039971d642Lennart Poettering ACTION_HELP,
1a14a53cfded6e78c6e8dfb73fdff0039971d642Lennart Poettering ACTION_VERSION,
1a14a53cfded6e78c6e8dfb73fdff0039971d642Lennart Poettering ACTION_TEST,
1a14a53cfded6e78c6e8dfb73fdff0039971d642Lennart Poettering ACTION_DUMP_CONFIGURATION_ITEMS,
1a14a53cfded6e78c6e8dfb73fdff0039971d642Lennart Poettering ACTION_DONE
1a14a53cfded6e78c6e8dfb73fdff0039971d642Lennart Poettering} arg_action = ACTION_RUN;
4a62c710b62a5a3c7a8a278b810b9d5b5a0c8f4fMichal Schmidtstatic char *arg_default_unit = NULL;
4a62c710b62a5a3c7a8a278b810b9d5b5a0c8f4fMichal Schmidtstatic SystemdRunningAs arg_running_as = _SYSTEMD_RUNNING_AS_INVALID;
1a14a53cfded6e78c6e8dfb73fdff0039971d642Lennart Poetteringstatic bool arg_dump_core = true;
1a14a53cfded6e78c6e8dfb73fdff0039971d642Lennart Poetteringstatic bool arg_crash_shell = false;
1a14a53cfded6e78c6e8dfb73fdff0039971d642Lennart Poetteringstatic int arg_crash_chvt = -1;
1a14a53cfded6e78c6e8dfb73fdff0039971d642Lennart Poetteringstatic bool arg_confirm_spawn = false;
cca1dfddd4ce4357113663532696488427cc54e4Lennart Poetteringstatic ShowStatus arg_show_status = _SHOW_STATUS_UNSET;
1af7211984a8dba3c5ba40fae794c4c55f5e6bd3Lennart Poetteringstatic bool arg_switched_root = false;
1af7211984a8dba3c5ba40fae794c4c55f5e6bd3Lennart Poetteringstatic char ***arg_join_controllers = NULL;
1af7211984a8dba3c5ba40fae794c4c55f5e6bd3Lennart Poetteringstatic ExecOutput arg_default_std_output = EXEC_OUTPUT_JOURNAL;
1af7211984a8dba3c5ba40fae794c4c55f5e6bd3Lennart Poetteringstatic ExecOutput arg_default_std_error = EXEC_OUTPUT_INHERIT;
1af7211984a8dba3c5ba40fae794c4c55f5e6bd3Lennart Poetteringstatic usec_t arg_default_restart_usec = DEFAULT_RESTART_USEC;
1af7211984a8dba3c5ba40fae794c4c55f5e6bd3Lennart Poetteringstatic usec_t arg_default_timeout_start_usec = DEFAULT_TIMEOUT_USEC;
1af7211984a8dba3c5ba40fae794c4c55f5e6bd3Lennart Poetteringstatic usec_t arg_default_timeout_stop_usec = DEFAULT_TIMEOUT_USEC;
1af7211984a8dba3c5ba40fae794c4c55f5e6bd3Lennart Poetteringstatic usec_t arg_default_start_limit_interval = DEFAULT_START_LIMIT_INTERVAL;
1af7211984a8dba3c5ba40fae794c4c55f5e6bd3Lennart Poetteringstatic unsigned arg_default_start_limit_burst = DEFAULT_START_LIMIT_BURST;
1af7211984a8dba3c5ba40fae794c4c55f5e6bd3Lennart Poetteringstatic usec_t arg_runtime_watchdog = 0;
1af7211984a8dba3c5ba40fae794c4c55f5e6bd3Lennart Poetteringstatic usec_t arg_shutdown_watchdog = 10 * USEC_PER_MINUTE;
1af7211984a8dba3c5ba40fae794c4c55f5e6bd3Lennart Poetteringstatic char **arg_default_environment = NULL;
1af7211984a8dba3c5ba40fae794c4c55f5e6bd3Lennart Poetteringstatic struct rlimit *arg_default_rlimit[_RLIMIT_MAX] = {};
1af7211984a8dba3c5ba40fae794c4c55f5e6bd3Lennart Poetteringstatic uint64_t arg_capability_bounding_set_drop = 0;
1af7211984a8dba3c5ba40fae794c4c55f5e6bd3Lennart Poetteringstatic nsec_t arg_timer_slack_nsec = (nsec_t) -1;
1af7211984a8dba3c5ba40fae794c4c55f5e6bd3Lennart Poetteringstatic usec_t arg_default_timer_accuracy_usec = 1 * USEC_PER_MINUTE;
1af7211984a8dba3c5ba40fae794c4c55f5e6bd3Lennart Poetteringstatic Set* arg_syscall_archs = NULL;
1af7211984a8dba3c5ba40fae794c4c55f5e6bd3Lennart Poetteringstatic FILE* arg_serialization = NULL;
1af7211984a8dba3c5ba40fae794c4c55f5e6bd3Lennart Poetteringstatic bool arg_default_cpu_accounting = false;
1af7211984a8dba3c5ba40fae794c4c55f5e6bd3Lennart Poetteringstatic bool arg_default_blockio_accounting = false;
1af7211984a8dba3c5ba40fae794c4c55f5e6bd3Lennart Poetteringstatic bool arg_default_memory_accounting = false;
1af7211984a8dba3c5ba40fae794c4c55f5e6bd3Lennart Poettering
821b2e792159e237a1e5a1ea4bb6ae2e55d64be5Lukas Nykrynstatic void nop_handler(int sig) {}
1af7211984a8dba3c5ba40fae794c4c55f5e6bd3Lennart Poettering
1af7211984a8dba3c5ba40fae794c4c55f5e6bd3Lennart Poetteringnoreturn static void crash(int sig) {
1af7211984a8dba3c5ba40fae794c4c55f5e6bd3Lennart Poettering
4a62c710b62a5a3c7a8a278b810b9d5b5a0c8f4fMichal Schmidt if (getpid() != 1)
4a62c710b62a5a3c7a8a278b810b9d5b5a0c8f4fMichal Schmidt /* Pass this on immediately, if this is not PID 1 */
1af7211984a8dba3c5ba40fae794c4c55f5e6bd3Lennart Poettering raise(sig);
1af7211984a8dba3c5ba40fae794c4c55f5e6bd3Lennart Poettering else if (!arg_dump_core)
1af7211984a8dba3c5ba40fae794c4c55f5e6bd3Lennart Poettering log_error("Caught <%s>, not dumping core.", signal_to_string(sig));
1af7211984a8dba3c5ba40fae794c4c55f5e6bd3Lennart Poettering else {
1af7211984a8dba3c5ba40fae794c4c55f5e6bd3Lennart Poettering struct sigaction sa = {
c3834f9b881f2b1a68dc7d797c134f0b66b47b57Lennart Poettering .sa_handler = nop_handler,
1af7211984a8dba3c5ba40fae794c4c55f5e6bd3Lennart Poettering .sa_flags = SA_NOCLDSTOP|SA_RESTART,
1af7211984a8dba3c5ba40fae794c4c55f5e6bd3Lennart Poettering };
1af7211984a8dba3c5ba40fae794c4c55f5e6bd3Lennart Poettering pid_t pid;
1af7211984a8dba3c5ba40fae794c4c55f5e6bd3Lennart Poettering
1af7211984a8dba3c5ba40fae794c4c55f5e6bd3Lennart Poettering /* We want to wait for the core process, hence let's enable SIGCHLD */
1af7211984a8dba3c5ba40fae794c4c55f5e6bd3Lennart Poettering sigaction(SIGCHLD, &sa, NULL);
1af7211984a8dba3c5ba40fae794c4c55f5e6bd3Lennart Poettering
1af7211984a8dba3c5ba40fae794c4c55f5e6bd3Lennart Poettering pid = fork();
1af7211984a8dba3c5ba40fae794c4c55f5e6bd3Lennart Poettering if (pid < 0)
1af7211984a8dba3c5ba40fae794c4c55f5e6bd3Lennart Poettering log_error("Caught <%s>, cannot fork for core dump: %m", signal_to_string(sig));
cca1dfddd4ce4357113663532696488427cc54e4Lennart Poettering
1af7211984a8dba3c5ba40fae794c4c55f5e6bd3Lennart Poettering else if (pid == 0) {
1af7211984a8dba3c5ba40fae794c4c55f5e6bd3Lennart Poettering struct rlimit rl = {};
cca1dfddd4ce4357113663532696488427cc54e4Lennart Poettering
1af7211984a8dba3c5ba40fae794c4c55f5e6bd3Lennart Poettering /* Enable default signal handler for core dump */
1af7211984a8dba3c5ba40fae794c4c55f5e6bd3Lennart Poettering zero(sa);
1af7211984a8dba3c5ba40fae794c4c55f5e6bd3Lennart Poettering sa.sa_handler = SIG_DFL;
4a62c710b62a5a3c7a8a278b810b9d5b5a0c8f4fMichal Schmidt sigaction(sig, &sa, NULL);
4a62c710b62a5a3c7a8a278b810b9d5b5a0c8f4fMichal Schmidt
1af7211984a8dba3c5ba40fae794c4c55f5e6bd3Lennart Poettering /* Don't limit the core dump size */
63c372cb9df3bee01e3bf8cd7f96f336bddda846Lennart Poettering rl.rlim_cur = RLIM_INFINITY;
1af7211984a8dba3c5ba40fae794c4c55f5e6bd3Lennart Poettering rl.rlim_max = RLIM_INFINITY;
1af7211984a8dba3c5ba40fae794c4c55f5e6bd3Lennart Poettering setrlimit(RLIMIT_CORE, &rl);
1af7211984a8dba3c5ba40fae794c4c55f5e6bd3Lennart Poettering
1af7211984a8dba3c5ba40fae794c4c55f5e6bd3Lennart Poettering /* Just to be sure... */
1af7211984a8dba3c5ba40fae794c4c55f5e6bd3Lennart Poettering chdir("/");
1af7211984a8dba3c5ba40fae794c4c55f5e6bd3Lennart Poettering
4a62c710b62a5a3c7a8a278b810b9d5b5a0c8f4fMichal Schmidt /* Raise the signal again */
4a62c710b62a5a3c7a8a278b810b9d5b5a0c8f4fMichal Schmidt raise(sig);
1af7211984a8dba3c5ba40fae794c4c55f5e6bd3Lennart Poettering
1af7211984a8dba3c5ba40fae794c4c55f5e6bd3Lennart Poettering assert_not_reached("We shouldn't be here...");
1af7211984a8dba3c5ba40fae794c4c55f5e6bd3Lennart Poettering _exit(1);
1af7211984a8dba3c5ba40fae794c4c55f5e6bd3Lennart Poettering
1af7211984a8dba3c5ba40fae794c4c55f5e6bd3Lennart Poettering } else {
1af7211984a8dba3c5ba40fae794c4c55f5e6bd3Lennart Poettering siginfo_t status;
1af7211984a8dba3c5ba40fae794c4c55f5e6bd3Lennart Poettering int r;
4a62c710b62a5a3c7a8a278b810b9d5b5a0c8f4fMichal Schmidt
4a62c710b62a5a3c7a8a278b810b9d5b5a0c8f4fMichal Schmidt /* Order things nicely. */
1af7211984a8dba3c5ba40fae794c4c55f5e6bd3Lennart Poettering r = wait_for_terminate(pid, &status);
1af7211984a8dba3c5ba40fae794c4c55f5e6bd3Lennart Poettering if (r < 0)
1af7211984a8dba3c5ba40fae794c4c55f5e6bd3Lennart Poettering log_error("Caught <%s>, waitpid() failed: %s", signal_to_string(sig), strerror(-r));
1af7211984a8dba3c5ba40fae794c4c55f5e6bd3Lennart Poettering else if (status.si_code != CLD_DUMPED)
1af7211984a8dba3c5ba40fae794c4c55f5e6bd3Lennart Poettering log_error("Caught <%s>, core dump failed.", signal_to_string(sig));
1af7211984a8dba3c5ba40fae794c4c55f5e6bd3Lennart Poettering else
1af7211984a8dba3c5ba40fae794c4c55f5e6bd3Lennart Poettering log_error("Caught <%s>, dumped core as pid "PID_FMT".", signal_to_string(sig), pid);
4a62c710b62a5a3c7a8a278b810b9d5b5a0c8f4fMichal Schmidt }
4a62c710b62a5a3c7a8a278b810b9d5b5a0c8f4fMichal Schmidt }
1af7211984a8dba3c5ba40fae794c4c55f5e6bd3Lennart Poettering
1af7211984a8dba3c5ba40fae794c4c55f5e6bd3Lennart Poettering if (arg_crash_chvt)
1af7211984a8dba3c5ba40fae794c4c55f5e6bd3Lennart Poettering chvt(arg_crash_chvt);
1af7211984a8dba3c5ba40fae794c4c55f5e6bd3Lennart Poettering
1af7211984a8dba3c5ba40fae794c4c55f5e6bd3Lennart Poettering if (arg_crash_shell) {
1af7211984a8dba3c5ba40fae794c4c55f5e6bd3Lennart Poettering struct sigaction sa = {
1af7211984a8dba3c5ba40fae794c4c55f5e6bd3Lennart Poettering .sa_handler = SIG_IGN,
1af7211984a8dba3c5ba40fae794c4c55f5e6bd3Lennart Poettering .sa_flags = SA_NOCLDSTOP|SA_NOCLDWAIT|SA_RESTART,
1af7211984a8dba3c5ba40fae794c4c55f5e6bd3Lennart Poettering };
1af7211984a8dba3c5ba40fae794c4c55f5e6bd3Lennart Poettering pid_t pid;
1af7211984a8dba3c5ba40fae794c4c55f5e6bd3Lennart Poettering
23bbb0de4e3f85d9704a5c12a5afa2dfa0159e41Michal Schmidt log_info("Executing crash shell in 10s...");
23bbb0de4e3f85d9704a5c12a5afa2dfa0159e41Michal Schmidt sleep(10);
1af7211984a8dba3c5ba40fae794c4c55f5e6bd3Lennart Poettering
1af7211984a8dba3c5ba40fae794c4c55f5e6bd3Lennart Poettering /* Let the kernel reap children for us */
1af7211984a8dba3c5ba40fae794c4c55f5e6bd3Lennart Poettering assert_se(sigaction(SIGCHLD, &sa, NULL) == 0);
1af7211984a8dba3c5ba40fae794c4c55f5e6bd3Lennart Poettering
1af7211984a8dba3c5ba40fae794c4c55f5e6bd3Lennart Poettering pid = fork();
1af7211984a8dba3c5ba40fae794c4c55f5e6bd3Lennart Poettering if (pid < 0)
1af7211984a8dba3c5ba40fae794c4c55f5e6bd3Lennart Poettering log_error("Failed to fork off crash shell: %m");
1af7211984a8dba3c5ba40fae794c4c55f5e6bd3Lennart Poettering else if (pid == 0) {
1af7211984a8dba3c5ba40fae794c4c55f5e6bd3Lennart Poettering make_console_stdio();
73b80ec2d999c45ce13f3e034704249d80829f7eLennart Poettering execl("/bin/sh", "/bin/sh", NULL);
73b80ec2d999c45ce13f3e034704249d80829f7eLennart Poettering
73b80ec2d999c45ce13f3e034704249d80829f7eLennart Poettering log_error("execl() failed: %m");
73b80ec2d999c45ce13f3e034704249d80829f7eLennart Poettering _exit(1);
73b80ec2d999c45ce13f3e034704249d80829f7eLennart Poettering }
cca1dfddd4ce4357113663532696488427cc54e4Lennart Poettering
73b80ec2d999c45ce13f3e034704249d80829f7eLennart Poettering log_info("Successfully spawned crash shell as pid "PID_FMT".", pid);
73b80ec2d999c45ce13f3e034704249d80829f7eLennart Poettering }
73b80ec2d999c45ce13f3e034704249d80829f7eLennart Poettering
1af7211984a8dba3c5ba40fae794c4c55f5e6bd3Lennart Poettering log_info("Freezing execution.");
1a14a53cfded6e78c6e8dfb73fdff0039971d642Lennart Poettering freeze();
e48fdd84432bbf9c2ecc339183258c7c33116032Lennart Poettering}
1a14a53cfded6e78c6e8dfb73fdff0039971d642Lennart Poettering
1af7211984a8dba3c5ba40fae794c4c55f5e6bd3Lennart Poetteringstatic void install_crash_handler(void) {
1af7211984a8dba3c5ba40fae794c4c55f5e6bd3Lennart Poettering struct sigaction sa = {
1af7211984a8dba3c5ba40fae794c4c55f5e6bd3Lennart Poettering .sa_handler = crash,
1af7211984a8dba3c5ba40fae794c4c55f5e6bd3Lennart Poettering .sa_flags = SA_NODEFER,
1af7211984a8dba3c5ba40fae794c4c55f5e6bd3Lennart Poettering };
73b80ec2d999c45ce13f3e034704249d80829f7eLennart Poettering
1a14a53cfded6e78c6e8dfb73fdff0039971d642Lennart Poettering sigaction_many(&sa, SIGNALS_CRASH_HANDLER, -1);
73b80ec2d999c45ce13f3e034704249d80829f7eLennart Poettering}
1af7211984a8dba3c5ba40fae794c4c55f5e6bd3Lennart Poettering
cca1dfddd4ce4357113663532696488427cc54e4Lennart Poetteringstatic int console_setup(bool do_reset) {
1af7211984a8dba3c5ba40fae794c4c55f5e6bd3Lennart Poettering int tty_fd, r;
1af7211984a8dba3c5ba40fae794c4c55f5e6bd3Lennart Poettering
1af7211984a8dba3c5ba40fae794c4c55f5e6bd3Lennart Poettering /* If we are init, we connect stdin/stdout/stderr to /dev/null
1af7211984a8dba3c5ba40fae794c4c55f5e6bd3Lennart Poettering * and make sure we don't have a controlling tty. */
1af7211984a8dba3c5ba40fae794c4c55f5e6bd3Lennart Poettering
1af7211984a8dba3c5ba40fae794c4c55f5e6bd3Lennart Poettering release_terminal();
1af7211984a8dba3c5ba40fae794c4c55f5e6bd3Lennart Poettering
e48fdd84432bbf9c2ecc339183258c7c33116032Lennart Poettering if (!do_reset)
1a14a53cfded6e78c6e8dfb73fdff0039971d642Lennart Poettering return 0;
1a14a53cfded6e78c6e8dfb73fdff0039971d642Lennart Poettering
1a14a53cfded6e78c6e8dfb73fdff0039971d642Lennart Poettering tty_fd = open_terminal("/dev/console", O_WRONLY|O_NOCTTY|O_CLOEXEC);
e48fdd84432bbf9c2ecc339183258c7c33116032Lennart Poettering if (tty_fd < 0) {
e48fdd84432bbf9c2ecc339183258c7c33116032Lennart Poettering log_error("Failed to open /dev/console: %s", strerror(-tty_fd));
e48fdd84432bbf9c2ecc339183258c7c33116032Lennart Poettering return -tty_fd;
e48fdd84432bbf9c2ecc339183258c7c33116032Lennart Poettering }
e48fdd84432bbf9c2ecc339183258c7c33116032Lennart Poettering
4a62c710b62a5a3c7a8a278b810b9d5b5a0c8f4fMichal Schmidt /* We don't want to force text mode.
4a62c710b62a5a3c7a8a278b810b9d5b5a0c8f4fMichal Schmidt * plymouth may be showing pictures already from initrd. */
1a14a53cfded6e78c6e8dfb73fdff0039971d642Lennart Poettering r = reset_terminal_fd(tty_fd, false);
1a14a53cfded6e78c6e8dfb73fdff0039971d642Lennart Poettering if (r < 0)
1a14a53cfded6e78c6e8dfb73fdff0039971d642Lennart Poettering log_error("Failed to reset /dev/console: %s", strerror(-r));
1a14a53cfded6e78c6e8dfb73fdff0039971d642Lennart Poettering
c3834f9b881f2b1a68dc7d797c134f0b66b47b57Lennart Poettering safe_close(tty_fd);
c3834f9b881f2b1a68dc7d797c134f0b66b47b57Lennart Poettering return r;
e48fdd84432bbf9c2ecc339183258c7c33116032Lennart Poettering}
e48fdd84432bbf9c2ecc339183258c7c33116032Lennart Poettering
73b80ec2d999c45ce13f3e034704249d80829f7eLennart Poetteringstatic int set_default_unit(const char *u) {
73b80ec2d999c45ce13f3e034704249d80829f7eLennart Poettering char *c;
73b80ec2d999c45ce13f3e034704249d80829f7eLennart Poettering
e48fdd84432bbf9c2ecc339183258c7c33116032Lennart Poettering assert(u);
e48fdd84432bbf9c2ecc339183258c7c33116032Lennart Poettering
e48fdd84432bbf9c2ecc339183258c7c33116032Lennart Poettering c = strdup(u);
e48fdd84432bbf9c2ecc339183258c7c33116032Lennart Poettering if (!c)
e48fdd84432bbf9c2ecc339183258c7c33116032Lennart Poettering return -ENOMEM;
e48fdd84432bbf9c2ecc339183258c7c33116032Lennart Poettering
1a14a53cfded6e78c6e8dfb73fdff0039971d642Lennart Poettering free(arg_default_unit);
1a14a53cfded6e78c6e8dfb73fdff0039971d642Lennart Poettering arg_default_unit = c;
1af7211984a8dba3c5ba40fae794c4c55f5e6bd3Lennart Poettering
1af7211984a8dba3c5ba40fae794c4c55f5e6bd3Lennart Poettering return 0;
1af7211984a8dba3c5ba40fae794c4c55f5e6bd3Lennart Poettering}
73b80ec2d999c45ce13f3e034704249d80829f7eLennart Poettering
73b80ec2d999c45ce13f3e034704249d80829f7eLennart Poetteringstatic int parse_proc_cmdline_item(const char *key, const char *value) {
73b80ec2d999c45ce13f3e034704249d80829f7eLennart Poettering
cca1dfddd4ce4357113663532696488427cc54e4Lennart Poettering static const char * const rlmap[] = {
1a14a53cfded6e78c6e8dfb73fdff0039971d642Lennart Poettering "emergency", SPECIAL_EMERGENCY_TARGET,
1a14a53cfded6e78c6e8dfb73fdff0039971d642Lennart Poettering "-b", SPECIAL_EMERGENCY_TARGET,
4a62c710b62a5a3c7a8a278b810b9d5b5a0c8f4fMichal Schmidt "single", SPECIAL_RESCUE_TARGET,
4a62c710b62a5a3c7a8a278b810b9d5b5a0c8f4fMichal Schmidt "-s", SPECIAL_RESCUE_TARGET,
1a14a53cfded6e78c6e8dfb73fdff0039971d642Lennart Poettering "s", SPECIAL_RESCUE_TARGET,
73b80ec2d999c45ce13f3e034704249d80829f7eLennart Poettering "S", SPECIAL_RESCUE_TARGET,
73b80ec2d999c45ce13f3e034704249d80829f7eLennart Poettering "1", SPECIAL_RESCUE_TARGET,
73b80ec2d999c45ce13f3e034704249d80829f7eLennart Poettering "2", SPECIAL_RUNLEVEL2_TARGET,
73b80ec2d999c45ce13f3e034704249d80829f7eLennart Poettering "3", SPECIAL_RUNLEVEL3_TARGET,
1a14a53cfded6e78c6e8dfb73fdff0039971d642Lennart Poettering "4", SPECIAL_RUNLEVEL4_TARGET,
73b80ec2d999c45ce13f3e034704249d80829f7eLennart Poettering "5", SPECIAL_RUNLEVEL5_TARGET,
4a62c710b62a5a3c7a8a278b810b9d5b5a0c8f4fMichal Schmidt };
4a62c710b62a5a3c7a8a278b810b9d5b5a0c8f4fMichal Schmidt int r;
1a14a53cfded6e78c6e8dfb73fdff0039971d642Lennart Poettering
1a14a53cfded6e78c6e8dfb73fdff0039971d642Lennart Poettering assert(key);
1a14a53cfded6e78c6e8dfb73fdff0039971d642Lennart Poettering
1a14a53cfded6e78c6e8dfb73fdff0039971d642Lennart Poettering if (streq(key, "systemd.unit") && value) {
1a14a53cfded6e78c6e8dfb73fdff0039971d642Lennart Poettering
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering if (!in_initrd())
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering return set_default_unit(value);
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering } else if (streq(key, "rd.systemd.unit") && value) {
cca1dfddd4ce4357113663532696488427cc54e4Lennart Poettering
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering if (in_initrd())
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering return set_default_unit(value);
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering } else if (streq(key, "systemd.log_target") && value) {
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering if (log_set_target_from_string(value) < 0)
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering log_warning("Failed to parse log target %s. Ignoring.", value);
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering } else if (streq(key, "systemd.log_level") && value) {
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering if (log_set_max_level_from_string(value) < 0)
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering log_warning("Failed to parse log level %s. Ignoring.", value);
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering } else if (streq(key, "systemd.log_color") && value) {
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering if (log_show_color_from_string(value) < 0)
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering log_warning("Failed to parse log color setting %s. Ignoring.", value);
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering } else if (streq(key, "systemd.log_location") && value) {
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering if (log_show_location_from_string(value) < 0)
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering log_warning("Failed to parse log location setting %s. Ignoring.", value);
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering } else if (streq(key, "systemd.dump_core") && value) {
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering r = parse_boolean(value);
56f64d95763a799ba4475daf44d8e9f72a1bd474Michal Schmidt if (r < 0)
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering log_warning("Failed to parse dump core switch %s. Ignoring.", value);
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering else
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering arg_dump_core = r;
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering } else if (streq(key, "systemd.crash_shell") && value) {
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering r = parse_boolean(value);
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering if (r < 0)
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering log_warning("Failed to parse crash shell switch %s. Ignoring.", value);
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering else
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering arg_crash_shell = r;
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering } else if (streq(key, "systemd.crash_chvt") && value) {
56f64d95763a799ba4475daf44d8e9f72a1bd474Michal Schmidt
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering if (safe_atoi(value, &r) < 0)
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering log_warning("Failed to parse crash chvt switch %s. Ignoring.", value);
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering else
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering arg_crash_chvt = r;
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering } else if (streq(key, "systemd.confirm_spawn") && value) {
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering r = parse_boolean(value);
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering if (r < 0)
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering log_warning("Failed to parse confirm spawn switch %s. Ignoring.", value);
cca1dfddd4ce4357113663532696488427cc54e4Lennart Poettering else
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering arg_confirm_spawn = r;
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering } else if (streq(key, "systemd.show_status") && value) {
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering r = parse_show_status(value, &arg_show_status);
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering if (r < 0)
1ca208fb4f93e5869704af1812cbff7130a2fc03Zbigniew Jędrzejewski-Szmek log_warning("Failed to parse show status switch %s. Ignoring.", value);
1ca208fb4f93e5869704af1812cbff7130a2fc03Zbigniew Jędrzejewski-Szmek
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering } else if (streq(key, "systemd.default_standard_output") && value) {
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering r = exec_output_from_string(value);
1a14a53cfded6e78c6e8dfb73fdff0039971d642Lennart Poettering if (r < 0)
e48fdd84432bbf9c2ecc339183258c7c33116032Lennart Poettering log_warning("Failed to parse default standard output switch %s. Ignoring.", value);
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering else
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering arg_default_std_output = r;
cca1dfddd4ce4357113663532696488427cc54e4Lennart Poettering
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering } else if (streq(key, "systemd.default_standard_error") && value) {
73b80ec2d999c45ce13f3e034704249d80829f7eLennart Poettering
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering r = exec_output_from_string(value);
1a14a53cfded6e78c6e8dfb73fdff0039971d642Lennart Poettering if (r < 0)
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering log_warning("Failed to parse default standard error switch %s. Ignoring.", value);
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering else
b47d419c25ecc735615a1088060c1ec8bef1e41fZbigniew Jędrzejewski-Szmek arg_default_std_error = r;
1a14a53cfded6e78c6e8dfb73fdff0039971d642Lennart Poettering
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering } else if (streq(key, "systemd.setenv") && value) {
1ca208fb4f93e5869704af1812cbff7130a2fc03Zbigniew Jędrzejewski-Szmek
1ca208fb4f93e5869704af1812cbff7130a2fc03Zbigniew Jędrzejewski-Szmek if (env_assignment_is_valid(value)) {
1a14a53cfded6e78c6e8dfb73fdff0039971d642Lennart Poettering char **env;
1a14a53cfded6e78c6e8dfb73fdff0039971d642Lennart Poettering
fa041593fe04b12ffd7e81d8b3598a7a6f313fb3Lennart Poettering env = strv_env_set(arg_default_environment, value);
fa041593fe04b12ffd7e81d8b3598a7a6f313fb3Lennart Poettering if (env)
9c4495ca561624c2f0085507dd1288ed5f1247c5Tomasz Torcz arg_default_environment = env;
fa041593fe04b12ffd7e81d8b3598a7a6f313fb3Lennart Poettering else
1a14a53cfded6e78c6e8dfb73fdff0039971d642Lennart Poettering log_warning("Setting environment variable '%s' failed, ignoring: %s", value, strerror(ENOMEM));
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering } else
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering log_warning("Environment variable name '%s' is not valid. Ignoring.", value);
fa041593fe04b12ffd7e81d8b3598a7a6f313fb3Lennart Poettering
fa041593fe04b12ffd7e81d8b3598a7a6f313fb3Lennart Poettering } else if (streq(key, "quiet") && !value) {
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering
fa041593fe04b12ffd7e81d8b3598a7a6f313fb3Lennart Poettering log_set_max_level(LOG_NOTICE);
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering if (arg_show_status == _SHOW_STATUS_UNSET)
fa041593fe04b12ffd7e81d8b3598a7a6f313fb3Lennart Poettering arg_show_status = SHOW_STATUS_AUTO;
fa041593fe04b12ffd7e81d8b3598a7a6f313fb3Lennart Poettering
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering } else if (streq(key, "debug") && !value) {
fa041593fe04b12ffd7e81d8b3598a7a6f313fb3Lennart Poettering
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering log_set_max_level(LOG_DEBUG);
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering if (detect_container(NULL) > 0)
fa041593fe04b12ffd7e81d8b3598a7a6f313fb3Lennart Poettering log_set_target(LOG_TARGET_CONSOLE);
fa041593fe04b12ffd7e81d8b3598a7a6f313fb3Lennart Poettering
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering } else if (!in_initrd() && !value) {
fa041593fe04b12ffd7e81d8b3598a7a6f313fb3Lennart Poettering unsigned i;
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering /* SysV compatibility */
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering for (i = 0; i < ELEMENTSOF(rlmap); i += 2)
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering if (streq(key, rlmap[i]))
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering return set_default_unit(rlmap[i+1]);
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering }
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering return 0;
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering}
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering#define DEFINE_SETTER(name, func, descr) \
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering static int name(const char *unit, \
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering const char *filename, \
56f64d95763a799ba4475daf44d8e9f72a1bd474Michal Schmidt unsigned line, \
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering const char *section, \
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering unsigned section_line, \
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering const char *lvalue, \
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering int ltype, \
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering const char *rvalue, \
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering void *data, \
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering void *userdata) { \
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering \
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering int r; \
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering \
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering assert(filename); \
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering assert(lvalue); \
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering assert(rvalue); \
56f64d95763a799ba4475daf44d8e9f72a1bd474Michal Schmidt \
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering r = func(rvalue); \
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering if (r < 0) \
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering log_syntax(unit, LOG_ERR, filename, line, -r, \
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering "Invalid " descr "'%s': %s", \
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering rvalue, strerror(-r)); \
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering \
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering return 0; \
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering }
56f64d95763a799ba4475daf44d8e9f72a1bd474Michal Schmidt
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart PoetteringDEFINE_SETTER(config_parse_level2, log_set_max_level_from_string, "log level")
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart PoetteringDEFINE_SETTER(config_parse_target, log_set_target_from_string, "target")
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart PoetteringDEFINE_SETTER(config_parse_color, log_show_color_from_string, "color" )
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart PoetteringDEFINE_SETTER(config_parse_location, log_show_location_from_string, "location")
fa041593fe04b12ffd7e81d8b3598a7a6f313fb3Lennart Poettering
fa041593fe04b12ffd7e81d8b3598a7a6f313fb3Lennart Poetteringstatic int config_parse_cpu_affinity2(
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering const char *unit,
fa041593fe04b12ffd7e81d8b3598a7a6f313fb3Lennart Poettering const char *filename,
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering unsigned line,
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering const char *section,
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering unsigned section_line,
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering const char *lvalue,
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering int ltype,
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering const char *rvalue,
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering void *data,
56f64d95763a799ba4475daf44d8e9f72a1bd474Michal Schmidt void *userdata) {
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering char *w;
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering size_t l;
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering char *state;
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering cpu_set_t *c = NULL;
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering unsigned ncpus = 0;
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering
1a14a53cfded6e78c6e8dfb73fdff0039971d642Lennart Poettering assert(filename);
1ca208fb4f93e5869704af1812cbff7130a2fc03Zbigniew Jędrzejewski-Szmek assert(lvalue);
1ca208fb4f93e5869704af1812cbff7130a2fc03Zbigniew Jędrzejewski-Szmek assert(rvalue);
1a14a53cfded6e78c6e8dfb73fdff0039971d642Lennart Poettering
1a14a53cfded6e78c6e8dfb73fdff0039971d642Lennart Poettering FOREACH_WORD_QUOTED(w, l, rvalue, state) {
1ca208fb4f93e5869704af1812cbff7130a2fc03Zbigniew Jędrzejewski-Szmek char *t;
1ca208fb4f93e5869704af1812cbff7130a2fc03Zbigniew Jędrzejewski-Szmek int r;
1a14a53cfded6e78c6e8dfb73fdff0039971d642Lennart Poettering unsigned cpu;
1a14a53cfded6e78c6e8dfb73fdff0039971d642Lennart Poettering
23bbb0de4e3f85d9704a5c12a5afa2dfa0159e41Michal Schmidt if (!(t = strndup(w, l)))
23bbb0de4e3f85d9704a5c12a5afa2dfa0159e41Michal Schmidt return log_oom();
1a14a53cfded6e78c6e8dfb73fdff0039971d642Lennart Poettering
1a14a53cfded6e78c6e8dfb73fdff0039971d642Lennart Poettering r = safe_atou(t, &cpu);
1a14a53cfded6e78c6e8dfb73fdff0039971d642Lennart Poettering free(t);
e48fdd84432bbf9c2ecc339183258c7c33116032Lennart Poettering
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering if (!c)
1a14a53cfded6e78c6e8dfb73fdff0039971d642Lennart Poettering if (!(c = cpu_set_malloc(&ncpus)))
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering return log_oom();
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering if (r < 0 || cpu >= ncpus) {
cca1dfddd4ce4357113663532696488427cc54e4Lennart Poettering log_syntax(unit, LOG_ERR, filename, line, -r,
1a14a53cfded6e78c6e8dfb73fdff0039971d642Lennart Poettering "Failed to parse CPU affinity '%s'", rvalue);
1a14a53cfded6e78c6e8dfb73fdff0039971d642Lennart Poettering CPU_FREE(c);
1ca208fb4f93e5869704af1812cbff7130a2fc03Zbigniew Jędrzejewski-Szmek return -EBADMSG;
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering }
1a14a53cfded6e78c6e8dfb73fdff0039971d642Lennart Poettering
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering CPU_SET_S(cpu, CPU_ALLOC_SIZE(ncpus), c);
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering }
1ca208fb4f93e5869704af1812cbff7130a2fc03Zbigniew Jędrzejewski-Szmek
1a14a53cfded6e78c6e8dfb73fdff0039971d642Lennart Poettering if (c) {
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering if (sched_setaffinity(0, CPU_ALLOC_SIZE(ncpus), c) < 0)
1ca208fb4f93e5869704af1812cbff7130a2fc03Zbigniew Jędrzejewski-Szmek log_warning_unit(unit, "Failed to set CPU affinity: %m");
1a14a53cfded6e78c6e8dfb73fdff0039971d642Lennart Poettering
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering CPU_FREE(c);
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering }
1a14a53cfded6e78c6e8dfb73fdff0039971d642Lennart Poettering
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering return 0;
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering}
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering
73b80ec2d999c45ce13f3e034704249d80829f7eLennart Poetteringstatic int config_parse_show_status(
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering const char* unit,
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering const char *filename,
73b80ec2d999c45ce13f3e034704249d80829f7eLennart Poettering unsigned line,
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering const char *section,
cca1dfddd4ce4357113663532696488427cc54e4Lennart Poettering unsigned section_line,
cca1dfddd4ce4357113663532696488427cc54e4Lennart Poettering const char *lvalue,
cca1dfddd4ce4357113663532696488427cc54e4Lennart Poettering int ltype,
cca1dfddd4ce4357113663532696488427cc54e4Lennart Poettering const char *rvalue,
cca1dfddd4ce4357113663532696488427cc54e4Lennart Poettering void *data,
cca1dfddd4ce4357113663532696488427cc54e4Lennart Poettering void *userdata) {
cca1dfddd4ce4357113663532696488427cc54e4Lennart Poettering
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering int k;
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering ShowStatus *b = data;
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering assert(filename);
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering assert(lvalue);
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering assert(rvalue);
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering assert(data);
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering k = parse_show_status(rvalue, b);
1ca208fb4f93e5869704af1812cbff7130a2fc03Zbigniew Jędrzejewski-Szmek if (k < 0) {
1a14a53cfded6e78c6e8dfb73fdff0039971d642Lennart Poettering log_syntax(unit, LOG_ERR, filename, line, -k,
73b80ec2d999c45ce13f3e034704249d80829f7eLennart Poettering "Failed to parse show status setting, ignoring: %s", rvalue);
73b80ec2d999c45ce13f3e034704249d80829f7eLennart Poettering return 0;
cca1dfddd4ce4357113663532696488427cc54e4Lennart Poettering }
cca1dfddd4ce4357113663532696488427cc54e4Lennart Poettering
cca1dfddd4ce4357113663532696488427cc54e4Lennart Poettering return 0;
cca1dfddd4ce4357113663532696488427cc54e4Lennart Poettering}
cca1dfddd4ce4357113663532696488427cc54e4Lennart Poettering
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poetteringstatic void strv_free_free(char ***l) {
73b80ec2d999c45ce13f3e034704249d80829f7eLennart Poettering char ***i;
73b80ec2d999c45ce13f3e034704249d80829f7eLennart Poettering
e48fdd84432bbf9c2ecc339183258c7c33116032Lennart Poettering if (!l)
73b80ec2d999c45ce13f3e034704249d80829f7eLennart Poettering return;
e48fdd84432bbf9c2ecc339183258c7c33116032Lennart Poettering
e48fdd84432bbf9c2ecc339183258c7c33116032Lennart Poettering for (i = l; *i; i++)
e48fdd84432bbf9c2ecc339183258c7c33116032Lennart Poettering strv_free(*i);
e48fdd84432bbf9c2ecc339183258c7c33116032Lennart Poettering
e48fdd84432bbf9c2ecc339183258c7c33116032Lennart Poettering free(l);
e48fdd84432bbf9c2ecc339183258c7c33116032Lennart Poettering}
cca1dfddd4ce4357113663532696488427cc54e4Lennart Poettering
e48fdd84432bbf9c2ecc339183258c7c33116032Lennart Poetteringstatic void free_join_controllers(void) {
e48fdd84432bbf9c2ecc339183258c7c33116032Lennart Poettering strv_free_free(arg_join_controllers);
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering arg_join_controllers = NULL;
e48fdd84432bbf9c2ecc339183258c7c33116032Lennart Poettering}
e48fdd84432bbf9c2ecc339183258c7c33116032Lennart Poettering
e48fdd84432bbf9c2ecc339183258c7c33116032Lennart Poetteringstatic int config_parse_join_controllers(const char *unit,
e48fdd84432bbf9c2ecc339183258c7c33116032Lennart Poettering const char *filename,
e48fdd84432bbf9c2ecc339183258c7c33116032Lennart Poettering unsigned line,
e48fdd84432bbf9c2ecc339183258c7c33116032Lennart Poettering const char *section,
e48fdd84432bbf9c2ecc339183258c7c33116032Lennart Poettering unsigned section_line,
e48fdd84432bbf9c2ecc339183258c7c33116032Lennart Poettering const char *lvalue,
e48fdd84432bbf9c2ecc339183258c7c33116032Lennart Poettering int ltype,
e48fdd84432bbf9c2ecc339183258c7c33116032Lennart Poettering const char *rvalue,
cca1dfddd4ce4357113663532696488427cc54e4Lennart Poettering void *data,
e48fdd84432bbf9c2ecc339183258c7c33116032Lennart Poettering void *userdata) {
e48fdd84432bbf9c2ecc339183258c7c33116032Lennart Poettering
e48fdd84432bbf9c2ecc339183258c7c33116032Lennart Poettering unsigned n = 0;
e48fdd84432bbf9c2ecc339183258c7c33116032Lennart Poettering char *state, *w;
e48fdd84432bbf9c2ecc339183258c7c33116032Lennart Poettering size_t length;
1a14a53cfded6e78c6e8dfb73fdff0039971d642Lennart Poettering
1a14a53cfded6e78c6e8dfb73fdff0039971d642Lennart Poettering assert(filename);
1a14a53cfded6e78c6e8dfb73fdff0039971d642Lennart Poettering assert(lvalue);
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering assert(rvalue);
cca1dfddd4ce4357113663532696488427cc54e4Lennart Poettering
73b80ec2d999c45ce13f3e034704249d80829f7eLennart Poettering free_join_controllers();
73b80ec2d999c45ce13f3e034704249d80829f7eLennart Poettering
73b80ec2d999c45ce13f3e034704249d80829f7eLennart Poettering FOREACH_WORD_QUOTED(w, length, rvalue, state) {
e48fdd84432bbf9c2ecc339183258c7c33116032Lennart Poettering char *s, **l;
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering
cca1dfddd4ce4357113663532696488427cc54e4Lennart Poettering s = strndup(w, length);
73b80ec2d999c45ce13f3e034704249d80829f7eLennart Poettering if (!s)
73b80ec2d999c45ce13f3e034704249d80829f7eLennart Poettering return log_oom();
73b80ec2d999c45ce13f3e034704249d80829f7eLennart Poettering
1a14a53cfded6e78c6e8dfb73fdff0039971d642Lennart Poettering l = strv_split(s, ",");
1a14a53cfded6e78c6e8dfb73fdff0039971d642Lennart Poettering free(s);
1a14a53cfded6e78c6e8dfb73fdff0039971d642Lennart Poettering
1a14a53cfded6e78c6e8dfb73fdff0039971d642Lennart Poettering strv_uniq(l);
1a14a53cfded6e78c6e8dfb73fdff0039971d642Lennart Poettering
1a14a53cfded6e78c6e8dfb73fdff0039971d642Lennart Poettering if (strv_length(l) <= 1) {
1a14a53cfded6e78c6e8dfb73fdff0039971d642Lennart Poettering strv_free(l);
1a14a53cfded6e78c6e8dfb73fdff0039971d642Lennart Poettering continue;
1a14a53cfded6e78c6e8dfb73fdff0039971d642Lennart Poettering }
1a14a53cfded6e78c6e8dfb73fdff0039971d642Lennart Poettering
1a14a53cfded6e78c6e8dfb73fdff0039971d642Lennart Poettering if (!arg_join_controllers) {
e48fdd84432bbf9c2ecc339183258c7c33116032Lennart Poettering arg_join_controllers = new(char**, 2);
1a14a53cfded6e78c6e8dfb73fdff0039971d642Lennart Poettering if (!arg_join_controllers) {
1a14a53cfded6e78c6e8dfb73fdff0039971d642Lennart Poettering strv_free(l);
1a14a53cfded6e78c6e8dfb73fdff0039971d642Lennart Poettering return log_oom();
1a14a53cfded6e78c6e8dfb73fdff0039971d642Lennart Poettering }
1a14a53cfded6e78c6e8dfb73fdff0039971d642Lennart Poettering
1a14a53cfded6e78c6e8dfb73fdff0039971d642Lennart Poettering arg_join_controllers[0] = l;
1a14a53cfded6e78c6e8dfb73fdff0039971d642Lennart Poettering arg_join_controllers[1] = NULL;
e48fdd84432bbf9c2ecc339183258c7c33116032Lennart Poettering
1a14a53cfded6e78c6e8dfb73fdff0039971d642Lennart Poettering n = 1;
1a14a53cfded6e78c6e8dfb73fdff0039971d642Lennart Poettering } else {
c51cf05646a11c65daf65c1123c77efb068f4f7bZbigniew Jędrzejewski-Szmek char ***a;
f9ac15442e4132f00eca5495d53c17062aae13e0Lennart Poettering char ***t;
1a14a53cfded6e78c6e8dfb73fdff0039971d642Lennart Poettering
1a14a53cfded6e78c6e8dfb73fdff0039971d642Lennart Poettering t = new0(char**, n+2);
1a14a53cfded6e78c6e8dfb73fdff0039971d642Lennart Poettering if (!t) {
1a14a53cfded6e78c6e8dfb73fdff0039971d642Lennart Poettering strv_free(l);
73b80ec2d999c45ce13f3e034704249d80829f7eLennart Poettering return log_oom();
73b80ec2d999c45ce13f3e034704249d80829f7eLennart Poettering }
1a14a53cfded6e78c6e8dfb73fdff0039971d642Lennart Poettering
73b80ec2d999c45ce13f3e034704249d80829f7eLennart Poettering n = 0;
1a14a53cfded6e78c6e8dfb73fdff0039971d642Lennart Poettering
73b80ec2d999c45ce13f3e034704249d80829f7eLennart Poettering for (a = arg_join_controllers; *a; a++) {
1a14a53cfded6e78c6e8dfb73fdff0039971d642Lennart Poettering
73b80ec2d999c45ce13f3e034704249d80829f7eLennart Poettering if (strv_overlap(*a, l)) {
73b80ec2d999c45ce13f3e034704249d80829f7eLennart Poettering if (strv_extend_strv(&l, *a) < 0) {
73b80ec2d999c45ce13f3e034704249d80829f7eLennart Poettering strv_free(l);
8086ffacdb1bfec5ec115d24626538bda6cc372eZbigniew Jędrzejewski-Szmek strv_free_free(t);
8086ffacdb1bfec5ec115d24626538bda6cc372eZbigniew Jędrzejewski-Szmek return log_oom();
1a14a53cfded6e78c6e8dfb73fdff0039971d642Lennart Poettering }
73b80ec2d999c45ce13f3e034704249d80829f7eLennart Poettering
73b80ec2d999c45ce13f3e034704249d80829f7eLennart Poettering } else {
73b80ec2d999c45ce13f3e034704249d80829f7eLennart Poettering char **c;
73b80ec2d999c45ce13f3e034704249d80829f7eLennart Poettering
73b80ec2d999c45ce13f3e034704249d80829f7eLennart Poettering c = strv_copy(*a);
73b80ec2d999c45ce13f3e034704249d80829f7eLennart Poettering if (!c) {
73b80ec2d999c45ce13f3e034704249d80829f7eLennart Poettering strv_free(l);
73b80ec2d999c45ce13f3e034704249d80829f7eLennart Poettering strv_free_free(t);
73b80ec2d999c45ce13f3e034704249d80829f7eLennart Poettering return log_oom();
73b80ec2d999c45ce13f3e034704249d80829f7eLennart Poettering }
73b80ec2d999c45ce13f3e034704249d80829f7eLennart Poettering
73b80ec2d999c45ce13f3e034704249d80829f7eLennart Poettering t[n++] = c;
73b80ec2d999c45ce13f3e034704249d80829f7eLennart Poettering }
73b80ec2d999c45ce13f3e034704249d80829f7eLennart Poettering }
73b80ec2d999c45ce13f3e034704249d80829f7eLennart Poettering
73b80ec2d999c45ce13f3e034704249d80829f7eLennart Poettering t[n++] = strv_uniq(l);
73b80ec2d999c45ce13f3e034704249d80829f7eLennart Poettering
73b80ec2d999c45ce13f3e034704249d80829f7eLennart Poettering strv_free_free(arg_join_controllers);
73b80ec2d999c45ce13f3e034704249d80829f7eLennart Poettering arg_join_controllers = t;
73b80ec2d999c45ce13f3e034704249d80829f7eLennart Poettering }
73b80ec2d999c45ce13f3e034704249d80829f7eLennart Poettering }
73b80ec2d999c45ce13f3e034704249d80829f7eLennart Poettering
73b80ec2d999c45ce13f3e034704249d80829f7eLennart Poettering return 0;
9a5cb1371b6d8b0a04bd08665bcf9b06cb40c64cZbigniew Jędrzejewski-Szmek}
9a5cb1371b6d8b0a04bd08665bcf9b06cb40c64cZbigniew Jędrzejewski-Szmek
73b80ec2d999c45ce13f3e034704249d80829f7eLennart Poetteringstatic int parse_config_file(void) {
73b80ec2d999c45ce13f3e034704249d80829f7eLennart Poettering
73b80ec2d999c45ce13f3e034704249d80829f7eLennart Poettering const ConfigTableItem items[] = {
73b80ec2d999c45ce13f3e034704249d80829f7eLennart Poettering { "Manager", "LogLevel", config_parse_level2, 0, NULL },
23bbb0de4e3f85d9704a5c12a5afa2dfa0159e41Michal Schmidt { "Manager", "LogTarget", config_parse_target, 0, NULL },
23bbb0de4e3f85d9704a5c12a5afa2dfa0159e41Michal Schmidt { "Manager", "LogColor", config_parse_color, 0, NULL },
1a14a53cfded6e78c6e8dfb73fdff0039971d642Lennart Poettering { "Manager", "LogLocation", config_parse_location, 0, NULL },
73b80ec2d999c45ce13f3e034704249d80829f7eLennart Poettering { "Manager", "DumpCore", config_parse_bool, 0, &arg_dump_core },
73b80ec2d999c45ce13f3e034704249d80829f7eLennart Poettering { "Manager", "CrashShell", config_parse_bool, 0, &arg_crash_shell },
73b80ec2d999c45ce13f3e034704249d80829f7eLennart Poettering { "Manager", "ShowStatus", config_parse_show_status, 0, &arg_show_status },
73b80ec2d999c45ce13f3e034704249d80829f7eLennart Poettering { "Manager", "CrashChVT", config_parse_int, 0, &arg_crash_chvt },
73b80ec2d999c45ce13f3e034704249d80829f7eLennart Poettering { "Manager", "CPUAffinity", config_parse_cpu_affinity2, 0, NULL },
73b80ec2d999c45ce13f3e034704249d80829f7eLennart Poettering { "Manager", "JoinControllers", config_parse_join_controllers, 0, &arg_join_controllers },
98b2f766b24c84fc49a4df954717b1f96bbce00dLennart Poettering { "Manager", "RuntimeWatchdogSec", config_parse_sec, 0, &arg_runtime_watchdog },
73b80ec2d999c45ce13f3e034704249d80829f7eLennart Poettering { "Manager", "ShutdownWatchdogSec", config_parse_sec, 0, &arg_shutdown_watchdog },
73b80ec2d999c45ce13f3e034704249d80829f7eLennart Poettering { "Manager", "CapabilityBoundingSet", config_parse_bounding_set, 0, &arg_capability_bounding_set_drop },
cca1dfddd4ce4357113663532696488427cc54e4Lennart Poettering#ifdef HAVE_SECCOMP
73b80ec2d999c45ce13f3e034704249d80829f7eLennart Poettering { "Manager", "SystemCallArchitectures", config_parse_syscall_archs, 0, &arg_syscall_archs },
73b80ec2d999c45ce13f3e034704249d80829f7eLennart Poettering#endif
73b80ec2d999c45ce13f3e034704249d80829f7eLennart Poettering { "Manager", "TimerSlackNSec", config_parse_nsec, 0, &arg_timer_slack_nsec },
73b80ec2d999c45ce13f3e034704249d80829f7eLennart Poettering { "Manager", "DefaultTimerAccuracySec", config_parse_sec, 0, &arg_default_timer_accuracy_usec },
73b80ec2d999c45ce13f3e034704249d80829f7eLennart Poettering { "Manager", "DefaultStandardOutput", config_parse_output, 0, &arg_default_std_output },
73b80ec2d999c45ce13f3e034704249d80829f7eLennart Poettering { "Manager", "DefaultStandardError", config_parse_output, 0, &arg_default_std_error },
73b80ec2d999c45ce13f3e034704249d80829f7eLennart Poettering { "Manager", "DefaultTimeoutStartSec", config_parse_sec, 0, &arg_default_timeout_start_usec },
73b80ec2d999c45ce13f3e034704249d80829f7eLennart Poettering { "Manager", "DefaultTimeoutStopSec", config_parse_sec, 0, &arg_default_timeout_stop_usec },
73b80ec2d999c45ce13f3e034704249d80829f7eLennart Poettering { "Manager", "DefaultRestartSec", config_parse_sec, 0, &arg_default_restart_usec },
73b80ec2d999c45ce13f3e034704249d80829f7eLennart Poettering { "Manager", "DefaultStartLimitInterval", config_parse_sec, 0, &arg_default_start_limit_interval },
73b80ec2d999c45ce13f3e034704249d80829f7eLennart Poettering { "Manager", "DefaultStartLimitBurst", config_parse_unsigned, 0, &arg_default_start_limit_burst },
3db604b907323b8df0fc810216f6112056d26a02Lennart Poettering { "Manager", "DefaultEnvironment", config_parse_environ, 0, &arg_default_environment },
23bbb0de4e3f85d9704a5c12a5afa2dfa0159e41Michal Schmidt { "Manager", "DefaultLimitCPU", config_parse_limit, 0, &arg_default_rlimit[RLIMIT_CPU] },
23bbb0de4e3f85d9704a5c12a5afa2dfa0159e41Michal Schmidt { "Manager", "DefaultLimitFSIZE", config_parse_limit, 0, &arg_default_rlimit[RLIMIT_FSIZE] },
23bbb0de4e3f85d9704a5c12a5afa2dfa0159e41Michal Schmidt { "Manager", "DefaultLimitDATA", config_parse_limit, 0, &arg_default_rlimit[RLIMIT_DATA] },
1a14a53cfded6e78c6e8dfb73fdff0039971d642Lennart Poettering { "Manager", "DefaultLimitSTACK", config_parse_limit, 0, &arg_default_rlimit[RLIMIT_STACK] },
73b80ec2d999c45ce13f3e034704249d80829f7eLennart Poettering { "Manager", "DefaultLimitCORE", config_parse_limit, 0, &arg_default_rlimit[RLIMIT_CORE] },
3db604b907323b8df0fc810216f6112056d26a02Lennart Poettering { "Manager", "DefaultLimitRSS", config_parse_limit, 0, &arg_default_rlimit[RLIMIT_RSS] },
3db604b907323b8df0fc810216f6112056d26a02Lennart Poettering { "Manager", "DefaultLimitNOFILE", config_parse_limit, 0, &arg_default_rlimit[RLIMIT_NOFILE] },
61331eab0a53cd9b8446eab6d1ebf1a046d8efc1Lennart Poettering { "Manager", "DefaultLimitAS", config_parse_limit, 0, &arg_default_rlimit[RLIMIT_AS] },
73b80ec2d999c45ce13f3e034704249d80829f7eLennart Poettering { "Manager", "DefaultLimitNPROC", config_parse_limit, 0, &arg_default_rlimit[RLIMIT_NPROC] },
73b80ec2d999c45ce13f3e034704249d80829f7eLennart Poettering { "Manager", "DefaultLimitMEMLOCK", config_parse_limit, 0, &arg_default_rlimit[RLIMIT_MEMLOCK] },
73b80ec2d999c45ce13f3e034704249d80829f7eLennart Poettering { "Manager", "DefaultLimitLOCKS", config_parse_limit, 0, &arg_default_rlimit[RLIMIT_LOCKS] },
73b80ec2d999c45ce13f3e034704249d80829f7eLennart Poettering { "Manager", "DefaultLimitSIGPENDING", config_parse_limit, 0, &arg_default_rlimit[RLIMIT_SIGPENDING] },
73b80ec2d999c45ce13f3e034704249d80829f7eLennart Poettering { "Manager", "DefaultLimitMSGQUEUE", config_parse_limit, 0, &arg_default_rlimit[RLIMIT_MSGQUEUE] },
73b80ec2d999c45ce13f3e034704249d80829f7eLennart Poettering { "Manager", "DefaultLimitNICE", config_parse_limit, 0, &arg_default_rlimit[RLIMIT_NICE] },
73b80ec2d999c45ce13f3e034704249d80829f7eLennart Poettering { "Manager", "DefaultLimitRTPRIO", config_parse_limit, 0, &arg_default_rlimit[RLIMIT_RTPRIO] },
73b80ec2d999c45ce13f3e034704249d80829f7eLennart Poettering { "Manager", "DefaultLimitRTTIME", config_parse_limit, 0, &arg_default_rlimit[RLIMIT_RTTIME] },
73b80ec2d999c45ce13f3e034704249d80829f7eLennart Poettering { "Manager", "DefaultCPUAccounting", config_parse_bool, 0, &arg_default_cpu_accounting },
73b80ec2d999c45ce13f3e034704249d80829f7eLennart Poettering { "Manager", "DefaultBlockIOAccounting", config_parse_bool, 0, &arg_default_blockio_accounting },
73b80ec2d999c45ce13f3e034704249d80829f7eLennart Poettering { "Manager", "DefaultMemoryAccounting", config_parse_bool, 0, &arg_default_memory_accounting },
73b80ec2d999c45ce13f3e034704249d80829f7eLennart Poettering {}
73b80ec2d999c45ce13f3e034704249d80829f7eLennart Poettering };
73b80ec2d999c45ce13f3e034704249d80829f7eLennart Poettering
73b80ec2d999c45ce13f3e034704249d80829f7eLennart Poettering _cleanup_fclose_ FILE *f;
73b80ec2d999c45ce13f3e034704249d80829f7eLennart Poettering const char *fn;
73b80ec2d999c45ce13f3e034704249d80829f7eLennart Poettering int r;
73b80ec2d999c45ce13f3e034704249d80829f7eLennart Poettering
73b80ec2d999c45ce13f3e034704249d80829f7eLennart Poettering fn = arg_running_as == SYSTEMD_SYSTEM ? PKGSYSCONFDIR "/system.conf" : PKGSYSCONFDIR "/user.conf";
73b80ec2d999c45ce13f3e034704249d80829f7eLennart Poettering f = fopen(fn, "re");
73b80ec2d999c45ce13f3e034704249d80829f7eLennart Poettering if (!f) {
e48fdd84432bbf9c2ecc339183258c7c33116032Lennart Poettering if (errno == ENOENT)
1a14a53cfded6e78c6e8dfb73fdff0039971d642Lennart Poettering return 0;
3db604b907323b8df0fc810216f6112056d26a02Lennart Poettering
b5884878a2874447b2a9f07f324a7cd909d96d48Lennart Poettering log_warning("Failed to open configuration file '%s': %m", fn);
b5884878a2874447b2a9f07f324a7cd909d96d48Lennart Poettering return 0;
da927ba997d68401563b927f92e6e40e021a8e5cMichal Schmidt }
1a14a53cfded6e78c6e8dfb73fdff0039971d642Lennart Poettering
73b80ec2d999c45ce13f3e034704249d80829f7eLennart Poettering r = config_parse(NULL, fn, f, "Manager\0", config_item_table_lookup, (void*) items, false, false, NULL);
73b80ec2d999c45ce13f3e034704249d80829f7eLennart Poettering if (r < 0)
73b80ec2d999c45ce13f3e034704249d80829f7eLennart Poettering log_warning("Failed to parse configuration file: %s", strerror(-r));
73b80ec2d999c45ce13f3e034704249d80829f7eLennart Poettering
73b80ec2d999c45ce13f3e034704249d80829f7eLennart Poettering return 0;
73b80ec2d999c45ce13f3e034704249d80829f7eLennart Poettering}
73b80ec2d999c45ce13f3e034704249d80829f7eLennart Poettering
73b80ec2d999c45ce13f3e034704249d80829f7eLennart Poetteringstatic int parse_argv(int argc, char *argv[]) {
73b80ec2d999c45ce13f3e034704249d80829f7eLennart Poettering
73b80ec2d999c45ce13f3e034704249d80829f7eLennart Poettering enum {
73b80ec2d999c45ce13f3e034704249d80829f7eLennart Poettering ARG_LOG_LEVEL = 0x100,
73b80ec2d999c45ce13f3e034704249d80829f7eLennart Poettering ARG_LOG_TARGET,
73b80ec2d999c45ce13f3e034704249d80829f7eLennart Poettering ARG_LOG_COLOR,
73b80ec2d999c45ce13f3e034704249d80829f7eLennart Poettering ARG_LOG_LOCATION,
73b80ec2d999c45ce13f3e034704249d80829f7eLennart Poettering ARG_UNIT,
73b80ec2d999c45ce13f3e034704249d80829f7eLennart Poettering ARG_SYSTEM,
73b80ec2d999c45ce13f3e034704249d80829f7eLennart Poettering ARG_USER,
1a14a53cfded6e78c6e8dfb73fdff0039971d642Lennart Poettering ARG_TEST,
ARG_VERSION,
ARG_DUMP_CONFIGURATION_ITEMS,
ARG_DUMP_CORE,
ARG_CRASH_SHELL,
ARG_CONFIRM_SPAWN,
ARG_SHOW_STATUS,
ARG_DESERIALIZE,
ARG_SWITCHED_ROOT,
ARG_DEFAULT_STD_OUTPUT,
ARG_DEFAULT_STD_ERROR
};
static const struct option options[] = {
{ "log-level", required_argument, NULL, ARG_LOG_LEVEL },
{ "log-target", required_argument, NULL, ARG_LOG_TARGET },
{ "log-color", optional_argument, NULL, ARG_LOG_COLOR },
{ "log-location", optional_argument, NULL, ARG_LOG_LOCATION },
{ "unit", required_argument, NULL, ARG_UNIT },
{ "system", no_argument, NULL, ARG_SYSTEM },
{ "user", no_argument, NULL, ARG_USER },
{ "test", no_argument, NULL, ARG_TEST },
{ "help", no_argument, NULL, 'h' },
{ "version", no_argument, NULL, ARG_VERSION },
{ "dump-configuration-items", no_argument, NULL, ARG_DUMP_CONFIGURATION_ITEMS },
{ "dump-core", optional_argument, NULL, ARG_DUMP_CORE },
{ "crash-shell", optional_argument, NULL, ARG_CRASH_SHELL },
{ "confirm-spawn", optional_argument, NULL, ARG_CONFIRM_SPAWN },
{ "show-status", optional_argument, NULL, ARG_SHOW_STATUS },
{ "deserialize", required_argument, NULL, ARG_DESERIALIZE },
{ "switched-root", no_argument, NULL, ARG_SWITCHED_ROOT },
{ "default-standard-output", required_argument, NULL, ARG_DEFAULT_STD_OUTPUT, },
{ "default-standard-error", required_argument, NULL, ARG_DEFAULT_STD_ERROR, },
{}
};
int c, r;
assert(argc >= 1);
assert(argv);
if (getpid() == 1)
opterr = 0;
while ((c = getopt_long(argc, argv, "hDbsz:", options, NULL)) >= 0)
switch (c) {
case ARG_LOG_LEVEL:
r = log_set_max_level_from_string(optarg);
if (r < 0) {
log_error("Failed to parse log level %s.", optarg);
return r;
}
break;
case ARG_LOG_TARGET:
r = log_set_target_from_string(optarg);
if (r < 0) {
log_error("Failed to parse log target %s.", optarg);
return r;
}
break;
case ARG_LOG_COLOR:
if (optarg) {
r = log_show_color_from_string(optarg);
if (r < 0) {
log_error("Failed to parse log color setting %s.", optarg);
return r;
}
} else
log_show_color(true);
break;
case ARG_LOG_LOCATION:
if (optarg) {
r = log_show_location_from_string(optarg);
if (r < 0) {
log_error("Failed to parse log location setting %s.", optarg);
return r;
}
} else
log_show_location(true);
break;
case ARG_DEFAULT_STD_OUTPUT:
r = exec_output_from_string(optarg);
if (r < 0) {
log_error("Failed to parse default standard output setting %s.", optarg);
return r;
} else
arg_default_std_output = r;
break;
case ARG_DEFAULT_STD_ERROR:
r = exec_output_from_string(optarg);
if (r < 0) {
log_error("Failed to parse default standard error output setting %s.", optarg);
return r;
} else
arg_default_std_error = r;
break;
case ARG_UNIT:
r = set_default_unit(optarg);
if (r < 0) {
log_error("Failed to set default unit %s: %s", optarg, strerror(-r));
return r;
}
break;
case ARG_SYSTEM:
arg_running_as = SYSTEMD_SYSTEM;
break;
case ARG_USER:
arg_running_as = SYSTEMD_USER;
break;
case ARG_TEST:
arg_action = ACTION_TEST;
break;
case ARG_VERSION:
arg_action = ACTION_VERSION;
break;
case ARG_DUMP_CONFIGURATION_ITEMS:
arg_action = ACTION_DUMP_CONFIGURATION_ITEMS;
break;
case ARG_DUMP_CORE:
r = optarg ? parse_boolean(optarg) : 1;
if (r < 0) {
log_error("Failed to parse dump core boolean %s.", optarg);
return r;
}
arg_dump_core = r;
break;
case ARG_CRASH_SHELL:
r = optarg ? parse_boolean(optarg) : 1;
if (r < 0) {
log_error("Failed to parse crash shell boolean %s.", optarg);
return r;
}
arg_crash_shell = r;
break;
case ARG_CONFIRM_SPAWN:
r = optarg ? parse_boolean(optarg) : 1;
if (r < 0) {
log_error("Failed to parse confirm spawn boolean %s.", optarg);
return r;
}
arg_confirm_spawn = r;
break;
case ARG_SHOW_STATUS:
if (optarg) {
r = parse_show_status(optarg, &arg_show_status);
if (r < 0) {
log_error("Failed to parse show status boolean %s.", optarg);
return r;
}
} else
arg_show_status = SHOW_STATUS_YES;
break;
case ARG_DESERIALIZE: {
int fd;
FILE *f;
r = safe_atoi(optarg, &fd);
if (r < 0 || fd < 0) {
log_error("Failed to parse deserialize option %s.", optarg);
return r < 0 ? r : -EINVAL;
}
fd_cloexec(fd, true);
f = fdopen(fd, "r");
if (!f) {
log_error("Failed to open serialization fd: %m");
return -errno;
}
if (arg_serialization)
fclose(arg_serialization);
arg_serialization = f;
break;
}
case ARG_SWITCHED_ROOT:
arg_switched_root = true;
break;
case 'h':
arg_action = ACTION_HELP;
break;
case 'D':
log_set_max_level(LOG_DEBUG);
break;
case 'b':
case 's':
case 'z':
/* Just to eat away the sysvinit kernel
* cmdline args without getopt() error
* messages that we'll parse in
* parse_proc_cmdline_word() or ignore. */
case '?':
default:
if (getpid() != 1) {
log_error("Unknown option code %c", c);
return -EINVAL;
}
break;
}
if (optind < argc && getpid() != 1) {
/* Hmm, when we aren't run as init system
* let's complain about excess arguments */
log_error("Excess arguments.");
return -EINVAL;
}
if (detect_container(NULL) > 0) {
char **a;
/* All /proc/cmdline arguments the kernel didn't
* understand it passed to us. We're not really
* interested in that usually since /proc/cmdline is
* more interesting and complete. With one exception:
* if we are run in a container /proc/cmdline is not
* relevant for the container, hence we rely on argv[]
* instead. */
for (a = argv; a < argv + argc; a++) {
_cleanup_free_ char *w;
char *value;
w = strdup(*a);
if (!w)
return log_oom();
value = strchr(w, '=');
if (value)
*(value++) = 0;
r = parse_proc_cmdline_item(w, value);
if (r < 0) {
log_error("Failed on cmdline argument %s: %s", *a, strerror(-r));
return r;
}
}
}
return 0;
}
static int help(void) {
printf("%s [OPTIONS...]\n\n"
"Starts up and maintains the system or user services.\n\n"
" -h --help Show this help\n"
" --test Determine startup sequence, dump it and exit\n"
" --dump-configuration-items Dump understood unit configuration items\n"
" --unit=UNIT Set default unit\n"
" --system Run a system instance, even if PID != 1\n"
" --user Run a user instance\n"
" --dump-core[=0|1] Dump core on crash\n"
" --crash-shell[=0|1] Run shell on crash\n"
" --confirm-spawn[=0|1] Ask for confirmation when spawning processes\n"
" --show-status[=0|1] Show status updates on the console during bootup\n"
" --log-target=TARGET Set log target (console, journal, syslog, kmsg, journal-or-kmsg, syslog-or-kmsg, null)\n"
" --log-level=LEVEL Set log level (debug, info, notice, warning, err, crit, alert, emerg)\n"
" --log-color[=0|1] Highlight important log messages\n"
" --log-location[=0|1] Include code location in log messages\n"
" --default-standard-output= Set default standard output for services\n"
" --default-standard-error= Set default standard error output for services\n",
program_invocation_short_name);
return 0;
}
static int version(void) {
puts(PACKAGE_STRING);
puts(SYSTEMD_FEATURES);
return 0;
}
static int prepare_reexecute(Manager *m, FILE **_f, FDSet **_fds, bool switching_root) {
FILE *f = NULL;
FDSet *fds = NULL;
int r;
assert(m);
assert(_f);
assert(_fds);
r = manager_open_serialization(m, &f);
if (r < 0) {
log_error("Failed to create serialization file: %s", strerror(-r));
goto fail;
}
/* Make sure nothing is really destructed when we shut down */
m->n_reloading ++;
bus_manager_send_reloading(m, true);
fds = fdset_new();
if (!fds) {
r = -ENOMEM;
log_error("Failed to allocate fd set: %s", strerror(-r));
goto fail;
}
r = manager_serialize(m, f, fds, switching_root);
if (r < 0) {
log_error("Failed to serialize state: %s", strerror(-r));
goto fail;
}
if (fseeko(f, 0, SEEK_SET) < 0) {
log_error("Failed to rewind serialization fd: %m");
goto fail;
}
r = fd_cloexec(fileno(f), false);
if (r < 0) {
log_error("Failed to disable O_CLOEXEC for serialization: %s", strerror(-r));
goto fail;
}
r = fdset_cloexec(fds, false);
if (r < 0) {
log_error("Failed to disable O_CLOEXEC for serialization fds: %s", strerror(-r));
goto fail;
}
*_f = f;
*_fds = fds;
return 0;
fail:
fdset_free(fds);
if (f)
fclose(f);
return r;
}
static int bump_rlimit_nofile(struct rlimit *saved_rlimit) {
struct rlimit nl;
int r;
assert(saved_rlimit);
/* Save the original RLIMIT_NOFILE so that we can reset it
* later when transitioning from the initrd to the main
* systemd or suchlike. */
if (getrlimit(RLIMIT_NOFILE, saved_rlimit) < 0) {
log_error("Reading RLIMIT_NOFILE failed: %m");
return -errno;
}
/* Make sure forked processes get the default kernel setting */
if (!arg_default_rlimit[RLIMIT_NOFILE]) {
struct rlimit *rl;
rl = newdup(struct rlimit, saved_rlimit, 1);
if (!rl)
return log_oom();
arg_default_rlimit[RLIMIT_NOFILE] = rl;
}
/* Bump up the resource limit for ourselves substantially */
nl.rlim_cur = nl.rlim_max = 64*1024;
r = setrlimit_closest(RLIMIT_NOFILE, &nl);
if (r < 0) {
log_error("Setting RLIMIT_NOFILE failed: %s", strerror(-r));
return r;
}
return 0;
}
static void test_mtab(void) {
static const char ok[] =
"/proc/self/mounts\0"
"/proc/mounts\0"
"../proc/self/mounts\0"
"../proc/mounts\0";
_cleanup_free_ char *p = NULL;
int r;
/* Check that /etc/mtab is a symlink to the right place or
* non-existing. But certainly not a file, or a symlink to
* some weird place... */
r = readlink_malloc("/etc/mtab", &p);
if (r == -ENOENT)
return;
if (r >= 0 && nulstr_contains(ok, p))
return;
log_warning("/etc/mtab is not a symlink or not pointing to /proc/self/mounts. "
"This is not supported anymore. "
"Please make sure to replace this file by a symlink to avoid incorrect or misleading mount(8) output.");
}
static void test_usr(void) {
/* Check that /usr is not a separate fs */
if (dir_is_empty("/usr") <= 0)
return;
log_warning("/usr appears to be on its own filesytem and is not already mounted. This is not a supported setup. "
"Some things will probably break (sometimes even silently) in mysterious ways. "
"Consult http://freedesktop.org/wiki/Software/systemd/separate-usr-is-broken for more information.");
}
static int initialize_join_controllers(void) {
/* By default, mount "cpu" + "cpuacct" together, and "net_cls"
* + "net_prio". We'd like to add "cpuset" to the mix, but
* "cpuset" does't really work for groups with no initialized
* attributes. */
arg_join_controllers = new(char**, 3);
if (!arg_join_controllers)
return -ENOMEM;
arg_join_controllers[0] = strv_new("cpu", "cpuacct", NULL);
arg_join_controllers[1] = strv_new("net_cls", "net_prio", NULL);
arg_join_controllers[2] = NULL;
if (!arg_join_controllers[0] || !arg_join_controllers[1]) {
free_join_controllers();
return -ENOMEM;
}
return 0;
}
static int enforce_syscall_archs(Set *archs) {
#ifdef HAVE_SECCOMP
scmp_filter_ctx *seccomp;
Iterator i;
void *id;
int r;
seccomp = seccomp_init(SCMP_ACT_ALLOW);
if (!seccomp)
return log_oom();
SET_FOREACH(id, arg_syscall_archs, i) {
r = seccomp_arch_add(seccomp, PTR_TO_UINT32(id) - 1);
if (r == -EEXIST)
continue;
if (r < 0) {
log_error("Failed to add architecture to seccomp: %s", strerror(-r));
goto finish;
}
}
r = seccomp_attr_set(seccomp, SCMP_FLTATR_CTL_NNP, 0);
if (r < 0) {
log_error("Failed to unset NO_NEW_PRIVS: %s", strerror(-r));
goto finish;
}
r = seccomp_load(seccomp);
if (r < 0)
log_error("Failed to add install architecture seccomp: %s", strerror(-r));
finish:
seccomp_release(seccomp);
return r;
#else
return 0;
#endif
}
static int status_welcome(void) {
_cleanup_free_ char *pretty_name = NULL, *ansi_color = NULL;
int r;
r = parse_env_file("/etc/os-release", NEWLINE,
"PRETTY_NAME", &pretty_name,
"ANSI_COLOR", &ansi_color,
NULL);
if (r == -ENOENT) {
r = parse_env_file("/usr/lib/os-release", NEWLINE,
"PRETTY_NAME", &pretty_name,
"ANSI_COLOR", &ansi_color,
NULL);
}
if (r < 0 && r != -ENOENT)
log_warning("Failed to read os-release file: %s", strerror(-r));
return status_printf(NULL, false, false,
"\nWelcome to \x1B[%sm%s\x1B[0m!\n",
isempty(ansi_color) ? "1" : ansi_color,
isempty(pretty_name) ? "Linux" : pretty_name);
}
static int write_container_id(void) {
const char *c;
c = getenv("container");
if (isempty(c))
return 0;
return write_string_file("/run/systemd/container", c);
}
int main(int argc, char *argv[]) {
Manager *m = NULL;
int r, retval = EXIT_FAILURE;
usec_t before_startup, after_startup;
char timespan[FORMAT_TIMESPAN_MAX];
FDSet *fds = NULL;
bool reexecute = false;
const char *shutdown_verb = NULL;
dual_timestamp initrd_timestamp = { 0ULL, 0ULL };
dual_timestamp userspace_timestamp = { 0ULL, 0ULL };
dual_timestamp kernel_timestamp = { 0ULL, 0ULL };
dual_timestamp security_start_timestamp = { 0ULL, 0ULL };
dual_timestamp security_finish_timestamp = { 0ULL, 0ULL };
static char systemd[] = "systemd";
bool skip_setup = false;
unsigned j;
bool loaded_policy = false;
bool arm_reboot_watchdog = false;
bool queue_default_job = false;
bool empty_etc = false;
char *switch_root_dir = NULL, *switch_root_init = NULL;
static struct rlimit saved_rlimit_nofile = { 0, 0 };
#ifdef HAVE_SYSV_COMPAT
if (getpid() != 1 && strstr(program_invocation_short_name, "init")) {
/* This is compatibility support for SysV, where
* calling init as a user is identical to telinit. */
errno = -ENOENT;
execv(SYSTEMCTL_BINARY_PATH, argv);
log_error("Failed to exec " SYSTEMCTL_BINARY_PATH ": %m");
return 1;
}
#endif
dual_timestamp_from_monotonic(&kernel_timestamp, 0);
dual_timestamp_get(&userspace_timestamp);
/* Determine if this is a reexecution or normal bootup. We do
* the full command line parsing much later, so let's just
* have a quick peek here. */
if (strv_find(argv+1, "--deserialize"))
skip_setup = true;
/* If we have switched root, do all the special setup
* things */
if (strv_find(argv+1, "--switched-root"))
skip_setup = false;
/* If we get started via the /sbin/init symlink then we are
called 'init'. After a subsequent reexecution we are then
called 'systemd'. That is confusing, hence let's call us
systemd right-away. */
program_invocation_short_name = systemd;
prctl(PR_SET_NAME, systemd);
saved_argv = argv;
saved_argc = argc;
log_show_color(isatty(STDERR_FILENO) > 0);
/* Disable the umask logic */
if (getpid() == 1)
umask(0);
if (getpid() == 1 && detect_container(NULL) <= 0) {
/* Running outside of a container as PID 1 */
arg_running_as = SYSTEMD_SYSTEM;
make_null_stdio();
log_set_target(LOG_TARGET_KMSG);
log_open();
if (in_initrd())
initrd_timestamp = userspace_timestamp;
if (!skip_setup) {
mount_setup_early();
dual_timestamp_get(&security_start_timestamp);
if (selinux_setup(&loaded_policy) < 0)
goto finish;
if (ima_setup() < 0)
goto finish;
if (smack_setup(&loaded_policy) < 0)
goto finish;
dual_timestamp_get(&security_finish_timestamp);
}
if (label_init(NULL) < 0)
goto finish;
if (!skip_setup) {
if (clock_is_localtime() > 0) {
int min;
/* The first-time call to settimeofday() does a time warp in the kernel */
r = clock_set_timezone(&min);
if (r < 0)
log_error("Failed to apply local time delta, ignoring: %s", strerror(-r));
else
log_info("RTC configured in localtime, applying delta of %i minutes to system time.", min);
} else if (!in_initrd()) {
/*
* Do dummy first-time call to seal the kernel's time warp magic
*
* Do not call this this from inside the initrd. The initrd might not
* carry /etc/adjtime with LOCAL, but the real system could be set up
* that way. In such case, we need to delay the time-warp or the sealing
* until we reach the real system.
*/
clock_reset_timezone();
/* Tell the kernel our timezone */
r = clock_set_timezone(NULL);
if (r < 0)
log_error("Failed to set the kernel's timezone, ignoring: %s", strerror(-r));
}
}
/* Set the default for later on, but don't actually
* open the logs like this for now. Note that if we
* are transitioning from the initrd there might still
* be journal fd open, and we shouldn't attempt
* opening that before we parsed /proc/cmdline which
* might redirect output elsewhere. */
log_set_target(LOG_TARGET_JOURNAL_OR_KMSG);
} else if (getpid() == 1) {
/* Running inside a container, as PID 1 */
arg_running_as = SYSTEMD_SYSTEM;
log_set_target(LOG_TARGET_CONSOLE);
log_close_console(); /* force reopen of /dev/console */
log_open();
/* For the later on, see above... */
log_set_target(LOG_TARGET_JOURNAL);
/* clear the kernel timestamp,
* because we are in a container */
kernel_timestamp.monotonic = 0ULL;
kernel_timestamp.realtime = 0ULL;
} else {
/* Running as user instance */
arg_running_as = SYSTEMD_USER;
log_set_target(LOG_TARGET_AUTO);
log_open();
/* clear the kernel timestamp,
* because we are not PID 1 */
kernel_timestamp.monotonic = 0ULL;
kernel_timestamp.realtime = 0ULL;
}
/* Initialize default unit */
r = set_default_unit(SPECIAL_DEFAULT_TARGET);
if (r < 0) {
log_error("Failed to set default unit %s: %s", SPECIAL_DEFAULT_TARGET, strerror(-r));
goto finish;
}
r = initialize_join_controllers();
if (r < 0)
goto finish;
/* Mount /proc, /sys and friends, so that /proc/cmdline and
* /proc/$PID/fd is available. */
if (getpid() == 1) {
r = mount_setup(loaded_policy);
if (r < 0)
goto finish;
}
/* Reset all signal handlers. */
assert_se(reset_all_signal_handlers() == 0);
ignore_signals(SIGNALS_IGNORE, -1);
if (parse_config_file() < 0)
goto finish;
if (arg_running_as == SYSTEMD_SYSTEM)
if (parse_proc_cmdline(parse_proc_cmdline_item) < 0)
goto finish;
log_parse_environment();
if (parse_argv(argc, argv) < 0)
goto finish;
if (arg_action == ACTION_TEST &&
geteuid() == 0) {
log_error("Don't run test mode as root.");
goto finish;
}
if (arg_running_as == SYSTEMD_USER &&
arg_action == ACTION_RUN &&
sd_booted() <= 0) {
log_error("Trying to run as user instance, but the system has not been booted with systemd.");
goto finish;
}
if (arg_running_as == SYSTEMD_SYSTEM &&
arg_action == ACTION_RUN &&
running_in_chroot() > 0) {
log_error("Cannot be run in a chroot() environment.");
goto finish;
}
if (arg_action == ACTION_HELP) {
retval = help();
goto finish;
} else if (arg_action == ACTION_VERSION) {
retval = version();
goto finish;
} else if (arg_action == ACTION_DUMP_CONFIGURATION_ITEMS) {
unit_dump_config_items(stdout);
retval = EXIT_SUCCESS;
goto finish;
} else if (arg_action == ACTION_DONE) {
retval = EXIT_SUCCESS;
goto finish;
}
if (arg_running_as == SYSTEMD_USER &&
!getenv("XDG_RUNTIME_DIR")) {
log_error("Trying to run as user instance, but $XDG_RUNTIME_DIR is not set.");
goto finish;
}
assert_se(arg_action == ACTION_RUN || arg_action == ACTION_TEST);
/* Close logging fds, in order not to confuse fdset below */
log_close();
/* Remember open file descriptors for later deserialization */
r = fdset_new_fill(&fds);
if (r < 0) {
log_error("Failed to allocate fd set: %s", strerror(-r));
goto finish;
} else
fdset_cloexec(fds, true);
if (arg_serialization)
assert_se(fdset_remove(fds, fileno(arg_serialization)) >= 0);
if (arg_running_as == SYSTEMD_SYSTEM)
/* Become a session leader if we aren't one yet. */
setsid();
/* Move out of the way, so that we won't block unmounts */
assert_se(chdir("/") == 0);
/* Reset the console, but only if this is really init and we
* are freshly booted */
if (arg_running_as == SYSTEMD_SYSTEM && arg_action == ACTION_RUN)
console_setup(getpid() == 1 && !skip_setup);
/* Open the logging devices, if possible and necessary */
log_open();
if (arg_show_status == _SHOW_STATUS_UNSET)
arg_show_status = SHOW_STATUS_YES;
/* Make sure we leave a core dump without panicing the
* kernel. */
if (getpid() == 1) {
install_crash_handler();
r = mount_cgroup_controllers(arg_join_controllers);
if (r < 0)
goto finish;
}
if (arg_running_as == SYSTEMD_SYSTEM) {
const char *virtualization = NULL;
log_info(PACKAGE_STRING " running in system mode. (" SYSTEMD_FEATURES ")");
detect_virtualization(&virtualization);
if (virtualization)
log_info("Detected virtualization '%s'.", virtualization);
write_container_id();
log_info("Detected architecture '%s'.", architecture_to_string(uname_architecture()));
if (in_initrd())
log_info("Running in initial RAM disk.");
/* Let's check whether /etc is already populated. We
* don't actually really check for that, but use
* /etc/machine-id as flag file. This allows container
* managers and installers to provision a couple of
* files already. If the container manager wants to
* provision the machine ID itself it should pass
* $container_uuid to PID 1.*/
empty_etc = access("/etc/machine-id", F_OK) < 0;
if (empty_etc)
log_info("Running with unpopulated /etc.");
} else {
_cleanup_free_ char *t;
t = uid_to_name(getuid());
log_debug(PACKAGE_STRING " running in user mode for user "UID_FMT"/%s. (" SYSTEMD_FEATURES ")", getuid(), strna(t));
}
if (arg_running_as == SYSTEMD_SYSTEM && !skip_setup) {
if (arg_show_status > 0 || plymouth_running())
status_welcome();
#ifdef HAVE_KMOD
kmod_setup();
#endif
hostname_setup();
machine_id_setup(NULL);
loopback_setup();
test_mtab();
test_usr();
}
if (arg_running_as == SYSTEMD_SYSTEM && arg_runtime_watchdog > 0)
watchdog_set_timeout(&arg_runtime_watchdog);
if (arg_timer_slack_nsec != (nsec_t) -1)
if (prctl(PR_SET_TIMERSLACK, arg_timer_slack_nsec) < 0)
log_error("Failed to adjust timer slack: %m");
if (arg_capability_bounding_set_drop) {
r = capability_bounding_set_drop_usermode(arg_capability_bounding_set_drop);
if (r < 0) {
log_error("Failed to drop capability bounding set of usermode helpers: %s", strerror(-r));
goto finish;
}
r = capability_bounding_set_drop(arg_capability_bounding_set_drop, true);
if (r < 0) {
log_error("Failed to drop capability bounding set: %s", strerror(-r));
goto finish;
}
}
if (arg_syscall_archs) {
r = enforce_syscall_archs(arg_syscall_archs);
if (r < 0)
goto finish;
}
if (arg_running_as == SYSTEMD_USER) {
/* Become reaper of our children */
if (prctl(PR_SET_CHILD_SUBREAPER, 1) < 0) {
log_warning("Failed to make us a subreaper: %m");
if (errno == EINVAL)
log_info("Perhaps the kernel version is too old (< 3.4?)");
}
}
if (arg_running_as == SYSTEMD_SYSTEM) {
bump_rlimit_nofile(&saved_rlimit_nofile);
if (empty_etc) {
r = unit_file_preset_all(UNIT_FILE_SYSTEM, false, NULL, UNIT_FILE_PRESET_FULL, false, NULL, 0);
if (r < 0)
log_warning("Failed to populate /etc with preset unit settings, ignoring: %s", strerror(-r));
else
log_info("Populated /etc with preset unit settings.");
}
}
r = manager_new(arg_running_as, &m);
if (r < 0) {
log_error("Failed to allocate manager object: %s", strerror(-r));
goto finish;
}
m->confirm_spawn = arg_confirm_spawn;
m->default_timer_accuracy_usec = arg_default_timer_accuracy_usec;
m->default_std_output = arg_default_std_output;
m->default_std_error = arg_default_std_error;
m->default_restart_usec = arg_default_restart_usec;
m->default_timeout_start_usec = arg_default_timeout_start_usec;
m->default_timeout_stop_usec = arg_default_timeout_stop_usec;
m->default_start_limit_interval = arg_default_start_limit_interval;
m->default_start_limit_burst = arg_default_start_limit_burst;
m->default_cpu_accounting = arg_default_cpu_accounting;
m->default_blockio_accounting = arg_default_blockio_accounting;
m->default_memory_accounting = arg_default_memory_accounting;
m->runtime_watchdog = arg_runtime_watchdog;
m->shutdown_watchdog = arg_shutdown_watchdog;
m->userspace_timestamp = userspace_timestamp;
m->kernel_timestamp = kernel_timestamp;
m->initrd_timestamp = initrd_timestamp;
m->security_start_timestamp = security_start_timestamp;
m->security_finish_timestamp = security_finish_timestamp;
manager_set_default_rlimits(m, arg_default_rlimit);
manager_environment_add(m, NULL, arg_default_environment);
manager_set_show_status(m, arg_show_status);
/* Remember whether we should queue the default job */
queue_default_job = !arg_serialization || arg_switched_root;
before_startup = now(CLOCK_MONOTONIC);
r = manager_startup(m, arg_serialization, fds);
if (r < 0)
log_error("Failed to fully start up daemon: %s", strerror(-r));
/* This will close all file descriptors that were opened, but
* not claimed by any unit. */
fdset_free(fds);
fds = NULL;
if (arg_serialization) {
fclose(arg_serialization);
arg_serialization = NULL;
}
if (queue_default_job) {
_cleanup_bus_error_free_ sd_bus_error error = SD_BUS_ERROR_NULL;
Unit *target = NULL;
Job *default_unit_job;
log_debug("Activating default unit: %s", arg_default_unit);
r = manager_load_unit(m, arg_default_unit, NULL, &error, &target);
if (r < 0)
log_error("Failed to load default target: %s", bus_error_message(&error, r));
else if (target->load_state == UNIT_ERROR || target->load_state == UNIT_NOT_FOUND)
log_error("Failed to load default target: %s", strerror(-target->load_error));
else if (target->load_state == UNIT_MASKED)
log_error("Default target masked.");
if (!target || target->load_state != UNIT_LOADED) {
log_info("Trying to load rescue target...");
r = manager_load_unit(m, SPECIAL_RESCUE_TARGET, NULL, &error, &target);
if (r < 0) {
log_error("Failed to load rescue target: %s", bus_error_message(&error, r));
goto finish;
} else if (target->load_state == UNIT_ERROR || target->load_state == UNIT_NOT_FOUND) {
log_error("Failed to load rescue target: %s", strerror(-target->load_error));
goto finish;
} else if (target->load_state == UNIT_MASKED) {
log_error("Rescue target masked.");
goto finish;
}
}
assert(target->load_state == UNIT_LOADED);
if (arg_action == ACTION_TEST) {
printf("-> By units:\n");
manager_dump_units(m, stdout, "\t");
}
r = manager_add_job(m, JOB_START, target, JOB_ISOLATE, false, &error, &default_unit_job);
if (r == -EPERM) {
log_debug("Default target could not be isolated, starting instead: %s", bus_error_message(&error, r));
r = manager_add_job(m, JOB_START, target, JOB_REPLACE, false, &error, &default_unit_job);
if (r < 0) {
log_error("Failed to start default target: %s", bus_error_message(&error, r));
goto finish;
}
} else if (r < 0) {
log_error("Failed to isolate default target: %s", bus_error_message(&error, r));
goto finish;
}
m->default_unit_job_id = default_unit_job->id;
after_startup = now(CLOCK_MONOTONIC);
log_full(arg_action == ACTION_TEST ? LOG_INFO : LOG_DEBUG,
"Loaded units and determined initial transaction in %s.",
format_timespan(timespan, sizeof(timespan), after_startup - before_startup, 0));
if (arg_action == ACTION_TEST) {
printf("-> By jobs:\n");
manager_dump_jobs(m, stdout, "\t");
retval = EXIT_SUCCESS;
goto finish;
}
}
for (;;) {
r = manager_loop(m);
if (r < 0) {
log_error("Failed to run mainloop: %s", strerror(-r));
goto finish;
}
switch (m->exit_code) {
case MANAGER_EXIT:
retval = EXIT_SUCCESS;
log_debug("Exit.");
goto finish;
case MANAGER_RELOAD:
log_info("Reloading.");
r = manager_reload(m);
if (r < 0)
log_error("Failed to reload: %s", strerror(-r));
break;
case MANAGER_REEXECUTE:
if (prepare_reexecute(m, &arg_serialization, &fds, false) < 0)
goto finish;
reexecute = true;
log_notice("Reexecuting.");
goto finish;
case MANAGER_SWITCH_ROOT:
/* Steal the switch root parameters */
switch_root_dir = m->switch_root;
switch_root_init = m->switch_root_init;
m->switch_root = m->switch_root_init = NULL;
if (!switch_root_init)
if (prepare_reexecute(m, &arg_serialization, &fds, true) < 0)
goto finish;
reexecute = true;
log_notice("Switching root.");
goto finish;
case MANAGER_REBOOT:
case MANAGER_POWEROFF:
case MANAGER_HALT:
case MANAGER_KEXEC: {
static const char * const table[_MANAGER_EXIT_CODE_MAX] = {
[MANAGER_REBOOT] = "reboot",
[MANAGER_POWEROFF] = "poweroff",
[MANAGER_HALT] = "halt",
[MANAGER_KEXEC] = "kexec"
};
assert_se(shutdown_verb = table[m->exit_code]);
arm_reboot_watchdog = m->exit_code == MANAGER_REBOOT;
log_notice("Shutting down.");
goto finish;
}
default:
assert_not_reached("Unknown exit code.");
}
}
finish:
if (m) {
manager_free(m);
m = NULL;
}
for (j = 0; j < ELEMENTSOF(arg_default_rlimit); j++) {
free(arg_default_rlimit[j]);
arg_default_rlimit[j] = NULL;
}
free(arg_default_unit);
arg_default_unit = NULL;
free_join_controllers();
strv_free(arg_default_environment);
arg_default_environment = NULL;
set_free(arg_syscall_archs);
arg_syscall_archs = NULL;
label_finish();
if (reexecute) {
const char **args;
unsigned i, args_size;
sigset_t ss;
/* Close and disarm the watchdog, so that the new
* instance can reinitialize it, but doesn't get
* rebooted while we do that */
watchdog_close(true);
/* Reset the RLIMIT_NOFILE to the kernel default, so
* that the new systemd can pass the kernel default to
* its child processes */
if (saved_rlimit_nofile.rlim_cur > 0)
setrlimit(RLIMIT_NOFILE, &saved_rlimit_nofile);
if (switch_root_dir) {
/* Kill all remaining processes from the
* initrd, but don't wait for them, so that we
* can handle the SIGCHLD for them after
* deserializing. */
broadcast_signal(SIGTERM, false, true);
/* And switch root */
r = switch_root(switch_root_dir);
if (r < 0)
log_error("Failed to switch root, ignoring: %s", strerror(-r));
}
args_size = MAX(6, argc+1);
args = newa(const char*, args_size);
if (!switch_root_init) {
char sfd[16];
/* First try to spawn ourselves with the right
* path, and with full serialization. We do
* this only if the user didn't specify an
* explicit init to spawn. */
assert(arg_serialization);
assert(fds);
snprintf(sfd, sizeof(sfd), "%i", fileno(arg_serialization));
char_array_0(sfd);
i = 0;
args[i++] = SYSTEMD_BINARY_PATH;
if (switch_root_dir)
args[i++] = "--switched-root";
args[i++] = arg_running_as == SYSTEMD_SYSTEM ? "--system" : "--user";
args[i++] = "--deserialize";
args[i++] = sfd;
args[i++] = NULL;
/* do not pass along the environment we inherit from the kernel or initrd */
if (switch_root_dir)
clearenv();
assert(i <= args_size);
execv(args[0], (char* const*) args);
}
/* Try the fallback, if there is any, without any
* serialization. We pass the original argv[] and
* envp[]. (Well, modulo the ordering changes due to
* getopt() in argv[], and some cleanups in envp[],
* but let's hope that doesn't matter.) */
if (arg_serialization) {
fclose(arg_serialization);
arg_serialization = NULL;
}
if (fds) {
fdset_free(fds);
fds = NULL;
}
/* Reopen the console */
make_console_stdio();
for (j = 1, i = 1; j < (unsigned) argc; j++)
args[i++] = argv[j];
args[i++] = NULL;
assert(i <= args_size);
/* reenable any blocked signals, especially important
* if we switch from initial ramdisk to init=... */
reset_all_signal_handlers();
assert_se(sigemptyset(&ss) == 0);
assert_se(sigprocmask(SIG_SETMASK, &ss, NULL) == 0);
if (switch_root_init) {
args[0] = switch_root_init;
execv(args[0], (char* const*) args);
log_warning("Failed to execute configured init, trying fallback: %m");
}
args[0] = "/sbin/init";
execv(args[0], (char* const*) args);
if (errno == ENOENT) {
log_warning("No /sbin/init, trying fallback");
args[0] = "/bin/sh";
args[1] = NULL;
execv(args[0], (char* const*) args);
log_error("Failed to execute /bin/sh, giving up: %m");
} else
log_warning("Failed to execute /sbin/init, giving up: %m");
}
if (arg_serialization) {
fclose(arg_serialization);
arg_serialization = NULL;
}
if (fds) {
fdset_free(fds);
fds = NULL;
}
#ifdef HAVE_VALGRIND_VALGRIND_H
/* If we are PID 1 and running under valgrind, then let's exit
* here explicitly. valgrind will only generate nice output on
* exit(), not on exec(), hence let's do the former not the
* latter here. */
if (getpid() == 1 && RUNNING_ON_VALGRIND)
return 0;
#endif
if (shutdown_verb) {
char log_level[DECIMAL_STR_MAX(int) + 1];
const char* command_line[9] = {
SYSTEMD_SHUTDOWN_BINARY_PATH,
shutdown_verb,
"--log-level", log_level,
"--log-target",
};
unsigned pos = 5;
_cleanup_strv_free_ char **env_block = NULL;
assert(command_line[pos] == NULL);
env_block = strv_copy(environ);
snprintf(log_level, sizeof(log_level), "%d", log_get_max_level());
switch (log_get_target()) {
case LOG_TARGET_KMSG:
case LOG_TARGET_JOURNAL_OR_KMSG:
case LOG_TARGET_SYSLOG_OR_KMSG:
command_line[pos++] = "kmsg";
break;
case LOG_TARGET_CONSOLE:
default:
command_line[pos++] = "console";
break;
};
if (log_get_show_color())
command_line[pos++] = "--log-color";
if (log_get_show_location())
command_line[pos++] = "--log-location";
assert(pos < ELEMENTSOF(command_line));
if (arm_reboot_watchdog && arg_shutdown_watchdog > 0) {
char *e;
/* If we reboot let's set the shutdown
* watchdog and tell the shutdown binary to
* repeatedly ping it */
watchdog_set_timeout(&arg_shutdown_watchdog);
watchdog_close(false);
/* Tell the binary how often to ping, ignore failure */
if (asprintf(&e, "WATCHDOG_USEC="USEC_FMT, arg_shutdown_watchdog) > 0)
strv_push(&env_block, e);
} else
watchdog_close(true);
/* Avoid the creation of new processes forked by the
* kernel; at this point, we will not listen to the
* signals anyway */
if (detect_container(NULL) <= 0)
cg_uninstall_release_agent(SYSTEMD_CGROUP_CONTROLLER);
execve(SYSTEMD_SHUTDOWN_BINARY_PATH, (char **) command_line, env_block);
log_error("Failed to execute shutdown binary, %s: %m",
getpid() == 1 ? "freezing" : "quitting");
}
if (getpid() == 1)
freeze();
return retval;
}