main.c revision c72aadd1851096ea979f68b4e32cca71746ccdc4
74b2466e14a1961bf3ac0e8a60cfaceec705bd59Lennart Poettering/*-*- Mode: C; c-basic-offset: 8; indent-tabs-mode: nil -*-*/
74b2466e14a1961bf3ac0e8a60cfaceec705bd59Lennart Poettering
74b2466e14a1961bf3ac0e8a60cfaceec705bd59Lennart Poettering/***
74b2466e14a1961bf3ac0e8a60cfaceec705bd59Lennart Poettering This file is part of systemd.
74b2466e14a1961bf3ac0e8a60cfaceec705bd59Lennart Poettering
74b2466e14a1961bf3ac0e8a60cfaceec705bd59Lennart Poettering Copyright 2010 Lennart Poettering
74b2466e14a1961bf3ac0e8a60cfaceec705bd59Lennart Poettering
74b2466e14a1961bf3ac0e8a60cfaceec705bd59Lennart Poettering systemd is free software; you can redistribute it and/or modify it
74b2466e14a1961bf3ac0e8a60cfaceec705bd59Lennart Poettering under the terms of the GNU Lesser General Public License as published by
74b2466e14a1961bf3ac0e8a60cfaceec705bd59Lennart Poettering the Free Software Foundation; either version 2.1 of the License, or
74b2466e14a1961bf3ac0e8a60cfaceec705bd59Lennart Poettering (at your option) any later version.
74b2466e14a1961bf3ac0e8a60cfaceec705bd59Lennart Poettering
74b2466e14a1961bf3ac0e8a60cfaceec705bd59Lennart Poettering systemd is distributed in the hope that it will be useful, but
74b2466e14a1961bf3ac0e8a60cfaceec705bd59Lennart Poettering WITHOUT ANY WARRANTY; without even the implied warranty of
74b2466e14a1961bf3ac0e8a60cfaceec705bd59Lennart Poettering MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
74b2466e14a1961bf3ac0e8a60cfaceec705bd59Lennart Poettering Lesser General Public License for more details.
74b2466e14a1961bf3ac0e8a60cfaceec705bd59Lennart Poettering
74b2466e14a1961bf3ac0e8a60cfaceec705bd59Lennart Poettering You should have received a copy of the GNU Lesser General Public License
74b2466e14a1961bf3ac0e8a60cfaceec705bd59Lennart Poettering along with systemd; If not, see <http://www.gnu.org/licenses/>.
74b2466e14a1961bf3ac0e8a60cfaceec705bd59Lennart Poettering***/
74b2466e14a1961bf3ac0e8a60cfaceec705bd59Lennart Poettering
0dae31d468b1a0e22d98921f7b0dbd92fd217167Zbigniew Jędrzejewski-Szmek#include <dbus/dbus.h>
0dae31d468b1a0e22d98921f7b0dbd92fd217167Zbigniew Jędrzejewski-Szmek
b5efdb8af40ea759a1ea584c1bc44ecc81dd00ceLennart Poettering#include <stdio.h>
4ad7f2761da661853dcc29d542efb4727abb1101Nick Owens#include <errno.h>
7263f72499e962b3fd54cdb7c79d49ca72121edeZbigniew Jędrzejewski-Szmek#include <string.h>
e4e73a632524c382139034d4271f53b6089ab4cbLennart Poettering#include <unistd.h>
07630cea1f3a845c09309f197ac7c4f11edd3b62Lennart Poettering#include <sys/types.h>
e4e73a632524c382139034d4271f53b6089ab4cbLennart Poettering#include <sys/stat.h>
8730bccfc59fe507bd3e0a3abcf411b497ac4f0eLennart Poettering#include <getopt.h>
07630cea1f3a845c09309f197ac7c4f11edd3b62Lennart Poettering#include <signal.h>
07630cea1f3a845c09309f197ac7c4f11edd3b62Lennart Poettering#include <sys/wait.h>
74b2466e14a1961bf3ac0e8a60cfaceec705bd59Lennart Poettering#include <fcntl.h>
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering#include <sys/prctl.h>
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering#include <sys/mount.h>
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering
74b2466e14a1961bf3ac0e8a60cfaceec705bd59Lennart Poettering#include "manager.h"
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering#include "log.h"
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering#include "load-fragment.h"
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering#include "fdset.h"
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering#include "special.h"
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering#include "conf-parser.h"
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering#include "dbus-common.h"
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering#include "missing.h"
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering#include "label.h"
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering#include "build.h"
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering#include "strv.h"
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering#include "def.h"
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering#include "virt.h"
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering#include "watchdog.h"
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering#include "path-util.h"
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering#include "switch-root.h"
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering#include "capability.h"
36d9205d669bcdcb04fa730d1f3549a9fc9a9001Tom Gundersen#include "killall.h"
58db254ade4fb2ef77de68f28c4f13814819f6a1Lennart Poettering#include "env-util.h"
58db254ade4fb2ef77de68f28c4f13814819f6a1Lennart Poettering#include "hwclock.h"
36d9205d669bcdcb04fa730d1f3549a9fc9a9001Tom Gundersen#include "sd-daemon.h"
36d9205d669bcdcb04fa730d1f3549a9fc9a9001Tom Gundersen#include "sd-messages.h"
36d9205d669bcdcb04fa730d1f3549a9fc9a9001Tom Gundersen
58db254ade4fb2ef77de68f28c4f13814819f6a1Lennart Poettering#include "mount-setup.h"
58db254ade4fb2ef77de68f28c4f13814819f6a1Lennart Poettering#include "loopback-setup.h"
58db254ade4fb2ef77de68f28c4f13814819f6a1Lennart Poettering#ifdef HAVE_KMOD
58db254ade4fb2ef77de68f28c4f13814819f6a1Lennart Poettering#include "kmod-setup.h"
58db254ade4fb2ef77de68f28c4f13814819f6a1Lennart Poettering#endif
58db254ade4fb2ef77de68f28c4f13814819f6a1Lennart Poettering#include "hostname-setup.h"
58db254ade4fb2ef77de68f28c4f13814819f6a1Lennart Poettering#include "machine-id-setup.h"
58db254ade4fb2ef77de68f28c4f13814819f6a1Lennart Poettering#include "locale-setup.h"
58db254ade4fb2ef77de68f28c4f13814819f6a1Lennart Poettering#include "selinux-setup.h"
58db254ade4fb2ef77de68f28c4f13814819f6a1Lennart Poettering#include "ima-setup.h"
58db254ade4fb2ef77de68f28c4f13814819f6a1Lennart Poettering#include "fileio.h"
58db254ade4fb2ef77de68f28c4f13814819f6a1Lennart Poettering#include "smack-setup.h"
58db254ade4fb2ef77de68f28c4f13814819f6a1Lennart Poettering
58db254ade4fb2ef77de68f28c4f13814819f6a1Lennart Poetteringstatic enum {
58db254ade4fb2ef77de68f28c4f13814819f6a1Lennart Poettering ACTION_RUN,
58db254ade4fb2ef77de68f28c4f13814819f6a1Lennart Poettering ACTION_HELP,
58db254ade4fb2ef77de68f28c4f13814819f6a1Lennart Poettering ACTION_VERSION,
58db254ade4fb2ef77de68f28c4f13814819f6a1Lennart Poettering ACTION_TEST,
58db254ade4fb2ef77de68f28c4f13814819f6a1Lennart Poettering ACTION_DUMP_CONFIGURATION_ITEMS,
58db254ade4fb2ef77de68f28c4f13814819f6a1Lennart Poettering ACTION_DONE
58db254ade4fb2ef77de68f28c4f13814819f6a1Lennart Poettering} arg_action = ACTION_RUN;
58db254ade4fb2ef77de68f28c4f13814819f6a1Lennart Poettering
36d9205d669bcdcb04fa730d1f3549a9fc9a9001Tom Gundersenstatic char *arg_default_unit = NULL;
36d9205d669bcdcb04fa730d1f3549a9fc9a9001Tom Gundersenstatic SystemdRunningAs arg_running_as = _SYSTEMD_RUNNING_AS_INVALID;
801ad6a6a9cd8fbd58b9f9c27f20dbb3c87d47ddLennart Poettering
801ad6a6a9cd8fbd58b9f9c27f20dbb3c87d47ddLennart Poetteringstatic bool arg_dump_core = true;
801ad6a6a9cd8fbd58b9f9c27f20dbb3c87d47ddLennart Poetteringstatic bool arg_crash_shell = false;
801ad6a6a9cd8fbd58b9f9c27f20dbb3c87d47ddLennart Poetteringstatic int arg_crash_chvt = -1;
801ad6a6a9cd8fbd58b9f9c27f20dbb3c87d47ddLennart Poetteringstatic bool arg_confirm_spawn = false;
801ad6a6a9cd8fbd58b9f9c27f20dbb3c87d47ddLennart Poetteringstatic bool arg_show_status = true;
801ad6a6a9cd8fbd58b9f9c27f20dbb3c87d47ddLennart Poetteringstatic bool arg_switched_root = false;
801ad6a6a9cd8fbd58b9f9c27f20dbb3c87d47ddLennart Poetteringstatic char ***arg_join_controllers = NULL;
801ad6a6a9cd8fbd58b9f9c27f20dbb3c87d47ddLennart Poetteringstatic ExecOutput arg_default_std_output = EXEC_OUTPUT_JOURNAL;
dc477e7385e8ab29efb8fadb72ec994077a105c6Lennart Poetteringstatic ExecOutput arg_default_std_error = EXEC_OUTPUT_INHERIT;
801ad6a6a9cd8fbd58b9f9c27f20dbb3c87d47ddLennart Poetteringstatic usec_t arg_runtime_watchdog = 0;
801ad6a6a9cd8fbd58b9f9c27f20dbb3c87d47ddLennart Poetteringstatic usec_t arg_shutdown_watchdog = 10 * USEC_PER_MINUTE;
801ad6a6a9cd8fbd58b9f9c27f20dbb3c87d47ddLennart Poetteringstatic char **arg_default_environment = NULL;
801ad6a6a9cd8fbd58b9f9c27f20dbb3c87d47ddLennart Poetteringstatic struct rlimit *arg_default_rlimit[RLIMIT_NLIMITS] = {};
801ad6a6a9cd8fbd58b9f9c27f20dbb3c87d47ddLennart Poetteringstatic uint64_t arg_capability_bounding_set_drop = 0;
801ad6a6a9cd8fbd58b9f9c27f20dbb3c87d47ddLennart Poetteringstatic nsec_t arg_timer_slack_nsec = (nsec_t) -1;
801ad6a6a9cd8fbd58b9f9c27f20dbb3c87d47ddLennart Poettering
801ad6a6a9cd8fbd58b9f9c27f20dbb3c87d47ddLennart Poetteringstatic FILE* serialization = NULL;
801ad6a6a9cd8fbd58b9f9c27f20dbb3c87d47ddLennart Poettering
801ad6a6a9cd8fbd58b9f9c27f20dbb3c87d47ddLennart Poetteringstatic void nop_handler(int sig) {
801ad6a6a9cd8fbd58b9f9c27f20dbb3c87d47ddLennart Poettering}
801ad6a6a9cd8fbd58b9f9c27f20dbb3c87d47ddLennart Poettering
801ad6a6a9cd8fbd58b9f9c27f20dbb3c87d47ddLennart Poettering_noreturn_ static void crash(int sig) {
801ad6a6a9cd8fbd58b9f9c27f20dbb3c87d47ddLennart Poettering
801ad6a6a9cd8fbd58b9f9c27f20dbb3c87d47ddLennart Poettering if (getpid() != 1)
801ad6a6a9cd8fbd58b9f9c27f20dbb3c87d47ddLennart Poettering /* Pass this on immediately, if this is not PID 1 */
801ad6a6a9cd8fbd58b9f9c27f20dbb3c87d47ddLennart Poettering raise(sig);
801ad6a6a9cd8fbd58b9f9c27f20dbb3c87d47ddLennart Poettering else if (!arg_dump_core)
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering log_error("Caught <%s>, not dumping core.", signal_to_string(sig));
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering else {
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering struct sigaction sa = {
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering .sa_handler = nop_handler,
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering .sa_flags = SA_NOCLDSTOP|SA_RESTART,
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering };
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering pid_t pid;
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering /* We want to wait for the core process, hence let's enable SIGCHLD */
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering sigaction(SIGCHLD, &sa, NULL);
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering pid = fork();
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering if (pid < 0)
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering log_error("Caught <%s>, cannot fork for core dump: %s", signal_to_string(sig), strerror(errno));
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering else if (pid == 0) {
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering struct rlimit rl = {};
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering /* Enable default signal handler for core dump */
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering zero(sa);
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering sa.sa_handler = SIG_DFL;
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering sigaction(sig, &sa, NULL);
1b4f6e79ec51a57003896a0b605fba427b4a98d2Lennart Poettering
1b4f6e79ec51a57003896a0b605fba427b4a98d2Lennart Poettering /* Don't limit the core dump size */
1b4f6e79ec51a57003896a0b605fba427b4a98d2Lennart Poettering rl.rlim_cur = RLIM_INFINITY;
1b4f6e79ec51a57003896a0b605fba427b4a98d2Lennart Poettering rl.rlim_max = RLIM_INFINITY;
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering setrlimit(RLIMIT_CORE, &rl);
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering /* Just to be sure... */
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering chdir("/");
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering /* Raise the signal again */
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering raise(sig);
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering assert_not_reached("We shouldn't be here...");
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering _exit(1);
1b4f6e79ec51a57003896a0b605fba427b4a98d2Lennart Poettering
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering } else {
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering siginfo_t status;
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering int r;
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering /* Order things nicely. */
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering r = wait_for_terminate(pid, &status);
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering if (r < 0)
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering log_error("Caught <%s>, waitpid() failed: %s", signal_to_string(sig), strerror(-r));
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering else if (status.si_code != CLD_DUMPED)
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering log_error("Caught <%s>, core dump failed.", signal_to_string(sig));
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering else
28b9b7640603f88cb49f95609331fa5072715f15Lennart Poettering log_error("Caught <%s>, dumped core as pid %lu.", signal_to_string(sig), (unsigned long) pid);
28b9b7640603f88cb49f95609331fa5072715f15Lennart Poettering }
28b9b7640603f88cb49f95609331fa5072715f15Lennart Poettering }
28b9b7640603f88cb49f95609331fa5072715f15Lennart Poettering
28b9b7640603f88cb49f95609331fa5072715f15Lennart Poettering if (arg_crash_chvt)
28b9b7640603f88cb49f95609331fa5072715f15Lennart Poettering chvt(arg_crash_chvt);
28b9b7640603f88cb49f95609331fa5072715f15Lennart Poettering
28b9b7640603f88cb49f95609331fa5072715f15Lennart Poettering if (arg_crash_shell) {
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering struct sigaction sa = {
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering .sa_handler = SIG_IGN,
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering .sa_flags = SA_NOCLDSTOP|SA_NOCLDWAIT|SA_RESTART,
4d247a6cd3f69acbc5a09e8ac7e4fbb50eaa3228Lennart Poettering };
4d247a6cd3f69acbc5a09e8ac7e4fbb50eaa3228Lennart Poettering pid_t pid;
4d247a6cd3f69acbc5a09e8ac7e4fbb50eaa3228Lennart Poettering
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering log_info("Executing crash shell in 10s...");
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering sleep(10);
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering /* Let the kernel reap children for us */
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering assert_se(sigaction(SIGCHLD, &sa, NULL) == 0);
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering pid = fork();
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering if (pid < 0)
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering log_error("Failed to fork off crash shell: %m");
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering else if (pid == 0) {
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering make_console_stdio();
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering execl("/bin/sh", "/bin/sh", NULL);
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering
105e151299dc1208855380be2b22d0db2d66ebc6Lennart Poettering log_error("execl() failed: %m");
801ad6a6a9cd8fbd58b9f9c27f20dbb3c87d47ddLennart Poettering _exit(1);
801ad6a6a9cd8fbd58b9f9c27f20dbb3c87d47ddLennart Poettering }
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering log_info("Successfully spawned crash shell as pid %lu.", (unsigned long) pid);
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering }
4d247a6cd3f69acbc5a09e8ac7e4fbb50eaa3228Lennart Poettering
4d247a6cd3f69acbc5a09e8ac7e4fbb50eaa3228Lennart Poettering log_info("Freezing execution.");
4d247a6cd3f69acbc5a09e8ac7e4fbb50eaa3228Lennart Poettering freeze();
801ad6a6a9cd8fbd58b9f9c27f20dbb3c87d47ddLennart Poettering}
801ad6a6a9cd8fbd58b9f9c27f20dbb3c87d47ddLennart Poettering
801ad6a6a9cd8fbd58b9f9c27f20dbb3c87d47ddLennart Poetteringstatic void install_crash_handler(void) {
801ad6a6a9cd8fbd58b9f9c27f20dbb3c87d47ddLennart Poettering struct sigaction sa = {
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering .sa_handler = crash,
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering .sa_flags = SA_NODEFER,
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering };
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering sigaction_many(&sa, SIGNALS_CRASH_HANDLER, -1);
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering}
801ad6a6a9cd8fbd58b9f9c27f20dbb3c87d47ddLennart Poettering
801ad6a6a9cd8fbd58b9f9c27f20dbb3c87d47ddLennart Poetteringstatic int console_setup(bool do_reset) {
801ad6a6a9cd8fbd58b9f9c27f20dbb3c87d47ddLennart Poettering int tty_fd, r;
801ad6a6a9cd8fbd58b9f9c27f20dbb3c87d47ddLennart Poettering
801ad6a6a9cd8fbd58b9f9c27f20dbb3c87d47ddLennart Poettering /* If we are init, we connect stdin/stdout/stderr to /dev/null
801ad6a6a9cd8fbd58b9f9c27f20dbb3c87d47ddLennart Poettering * and make sure we don't have a controlling tty. */
801ad6a6a9cd8fbd58b9f9c27f20dbb3c87d47ddLennart Poettering
801ad6a6a9cd8fbd58b9f9c27f20dbb3c87d47ddLennart Poettering release_terminal();
801ad6a6a9cd8fbd58b9f9c27f20dbb3c87d47ddLennart Poettering
801ad6a6a9cd8fbd58b9f9c27f20dbb3c87d47ddLennart Poettering if (!do_reset)
801ad6a6a9cd8fbd58b9f9c27f20dbb3c87d47ddLennart Poettering return 0;
801ad6a6a9cd8fbd58b9f9c27f20dbb3c87d47ddLennart Poettering
801ad6a6a9cd8fbd58b9f9c27f20dbb3c87d47ddLennart Poettering tty_fd = open_terminal("/dev/console", O_WRONLY|O_NOCTTY|O_CLOEXEC);
801ad6a6a9cd8fbd58b9f9c27f20dbb3c87d47ddLennart Poettering if (tty_fd < 0) {
801ad6a6a9cd8fbd58b9f9c27f20dbb3c87d47ddLennart Poettering log_error("Failed to open /dev/console: %s", strerror(-tty_fd));
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering return -tty_fd;
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering }
5d27351f8546530cf779847b0b04b0172c09f9d0Tom Gundersen
801ad6a6a9cd8fbd58b9f9c27f20dbb3c87d47ddLennart Poettering /* We don't want to force text mode.
801ad6a6a9cd8fbd58b9f9c27f20dbb3c87d47ddLennart Poettering * plymouth may be showing pictures already from initrd. */
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering r = reset_terminal_fd(tty_fd, false);
5d27351f8546530cf779847b0b04b0172c09f9d0Tom Gundersen if (r < 0)
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering log_error("Failed to reset /dev/console: %s", strerror(-r));
5d27351f8546530cf779847b0b04b0172c09f9d0Tom Gundersen
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering close_nointr_nofail(tty_fd);
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering return r;
5d27351f8546530cf779847b0b04b0172c09f9d0Tom Gundersen}
5d27351f8546530cf779847b0b04b0172c09f9d0Tom Gundersen
5d27351f8546530cf779847b0b04b0172c09f9d0Tom Gundersenstatic int set_default_unit(const char *u) {
5d27351f8546530cf779847b0b04b0172c09f9d0Tom Gundersen char *c;
58db254ade4fb2ef77de68f28c4f13814819f6a1Lennart Poettering
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering assert(u);
801ad6a6a9cd8fbd58b9f9c27f20dbb3c87d47ddLennart Poettering
801ad6a6a9cd8fbd58b9f9c27f20dbb3c87d47ddLennart Poettering c = strdup(u);
801ad6a6a9cd8fbd58b9f9c27f20dbb3c87d47ddLennart Poettering if (!c)
801ad6a6a9cd8fbd58b9f9c27f20dbb3c87d47ddLennart Poettering return -ENOMEM;
801ad6a6a9cd8fbd58b9f9c27f20dbb3c87d47ddLennart Poettering
801ad6a6a9cd8fbd58b9f9c27f20dbb3c87d47ddLennart Poettering free(arg_default_unit);
801ad6a6a9cd8fbd58b9f9c27f20dbb3c87d47ddLennart Poettering arg_default_unit = c;
801ad6a6a9cd8fbd58b9f9c27f20dbb3c87d47ddLennart Poettering
801ad6a6a9cd8fbd58b9f9c27f20dbb3c87d47ddLennart Poettering return 0;
801ad6a6a9cd8fbd58b9f9c27f20dbb3c87d47ddLennart Poettering}
801ad6a6a9cd8fbd58b9f9c27f20dbb3c87d47ddLennart Poettering
5d27351f8546530cf779847b0b04b0172c09f9d0Tom Gundersenstatic int parse_proc_cmdline_word(const char *word) {
5d27351f8546530cf779847b0b04b0172c09f9d0Tom Gundersen
5d27351f8546530cf779847b0b04b0172c09f9d0Tom Gundersen static const char * const rlmap[] = {
5d27351f8546530cf779847b0b04b0172c09f9d0Tom Gundersen "emergency", SPECIAL_EMERGENCY_TARGET,
801ad6a6a9cd8fbd58b9f9c27f20dbb3c87d47ddLennart Poettering "-b", SPECIAL_EMERGENCY_TARGET,
801ad6a6a9cd8fbd58b9f9c27f20dbb3c87d47ddLennart Poettering "single", SPECIAL_RESCUE_TARGET,
801ad6a6a9cd8fbd58b9f9c27f20dbb3c87d47ddLennart Poettering "-s", SPECIAL_RESCUE_TARGET,
547973dea7abd6c124ff6c79fe2bbe322a7314aeLennart Poettering "s", SPECIAL_RESCUE_TARGET,
547973dea7abd6c124ff6c79fe2bbe322a7314aeLennart Poettering "S", SPECIAL_RESCUE_TARGET,
547973dea7abd6c124ff6c79fe2bbe322a7314aeLennart Poettering "1", SPECIAL_RESCUE_TARGET,
547973dea7abd6c124ff6c79fe2bbe322a7314aeLennart Poettering "2", SPECIAL_RUNLEVEL2_TARGET,
547973dea7abd6c124ff6c79fe2bbe322a7314aeLennart Poettering "3", SPECIAL_RUNLEVEL3_TARGET,
547973dea7abd6c124ff6c79fe2bbe322a7314aeLennart Poettering "4", SPECIAL_RUNLEVEL4_TARGET,
547973dea7abd6c124ff6c79fe2bbe322a7314aeLennart Poettering "5", SPECIAL_RUNLEVEL5_TARGET,
547973dea7abd6c124ff6c79fe2bbe322a7314aeLennart Poettering };
65b200e70d01cb7c513114b602c9b5cf41348f77Lennart Poettering
547973dea7abd6c124ff6c79fe2bbe322a7314aeLennart Poettering assert(word);
801ad6a6a9cd8fbd58b9f9c27f20dbb3c87d47ddLennart Poettering
547973dea7abd6c124ff6c79fe2bbe322a7314aeLennart Poettering if (startswith(word, "systemd.unit=")) {
547973dea7abd6c124ff6c79fe2bbe322a7314aeLennart Poettering
547973dea7abd6c124ff6c79fe2bbe322a7314aeLennart Poettering if (!in_initrd())
0936416a1cba187dc76903dae73246f4cbddf312Lennart Poettering return set_default_unit(word + 13);
74b2466e14a1961bf3ac0e8a60cfaceec705bd59Lennart Poettering
74b2466e14a1961bf3ac0e8a60cfaceec705bd59Lennart Poettering } else if (startswith(word, "rd.systemd.unit=")) {
b826ab586c9e0a9c0d438a75c28cf3a8ab485929Tom Gundersen
322345fdb9865ef2477fba8e4bdde0e1183ef505Lennart Poettering if (in_initrd())
322345fdb9865ef2477fba8e4bdde0e1183ef505Lennart Poettering return set_default_unit(word + 16);
b826ab586c9e0a9c0d438a75c28cf3a8ab485929Tom Gundersen
322345fdb9865ef2477fba8e4bdde0e1183ef505Lennart Poettering } else if (startswith(word, "systemd.log_target=")) {
b826ab586c9e0a9c0d438a75c28cf3a8ab485929Tom Gundersen
b826ab586c9e0a9c0d438a75c28cf3a8ab485929Tom Gundersen if (log_set_target_from_string(word + 19) < 0)
b826ab586c9e0a9c0d438a75c28cf3a8ab485929Tom Gundersen log_warning("Failed to parse log target %s. Ignoring.", word + 19);
322345fdb9865ef2477fba8e4bdde0e1183ef505Lennart Poettering
322345fdb9865ef2477fba8e4bdde0e1183ef505Lennart Poettering } else if (startswith(word, "systemd.log_level=")) {
d5099efc47d4e6ac60816b5381a5f607ab03f06eMichal Schmidt
322345fdb9865ef2477fba8e4bdde0e1183ef505Lennart Poettering if (log_set_max_level_from_string(word + 18) < 0)
322345fdb9865ef2477fba8e4bdde0e1183ef505Lennart Poettering log_warning("Failed to parse log level %s. Ignoring.", word + 18);
322345fdb9865ef2477fba8e4bdde0e1183ef505Lennart Poettering
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering } else if (startswith(word, "systemd.log_color=")) {
322345fdb9865ef2477fba8e4bdde0e1183ef505Lennart Poettering
322345fdb9865ef2477fba8e4bdde0e1183ef505Lennart Poettering if (log_show_color_from_string(word + 18) < 0)
322345fdb9865ef2477fba8e4bdde0e1183ef505Lennart Poettering log_warning("Failed to parse log color setting %s. Ignoring.", word + 18);
322345fdb9865ef2477fba8e4bdde0e1183ef505Lennart Poettering
322345fdb9865ef2477fba8e4bdde0e1183ef505Lennart Poettering } else if (startswith(word, "systemd.log_location=")) {
322345fdb9865ef2477fba8e4bdde0e1183ef505Lennart Poettering
322345fdb9865ef2477fba8e4bdde0e1183ef505Lennart Poettering if (log_show_location_from_string(word + 21) < 0)
322345fdb9865ef2477fba8e4bdde0e1183ef505Lennart Poettering log_warning("Failed to parse log location setting %s. Ignoring.", word + 21);
322345fdb9865ef2477fba8e4bdde0e1183ef505Lennart Poettering
322345fdb9865ef2477fba8e4bdde0e1183ef505Lennart Poettering } else if (startswith(word, "systemd.dump_core=")) {
322345fdb9865ef2477fba8e4bdde0e1183ef505Lennart Poettering int r;
322345fdb9865ef2477fba8e4bdde0e1183ef505Lennart Poettering
322345fdb9865ef2477fba8e4bdde0e1183ef505Lennart Poettering if ((r = parse_boolean(word + 18)) < 0)
322345fdb9865ef2477fba8e4bdde0e1183ef505Lennart Poettering log_warning("Failed to parse dump core switch %s. Ignoring.", word + 18);
322345fdb9865ef2477fba8e4bdde0e1183ef505Lennart Poettering else
322345fdb9865ef2477fba8e4bdde0e1183ef505Lennart Poettering arg_dump_core = r;
d5099efc47d4e6ac60816b5381a5f607ab03f06eMichal Schmidt
d5099efc47d4e6ac60816b5381a5f607ab03f06eMichal Schmidt } else if (startswith(word, "systemd.crash_shell=")) {
d5099efc47d4e6ac60816b5381a5f607ab03f06eMichal Schmidt int r;
d5099efc47d4e6ac60816b5381a5f607ab03f06eMichal Schmidt
d5099efc47d4e6ac60816b5381a5f607ab03f06eMichal Schmidt if ((r = parse_boolean(word + 20)) < 0)
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering log_warning("Failed to parse crash shell switch %s. Ignoring.", word + 20);
d23a27a964748967e1ad20e86de869a753af555bTom Gundersen else
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering arg_crash_shell = r;
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering } else if (startswith(word, "systemd.confirm_spawn=")) {
6af47493de0ef2b66d4c3fbcdd4a2e12fec4bfbaLennart Poettering int r;
6af47493de0ef2b66d4c3fbcdd4a2e12fec4bfbaLennart Poettering
6af47493de0ef2b66d4c3fbcdd4a2e12fec4bfbaLennart Poettering if ((r = parse_boolean(word + 22)) < 0)
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering log_warning("Failed to parse confirm spawn switch %s. Ignoring.", word + 22);
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering else
d23a27a964748967e1ad20e86de869a753af555bTom Gundersen arg_confirm_spawn = r;
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering } else if (startswith(word, "systemd.crash_chvt=")) {
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering int k;
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering if (safe_atoi(word + 19, &k) < 0)
d23a27a964748967e1ad20e86de869a753af555bTom Gundersen log_warning("Failed to parse crash chvt switch %s. Ignoring.", word + 19);
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering else
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering arg_crash_chvt = k;
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering
c52a97b896c914e17ba5be73c0e806455fd9ad4dLennart Poettering } else if (startswith(word, "systemd.show_status=")) {
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering int r;
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering if ((r = parse_boolean(word + 20)) < 0)
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering log_warning("Failed to parse show status switch %s. Ignoring.", word + 20);
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering else
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering arg_show_status = r;
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering } else if (startswith(word, "systemd.default_standard_output=")) {
74b2466e14a1961bf3ac0e8a60cfaceec705bd59Lennart Poettering int r;
74b2466e14a1961bf3ac0e8a60cfaceec705bd59Lennart Poettering
74b2466e14a1961bf3ac0e8a60cfaceec705bd59Lennart Poettering if ((r = exec_output_from_string(word + 32)) < 0)
74b2466e14a1961bf3ac0e8a60cfaceec705bd59Lennart Poettering log_warning("Failed to parse default standard output switch %s. Ignoring.", word + 32);
74b2466e14a1961bf3ac0e8a60cfaceec705bd59Lennart Poettering else
74b2466e14a1961bf3ac0e8a60cfaceec705bd59Lennart Poettering arg_default_std_output = r;
74b2466e14a1961bf3ac0e8a60cfaceec705bd59Lennart Poettering } else if (startswith(word, "systemd.default_standard_error=")) {
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering int r;
ee3d6aff9bd73c1b23e29d1fa1fa6f7a1ef0533bLennart Poettering
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering if ((r = exec_output_from_string(word + 31)) < 0)
74b2466e14a1961bf3ac0e8a60cfaceec705bd59Lennart Poettering log_warning("Failed to parse default standard error switch %s. Ignoring.", word + 31);
74b2466e14a1961bf3ac0e8a60cfaceec705bd59Lennart Poettering else
74b2466e14a1961bf3ac0e8a60cfaceec705bd59Lennart Poettering arg_default_std_error = r;
8bf52d3d17d364438191077d0750b8b80b5dc53aLennart Poettering } else if (startswith(word, "systemd.setenv=")) {
8bf52d3d17d364438191077d0750b8b80b5dc53aLennart Poettering _cleanup_free_ char *cenv = NULL;
8bf52d3d17d364438191077d0750b8b80b5dc53aLennart Poettering char *eq;
8bf52d3d17d364438191077d0750b8b80b5dc53aLennart Poettering int r;
8bf52d3d17d364438191077d0750b8b80b5dc53aLennart Poettering
8bf52d3d17d364438191077d0750b8b80b5dc53aLennart Poettering cenv = strdup(word + 15);
8bf52d3d17d364438191077d0750b8b80b5dc53aLennart Poettering if (!cenv)
8bf52d3d17d364438191077d0750b8b80b5dc53aLennart Poettering return -ENOMEM;
8bf52d3d17d364438191077d0750b8b80b5dc53aLennart Poettering
8bf52d3d17d364438191077d0750b8b80b5dc53aLennart Poettering eq = strchr(cenv, '=');
74b2466e14a1961bf3ac0e8a60cfaceec705bd59Lennart Poettering if (!eq) {
74b2466e14a1961bf3ac0e8a60cfaceec705bd59Lennart Poettering if (!env_name_is_valid(cenv))
74b2466e14a1961bf3ac0e8a60cfaceec705bd59Lennart Poettering log_warning("Environment variable name '%s' is not valid. Ignoring.", cenv);
74b2466e14a1961bf3ac0e8a60cfaceec705bd59Lennart Poettering else {
74b2466e14a1961bf3ac0e8a60cfaceec705bd59Lennart Poettering r = unsetenv(cenv);
74b2466e14a1961bf3ac0e8a60cfaceec705bd59Lennart Poettering if (r < 0)
74b2466e14a1961bf3ac0e8a60cfaceec705bd59Lennart Poettering log_warning("Unsetting environment variable '%s' failed, ignoring: %m", cenv);
74b2466e14a1961bf3ac0e8a60cfaceec705bd59Lennart Poettering }
74b2466e14a1961bf3ac0e8a60cfaceec705bd59Lennart Poettering } else {
74b2466e14a1961bf3ac0e8a60cfaceec705bd59Lennart Poettering if (!env_assignment_is_valid(cenv))
74b2466e14a1961bf3ac0e8a60cfaceec705bd59Lennart Poettering log_warning("Environment variable assignment '%s' is not valid. Ignoring.", cenv);
74b2466e14a1961bf3ac0e8a60cfaceec705bd59Lennart Poettering else {
74b2466e14a1961bf3ac0e8a60cfaceec705bd59Lennart Poettering *eq = 0;
74b2466e14a1961bf3ac0e8a60cfaceec705bd59Lennart Poettering r = setenv(cenv, eq + 1, 1);
74b2466e14a1961bf3ac0e8a60cfaceec705bd59Lennart Poettering if (r < 0)
74b2466e14a1961bf3ac0e8a60cfaceec705bd59Lennart Poettering log_warning("Setting environment variable '%s=%s' failed, ignoring: %m", cenv, eq + 1);
74b2466e14a1961bf3ac0e8a60cfaceec705bd59Lennart Poettering }
74b2466e14a1961bf3ac0e8a60cfaceec705bd59Lennart Poettering }
74b2466e14a1961bf3ac0e8a60cfaceec705bd59Lennart Poettering
74b2466e14a1961bf3ac0e8a60cfaceec705bd59Lennart Poettering } else if (startswith(word, "systemd.") ||
74b2466e14a1961bf3ac0e8a60cfaceec705bd59Lennart Poettering (in_initrd() && startswith(word, "rd.systemd."))) {
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering
9de3e3294065e8697ff10130b53f274319cdcf6fZbigniew Jędrzejewski-Szmek const char *c;
9c92ce6d67f88beb31dd6555d12ae3f632218a39Lennart Poettering
9c92ce6d67f88beb31dd6555d12ae3f632218a39Lennart Poettering /* Ignore systemd.journald.xyz and friends */
9c92ce6d67f88beb31dd6555d12ae3f632218a39Lennart Poettering c = word;
9c92ce6d67f88beb31dd6555d12ae3f632218a39Lennart Poettering if (startswith(c, "rd."))
9c92ce6d67f88beb31dd6555d12ae3f632218a39Lennart Poettering c += 3;
9de3e3294065e8697ff10130b53f274319cdcf6fZbigniew Jędrzejewski-Szmek if (startswith(c, "systemd."))
9de3e3294065e8697ff10130b53f274319cdcf6fZbigniew Jędrzejewski-Szmek c += 8;
9de3e3294065e8697ff10130b53f274319cdcf6fZbigniew Jędrzejewski-Szmek if (c[strcspn(c, ".=")] != '.') {
8ac4e9e1e54397f6d1745c2a7a806132418c7da2Lennart Poettering
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering log_warning("Unknown kernel switch %s. Ignoring.", word);
9de3e3294065e8697ff10130b53f274319cdcf6fZbigniew Jędrzejewski-Szmek
9c92ce6d67f88beb31dd6555d12ae3f632218a39Lennart Poettering log_info("Supported kernel switches:\n"
9de3e3294065e8697ff10130b53f274319cdcf6fZbigniew Jędrzejewski-Szmek "systemd.unit=UNIT Default unit to start\n"
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering "rd.systemd.unit=UNIT Default unit to start when run in initrd\n"
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering "systemd.dump_core=0|1 Dump core on crash\n"
9de3e3294065e8697ff10130b53f274319cdcf6fZbigniew Jędrzejewski-Szmek "systemd.crash_shell=0|1 Run shell on crash\n"
9c92ce6d67f88beb31dd6555d12ae3f632218a39Lennart Poettering "systemd.crash_chvt=N Change to VT #N on crash\n"
9de3e3294065e8697ff10130b53f274319cdcf6fZbigniew Jędrzejewski-Szmek "systemd.confirm_spawn=0|1 Confirm every process spawn\n"
9c92ce6d67f88beb31dd6555d12ae3f632218a39Lennart Poettering "systemd.show_status=0|1 Show status updates on the console during bootup\n"
2001c80560e3dae69e14fd994d3978c187af48b8Lennart Poettering "systemd.log_target=console|kmsg|journal|journal-or-kmsg|syslog|syslog-or-kmsg|null\n"
9de3e3294065e8697ff10130b53f274319cdcf6fZbigniew Jędrzejewski-Szmek " Log target\n"
9c92ce6d67f88beb31dd6555d12ae3f632218a39Lennart Poettering "systemd.log_level=LEVEL Log level\n"
9de3e3294065e8697ff10130b53f274319cdcf6fZbigniew Jędrzejewski-Szmek "systemd.log_color=0|1 Highlight important log messages\n"
7e8e0422aeb16f2a09a40546c61df753d10029b6Lennart Poettering "systemd.log_location=0|1 Include code location in log messages\n"
7e8e0422aeb16f2a09a40546c61df753d10029b6Lennart Poettering "systemd.default_standard_output=null|tty|syslog|syslog+console|kmsg|kmsg+console|journal|journal+console\n"
9de3e3294065e8697ff10130b53f274319cdcf6fZbigniew Jędrzejewski-Szmek " Set default log output for services\n"
9c92ce6d67f88beb31dd6555d12ae3f632218a39Lennart Poettering "systemd.default_standard_error=null|tty|syslog|syslog+console|kmsg|kmsg+console|journal|journal+console\n"
9de3e3294065e8697ff10130b53f274319cdcf6fZbigniew Jędrzejewski-Szmek " Set default log error output for services\n"
946c70944ebdf428ffeb9991a7449edbd4011461Zbigniew Jędrzejewski-Szmek "systemd.setenv=ASSIGNMENT Set an environment variable for all spawned processes\n");
9de3e3294065e8697ff10130b53f274319cdcf6fZbigniew Jędrzejewski-Szmek }
9c92ce6d67f88beb31dd6555d12ae3f632218a39Lennart Poettering
abf126a355e2f2b62b6c51ab3bb37895d1e3eee7Tom Gundersen } else if (streq(word, "quiet"))
abf126a355e2f2b62b6c51ab3bb37895d1e3eee7Tom Gundersen arg_show_status = false;
abf126a355e2f2b62b6c51ab3bb37895d1e3eee7Tom Gundersen else if (streq(word, "debug"))
abf126a355e2f2b62b6c51ab3bb37895d1e3eee7Tom Gundersen log_set_max_level(LOG_DEBUG);
42cc2eebb01056beb7acd3ecfe8e533558237f84Lennart Poettering else if (!in_initrd()) {
549c1a2564b56f2bb38f1203d59c747ea15817f3Tom Gundersen unsigned i;
42cc2eebb01056beb7acd3ecfe8e533558237f84Lennart Poettering
42cc2eebb01056beb7acd3ecfe8e533558237f84Lennart Poettering /* SysV compatibility */
8db0d2f5c37e7e8f5bfce016cfdad7947a3ea939Zbigniew Jędrzejewski-Szmek for (i = 0; i < ELEMENTSOF(rlmap); i += 2)
8db0d2f5c37e7e8f5bfce016cfdad7947a3ea939Zbigniew Jędrzejewski-Szmek if (streq(word, rlmap[i]))
8db0d2f5c37e7e8f5bfce016cfdad7947a3ea939Zbigniew Jędrzejewski-Szmek return set_default_unit(rlmap[i+1]);
8db0d2f5c37e7e8f5bfce016cfdad7947a3ea939Zbigniew Jędrzejewski-Szmek }
151226ab4bf276d60d51864330a99f886b923697Zbigniew Jędrzejewski-Szmek
151226ab4bf276d60d51864330a99f886b923697Zbigniew Jędrzejewski-Szmek return 0;
151226ab4bf276d60d51864330a99f886b923697Zbigniew Jędrzejewski-Szmek}
151226ab4bf276d60d51864330a99f886b923697Zbigniew Jędrzejewski-Szmek
151226ab4bf276d60d51864330a99f886b923697Zbigniew Jędrzejewski-Szmek#define DEFINE_SETTER(name, func, descr) \
50f1e641a93cacfc693b0c3d300bee5df0c8c460Tom Gundersen static int name(const char *unit, \
50f1e641a93cacfc693b0c3d300bee5df0c8c460Tom Gundersen const char *filename, \
50f1e641a93cacfc693b0c3d300bee5df0c8c460Tom Gundersen unsigned line, \
50f1e641a93cacfc693b0c3d300bee5df0c8c460Tom Gundersen const char *section, \
50f1e641a93cacfc693b0c3d300bee5df0c8c460Tom Gundersen const char *lvalue, \
5d45a8808431987c370706d365fb0cc95cf03d52Tom Gundersen int ltype, \
5d45a8808431987c370706d365fb0cc95cf03d52Tom Gundersen const char *rvalue, \
5d45a8808431987c370706d365fb0cc95cf03d52Tom Gundersen void *data, \
5d45a8808431987c370706d365fb0cc95cf03d52Tom Gundersen void *userdata) { \
5d45a8808431987c370706d365fb0cc95cf03d52Tom Gundersen \
5d45a8808431987c370706d365fb0cc95cf03d52Tom Gundersen int r; \
0dae31d468b1a0e22d98921f7b0dbd92fd217167Zbigniew Jędrzejewski-Szmek \
9de3e3294065e8697ff10130b53f274319cdcf6fZbigniew Jędrzejewski-Szmek assert(filename); \
9de3e3294065e8697ff10130b53f274319cdcf6fZbigniew Jędrzejewski-Szmek assert(lvalue); \
9de3e3294065e8697ff10130b53f274319cdcf6fZbigniew Jędrzejewski-Szmek assert(rvalue); \
9c92ce6d67f88beb31dd6555d12ae3f632218a39Lennart Poettering \
9de3e3294065e8697ff10130b53f274319cdcf6fZbigniew Jędrzejewski-Szmek r = func(rvalue); \
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering if (r < 0) \
9de3e3294065e8697ff10130b53f274319cdcf6fZbigniew Jędrzejewski-Szmek log_syntax(unit, LOG_ERR, filename, line, -r, \
322345fdb9865ef2477fba8e4bdde0e1183ef505Lennart Poettering "Invalid " descr "'%s': %s", \
a8812dd7f161a3e459c1730ac92ff2bbc9986ff1Lennart Poettering rvalue, strerror(-r)); \
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering \
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poettering return 0; \
322345fdb9865ef2477fba8e4bdde0e1183ef505Lennart Poettering }
7b50eb2efa122200e39646c19a29abab302f7d24Lennart Poettering
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart PoetteringDEFINE_SETTER(config_parse_level2, log_set_max_level_from_string, "log level")
322345fdb9865ef2477fba8e4bdde0e1183ef505Lennart PoetteringDEFINE_SETTER(config_parse_target, log_set_target_from_string, "target")
322345fdb9865ef2477fba8e4bdde0e1183ef505Lennart PoetteringDEFINE_SETTER(config_parse_color, log_show_color_from_string, "color" )
322345fdb9865ef2477fba8e4bdde0e1183ef505Lennart PoetteringDEFINE_SETTER(config_parse_location, log_show_location_from_string, "location")
322345fdb9865ef2477fba8e4bdde0e1183ef505Lennart Poettering
623a4c97b9175f95c4b1c6fc34e36c56f1e4ddbfLennart Poettering
623a4c97b9175f95c4b1c6fc34e36c56f1e4ddbfLennart Poetteringstatic int config_parse_cpu_affinity2(const char *unit,
623a4c97b9175f95c4b1c6fc34e36c56f1e4ddbfLennart Poettering const char *filename,
623a4c97b9175f95c4b1c6fc34e36c56f1e4ddbfLennart Poettering unsigned line,
623a4c97b9175f95c4b1c6fc34e36c56f1e4ddbfLennart Poettering const char *section,
623a4c97b9175f95c4b1c6fc34e36c56f1e4ddbfLennart Poettering const char *lvalue,
623a4c97b9175f95c4b1c6fc34e36c56f1e4ddbfLennart Poettering int ltype,
623a4c97b9175f95c4b1c6fc34e36c56f1e4ddbfLennart Poettering const char *rvalue,
623a4c97b9175f95c4b1c6fc34e36c56f1e4ddbfLennart Poettering void *data,
623a4c97b9175f95c4b1c6fc34e36c56f1e4ddbfLennart Poettering void *userdata) {
623a4c97b9175f95c4b1c6fc34e36c56f1e4ddbfLennart Poettering
623a4c97b9175f95c4b1c6fc34e36c56f1e4ddbfLennart Poettering char *w;
623a4c97b9175f95c4b1c6fc34e36c56f1e4ddbfLennart Poettering size_t l;
623a4c97b9175f95c4b1c6fc34e36c56f1e4ddbfLennart Poettering char *state;
623a4c97b9175f95c4b1c6fc34e36c56f1e4ddbfLennart Poettering cpu_set_t *c = NULL;
623a4c97b9175f95c4b1c6fc34e36c56f1e4ddbfLennart Poettering unsigned ncpus = 0;
623a4c97b9175f95c4b1c6fc34e36c56f1e4ddbfLennart Poettering
623a4c97b9175f95c4b1c6fc34e36c56f1e4ddbfLennart Poettering assert(filename);
623a4c97b9175f95c4b1c6fc34e36c56f1e4ddbfLennart Poettering assert(lvalue);
623a4c97b9175f95c4b1c6fc34e36c56f1e4ddbfLennart Poettering assert(rvalue);
623a4c97b9175f95c4b1c6fc34e36c56f1e4ddbfLennart Poettering
623a4c97b9175f95c4b1c6fc34e36c56f1e4ddbfLennart Poettering FOREACH_WORD_QUOTED(w, l, rvalue, state) {
623a4c97b9175f95c4b1c6fc34e36c56f1e4ddbfLennart Poettering char *t;
623a4c97b9175f95c4b1c6fc34e36c56f1e4ddbfLennart Poettering int r;
623a4c97b9175f95c4b1c6fc34e36c56f1e4ddbfLennart Poettering unsigned cpu;
623a4c97b9175f95c4b1c6fc34e36c56f1e4ddbfLennart Poettering
623a4c97b9175f95c4b1c6fc34e36c56f1e4ddbfLennart Poettering if (!(t = strndup(w, l)))
623a4c97b9175f95c4b1c6fc34e36c56f1e4ddbfLennart Poettering return log_oom();
623a4c97b9175f95c4b1c6fc34e36c56f1e4ddbfLennart Poettering
623a4c97b9175f95c4b1c6fc34e36c56f1e4ddbfLennart Poettering r = safe_atou(t, &cpu);
623a4c97b9175f95c4b1c6fc34e36c56f1e4ddbfLennart Poettering free(t);
623a4c97b9175f95c4b1c6fc34e36c56f1e4ddbfLennart Poettering
623a4c97b9175f95c4b1c6fc34e36c56f1e4ddbfLennart Poettering if (!c)
623a4c97b9175f95c4b1c6fc34e36c56f1e4ddbfLennart Poettering if (!(c = cpu_set_malloc(&ncpus)))
78c6a153c47f8d597c827bdcaf8c4e42ac87f738Lennart Poettering return log_oom();
78c6a153c47f8d597c827bdcaf8c4e42ac87f738Lennart Poettering
78c6a153c47f8d597c827bdcaf8c4e42ac87f738Lennart Poettering if (r < 0 || cpu >= ncpus) {
78c6a153c47f8d597c827bdcaf8c4e42ac87f738Lennart Poettering log_syntax(unit, LOG_ERR, filename, line, -r,
78c6a153c47f8d597c827bdcaf8c4e42ac87f738Lennart Poettering "Failed to parse CPU affinity '%s'", rvalue);
78c6a153c47f8d597c827bdcaf8c4e42ac87f738Lennart Poettering CPU_FREE(c);
78c6a153c47f8d597c827bdcaf8c4e42ac87f738Lennart Poettering return -EBADMSG;
78c6a153c47f8d597c827bdcaf8c4e42ac87f738Lennart Poettering }
78c6a153c47f8d597c827bdcaf8c4e42ac87f738Lennart Poettering
78c6a153c47f8d597c827bdcaf8c4e42ac87f738Lennart Poettering CPU_SET_S(cpu, CPU_ALLOC_SIZE(ncpus), c);
78c6a153c47f8d597c827bdcaf8c4e42ac87f738Lennart Poettering }
78c6a153c47f8d597c827bdcaf8c4e42ac87f738Lennart Poettering
78c6a153c47f8d597c827bdcaf8c4e42ac87f738Lennart Poettering if (c) {
78c6a153c47f8d597c827bdcaf8c4e42ac87f738Lennart Poettering if (sched_setaffinity(0, CPU_ALLOC_SIZE(ncpus), c) < 0)
78c6a153c47f8d597c827bdcaf8c4e42ac87f738Lennart Poettering log_warning_unit(unit, "Failed to set CPU affinity: %m");
78c6a153c47f8d597c827bdcaf8c4e42ac87f738Lennart Poettering
78c6a153c47f8d597c827bdcaf8c4e42ac87f738Lennart Poettering CPU_FREE(c);
78c6a153c47f8d597c827bdcaf8c4e42ac87f738Lennart Poettering }
78c6a153c47f8d597c827bdcaf8c4e42ac87f738Lennart Poettering
78c6a153c47f8d597c827bdcaf8c4e42ac87f738Lennart Poettering return 0;
78c6a153c47f8d597c827bdcaf8c4e42ac87f738Lennart Poettering}
78c6a153c47f8d597c827bdcaf8c4e42ac87f738Lennart Poettering
78c6a153c47f8d597c827bdcaf8c4e42ac87f738Lennart Poetteringstatic void strv_free_free(char ***l) {
78c6a153c47f8d597c827bdcaf8c4e42ac87f738Lennart Poettering char ***i;
78c6a153c47f8d597c827bdcaf8c4e42ac87f738Lennart Poettering
78c6a153c47f8d597c827bdcaf8c4e42ac87f738Lennart Poettering if (!l)
78c6a153c47f8d597c827bdcaf8c4e42ac87f738Lennart Poettering return;
78c6a153c47f8d597c827bdcaf8c4e42ac87f738Lennart Poettering
78c6a153c47f8d597c827bdcaf8c4e42ac87f738Lennart Poettering for (i = l; *i; i++)
78c6a153c47f8d597c827bdcaf8c4e42ac87f738Lennart Poettering strv_free(*i);
322345fdb9865ef2477fba8e4bdde0e1183ef505Lennart Poettering
322345fdb9865ef2477fba8e4bdde0e1183ef505Lennart Poettering free(l);
322345fdb9865ef2477fba8e4bdde0e1183ef505Lennart Poettering}
322345fdb9865ef2477fba8e4bdde0e1183ef505Lennart Poettering
322345fdb9865ef2477fba8e4bdde0e1183ef505Lennart Poetteringstatic void free_join_controllers(void) {
322345fdb9865ef2477fba8e4bdde0e1183ef505Lennart Poettering strv_free_free(arg_join_controllers);
4d247a6cd3f69acbc5a09e8ac7e4fbb50eaa3228Lennart Poettering arg_join_controllers = NULL;
4d247a6cd3f69acbc5a09e8ac7e4fbb50eaa3228Lennart Poettering}
4d247a6cd3f69acbc5a09e8ac7e4fbb50eaa3228Lennart Poettering
faa133f3aa7a18f26563dc5d6b95898cb315c37aLennart Poetteringstatic int config_parse_join_controllers(const char *unit,
322345fdb9865ef2477fba8e4bdde0e1183ef505Lennart Poettering const char *filename,
322345fdb9865ef2477fba8e4bdde0e1183ef505Lennart Poettering unsigned line,
322345fdb9865ef2477fba8e4bdde0e1183ef505Lennart Poettering const char *section,
fd0b4602f6332c3f1660eb208c8f5c719709a009Lennart Poettering const char *lvalue,
fd0b4602f6332c3f1660eb208c8f5c719709a009Lennart Poettering int ltype,
fd0b4602f6332c3f1660eb208c8f5c719709a009Lennart Poettering const char *rvalue,
fd0b4602f6332c3f1660eb208c8f5c719709a009Lennart Poettering void *data,
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering void *userdata) {
9c92ce6d67f88beb31dd6555d12ae3f632218a39Lennart Poettering
9c92ce6d67f88beb31dd6555d12ae3f632218a39Lennart Poettering unsigned n = 0;
9c92ce6d67f88beb31dd6555d12ae3f632218a39Lennart Poettering char *state, *w;
9c92ce6d67f88beb31dd6555d12ae3f632218a39Lennart Poettering size_t length;
9c92ce6d67f88beb31dd6555d12ae3f632218a39Lennart Poettering
9c92ce6d67f88beb31dd6555d12ae3f632218a39Lennart Poettering assert(filename);
9c92ce6d67f88beb31dd6555d12ae3f632218a39Lennart Poettering assert(lvalue);
9c92ce6d67f88beb31dd6555d12ae3f632218a39Lennart Poettering assert(rvalue);
9c92ce6d67f88beb31dd6555d12ae3f632218a39Lennart Poettering
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering free_join_controllers();
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering FOREACH_WORD_QUOTED(w, length, rvalue, state) {
8ac4e9e1e54397f6d1745c2a7a806132418c7da2Lennart Poettering char *s, **l;
322345fdb9865ef2477fba8e4bdde0e1183ef505Lennart Poettering
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering s = strndup(w, length);
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering if (!s)
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering return log_oom();
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering l = strv_split(s, ",");
9de3e3294065e8697ff10130b53f274319cdcf6fZbigniew Jędrzejewski-Szmek free(s);
0f84a72e3c0f58d71cff2121e6df1611eaf9c9eaDavid Herrmann
2001c80560e3dae69e14fd994d3978c187af48b8Lennart Poettering strv_uniq(l);
2e276efc7b0398a3086629a52970bdd4ab7252f9Zbigniew Jędrzejewski-Szmek
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering if (strv_length(l) <= 1) {
322345fdb9865ef2477fba8e4bdde0e1183ef505Lennart Poettering strv_free(l);
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering continue;
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering }
322345fdb9865ef2477fba8e4bdde0e1183ef505Lennart Poettering
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering if (!arg_join_controllers) {
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering arg_join_controllers = new(char**, 2);
7e8e0422aeb16f2a09a40546c61df753d10029b6Lennart Poettering if (!arg_join_controllers) {
7e8e0422aeb16f2a09a40546c61df753d10029b6Lennart Poettering strv_free(l);
7e8e0422aeb16f2a09a40546c61df753d10029b6Lennart Poettering return log_oom();
7e8e0422aeb16f2a09a40546c61df753d10029b6Lennart Poettering }
7e8e0422aeb16f2a09a40546c61df753d10029b6Lennart Poettering
7e8e0422aeb16f2a09a40546c61df753d10029b6Lennart Poettering arg_join_controllers[0] = l;
7e8e0422aeb16f2a09a40546c61df753d10029b6Lennart Poettering arg_join_controllers[1] = NULL;
7e8e0422aeb16f2a09a40546c61df753d10029b6Lennart Poettering
7e8e0422aeb16f2a09a40546c61df753d10029b6Lennart Poettering n = 1;
7e8e0422aeb16f2a09a40546c61df753d10029b6Lennart Poettering } else {
7e8e0422aeb16f2a09a40546c61df753d10029b6Lennart Poettering char ***a;
7e8e0422aeb16f2a09a40546c61df753d10029b6Lennart Poettering char ***t;
9c92ce6d67f88beb31dd6555d12ae3f632218a39Lennart Poettering
946c70944ebdf428ffeb9991a7449edbd4011461Zbigniew Jędrzejewski-Szmek t = new0(char**, n+2);
946c70944ebdf428ffeb9991a7449edbd4011461Zbigniew Jędrzejewski-Szmek if (!t) {
946c70944ebdf428ffeb9991a7449edbd4011461Zbigniew Jędrzejewski-Szmek strv_free(l);
946c70944ebdf428ffeb9991a7449edbd4011461Zbigniew Jędrzejewski-Szmek return log_oom();
946c70944ebdf428ffeb9991a7449edbd4011461Zbigniew Jędrzejewski-Szmek }
946c70944ebdf428ffeb9991a7449edbd4011461Zbigniew Jędrzejewski-Szmek
0dae31d468b1a0e22d98921f7b0dbd92fd217167Zbigniew Jędrzejewski-Szmek n = 0;
0dae31d468b1a0e22d98921f7b0dbd92fd217167Zbigniew Jędrzejewski-Szmek
0dae31d468b1a0e22d98921f7b0dbd92fd217167Zbigniew Jędrzejewski-Szmek for (a = arg_join_controllers; *a; a++) {
0dae31d468b1a0e22d98921f7b0dbd92fd217167Zbigniew Jędrzejewski-Szmek
0dae31d468b1a0e22d98921f7b0dbd92fd217167Zbigniew Jędrzejewski-Szmek if (strv_overlap(*a, l)) {
0dae31d468b1a0e22d98921f7b0dbd92fd217167Zbigniew Jędrzejewski-Szmek char **c;
0dae31d468b1a0e22d98921f7b0dbd92fd217167Zbigniew Jędrzejewski-Szmek
0dae31d468b1a0e22d98921f7b0dbd92fd217167Zbigniew Jędrzejewski-Szmek c = strv_merge(*a, l);
0dae31d468b1a0e22d98921f7b0dbd92fd217167Zbigniew Jędrzejewski-Szmek if (!c) {
0dae31d468b1a0e22d98921f7b0dbd92fd217167Zbigniew Jędrzejewski-Szmek strv_free(l);
abf126a355e2f2b62b6c51ab3bb37895d1e3eee7Tom Gundersen strv_free_free(t);
abf126a355e2f2b62b6c51ab3bb37895d1e3eee7Tom Gundersen return log_oom();
abf126a355e2f2b62b6c51ab3bb37895d1e3eee7Tom Gundersen }
abf126a355e2f2b62b6c51ab3bb37895d1e3eee7Tom Gundersen
abf126a355e2f2b62b6c51ab3bb37895d1e3eee7Tom Gundersen strv_free(l);
abf126a355e2f2b62b6c51ab3bb37895d1e3eee7Tom Gundersen l = c;
abf126a355e2f2b62b6c51ab3bb37895d1e3eee7Tom Gundersen } else {
42cc2eebb01056beb7acd3ecfe8e533558237f84Lennart Poettering char **c;
42cc2eebb01056beb7acd3ecfe8e533558237f84Lennart Poettering
42cc2eebb01056beb7acd3ecfe8e533558237f84Lennart Poettering c = strv_copy(*a);
549c1a2564b56f2bb38f1203d59c747ea15817f3Tom Gundersen if (!c) {
549c1a2564b56f2bb38f1203d59c747ea15817f3Tom Gundersen strv_free(l);
42cc2eebb01056beb7acd3ecfe8e533558237f84Lennart Poettering strv_free_free(t);
8db0d2f5c37e7e8f5bfce016cfdad7947a3ea939Zbigniew Jędrzejewski-Szmek return log_oom();
f91dc2400dc33e9a0745ecaaef7489af116dca38Lennart Poettering }
f91dc2400dc33e9a0745ecaaef7489af116dca38Lennart Poettering
8db0d2f5c37e7e8f5bfce016cfdad7947a3ea939Zbigniew Jędrzejewski-Szmek t[n++] = c;
8db0d2f5c37e7e8f5bfce016cfdad7947a3ea939Zbigniew Jędrzejewski-Szmek }
8db0d2f5c37e7e8f5bfce016cfdad7947a3ea939Zbigniew Jędrzejewski-Szmek }
8db0d2f5c37e7e8f5bfce016cfdad7947a3ea939Zbigniew Jędrzejewski-Szmek
151226ab4bf276d60d51864330a99f886b923697Zbigniew Jędrzejewski-Szmek t[n++] = strv_uniq(l);
151226ab4bf276d60d51864330a99f886b923697Zbigniew Jędrzejewski-Szmek
03664a62914782dbd8f069bbcf8a0c8ca1df7010Lukas Nykryn strv_free_free(arg_join_controllers);
03664a62914782dbd8f069bbcf8a0c8ca1df7010Lukas Nykryn arg_join_controllers = t;
03664a62914782dbd8f069bbcf8a0c8ca1df7010Lukas Nykryn }
03664a62914782dbd8f069bbcf8a0c8ca1df7010Lukas Nykryn }
03664a62914782dbd8f069bbcf8a0c8ca1df7010Lukas Nykryn
03664a62914782dbd8f069bbcf8a0c8ca1df7010Lukas Nykryn return 0;
03664a62914782dbd8f069bbcf8a0c8ca1df7010Lukas Nykryn}
151226ab4bf276d60d51864330a99f886b923697Zbigniew Jędrzejewski-Szmek
151226ab4bf276d60d51864330a99f886b923697Zbigniew Jędrzejewski-Szmekstatic int parse_config_file(void) {
151226ab4bf276d60d51864330a99f886b923697Zbigniew Jędrzejewski-Szmek
151226ab4bf276d60d51864330a99f886b923697Zbigniew Jędrzejewski-Szmek const ConfigTableItem items[] = {
151226ab4bf276d60d51864330a99f886b923697Zbigniew Jędrzejewski-Szmek { "Manager", "LogLevel", config_parse_level2, 0, NULL },
151226ab4bf276d60d51864330a99f886b923697Zbigniew Jędrzejewski-Szmek { "Manager", "LogTarget", config_parse_target, 0, NULL },
50f1e641a93cacfc693b0c3d300bee5df0c8c460Tom Gundersen { "Manager", "LogColor", config_parse_color, 0, NULL },
50f1e641a93cacfc693b0c3d300bee5df0c8c460Tom Gundersen { "Manager", "LogLocation", config_parse_location, 0, NULL },
50f1e641a93cacfc693b0c3d300bee5df0c8c460Tom Gundersen { "Manager", "DumpCore", config_parse_bool, 0, &arg_dump_core },
50f1e641a93cacfc693b0c3d300bee5df0c8c460Tom Gundersen { "Manager", "CrashShell", config_parse_bool, 0, &arg_crash_shell },
5d45a8808431987c370706d365fb0cc95cf03d52Tom Gundersen { "Manager", "ShowStatus", config_parse_bool, 0, &arg_show_status },
5d45a8808431987c370706d365fb0cc95cf03d52Tom Gundersen { "Manager", "CrashChVT", config_parse_int, 0, &arg_crash_chvt },
5d45a8808431987c370706d365fb0cc95cf03d52Tom Gundersen { "Manager", "CPUAffinity", config_parse_cpu_affinity2, 0, NULL },
5d45a8808431987c370706d365fb0cc95cf03d52Tom Gundersen { "Manager", "DefaultStandardOutput", config_parse_output, 0, &arg_default_std_output },
5d45a8808431987c370706d365fb0cc95cf03d52Tom Gundersen { "Manager", "DefaultStandardError", config_parse_output, 0, &arg_default_std_error },
5d45a8808431987c370706d365fb0cc95cf03d52Tom Gundersen { "Manager", "JoinControllers", config_parse_join_controllers, 0, &arg_join_controllers },
5d45a8808431987c370706d365fb0cc95cf03d52Tom Gundersen { "Manager", "RuntimeWatchdogSec", config_parse_sec, 0, &arg_runtime_watchdog },
5d45a8808431987c370706d365fb0cc95cf03d52Tom Gundersen { "Manager", "ShutdownWatchdogSec", config_parse_sec, 0, &arg_shutdown_watchdog },
5d45a8808431987c370706d365fb0cc95cf03d52Tom Gundersen { "Manager", "CapabilityBoundingSet", config_parse_bounding_set, 0, &arg_capability_bounding_set_drop },
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering { "Manager", "TimerSlackNSec", config_parse_nsec, 0, &arg_timer_slack_nsec },
322345fdb9865ef2477fba8e4bdde0e1183ef505Lennart Poettering { "Manager", "DefaultEnvironment", config_parse_environ, 0, &arg_default_environment },
322345fdb9865ef2477fba8e4bdde0e1183ef505Lennart Poettering { "Manager", "DefaultLimitCPU", config_parse_limit, 0, &arg_default_rlimit[RLIMIT_CPU]},
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering { "Manager", "DefaultLimitFSIZE", config_parse_limit, 0, &arg_default_rlimit[RLIMIT_FSIZE]},
322345fdb9865ef2477fba8e4bdde0e1183ef505Lennart Poettering { "Manager", "DefaultLimitDATA", config_parse_limit, 0, &arg_default_rlimit[RLIMIT_DATA]},
322345fdb9865ef2477fba8e4bdde0e1183ef505Lennart Poettering { "Manager", "DefaultLimitSTACK", config_parse_limit, 0, &arg_default_rlimit[RLIMIT_STACK]},
0dae31d468b1a0e22d98921f7b0dbd92fd217167Zbigniew Jędrzejewski-Szmek { "Manager", "DefaultLimitCORE", config_parse_limit, 0, &arg_default_rlimit[RLIMIT_CORE]},
0dae31d468b1a0e22d98921f7b0dbd92fd217167Zbigniew Jędrzejewski-Szmek { "Manager", "DefaultLimitRSS", config_parse_limit, 0, &arg_default_rlimit[RLIMIT_RSS]},
0dae31d468b1a0e22d98921f7b0dbd92fd217167Zbigniew Jędrzejewski-Szmek { "Manager", "DefaultLimitNOFILE", config_parse_limit, 0, &arg_default_rlimit[RLIMIT_NOFILE]},
0dae31d468b1a0e22d98921f7b0dbd92fd217167Zbigniew Jędrzejewski-Szmek { "Manager", "DefaultLimitAS", config_parse_limit, 0, &arg_default_rlimit[RLIMIT_AS]},
0dae31d468b1a0e22d98921f7b0dbd92fd217167Zbigniew Jędrzejewski-Szmek { "Manager", "DefaultLimitNPROC", config_parse_limit, 0, &arg_default_rlimit[RLIMIT_NPROC]},
0dae31d468b1a0e22d98921f7b0dbd92fd217167Zbigniew Jędrzejewski-Szmek { "Manager", "DefaultLimitMEMLOCK", config_parse_limit, 0, &arg_default_rlimit[RLIMIT_MEMLOCK]},
0dae31d468b1a0e22d98921f7b0dbd92fd217167Zbigniew Jędrzejewski-Szmek { "Manager", "DefaultLimitLOCKS", config_parse_limit, 0, &arg_default_rlimit[RLIMIT_LOCKS]},
0dae31d468b1a0e22d98921f7b0dbd92fd217167Zbigniew Jędrzejewski-Szmek { "Manager", "DefaultLimitSIGPENDING",config_parse_limit, 0, &arg_default_rlimit[RLIMIT_SIGPENDING]},
0dae31d468b1a0e22d98921f7b0dbd92fd217167Zbigniew Jędrzejewski-Szmek { "Manager", "DefaultLimitMSGQUEUE", config_parse_limit, 0, &arg_default_rlimit[RLIMIT_MSGQUEUE]},
0dae31d468b1a0e22d98921f7b0dbd92fd217167Zbigniew Jędrzejewski-Szmek { "Manager", "DefaultLimitNICE", config_parse_limit, 0, &arg_default_rlimit[RLIMIT_NICE]},
0dae31d468b1a0e22d98921f7b0dbd92fd217167Zbigniew Jędrzejewski-Szmek { "Manager", "DefaultLimitRTPRIO", config_parse_limit, 0, &arg_default_rlimit[RLIMIT_RTPRIO]},
0dae31d468b1a0e22d98921f7b0dbd92fd217167Zbigniew Jędrzejewski-Szmek { "Manager", "DefaultLimitRTTIME", config_parse_limit, 0, &arg_default_rlimit[RLIMIT_RTTIME]},
0dae31d468b1a0e22d98921f7b0dbd92fd217167Zbigniew Jędrzejewski-Szmek { NULL, NULL, NULL, 0, NULL }
0dae31d468b1a0e22d98921f7b0dbd92fd217167Zbigniew Jędrzejewski-Szmek };
0dae31d468b1a0e22d98921f7b0dbd92fd217167Zbigniew Jędrzejewski-Szmek
0dae31d468b1a0e22d98921f7b0dbd92fd217167Zbigniew Jędrzejewski-Szmek _cleanup_fclose_ FILE *f;
0dae31d468b1a0e22d98921f7b0dbd92fd217167Zbigniew Jędrzejewski-Szmek const char *fn;
0dae31d468b1a0e22d98921f7b0dbd92fd217167Zbigniew Jędrzejewski-Szmek int r;
0dae31d468b1a0e22d98921f7b0dbd92fd217167Zbigniew Jędrzejewski-Szmek
0dae31d468b1a0e22d98921f7b0dbd92fd217167Zbigniew Jędrzejewski-Szmek fn = arg_running_as == SYSTEMD_SYSTEM ? PKGSYSCONFDIR "/system.conf" : PKGSYSCONFDIR "/user.conf";
0dae31d468b1a0e22d98921f7b0dbd92fd217167Zbigniew Jędrzejewski-Szmek f = fopen(fn, "re");
0dae31d468b1a0e22d98921f7b0dbd92fd217167Zbigniew Jędrzejewski-Szmek if (!f) {
0dae31d468b1a0e22d98921f7b0dbd92fd217167Zbigniew Jędrzejewski-Szmek if (errno == ENOENT)
0dae31d468b1a0e22d98921f7b0dbd92fd217167Zbigniew Jędrzejewski-Szmek return 0;
0dae31d468b1a0e22d98921f7b0dbd92fd217167Zbigniew Jędrzejewski-Szmek
0dae31d468b1a0e22d98921f7b0dbd92fd217167Zbigniew Jędrzejewski-Szmek log_warning("Failed to open configuration file '%s': %m", fn);
0dae31d468b1a0e22d98921f7b0dbd92fd217167Zbigniew Jędrzejewski-Szmek return 0;
0dae31d468b1a0e22d98921f7b0dbd92fd217167Zbigniew Jędrzejewski-Szmek }
0dae31d468b1a0e22d98921f7b0dbd92fd217167Zbigniew Jędrzejewski-Szmek
0dae31d468b1a0e22d98921f7b0dbd92fd217167Zbigniew Jędrzejewski-Szmek r = config_parse(NULL, fn, f, "Manager\0", config_item_table_lookup, (void*) items, false, false, NULL);
0dae31d468b1a0e22d98921f7b0dbd92fd217167Zbigniew Jędrzejewski-Szmek if (r < 0)
7c6423e19136a7b7b6ef3fe06b94822e582dda27Tom Gundersen log_warning("Failed to parse configuration file: %s", strerror(-r));
7c6423e19136a7b7b6ef3fe06b94822e582dda27Tom Gundersen
7c6423e19136a7b7b6ef3fe06b94822e582dda27Tom Gundersen return 0;
7c6423e19136a7b7b6ef3fe06b94822e582dda27Tom Gundersen}
7c6423e19136a7b7b6ef3fe06b94822e582dda27Tom Gundersen
7c6423e19136a7b7b6ef3fe06b94822e582dda27Tom Gundersenstatic int parse_proc_cmdline(void) {
7c6423e19136a7b7b6ef3fe06b94822e582dda27Tom Gundersen _cleanup_free_ char *line = NULL;
7c6423e19136a7b7b6ef3fe06b94822e582dda27Tom Gundersen char *w, *state;
7c6423e19136a7b7b6ef3fe06b94822e582dda27Tom Gundersen int r;
7c6423e19136a7b7b6ef3fe06b94822e582dda27Tom Gundersen size_t l;
7c6423e19136a7b7b6ef3fe06b94822e582dda27Tom Gundersen
7c6423e19136a7b7b6ef3fe06b94822e582dda27Tom Gundersen /* Don't read /proc/cmdline if we are in a container, since
7c6423e19136a7b7b6ef3fe06b94822e582dda27Tom Gundersen * that is only relevant for the host system */
7c6423e19136a7b7b6ef3fe06b94822e582dda27Tom Gundersen if (detect_container(NULL) > 0)
7c6423e19136a7b7b6ef3fe06b94822e582dda27Tom Gundersen return 0;
50f1e641a93cacfc693b0c3d300bee5df0c8c460Tom Gundersen
50f1e641a93cacfc693b0c3d300bee5df0c8c460Tom Gundersen r = read_one_line_file("/proc/cmdline", &line);
50f1e641a93cacfc693b0c3d300bee5df0c8c460Tom Gundersen if (r < 0) {
cb57dd41595adddb08095298bb1ed258c8ea4877Tom Gundersen log_warning("Failed to read /proc/cmdline, ignoring: %s", strerror(-r));
50f1e641a93cacfc693b0c3d300bee5df0c8c460Tom Gundersen return 0;
50f1e641a93cacfc693b0c3d300bee5df0c8c460Tom Gundersen }
50f1e641a93cacfc693b0c3d300bee5df0c8c460Tom Gundersen
cb57dd41595adddb08095298bb1ed258c8ea4877Tom Gundersen FOREACH_WORD_QUOTED(w, l, line, state) {
50f1e641a93cacfc693b0c3d300bee5df0c8c460Tom Gundersen _cleanup_free_ char *word;
2c1fb4f71206bf970d493294208c5d7597194856Lennart Poettering
50f1e641a93cacfc693b0c3d300bee5df0c8c460Tom Gundersen word = strndup(w, l);
50f1e641a93cacfc693b0c3d300bee5df0c8c460Tom Gundersen if (!word)
50f1e641a93cacfc693b0c3d300bee5df0c8c460Tom Gundersen return log_oom();
50f1e641a93cacfc693b0c3d300bee5df0c8c460Tom Gundersen
50f1e641a93cacfc693b0c3d300bee5df0c8c460Tom Gundersen r = parse_proc_cmdline_word(word);
50f1e641a93cacfc693b0c3d300bee5df0c8c460Tom Gundersen if (r < 0) {
50f1e641a93cacfc693b0c3d300bee5df0c8c460Tom Gundersen log_error("Failed on cmdline argument %s: %s", word, strerror(-r));
50f1e641a93cacfc693b0c3d300bee5df0c8c460Tom Gundersen return r;
50f1e641a93cacfc693b0c3d300bee5df0c8c460Tom Gundersen }
2c1fb4f71206bf970d493294208c5d7597194856Lennart Poettering }
50f1e641a93cacfc693b0c3d300bee5df0c8c460Tom Gundersen
50f1e641a93cacfc693b0c3d300bee5df0c8c460Tom Gundersen return 0;
50f1e641a93cacfc693b0c3d300bee5df0c8c460Tom Gundersen}
50f1e641a93cacfc693b0c3d300bee5df0c8c460Tom Gundersen
50f1e641a93cacfc693b0c3d300bee5df0c8c460Tom Gundersenstatic int parse_argv(int argc, char *argv[]) {
50f1e641a93cacfc693b0c3d300bee5df0c8c460Tom Gundersen
50f1e641a93cacfc693b0c3d300bee5df0c8c460Tom Gundersen enum {
50f1e641a93cacfc693b0c3d300bee5df0c8c460Tom Gundersen ARG_LOG_LEVEL = 0x100,
50f1e641a93cacfc693b0c3d300bee5df0c8c460Tom Gundersen ARG_LOG_TARGET,
50f1e641a93cacfc693b0c3d300bee5df0c8c460Tom Gundersen ARG_LOG_COLOR,
50f1e641a93cacfc693b0c3d300bee5df0c8c460Tom Gundersen ARG_LOG_LOCATION,
50f1e641a93cacfc693b0c3d300bee5df0c8c460Tom Gundersen ARG_UNIT,
2001c80560e3dae69e14fd994d3978c187af48b8Lennart Poettering ARG_SYSTEM,
2001c80560e3dae69e14fd994d3978c187af48b8Lennart Poettering ARG_USER,
2001c80560e3dae69e14fd994d3978c187af48b8Lennart Poettering ARG_TEST,
2001c80560e3dae69e14fd994d3978c187af48b8Lennart Poettering ARG_VERSION,
2001c80560e3dae69e14fd994d3978c187af48b8Lennart Poettering ARG_DUMP_CONFIGURATION_ITEMS,
2001c80560e3dae69e14fd994d3978c187af48b8Lennart Poettering ARG_DUMP_CORE,
2001c80560e3dae69e14fd994d3978c187af48b8Lennart Poettering ARG_CRASH_SHELL,
2001c80560e3dae69e14fd994d3978c187af48b8Lennart Poettering ARG_CONFIRM_SPAWN,
2001c80560e3dae69e14fd994d3978c187af48b8Lennart Poettering ARG_SHOW_STATUS,
2001c80560e3dae69e14fd994d3978c187af48b8Lennart Poettering ARG_DESERIALIZE,
2001c80560e3dae69e14fd994d3978c187af48b8Lennart Poettering ARG_SWITCHED_ROOT,
2001c80560e3dae69e14fd994d3978c187af48b8Lennart Poettering ARG_INTROSPECT,
2001c80560e3dae69e14fd994d3978c187af48b8Lennart Poettering ARG_DEFAULT_STD_OUTPUT,
2001c80560e3dae69e14fd994d3978c187af48b8Lennart Poettering ARG_DEFAULT_STD_ERROR
2001c80560e3dae69e14fd994d3978c187af48b8Lennart Poettering };
2001c80560e3dae69e14fd994d3978c187af48b8Lennart Poettering
2001c80560e3dae69e14fd994d3978c187af48b8Lennart Poettering static const struct option options[] = {
2001c80560e3dae69e14fd994d3978c187af48b8Lennart Poettering { "log-level", required_argument, NULL, ARG_LOG_LEVEL },
2001c80560e3dae69e14fd994d3978c187af48b8Lennart Poettering { "log-target", required_argument, NULL, ARG_LOG_TARGET },
2001c80560e3dae69e14fd994d3978c187af48b8Lennart Poettering { "log-color", optional_argument, NULL, ARG_LOG_COLOR },
2001c80560e3dae69e14fd994d3978c187af48b8Lennart Poettering { "log-location", optional_argument, NULL, ARG_LOG_LOCATION },
2001c80560e3dae69e14fd994d3978c187af48b8Lennart Poettering { "unit", required_argument, NULL, ARG_UNIT },
2001c80560e3dae69e14fd994d3978c187af48b8Lennart Poettering { "system", no_argument, NULL, ARG_SYSTEM },
2001c80560e3dae69e14fd994d3978c187af48b8Lennart Poettering { "user", no_argument, NULL, ARG_USER },
2001c80560e3dae69e14fd994d3978c187af48b8Lennart Poettering { "test", no_argument, NULL, ARG_TEST },
2001c80560e3dae69e14fd994d3978c187af48b8Lennart Poettering { "help", no_argument, NULL, 'h' },
2001c80560e3dae69e14fd994d3978c187af48b8Lennart Poettering { "version", no_argument, NULL, ARG_VERSION },
2001c80560e3dae69e14fd994d3978c187af48b8Lennart Poettering { "dump-configuration-items", no_argument, NULL, ARG_DUMP_CONFIGURATION_ITEMS },
2001c80560e3dae69e14fd994d3978c187af48b8Lennart Poettering { "dump-core", optional_argument, NULL, ARG_DUMP_CORE },
2001c80560e3dae69e14fd994d3978c187af48b8Lennart Poettering { "crash-shell", optional_argument, NULL, ARG_CRASH_SHELL },
2001c80560e3dae69e14fd994d3978c187af48b8Lennart Poettering { "confirm-spawn", optional_argument, NULL, ARG_CONFIRM_SPAWN },
2001c80560e3dae69e14fd994d3978c187af48b8Lennart Poettering { "show-status", optional_argument, NULL, ARG_SHOW_STATUS },
2001c80560e3dae69e14fd994d3978c187af48b8Lennart Poettering { "deserialize", required_argument, NULL, ARG_DESERIALIZE },
2001c80560e3dae69e14fd994d3978c187af48b8Lennart Poettering { "switched-root", no_argument, NULL, ARG_SWITCHED_ROOT },
2001c80560e3dae69e14fd994d3978c187af48b8Lennart Poettering { "introspect", optional_argument, NULL, ARG_INTROSPECT },
2001c80560e3dae69e14fd994d3978c187af48b8Lennart Poettering { "default-standard-output", required_argument, NULL, ARG_DEFAULT_STD_OUTPUT, },
2001c80560e3dae69e14fd994d3978c187af48b8Lennart Poettering { "default-standard-error", required_argument, NULL, ARG_DEFAULT_STD_ERROR, },
7b50eb2efa122200e39646c19a29abab302f7d24Lennart Poettering { NULL, 0, NULL, 0 }
8db0d2f5c37e7e8f5bfce016cfdad7947a3ea939Zbigniew Jędrzejewski-Szmek };
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering int c, r;
322345fdb9865ef2477fba8e4bdde0e1183ef505Lennart Poettering
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering assert(argc >= 1);
322345fdb9865ef2477fba8e4bdde0e1183ef505Lennart Poettering assert(argv);
7b50eb2efa122200e39646c19a29abab302f7d24Lennart Poettering
7b50eb2efa122200e39646c19a29abab302f7d24Lennart Poettering if (getpid() == 1)
7b50eb2efa122200e39646c19a29abab302f7d24Lennart Poettering opterr = 0;
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering while ((c = getopt_long(argc, argv, "hDbsz:", options, NULL)) >= 0)
7b50eb2efa122200e39646c19a29abab302f7d24Lennart Poettering
322345fdb9865ef2477fba8e4bdde0e1183ef505Lennart Poettering switch (c) {
0dae31d468b1a0e22d98921f7b0dbd92fd217167Zbigniew Jędrzejewski-Szmek
322345fdb9865ef2477fba8e4bdde0e1183ef505Lennart Poettering case ARG_LOG_LEVEL:
9c92ce6d67f88beb31dd6555d12ae3f632218a39Lennart Poettering if ((r = log_set_max_level_from_string(optarg)) < 0) {
9c92ce6d67f88beb31dd6555d12ae3f632218a39Lennart Poettering log_error("Failed to parse log level %s.", optarg);
9c92ce6d67f88beb31dd6555d12ae3f632218a39Lennart Poettering return r;
9c92ce6d67f88beb31dd6555d12ae3f632218a39Lennart Poettering }
9c92ce6d67f88beb31dd6555d12ae3f632218a39Lennart Poettering
9c92ce6d67f88beb31dd6555d12ae3f632218a39Lennart Poettering break;
9c92ce6d67f88beb31dd6555d12ae3f632218a39Lennart Poettering
9c92ce6d67f88beb31dd6555d12ae3f632218a39Lennart Poettering case ARG_LOG_TARGET:
7b50eb2efa122200e39646c19a29abab302f7d24Lennart Poettering
9c92ce6d67f88beb31dd6555d12ae3f632218a39Lennart Poettering if ((r = log_set_target_from_string(optarg)) < 0) {
9c92ce6d67f88beb31dd6555d12ae3f632218a39Lennart Poettering log_error("Failed to parse log target %s.", optarg);
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering return r;
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering }
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering
8ac4e9e1e54397f6d1745c2a7a806132418c7da2Lennart Poettering break;
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering case ARG_LOG_COLOR:
7b50eb2efa122200e39646c19a29abab302f7d24Lennart Poettering
322345fdb9865ef2477fba8e4bdde0e1183ef505Lennart Poettering if (optarg) {
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering if ((r = log_show_color_from_string(optarg)) < 0) {
322345fdb9865ef2477fba8e4bdde0e1183ef505Lennart Poettering log_error("Failed to parse log color setting %s.", optarg);
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering return r;
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering }
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering } else
7b50eb2efa122200e39646c19a29abab302f7d24Lennart Poettering log_show_color(true);
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering
322345fdb9865ef2477fba8e4bdde0e1183ef505Lennart Poettering break;
9de3e3294065e8697ff10130b53f274319cdcf6fZbigniew Jędrzejewski-Szmek
8db0d2f5c37e7e8f5bfce016cfdad7947a3ea939Zbigniew Jędrzejewski-Szmek case ARG_LOG_LOCATION:
2001c80560e3dae69e14fd994d3978c187af48b8Lennart Poettering
2e276efc7b0398a3086629a52970bdd4ab7252f9Zbigniew Jędrzejewski-Szmek if (optarg) {
7b50eb2efa122200e39646c19a29abab302f7d24Lennart Poettering if ((r = log_show_location_from_string(optarg)) < 0) {
2e276efc7b0398a3086629a52970bdd4ab7252f9Zbigniew Jędrzejewski-Szmek log_error("Failed to parse log location setting %s.", optarg);
2e276efc7b0398a3086629a52970bdd4ab7252f9Zbigniew Jędrzejewski-Szmek return r;
2e276efc7b0398a3086629a52970bdd4ab7252f9Zbigniew Jędrzejewski-Szmek }
7b50eb2efa122200e39646c19a29abab302f7d24Lennart Poettering } else
2e276efc7b0398a3086629a52970bdd4ab7252f9Zbigniew Jędrzejewski-Szmek log_show_location(true);
2e276efc7b0398a3086629a52970bdd4ab7252f9Zbigniew Jędrzejewski-Szmek
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering break;
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering
322345fdb9865ef2477fba8e4bdde0e1183ef505Lennart Poettering case ARG_DEFAULT_STD_OUTPUT:
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering if ((r = exec_output_from_string(optarg)) < 0) {
7b50eb2efa122200e39646c19a29abab302f7d24Lennart Poettering log_error("Failed to parse default standard output setting %s.", optarg);
322345fdb9865ef2477fba8e4bdde0e1183ef505Lennart Poettering return r;
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering } else
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering arg_default_std_output = r;
7b50eb2efa122200e39646c19a29abab302f7d24Lennart Poettering break;
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering case ARG_DEFAULT_STD_ERROR:
322345fdb9865ef2477fba8e4bdde0e1183ef505Lennart Poettering
8db0d2f5c37e7e8f5bfce016cfdad7947a3ea939Zbigniew Jędrzejewski-Szmek if ((r = exec_output_from_string(optarg)) < 0) {
8db0d2f5c37e7e8f5bfce016cfdad7947a3ea939Zbigniew Jędrzejewski-Szmek log_error("Failed to parse default standard error output setting %s.", optarg);
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering return r;
7b50eb2efa122200e39646c19a29abab302f7d24Lennart Poettering } else
322345fdb9865ef2477fba8e4bdde0e1183ef505Lennart Poettering arg_default_std_error = r;
8db0d2f5c37e7e8f5bfce016cfdad7947a3ea939Zbigniew Jędrzejewski-Szmek break;
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering
7b50eb2efa122200e39646c19a29abab302f7d24Lennart Poettering case ARG_UNIT:
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering
322345fdb9865ef2477fba8e4bdde0e1183ef505Lennart Poettering if ((r = set_default_unit(optarg)) < 0) {
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering log_error("Failed to set default unit %s: %s", optarg, strerror(-r));
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering return r;
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering }
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering break;
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering case ARG_SYSTEM:
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering arg_running_as = SYSTEMD_SYSTEM;
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering break;
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering case ARG_USER:
7b50eb2efa122200e39646c19a29abab302f7d24Lennart Poettering arg_running_as = SYSTEMD_USER;
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering break;
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering
946c70944ebdf428ffeb9991a7449edbd4011461Zbigniew Jędrzejewski-Szmek case ARG_TEST:
946c70944ebdf428ffeb9991a7449edbd4011461Zbigniew Jędrzejewski-Szmek arg_action = ACTION_TEST;
946c70944ebdf428ffeb9991a7449edbd4011461Zbigniew Jędrzejewski-Szmek break;
946c70944ebdf428ffeb9991a7449edbd4011461Zbigniew Jędrzejewski-Szmek
946c70944ebdf428ffeb9991a7449edbd4011461Zbigniew Jędrzejewski-Szmek case ARG_VERSION:
946c70944ebdf428ffeb9991a7449edbd4011461Zbigniew Jędrzejewski-Szmek arg_action = ACTION_VERSION;
7b50eb2efa122200e39646c19a29abab302f7d24Lennart Poettering break;
946c70944ebdf428ffeb9991a7449edbd4011461Zbigniew Jędrzejewski-Szmek
946c70944ebdf428ffeb9991a7449edbd4011461Zbigniew Jędrzejewski-Szmek case ARG_DUMP_CONFIGURATION_ITEMS:
8db0d2f5c37e7e8f5bfce016cfdad7947a3ea939Zbigniew Jędrzejewski-Szmek arg_action = ACTION_DUMP_CONFIGURATION_ITEMS;
0dae31d468b1a0e22d98921f7b0dbd92fd217167Zbigniew Jędrzejewski-Szmek break;
0dae31d468b1a0e22d98921f7b0dbd92fd217167Zbigniew Jędrzejewski-Szmek
8db0d2f5c37e7e8f5bfce016cfdad7947a3ea939Zbigniew Jędrzejewski-Szmek case ARG_DUMP_CORE:
8db0d2f5c37e7e8f5bfce016cfdad7947a3ea939Zbigniew Jędrzejewski-Szmek r = optarg ? parse_boolean(optarg) : 1;
8db0d2f5c37e7e8f5bfce016cfdad7947a3ea939Zbigniew Jędrzejewski-Szmek if (r < 0) {
8db0d2f5c37e7e8f5bfce016cfdad7947a3ea939Zbigniew Jędrzejewski-Szmek log_error("Failed to parse dump core boolean %s.", optarg);
8db0d2f5c37e7e8f5bfce016cfdad7947a3ea939Zbigniew Jędrzejewski-Szmek return r;
8db0d2f5c37e7e8f5bfce016cfdad7947a3ea939Zbigniew Jędrzejewski-Szmek }
8db0d2f5c37e7e8f5bfce016cfdad7947a3ea939Zbigniew Jędrzejewski-Szmek arg_dump_core = r;
7b50eb2efa122200e39646c19a29abab302f7d24Lennart Poettering break;
0dae31d468b1a0e22d98921f7b0dbd92fd217167Zbigniew Jędrzejewski-Szmek
8db0d2f5c37e7e8f5bfce016cfdad7947a3ea939Zbigniew Jędrzejewski-Szmek case ARG_CRASH_SHELL:
0dae31d468b1a0e22d98921f7b0dbd92fd217167Zbigniew Jędrzejewski-Szmek r = optarg ? parse_boolean(optarg) : 1;
7b50eb2efa122200e39646c19a29abab302f7d24Lennart Poettering if (r < 0) {
0dae31d468b1a0e22d98921f7b0dbd92fd217167Zbigniew Jędrzejewski-Szmek log_error("Failed to parse crash shell boolean %s.", optarg);
0dae31d468b1a0e22d98921f7b0dbd92fd217167Zbigniew Jędrzejewski-Szmek return r;
abf126a355e2f2b62b6c51ab3bb37895d1e3eee7Tom Gundersen }
abf126a355e2f2b62b6c51ab3bb37895d1e3eee7Tom Gundersen arg_crash_shell = r;
abf126a355e2f2b62b6c51ab3bb37895d1e3eee7Tom Gundersen break;
7b50eb2efa122200e39646c19a29abab302f7d24Lennart Poettering
abf126a355e2f2b62b6c51ab3bb37895d1e3eee7Tom Gundersen case ARG_CONFIRM_SPAWN:
abf126a355e2f2b62b6c51ab3bb37895d1e3eee7Tom Gundersen r = optarg ? parse_boolean(optarg) : 1;
abf126a355e2f2b62b6c51ab3bb37895d1e3eee7Tom Gundersen if (r < 0) {
abf126a355e2f2b62b6c51ab3bb37895d1e3eee7Tom Gundersen log_error("Failed to parse confirm spawn boolean %s.", optarg);
abf126a355e2f2b62b6c51ab3bb37895d1e3eee7Tom Gundersen return r;
abf126a355e2f2b62b6c51ab3bb37895d1e3eee7Tom Gundersen }
abf126a355e2f2b62b6c51ab3bb37895d1e3eee7Tom Gundersen arg_confirm_spawn = r;
abf126a355e2f2b62b6c51ab3bb37895d1e3eee7Tom Gundersen break;
7b50eb2efa122200e39646c19a29abab302f7d24Lennart Poettering
abf126a355e2f2b62b6c51ab3bb37895d1e3eee7Tom Gundersen case ARG_SHOW_STATUS:
abf126a355e2f2b62b6c51ab3bb37895d1e3eee7Tom Gundersen r = optarg ? parse_boolean(optarg) : 1;
8db0d2f5c37e7e8f5bfce016cfdad7947a3ea939Zbigniew Jędrzejewski-Szmek if (r < 0) {
549c1a2564b56f2bb38f1203d59c747ea15817f3Tom Gundersen log_error("Failed to parse show status boolean %s.", optarg);
8db0d2f5c37e7e8f5bfce016cfdad7947a3ea939Zbigniew Jędrzejewski-Szmek return r;
7b50eb2efa122200e39646c19a29abab302f7d24Lennart Poettering }
42cc2eebb01056beb7acd3ecfe8e533558237f84Lennart Poettering arg_show_status = r;
42cc2eebb01056beb7acd3ecfe8e533558237f84Lennart Poettering break;
42cc2eebb01056beb7acd3ecfe8e533558237f84Lennart Poettering
42cc2eebb01056beb7acd3ecfe8e533558237f84Lennart Poettering case ARG_DESERIALIZE: {
42cc2eebb01056beb7acd3ecfe8e533558237f84Lennart Poettering int fd;
8db0d2f5c37e7e8f5bfce016cfdad7947a3ea939Zbigniew Jędrzejewski-Szmek FILE *f;
42cc2eebb01056beb7acd3ecfe8e533558237f84Lennart Poettering
7b50eb2efa122200e39646c19a29abab302f7d24Lennart Poettering r = safe_atoi(optarg, &fd);
42cc2eebb01056beb7acd3ecfe8e533558237f84Lennart Poettering if (r < 0 || fd < 0) {
42cc2eebb01056beb7acd3ecfe8e533558237f84Lennart Poettering log_error("Failed to parse deserialize option %s.", optarg);
ff3d6560bead6879a2fed1bf99bfe8273b3723f1Zbigniew Jędrzejewski-Szmek return r < 0 ? r : -EINVAL;
ff3d6560bead6879a2fed1bf99bfe8273b3723f1Zbigniew Jędrzejewski-Szmek }
ff3d6560bead6879a2fed1bf99bfe8273b3723f1Zbigniew Jędrzejewski-Szmek
ff3d6560bead6879a2fed1bf99bfe8273b3723f1Zbigniew Jędrzejewski-Szmek fd_cloexec(fd, true);
ff3d6560bead6879a2fed1bf99bfe8273b3723f1Zbigniew Jędrzejewski-Szmek
1bf968f36393666f2c57953b1748e6219c027deeTom Gundersen f = fdopen(fd, "r");
8db0d2f5c37e7e8f5bfce016cfdad7947a3ea939Zbigniew Jędrzejewski-Szmek if (!f) {
7b50eb2efa122200e39646c19a29abab302f7d24Lennart Poettering log_error("Failed to open serialization fd: %m");
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering return -errno;
f91dc2400dc33e9a0745ecaaef7489af116dca38Lennart Poettering }
8db0d2f5c37e7e8f5bfce016cfdad7947a3ea939Zbigniew Jędrzejewski-Szmek
f91dc2400dc33e9a0745ecaaef7489af116dca38Lennart Poettering if (serialization)
f91dc2400dc33e9a0745ecaaef7489af116dca38Lennart Poettering fclose(serialization);
ff3d6560bead6879a2fed1bf99bfe8273b3723f1Zbigniew Jędrzejewski-Szmek
ff3d6560bead6879a2fed1bf99bfe8273b3723f1Zbigniew Jędrzejewski-Szmek serialization = f;
8db0d2f5c37e7e8f5bfce016cfdad7947a3ea939Zbigniew Jędrzejewski-Szmek
8db0d2f5c37e7e8f5bfce016cfdad7947a3ea939Zbigniew Jędrzejewski-Szmek break;
7b50eb2efa122200e39646c19a29abab302f7d24Lennart Poettering }
8db0d2f5c37e7e8f5bfce016cfdad7947a3ea939Zbigniew Jędrzejewski-Szmek
ff3d6560bead6879a2fed1bf99bfe8273b3723f1Zbigniew Jędrzejewski-Szmek case ARG_SWITCHED_ROOT:
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering arg_switched_root = true;
151226ab4bf276d60d51864330a99f886b923697Zbigniew Jędrzejewski-Szmek break;
151226ab4bf276d60d51864330a99f886b923697Zbigniew Jędrzejewski-Szmek
7c6423e19136a7b7b6ef3fe06b94822e582dda27Tom Gundersen case ARG_INTROSPECT: {
151226ab4bf276d60d51864330a99f886b923697Zbigniew Jędrzejewski-Szmek const char * const * i = NULL;
151226ab4bf276d60d51864330a99f886b923697Zbigniew Jędrzejewski-Szmek
151226ab4bf276d60d51864330a99f886b923697Zbigniew Jędrzejewski-Szmek for (i = bus_interface_table; *i; i += 2)
151226ab4bf276d60d51864330a99f886b923697Zbigniew Jędrzejewski-Szmek if (!optarg || streq(i[0], optarg)) {
1bf968f36393666f2c57953b1748e6219c027deeTom Gundersen fputs(DBUS_INTROSPECT_1_0_XML_DOCTYPE_DECL_NODE
151226ab4bf276d60d51864330a99f886b923697Zbigniew Jędrzejewski-Szmek "<node>\n", stdout);
7b50eb2efa122200e39646c19a29abab302f7d24Lennart Poettering fputs(i[1], stdout);
151226ab4bf276d60d51864330a99f886b923697Zbigniew Jędrzejewski-Szmek fputs("</node>\n", stdout);
7c6423e19136a7b7b6ef3fe06b94822e582dda27Tom Gundersen
7c6423e19136a7b7b6ef3fe06b94822e582dda27Tom Gundersen if (optarg)
7b50eb2efa122200e39646c19a29abab302f7d24Lennart Poettering break;
7c6423e19136a7b7b6ef3fe06b94822e582dda27Tom Gundersen }
7c6423e19136a7b7b6ef3fe06b94822e582dda27Tom Gundersen
7c6423e19136a7b7b6ef3fe06b94822e582dda27Tom Gundersen if (!i[0] && optarg)
7b50eb2efa122200e39646c19a29abab302f7d24Lennart Poettering log_error("Unknown interface %s.", optarg);
7c6423e19136a7b7b6ef3fe06b94822e582dda27Tom Gundersen
151226ab4bf276d60d51864330a99f886b923697Zbigniew Jędrzejewski-Szmek arg_action = ACTION_DONE;
151226ab4bf276d60d51864330a99f886b923697Zbigniew Jędrzejewski-Szmek break;
151226ab4bf276d60d51864330a99f886b923697Zbigniew Jędrzejewski-Szmek }
7c6423e19136a7b7b6ef3fe06b94822e582dda27Tom Gundersen
151226ab4bf276d60d51864330a99f886b923697Zbigniew Jędrzejewski-Szmek case 'h':
151226ab4bf276d60d51864330a99f886b923697Zbigniew Jędrzejewski-Szmek arg_action = ACTION_HELP;
151226ab4bf276d60d51864330a99f886b923697Zbigniew Jędrzejewski-Szmek break;
151226ab4bf276d60d51864330a99f886b923697Zbigniew Jędrzejewski-Szmek
151226ab4bf276d60d51864330a99f886b923697Zbigniew Jędrzejewski-Szmek case 'D':
151226ab4bf276d60d51864330a99f886b923697Zbigniew Jędrzejewski-Szmek log_set_max_level(LOG_DEBUG);
151226ab4bf276d60d51864330a99f886b923697Zbigniew Jędrzejewski-Szmek break;
7c6423e19136a7b7b6ef3fe06b94822e582dda27Tom Gundersen
7c6423e19136a7b7b6ef3fe06b94822e582dda27Tom Gundersen case 'b':
151226ab4bf276d60d51864330a99f886b923697Zbigniew Jędrzejewski-Szmek case 's':
151226ab4bf276d60d51864330a99f886b923697Zbigniew Jędrzejewski-Szmek case 'z':
151226ab4bf276d60d51864330a99f886b923697Zbigniew Jędrzejewski-Szmek /* Just to eat away the sysvinit kernel
151226ab4bf276d60d51864330a99f886b923697Zbigniew Jędrzejewski-Szmek * cmdline args without getopt() error
7b50eb2efa122200e39646c19a29abab302f7d24Lennart Poettering * messages that we'll parse in
151226ab4bf276d60d51864330a99f886b923697Zbigniew Jędrzejewski-Szmek * parse_proc_cmdline_word() or ignore. */
151226ab4bf276d60d51864330a99f886b923697Zbigniew Jędrzejewski-Szmek
151226ab4bf276d60d51864330a99f886b923697Zbigniew Jędrzejewski-Szmek case '?':
50f1e641a93cacfc693b0c3d300bee5df0c8c460Tom Gundersen default:
50f1e641a93cacfc693b0c3d300bee5df0c8c460Tom Gundersen if (getpid() != 1) {
50f1e641a93cacfc693b0c3d300bee5df0c8c460Tom Gundersen log_error("Unknown option code %c", c);
7b50eb2efa122200e39646c19a29abab302f7d24Lennart Poettering return -EINVAL;
50f1e641a93cacfc693b0c3d300bee5df0c8c460Tom Gundersen }
50f1e641a93cacfc693b0c3d300bee5df0c8c460Tom Gundersen
50f1e641a93cacfc693b0c3d300bee5df0c8c460Tom Gundersen break;
50f1e641a93cacfc693b0c3d300bee5df0c8c460Tom Gundersen }
50f1e641a93cacfc693b0c3d300bee5df0c8c460Tom Gundersen
50f1e641a93cacfc693b0c3d300bee5df0c8c460Tom Gundersen if (optind < argc && getpid() != 1) {
7b50eb2efa122200e39646c19a29abab302f7d24Lennart Poettering /* Hmm, when we aren't run as init system
50f1e641a93cacfc693b0c3d300bee5df0c8c460Tom Gundersen * let's complain about excess arguments */
50f1e641a93cacfc693b0c3d300bee5df0c8c460Tom Gundersen
5d45a8808431987c370706d365fb0cc95cf03d52Tom Gundersen log_error("Excess arguments.");
5d45a8808431987c370706d365fb0cc95cf03d52Tom Gundersen return -EINVAL;
5d45a8808431987c370706d365fb0cc95cf03d52Tom Gundersen }
f5430a3ef308f3a102899fcaf7fbece757082f2aLennart Poettering
5d45a8808431987c370706d365fb0cc95cf03d52Tom Gundersen if (detect_container(NULL) > 0) {
5d45a8808431987c370706d365fb0cc95cf03d52Tom Gundersen char **a;
7b50eb2efa122200e39646c19a29abab302f7d24Lennart Poettering
5d45a8808431987c370706d365fb0cc95cf03d52Tom Gundersen /* All /proc/cmdline arguments the kernel didn't
5d45a8808431987c370706d365fb0cc95cf03d52Tom Gundersen * understand it passed to us. We're not really
5d45a8808431987c370706d365fb0cc95cf03d52Tom Gundersen * interested in that usually since /proc/cmdline is
5d45a8808431987c370706d365fb0cc95cf03d52Tom Gundersen * more interesting and complete. With one exception:
7b50eb2efa122200e39646c19a29abab302f7d24Lennart Poettering * if we are run in a container /proc/cmdline is not
5d45a8808431987c370706d365fb0cc95cf03d52Tom Gundersen * relevant for the container, hence we rely on argv[]
5d45a8808431987c370706d365fb0cc95cf03d52Tom Gundersen * instead. */
5d45a8808431987c370706d365fb0cc95cf03d52Tom Gundersen
7b50eb2efa122200e39646c19a29abab302f7d24Lennart Poettering for (a = argv; a < argv + argc; a++)
5d45a8808431987c370706d365fb0cc95cf03d52Tom Gundersen if ((r = parse_proc_cmdline_word(*a)) < 0) {
5d45a8808431987c370706d365fb0cc95cf03d52Tom Gundersen log_error("Failed on cmdline argument %s: %s", *a, strerror(-r));
5d45a8808431987c370706d365fb0cc95cf03d52Tom Gundersen return r;
5d45a8808431987c370706d365fb0cc95cf03d52Tom Gundersen }
5d45a8808431987c370706d365fb0cc95cf03d52Tom Gundersen }
5d45a8808431987c370706d365fb0cc95cf03d52Tom Gundersen
f5430a3ef308f3a102899fcaf7fbece757082f2aLennart Poettering return 0;
5d45a8808431987c370706d365fb0cc95cf03d52Tom Gundersen}
5d45a8808431987c370706d365fb0cc95cf03d52Tom Gundersen
5d45a8808431987c370706d365fb0cc95cf03d52Tom Gundersenstatic int help(void) {
7b50eb2efa122200e39646c19a29abab302f7d24Lennart Poettering
5d45a8808431987c370706d365fb0cc95cf03d52Tom Gundersen printf("%s [OPTIONS...]\n\n"
5d45a8808431987c370706d365fb0cc95cf03d52Tom Gundersen "Starts up and maintains the system or user services.\n\n"
5d45a8808431987c370706d365fb0cc95cf03d52Tom Gundersen " -h --help Show this help\n"
5d45a8808431987c370706d365fb0cc95cf03d52Tom Gundersen " --test Determine startup sequence, dump it and exit\n"
8db0d2f5c37e7e8f5bfce016cfdad7947a3ea939Zbigniew Jędrzejewski-Szmek " --dump-configuration-items Dump understood unit configuration items\n"
8db0d2f5c37e7e8f5bfce016cfdad7947a3ea939Zbigniew Jędrzejewski-Szmek " --introspect[=INTERFACE] Extract D-Bus interface data\n"
8db0d2f5c37e7e8f5bfce016cfdad7947a3ea939Zbigniew Jędrzejewski-Szmek " --unit=UNIT Set default unit\n"
7b50eb2efa122200e39646c19a29abab302f7d24Lennart Poettering " --system Run a system instance, even if PID != 1\n"
8db0d2f5c37e7e8f5bfce016cfdad7947a3ea939Zbigniew Jędrzejewski-Szmek " --user Run a user instance\n"
6af47493de0ef2b66d4c3fbcdd4a2e12fec4bfbaLennart Poettering " --dump-core[=0|1] Dump core on crash\n"
f5430a3ef308f3a102899fcaf7fbece757082f2aLennart Poettering " --crash-shell[=0|1] Run shell on crash\n"
d23a27a964748967e1ad20e86de869a753af555bTom Gundersen " --confirm-spawn[=0|1] Ask for confirmation when spawning processes\n"
7b50eb2efa122200e39646c19a29abab302f7d24Lennart Poettering " --show-status[=0|1] Show status updates on the console during bootup\n"
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering " --log-target=TARGET Set log target (console, journal, syslog, kmsg, journal-or-kmsg, syslog-or-kmsg, null)\n"
8db0d2f5c37e7e8f5bfce016cfdad7947a3ea939Zbigniew Jędrzejewski-Szmek " --log-level=LEVEL Set log level (debug, info, notice, warning, err, crit, alert, emerg)\n"
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering " --log-color[=0|1] Highlight important log messages\n"
7b50eb2efa122200e39646c19a29abab302f7d24Lennart Poettering " --log-location[=0|1] Include code location in log messages\n"
7b50eb2efa122200e39646c19a29abab302f7d24Lennart Poettering " --default-standard-output= Set default standard output for services\n"
2d4c5cbc0ed3ccb09dc086a040088b454c22c644Lennart Poettering " --default-standard-error= Set default standard error output for services\n",
322345fdb9865ef2477fba8e4bdde0e1183ef505Lennart Poettering program_invocation_short_name);
a8812dd7f161a3e459c1730ac92ff2bbc9986ff1Lennart Poettering
a8812dd7f161a3e459c1730ac92ff2bbc9986ff1Lennart Poettering return 0;
a8812dd7f161a3e459c1730ac92ff2bbc9986ff1Lennart Poettering}
a8812dd7f161a3e459c1730ac92ff2bbc9986ff1Lennart Poettering
a8812dd7f161a3e459c1730ac92ff2bbc9986ff1Lennart Poetteringstatic int version(void) {
a8812dd7f161a3e459c1730ac92ff2bbc9986ff1Lennart Poettering puts(PACKAGE_STRING);
a8812dd7f161a3e459c1730ac92ff2bbc9986ff1Lennart Poettering puts(SYSTEMD_FEATURES);
a8812dd7f161a3e459c1730ac92ff2bbc9986ff1Lennart Poettering
a8812dd7f161a3e459c1730ac92ff2bbc9986ff1Lennart Poettering return 0;
a8812dd7f161a3e459c1730ac92ff2bbc9986ff1Lennart Poettering}
a8812dd7f161a3e459c1730ac92ff2bbc9986ff1Lennart Poettering
a8812dd7f161a3e459c1730ac92ff2bbc9986ff1Lennart Poetteringstatic int prepare_reexecute(Manager *m, FILE **_f, FDSet **_fds, bool switching_root) {
a8812dd7f161a3e459c1730ac92ff2bbc9986ff1Lennart Poettering FILE *f = NULL;
a8812dd7f161a3e459c1730ac92ff2bbc9986ff1Lennart Poettering FDSet *fds = NULL;
a8812dd7f161a3e459c1730ac92ff2bbc9986ff1Lennart Poettering int r;
a8812dd7f161a3e459c1730ac92ff2bbc9986ff1Lennart Poettering
a8812dd7f161a3e459c1730ac92ff2bbc9986ff1Lennart Poettering assert(m);
a8812dd7f161a3e459c1730ac92ff2bbc9986ff1Lennart Poettering assert(_f);
a8812dd7f161a3e459c1730ac92ff2bbc9986ff1Lennart Poettering assert(_fds);
a8812dd7f161a3e459c1730ac92ff2bbc9986ff1Lennart Poettering
a8812dd7f161a3e459c1730ac92ff2bbc9986ff1Lennart Poettering r = manager_open_serialization(m, &f);
a8812dd7f161a3e459c1730ac92ff2bbc9986ff1Lennart Poettering if (r < 0) {
a8812dd7f161a3e459c1730ac92ff2bbc9986ff1Lennart Poettering log_error("Failed to create serialization file: %s", strerror(-r));
a8812dd7f161a3e459c1730ac92ff2bbc9986ff1Lennart Poettering goto fail;
a8812dd7f161a3e459c1730ac92ff2bbc9986ff1Lennart Poettering }
a8812dd7f161a3e459c1730ac92ff2bbc9986ff1Lennart Poettering
a8812dd7f161a3e459c1730ac92ff2bbc9986ff1Lennart Poettering /* Make sure nothing is really destructed when we shut down */
a8812dd7f161a3e459c1730ac92ff2bbc9986ff1Lennart Poettering m->n_reloading ++;
a8812dd7f161a3e459c1730ac92ff2bbc9986ff1Lennart Poettering bus_broadcast_reloading(m, true);
a8812dd7f161a3e459c1730ac92ff2bbc9986ff1Lennart Poettering
a8812dd7f161a3e459c1730ac92ff2bbc9986ff1Lennart Poettering fds = fdset_new();
a8812dd7f161a3e459c1730ac92ff2bbc9986ff1Lennart Poettering if (!fds) {
a8812dd7f161a3e459c1730ac92ff2bbc9986ff1Lennart Poettering r = -ENOMEM;
a8812dd7f161a3e459c1730ac92ff2bbc9986ff1Lennart Poettering log_error("Failed to allocate fd set: %s", strerror(-r));
a8812dd7f161a3e459c1730ac92ff2bbc9986ff1Lennart Poettering goto fail;
a8812dd7f161a3e459c1730ac92ff2bbc9986ff1Lennart Poettering }
a8812dd7f161a3e459c1730ac92ff2bbc9986ff1Lennart Poettering
a8812dd7f161a3e459c1730ac92ff2bbc9986ff1Lennart Poettering r = manager_serialize(m, f, fds, switching_root);
a8812dd7f161a3e459c1730ac92ff2bbc9986ff1Lennart Poettering if (r < 0) {
a8812dd7f161a3e459c1730ac92ff2bbc9986ff1Lennart Poettering log_error("Failed to serialize state: %s", strerror(-r));
a8812dd7f161a3e459c1730ac92ff2bbc9986ff1Lennart Poettering goto fail;
a8812dd7f161a3e459c1730ac92ff2bbc9986ff1Lennart Poettering }
a8812dd7f161a3e459c1730ac92ff2bbc9986ff1Lennart Poettering
a8812dd7f161a3e459c1730ac92ff2bbc9986ff1Lennart Poettering if (fseeko(f, 0, SEEK_SET) < 0) {
a8812dd7f161a3e459c1730ac92ff2bbc9986ff1Lennart Poettering log_error("Failed to rewind serialization fd: %m");
2001c80560e3dae69e14fd994d3978c187af48b8Lennart Poettering goto fail;
2001c80560e3dae69e14fd994d3978c187af48b8Lennart Poettering }
2001c80560e3dae69e14fd994d3978c187af48b8Lennart Poettering
2001c80560e3dae69e14fd994d3978c187af48b8Lennart Poettering r = fd_cloexec(fileno(f), false);
2001c80560e3dae69e14fd994d3978c187af48b8Lennart Poettering if (r < 0) {
2001c80560e3dae69e14fd994d3978c187af48b8Lennart Poettering log_error("Failed to disable O_CLOEXEC for serialization: %s", strerror(-r));
2001c80560e3dae69e14fd994d3978c187af48b8Lennart Poettering goto fail;
2001c80560e3dae69e14fd994d3978c187af48b8Lennart Poettering }
2001c80560e3dae69e14fd994d3978c187af48b8Lennart Poettering
2001c80560e3dae69e14fd994d3978c187af48b8Lennart Poettering r = fdset_cloexec(fds, false);
2001c80560e3dae69e14fd994d3978c187af48b8Lennart Poettering if (r < 0) {
2001c80560e3dae69e14fd994d3978c187af48b8Lennart Poettering log_error("Failed to disable O_CLOEXEC for serialization fds: %s", strerror(-r));
2001c80560e3dae69e14fd994d3978c187af48b8Lennart Poettering goto fail;
2001c80560e3dae69e14fd994d3978c187af48b8Lennart Poettering }
4d247a6cd3f69acbc5a09e8ac7e4fbb50eaa3228Lennart Poettering
4d247a6cd3f69acbc5a09e8ac7e4fbb50eaa3228Lennart Poettering *_f = f;
4d247a6cd3f69acbc5a09e8ac7e4fbb50eaa3228Lennart Poettering *_fds = fds;
2001c80560e3dae69e14fd994d3978c187af48b8Lennart Poettering
2001c80560e3dae69e14fd994d3978c187af48b8Lennart Poettering return 0;
2001c80560e3dae69e14fd994d3978c187af48b8Lennart Poettering
2001c80560e3dae69e14fd994d3978c187af48b8Lennart Poetteringfail:
2001c80560e3dae69e14fd994d3978c187af48b8Lennart Poettering fdset_free(fds);
2001c80560e3dae69e14fd994d3978c187af48b8Lennart Poettering
2001c80560e3dae69e14fd994d3978c187af48b8Lennart Poettering if (f)
2001c80560e3dae69e14fd994d3978c187af48b8Lennart Poettering fclose(f);
2001c80560e3dae69e14fd994d3978c187af48b8Lennart Poettering
2001c80560e3dae69e14fd994d3978c187af48b8Lennart Poettering return r;
2001c80560e3dae69e14fd994d3978c187af48b8Lennart Poettering}
2001c80560e3dae69e14fd994d3978c187af48b8Lennart Poettering
2001c80560e3dae69e14fd994d3978c187af48b8Lennart Poetteringstatic int bump_rlimit_nofile(struct rlimit *saved_rlimit) {
2001c80560e3dae69e14fd994d3978c187af48b8Lennart Poettering struct rlimit nl;
8730bccfc59fe507bd3e0a3abcf411b497ac4f0eLennart Poettering int r;
8730bccfc59fe507bd3e0a3abcf411b497ac4f0eLennart Poettering
6f717d0817573a76c3e586eae02793d8b23a0581Lennart Poettering assert(saved_rlimit);
8730bccfc59fe507bd3e0a3abcf411b497ac4f0eLennart Poettering
8730bccfc59fe507bd3e0a3abcf411b497ac4f0eLennart Poettering /* Save the original RLIMIT_NOFILE so that we can reset it
8730bccfc59fe507bd3e0a3abcf411b497ac4f0eLennart Poettering * later when transitioning from the initrd to the main
8730bccfc59fe507bd3e0a3abcf411b497ac4f0eLennart Poettering * systemd or suchlike. */
8730bccfc59fe507bd3e0a3abcf411b497ac4f0eLennart Poettering if (getrlimit(RLIMIT_NOFILE, saved_rlimit) < 0) {
8730bccfc59fe507bd3e0a3abcf411b497ac4f0eLennart Poettering log_error("Reading RLIMIT_NOFILE failed: %m");
8730bccfc59fe507bd3e0a3abcf411b497ac4f0eLennart Poettering return -errno;
8730bccfc59fe507bd3e0a3abcf411b497ac4f0eLennart Poettering }
8730bccfc59fe507bd3e0a3abcf411b497ac4f0eLennart Poettering
6f717d0817573a76c3e586eae02793d8b23a0581Lennart Poettering /* Make sure forked processes get the default kernel setting */
6f717d0817573a76c3e586eae02793d8b23a0581Lennart Poettering if (!arg_default_rlimit[RLIMIT_NOFILE]) {
6f717d0817573a76c3e586eae02793d8b23a0581Lennart Poettering struct rlimit *rl;
8730bccfc59fe507bd3e0a3abcf411b497ac4f0eLennart Poettering
8730bccfc59fe507bd3e0a3abcf411b497ac4f0eLennart Poettering rl = newdup(struct rlimit, saved_rlimit, 1);
8730bccfc59fe507bd3e0a3abcf411b497ac4f0eLennart Poettering if (!rl)
8730bccfc59fe507bd3e0a3abcf411b497ac4f0eLennart Poettering return log_oom();
8730bccfc59fe507bd3e0a3abcf411b497ac4f0eLennart Poettering
8730bccfc59fe507bd3e0a3abcf411b497ac4f0eLennart Poettering arg_default_rlimit[RLIMIT_NOFILE] = rl;
8730bccfc59fe507bd3e0a3abcf411b497ac4f0eLennart Poettering }
6f717d0817573a76c3e586eae02793d8b23a0581Lennart Poettering
6f717d0817573a76c3e586eae02793d8b23a0581Lennart Poettering /* Bump up the resource limit for ourselves substantially */
6f717d0817573a76c3e586eae02793d8b23a0581Lennart Poettering nl.rlim_cur = nl.rlim_max = 64*1024;
6f717d0817573a76c3e586eae02793d8b23a0581Lennart Poettering r = setrlimit_closest(RLIMIT_NOFILE, &nl);
6f717d0817573a76c3e586eae02793d8b23a0581Lennart Poettering if (r < 0) {
8730bccfc59fe507bd3e0a3abcf411b497ac4f0eLennart Poettering log_error("Setting RLIMIT_NOFILE failed: %s", strerror(-r));
8730bccfc59fe507bd3e0a3abcf411b497ac4f0eLennart Poettering 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;
}
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 };
static char systemd[] = "systemd";
bool skip_setup = false;
int 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);
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()) {
char *rd_timestamp = NULL;
initrd_timestamp = userspace_timestamp;
asprintf(&rd_timestamp, "%llu %llu",
(unsigned long long) initrd_timestamp.realtime,
(unsigned long long) initrd_timestamp.monotonic);
if (rd_timestamp) {
setenv("RD_TIMESTAMP", rd_timestamp, 1);
free(rd_timestamp);
}
}
if (!skip_setup) {
mount_setup_early();
if (selinux_setup(&loaded_policy) < 0)
goto finish;
if (ima_setup() < 0)
goto finish;
if (smack_setup() < 0)
goto finish;
}
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_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;
}
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 (serialization)
assert_se(fdset_remove(fds, fileno(serialization)) >= 0);
/* Set up PATH unless it is already set */
setenv("PATH",
#ifdef HAVE_SPLIT_USR
"/usr/local/sbin:/usr/local/bin:/usr/sbin:/usr/bin:/sbin:/bin",
#else
"/usr/local/sbin:/usr/local/bin:/usr/sbin:/usr/bin",
#endif
arg_running_as == SYSTEMD_SYSTEM);
if (arg_running_as == SYSTEMD_SYSTEM) {
/* Unset some environment variables passed in from the
* kernel that don't really make sense for us. */
unsetenv("HOME");
unsetenv("TERM");
/* When we are invoked by a shell, these might be set,
* but make little sense to pass on */
unsetenv("PWD");
unsetenv("SHLVL");
unsetenv("_");
/* When we are invoked by a chroot-like tool such as
* nspawn, these might be set, but make little sense
* to pass on */
unsetenv("USER");
unsetenv("LOGNAME");
/* We suppress the socket activation env vars, as
* we'll try to match *any* open fd to units if
* possible. */
unsetenv("LISTEN_FDS");
unsetenv("LISTEN_PID");
/* All other variables are left as is, so that clients
* can still read them via /proc/1/environ */
/* Become a session leader if we aren't one yet. */
setsid();
/* Disable the umask logic */
umask(0);
}
/* Move out of the way, so that we won't block unmounts */
assert_se(chdir("/") == 0);
/* Make sure D-Bus doesn't fiddle with the SIGPIPE handlers */
dbus_connection_set_change_sigpipe(FALSE);
/* 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
log_debug(PACKAGE_STRING " running in user mode. (" SYSTEMD_FEATURES ")");
if (arg_running_as == SYSTEMD_SYSTEM && !skip_setup) {
locale_setup();
if (arg_show_status || plymouth_running())
status_welcome();
#ifdef HAVE_KMOD
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(arg_capability_bounding_set_drop, true);
if (r < 0) {
log_error("Failed to drop capability bounding set: %s", strerror(-r));
goto finish;
}
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;
}
}
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, !!serialization, &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->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;
manager_set_default_rlimits(m, arg_default_rlimit);
if (arg_default_environment)
manager_set_default_environment(m, arg_default_environment);
manager_set_show_status(m, arg_show_status);
/* Remember whether we should queue the default job */
queue_default_job = !serialization || arg_switched_root;
before_startup = now(CLOCK_MONOTONIC);
r = manager_startup(m, 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);
if (serialization) {
fclose(serialization);
serialization = NULL;
}
if (queue_default_job) {
DBusError error;
Unit *target = NULL;
Job *default_unit_job;
dbus_error_init(&error);
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(&error, r));
dbus_error_free(&error);
} 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(&error, r));
dbus_error_free(&error);
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(&error, r));
dbus_error_free(&error);
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(&error, r));
dbus_error_free(&error);
goto finish;
}
} else if (r < 0) {
log_error("Failed to isolate default target: %s", bus_error(&error, r));
dbus_error_free(&error);
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, &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, &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);
for (j = 0; j < RLIMIT_NLIMITS; j++)
free(arg_default_rlimit[j]);
free(arg_default_unit);
free_join_controllers();
dbus_shutdown();
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);
/* 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(serialization);
assert(fds);
snprintf(sfd, sizeof(sfd), "%i", fileno(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;
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 (serialization) {
fclose(serialization);
serialization = NULL;
}
if (fds) {
fdset_free(fds);
fds = NULL;
}
/* Reopen the console */
make_console_stdio();
for (j = 1, i = 1; j < 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 (serialization)
fclose(serialization);
if (fds)
fdset_free(fds);
if (shutdown_verb) {
const char * command_line[] = {
SYSTEMD_SHUTDOWN_BINARY_PATH,
shutdown_verb,
NULL
};
char **env_block;
if (arm_reboot_watchdog && arg_shutdown_watchdog > 0) {
char e[32];
/* 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 */
snprintf(e, sizeof(e), "WATCHDOG_USEC=%llu", (unsigned long long) arg_shutdown_watchdog);
char_array_0(e);
env_block = strv_append(environ, e);
} else {
env_block = strv_copy(environ);
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);
free(env_block);
log_error("Failed to execute shutdown binary, freezing: %m");
}
if (getpid() == 1)
freeze();
return retval;
}