main.c revision 15a900327aba7dc4dc886affe1ae22d3b759b193
74b2466e14a1961bf3ac0e8a60cfaceec705bd59Lennart Poettering/***
74b2466e14a1961bf3ac0e8a60cfaceec705bd59Lennart Poettering This file is part of systemd.
74b2466e14a1961bf3ac0e8a60cfaceec705bd59Lennart Poettering
74b2466e14a1961bf3ac0e8a60cfaceec705bd59Lennart Poettering Copyright 2010 Lennart Poettering
74b2466e14a1961bf3ac0e8a60cfaceec705bd59Lennart Poettering
74b2466e14a1961bf3ac0e8a60cfaceec705bd59Lennart Poettering systemd is free software; you can redistribute it and/or modify it
74b2466e14a1961bf3ac0e8a60cfaceec705bd59Lennart Poettering under the terms of the GNU Lesser General Public License as published by
74b2466e14a1961bf3ac0e8a60cfaceec705bd59Lennart Poettering the Free Software Foundation; either version 2.1 of the License, or
74b2466e14a1961bf3ac0e8a60cfaceec705bd59Lennart Poettering (at your option) any later version.
74b2466e14a1961bf3ac0e8a60cfaceec705bd59Lennart Poettering
74b2466e14a1961bf3ac0e8a60cfaceec705bd59Lennart Poettering systemd is distributed in the hope that it will be useful, but
74b2466e14a1961bf3ac0e8a60cfaceec705bd59Lennart Poettering WITHOUT ANY WARRANTY; without even the implied warranty of
74b2466e14a1961bf3ac0e8a60cfaceec705bd59Lennart Poettering MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
74b2466e14a1961bf3ac0e8a60cfaceec705bd59Lennart Poettering Lesser General Public License for more details.
74b2466e14a1961bf3ac0e8a60cfaceec705bd59Lennart Poettering
74b2466e14a1961bf3ac0e8a60cfaceec705bd59Lennart Poettering You should have received a copy of the GNU Lesser General Public License
74b2466e14a1961bf3ac0e8a60cfaceec705bd59Lennart Poettering along with systemd; If not, see <http://www.gnu.org/licenses/>.
74b2466e14a1961bf3ac0e8a60cfaceec705bd59Lennart Poettering***/
74b2466e14a1961bf3ac0e8a60cfaceec705bd59Lennart Poettering
74b2466e14a1961bf3ac0e8a60cfaceec705bd59Lennart Poettering#include <errno.h>
74b2466e14a1961bf3ac0e8a60cfaceec705bd59Lennart Poettering#include <fcntl.h>
0dae31d468b1a0e22d98921f7b0dbd92fd217167Zbigniew Jędrzejewski-Szmek#include <getopt.h>
0dae31d468b1a0e22d98921f7b0dbd92fd217167Zbigniew Jędrzejewski-Szmek#include <signal.h>
2e276efc7b0398a3086629a52970bdd4ab7252f9Zbigniew Jędrzejewski-Szmek#include <stdio.h>
2e276efc7b0398a3086629a52970bdd4ab7252f9Zbigniew Jędrzejewski-Szmek#include <string.h>
4ad7f2761da661853dcc29d542efb4727abb1101Nick Owens#include <sys/mount.h>
74b2466e14a1961bf3ac0e8a60cfaceec705bd59Lennart Poettering#include <sys/prctl.h>
8db0d2f5c37e7e8f5bfce016cfdad7947a3ea939Zbigniew Jędrzejewski-Szmek#include <sys/reboot.h>
7263f72499e962b3fd54cdb7c79d49ca72121edeZbigniew Jędrzejewski-Szmek#include <sys/stat.h>
74b2466e14a1961bf3ac0e8a60cfaceec705bd59Lennart Poettering#include <unistd.h>
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering#ifdef HAVE_SECCOMP
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering#include <seccomp.h>
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering#endif
74b2466e14a1961bf3ac0e8a60cfaceec705bd59Lennart Poettering#ifdef HAVE_VALGRIND_VALGRIND_H
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering#include <valgrind/valgrind.h>
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering#endif
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering#include "sd-bus.h"
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering#include "sd-daemon.h"
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering#include "alloc-util.h"
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering#include "architecture.h"
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering#include "build.h"
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering#include "bus-error.h"
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering#include "bus-util.h"
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering#include "capability-util.h"
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering#include "clock-util.h"
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering#include "conf-parser.h"
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering#include "cpu-set-util.h"
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering#include "dbus-manager.h"
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering#include "def.h"
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering#include "env-util.h"
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering#include "fd-util.h"
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering#include "fdset.h"
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering#include "fileio.h"
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering#include "formats-util.h"
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering#include "fs-util.h"
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering#include "hostname-setup.h"
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering#include "ima-setup.h"
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering#include "killall.h"
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering#include "kmod-setup.h"
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering#include "load-fragment.h"
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering#include "log.h"
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering#include "loopback-setup.h"
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering#include "machine-id-setup.h"
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering#include "manager.h"
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering#include "missing.h"
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering#include "mount-setup.h"
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering#include "pager.h"
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering#include "parse-util.h"
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering#include "proc-cmdline.h"
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering#include "process-util.h"
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering#include "rlimit-util.h"
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering#include "selinux-setup.h"
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering#include "selinux-util.h"
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering#include "signal-util.h"
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering#include "smack-setup.h"
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering#include "special.h"
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering#include "stat-util.h"
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering#include "stdio-util.h"
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering#include "strv.h"
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering#include "switch-root.h"
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering#include "terminal-util.h"
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering#include "user-util.h"
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering#include "virt.h"
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering#include "watchdog.h"
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poetteringstatic enum {
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering ACTION_RUN,
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering ACTION_HELP,
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering ACTION_VERSION,
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering ACTION_TEST,
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering ACTION_DUMP_CONFIGURATION_ITEMS,
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering ACTION_DONE
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering} arg_action = ACTION_RUN;
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poetteringstatic char *arg_default_unit = NULL;
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poetteringstatic ManagerRunningAs arg_running_as = _MANAGER_RUNNING_AS_INVALID;
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poetteringstatic bool arg_dump_core = true;
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poetteringstatic int arg_crash_chvt = -1;
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poetteringstatic bool arg_crash_shell = false;
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poetteringstatic bool arg_crash_reboot = false;
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poetteringstatic bool arg_confirm_spawn = false;
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poetteringstatic ShowStatus arg_show_status = _SHOW_STATUS_UNSET;
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poetteringstatic bool arg_switched_root = false;
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poetteringstatic int arg_no_pager = -1;
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poetteringstatic char ***arg_join_controllers = NULL;
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poetteringstatic ExecOutput arg_default_std_output = EXEC_OUTPUT_JOURNAL;
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poetteringstatic ExecOutput arg_default_std_error = EXEC_OUTPUT_INHERIT;
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poetteringstatic usec_t arg_default_restart_usec = DEFAULT_RESTART_USEC;
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poetteringstatic usec_t arg_default_timeout_start_usec = DEFAULT_TIMEOUT_USEC;
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poetteringstatic usec_t arg_default_timeout_stop_usec = DEFAULT_TIMEOUT_USEC;
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poetteringstatic usec_t arg_default_start_limit_interval = DEFAULT_START_LIMIT_INTERVAL;
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poetteringstatic unsigned arg_default_start_limit_burst = DEFAULT_START_LIMIT_BURST;
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poetteringstatic usec_t arg_runtime_watchdog = 0;
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poetteringstatic usec_t arg_shutdown_watchdog = 10 * USEC_PER_MINUTE;
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poetteringstatic char **arg_default_environment = NULL;
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poetteringstatic struct rlimit *arg_default_rlimit[_RLIMIT_MAX] = {};
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poetteringstatic uint64_t arg_capability_bounding_set = CAP_ALL;
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poetteringstatic nsec_t arg_timer_slack_nsec = NSEC_INFINITY;
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poetteringstatic usec_t arg_default_timer_accuracy_usec = 1 * USEC_PER_MINUTE;
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poetteringstatic Set* arg_syscall_archs = NULL;
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poetteringstatic FILE* arg_serialization = NULL;
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poetteringstatic bool arg_default_cpu_accounting = false;
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poetteringstatic bool arg_default_blockio_accounting = false;
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poetteringstatic bool arg_default_memory_accounting = false;
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poetteringstatic bool arg_default_tasks_accounting = true;
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poetteringstatic uint64_t arg_default_tasks_max = UINT64_C(512);
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poetteringstatic sd_id128_t arg_machine_id = {};
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poetteringstatic void pager_open_if_enabled(void) {
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering if (arg_no_pager <= 0)
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering return;
74b2466e14a1961bf3ac0e8a60cfaceec705bd59Lennart Poettering
74b2466e14a1961bf3ac0e8a60cfaceec705bd59Lennart Poettering pager_open(false);
d5099efc47d4e6ac60816b5381a5f607ab03f06eMichal Schmidt}
322345fdb9865ef2477fba8e4bdde0e1183ef505Lennart Poettering
322345fdb9865ef2477fba8e4bdde0e1183ef505Lennart Poetteringnoreturn static void freeze_or_reboot(void) {
322345fdb9865ef2477fba8e4bdde0e1183ef505Lennart Poettering
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering if (arg_crash_reboot) {
322345fdb9865ef2477fba8e4bdde0e1183ef505Lennart Poettering log_notice("Rebooting in 10s...");
322345fdb9865ef2477fba8e4bdde0e1183ef505Lennart Poettering (void) sleep(10);
322345fdb9865ef2477fba8e4bdde0e1183ef505Lennart Poettering
322345fdb9865ef2477fba8e4bdde0e1183ef505Lennart Poettering log_notice("Rebooting now...");
322345fdb9865ef2477fba8e4bdde0e1183ef505Lennart Poettering (void) reboot(RB_AUTOBOOT);
322345fdb9865ef2477fba8e4bdde0e1183ef505Lennart Poettering log_emergency_errno(errno, "Failed to reboot: %m");
d5099efc47d4e6ac60816b5381a5f607ab03f06eMichal Schmidt }
322345fdb9865ef2477fba8e4bdde0e1183ef505Lennart Poettering
322345fdb9865ef2477fba8e4bdde0e1183ef505Lennart Poettering log_emergency("Freezing execution.");
322345fdb9865ef2477fba8e4bdde0e1183ef505Lennart Poettering freeze();
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering}
322345fdb9865ef2477fba8e4bdde0e1183ef505Lennart Poettering
322345fdb9865ef2477fba8e4bdde0e1183ef505Lennart Poetteringnoreturn static void crash(int sig) {
322345fdb9865ef2477fba8e4bdde0e1183ef505Lennart Poettering struct sigaction sa;
322345fdb9865ef2477fba8e4bdde0e1183ef505Lennart Poettering pid_t pid;
322345fdb9865ef2477fba8e4bdde0e1183ef505Lennart Poettering
322345fdb9865ef2477fba8e4bdde0e1183ef505Lennart Poettering if (getpid() != 1)
322345fdb9865ef2477fba8e4bdde0e1183ef505Lennart Poettering /* Pass this on immediately, if this is not PID 1 */
322345fdb9865ef2477fba8e4bdde0e1183ef505Lennart Poettering (void) raise(sig);
322345fdb9865ef2477fba8e4bdde0e1183ef505Lennart Poettering else if (!arg_dump_core)
322345fdb9865ef2477fba8e4bdde0e1183ef505Lennart Poettering log_emergency("Caught <%s>, not dumping core.", signal_to_string(sig));
322345fdb9865ef2477fba8e4bdde0e1183ef505Lennart Poettering else {
322345fdb9865ef2477fba8e4bdde0e1183ef505Lennart Poettering sa = (struct sigaction) {
322345fdb9865ef2477fba8e4bdde0e1183ef505Lennart Poettering .sa_handler = nop_signal_handler,
322345fdb9865ef2477fba8e4bdde0e1183ef505Lennart Poettering .sa_flags = SA_NOCLDSTOP|SA_RESTART,
322345fdb9865ef2477fba8e4bdde0e1183ef505Lennart Poettering };
322345fdb9865ef2477fba8e4bdde0e1183ef505Lennart Poettering
d5099efc47d4e6ac60816b5381a5f607ab03f06eMichal Schmidt /* We want to wait for the core process, hence let's enable SIGCHLD */
d5099efc47d4e6ac60816b5381a5f607ab03f06eMichal Schmidt (void) sigaction(SIGCHLD, &sa, NULL);
d5099efc47d4e6ac60816b5381a5f607ab03f06eMichal Schmidt
d5099efc47d4e6ac60816b5381a5f607ab03f06eMichal Schmidt pid = raw_clone(SIGCHLD, NULL);
d5099efc47d4e6ac60816b5381a5f607ab03f06eMichal Schmidt if (pid < 0)
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering log_emergency_errno(errno, "Caught <%s>, cannot fork for core dump: %m", signal_to_string(sig));
d23a27a964748967e1ad20e86de869a753af555bTom Gundersen else if (pid == 0) {
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering /* Enable default signal handler for core dump */
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering sa = (struct sigaction) {
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering .sa_handler = SIG_DFL,
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering };
d23a27a964748967e1ad20e86de869a753af555bTom Gundersen (void) sigaction(sig, &sa, NULL);
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering /* Don't limit the coredump size */
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering (void) setrlimit(RLIMIT_CORE, &RLIMIT_MAKE_CONST(RLIM_INFINITY));
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering /* Just to be sure... */
d23a27a964748967e1ad20e86de869a753af555bTom Gundersen (void) chdir("/");
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering /* Raise the signal again */
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering pid = raw_getpid();
23432a1c249b9c513e438bffe9778a7ce2ee74feZbigniew Jędrzejewski-Szmek (void) kill(pid, sig); /* raise() would kill the parent */
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering assert_not_reached("We shouldn't be here...");
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering _exit(EXIT_FAILURE);
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering } else {
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering siginfo_t status;
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering int r;
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering
74b2466e14a1961bf3ac0e8a60cfaceec705bd59Lennart Poettering /* Order things nicely. */
74b2466e14a1961bf3ac0e8a60cfaceec705bd59Lennart Poettering r = wait_for_terminate(pid, &status);
74b2466e14a1961bf3ac0e8a60cfaceec705bd59Lennart Poettering if (r < 0)
74b2466e14a1961bf3ac0e8a60cfaceec705bd59Lennart Poettering log_emergency_errno(r, "Caught <%s>, waitpid() failed: %m", signal_to_string(sig));
74b2466e14a1961bf3ac0e8a60cfaceec705bd59Lennart Poettering else if (status.si_code != CLD_DUMPED)
74b2466e14a1961bf3ac0e8a60cfaceec705bd59Lennart Poettering log_emergency("Caught <%s>, core dump failed (child "PID_FMT", code=%s, status=%i/%s).",
74b2466e14a1961bf3ac0e8a60cfaceec705bd59Lennart Poettering signal_to_string(sig),
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering pid, sigchld_code_to_string(status.si_code),
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering status.si_status,
74b2466e14a1961bf3ac0e8a60cfaceec705bd59Lennart Poettering strna(status.si_code == CLD_EXITED
74b2466e14a1961bf3ac0e8a60cfaceec705bd59Lennart Poettering ? exit_status_to_string(status.si_status, EXIT_STATUS_FULL)
74b2466e14a1961bf3ac0e8a60cfaceec705bd59Lennart Poettering : signal_to_string(status.si_status)));
8bf52d3d17d364438191077d0750b8b80b5dc53aLennart Poettering else
8bf52d3d17d364438191077d0750b8b80b5dc53aLennart Poettering log_emergency("Caught <%s>, dumped core as pid "PID_FMT".", signal_to_string(sig), pid);
8bf52d3d17d364438191077d0750b8b80b5dc53aLennart Poettering }
8bf52d3d17d364438191077d0750b8b80b5dc53aLennart Poettering }
8bf52d3d17d364438191077d0750b8b80b5dc53aLennart Poettering
8bf52d3d17d364438191077d0750b8b80b5dc53aLennart Poettering if (arg_crash_chvt >= 0)
8bf52d3d17d364438191077d0750b8b80b5dc53aLennart Poettering (void) chvt(arg_crash_chvt);
8bf52d3d17d364438191077d0750b8b80b5dc53aLennart Poettering
8bf52d3d17d364438191077d0750b8b80b5dc53aLennart Poettering sa = (struct sigaction) {
8bf52d3d17d364438191077d0750b8b80b5dc53aLennart Poettering .sa_handler = SIG_IGN,
74b2466e14a1961bf3ac0e8a60cfaceec705bd59Lennart Poettering .sa_flags = SA_NOCLDSTOP|SA_NOCLDWAIT|SA_RESTART,
74b2466e14a1961bf3ac0e8a60cfaceec705bd59Lennart Poettering };
74b2466e14a1961bf3ac0e8a60cfaceec705bd59Lennart Poettering
74b2466e14a1961bf3ac0e8a60cfaceec705bd59Lennart Poettering /* Let the kernel reap children for us */
74b2466e14a1961bf3ac0e8a60cfaceec705bd59Lennart Poettering (void) sigaction(SIGCHLD, &sa, NULL);
74b2466e14a1961bf3ac0e8a60cfaceec705bd59Lennart Poettering
74b2466e14a1961bf3ac0e8a60cfaceec705bd59Lennart Poettering if (arg_crash_shell) {
74b2466e14a1961bf3ac0e8a60cfaceec705bd59Lennart Poettering log_notice("Executing crash shell in 10s...");
74b2466e14a1961bf3ac0e8a60cfaceec705bd59Lennart Poettering (void) sleep(10);
74b2466e14a1961bf3ac0e8a60cfaceec705bd59Lennart Poettering
74b2466e14a1961bf3ac0e8a60cfaceec705bd59Lennart Poettering pid = raw_clone(SIGCHLD, NULL);
74b2466e14a1961bf3ac0e8a60cfaceec705bd59Lennart Poettering if (pid < 0)
74b2466e14a1961bf3ac0e8a60cfaceec705bd59Lennart Poettering log_emergency_errno(errno, "Failed to fork off crash shell: %m");
74b2466e14a1961bf3ac0e8a60cfaceec705bd59Lennart Poettering else if (pid == 0) {
74b2466e14a1961bf3ac0e8a60cfaceec705bd59Lennart Poettering (void) setsid();
74b2466e14a1961bf3ac0e8a60cfaceec705bd59Lennart Poettering (void) make_console_stdio();
74b2466e14a1961bf3ac0e8a60cfaceec705bd59Lennart Poettering (void) execle("/bin/sh", "/bin/sh", NULL, environ);
74b2466e14a1961bf3ac0e8a60cfaceec705bd59Lennart Poettering
74b2466e14a1961bf3ac0e8a60cfaceec705bd59Lennart Poettering log_emergency_errno(errno, "execle() failed: %m");
74b2466e14a1961bf3ac0e8a60cfaceec705bd59Lennart Poettering _exit(EXIT_FAILURE);
74b2466e14a1961bf3ac0e8a60cfaceec705bd59Lennart Poettering } else {
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering log_info("Spawned crash shell as PID "PID_FMT".", pid);
9de3e3294065e8697ff10130b53f274319cdcf6fZbigniew Jędrzejewski-Szmek (void) wait_for_terminate(pid, NULL);
9c92ce6d67f88beb31dd6555d12ae3f632218a39Lennart Poettering }
9c92ce6d67f88beb31dd6555d12ae3f632218a39Lennart Poettering }
9c92ce6d67f88beb31dd6555d12ae3f632218a39Lennart Poettering
9c92ce6d67f88beb31dd6555d12ae3f632218a39Lennart Poettering freeze_or_reboot();
9c92ce6d67f88beb31dd6555d12ae3f632218a39Lennart Poettering}
9de3e3294065e8697ff10130b53f274319cdcf6fZbigniew Jędrzejewski-Szmek
9de3e3294065e8697ff10130b53f274319cdcf6fZbigniew Jędrzejewski-Szmekstatic void install_crash_handler(void) {
9de3e3294065e8697ff10130b53f274319cdcf6fZbigniew Jędrzejewski-Szmek static const struct sigaction sa = {
8ac4e9e1e54397f6d1745c2a7a806132418c7da2Lennart Poettering .sa_handler = crash,
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering .sa_flags = SA_NODEFER, /* So that we can raise the signal again from the signal handler */
9de3e3294065e8697ff10130b53f274319cdcf6fZbigniew Jędrzejewski-Szmek };
9c92ce6d67f88beb31dd6555d12ae3f632218a39Lennart Poettering int r;
9de3e3294065e8697ff10130b53f274319cdcf6fZbigniew Jędrzejewski-Szmek
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering /* We ignore the return value here, since, we don't mind if we
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering * cannot set up a crash handler */
9de3e3294065e8697ff10130b53f274319cdcf6fZbigniew Jędrzejewski-Szmek r = sigaction_many(&sa, SIGNALS_CRASH_HANDLER, -1);
9c92ce6d67f88beb31dd6555d12ae3f632218a39Lennart Poettering if (r < 0)
9de3e3294065e8697ff10130b53f274319cdcf6fZbigniew Jędrzejewski-Szmek log_debug_errno(r, "I had trouble setting up the crash handler, ignoring: %m");
9c92ce6d67f88beb31dd6555d12ae3f632218a39Lennart Poettering}
2e276efc7b0398a3086629a52970bdd4ab7252f9Zbigniew Jędrzejewski-Szmek
9de3e3294065e8697ff10130b53f274319cdcf6fZbigniew Jędrzejewski-Szmekstatic int console_setup(void) {
9c92ce6d67f88beb31dd6555d12ae3f632218a39Lennart Poettering _cleanup_close_ int tty_fd = -1;
9de3e3294065e8697ff10130b53f274319cdcf6fZbigniew Jędrzejewski-Szmek int r;
7e8e0422aeb16f2a09a40546c61df753d10029b6Lennart Poettering
7e8e0422aeb16f2a09a40546c61df753d10029b6Lennart Poettering tty_fd = open_terminal("/dev/console", O_WRONLY|O_NOCTTY|O_CLOEXEC);
9de3e3294065e8697ff10130b53f274319cdcf6fZbigniew Jędrzejewski-Szmek if (tty_fd < 0)
9c92ce6d67f88beb31dd6555d12ae3f632218a39Lennart Poettering return log_error_errno(tty_fd, "Failed to open /dev/console: %m");
9de3e3294065e8697ff10130b53f274319cdcf6fZbigniew Jędrzejewski-Szmek
946c70944ebdf428ffeb9991a7449edbd4011461Zbigniew Jędrzejewski-Szmek /* We don't want to force text mode. plymouth may be showing
9de3e3294065e8697ff10130b53f274319cdcf6fZbigniew Jędrzejewski-Szmek * pictures already from initrd. */
9c92ce6d67f88beb31dd6555d12ae3f632218a39Lennart Poettering r = reset_terminal_fd(tty_fd, false);
abf126a355e2f2b62b6c51ab3bb37895d1e3eee7Tom Gundersen if (r < 0)
abf126a355e2f2b62b6c51ab3bb37895d1e3eee7Tom Gundersen return log_error_errno(r, "Failed to reset /dev/console: %m");
abf126a355e2f2b62b6c51ab3bb37895d1e3eee7Tom Gundersen
abf126a355e2f2b62b6c51ab3bb37895d1e3eee7Tom Gundersen return 0;
42cc2eebb01056beb7acd3ecfe8e533558237f84Lennart Poettering}
42cc2eebb01056beb7acd3ecfe8e533558237f84Lennart Poettering
42cc2eebb01056beb7acd3ecfe8e533558237f84Lennart Poetteringstatic int parse_crash_chvt(const char *value) {
42cc2eebb01056beb7acd3ecfe8e533558237f84Lennart Poettering int b;
8db0d2f5c37e7e8f5bfce016cfdad7947a3ea939Zbigniew Jędrzejewski-Szmek
8db0d2f5c37e7e8f5bfce016cfdad7947a3ea939Zbigniew Jędrzejewski-Szmek if (safe_atoi(value, &arg_crash_chvt) >= 0)
8db0d2f5c37e7e8f5bfce016cfdad7947a3ea939Zbigniew Jędrzejewski-Szmek return 0;
8db0d2f5c37e7e8f5bfce016cfdad7947a3ea939Zbigniew Jędrzejewski-Szmek
151226ab4bf276d60d51864330a99f886b923697Zbigniew Jędrzejewski-Szmek b = parse_boolean(value);
151226ab4bf276d60d51864330a99f886b923697Zbigniew Jędrzejewski-Szmek if (b < 0)
151226ab4bf276d60d51864330a99f886b923697Zbigniew Jędrzejewski-Szmek return b;
151226ab4bf276d60d51864330a99f886b923697Zbigniew Jędrzejewski-Szmek
151226ab4bf276d60d51864330a99f886b923697Zbigniew Jędrzejewski-Szmek if (b > 0)
50f1e641a93cacfc693b0c3d300bee5df0c8c460Tom Gundersen arg_crash_chvt = 0; /* switch to where kmsg goes */
50f1e641a93cacfc693b0c3d300bee5df0c8c460Tom Gundersen else
50f1e641a93cacfc693b0c3d300bee5df0c8c460Tom Gundersen arg_crash_chvt = -1; /* turn off switching */
50f1e641a93cacfc693b0c3d300bee5df0c8c460Tom Gundersen
50f1e641a93cacfc693b0c3d300bee5df0c8c460Tom Gundersen return 0;
5d45a8808431987c370706d365fb0cc95cf03d52Tom Gundersen}
5d45a8808431987c370706d365fb0cc95cf03d52Tom Gundersen
5d45a8808431987c370706d365fb0cc95cf03d52Tom Gundersenstatic int set_machine_id(const char *m) {
5d45a8808431987c370706d365fb0cc95cf03d52Tom Gundersen
5d45a8808431987c370706d365fb0cc95cf03d52Tom Gundersen if (sd_id128_from_string(m, &arg_machine_id) < 0)
5d45a8808431987c370706d365fb0cc95cf03d52Tom Gundersen return -EINVAL;
0dae31d468b1a0e22d98921f7b0dbd92fd217167Zbigniew Jędrzejewski-Szmek
9de3e3294065e8697ff10130b53f274319cdcf6fZbigniew Jędrzejewski-Szmek if (sd_id128_is_null(arg_machine_id))
9de3e3294065e8697ff10130b53f274319cdcf6fZbigniew Jędrzejewski-Szmek return -EINVAL;
9de3e3294065e8697ff10130b53f274319cdcf6fZbigniew Jędrzejewski-Szmek
9c92ce6d67f88beb31dd6555d12ae3f632218a39Lennart Poettering return 0;
9de3e3294065e8697ff10130b53f274319cdcf6fZbigniew Jędrzejewski-Szmek}
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering
9de3e3294065e8697ff10130b53f274319cdcf6fZbigniew Jędrzejewski-Szmekstatic int parse_proc_cmdline_item(const char *key, const char *value) {
322345fdb9865ef2477fba8e4bdde0e1183ef505Lennart Poettering
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering int r;
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering
322345fdb9865ef2477fba8e4bdde0e1183ef505Lennart Poettering assert(key);
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering
322345fdb9865ef2477fba8e4bdde0e1183ef505Lennart Poettering if (streq(key, "systemd.unit") && value) {
322345fdb9865ef2477fba8e4bdde0e1183ef505Lennart Poettering
322345fdb9865ef2477fba8e4bdde0e1183ef505Lennart Poettering if (!in_initrd())
322345fdb9865ef2477fba8e4bdde0e1183ef505Lennart Poettering return free_and_strdup(&arg_default_unit, value);
623a4c97b9175f95c4b1c6fc34e36c56f1e4ddbfLennart Poettering
623a4c97b9175f95c4b1c6fc34e36c56f1e4ddbfLennart Poettering } else if (streq(key, "rd.systemd.unit") && value) {
623a4c97b9175f95c4b1c6fc34e36c56f1e4ddbfLennart Poettering
623a4c97b9175f95c4b1c6fc34e36c56f1e4ddbfLennart Poettering if (in_initrd())
623a4c97b9175f95c4b1c6fc34e36c56f1e4ddbfLennart Poettering return free_and_strdup(&arg_default_unit, value);
623a4c97b9175f95c4b1c6fc34e36c56f1e4ddbfLennart Poettering
623a4c97b9175f95c4b1c6fc34e36c56f1e4ddbfLennart Poettering } else if (streq(key, "systemd.dump_core") && value) {
623a4c97b9175f95c4b1c6fc34e36c56f1e4ddbfLennart Poettering
623a4c97b9175f95c4b1c6fc34e36c56f1e4ddbfLennart Poettering r = parse_boolean(value);
623a4c97b9175f95c4b1c6fc34e36c56f1e4ddbfLennart Poettering if (r < 0)
623a4c97b9175f95c4b1c6fc34e36c56f1e4ddbfLennart Poettering log_warning("Failed to parse dump core switch %s. Ignoring.", value);
623a4c97b9175f95c4b1c6fc34e36c56f1e4ddbfLennart Poettering else
623a4c97b9175f95c4b1c6fc34e36c56f1e4ddbfLennart Poettering arg_dump_core = r;
623a4c97b9175f95c4b1c6fc34e36c56f1e4ddbfLennart Poettering
623a4c97b9175f95c4b1c6fc34e36c56f1e4ddbfLennart Poettering } else if (streq(key, "systemd.crash_chvt") && value) {
623a4c97b9175f95c4b1c6fc34e36c56f1e4ddbfLennart Poettering
623a4c97b9175f95c4b1c6fc34e36c56f1e4ddbfLennart Poettering if (parse_crash_chvt(value) < 0)
623a4c97b9175f95c4b1c6fc34e36c56f1e4ddbfLennart Poettering log_warning("Failed to parse crash chvt switch %s. Ignoring.", value);
623a4c97b9175f95c4b1c6fc34e36c56f1e4ddbfLennart Poettering
623a4c97b9175f95c4b1c6fc34e36c56f1e4ddbfLennart Poettering } else if (streq(key, "systemd.crash_shell") && value) {
623a4c97b9175f95c4b1c6fc34e36c56f1e4ddbfLennart Poettering
623a4c97b9175f95c4b1c6fc34e36c56f1e4ddbfLennart Poettering r = parse_boolean(value);
623a4c97b9175f95c4b1c6fc34e36c56f1e4ddbfLennart Poettering if (r < 0)
623a4c97b9175f95c4b1c6fc34e36c56f1e4ddbfLennart Poettering log_warning("Failed to parse crash shell switch %s. Ignoring.", value);
623a4c97b9175f95c4b1c6fc34e36c56f1e4ddbfLennart Poettering else
623a4c97b9175f95c4b1c6fc34e36c56f1e4ddbfLennart Poettering arg_crash_shell = r;
623a4c97b9175f95c4b1c6fc34e36c56f1e4ddbfLennart Poettering
623a4c97b9175f95c4b1c6fc34e36c56f1e4ddbfLennart Poettering } else if (streq(key, "systemd.crash_reboot") && value) {
623a4c97b9175f95c4b1c6fc34e36c56f1e4ddbfLennart Poettering
623a4c97b9175f95c4b1c6fc34e36c56f1e4ddbfLennart Poettering r = parse_boolean(value);
623a4c97b9175f95c4b1c6fc34e36c56f1e4ddbfLennart Poettering if (r < 0)
623a4c97b9175f95c4b1c6fc34e36c56f1e4ddbfLennart Poettering log_warning("Failed to parse crash reboot switch %s. Ignoring.", value);
623a4c97b9175f95c4b1c6fc34e36c56f1e4ddbfLennart Poettering else
623a4c97b9175f95c4b1c6fc34e36c56f1e4ddbfLennart Poettering arg_crash_reboot = r;
322345fdb9865ef2477fba8e4bdde0e1183ef505Lennart Poettering
322345fdb9865ef2477fba8e4bdde0e1183ef505Lennart Poettering } else if (streq(key, "systemd.confirm_spawn") && value) {
322345fdb9865ef2477fba8e4bdde0e1183ef505Lennart Poettering
322345fdb9865ef2477fba8e4bdde0e1183ef505Lennart Poettering r = parse_boolean(value);
322345fdb9865ef2477fba8e4bdde0e1183ef505Lennart Poettering if (r < 0)
322345fdb9865ef2477fba8e4bdde0e1183ef505Lennart Poettering log_warning("Failed to parse confirm spawn switch %s. Ignoring.", value);
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering else
322345fdb9865ef2477fba8e4bdde0e1183ef505Lennart Poettering arg_confirm_spawn = r;
322345fdb9865ef2477fba8e4bdde0e1183ef505Lennart Poettering
322345fdb9865ef2477fba8e4bdde0e1183ef505Lennart Poettering } else if (streq(key, "systemd.show_status") && value) {
fd0b4602f6332c3f1660eb208c8f5c719709a009Lennart Poettering
fd0b4602f6332c3f1660eb208c8f5c719709a009Lennart Poettering r = parse_show_status(value, &arg_show_status);
fd0b4602f6332c3f1660eb208c8f5c719709a009Lennart Poettering if (r < 0)
fd0b4602f6332c3f1660eb208c8f5c719709a009Lennart Poettering log_warning("Failed to parse show status switch %s. Ignoring.", value);
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering
9c92ce6d67f88beb31dd6555d12ae3f632218a39Lennart Poettering } else if (streq(key, "systemd.default_standard_output") && value) {
9c92ce6d67f88beb31dd6555d12ae3f632218a39Lennart Poettering
9c92ce6d67f88beb31dd6555d12ae3f632218a39Lennart Poettering r = exec_output_from_string(value);
9c92ce6d67f88beb31dd6555d12ae3f632218a39Lennart Poettering if (r < 0)
9c92ce6d67f88beb31dd6555d12ae3f632218a39Lennart Poettering log_warning("Failed to parse default standard output switch %s. Ignoring.", value);
9c92ce6d67f88beb31dd6555d12ae3f632218a39Lennart Poettering else
9c92ce6d67f88beb31dd6555d12ae3f632218a39Lennart Poettering arg_default_std_output = r;
9c92ce6d67f88beb31dd6555d12ae3f632218a39Lennart Poettering
9c92ce6d67f88beb31dd6555d12ae3f632218a39Lennart Poettering } else if (streq(key, "systemd.default_standard_error") && value) {
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering r = exec_output_from_string(value);
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering if (r < 0)
8ac4e9e1e54397f6d1745c2a7a806132418c7da2Lennart Poettering log_warning("Failed to parse default standard error switch %s. Ignoring.", value);
322345fdb9865ef2477fba8e4bdde0e1183ef505Lennart Poettering else
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering arg_default_std_error = r;
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering } else if (streq(key, "systemd.setenv") && value) {
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering if (env_assignment_is_valid(value)) {
9de3e3294065e8697ff10130b53f274319cdcf6fZbigniew Jędrzejewski-Szmek char **env;
0f84a72e3c0f58d71cff2121e6df1611eaf9c9eaDavid Herrmann
0f84a72e3c0f58d71cff2121e6df1611eaf9c9eaDavid Herrmann env = strv_env_set(arg_default_environment, value);
2e276efc7b0398a3086629a52970bdd4ab7252f9Zbigniew Jędrzejewski-Szmek if (env)
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering arg_default_environment = env;
322345fdb9865ef2477fba8e4bdde0e1183ef505Lennart Poettering else
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering log_warning_errno(ENOMEM, "Setting environment variable '%s' failed, ignoring: %m", value);
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering } else
322345fdb9865ef2477fba8e4bdde0e1183ef505Lennart Poettering log_warning("Environment variable name '%s' is not valid. Ignoring.", value);
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering } else if (streq(key, "systemd.machine_id") && value) {
7e8e0422aeb16f2a09a40546c61df753d10029b6Lennart Poettering
7e8e0422aeb16f2a09a40546c61df753d10029b6Lennart Poettering r = set_machine_id(value);
7e8e0422aeb16f2a09a40546c61df753d10029b6Lennart Poettering if (r < 0)
7e8e0422aeb16f2a09a40546c61df753d10029b6Lennart Poettering log_warning("MachineID '%s' is not valid. Ignoring.", value);
7e8e0422aeb16f2a09a40546c61df753d10029b6Lennart Poettering
7e8e0422aeb16f2a09a40546c61df753d10029b6Lennart Poettering } else if (streq(key, "quiet") && !value) {
7e8e0422aeb16f2a09a40546c61df753d10029b6Lennart Poettering
7e8e0422aeb16f2a09a40546c61df753d10029b6Lennart Poettering if (arg_show_status == _SHOW_STATUS_UNSET)
7e8e0422aeb16f2a09a40546c61df753d10029b6Lennart Poettering arg_show_status = SHOW_STATUS_AUTO;
7e8e0422aeb16f2a09a40546c61df753d10029b6Lennart Poettering
7e8e0422aeb16f2a09a40546c61df753d10029b6Lennart Poettering } else if (streq(key, "debug") && !value) {
7e8e0422aeb16f2a09a40546c61df753d10029b6Lennart Poettering
9c92ce6d67f88beb31dd6555d12ae3f632218a39Lennart Poettering /* Note that log_parse_environment() handles 'debug'
946c70944ebdf428ffeb9991a7449edbd4011461Zbigniew Jędrzejewski-Szmek * too, and sets the log level to LOG_DEBUG. */
946c70944ebdf428ffeb9991a7449edbd4011461Zbigniew Jędrzejewski-Szmek
946c70944ebdf428ffeb9991a7449edbd4011461Zbigniew Jędrzejewski-Szmek if (detect_container() > 0)
946c70944ebdf428ffeb9991a7449edbd4011461Zbigniew Jędrzejewski-Szmek log_set_target(LOG_TARGET_CONSOLE);
946c70944ebdf428ffeb9991a7449edbd4011461Zbigniew Jędrzejewski-Szmek
946c70944ebdf428ffeb9991a7449edbd4011461Zbigniew Jędrzejewski-Szmek } else if (!in_initrd() && !value) {
0dae31d468b1a0e22d98921f7b0dbd92fd217167Zbigniew Jędrzejewski-Szmek const char *target;
0dae31d468b1a0e22d98921f7b0dbd92fd217167Zbigniew Jędrzejewski-Szmek
0dae31d468b1a0e22d98921f7b0dbd92fd217167Zbigniew Jędrzejewski-Szmek /* SysV compatibility */
0dae31d468b1a0e22d98921f7b0dbd92fd217167Zbigniew Jędrzejewski-Szmek target = runlevel_to_target(key);
0dae31d468b1a0e22d98921f7b0dbd92fd217167Zbigniew Jędrzejewski-Szmek if (target)
0dae31d468b1a0e22d98921f7b0dbd92fd217167Zbigniew Jędrzejewski-Szmek return free_and_strdup(&arg_default_unit, target);
0dae31d468b1a0e22d98921f7b0dbd92fd217167Zbigniew Jędrzejewski-Szmek }
0dae31d468b1a0e22d98921f7b0dbd92fd217167Zbigniew Jędrzejewski-Szmek
0dae31d468b1a0e22d98921f7b0dbd92fd217167Zbigniew Jędrzejewski-Szmek return 0;
0dae31d468b1a0e22d98921f7b0dbd92fd217167Zbigniew Jędrzejewski-Szmek}
abf126a355e2f2b62b6c51ab3bb37895d1e3eee7Tom Gundersen
abf126a355e2f2b62b6c51ab3bb37895d1e3eee7Tom Gundersen#define DEFINE_SETTER(name, func, descr) \
abf126a355e2f2b62b6c51ab3bb37895d1e3eee7Tom Gundersen static int name(const char *unit, \
abf126a355e2f2b62b6c51ab3bb37895d1e3eee7Tom Gundersen const char *filename, \
abf126a355e2f2b62b6c51ab3bb37895d1e3eee7Tom Gundersen unsigned line, \
abf126a355e2f2b62b6c51ab3bb37895d1e3eee7Tom Gundersen const char *section, \
abf126a355e2f2b62b6c51ab3bb37895d1e3eee7Tom Gundersen unsigned section_line, \
42cc2eebb01056beb7acd3ecfe8e533558237f84Lennart Poettering const char *lvalue, \
42cc2eebb01056beb7acd3ecfe8e533558237f84Lennart Poettering int ltype, \
42cc2eebb01056beb7acd3ecfe8e533558237f84Lennart Poettering const char *rvalue, \
42cc2eebb01056beb7acd3ecfe8e533558237f84Lennart Poettering void *data, \
42cc2eebb01056beb7acd3ecfe8e533558237f84Lennart Poettering void *userdata) { \
42cc2eebb01056beb7acd3ecfe8e533558237f84Lennart Poettering \
8db0d2f5c37e7e8f5bfce016cfdad7947a3ea939Zbigniew Jędrzejewski-Szmek int r; \
8db0d2f5c37e7e8f5bfce016cfdad7947a3ea939Zbigniew Jędrzejewski-Szmek \
8db0d2f5c37e7e8f5bfce016cfdad7947a3ea939Zbigniew Jędrzejewski-Szmek assert(filename); \
8db0d2f5c37e7e8f5bfce016cfdad7947a3ea939Zbigniew Jędrzejewski-Szmek assert(lvalue); \
8db0d2f5c37e7e8f5bfce016cfdad7947a3ea939Zbigniew Jędrzejewski-Szmek assert(rvalue); \
8db0d2f5c37e7e8f5bfce016cfdad7947a3ea939Zbigniew Jędrzejewski-Szmek \
8db0d2f5c37e7e8f5bfce016cfdad7947a3ea939Zbigniew Jędrzejewski-Szmek r = func(rvalue); \
151226ab4bf276d60d51864330a99f886b923697Zbigniew Jędrzejewski-Szmek if (r < 0) \
151226ab4bf276d60d51864330a99f886b923697Zbigniew Jędrzejewski-Szmek log_syntax(unit, LOG_ERR, filename, line, r, \
03664a62914782dbd8f069bbcf8a0c8ca1df7010Lukas Nykryn "Invalid " descr "'%s': %m", \
03664a62914782dbd8f069bbcf8a0c8ca1df7010Lukas Nykryn rvalue); \
03664a62914782dbd8f069bbcf8a0c8ca1df7010Lukas Nykryn \
03664a62914782dbd8f069bbcf8a0c8ca1df7010Lukas Nykryn return 0; \
03664a62914782dbd8f069bbcf8a0c8ca1df7010Lukas Nykryn }
03664a62914782dbd8f069bbcf8a0c8ca1df7010Lukas Nykryn
03664a62914782dbd8f069bbcf8a0c8ca1df7010Lukas NykrynDEFINE_SETTER(config_parse_level2, log_set_max_level_from_string, "log level")
151226ab4bf276d60d51864330a99f886b923697Zbigniew Jędrzejewski-SzmekDEFINE_SETTER(config_parse_target, log_set_target_from_string, "target")
151226ab4bf276d60d51864330a99f886b923697Zbigniew Jędrzejewski-SzmekDEFINE_SETTER(config_parse_color, log_show_color_from_string, "color" )
151226ab4bf276d60d51864330a99f886b923697Zbigniew Jędrzejewski-SzmekDEFINE_SETTER(config_parse_location, log_show_location_from_string, "location")
151226ab4bf276d60d51864330a99f886b923697Zbigniew Jędrzejewski-Szmek
151226ab4bf276d60d51864330a99f886b923697Zbigniew Jędrzejewski-Szmekstatic int config_parse_cpu_affinity2(
151226ab4bf276d60d51864330a99f886b923697Zbigniew Jędrzejewski-Szmek const char *unit,
50f1e641a93cacfc693b0c3d300bee5df0c8c460Tom Gundersen const char *filename,
50f1e641a93cacfc693b0c3d300bee5df0c8c460Tom Gundersen unsigned line,
50f1e641a93cacfc693b0c3d300bee5df0c8c460Tom Gundersen const char *section,
50f1e641a93cacfc693b0c3d300bee5df0c8c460Tom Gundersen unsigned section_line,
5d45a8808431987c370706d365fb0cc95cf03d52Tom Gundersen const char *lvalue,
5d45a8808431987c370706d365fb0cc95cf03d52Tom Gundersen int ltype,
5d45a8808431987c370706d365fb0cc95cf03d52Tom Gundersen const char *rvalue,
5d45a8808431987c370706d365fb0cc95cf03d52Tom Gundersen void *data,
5d45a8808431987c370706d365fb0cc95cf03d52Tom Gundersen void *userdata) {
5d45a8808431987c370706d365fb0cc95cf03d52Tom Gundersen
5d45a8808431987c370706d365fb0cc95cf03d52Tom Gundersen _cleanup_cpu_free_ cpu_set_t *c = NULL;
5d45a8808431987c370706d365fb0cc95cf03d52Tom Gundersen int ncpus;
5d45a8808431987c370706d365fb0cc95cf03d52Tom Gundersen
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering ncpus = parse_cpu_set_and_warn(rvalue, &c, unit, filename, line, lvalue);
322345fdb9865ef2477fba8e4bdde0e1183ef505Lennart Poettering if (ncpus < 0)
322345fdb9865ef2477fba8e4bdde0e1183ef505Lennart Poettering return ncpus;
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering
322345fdb9865ef2477fba8e4bdde0e1183ef505Lennart Poettering if (sched_setaffinity(0, CPU_ALLOC_SIZE(ncpus), c) < 0)
322345fdb9865ef2477fba8e4bdde0e1183ef505Lennart Poettering log_warning("Failed to set CPU affinity: %m");
0dae31d468b1a0e22d98921f7b0dbd92fd217167Zbigniew Jędrzejewski-Szmek
0dae31d468b1a0e22d98921f7b0dbd92fd217167Zbigniew Jędrzejewski-Szmek return 0;
0dae31d468b1a0e22d98921f7b0dbd92fd217167Zbigniew Jędrzejewski-Szmek}
0dae31d468b1a0e22d98921f7b0dbd92fd217167Zbigniew Jędrzejewski-Szmek
0dae31d468b1a0e22d98921f7b0dbd92fd217167Zbigniew Jędrzejewski-Szmekstatic int config_parse_show_status(
0dae31d468b1a0e22d98921f7b0dbd92fd217167Zbigniew Jędrzejewski-Szmek const char* unit,
0dae31d468b1a0e22d98921f7b0dbd92fd217167Zbigniew Jędrzejewski-Szmek const char *filename,
0dae31d468b1a0e22d98921f7b0dbd92fd217167Zbigniew Jędrzejewski-Szmek unsigned line,
0dae31d468b1a0e22d98921f7b0dbd92fd217167Zbigniew Jędrzejewski-Szmek const char *section,
0dae31d468b1a0e22d98921f7b0dbd92fd217167Zbigniew Jędrzejewski-Szmek unsigned section_line,
0dae31d468b1a0e22d98921f7b0dbd92fd217167Zbigniew Jędrzejewski-Szmek const char *lvalue,
0dae31d468b1a0e22d98921f7b0dbd92fd217167Zbigniew Jędrzejewski-Szmek int ltype,
0dae31d468b1a0e22d98921f7b0dbd92fd217167Zbigniew Jędrzejewski-Szmek const char *rvalue,
0dae31d468b1a0e22d98921f7b0dbd92fd217167Zbigniew Jędrzejewski-Szmek void *data,
0dae31d468b1a0e22d98921f7b0dbd92fd217167Zbigniew Jędrzejewski-Szmek void *userdata) {
0dae31d468b1a0e22d98921f7b0dbd92fd217167Zbigniew Jędrzejewski-Szmek
0dae31d468b1a0e22d98921f7b0dbd92fd217167Zbigniew Jędrzejewski-Szmek int k;
0dae31d468b1a0e22d98921f7b0dbd92fd217167Zbigniew Jędrzejewski-Szmek ShowStatus *b = data;
0dae31d468b1a0e22d98921f7b0dbd92fd217167Zbigniew Jędrzejewski-Szmek
0dae31d468b1a0e22d98921f7b0dbd92fd217167Zbigniew Jędrzejewski-Szmek assert(filename);
0dae31d468b1a0e22d98921f7b0dbd92fd217167Zbigniew Jędrzejewski-Szmek assert(lvalue);
0dae31d468b1a0e22d98921f7b0dbd92fd217167Zbigniew Jędrzejewski-Szmek assert(rvalue);
0dae31d468b1a0e22d98921f7b0dbd92fd217167Zbigniew Jędrzejewski-Szmek assert(data);
0dae31d468b1a0e22d98921f7b0dbd92fd217167Zbigniew Jędrzejewski-Szmek
0dae31d468b1a0e22d98921f7b0dbd92fd217167Zbigniew Jędrzejewski-Szmek k = parse_show_status(rvalue, b);
0dae31d468b1a0e22d98921f7b0dbd92fd217167Zbigniew Jędrzejewski-Szmek if (k < 0) {
0dae31d468b1a0e22d98921f7b0dbd92fd217167Zbigniew Jędrzejewski-Szmek log_syntax(unit, LOG_ERR, filename, line, k, "Failed to parse show status setting, ignoring: %s", rvalue);
0dae31d468b1a0e22d98921f7b0dbd92fd217167Zbigniew Jędrzejewski-Szmek return 0;
0dae31d468b1a0e22d98921f7b0dbd92fd217167Zbigniew Jędrzejewski-Szmek }
0dae31d468b1a0e22d98921f7b0dbd92fd217167Zbigniew Jędrzejewski-Szmek
0dae31d468b1a0e22d98921f7b0dbd92fd217167Zbigniew Jędrzejewski-Szmek return 0;
7c6423e19136a7b7b6ef3fe06b94822e582dda27Tom Gundersen}
7c6423e19136a7b7b6ef3fe06b94822e582dda27Tom Gundersen
7c6423e19136a7b7b6ef3fe06b94822e582dda27Tom Gundersenstatic int config_parse_crash_chvt(
7c6423e19136a7b7b6ef3fe06b94822e582dda27Tom Gundersen const char* unit,
7c6423e19136a7b7b6ef3fe06b94822e582dda27Tom Gundersen const char *filename,
7c6423e19136a7b7b6ef3fe06b94822e582dda27Tom Gundersen unsigned line,
7c6423e19136a7b7b6ef3fe06b94822e582dda27Tom Gundersen const char *section,
7c6423e19136a7b7b6ef3fe06b94822e582dda27Tom Gundersen unsigned section_line,
7c6423e19136a7b7b6ef3fe06b94822e582dda27Tom Gundersen const char *lvalue,
7c6423e19136a7b7b6ef3fe06b94822e582dda27Tom Gundersen int ltype,
7c6423e19136a7b7b6ef3fe06b94822e582dda27Tom Gundersen const char *rvalue,
7c6423e19136a7b7b6ef3fe06b94822e582dda27Tom Gundersen void *data,
7c6423e19136a7b7b6ef3fe06b94822e582dda27Tom Gundersen void *userdata) {
7c6423e19136a7b7b6ef3fe06b94822e582dda27Tom Gundersen
7c6423e19136a7b7b6ef3fe06b94822e582dda27Tom Gundersen int r;
50f1e641a93cacfc693b0c3d300bee5df0c8c460Tom Gundersen
50f1e641a93cacfc693b0c3d300bee5df0c8c460Tom Gundersen assert(filename);
50f1e641a93cacfc693b0c3d300bee5df0c8c460Tom Gundersen assert(lvalue);
cb57dd41595adddb08095298bb1ed258c8ea4877Tom Gundersen assert(rvalue);
50f1e641a93cacfc693b0c3d300bee5df0c8c460Tom Gundersen
50f1e641a93cacfc693b0c3d300bee5df0c8c460Tom Gundersen r = parse_crash_chvt(rvalue);
50f1e641a93cacfc693b0c3d300bee5df0c8c460Tom Gundersen if (r < 0) {
cb57dd41595adddb08095298bb1ed258c8ea4877Tom Gundersen log_syntax(unit, LOG_ERR, filename, line, r, "Failed to parse CrashChangeVT= setting, ignoring: %s", rvalue);
50f1e641a93cacfc693b0c3d300bee5df0c8c460Tom Gundersen return 0;
50f1e641a93cacfc693b0c3d300bee5df0c8c460Tom Gundersen }
50f1e641a93cacfc693b0c3d300bee5df0c8c460Tom Gundersen
50f1e641a93cacfc693b0c3d300bee5df0c8c460Tom Gundersen return 0;
50f1e641a93cacfc693b0c3d300bee5df0c8c460Tom Gundersen}
50f1e641a93cacfc693b0c3d300bee5df0c8c460Tom Gundersen
50f1e641a93cacfc693b0c3d300bee5df0c8c460Tom Gundersenstatic int config_parse_join_controllers(const char *unit,
50f1e641a93cacfc693b0c3d300bee5df0c8c460Tom Gundersen const char *filename,
50f1e641a93cacfc693b0c3d300bee5df0c8c460Tom Gundersen unsigned line,
50f1e641a93cacfc693b0c3d300bee5df0c8c460Tom Gundersen const char *section,
50f1e641a93cacfc693b0c3d300bee5df0c8c460Tom Gundersen unsigned section_line,
50f1e641a93cacfc693b0c3d300bee5df0c8c460Tom Gundersen const char *lvalue,
50f1e641a93cacfc693b0c3d300bee5df0c8c460Tom Gundersen int ltype,
50f1e641a93cacfc693b0c3d300bee5df0c8c460Tom Gundersen const char *rvalue,
50f1e641a93cacfc693b0c3d300bee5df0c8c460Tom Gundersen void *data,
50f1e641a93cacfc693b0c3d300bee5df0c8c460Tom Gundersen void *userdata) {
50f1e641a93cacfc693b0c3d300bee5df0c8c460Tom Gundersen
50f1e641a93cacfc693b0c3d300bee5df0c8c460Tom Gundersen const char *whole_rvalue = rvalue;
50f1e641a93cacfc693b0c3d300bee5df0c8c460Tom Gundersen unsigned n = 0;
50f1e641a93cacfc693b0c3d300bee5df0c8c460Tom Gundersen
50f1e641a93cacfc693b0c3d300bee5df0c8c460Tom Gundersen assert(filename);
50f1e641a93cacfc693b0c3d300bee5df0c8c460Tom Gundersen assert(lvalue);
50f1e641a93cacfc693b0c3d300bee5df0c8c460Tom Gundersen assert(rvalue);
50f1e641a93cacfc693b0c3d300bee5df0c8c460Tom Gundersen
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering arg_join_controllers = strv_free_free(arg_join_controllers);
8db0d2f5c37e7e8f5bfce016cfdad7947a3ea939Zbigniew Jędrzejewski-Szmek
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering for (;;) {
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering _cleanup_free_ char *word = NULL;
322345fdb9865ef2477fba8e4bdde0e1183ef505Lennart Poettering char **l;
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering int r;
322345fdb9865ef2477fba8e4bdde0e1183ef505Lennart Poettering
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering r = extract_first_word(&rvalue, &word, WHITESPACE, EXTRACT_QUOTES);
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering if (r < 0) {
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering log_syntax(unit, LOG_ERR, filename, line, r, "Invalid value for %s: %s", lvalue, whole_rvalue);
322345fdb9865ef2477fba8e4bdde0e1183ef505Lennart Poettering return r;
0dae31d468b1a0e22d98921f7b0dbd92fd217167Zbigniew Jędrzejewski-Szmek }
322345fdb9865ef2477fba8e4bdde0e1183ef505Lennart Poettering if (r == 0)
9c92ce6d67f88beb31dd6555d12ae3f632218a39Lennart Poettering break;
9c92ce6d67f88beb31dd6555d12ae3f632218a39Lennart Poettering
9c92ce6d67f88beb31dd6555d12ae3f632218a39Lennart Poettering l = strv_split(word, ",");
9c92ce6d67f88beb31dd6555d12ae3f632218a39Lennart Poettering if (!l)
9c92ce6d67f88beb31dd6555d12ae3f632218a39Lennart Poettering return log_oom();
9c92ce6d67f88beb31dd6555d12ae3f632218a39Lennart Poettering strv_uniq(l);
9c92ce6d67f88beb31dd6555d12ae3f632218a39Lennart Poettering
9c92ce6d67f88beb31dd6555d12ae3f632218a39Lennart Poettering if (strv_length(l) <= 1) {
9c92ce6d67f88beb31dd6555d12ae3f632218a39Lennart Poettering strv_free(l);
9c92ce6d67f88beb31dd6555d12ae3f632218a39Lennart Poettering continue;
9c92ce6d67f88beb31dd6555d12ae3f632218a39Lennart Poettering }
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering if (!arg_join_controllers) {
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering arg_join_controllers = new(char**, 2);
8ac4e9e1e54397f6d1745c2a7a806132418c7da2Lennart Poettering if (!arg_join_controllers) {
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering strv_free(l);
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering return log_oom();
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering }
322345fdb9865ef2477fba8e4bdde0e1183ef505Lennart Poettering
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering arg_join_controllers[0] = l;
322345fdb9865ef2477fba8e4bdde0e1183ef505Lennart Poettering arg_join_controllers[1] = NULL;
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering n = 1;
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering } else {
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering char ***a;
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering char ***t;
322345fdb9865ef2477fba8e4bdde0e1183ef505Lennart Poettering
9de3e3294065e8697ff10130b53f274319cdcf6fZbigniew Jędrzejewski-Szmek t = new0(char**, n+2);
8db0d2f5c37e7e8f5bfce016cfdad7947a3ea939Zbigniew Jędrzejewski-Szmek if (!t) {
2e276efc7b0398a3086629a52970bdd4ab7252f9Zbigniew Jędrzejewski-Szmek strv_free(l);
2e276efc7b0398a3086629a52970bdd4ab7252f9Zbigniew Jędrzejewski-Szmek return log_oom();
2e276efc7b0398a3086629a52970bdd4ab7252f9Zbigniew Jędrzejewski-Szmek }
2e276efc7b0398a3086629a52970bdd4ab7252f9Zbigniew Jędrzejewski-Szmek
2e276efc7b0398a3086629a52970bdd4ab7252f9Zbigniew Jędrzejewski-Szmek n = 0;
2e276efc7b0398a3086629a52970bdd4ab7252f9Zbigniew Jędrzejewski-Szmek
2e276efc7b0398a3086629a52970bdd4ab7252f9Zbigniew Jędrzejewski-Szmek for (a = arg_join_controllers; *a; a++) {
2e276efc7b0398a3086629a52970bdd4ab7252f9Zbigniew Jędrzejewski-Szmek
2e276efc7b0398a3086629a52970bdd4ab7252f9Zbigniew Jędrzejewski-Szmek if (strv_overlap(*a, l)) {
2e276efc7b0398a3086629a52970bdd4ab7252f9Zbigniew Jędrzejewski-Szmek if (strv_extend_strv(&l, *a, false) < 0) {
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering strv_free(l);
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering strv_free_free(t);
322345fdb9865ef2477fba8e4bdde0e1183ef505Lennart Poettering return log_oom();
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering }
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering } else {
322345fdb9865ef2477fba8e4bdde0e1183ef505Lennart Poettering char **c;
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering c = strv_copy(*a);
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering if (!c) {
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering strv_free(l);
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering strv_free_free(t);
322345fdb9865ef2477fba8e4bdde0e1183ef505Lennart Poettering return log_oom();
8db0d2f5c37e7e8f5bfce016cfdad7947a3ea939Zbigniew Jędrzejewski-Szmek }
8db0d2f5c37e7e8f5bfce016cfdad7947a3ea939Zbigniew Jędrzejewski-Szmek
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering t[n++] = c;
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering }
322345fdb9865ef2477fba8e4bdde0e1183ef505Lennart Poettering }
8db0d2f5c37e7e8f5bfce016cfdad7947a3ea939Zbigniew Jędrzejewski-Szmek
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering t[n++] = strv_uniq(l);
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering strv_free_free(arg_join_controllers);
322345fdb9865ef2477fba8e4bdde0e1183ef505Lennart Poettering arg_join_controllers = t;
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering }
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering }
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering if (!isempty(rvalue))
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering log_syntax(unit, LOG_ERR, filename, line, 0, "Trailing garbage, ignoring.");
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering return 0;
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering}
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poetteringstatic int parse_config_file(void) {
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering const ConfigTableItem items[] = {
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering { "Manager", "LogLevel", config_parse_level2, 0, NULL },
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering { "Manager", "LogTarget", config_parse_target, 0, NULL },
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering { "Manager", "LogColor", config_parse_color, 0, NULL },
946c70944ebdf428ffeb9991a7449edbd4011461Zbigniew Jędrzejewski-Szmek { "Manager", "LogLocation", config_parse_location, 0, NULL },
946c70944ebdf428ffeb9991a7449edbd4011461Zbigniew Jędrzejewski-Szmek { "Manager", "DumpCore", config_parse_bool, 0, &arg_dump_core },
946c70944ebdf428ffeb9991a7449edbd4011461Zbigniew Jędrzejewski-Szmek { "Manager", "CrashChVT", /* legacy */ config_parse_crash_chvt, 0, NULL },
946c70944ebdf428ffeb9991a7449edbd4011461Zbigniew Jędrzejewski-Szmek { "Manager", "CrashChangeVT", config_parse_crash_chvt, 0, NULL },
946c70944ebdf428ffeb9991a7449edbd4011461Zbigniew Jędrzejewski-Szmek { "Manager", "CrashShell", config_parse_bool, 0, &arg_crash_shell },
946c70944ebdf428ffeb9991a7449edbd4011461Zbigniew Jędrzejewski-Szmek { "Manager", "CrashReboot", config_parse_bool, 0, &arg_crash_reboot },
946c70944ebdf428ffeb9991a7449edbd4011461Zbigniew Jędrzejewski-Szmek { "Manager", "ShowStatus", config_parse_show_status, 0, &arg_show_status },
946c70944ebdf428ffeb9991a7449edbd4011461Zbigniew Jędrzejewski-Szmek { "Manager", "CPUAffinity", config_parse_cpu_affinity2, 0, NULL },
946c70944ebdf428ffeb9991a7449edbd4011461Zbigniew Jędrzejewski-Szmek { "Manager", "JoinControllers", config_parse_join_controllers, 0, &arg_join_controllers },
8db0d2f5c37e7e8f5bfce016cfdad7947a3ea939Zbigniew Jędrzejewski-Szmek { "Manager", "RuntimeWatchdogSec", config_parse_sec, 0, &arg_runtime_watchdog },
0dae31d468b1a0e22d98921f7b0dbd92fd217167Zbigniew Jędrzejewski-Szmek { "Manager", "ShutdownWatchdogSec", config_parse_sec, 0, &arg_shutdown_watchdog },
0dae31d468b1a0e22d98921f7b0dbd92fd217167Zbigniew Jędrzejewski-Szmek { "Manager", "CapabilityBoundingSet", config_parse_capability_set, 0, &arg_capability_bounding_set },
8db0d2f5c37e7e8f5bfce016cfdad7947a3ea939Zbigniew Jędrzejewski-Szmek#ifdef HAVE_SECCOMP
8db0d2f5c37e7e8f5bfce016cfdad7947a3ea939Zbigniew Jędrzejewski-Szmek { "Manager", "SystemCallArchitectures", config_parse_syscall_archs, 0, &arg_syscall_archs },
8db0d2f5c37e7e8f5bfce016cfdad7947a3ea939Zbigniew Jędrzejewski-Szmek#endif
8db0d2f5c37e7e8f5bfce016cfdad7947a3ea939Zbigniew Jędrzejewski-Szmek { "Manager", "TimerSlackNSec", config_parse_nsec, 0, &arg_timer_slack_nsec },
8db0d2f5c37e7e8f5bfce016cfdad7947a3ea939Zbigniew Jędrzejewski-Szmek { "Manager", "DefaultTimerAccuracySec", config_parse_sec, 0, &arg_default_timer_accuracy_usec },
8db0d2f5c37e7e8f5bfce016cfdad7947a3ea939Zbigniew Jędrzejewski-Szmek { "Manager", "DefaultStandardOutput", config_parse_output, 0, &arg_default_std_output },
8db0d2f5c37e7e8f5bfce016cfdad7947a3ea939Zbigniew Jędrzejewski-Szmek { "Manager", "DefaultStandardError", config_parse_output, 0, &arg_default_std_error },
0dae31d468b1a0e22d98921f7b0dbd92fd217167Zbigniew Jędrzejewski-Szmek { "Manager", "DefaultTimeoutStartSec", config_parse_sec, 0, &arg_default_timeout_start_usec },
0dae31d468b1a0e22d98921f7b0dbd92fd217167Zbigniew Jędrzejewski-Szmek { "Manager", "DefaultTimeoutStopSec", config_parse_sec, 0, &arg_default_timeout_stop_usec },
8db0d2f5c37e7e8f5bfce016cfdad7947a3ea939Zbigniew Jędrzejewski-Szmek { "Manager", "DefaultRestartSec", config_parse_sec, 0, &arg_default_restart_usec },
0dae31d468b1a0e22d98921f7b0dbd92fd217167Zbigniew Jędrzejewski-Szmek { "Manager", "DefaultStartLimitInterval", config_parse_sec, 0, &arg_default_start_limit_interval },
0dae31d468b1a0e22d98921f7b0dbd92fd217167Zbigniew Jędrzejewski-Szmek { "Manager", "DefaultStartLimitBurst", config_parse_unsigned, 0, &arg_default_start_limit_burst },
0dae31d468b1a0e22d98921f7b0dbd92fd217167Zbigniew Jędrzejewski-Szmek { "Manager", "DefaultEnvironment", config_parse_environ, 0, &arg_default_environment },
0dae31d468b1a0e22d98921f7b0dbd92fd217167Zbigniew Jędrzejewski-Szmek { "Manager", "DefaultLimitCPU", config_parse_limit, RLIMIT_CPU, arg_default_rlimit },
abf126a355e2f2b62b6c51ab3bb37895d1e3eee7Tom Gundersen { "Manager", "DefaultLimitFSIZE", config_parse_limit, RLIMIT_FSIZE, arg_default_rlimit },
abf126a355e2f2b62b6c51ab3bb37895d1e3eee7Tom Gundersen { "Manager", "DefaultLimitDATA", config_parse_limit, RLIMIT_DATA, arg_default_rlimit },
abf126a355e2f2b62b6c51ab3bb37895d1e3eee7Tom Gundersen { "Manager", "DefaultLimitSTACK", config_parse_limit, RLIMIT_STACK, arg_default_rlimit },
abf126a355e2f2b62b6c51ab3bb37895d1e3eee7Tom Gundersen { "Manager", "DefaultLimitCORE", config_parse_limit, RLIMIT_CORE, arg_default_rlimit },
abf126a355e2f2b62b6c51ab3bb37895d1e3eee7Tom Gundersen { "Manager", "DefaultLimitRSS", config_parse_limit, RLIMIT_RSS, arg_default_rlimit },
abf126a355e2f2b62b6c51ab3bb37895d1e3eee7Tom Gundersen { "Manager", "DefaultLimitNOFILE", config_parse_limit, RLIMIT_NOFILE, arg_default_rlimit },
abf126a355e2f2b62b6c51ab3bb37895d1e3eee7Tom Gundersen { "Manager", "DefaultLimitAS", config_parse_limit, RLIMIT_AS, arg_default_rlimit },
abf126a355e2f2b62b6c51ab3bb37895d1e3eee7Tom Gundersen { "Manager", "DefaultLimitNPROC", config_parse_limit, RLIMIT_NPROC, arg_default_rlimit },
abf126a355e2f2b62b6c51ab3bb37895d1e3eee7Tom Gundersen { "Manager", "DefaultLimitMEMLOCK", config_parse_limit, RLIMIT_MEMLOCK, arg_default_rlimit },
abf126a355e2f2b62b6c51ab3bb37895d1e3eee7Tom Gundersen { "Manager", "DefaultLimitLOCKS", config_parse_limit, RLIMIT_LOCKS, arg_default_rlimit },
abf126a355e2f2b62b6c51ab3bb37895d1e3eee7Tom Gundersen { "Manager", "DefaultLimitSIGPENDING", config_parse_limit, RLIMIT_SIGPENDING, arg_default_rlimit },
abf126a355e2f2b62b6c51ab3bb37895d1e3eee7Tom Gundersen { "Manager", "DefaultLimitMSGQUEUE", config_parse_limit, RLIMIT_MSGQUEUE, arg_default_rlimit },
abf126a355e2f2b62b6c51ab3bb37895d1e3eee7Tom Gundersen { "Manager", "DefaultLimitNICE", config_parse_limit, RLIMIT_NICE, arg_default_rlimit },
abf126a355e2f2b62b6c51ab3bb37895d1e3eee7Tom Gundersen { "Manager", "DefaultLimitRTPRIO", config_parse_limit, RLIMIT_RTPRIO, arg_default_rlimit },
abf126a355e2f2b62b6c51ab3bb37895d1e3eee7Tom Gundersen { "Manager", "DefaultLimitRTTIME", config_parse_limit, RLIMIT_RTTIME, arg_default_rlimit },
8db0d2f5c37e7e8f5bfce016cfdad7947a3ea939Zbigniew Jędrzejewski-Szmek { "Manager", "DefaultCPUAccounting", config_parse_bool, 0, &arg_default_cpu_accounting },
8db0d2f5c37e7e8f5bfce016cfdad7947a3ea939Zbigniew Jędrzejewski-Szmek { "Manager", "DefaultBlockIOAccounting", config_parse_bool, 0, &arg_default_blockio_accounting },
8db0d2f5c37e7e8f5bfce016cfdad7947a3ea939Zbigniew Jędrzejewski-Szmek { "Manager", "DefaultMemoryAccounting", config_parse_bool, 0, &arg_default_memory_accounting },
42cc2eebb01056beb7acd3ecfe8e533558237f84Lennart Poettering { "Manager", "DefaultTasksAccounting", config_parse_bool, 0, &arg_default_tasks_accounting },
42cc2eebb01056beb7acd3ecfe8e533558237f84Lennart Poettering { "Manager", "DefaultTasksMax", config_parse_tasks_max, 0, &arg_default_tasks_max },
42cc2eebb01056beb7acd3ecfe8e533558237f84Lennart Poettering {}
42cc2eebb01056beb7acd3ecfe8e533558237f84Lennart Poettering };
42cc2eebb01056beb7acd3ecfe8e533558237f84Lennart Poettering
42cc2eebb01056beb7acd3ecfe8e533558237f84Lennart Poettering const char *fn, *conf_dirs_nulstr;
8db0d2f5c37e7e8f5bfce016cfdad7947a3ea939Zbigniew Jędrzejewski-Szmek
42cc2eebb01056beb7acd3ecfe8e533558237f84Lennart Poettering fn = arg_running_as == MANAGER_SYSTEM ?
42cc2eebb01056beb7acd3ecfe8e533558237f84Lennart Poettering PKGSYSCONFDIR "/system.conf" :
42cc2eebb01056beb7acd3ecfe8e533558237f84Lennart Poettering PKGSYSCONFDIR "/user.conf";
42cc2eebb01056beb7acd3ecfe8e533558237f84Lennart Poettering
ff3d6560bead6879a2fed1bf99bfe8273b3723f1Zbigniew Jędrzejewski-Szmek conf_dirs_nulstr = arg_running_as == MANAGER_SYSTEM ?
ff3d6560bead6879a2fed1bf99bfe8273b3723f1Zbigniew Jędrzejewski-Szmek CONF_PATHS_NULSTR("systemd/system.conf.d") :
ff3d6560bead6879a2fed1bf99bfe8273b3723f1Zbigniew Jędrzejewski-Szmek CONF_PATHS_NULSTR("systemd/user.conf.d");
ff3d6560bead6879a2fed1bf99bfe8273b3723f1Zbigniew Jędrzejewski-Szmek
ff3d6560bead6879a2fed1bf99bfe8273b3723f1Zbigniew Jędrzejewski-Szmek config_parse_many(fn, conf_dirs_nulstr, "Manager\0", config_item_table_lookup, items, false, NULL);
1bf968f36393666f2c57953b1748e6219c027deeTom Gundersen
8db0d2f5c37e7e8f5bfce016cfdad7947a3ea939Zbigniew Jędrzejewski-Szmek /* Traditionally "0" was used to turn off the default unit timeouts. Fix this up so that we used USEC_INFINITY
8db0d2f5c37e7e8f5bfce016cfdad7947a3ea939Zbigniew Jędrzejewski-Szmek * like everywhere else. */
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering if (arg_default_timeout_start_usec <= 0)
ff3d6560bead6879a2fed1bf99bfe8273b3723f1Zbigniew Jędrzejewski-Szmek arg_default_timeout_start_usec = USEC_INFINITY;
8db0d2f5c37e7e8f5bfce016cfdad7947a3ea939Zbigniew Jędrzejewski-Szmek if (arg_default_timeout_stop_usec <= 0)
8db0d2f5c37e7e8f5bfce016cfdad7947a3ea939Zbigniew Jędrzejewski-Szmek arg_default_timeout_stop_usec = USEC_INFINITY;
ff3d6560bead6879a2fed1bf99bfe8273b3723f1Zbigniew Jędrzejewski-Szmek
ff3d6560bead6879a2fed1bf99bfe8273b3723f1Zbigniew Jędrzejewski-Szmek return 0;
8db0d2f5c37e7e8f5bfce016cfdad7947a3ea939Zbigniew Jędrzejewski-Szmek}
8db0d2f5c37e7e8f5bfce016cfdad7947a3ea939Zbigniew Jędrzejewski-Szmek
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poetteringstatic void manager_set_defaults(Manager *m) {
8db0d2f5c37e7e8f5bfce016cfdad7947a3ea939Zbigniew Jędrzejewski-Szmek
ff3d6560bead6879a2fed1bf99bfe8273b3723f1Zbigniew Jędrzejewski-Szmek assert(m);
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering
151226ab4bf276d60d51864330a99f886b923697Zbigniew Jędrzejewski-Szmek m->default_timer_accuracy_usec = arg_default_timer_accuracy_usec;
151226ab4bf276d60d51864330a99f886b923697Zbigniew Jędrzejewski-Szmek m->default_std_output = arg_default_std_output;
7c6423e19136a7b7b6ef3fe06b94822e582dda27Tom Gundersen m->default_std_error = arg_default_std_error;
151226ab4bf276d60d51864330a99f886b923697Zbigniew Jędrzejewski-Szmek m->default_timeout_start_usec = arg_default_timeout_start_usec;
151226ab4bf276d60d51864330a99f886b923697Zbigniew Jędrzejewski-Szmek m->default_timeout_stop_usec = arg_default_timeout_stop_usec;
151226ab4bf276d60d51864330a99f886b923697Zbigniew Jędrzejewski-Szmek m->default_restart_usec = arg_default_restart_usec;
151226ab4bf276d60d51864330a99f886b923697Zbigniew Jędrzejewski-Szmek m->default_start_limit_interval = arg_default_start_limit_interval;
1bf968f36393666f2c57953b1748e6219c027deeTom Gundersen m->default_start_limit_burst = arg_default_start_limit_burst;
151226ab4bf276d60d51864330a99f886b923697Zbigniew Jędrzejewski-Szmek m->default_cpu_accounting = arg_default_cpu_accounting;
151226ab4bf276d60d51864330a99f886b923697Zbigniew Jędrzejewski-Szmek m->default_blockio_accounting = arg_default_blockio_accounting;
151226ab4bf276d60d51864330a99f886b923697Zbigniew Jędrzejewski-Szmek m->default_memory_accounting = arg_default_memory_accounting;
7c6423e19136a7b7b6ef3fe06b94822e582dda27Tom Gundersen m->default_tasks_accounting = arg_default_tasks_accounting;
7c6423e19136a7b7b6ef3fe06b94822e582dda27Tom Gundersen m->default_tasks_max = arg_default_tasks_max;
7c6423e19136a7b7b6ef3fe06b94822e582dda27Tom Gundersen
7c6423e19136a7b7b6ef3fe06b94822e582dda27Tom Gundersen manager_set_default_rlimits(m, arg_default_rlimit);
7c6423e19136a7b7b6ef3fe06b94822e582dda27Tom Gundersen manager_environment_add(m, NULL, arg_default_environment);
7c6423e19136a7b7b6ef3fe06b94822e582dda27Tom Gundersen}
7c6423e19136a7b7b6ef3fe06b94822e582dda27Tom Gundersen
7c6423e19136a7b7b6ef3fe06b94822e582dda27Tom Gundersenstatic int parse_argv(int argc, char *argv[]) {
151226ab4bf276d60d51864330a99f886b923697Zbigniew Jędrzejewski-Szmek
151226ab4bf276d60d51864330a99f886b923697Zbigniew Jędrzejewski-Szmek enum {
151226ab4bf276d60d51864330a99f886b923697Zbigniew Jędrzejewski-Szmek ARG_LOG_LEVEL = 0x100,
7c6423e19136a7b7b6ef3fe06b94822e582dda27Tom Gundersen ARG_LOG_TARGET,
151226ab4bf276d60d51864330a99f886b923697Zbigniew Jędrzejewski-Szmek ARG_LOG_COLOR,
151226ab4bf276d60d51864330a99f886b923697Zbigniew Jędrzejewski-Szmek ARG_LOG_LOCATION,
151226ab4bf276d60d51864330a99f886b923697Zbigniew Jędrzejewski-Szmek ARG_UNIT,
151226ab4bf276d60d51864330a99f886b923697Zbigniew Jędrzejewski-Szmek ARG_SYSTEM,
151226ab4bf276d60d51864330a99f886b923697Zbigniew Jędrzejewski-Szmek ARG_USER,
151226ab4bf276d60d51864330a99f886b923697Zbigniew Jędrzejewski-Szmek ARG_TEST,
151226ab4bf276d60d51864330a99f886b923697Zbigniew Jędrzejewski-Szmek ARG_NO_PAGER,
7c6423e19136a7b7b6ef3fe06b94822e582dda27Tom Gundersen ARG_VERSION,
7c6423e19136a7b7b6ef3fe06b94822e582dda27Tom Gundersen ARG_DUMP_CONFIGURATION_ITEMS,
151226ab4bf276d60d51864330a99f886b923697Zbigniew Jędrzejewski-Szmek ARG_DUMP_CORE,
151226ab4bf276d60d51864330a99f886b923697Zbigniew Jędrzejewski-Szmek ARG_CRASH_CHVT,
151226ab4bf276d60d51864330a99f886b923697Zbigniew Jędrzejewski-Szmek ARG_CRASH_SHELL,
151226ab4bf276d60d51864330a99f886b923697Zbigniew Jędrzejewski-Szmek ARG_CRASH_REBOOT,
151226ab4bf276d60d51864330a99f886b923697Zbigniew Jędrzejewski-Szmek ARG_CONFIRM_SPAWN,
151226ab4bf276d60d51864330a99f886b923697Zbigniew Jędrzejewski-Szmek ARG_SHOW_STATUS,
151226ab4bf276d60d51864330a99f886b923697Zbigniew Jędrzejewski-Szmek ARG_DESERIALIZE,
151226ab4bf276d60d51864330a99f886b923697Zbigniew Jędrzejewski-Szmek ARG_SWITCHED_ROOT,
50f1e641a93cacfc693b0c3d300bee5df0c8c460Tom Gundersen ARG_DEFAULT_STD_OUTPUT,
50f1e641a93cacfc693b0c3d300bee5df0c8c460Tom Gundersen ARG_DEFAULT_STD_ERROR,
50f1e641a93cacfc693b0c3d300bee5df0c8c460Tom Gundersen ARG_MACHINE_ID
50f1e641a93cacfc693b0c3d300bee5df0c8c460Tom Gundersen };
50f1e641a93cacfc693b0c3d300bee5df0c8c460Tom Gundersen
50f1e641a93cacfc693b0c3d300bee5df0c8c460Tom Gundersen static const struct option options[] = {
50f1e641a93cacfc693b0c3d300bee5df0c8c460Tom Gundersen { "log-level", required_argument, NULL, ARG_LOG_LEVEL },
50f1e641a93cacfc693b0c3d300bee5df0c8c460Tom Gundersen { "log-target", required_argument, NULL, ARG_LOG_TARGET },
50f1e641a93cacfc693b0c3d300bee5df0c8c460Tom Gundersen { "log-color", optional_argument, NULL, ARG_LOG_COLOR },
50f1e641a93cacfc693b0c3d300bee5df0c8c460Tom Gundersen { "log-location", optional_argument, NULL, ARG_LOG_LOCATION },
50f1e641a93cacfc693b0c3d300bee5df0c8c460Tom Gundersen { "unit", required_argument, NULL, ARG_UNIT },
50f1e641a93cacfc693b0c3d300bee5df0c8c460Tom Gundersen { "system", no_argument, NULL, ARG_SYSTEM },
50f1e641a93cacfc693b0c3d300bee5df0c8c460Tom Gundersen { "user", no_argument, NULL, ARG_USER },
5d45a8808431987c370706d365fb0cc95cf03d52Tom Gundersen { "test", no_argument, NULL, ARG_TEST },
5d45a8808431987c370706d365fb0cc95cf03d52Tom Gundersen { "no-pager", no_argument, NULL, ARG_NO_PAGER },
5d45a8808431987c370706d365fb0cc95cf03d52Tom Gundersen { "help", no_argument, NULL, 'h' },
f5430a3ef308f3a102899fcaf7fbece757082f2aLennart Poettering { "version", no_argument, NULL, ARG_VERSION },
5d45a8808431987c370706d365fb0cc95cf03d52Tom Gundersen { "dump-configuration-items", no_argument, NULL, ARG_DUMP_CONFIGURATION_ITEMS },
5d45a8808431987c370706d365fb0cc95cf03d52Tom Gundersen { "dump-core", optional_argument, NULL, ARG_DUMP_CORE },
5d45a8808431987c370706d365fb0cc95cf03d52Tom Gundersen { "crash-chvt", required_argument, NULL, ARG_CRASH_CHVT },
5d45a8808431987c370706d365fb0cc95cf03d52Tom Gundersen { "crash-shell", optional_argument, NULL, ARG_CRASH_SHELL },
5d45a8808431987c370706d365fb0cc95cf03d52Tom Gundersen { "crash-reboot", optional_argument, NULL, ARG_CRASH_REBOOT },
5d45a8808431987c370706d365fb0cc95cf03d52Tom Gundersen { "confirm-spawn", optional_argument, NULL, ARG_CONFIRM_SPAWN },
5d45a8808431987c370706d365fb0cc95cf03d52Tom Gundersen { "show-status", optional_argument, NULL, ARG_SHOW_STATUS },
5d45a8808431987c370706d365fb0cc95cf03d52Tom Gundersen { "deserialize", required_argument, NULL, ARG_DESERIALIZE },
5d45a8808431987c370706d365fb0cc95cf03d52Tom Gundersen { "switched-root", no_argument, NULL, ARG_SWITCHED_ROOT },
5d45a8808431987c370706d365fb0cc95cf03d52Tom Gundersen { "default-standard-output", required_argument, NULL, ARG_DEFAULT_STD_OUTPUT, },
5d45a8808431987c370706d365fb0cc95cf03d52Tom Gundersen { "default-standard-error", required_argument, NULL, ARG_DEFAULT_STD_ERROR, },
5d45a8808431987c370706d365fb0cc95cf03d52Tom Gundersen { "machine-id", required_argument, NULL, ARG_MACHINE_ID },
5d45a8808431987c370706d365fb0cc95cf03d52Tom Gundersen {}
5d45a8808431987c370706d365fb0cc95cf03d52Tom Gundersen };
5d45a8808431987c370706d365fb0cc95cf03d52Tom Gundersen
5d45a8808431987c370706d365fb0cc95cf03d52Tom Gundersen int c, r;
5d45a8808431987c370706d365fb0cc95cf03d52Tom Gundersen
5d45a8808431987c370706d365fb0cc95cf03d52Tom Gundersen assert(argc >= 1);
f5430a3ef308f3a102899fcaf7fbece757082f2aLennart Poettering assert(argv);
5d45a8808431987c370706d365fb0cc95cf03d52Tom Gundersen
5d45a8808431987c370706d365fb0cc95cf03d52Tom Gundersen if (getpid() == 1)
5d45a8808431987c370706d365fb0cc95cf03d52Tom Gundersen opterr = 0;
5d45a8808431987c370706d365fb0cc95cf03d52Tom Gundersen
5d45a8808431987c370706d365fb0cc95cf03d52Tom Gundersen while ((c = getopt_long(argc, argv, "hDbsz:", options, NULL)) >= 0)
5d45a8808431987c370706d365fb0cc95cf03d52Tom Gundersen
5d45a8808431987c370706d365fb0cc95cf03d52Tom Gundersen switch (c) {
5d45a8808431987c370706d365fb0cc95cf03d52Tom Gundersen
8db0d2f5c37e7e8f5bfce016cfdad7947a3ea939Zbigniew Jędrzejewski-Szmek case ARG_LOG_LEVEL:
8db0d2f5c37e7e8f5bfce016cfdad7947a3ea939Zbigniew Jędrzejewski-Szmek r = log_set_max_level_from_string(optarg);
8db0d2f5c37e7e8f5bfce016cfdad7947a3ea939Zbigniew Jędrzejewski-Szmek if (r < 0) {
8db0d2f5c37e7e8f5bfce016cfdad7947a3ea939Zbigniew Jędrzejewski-Szmek log_error("Failed to parse log level %s.", optarg);
8db0d2f5c37e7e8f5bfce016cfdad7947a3ea939Zbigniew Jędrzejewski-Szmek return r;
f5430a3ef308f3a102899fcaf7fbece757082f2aLennart Poettering }
d23a27a964748967e1ad20e86de869a753af555bTom Gundersen
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering break;
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering
8db0d2f5c37e7e8f5bfce016cfdad7947a3ea939Zbigniew Jędrzejewski-Szmek case ARG_LOG_TARGET:
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering r = log_set_target_from_string(optarg);
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering if (r < 0) {
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering log_error("Failed to parse log target %s.", optarg);
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering return r;
322345fdb9865ef2477fba8e4bdde0e1183ef505Lennart Poettering }
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering
322345fdb9865ef2477fba8e4bdde0e1183ef505Lennart Poettering break;
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering
322345fdb9865ef2477fba8e4bdde0e1183ef505Lennart Poettering case ARG_LOG_COLOR:
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering if (optarg) {
322345fdb9865ef2477fba8e4bdde0e1183ef505Lennart Poettering r = log_show_color_from_string(optarg);
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering if (r < 0) {
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering log_error("Failed to parse log color setting %s.", optarg);
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering return r;
322345fdb9865ef2477fba8e4bdde0e1183ef505Lennart Poettering }
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering } else
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering log_show_color(true);
322345fdb9865ef2477fba8e4bdde0e1183ef505Lennart Poettering
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering break;
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering case ARG_LOG_LOCATION:
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering if (optarg) {
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering r = log_show_location_from_string(optarg);
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering if (r < 0) {
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering log_error("Failed to parse log location setting %s.", optarg);
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering return r;
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering }
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering } else
322345fdb9865ef2477fba8e4bdde0e1183ef505Lennart Poettering log_show_location(true);
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering 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 = free_and_strdup(&arg_default_unit, optarg);
if (r < 0)
return log_error_errno(r, "Failed to set default unit %s: %m", optarg);
break;
case ARG_SYSTEM:
arg_running_as = MANAGER_SYSTEM;
break;
case ARG_USER:
arg_running_as = MANAGER_USER;
break;
case ARG_TEST:
arg_action = ACTION_TEST;
if (arg_no_pager < 0)
arg_no_pager = true;
break;
case ARG_NO_PAGER:
arg_no_pager = true;
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:
if (!optarg)
arg_dump_core = true;
else {
r = parse_boolean(optarg);
if (r < 0)
return log_error_errno(r, "Failed to parse dump core boolean: %s", optarg);
arg_dump_core = r;
}
break;
case ARG_CRASH_CHVT:
r = parse_crash_chvt(optarg);
if (r < 0)
return log_error_errno(r, "Failed to parse crash virtual terminal index: %s", optarg);
break;
case ARG_CRASH_SHELL:
if (!optarg)
arg_crash_shell = true;
else {
r = parse_boolean(optarg);
if (r < 0)
return log_error_errno(r, "Failed to parse crash shell boolean: %s", optarg);
arg_crash_shell = r;
}
break;
case ARG_CRASH_REBOOT:
if (!optarg)
arg_crash_reboot = true;
else {
r = parse_boolean(optarg);
if (r < 0)
return log_error_errno(r, "Failed to parse crash shell boolean: %s", optarg);
arg_crash_reboot = 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 -EINVAL;
}
(void) fd_cloexec(fd, true);
f = fdopen(fd, "r");
if (!f)
return log_error_errno(errno, "Failed to open serialization fd: %m");
safe_fclose(arg_serialization);
arg_serialization = f;
break;
}
case ARG_SWITCHED_ROOT:
arg_switched_root = true;
break;
case ARG_MACHINE_ID:
r = set_machine_id(optarg);
if (r < 0) {
log_error("MachineID '%s' is not valid.", optarg);
return r;
}
break;
case 'h':
arg_action = ACTION_HELP;
if (arg_no_pager < 0)
arg_no_pager = true;
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 '?':
if (getpid() != 1)
return -EINVAL;
else
return 0;
default:
assert_not_reached("Unhandled option code.");
}
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;
}
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"
" --no-pager Do not pipe output into a pager\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[=BOOL] Dump core on crash\n"
" --crash-vt=NR Change to specified VT on crash\n"
" --crash-reboot[=BOOL] Reboot on crash\n"
" --crash-shell[=BOOL] Run shell on crash\n"
" --confirm-spawn[=BOOL] Ask for confirmation when spawning processes\n"
" --show-status[=BOOL] Show status updates on the console during bootup\n"
" --log-target=TARGET Set log target (console, journal, kmsg, journal-or-kmsg, null)\n"
" --log-level=LEVEL Set log level (debug, info, notice, warning, err, crit, alert, emerg)\n"
" --log-color[=BOOL] Highlight important log messages\n"
" --log-location[=BOOL] 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 prepare_reexecute(Manager *m, FILE **_f, FDSet **_fds, bool switching_root) {
_cleanup_fdset_free_ FDSet *fds = NULL;
_cleanup_fclose_ FILE *f = NULL;
int r;
assert(m);
assert(_f);
assert(_fds);
r = manager_open_serialization(m, &f);
if (r < 0)
return log_error_errno(r, "Failed to create serialization file: %m");
/* Make sure nothing is really destructed when we shut down */
m->n_reloading ++;
bus_manager_send_reloading(m, true);
fds = fdset_new();
if (!fds)
return log_oom();
r = manager_serialize(m, f, fds, switching_root);
if (r < 0)
return log_error_errno(r, "Failed to serialize state: %m");
if (fseeko(f, 0, SEEK_SET) == (off_t) -1)
return log_error_errno(errno, "Failed to rewind serialization fd: %m");
r = fd_cloexec(fileno(f), false);
if (r < 0)
return log_error_errno(r, "Failed to disable O_CLOEXEC for serialization: %m");
r = fdset_cloexec(fds, false);
if (r < 0)
return log_error_errno(r, "Failed to disable O_CLOEXEC for serialization fds: %m");
*_f = f;
*_fds = fds;
f = NULL;
fds = NULL;
return 0;
}
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)
return log_error_errno(errno, "Reading RLIMIT_NOFILE failed: %m");
/* 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)
return log_error_errno(r, "Setting RLIMIT_NOFILE failed: %m");
return 0;
}
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 filesystem 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" doesn'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);
if (!arg_join_controllers[0])
goto oom;
arg_join_controllers[1] = strv_new("net_cls", "net_prio", NULL);
if (!arg_join_controllers[1])
goto oom;
arg_join_controllers[2] = NULL;
return 0;
oom:
arg_join_controllers = strv_free_free(arg_join_controllers);
return -ENOMEM;
}
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_errno(r, "Failed to add architecture to seccomp: %m");
goto finish;
}
}
r = seccomp_attr_set(seccomp, SCMP_FLTATR_CTL_NNP, 0);
if (r < 0) {
log_error_errno(r, "Failed to unset NO_NEW_PRIVS: %m");
goto finish;
}
r = seccomp_load(seccomp);
if (r < 0)
log_error_errno(r, "Failed to add install architecture seccomp: %m");
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_errno(r, "Failed to read os-release file: %m");
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;
int r;
c = getenv("container");
if (isempty(c))
return 0;
r = write_string_file("/run/systemd/container", c, WRITE_STRING_FILE_CREATE);
if (r < 0)
return log_warning_errno(r, "Failed to write /run/systemd/container, ignoring: %m");
return 1;
}
static int bump_unix_max_dgram_qlen(void) {
_cleanup_free_ char *qlen = NULL;
unsigned long v;
int r;
/* Let's bump the net.unix.max_dgram_qlen sysctl. The kernel
* default of 16 is simply too low. We set the value really
* really early during boot, so that it is actually applied to
* all our sockets, including the $NOTIFY_SOCKET one. */
r = read_one_line_file("/proc/sys/net/unix/max_dgram_qlen", &qlen);
if (r < 0)
return log_warning_errno(r, "Failed to read AF_UNIX datagram queue length, ignoring: %m");
r = safe_atolu(qlen, &v);
if (r < 0)
return log_warning_errno(r, "Failed to parse AF_UNIX datagram queue length, ignoring: %m");
if (v >= DEFAULT_UNIX_MAX_DGRAM_QLEN)
return 0;
qlen = mfree(qlen);
if (asprintf(&qlen, "%lu\n", DEFAULT_UNIX_MAX_DGRAM_QLEN) < 0)
return log_oom();
r = write_string_file("/proc/sys/net/unix/max_dgram_qlen", qlen, 0);
if (r < 0)
return log_full_errno(IN_SET(r, -EROFS, -EPERM, -EACCES) ? LOG_DEBUG : LOG_WARNING, r,
"Failed to bump AF_UNIX datagram queue length, ignoring: %m");
return 1;
}
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 = DUAL_TIMESTAMP_NULL;
dual_timestamp userspace_timestamp = DUAL_TIMESTAMP_NULL;
dual_timestamp kernel_timestamp = DUAL_TIMESTAMP_NULL;
dual_timestamp security_start_timestamp = DUAL_TIMESTAMP_NULL;
dual_timestamp security_finish_timestamp = DUAL_TIMESTAMP_NULL;
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;
struct rlimit saved_rlimit_nofile = RLIMIT_MAKE_CONST(0);
const char *error_message = NULL;
#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_errno(errno, "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);
log_set_upgrade_syslog_to_journal(true);
/* Disable the umask logic */
if (getpid() == 1)
umask(0);
if (getpid() == 1 && detect_container() <= 0) {
/* Running outside of a container as PID 1 */
arg_running_as = MANAGER_SYSTEM;
make_null_stdio();
log_set_target(LOG_TARGET_KMSG);
log_open();
if (in_initrd())
initrd_timestamp = userspace_timestamp;
if (!skip_setup) {
r = mount_setup_early();
if (r < 0) {
error_message = "Failed to early mount API filesystems";
goto finish;
}
dual_timestamp_get(&security_start_timestamp);
if (mac_selinux_setup(&loaded_policy) < 0) {
error_message = "Failed to load SELinux policy";
goto finish;
} else if (ima_setup() < 0) {
error_message = "Failed to load IMA policy";
goto finish;
} else if (mac_smack_setup(&loaded_policy) < 0) {
error_message = "Failed to load SMACK policy";
goto finish;
}
dual_timestamp_get(&security_finish_timestamp);
}
if (mac_selinux_init(NULL) < 0) {
error_message = "Failed to initialize SELinux policy";
goto finish;
}
if (!skip_setup) {
if (clock_is_localtime() > 0) {
int min;
/*
* The very first call of settimeofday() also does a time warp in the kernel.
*
* In the rtc-in-local time mode, we set the kernel's timezone, and rely on
* external tools to take care of maintaining the RTC and do all adjustments.
* This matches the behavior of Windows, which leaves the RTC alone if the
* registry tells that the RTC runs in UTC.
*/
r = clock_set_timezone(&min);
if (r < 0)
log_error_errno(r, "Failed to apply local time delta, ignoring: %m");
else
log_info("RTC configured in localtime, applying delta of %i minutes to system time.", min);
} else if (!in_initrd()) {
/*
* Do a dummy very first 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.
*
* Do no set the kernel's timezone. The concept of local time cannot
* be supported reliably, the time will jump or be incorrect at every daylight
* saving time change. All kernel local time concepts will be treated
* as UTC that way.
*/
(void) clock_reset_timewarp();
}
r = clock_apply_epoch();
if (r < 0)
log_error_errno(r, "Current system time is before build time, but cannot correct: %m");
else if (r > 0)
log_info("System time before build time, advancing clock.");
}
/* 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 = MANAGER_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 = MANAGER_USER;
log_set_target(LOG_TARGET_AUTO);
log_open();
/* clear the kernel timestamp,
* because we are not PID 1 */
kernel_timestamp = DUAL_TIMESTAMP_NULL;
}
if (getpid() == 1) {
/* Don't limit the core dump size, so that coredump handlers such as systemd-coredump (which honour the limit)
* will process core dumps for system services by default. */
(void) setrlimit(RLIMIT_CORE, &RLIMIT_MAKE_CONST(RLIM_INFINITY));
/* But at the same time, turn off the core_pattern logic by default, so that no coredumps are stored
* until the systemd-coredump tool is enabled via sysctl. */
if (!skip_setup)
(void) write_string_file("/proc/sys/kernel/core_pattern", "|/bin/false", 0);
}
/* Initialize default unit */
r = free_and_strdup(&arg_default_unit, SPECIAL_DEFAULT_TARGET);
if (r < 0) {
log_emergency_errno(r, "Failed to set default unit %s: %m", SPECIAL_DEFAULT_TARGET);
error_message = "Failed to set default unit";
goto finish;
}
r = initialize_join_controllers();
if (r < 0) {
error_message = "Failed to initialize cgroup controllers";
goto finish;
}
/* Mount /proc, /sys and friends, so that /proc/cmdline and
* /proc/$PID/fd is available. */
if (getpid() == 1) {
/* Load the kernel modules early, so that we kdbus.ko is loaded before kdbusfs shall be mounted */
if (!skip_setup)
kmod_setup();
r = mount_setup(loaded_policy);
if (r < 0) {
error_message = "Failed to mount API filesystems";
goto finish;
}
}
/* Reset all signal handlers. */
(void) reset_all_signal_handlers();
(void) ignore_signals(SIGNALS_IGNORE, -1);
if (parse_config_file() < 0) {
error_message = "Failed to parse config file";
goto finish;
}
if (arg_running_as == MANAGER_SYSTEM) {
r = parse_proc_cmdline(parse_proc_cmdline_item);
if (r < 0)
log_warning_errno(r, "Failed to parse kernel command line, ignoring: %m");
}
/* Note that this also parses bits from the kernel command
* line, including "debug". */
log_parse_environment();
if (parse_argv(argc, argv) < 0) {
error_message = "Failed to parse commandline arguments";
goto finish;
}
if (arg_action == ACTION_TEST &&
geteuid() == 0) {
log_error("Don't run test mode as root.");
goto finish;
}
if (arg_running_as == MANAGER_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 == MANAGER_SYSTEM &&
arg_action == ACTION_RUN &&
running_in_chroot() > 0) {
log_error("Cannot be run in a chroot() environment.");
goto finish;
}
if (arg_action == ACTION_TEST)
skip_setup = true;
pager_open_if_enabled();
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 == MANAGER_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_emergency_errno(r, "Failed to allocate fd set: %m");
error_message = "Failed to allocate fd set";
goto finish;
} else
fdset_cloexec(fds, true);
if (arg_serialization)
assert_se(fdset_remove(fds, fileno(arg_serialization)) >= 0);
if (arg_running_as == MANAGER_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 == MANAGER_SYSTEM && arg_action == ACTION_RUN) {
/* If we are init, we connect stdin/stdout/stderr to
* /dev/null and make sure we don't have a controlling
* tty. */
release_terminal();
if (getpid() == 1 && !skip_setup)
console_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 == MANAGER_SYSTEM) {
int v;
log_info(PACKAGE_STRING " running in %ssystem mode. (" SYSTEMD_FEATURES ")",
arg_action == ACTION_TEST ? "test " : "" );
v = detect_virtualization();
if (v > 0)
log_info("Detected virtualization %s.", virtualization_to_string(v));
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 %suser mode for user "UID_FMT"/%s. (" SYSTEMD_FEATURES ")",
arg_action == ACTION_TEST ? " test" : "", getuid(), t);
}
if (arg_running_as == MANAGER_SYSTEM && !skip_setup) {
if (arg_show_status > 0)
status_welcome();
hostname_setup();
machine_id_setup(NULL, arg_machine_id);
loopback_setup();
bump_unix_max_dgram_qlen();
test_usr();
}
if (arg_running_as == MANAGER_SYSTEM && arg_runtime_watchdog > 0)
watchdog_set_timeout(&arg_runtime_watchdog);
if (arg_timer_slack_nsec != NSEC_INFINITY)
if (prctl(PR_SET_TIMERSLACK, arg_timer_slack_nsec) < 0)
log_error_errno(errno, "Failed to adjust timer slack: %m");
if (!cap_test_all(arg_capability_bounding_set)) {
r = capability_bounding_set_drop_usermode(arg_capability_bounding_set);
if (r < 0) {
log_emergency_errno(r, "Failed to drop capability bounding set of usermode helpers: %m");
error_message = "Failed to drop capability bounding set of usermode helpers";
goto finish;
}
r = capability_bounding_set_drop(arg_capability_bounding_set, true);
if (r < 0) {
log_emergency_errno(r, "Failed to drop capability bounding set: %m");
error_message = "Failed to drop capability bounding set";
goto finish;
}
}
if (arg_syscall_archs) {
r = enforce_syscall_archs(arg_syscall_archs);
if (r < 0) {
error_message = "Failed to set syscall architectures";
goto finish;
}
}
if (arg_running_as == MANAGER_USER)
/* Become reaper of our children */
if (prctl(PR_SET_CHILD_SUBREAPER, 1) < 0)
log_warning_errno(errno, "Failed to make us a subreaper: %m");
if (arg_running_as == MANAGER_SYSTEM) {
bump_rlimit_nofile(&saved_rlimit_nofile);
if (empty_etc) {
r = unit_file_preset_all(UNIT_FILE_SYSTEM, false, NULL, UNIT_FILE_PRESET_ENABLE_ONLY, false, NULL, 0);
if (r < 0)
log_full_errno(r == -EEXIST ? LOG_NOTICE : LOG_WARNING, r, "Failed to populate /etc with preset unit settings, ignoring: %m");
else
log_info("Populated /etc with preset unit settings.");
}
}
r = manager_new(arg_running_as, arg_action == ACTION_TEST, &m);
if (r < 0) {
log_emergency_errno(r, "Failed to allocate manager object: %m");
error_message = "Failed to allocate manager object";
goto finish;
}
m->confirm_spawn = arg_confirm_spawn;
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_defaults(m);
manager_set_show_status(m, arg_show_status);
manager_set_first_boot(m, empty_etc);
/* 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_errno(r, "Failed to fully start up daemon: %m");
/* This will close all file descriptors that were opened, but
* not claimed by any unit. */
fds = fdset_free(fds);
arg_serialization = safe_fclose(arg_serialization);
if (queue_default_job) {
_cleanup_(sd_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_errno(target->load_error, "Failed to load default target: %m");
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_emergency("Failed to load rescue target: %s", bus_error_message(&error, r));
error_message = "Failed to load rescue target";
goto finish;
} else if (target->load_state == UNIT_ERROR || target->load_state == UNIT_NOT_FOUND) {
log_emergency_errno(target->load_error, "Failed to load rescue target: %m");
error_message = "Failed to load rescue target";
goto finish;
} else if (target->load_state == UNIT_MASKED) {
log_emergency("Rescue target masked.");
error_message = "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, &error, &default_unit_job);
if (r == -EPERM) {
log_debug("Default target could not be isolated, starting instead: %s", bus_error_message(&error, r));
sd_bus_error_free(&error);
r = manager_add_job(m, JOB_START, target, JOB_REPLACE, &error, &default_unit_job);
if (r < 0) {
log_emergency("Failed to start default target: %s", bus_error_message(&error, r));
error_message = "Failed to start default target";
goto finish;
}
} else if (r < 0) {
log_emergency("Failed to isolate default target: %s", bus_error_message(&error, r));
error_message = "Failed to isolate default target";
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, 100 * USEC_PER_MSEC));
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_emergency_errno(r, "Failed to run main loop: %m");
error_message = "Failed to run main loop";
goto finish;
}
switch (m->exit_code) {
case MANAGER_RELOAD:
log_info("Reloading.");
r = parse_config_file();
if (r < 0)
log_error("Failed to parse config file.");
manager_set_defaults(m);
r = manager_reload(m);
if (r < 0)
log_error_errno(r, "Failed to reload: %m");
break;
case MANAGER_REEXECUTE:
if (prepare_reexecute(m, &arg_serialization, &fds, false) < 0) {
error_message = "Failed to prepare for reexecution";
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) {
error_message = "Failed to prepare for reexecution";
goto finish;
}
reexecute = true;
log_notice("Switching root.");
goto finish;
case MANAGER_EXIT:
retval = m->return_value;
if (m->running_as == MANAGER_USER) {
log_debug("Exit.");
goto finish;
}
/* fallthrough */
case MANAGER_REBOOT:
case MANAGER_POWEROFF:
case MANAGER_HALT:
case MANAGER_KEXEC: {
static const char * const table[_MANAGER_EXIT_CODE_MAX] = {
[MANAGER_EXIT] = "exit",
[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:
pager_close();
if (m)
arg_shutdown_watchdog = m->shutdown_watchdog;
m = manager_free(m);
for (j = 0; j < ELEMENTSOF(arg_default_rlimit); j++)
arg_default_rlimit[j] = mfree(arg_default_rlimit[j]);
arg_default_unit = mfree(arg_default_unit);
arg_join_controllers = strv_free_free(arg_join_controllers);
arg_default_environment = strv_free(arg_default_environment);
arg_syscall_archs = set_free(arg_syscall_archs);
mac_selinux_finish();
if (reexecute) {
const char **args;
unsigned i, args_size;
/* 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)
(void) 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 with MS_MOVE, because we remove the old directory afterwards and detach it. */
r = switch_root(switch_root_dir, "/mnt", true, MS_MOVE);
if (r < 0)
log_error_errno(r, "Failed to switch root, trying to continue: %m");
}
args_size = MAX(6, argc+1);
args = newa(const char*, args_size);
if (!switch_root_init) {
char sfd[DECIMAL_STR_MAX(int) + 1];
/* 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);
xsprintf(sfd, "%i", fileno(arg_serialization));
i = 0;
args[i++] = SYSTEMD_BINARY_PATH;
if (switch_root_dir)
args[i++] = "--switched-root";
args[i++] = arg_running_as == MANAGER_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)
(void) clearenv();
assert(i <= args_size);
/*
* We want valgrind to print its memory usage summary before reexecution.
* Valgrind won't do this is on its own on exec(), but it will do it on exit().
* Hence, to ensure we get a summary here, fork() off a child, let it exit() cleanly,
* so that it prints the summary, and wait() for it in the parent, before proceeding into the exec().
*/
valgrind_summary_hack();
(void) 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.) */
arg_serialization = safe_fclose(arg_serialization);
fds = fdset_free(fds);
/* Reopen the console */
(void) 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=... */
(void) reset_all_signal_handlers();
(void) reset_signal_mask();
if (switch_root_init) {
args[0] = switch_root_init;
(void) execv(args[0], (char* const*) args);
log_warning_errno(errno, "Failed to execute configured init, trying fallback: %m");
}
args[0] = "/sbin/init";
(void) execv(args[0], (char* const*) args);
if (errno == ENOENT) {
log_warning("No /sbin/init, trying fallback");
args[0] = "/bin/sh";
args[1] = NULL;
(void) execv(args[0], (char* const*) args);
log_error_errno(errno, "Failed to execute /bin/sh, giving up: %m");
} else
log_warning_errno(errno, "Failed to execute /sbin/init, giving up: %m");
}
arg_serialization = safe_fclose(arg_serialization);
fds = fdset_free(fds);
#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];
char exit_code[DECIMAL_STR_MAX(uint8_t) + 1];
const char* command_line[11] = {
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);
xsprintf(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_NULL:
command_line[pos++] = "null";
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";
if (streq(shutdown_verb, "exit")) {
command_line[pos++] = "--exit-code";
command_line[pos++] = exit_code;
xsprintf(exit_code, "%d", retval);
}
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 */
r = watchdog_set_timeout(&arg_shutdown_watchdog);
watchdog_close(r < 0);
/* Tell the binary how often to ping, ignore failure */
if (asprintf(&e, "WATCHDOG_USEC="USEC_FMT, arg_shutdown_watchdog) > 0)
(void) 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() <= 0)
(void) cg_uninstall_release_agent(SYSTEMD_CGROUP_CONTROLLER);
execve(SYSTEMD_SHUTDOWN_BINARY_PATH, (char **) command_line, env_block);
log_error_errno(errno, "Failed to execute shutdown binary, %s: %m",
getpid() == 1 ? "freezing" : "quitting");
}
if (getpid() == 1) {
if (error_message)
manager_status_printf(NULL, STATUS_TYPE_EMERGENCY,
ANSI_HIGHLIGHT_RED "!!!!!!" ANSI_NORMAL,
"%s, freezing.", error_message);
freeze_or_reboot();
}
return retval;
}