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