systemctl.c revision 8d5ba5a946388c965632713f6c1abfb3acba17f7
/*-*- Mode: C; c-basic-offset: 8; indent-tabs-mode: nil -*-*/
/***
This file is part of systemd.
Copyright 2010 Lennart Poettering
Copyright 2013 Marc-Antoine Perennou
under the terms of the GNU Lesser General Public License as published by
the Free Software Foundation; either version 2.1 of the License, or
(at your option) any later version.
systemd is distributed in the hope that it will be useful, but
WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
Lesser General Public License for more details.
You should have received a copy of the GNU Lesser General Public License
along with systemd; If not, see <http://www.gnu.org/licenses/>.
***/
#include <stdio.h>
#include <getopt.h>
#include <locale.h>
#include <stdbool.h>
#include <string.h>
#include <errno.h>
#include <termios.h>
#include <unistd.h>
#include <fcntl.h>
#include <stddef.h>
#include <fnmatch.h>
#include "sd-daemon.h"
#include "sd-shutdown.h"
#include "sd-login.h"
#include "sd-bus.h"
#include "log.h"
#include "util.h"
#include "macro.h"
#include "set.h"
#include "utmp-wtmp.h"
#include "special.h"
#include "initreq.h"
#include "path-util.h"
#include "strv.h"
#include "cgroup-show.h"
#include "cgroup-util.h"
#include "list.h"
#include "path-lookup.h"
#include "conf-parser.h"
#include "exit-status.h"
#include "bus-errors.h"
#include "build.h"
#include "unit-name.h"
#include "pager.h"
#include "spawn-ask-password-agent.h"
#include "spawn-polkit-agent.h"
#include "install.h"
#include "logs-show.h"
#include "socket-util.h"
#include "fileio.h"
#include "bus-util.h"
#include "bus-message.h"
#include "bus-error.h"
#include "bus-errors.h"
static char **arg_states = NULL;
static char **arg_properties = NULL;
static bool arg_all = false;
static bool original_stdout_is_tty;
static enum dependency {
static const char *arg_job_mode = "replace";
static bool arg_no_block = false;
static bool arg_no_legend = false;
static bool arg_no_pager = false;
static bool arg_no_wtmp = false;
static bool arg_no_wall = false;
static bool arg_no_reload = false;
static bool arg_show_types = false;
static bool arg_ignore_inhibitors = false;
static bool arg_dry = false;
static bool arg_quiet = false;
static bool arg_full = false;
static int arg_force = 0;
static bool arg_ask_password = true;
static bool arg_runtime = false;
static const char *arg_kill_who = NULL;
static int arg_signal = SIGTERM;
static enum action {
static unsigned arg_lines = 10;
static bool arg_plain = false;
static char** strv_skip_first(char **strv) {
if (strv_length(strv) > 0)
return strv + 1;
return NULL;
}
static void pager_open_if_enabled(void) {
if (arg_no_pager)
return;
pager_open(false);
}
static void ask_password_agent_open_if_enabled(void) {
/* Open the password agent as a child process if necessary */
if (!arg_ask_password)
return;
if (arg_scope != UNIT_FILE_SYSTEM)
return;
if (arg_transport != BUS_TRANSPORT_LOCAL)
return;
}
#ifdef HAVE_LOGIND
static void polkit_agent_open_if_enabled(void) {
/* Open the polkit agent as a child process if necessary */
if (!arg_ask_password)
return;
if (arg_scope != UNIT_FILE_SYSTEM)
return;
if (arg_transport != BUS_TRANSPORT_LOCAL)
return;
}
#endif
if (!sd_bus_error_is_set(error))
return r;
return EXIT_NOPERMISSION;
return EXIT_NOTINSTALLED;
return EXIT_NOTIMPLEMENTED;
return EXIT_NOTCONFIGURED;
if (r != 0)
return r;
return EXIT_FAILURE;
}
static const char *table[_ACTION_MAX] = {
[ACTION_HALT] = "The system is going down for system halt NOW!",
[ACTION_REBOOT] = "The system is going down for reboot NOW!",
[ACTION_POWEROFF] = "The system is going down for power-off NOW!",
[ACTION_KEXEC] = "The system is going down for kexec reboot NOW!",
[ACTION_RESCUE] = "The system is going down to rescue mode NOW!",
[ACTION_EMERGENCY] = "The system is going down to emergency mode NOW!",
[ACTION_CANCEL_SHUTDOWN] = "The system shutdown has been cancelled NOW!"
};
if (arg_no_wall)
return;
if (arg_wall) {
_cleanup_free_ char *p;
if (!p) {
log_oom();
return;
}
if (*p) {
return;
}
}
if (!table[a])
return;
}
static bool avoid_bus(void) {
if (running_in_chroot() > 0)
return true;
if (sd_booted() <= 0)
return true;
return true;
if (arg_scope == UNIT_FILE_GLOBAL)
return true;
return false;
}
static int compare_unit_info(const void *a, const void *b) {
const UnitInfo *u = a, *v = b;
int r;
if (r != 0)
return r;
}
}
const char *dot;
if (!strv_isempty(arg_states))
return
if (!strv_isempty(patterns)) {
char **pattern;
return true;
return false;
}
}
const UnitInfo *u;
unsigned n_shown = 0;
int job_count = 0;
desc_len = 0;
for (u = unit_infos; u < unit_infos + c; u++) {
if (u->job_id != 0) {
job_count++;
}
}
if (!arg_full && original_stdout_is_tty) {
unsigned basic_len;
if (job_count)
/* Either UNIT already got 25, or is fully satisfied.
* Grant up to 25 to DESC now. */
/* split the remaining space between UNIT and DESC,
* but do not give UNIT more than it needs. */
if (extra_len > 0) {
}
}
} else
id_len = max_id_len;
for (u = unit_infos; u < unit_infos + c; u++) {
_cleanup_free_ char *e = NULL;
if (!n_shown && !arg_no_legend) {
printf("%-*s %-*s %-*s %-*s ",
id_len, "UNIT",
load_len, "LOAD",
active_len, "ACTIVE",
sub_len, "SUB");
if (job_count)
if (!arg_full && arg_no_pager)
else
}
n_shown++;
} else
} else
printf("%s%-*s%s %s%-*s%s %s%-*s %-*s%s %-*s",
if (desc_len > 0)
else
}
if (!arg_no_legend) {
if (n_shown) {
printf("\nLOAD = Reflects whether the unit definition was properly loaded.\n"
"ACTIVE = The high-level unit activation state, i.e. generalization of SUB.\n"
"SUB = The low-level unit activation state, values depend on unit type.\n");
if (job_count)
printf("JOB = Pending job for the unit.\n");
puts("");
on = ansi_highlight();
off = ansi_highlight_off();
} else {
on = ansi_highlight_red();
off = ansi_highlight_off();
}
if (arg_all)
printf("%s%u loaded units listed.%s\n"
"To show all installed unit files use 'systemctl list-unit-files'.\n",
else
printf("%s%u loaded units listed.%s Pass --all to see loaded but inactive units, too.\n"
"To show all installed unit files use 'systemctl list-unit-files'.\n",
}
}
static int get_unit_list(
char **patterns) {
int r, c = 0;
UnitInfo u;
r = sd_bus_call_method(
bus,
"org.freedesktop.systemd1",
"/org/freedesktop/systemd1",
"org.freedesktop.systemd1.Manager",
"ListUnits",
&error,
&reply,
NULL);
if (r < 0) {
return r;
}
if (r < 0)
return bus_log_parse_error(r);
while ((r = bus_parse_unit_info(reply, &u)) > 0) {
if (!output_show_unit(&u, patterns))
continue;
return log_oom();
unit_infos[c++] = u;
}
if (r < 0)
return bus_log_parse_error(r);
if (r < 0)
return bus_log_parse_error(r);
unit_infos = NULL;
return c;
}
int r;
if (r < 0)
return r;
return 0;
}
static int get_triggered_units(
const char* path,
char*** ret) {
int r;
bus,
"org.freedesktop.systemd1",
path,
"org.freedesktop.systemd1.Unit",
"Triggers",
&error,
ret);
if (r < 0)
return 0;
}
static int get_listening(
const char* unit_path,
char*** listening) {
int r, n = 0;
r = sd_bus_get_property(
bus,
"org.freedesktop.systemd1",
"org.freedesktop.systemd1.Socket",
"Listen",
&error,
&reply,
"a(ss)");
if (r < 0) {
return r;
}
if (r < 0)
return bus_log_parse_error(r);
if (r < 0)
return log_oom();
if (r < 0)
return log_oom();
n++;
}
if (r < 0)
return bus_log_parse_error(r);
if (r < 0)
return bus_log_parse_error(r);
return n;
}
struct socket_info {
const char* id;
char* type;
char* path;
/* Note: triggered is a list here, although it almost certainly
* will always be one unit. Nevertheless, dbus API allows for multiple
* values, so let's follow that.*/
char** triggered;
/* The strv above is shared. free is set only in the first one. */
bool own_triggered;
};
int o;
assert(a);
assert(b);
if (o == 0)
return o;
}
struct socket_info *s;
unsigned tmp = 0;
char **a;
if (arg_show_types)
STRV_FOREACH(a, s->triggered)
}
if (cs) {
if (!arg_no_legend)
printf("%-*s %-*.*s%-*s %s\n",
pathlen, "LISTEN",
socklen, "UNIT",
"ACTIVATES");
char **a;
if (arg_show_types)
printf("%-*s %-*s %-*s",
else
printf("%-*s %-*s",
STRV_FOREACH(a, s->triggered)
printf("%s %s",
printf("\n");
}
on = ansi_highlight();
off = ansi_highlight_off();
if (!arg_no_legend)
printf("\n");
} else {
on = ansi_highlight_red();
off = ansi_highlight_off();
}
if (!arg_no_legend) {
if (!arg_all)
printf("Pass --all to see loaded but inactive sockets, too.\n");
}
return 0;
}
const UnitInfo *u;
struct socket_info *s;
unsigned cs = 0;
int r = 0, n;
if (n < 0)
return n;
for (u = unit_infos; u < unit_infos + n; u++) {
int i, c;
continue;
if (r < 0)
goto cleanup;
if (c < 0) {
r = c;
goto cleanup;
}
r = log_oom();
goto cleanup;
}
for (i = 0; i < c; i++)
.own_triggered = i==0,
};
/* from this point on we will cleanup those socket_infos */
cs += c;
}
if (s->own_triggered)
}
return r;
}
static int get_next_elapse(
const char *path,
dual_timestamp *next) {
int r;
bus,
"org.freedesktop.systemd1",
path,
"org.freedesktop.systemd1.Timer",
"NextElapseUSecMonotonic",
&error,
't',
&t.monotonic);
if (r < 0) {
return r;
}
bus,
"org.freedesktop.systemd1",
path,
"org.freedesktop.systemd1.Timer",
"NextElapseUSecRealtime",
&error,
't',
&t.realtime);
if (r < 0) {
return r;
}
*next = t;
return 0;
}
struct timer_info {
const char* id;
char** triggered;
};
assert(a);
assert(b);
if (a->next_elapse < b->next_elapse)
return -1;
if (a->next_elapse > b->next_elapse)
return 1;
}
struct timer_info *t;
unsigned
assert(timer_infos || n == 0);
for (t = timer_infos; t < timer_infos + n; t++) {
unsigned ul = 0;
char **a;
if (t->next_elapse > 0) {
}
STRV_FOREACH(a, t->triggered)
}
if (n > 0) {
if (!arg_no_legend)
printf("%-*s %-*s %-*s %s\n",
nextlen, "NEXT",
leftlen, "LEFT",
unitlen, "UNIT",
"ACTIVATES");
for (t = timer_infos; t < timer_infos + n; t++) {
char **a;
printf("%-*s %-*s %-*s",
STRV_FOREACH(a, t->triggered)
printf("%s %s",
printf("\n");
}
on = ansi_highlight();
off = ansi_highlight_off();
if (!arg_no_legend)
printf("\n");
} else {
on = ansi_highlight_red();
off = ansi_highlight_off();
}
if (!arg_no_legend) {
if (!arg_all)
printf("Pass --all to see loaded but inactive timers, too.\n");
}
return 0;
}
struct timer_info *t;
const UnitInfo *u;
int n, c = 0;
int r = 0;
if (n < 0)
return n;
for (u = unit_infos; u < unit_infos + n; u++) {
usec_t m;
continue;
if (r < 0)
goto cleanup;
if (r < 0)
goto cleanup;
else
else
m = converted;
} else
r = log_oom();
goto cleanup;
}
timer_infos[c++] = (struct timer_info) {
.next_elapse = m,
};
}
for (t = timer_infos; t < timer_infos + c; t++)
return r;
}
static int compare_unit_file_list(const void *a, const void *b) {
const UnitFileList *u = a, *v = b;
int r;
if (r != 0)
return r;
}
}
const char *dot;
if (!strv_isempty(patterns)) {
char **pattern;
return true;
return false;
}
}
const UnitFileList *u;
}
if (!arg_full) {
unsigned basic_cols;
if (basic_cols < (unsigned) columns())
} else
if (!arg_no_legend)
printf("%-*s %-*s\n",
id_cols, "UNIT FILE",
state_cols, "STATE");
_cleanup_free_ char *e = NULL;
const char *id;
n_shown++;
if (u->state == UNIT_FILE_MASKED ||
u->state == UNIT_FILE_MASKED_RUNTIME ||
u->state == UNIT_FILE_DISABLED ||
u->state == UNIT_FILE_INVALID) {
on = ansi_highlight_red();
off = ansi_highlight_off();
} else if (u->state == UNIT_FILE_ENABLED) {
on = ansi_highlight_green();
off = ansi_highlight_off();
} else
printf("%-*s %s%-*s%s\n",
}
if (!arg_no_legend)
}
unsigned c = 0;
const char *state;
char *path;
int r;
if (avoid_bus()) {
Hashmap *h;
UnitFileList *u;
Iterator i;
unsigned n_units;
if (!h)
return log_oom();
if (r < 0) {
return r;
}
n_units = hashmap_size(h);
if (!units) {
return log_oom();
}
HASHMAP_FOREACH(u, h, i) {
continue;
units[c++] = *u;
free(u);
}
hashmap_free(h);
} else {
r = sd_bus_call_method(
bus,
"org.freedesktop.systemd1",
"/org/freedesktop/systemd1",
"org.freedesktop.systemd1.Manager",
"ListUnitFiles",
&error,
&reply,
NULL);
if (r < 0) {
return r;
}
if (r < 0)
return bus_log_parse_error(r);
return log_oom();
units[c] = (struct UnitFileList) {
path,
};
c ++;
}
if (r < 0)
return bus_log_parse_error(r);
if (r < 0)
return bus_log_parse_error(r);
}
if (c > 0) {
}
if (avoid_bus())
return 0;
}
_cleanup_free_ char *n = NULL;
int i;
if (!arg_plain) {
for (i = level - 1; i >= 0; i--) {
len += 2;
return 0;
}
}
len += 2;
return 0;
}
}
if (arg_full){
return 0;
}
if (!n)
return log_oom();
printf("%s\n", n);
return 0;
}
static const char *dependencies[_DEPENDENCY_MAX] = {
[DEPENDENCY_FORWARD] = "Requires\0"
"RequiresOverridable\0"
"Requisite\0"
"RequisiteOverridable\0"
"Wants\0",
[DEPENDENCY_REVERSE] = "RequiredBy\0"
"RequiredByOverridable\0"
"WantedBy\0"
"PartOf\0",
[DEPENDENCY_AFTER] = "After\0",
[DEPENDENCY_BEFORE] = "Before\0",
};
int r;
if (!path)
return log_oom();
r = sd_bus_call_method(
bus,
"org.freedesktop.systemd1",
path,
"org.freedesktop.DBus.Properties",
"GetAll",
&error,
&reply,
"s", "org.freedesktop.systemd1.Unit");
if (r < 0) {
return r;
}
if (r < 0)
return bus_log_parse_error(r);
const char *prop;
if (r < 0)
return bus_log_parse_error(r);
if (r < 0)
return bus_log_parse_error(r);
} else {
if (r < 0)
return bus_log_parse_error(r);
if (r < 0)
return bus_log_parse_error(r);
if (r < 0)
return bus_log_parse_error(r);
}
if (r < 0)
return bus_log_parse_error(r);
}
if (r < 0)
return bus_log_parse_error(r);
if (r < 0)
return bus_log_parse_error(r);
return 0;
}
return 1;
return -1;
return strcasecmp(*a, *b);
}
static int list_dependencies_one(
const char *name,
int level,
char ***units,
unsigned int branches) {
char **c;
int r = 0;
if (!u)
return log_oom();
if (r < 0)
return r;
STRV_FOREACH(c, deps) {
int state;
if (strv_contains(u, *c)) {
if (!arg_plain) {
if (r < 0)
return r;
}
continue;
}
if (state > 0)
printf("%s%s%s", ansi_highlight_green(), draw_special_char(DRAW_BLACK_CIRCLE), ansi_highlight_off());
else
if (r < 0)
return r;
if (r < 0)
return r;
}
}
if (arg_plain) {
*units = u;
u = NULL;
}
return 0;
}
const char *u;
if (args[1]) {
if (!unit)
return log_oom();
u = unit;
} else
puts(u);
}
const char *path;
int r;
if (r < 0) {
return r;
}
} else {
r = sd_bus_call_method(
bus,
"org.freedesktop.systemd1",
"/org/freedesktop/systemd1",
"org.freedesktop.systemd1.Manager",
"GetDefaultTarget",
&error,
&reply,
NULL);
if (r < 0) {
return r;
}
if (r < 0)
return bus_log_parse_error(r);
}
if (path)
return 0;
}
unsigned i;
for (i = 0; i < n_changes; i++) {
else
}
}
static int deserialize_and_dump_unit_file_changes(sd_bus_message *m) {
int r;
if (r < 0)
return bus_log_parse_error(r);
if (!arg_quiet) {
else
}
}
if (r < 0)
return bus_log_parse_error(r);
r = sd_bus_message_exit_container(m);
if (r < 0)
return bus_log_parse_error(r);
return 0;
}
unsigned n_changes = 0;
int r;
if (!unit)
return log_oom();
if (r < 0) {
return r;
}
if (!arg_quiet)
r = 0;
} else {
r = sd_bus_call_method(
bus,
"org.freedesktop.systemd1",
"/org/freedesktop/systemd1",
"org.freedesktop.systemd1.Manager",
"SetDefaultTarget",
&error,
&reply,
if (r < 0) {
return r;
}
if (r < 0)
return r;
/* Try to reload if enabeld */
if (!arg_no_reload)
else
r = 0;
}
return r;
}
struct job_info {
};
const struct job_info *j;
bool shorten = false;
if (n == 0) {
on = ansi_highlight_green();
off = ansi_highlight_off();
return;
}
}
shorten = true;
}
if (!arg_no_legend)
printf("%*s %-*s %-*s %-*s\n",
id_len, "JOB",
unit_len, "UNIT",
type_len, "TYPE",
state_len, "STATE");
_cleanup_free_ char *e = NULL;
on = ansi_highlight();
off = ansi_highlight_off();
} else
printf("%*u %s%-*s%s %-*s %s%-*s%s\n",
}
if (!arg_no_legend) {
on = ansi_highlight();
off = ansi_highlight_off();
}
}
if (!strv_isempty(patterns)) {
char **pattern;
return true;
return false;
}
return true;
}
unsigned c = 0;
int r;
bool skipped = false;
r = sd_bus_call_method(
bus,
"org.freedesktop.systemd1",
"/org/freedesktop/systemd1",
"org.freedesktop.systemd1.Manager",
"ListJobs",
&error,
&reply,
NULL);
if (r < 0) {
return r;
}
if (r < 0)
return bus_log_parse_error(r);
while ((r = sd_bus_message_read(reply, "(usssoo)", &id, &name, &type, &state, &job_path, &unit_path)) > 0) {
skipped = true;
continue;
}
return log_oom();
}
if (r < 0)
return bus_log_parse_error(r);
if (r < 0)
return bus_log_parse_error(r);
return r;
}
char **name;
int r;
if (r < 0) {
return r;
}
r = sd_bus_call_method(
bus,
"org.freedesktop.systemd1",
"/org/freedesktop/systemd1",
"org.freedesktop.systemd1.Manager",
"CancelJob",
&error,
NULL,
"u", id);
if (r < 0) {
return r;
}
}
return 0;
}
_cleanup_free_ char *n = NULL;
const char *path;
int b, r;
/* We ignore all errors here, since this is used to show a
* warning only */
n = unit_name_mangle(unit);
if (!n)
return -ENOMEM;
/* We don't use unit_dbus_path_from_name() directly since we
* don't want to load the unit if it isn't loaded. */
r = sd_bus_call_method(
bus,
"org.freedesktop.systemd1",
"/org/freedesktop/systemd1",
"org.freedesktop.systemd1.Manager",
"GetUnit",
NULL,
&reply,
"s", n);
if (r < 0)
return r;
if (r < 0)
return r;
bus,
"org.freedesktop.systemd1",
path,
"org.freedesktop.systemd1.Unit",
"NeedDaemonReload",
NULL,
'b', &b);
if (r < 0)
return r;
return b;
}
typedef struct WaitData {
char *name;
char *result;
} WaitData;
assert(m);
assert(d);
log_debug("Got D-Bus request: %s.%s() on %s",
log_error("Warning! D-Bus connection terminated.");
char *ret;
int r;
if (r >= 0) {
if (!ret)
return 0;
return 0;
}
#ifndef NOLEGACY
if (r >= 0) {
if (!ret)
return 0;
if (*result)
return 0;
}
#endif
}
return 0;
}
int r;
r = sd_bus_add_match(
bus,
"type='signal',"
"sender='org.freedesktop.systemd1',"
"interface='org.freedesktop.systemd1.Manager',"
"member='JobRemoved',"
"path='/org/freedesktop/systemd1'",
if (r < 0) {
log_error("Failed to add match");
return -EIO;
}
/* This is slightly dirty, since we don't undo the match registrations. */
return 0;
}
int r;
for (;;) {
if (r < 0)
return r;
if (r > 0)
return 0;
if (r < 0)
return r;
}
}
static int check_wait_response(WaitData *d) {
int r = 0;
if (!arg_quiet) {
log_error("Job for %s failed. See 'systemctl status %s' and 'journalctl -xn' for details.", strna(d->name), strna(d->name));
}
r = -ETIME;
r = -ECANCELED;
r = -EIO;
r = -EIO;
return r;
}
int r = 0, q;
assert(s);
if (q < 0)
return log_oom();
while (!set_isempty(s)) {
q = bus_process_wait(bus);
if (q < 0)
return q;
if (d.result) {
q = check_wait_response(&d);
/* Return the first error as it is most likely to be
* meaningful. */
if (q < 0 && r == 0)
r = q;
}
}
if (q < 0 && r == 0)
r = q;
return r;
}
const char *path;
int r;
n = unit_name_mangle(name);
if (!n)
return log_oom();
/* We don't use unit_dbus_path_from_name() directly since we
* don't want to load the unit if it isn't loaded. */
r = sd_bus_call_method(
bus,
"org.freedesktop.systemd1",
"/org/freedesktop/systemd1",
"org.freedesktop.systemd1.Manager",
"GetUnit",
NULL,
&reply,
"s", n);
if (r < 0) {
if (!quiet)
puts("unknown");
return 0;
}
if (r < 0)
return bus_log_parse_error(r);
bus,
"org.freedesktop.systemd1",
path,
"org.freedesktop.systemd1.Unit",
"ActiveState",
NULL,
&state);
if (r < 0) {
if (!quiet)
puts("unknown");
return 0;
}
if (!quiet)
}
static int check_triggering_units(
const char *name) {
bool print_warning_label = true;
char **i;
int r;
n = unit_name_mangle(name);
if (!n)
return log_oom();
path = unit_dbus_path_from_name(n);
if (!path)
return log_oom();
bus,
"org.freedesktop.systemd1",
path,
"org.freedesktop.systemd1.Unit",
"LoadState",
&error,
&state);
if (r < 0) {
return r;
}
return 0;
bus,
"org.freedesktop.systemd1",
path,
"org.freedesktop.systemd1.Unit",
"TriggeredBy",
&error,
&triggered_by);
if (r < 0) {
return r;
}
STRV_FOREACH(i, triggered_by) {
if (r < 0) {
return r;
}
if (r == 0)
continue;
if (print_warning_label) {
log_warning("Warning: Stopping %s, but it can still be activated by:", n);
print_warning_label = false;
}
log_warning(" %s", *i);
}
return 0;
}
static int start_unit_one(
const char *method,
const char *name,
const char *mode,
Set *s) {
_cleanup_free_ char *n;
const char *path;
int r;
n = unit_name_mangle(name);
if (!n)
return log_oom();
r = sd_bus_call_method(
bus,
"org.freedesktop.systemd1",
"/org/freedesktop/systemd1",
"org.freedesktop.systemd1.Manager",
&reply,
"ss", n, mode);
if (r < 0) {
/* There's always a fallback possible for
* legacy actions. */
return -EADDRNOTAVAIL;
return r;
}
if (r < 0)
return bus_log_parse_error(r);
if (need_daemon_reload(bus, n) > 0)
log_warning("Warning: Unit file of %s changed on disk, 'systemctl%s daemon-reload' recommended.",
if (s) {
char *p;
if (!p)
return log_oom();
r = set_consume(s, p);
if (r < 0)
return log_oom();
}
return 0;
}
static const struct {
const char *target;
const char *verb;
const char *mode;
} action_table[_ACTION_MAX] = {
};
enum action i;
for (i = _ACTION_INVALID; i < _ACTION_MAX; i++)
return i;
return _ACTION_INVALID;
}
char **name;
int r = 0;
if (arg_action == ACTION_SYSTEMCTL) {
method =
"StartUnit";
} else {
method = "StartUnit";
}
if (strv[0])
else
if (!arg_no_block) {
r = enable_wait_for_jobs(bus);
if (r < 0) {
return r;
}
if (!s)
return log_oom();
}
int q;
if (r == 0 && q < 0) {
r = translate_bus_error_to_exit_status(q, &error);
}
}
if (!arg_no_block) {
int q;
q = wait_for_jobs(bus, s);
if (q < 0)
return q;
/* When stopping units, warn if they can still be triggered by
* another active unit (socket, path, timer) */
}
return r;
}
/* Ask systemd-logind, which might grant access to unprivileged users
* through PolicyKit */
#ifdef HAVE_LOGIND
const char *method;
int r;
if (!bus)
return -EIO;
switch (a) {
case ACTION_REBOOT:
method = "Reboot";
break;
case ACTION_POWEROFF:
method = "PowerOff";
break;
case ACTION_SUSPEND:
method = "Suspend";
break;
case ACTION_HIBERNATE:
method = "Hibernate";
break;
case ACTION_HYBRID_SLEEP:
method = "HybridSleep";
break;
default:
return -EINVAL;
}
r = sd_bus_call_method(
bus,
"org.freedesktop.login1",
"/org/freedesktop/login1",
"org.freedesktop.login1.Manager",
&error,
NULL,
"b", true);
if (r < 0)
return r;
#else
return -ENOSYS;
#endif
}
#ifdef HAVE_LOGIND
unsigned c = 0;
char **s;
int r;
if (!bus)
return 0;
if (arg_ignore_inhibitors || arg_force > 0)
return 0;
if (arg_when > 0)
return 0;
if (geteuid() == 0)
return 0;
if (!on_tty())
return 0;
r = sd_bus_call_method(
bus,
"org.freedesktop.login1",
"/org/freedesktop/login1",
"org.freedesktop.login1.Manager",
"ListInhibitors",
NULL,
&reply,
NULL);
if (r < 0)
/* If logind is not around, then there are no inhibitors... */
return 0;
if (r < 0)
return bus_log_parse_error(r);
continue;
if (!sv)
return log_oom();
if (!strv_contains(sv,
a == ACTION_HALT ||
a == ACTION_POWEROFF ||
a == ACTION_REBOOT ||
continue;
log_warning("Operation inhibited by \"%s\" (PID %lu \"%s\", user %s), reason is \"%s\".",
c++;
}
if (r < 0)
return bus_log_parse_error(r);
if (r < 0)
return bus_log_parse_error(r);
/* Check for current sessions */
STRV_FOREACH(s, sessions) {
_cleanup_free_ char *type = NULL, *tty = NULL, *seat = NULL, *user = NULL, *service = NULL, *class = NULL;
continue;
continue;
continue;
sd_session_get_tty(*s, &tty);
sd_session_get_seat(*s, &seat);
sd_session_get_service(*s, &service);
log_warning("User %s is logged in on %s.", strna(user), isempty(tty) ? (isempty(seat) ? strna(service) : seat) : tty);
c++;
}
if (c <= 0)
return 0;
log_error("Please retry operation after closing inhibitors and logging out other users.\nAlternatively, ignore inhibitors and users with 'systemctl %s -i'.",
action_table[a].verb);
return -EPERM;
#else
return 0;
#endif
}
enum action a;
int r;
a = verb_to_action(args[0]);
r = check_inhibitors(bus, a);
if (r < 0)
return r;
log_error("Must be root.");
return -EPERM;
}
if (arg_force >= 2 &&
(a == ACTION_HALT ||
a == ACTION_POWEROFF ||
a == ACTION_REBOOT))
return halt_now(a);
if (arg_force >= 1 &&
(a == ACTION_HALT ||
a == ACTION_POWEROFF ||
a == ACTION_REBOOT ||
a == ACTION_KEXEC ||
a == ACTION_EXIT))
/* first try logind, to allow authentication with polkit */
if (geteuid() != 0 &&
(a == ACTION_POWEROFF ||
a == ACTION_REBOOT ||
a == ACTION_SUSPEND ||
a == ACTION_HIBERNATE ||
a == ACTION_HYBRID_SLEEP)) {
r = reboot_with_logind(bus, a);
if (r >= 0)
return r;
}
if (r == EXIT_SUCCESS)
warn_wall(a);
return r;
}
char **name;
int r = 3; /* According to LSB: "program is not running" */
int state;
if (state < 0)
return state;
if (state > 0)
r = 0;
}
return r;
}
char **name;
int r = 1;
int state;
if (state < 0)
return state;
if (state > 0)
r = 0;
}
return r;
}
char **name;
int r = 0;
if (!arg_kill_who)
arg_kill_who = "all";
_cleanup_free_ char *n = NULL;
n = unit_name_mangle(*name);
if (!n)
return log_oom();
r = sd_bus_call_method(
bus,
"org.freedesktop.systemd1",
"/org/freedesktop/systemd1",
"org.freedesktop.systemd1.Manager",
"KillUnit",
&error,
NULL,
if (r < 0) {
return r;
}
}
return 0;
}
typedef struct ExecStatusInfo {
char *name;
char *path;
char **argv;
bool ignore;
int code;
int status;
static void exec_status_info_free(ExecStatusInfo *i) {
assert(i);
free(i);
}
const char *path;
int ignore, r;
assert(m);
assert(i);
if (r < 0)
return bus_log_parse_error(r);
else if (r == 0)
return 0;
if (r < 0)
return bus_log_parse_error(r);
if (!i->path)
return log_oom();
r = sd_bus_message_read_strv(m, &i->argv);
if (r < 0)
return bus_log_parse_error(r);
r = sd_bus_message_read(m,
"bttttuii",
&ignore,
&pid,
if (r < 0)
return bus_log_parse_error(r);
r = sd_bus_message_exit_container(m);
if (r < 0)
return bus_log_parse_error(r);
return 1;
}
typedef struct UnitStatusInfo {
const char *id;
const char *load_state;
const char *active_state;
const char *sub_state;
const char *unit_file_state;
const char *description;
const char *following;
char **documentation;
const char *fragment_path;
const char *source_path;
const char *control_group;
char **dropin_paths;
const char *load_error;
const char *result;
bool need_daemon_reload;
/* Service */
const char *status_text;
const char *pid_file;
bool running:1;
int exit_code, exit_status;
bool condition_result;
bool failed_condition_trigger;
bool failed_condition_negate;
const char *failed_condition;
const char *failed_condition_param;
/* Socket */
unsigned n_accepted;
unsigned n_connections;
bool accept;
/* Pairs of type, path */
char **listen;
/* Device */
const char *sysfs_path;
/* Mount, Automount */
const char *where;
/* Swap */
const char *what;
static void print_status_info(
UnitStatusInfo *i,
bool *ellipsized) {
ExecStatusInfo *p;
const char *path;
int flags =
on_tty() * OUTPUT_COLOR |
char **t, **t2;
assert(i);
/* This shows pretty information about a unit. See
* print_property() for a low-level property printer */
printf("\n");
if (i->following)
on = ansi_highlight_red();
off = ansi_highlight_off();
} else
if (i->load_error)
printf(" Loaded: %s%s%s (Reason: %s)\n",
else if (path && i->unit_file_state)
printf(" Loaded: %s%s%s (%s; %s)\n",
else if (path)
printf(" Loaded: %s%s%s (%s)\n",
else
printf(" Loaded: %s%s%s\n",
if (!strv_isempty(i->dropin_paths)) {
bool last = false;
char ** dropin;
log_oom();
return;
}
}
}
}
on = ansi_highlight_red();
off = ansi_highlight_off();
on = ansi_highlight_green();
off = ansi_highlight_off();
} else
if (ss)
printf(" Active: %s%s (%s)%s",
else
printf(" Active: %s%s%s",
if (s1)
else if (s2)
else
printf("\n");
if (!i->condition_result && i->condition_timestamp > 0) {
printf(" start condition failed at %s%s%s\n",
if (i->failed_condition_trigger)
printf(" none of the trigger conditions were met\n");
else if (i->failed_condition)
printf(" %s=%s%s was not met\n",
i->failed_condition,
}
if (i->sysfs_path)
if (i->where)
if (i->what)
STRV_FOREACH(t, i->documentation)
if (i->accept)
bool good;
/* Only show exited processes here */
if (p->code == 0)
continue;
if (!good) {
on = ansi_highlight_red();
off = ansi_highlight_off();
} else
if (p->code == CLD_EXITED) {
const char *c;
if (c)
printf("/%s", c);
} else
i->start_timestamp == p->start_timestamp &&
i->exit_timestamp == p->start_timestamp)
/* Let's not show this twice */
i->main_pid = 0;
if (p->pid == i->control_pid)
i->control_pid = 0;
}
if (i->main_pid > 0 || i->control_pid > 0) {
if (i->main_pid > 0) {
if (i->running) {
if (comm)
} else if (i->exit_code > 0) {
if (i->exit_code == CLD_EXITED) {
const char *c;
if (c)
printf("/%s", c);
} else
printf(")");
}
if (i->control_pid > 0)
printf(";");
}
if (i->control_pid > 0) {
_cleanup_free_ char *c = NULL;
get_process_comm(i->control_pid, &c);
if (c)
printf(" (%s)", c);
}
printf("\n");
}
if (i->status_text)
if (i->control_group &&
(i->main_pid > 0 || i->control_pid > 0 || cg_is_empty_recursive(SYSTEMD_CGROUP_CONTROLLER, i->control_group, false) == 0)) {
unsigned c;
unsigned k = 0;
char prefix[] = " ";
c = columns();
if (c > sizeof(prefix) - 1)
c -= sizeof(prefix) - 1;
else
c = 0;
if (i->main_pid > 0)
if (i->control_pid > 0)
extra[k++] = i->control_pid;
}
}
printf("\n");
i->id,
0,
getuid(),
}
if (i->need_daemon_reload)
printf("\n%sWarning:%s Unit file changed on disk, 'systemctl %sdaemon-reload' recommended.\n",
}
static void show_unit_help(UnitStatusInfo *i) {
char **p;
assert(i);
if (!i->documentation) {
return;
}
STRV_FOREACH(p, i->documentation) {
if (startswith(*p, "man:")) {
char *e = NULL;
size_t k;
k = strlen(*p);
if ((*p)[k-1] == ')')
e = strrchr(*p, '(');
if (e) {
log_oom();
return;
}
} else
if (pid < 0) {
log_error("Failed to fork: %m");
continue;
}
if (pid == 0) {
/* Child */
log_error("Failed to execute man: %m");
}
} else
log_info("Can't show: %s", *p);
}
}
static int status_property(const char *name, sd_bus_message *m, UnitStatusInfo *i, const char *contents) {
int r;
assert(m);
assert(i);
switch (contents[0]) {
case SD_BUS_TYPE_STRING: {
const char *s;
r = sd_bus_message_read(m, "s", &s);
if (r < 0)
return bus_log_parse_error(r);
if (!isempty(s)) {
i->id = s;
i->load_state = s;
i->active_state = s;
i->sub_state = s;
i->description = s;
i->fragment_path = s;
i->source_path = s;
#ifndef NOLEGACY
const char *e;
if (e)
i->control_group = e;
}
#endif
i->control_group = s;
i->status_text = s;
i->pid_file = s;
i->sysfs_path = s;
i->where = s;
i->what = s;
i->following = s;
i->unit_file_state = s;
i->result = s;
}
break;
}
case SD_BUS_TYPE_BOOLEAN: {
int b;
r = sd_bus_message_read(m, "b", &b);
if (r < 0)
return bus_log_parse_error(r);
i->accept = b;
i->need_daemon_reload = b;
i->condition_result = b;
break;
}
case SD_BUS_TYPE_UINT32: {
uint32_t u;
r = sd_bus_message_read(m, "u", &u);
if (r < 0)
return bus_log_parse_error(r);
if (u > 0) {
i->running = true;
}
i->control_pid = (pid_t) u;
if (u > 0)
i->n_accepted = u;
i->n_connections = u;
break;
}
case SD_BUS_TYPE_INT32: {
int32_t j;
r = sd_bus_message_read(m, "i", &j);
if (r < 0)
return bus_log_parse_error(r);
i->exit_code = (int) j;
i->exit_status = (int) j;
break;
}
case SD_BUS_TYPE_UINT64: {
uint64_t u;
r = sd_bus_message_read(m, "t", &u);
if (r < 0)
return bus_log_parse_error(r);
i->start_timestamp = (usec_t) u;
i->exit_timestamp = (usec_t) u;
i->active_enter_timestamp = (usec_t) u;
i->inactive_enter_timestamp = (usec_t) u;
i->inactive_exit_timestamp = (usec_t) u;
i->inactive_exit_timestamp_monotonic = (usec_t) u;
i->active_exit_timestamp = (usec_t) u;
i->condition_timestamp = (usec_t) u;
break;
}
case SD_BUS_TYPE_ARRAY:
if (r < 0)
return bus_log_parse_error(r);
if (!info)
return log_oom();
while ((r = exec_status_info_deserialize(m, info)) > 0) {
log_oom();
if (!info)
log_oom();
}
if (r < 0)
return bus_log_parse_error(r);
r = sd_bus_message_exit_container(m);
if (r < 0)
return bus_log_parse_error(r);
return 0;
if (r < 0)
return bus_log_parse_error(r);
if (r < 0)
return r;
if (r < 0)
return r;
}
if (r < 0)
return bus_log_parse_error(r);
r = sd_bus_message_exit_container(m);
if (r < 0)
return bus_log_parse_error(r);
return 0;
r = sd_bus_message_read_strv(m, &i->dropin_paths);
if (r < 0)
return bus_log_parse_error(r);
r = sd_bus_message_read_strv(m, &i->documentation);
if (r < 0)
return bus_log_parse_error(r);
if (r < 0)
return bus_log_parse_error(r);
i->failed_condition = cond;
i->failed_condition_param = param;
}
}
if (r < 0)
return bus_log_parse_error(r);
r = sd_bus_message_exit_container(m);
if (r < 0)
return bus_log_parse_error(r);
} else
goto skip;
break;
case SD_BUS_TYPE_STRUCT_BEGIN:
const char *n, *message;
if (r < 0)
return bus_log_parse_error(r);
i->load_error = message;
} else
goto skip;
break;
default:
goto skip;
}
return 0;
skip:
r = sd_bus_message_skip(m, contents);
if (r < 0)
return bus_log_parse_error(r);
return 0;
}
int r;
assert(m);
/* This is a low-level property printer, see
* print_status_info() for the nicer output */
/* skip what we didn't read */
r = sd_bus_message_skip(m, contents);
return r;
}
switch (contents[0]) {
case SD_BUS_TYPE_STRUCT_BEGIN:
uint32_t u;
if (r < 0)
return bus_log_parse_error(r);
if (u > 0)
else if (arg_all)
return 0;
const char *s;
if (r < 0)
return bus_log_parse_error(r);
return 0;
r = sd_bus_message_read(m, "(ss)", &a, &b);
if (r < 0)
return bus_log_parse_error(r);
return 0;
}
break;
case SD_BUS_TYPE_ARRAY:
const char *path;
int ignore;
if (r < 0)
return bus_log_parse_error(r);
if (r < 0)
return bus_log_parse_error(r);
r = sd_bus_message_exit_container(m);
if (r < 0)
return bus_log_parse_error(r);
return 0;
if (r < 0)
return bus_log_parse_error(r);
if (r < 0)
return bus_log_parse_error(r);
r = sd_bus_message_exit_container(m);
if (r < 0)
return bus_log_parse_error(r);
return 0;
if (r < 0)
return bus_log_parse_error(r);
if (r < 0)
return bus_log_parse_error(r);
r = sd_bus_message_exit_container(m);
if (r < 0)
return bus_log_parse_error(r);
return 0;
const char *base;
if (r < 0)
return bus_log_parse_error(r);
printf("%s={ value=%s ; next_elapse=%s }\n",
base,
}
if (r < 0)
return bus_log_parse_error(r);
r = sd_bus_message_exit_container(m);
if (r < 0)
return bus_log_parse_error(r);
return 0;
ExecStatusInfo info = {};
if (r < 0)
return bus_log_parse_error(r);
while ((r = exec_status_info_deserialize(m, &info)) > 0) {
_cleanup_free_ char *tt;
printf("%s={ path=%s ; argv[]=%s ; ignore_errors=%s ; start_time=[%s] ; stop_time=[%s] ; pid=%u ; code=%s ; status=%i%s%s }\n",
name,
}
r = sd_bus_message_exit_container(m);
if (r < 0)
return bus_log_parse_error(r);
return 0;
if (r < 0)
return bus_log_parse_error(r);
if (r < 0)
return bus_log_parse_error(r);
r = sd_bus_message_exit_container(m);
if (r < 0)
return bus_log_parse_error(r);
return 0;
const char *path;
if (r < 0)
return bus_log_parse_error(r);
if (r < 0)
return bus_log_parse_error(r);
r = sd_bus_message_exit_container(m);
if (r < 0)
return bus_log_parse_error(r);
return 0;
} else if (contents[1] == SD_BUS_TYPE_STRUCT_BEGIN && (streq(name, "BlockIOReadBandwidth") || streq(name, "BlockIOWriteBandwidth"))) {
const char *path;
if (r < 0)
return bus_log_parse_error(r);
if (r < 0)
return bus_log_parse_error(r);
r = sd_bus_message_exit_container(m);
if (r < 0)
return bus_log_parse_error(r);
return 0;
}
break;
}
if (r < 0)
return bus_log_parse_error(r);
if (r == 0) {
r = sd_bus_message_skip(m, contents);
if (r < 0)
return bus_log_parse_error(r);
if (arg_all)
}
return 0;
}
static int show_one(
const char *verb,
const char *path,
bool show_properties,
bool *new_line,
bool *ellipsized) {
UnitStatusInfo info = {};
ExecStatusInfo *p;
int r;
r = sd_bus_call_method(
bus,
"org.freedesktop.systemd1",
path,
"org.freedesktop.DBus.Properties",
"GetAll",
&error,
&reply,
"s", "");
if (r < 0) {
return r;
}
if (r < 0)
return bus_log_parse_error(r);
if (*new_line)
printf("\n");
*new_line = true;
if (r < 0)
return bus_log_parse_error(r);
if (r < 0)
return bus_log_parse_error(r);
if (r < 0)
return bus_log_parse_error(r);
if (show_properties)
else
if (r < 0)
return r;
if (r < 0)
return bus_log_parse_error(r);
if (r < 0)
return bus_log_parse_error(r);
}
if (r < 0)
return bus_log_parse_error(r);
if (r < 0)
return bus_log_parse_error(r);
r = 0;
if (!show_properties) {
else
}
/* According to LSB: "program not running" */
/* 0: program is running or service is OK
* 3: program is not running
* 4: program or service status is unknown
*/
r = 1;
else
r = 3;
}
}
return r;
}
static int get_unit_dbus_path_by_pid(
char **unit) {
char *u;
int r;
r = sd_bus_call_method(
bus,
"org.freedesktop.systemd1",
"/org/freedesktop/systemd1",
"org.freedesktop.systemd1.Manager",
"GetUnitByPID",
&error,
&reply,
"u", pid);
if (r < 0) {
return r;
}
if (r < 0)
return bus_log_parse_error(r);
u = strdup(u);
if (!u)
return log_oom();
*unit = u;
return 0;
}
static int show_all(
const char* verb,
bool show_properties,
bool *new_line,
bool *ellipsized) {
const UnitInfo *u;
unsigned c;
int r;
if (r < 0)
return r;
c = (unsigned) r;
for (u = unit_infos; u < unit_infos + c; u++) {
_cleanup_free_ char *p = NULL;
p = unit_dbus_path_from_name(u->id);
if (!p)
return log_oom();
if (r != 0)
return r;
}
return 0;
}
int r = 0;
char **name;
bool first = true;
char **path;
n = unit_name_mangle(*name);
if (!n)
return log_oom();
unit = unit_dbus_path_from_name(n);
if (!unit)
return log_oom();
if (need_daemon_reload(bus, n) > 0)
log_warning("Unit file of %s changed on disk. Run 'systemctl%s daemon-reload'.",
bus,
"org.freedesktop.systemd1",
unit,
"org.freedesktop.systemd1.Unit",
"FragmentPath",
&error,
if (r < 0) {
continue;
}
bus,
"org.freedesktop.systemd1",
unit,
"org.freedesktop.systemd1.Unit",
"DropInPaths",
&error,
&dropin_paths);
if (r < 0) {
continue;
}
if (first)
first = false;
else
puts("");
if (!isempty(fragment_path)) {
printf("%s# %s%s\n",
if (r < 0) {
continue;
}
}
printf("%s%s# %s%s\n",
*path,
if (r < 0) {
continue;
}
}
}
return r < 0 ? r : 0;
}
int r, ret = 0;
char **name;
bool ellipsized = false;
if (show_properties)
/* If no argument is specified inspect the manager itself */
return show_one(args[0], bus, "/org/freedesktop/systemd1", show_properties, &new_line, &ellipsized);
else
_cleanup_free_ char *n = NULL;
/* Interpret as unit name */
n = unit_name_mangle(*name);
if (!n)
return log_oom();
unit = unit_dbus_path_from_name(n);
if (!unit)
return log_oom();
} else if (show_properties) {
/* Interpret as job id */
return log_oom();
} else {
/* Interpret as PID */
if (r < 0) {
ret = r;
continue;
}
}
}
if (ellipsized && !arg_quiet)
printf("Hint: Some lines were ellipsized, use -l to show in full.\n");
return ret;
}
const char *eq;
char *field;
int r;
assert(m);
if (!eq) {
return -EINVAL;
}
eq ++;
if (r < 0)
return bus_log_create_error(r);
r = parse_boolean(eq);
if (r < 0) {
return -EINVAL;
}
if (r < 0) {
return -EINVAL;
}
uint64_t u;
r = safe_atou64(eq, &u);
if (r < 0) {
return -EINVAL;
}
else {
char *e;
if (e) {
rwm = e+1;
} else {
rwm = "";
}
return -EINVAL;
}
}
else {
char *e;
if (e) {
bandwidth = e+1;
} else {
return -EINVAL;
}
return -EINVAL;
}
if (r < 0) {
return -EINVAL;
}
}
else {
uint64_t u;
char *e;
if (e) {
weight = e+1;
} else {
return -EINVAL;
}
return -EINVAL;
}
r = safe_atou64(weight, &u);
if (r < 0) {
return -EINVAL;
}
}
} else {
return -EINVAL;
}
if (r < 0)
return bus_log_create_error(r);
return 0;
}
_cleanup_free_ char *n = NULL;
char **i;
int r;
bus,
"org.freedesktop.systemd1",
"/org/freedesktop/systemd1",
"org.freedesktop.systemd1.Manager",
"SetUnitProperties",
&m);
if (r < 0)
return bus_log_create_error(r);
if (!n)
return log_oom();
if (r < 0)
return bus_log_create_error(r);
if (r < 0)
return bus_log_create_error(r);
if (r < 0)
return bus_log_create_error(r);
r = append_assignment(m, *i);
if (r < 0)
return r;
r = sd_bus_message_close_container(m);
if (r < 0)
return bus_log_create_error(r);
}
r = sd_bus_message_close_container(m);
if (r < 0)
return bus_log_create_error(r);
if (r < 0) {
return r;
}
return 0;
}
const char *path;
int r;
else
n = strdup("");
if (!n)
return log_oom();
r = sd_bus_call_method(
bus,
"org.freedesktop.systemd1",
"/org/freedesktop/systemd1",
"org.freedesktop.systemd1.Manager",
"CreateSnapshot",
&error,
&reply,
"sb", n, false);
if (r < 0) {
return r;
}
if (r < 0)
return bus_log_parse_error(r);
bus,
"org.freedesktop.systemd1",
path,
"org.freedesktop.systemd1.Unit",
"Id",
&error,
&id);
if (r < 0) {
return r;
}
if (!arg_quiet)
return 0;
}
char **name;
int r;
_cleanup_free_ char *n = NULL;
if (!n)
return log_oom();
r = sd_bus_call_method(
bus,
"org.freedesktop.systemd1",
"/org/freedesktop/systemd1",
"org.freedesktop.systemd1.Manager",
"RemoveSnapshot",
&error,
NULL,
"s", n);
if (r < 0) {
return r;
}
}
return 0;
}
const char *method;
int r;
if (arg_action == ACTION_RELOAD)
method = "Reload";
else if (arg_action == ACTION_REEXEC)
method = "Reexecute";
else {
method =
/* "daemon-reload" */ "Reload";
}
r = sd_bus_call_method(
bus,
"org.freedesktop.systemd1",
"/org/freedesktop/systemd1",
"org.freedesktop.systemd1.Manager",
&error,
NULL,
NULL);
/* There's always a fallback possible for
* legacy actions. */
r = -EADDRNOTAVAIL;
/* On reexecution, we expect a disconnect, not a
* reply */
r = 0;
else if (r < 0)
return r < 0 ? r : 0;
}
char **name;
int r;
_cleanup_free_ char *n;
n = unit_name_mangle(*name);
if (!n)
return log_oom();
r = sd_bus_call_method(
bus,
"org.freedesktop.systemd1",
"/org/freedesktop/systemd1",
"org.freedesktop.systemd1.Manager",
"ResetFailedUnit",
&error,
NULL,
"s", n);
if (r < 0) {
return r;
}
}
return 0;
}
const char *text;
int r;
r = sd_bus_get_property(
bus,
"org.freedesktop.systemd1",
"/org/freedesktop/systemd1",
"org.freedesktop.systemd1.Manager",
"Environment",
&error,
&reply,
"as");
if (r < 0) {
return r;
}
if (r < 0)
return bus_log_parse_error(r);
if (r < 0)
return bus_log_parse_error(r);
if (r < 0)
return bus_log_parse_error(r);
return 0;
}
const char *root;
unsigned l;
int r;
l = strv_length(args);
if (l < 2 || l > 3) {
log_error("Wrong number of arguments.");
return -EINVAL;
}
if (l >= 3)
else {
"init", &init,
NULL);
if (!init)
}
if (!init)
return log_oom();
r = sd_bus_call_method(
bus,
"org.freedesktop.systemd1",
"/org/freedesktop/systemd1",
"org.freedesktop.systemd1.Manager",
"SwitchRoot",
&error,
NULL,
if (r < 0) {
return r;
}
return 0;
}
const char *method;
int r;
? "SetEnvironment"
: "UnsetEnvironment";
bus,
"org.freedesktop.systemd1",
"/org/freedesktop/systemd1",
"org.freedesktop.systemd1.Manager",
&m);
if (r < 0)
return bus_log_create_error(r);
if (r < 0)
return bus_log_create_error(r);
if (r < 0) {
return r;
}
return 0;
}
int r = 0;
#if defined(HAVE_SYSV_COMPAT) && defined(HAVE_CHKCONFIG)
unsigned f = 1, t = 1;
if (arg_scope != UNIT_FILE_SYSTEM)
return 0;
return 0;
/* Processes all SysV units, and reshuffles the array so that
* afterwards only the native units remain */
if (r < 0)
return r;
r = 0;
for (f = 0; args[f]; f++) {
const char *name;
bool found_native = false, found_sysv;
unsigned c = 1;
char **k, *l;
int j;
continue;
if (path_is_absolute(name))
continue;
else
if (!p) {
r = log_oom();
goto finish;
}
free(p);
p = NULL;
if (found_native)
break;
}
if (found_native)
continue;
else
if (!p) {
r = log_oom();
goto finish;
}
if (!found_sysv)
continue;
/* Mark this entry, so that we don't try enabling it as native unit */
args[f] = (char*) "";
argv[c++] =
if (!l) {
r = log_oom();
goto finish;
}
log_info("Executing %s", l);
free(l);
if (pid < 0) {
log_error("Failed to fork: %m");
r = -errno;
goto finish;
} else if (pid == 0) {
/* Child */
}
if (j < 0) {
r = j;
goto finish;
}
if (!arg_quiet)
puts("enabled");
r = 1;
} else {
if (!arg_quiet)
puts("disabled");
}
r = -EINVAL;
goto finish;
}
} else {
r = -EPROTO;
goto finish;
}
}
/* Drop all SysV units */
for (f = 0, t = 0; args[f]; f++) {
continue;
}
#endif
return r;
}
char **i, **l, **name;
if (!l)
return log_oom();
i = l;
/* When enabling units qualified path names are OK,
* too, hence allow them explicitly. */
else
*i = unit_name_mangle(*name);
if (!*i) {
strv_free(l);
return log_oom();
}
i++;
}
*i = NULL;
*mangled_names = l;
return 0;
}
unsigned n_changes = 0;
int carries_install_info = -1;
int r;
if (!args[1])
return 0;
if (r < 0)
return r;
if (r < 0)
return r;
r = unit_file_enable(arg_scope, arg_runtime, arg_root, mangled_names, arg_force, &changes, &n_changes);
carries_install_info = r;
r = unit_file_reenable(arg_scope, arg_runtime, arg_root, mangled_names, arg_force, &changes, &n_changes);
carries_install_info = r;
r = unit_file_link(arg_scope, arg_runtime, arg_root, mangled_names, arg_force, &changes, &n_changes);
r = unit_file_preset(arg_scope, arg_runtime, arg_root, mangled_names, arg_force, &changes, &n_changes);
carries_install_info = r;
r = unit_file_mask(arg_scope, arg_runtime, arg_root, mangled_names, arg_force, &changes, &n_changes);
else
assert_not_reached("Unknown verb");
if (r < 0) {
goto finish;
}
if (!arg_quiet)
r = 0;
} else {
int expect_carries_install_info = false;
bool send_force = true;
const char *method;
method = "EnableUnitFiles";
expect_carries_install_info = true;
method = "DisableUnitFiles";
send_force = false;
method = "ReenableUnitFiles";
expect_carries_install_info = true;
method = "LinkUnitFiles";
method = "PresetUnitFiles";
expect_carries_install_info = true;
method = "MaskUnitFiles";
method = "UnmaskUnitFiles";
send_force = false;
} else
assert_not_reached("Unknown verb");
bus,
"org.freedesktop.systemd1",
"/org/freedesktop/systemd1",
"org.freedesktop.systemd1.Manager",
&m);
if (r < 0)
return bus_log_create_error(r);
r = sd_bus_message_append_strv(m, mangled_names);
if (r < 0)
return bus_log_create_error(r);
if (r < 0)
return bus_log_create_error(r);
if (send_force) {
if (r < 0)
return bus_log_create_error(r);
}
if (r < 0) {
return r;
}
if (expect_carries_install_info) {
if (r < 0)
return bus_log_parse_error(r);
}
if (r < 0)
return r;
/* Try to reload if enabeld */
if (!arg_no_reload)
else
r = 0;
}
if (carries_install_info == 0)
log_warning("The unit files have no [Install] section. They are not meant to be enabled\n"
"using systemctl.\n"
"Possible reasons for having this kind of units are:\n"
"1) A unit may be statically enabled by being symlinked from another unit's\n"
" .wants/ or .requires/ directory.\n"
"2) A unit's purpose may be to act as a helper for some other unit which has\n"
" a requirement dependency on it.\n"
"3) A unit may be started when needed via activation (socket, path, timer,\n"
" D-Bus, udev, scripted systemctl call, ...).\n");
return r;
}
bool enabled;
char **name;
int r;
if (r < 0)
return r;
if (r < 0)
return r;
enabled = r > 0;
if (state < 0) {
return state;
}
if (state == UNIT_FILE_ENABLED ||
enabled = true;
if (!arg_quiet)
}
} else {
const char *s;
r = sd_bus_call_method(
bus,
"org.freedesktop.systemd1",
"/org/freedesktop/systemd1",
"org.freedesktop.systemd1.Manager",
"GetUnitFileState",
&error,
&reply,
"s", *name);
if (r < 0) {
return r;
}
if (r < 0)
return bus_log_parse_error(r);
if (streq(s, "enabled") ||
streq(s, "enabled-runtime") ||
streq(s, "static"))
enabled = true;
if (!arg_quiet)
puts(s);
}
}
return !enabled;
}
static int systemctl_help(void) {
printf("%s [OPTIONS...] {COMMAND} ...\n\n"
"Query or send control commands to the systemd manager.\n\n"
" -h --help Show this help\n"
" --version Show package version\n"
" --system Connect to system manager\n"
" --user Connect to user service manager\n"
" -H --host=[USER@]HOST\n"
" Operate on remote host\n"
" -M --machine=CONTAINER\n"
" Operate on local container\n"
" -t --type=TYPE List only units of a particular type\n"
" --state=STATE List only units with particular LOAD or SUB or ACTIVE state\n"
" -p --property=NAME Show only properties by this name\n"
" -a --all Show all loaded units/properties, including dead/empty\n"
" ones. To list all units installed on the system, use\n"
" the 'list-unit-files' command instead.\n"
" -l --full Don't ellipsize unit names on output\n"
" --reverse Show reverse dependencies with 'list-dependencies'\n"
" --job-mode=MODE Specify how to deal with already queued jobs, when\n"
" queueing a new job\n"
" --show-types When showing sockets, explicitly show their type\n"
" -i --ignore-inhibitors\n"
" When shutting down or sleeping, ignore inhibitors\n"
" --kill-who=WHO Who to send signal to\n"
" -s --signal=SIGNAL Which signal to send\n"
" -q --quiet Suppress output\n"
" --no-block Do not wait until operation finished\n"
" configuration\n"
" --no-legend Do not print a legend (column headers and hints)\n"
" --no-pager Do not pipe output into a pager\n"
" --no-ask-password\n"
" Do not ask for system passwords\n"
" --runtime Enable unit files only temporarily until next reboot\n"
" -f --force When enabling unit files, override existing symlinks\n"
" When shutting down, execute action immediately\n"
" --root=PATH Enable unit files in the specified root directory\n"
" -n --lines=INTEGER Number of journal entries to show\n"
" -o --output=STRING Change journal output mode (short, short-monotonic,\n"
" verbose, export, json, json-pretty, json-sse, cat)\n"
" --plain Print unit dependencies as a list instead of a tree\n\n"
"Unit Commands:\n"
" list-units [PATTERN...] List loaded units\n"
" list-sockets [PATTERN...] List loaded sockets ordered by address\n"
" list-timers [PATTERN...] List loaded timers ordered by next elapse\n"
" start NAME... Start (activate) one or more units\n"
" stop NAME... Stop (deactivate) one or more units\n"
" reload NAME... Reload one or more units\n"
" restart NAME... Start or restart one or more units\n"
" try-restart NAME... Restart one or more units if active\n"
" reload-or-restart NAME... Reload one or more units if possible,\n"
" otherwise start or restart\n"
" reload-or-try-restart NAME... Reload one or more units if possible,\n"
" otherwise restart if active\n"
" isolate NAME Start one unit and stop all others\n"
" kill NAME... Send signal to processes of a unit\n"
" is-active NAME... Check whether units are active\n"
" is-failed NAME... Check whether units are failed\n"
" status [NAME...|PID...] Show runtime status of one or more units\n"
" show [NAME...|JOB...] Show properties of one or more\n"
" cat NAME... Show files and drop-ins of one or more units\n"
" set-property NAME ASSIGNMENT... Sets one or more properties of a unit\n"
" help NAME...|PID... Show manual for one or more units\n"
" reset-failed [NAME...] Reset failed state for all, one, or more\n"
" units\n"
" list-dependencies [NAME] Recursively show units which are required\n"
" or wanted by this unit or by which this\n"
" unit is required or wanted\n\n"
"Unit File Commands:\n"
" list-unit-files [PATTERN...] List installed unit files\n"
" enable NAME... Enable one or more unit files\n"
" disable NAME... Disable one or more unit files\n"
" reenable NAME... Reenable one or more unit files\n"
" based on preset configuration\n"
" is-enabled NAME... Check whether unit files are enabled\n\n"
" mask NAME... Mask one or more units\n"
" unmask NAME... Unmask one or more units\n"
" link PATH... Link one or more units files into\n"
" the search path\n"
" get-default Get the name of the default target\n"
" set-default NAME Set the default target\n\n"
"Job Commands:\n"
" list-jobs [PATTERN...] List jobs\n"
" cancel [JOB...] Cancel all, one, or more jobs\n\n"
"Snapshot Commands:\n"
" snapshot [NAME] Create a snapshot\n"
" delete NAME... Remove one or more snapshots\n\n"
"Environment Commands:\n"
" show-environment Dump environment\n"
" set-environment NAME=VALUE... Set one or more environment variables\n"
" unset-environment NAME... Unset one or more environment variables\n\n"
"Manager Lifecycle Commands:\n"
" daemon-reload Reload systemd manager configuration\n"
" daemon-reexec Reexecute systemd manager\n\n"
"System Commands:\n"
" default Enter system default mode\n"
" rescue Enter system rescue mode\n"
" emergency Enter system emergency mode\n"
" halt Shut down and halt the system\n"
" poweroff Shut down and power-off the system\n"
" reboot [ARG] Shut down and reboot the system\n"
" kexec Shut down and reboot the system with kexec\n"
" exit Request user instance exit\n"
" switch-root ROOT [INIT] Change to a different root file system\n"
" suspend Suspend the system\n"
" hibernate Hibernate the system\n"
" hybrid-sleep Hibernate and suspend the system\n",
return 0;
}
static int halt_help(void) {
printf("%s [OPTIONS...]%s\n\n"
"%s the system.\n\n"
" --help Show this help\n"
" --halt Halt the machine\n"
" -p --poweroff Switch off the machine\n"
" --reboot Reboot the machine\n"
" -d --no-wtmp Don't write wtmp record\n"
"Halt");
return 0;
}
static int shutdown_help(void) {
printf("%s [OPTIONS...] [TIME] [WALL...]\n\n"
"Shut down the system.\n\n"
" --help Show this help\n"
" -H --halt Halt the machine\n"
" -P --poweroff Power-off the machine\n"
" -r --reboot Reboot the machine\n"
" -h Equivalent to --poweroff, overridden by --halt\n"
" -c Cancel a pending shutdown\n",
return 0;
}
static int telinit_help(void) {
printf("%s [OPTIONS...] {COMMAND}\n\n"
"Send control commands to the init daemon.\n\n"
" --help Show this help\n"
"Commands:\n"
" 0 Power-off the machine\n"
" 6 Reboot the machine\n"
" 2, 3, 4, 5 Start runlevelX.target unit\n"
" 1, s, S Enter rescue mode\n"
" q, Q Reload init daemon configuration\n"
" u, U Reexecute init daemon\n",
return 0;
}
static int runlevel_help(void) {
printf("%s [OPTIONS...]\n\n"
"Prints the previous and current runlevel of the init system.\n\n"
" --help Show this help\n",
return 0;
}
static int help_types(void) {
int i;
const char *t;
puts("Available unit types:");
for (i = 0; i < _UNIT_TYPE_MAX; i++) {
t = unit_type_to_string(i);
if (t)
puts(t);
}
return 0;
}
enum {
ARG_FAIL = 0x100,
};
{}
};
int c;
switch (c) {
case 'h':
return systemctl_help();
case ARG_VERSION:
return 0;
case 't': {
_cleanup_free_ char *type;
if (!type)
return -ENOMEM;
help_types();
return 0;
}
if (unit_type_from_string(type) >= 0) {
return log_oom();
continue;
}
/* It's much nicer to use --state= for
* load states, but let's support this
* in --types= too for compatibility
* with old versions */
if (unit_load_state_from_string(optarg) >= 0) {
return log_oom();
continue;
}
log_info("Use -t help to see a list of allowed values.");
return -EINVAL;
}
break;
}
case 'p': {
/* Make sure that if the empty property list
was specified, we won't show any properties. */
if (!arg_properties)
return log_oom();
} else {
char *prop;
if (!prop)
return log_oom();
return log_oom();
}
}
}
/* If the user asked for a particular
* property, show it to him, even if it is
* empty. */
arg_all = true;
break;
}
case 'a':
arg_all = true;
break;
case ARG_REVERSE:
break;
case ARG_AFTER:
break;
case ARG_BEFORE:
break;
case ARG_SHOW_TYPES:
arg_show_types = true;
break;
case ARG_JOB_MODE:
break;
case ARG_FAIL:
arg_job_mode = "fail";
break;
case ARG_IRREVERSIBLE:
arg_job_mode = "replace-irreversibly";
break;
case ARG_IGNORE_DEPENDENCIES:
arg_job_mode = "ignore-dependencies";
break;
case ARG_USER:
break;
case ARG_SYSTEM:
break;
case ARG_GLOBAL:
break;
case ARG_NO_BLOCK:
arg_no_block = true;
break;
case ARG_NO_LEGEND:
arg_no_legend = true;
break;
case ARG_NO_PAGER:
arg_no_pager = true;
break;
case ARG_NO_WALL:
arg_no_wall = true;
break;
case ARG_ROOT:
break;
case 'l':
arg_full = true;
break;
case ARG_FAILED:
return log_oom();
break;
case 'q':
arg_quiet = true;
break;
case ARG_FORCE:
arg_force ++;
break;
case 'f':
arg_force ++;
break;
case ARG_NO_RELOAD:
arg_no_reload = true;
break;
case ARG_KILL_WHO:
break;
case 's':
return -EINVAL;
}
break;
case ARG_NO_ASK_PASSWORD:
arg_ask_password = false;
break;
case 'H':
break;
case 'M':
break;
case ARG_RUNTIME:
arg_runtime = true;
break;
case 'n':
return -EINVAL;
}
break;
case 'o':
if (arg_output < 0) {
return -EINVAL;
}
break;
case 'i':
arg_ignore_inhibitors = true;
break;
case ARG_PLAIN:
arg_plain = true;
break;
case ARG_STATE: {
char *s;
if (!s)
return log_oom();
if (strv_push(&arg_states, s) < 0) {
free(s);
return log_oom();
}
}
break;
}
case '?':
return -EINVAL;
default:
assert_not_reached("Unhandled option");
}
}
log_error("Cannot access user instance remotely.");
return -EINVAL;
}
return 1;
}
enum {
ARG_HELP = 0x100,
};
{}
};
int c, r, runlevel;
arg_force = 2;
switch (c) {
case ARG_HELP:
return halt_help();
case ARG_HALT:
break;
case 'p':
if (arg_action != ACTION_REBOOT)
break;
case ARG_REBOOT:
break;
case 'f':
arg_force = 2;
break;
case 'w':
arg_dry = true;
break;
case 'd':
arg_no_wtmp = true;
break;
case ARG_NO_WALL:
arg_no_wall = true;
break;
case 'i':
case 'h':
case 'n':
/* Compatibility nops */
break;
case '?':
return -EINVAL;
default:
assert_not_reached("Unhandled option");
}
}
if (r < 0) {
log_error("Failed to write reboot param to "
return r;
}
log_error("Too many arguments.");
return -EINVAL;
}
return 1;
}
assert(t);
if (streq(t, "now"))
*_u = 0;
else if (!strchr(t, ':')) {
uint64_t u;
if (safe_atou64(t, &u) < 0)
return -EINVAL;
} else {
char *e = NULL;
time_t s;
usec_t n;
errno = 0;
return -EINVAL;
return -EINVAL;
n = now(CLOCK_REALTIME);
s = (time_t) (n / USEC_PER_SEC);
while (*_u <= n)
*_u += USEC_PER_DAY;
}
return 0;
}
enum {
ARG_HELP = 0x100,
};
{}
};
int c, r;
switch (c) {
case ARG_HELP:
return shutdown_help();
case 'H':
break;
case 'P':
break;
case 'r':
if (kexec_loaded())
else
break;
case 'K':
break;
case 'h':
if (arg_action != ACTION_HALT)
break;
case 'k':
arg_dry = true;
break;
case ARG_NO_WALL:
arg_no_wall = true;
break;
case 't':
case 'a':
/* Compatibility nops */
break;
case 'c':
break;
case '?':
return -EINVAL;
default:
assert_not_reached("Unhandled option");
}
}
if (r < 0) {
return r;
}
} else
/* No time argument for shutdown cancel */
/* We skip the time argument */
return 1;
}
enum {
ARG_HELP = 0x100,
};
{}
};
static const struct {
char from;
} table[] = {
{ '0', ACTION_POWEROFF },
{ '6', ACTION_REBOOT },
{ '1', ACTION_RESCUE },
{ '2', ACTION_RUNLEVEL2 },
{ '3', ACTION_RUNLEVEL3 },
{ '4', ACTION_RUNLEVEL4 },
{ '5', ACTION_RUNLEVEL5 },
{ 's', ACTION_RESCUE },
{ 'S', ACTION_RESCUE },
{ 'q', ACTION_RELOAD },
{ 'Q', ACTION_RELOAD },
{ 'u', ACTION_REEXEC },
{ 'U', ACTION_REEXEC }
};
unsigned i;
int c;
switch (c) {
case ARG_HELP:
return telinit_help();
case ARG_NO_WALL:
arg_no_wall = true;
break;
case '?':
return -EINVAL;
default:
assert_not_reached("Unhandled option");
}
}
telinit_help();
return -EINVAL;
}
log_error("Too many arguments.");
return -EINVAL;
}
log_error("Expected single character argument.");
return -EINVAL;
}
for (i = 0; i < ELEMENTSOF(table); i++)
break;
if (i >= ELEMENTSOF(table)) {
return -EINVAL;
}
optind ++;
return 1;
}
enum {
ARG_HELP = 0x100,
};
{}
};
int c;
switch (c) {
case ARG_HELP:
return runlevel_help();
return 0;
case '?':
return -EINVAL;
default:
assert_not_reached("Unhandled option");
}
}
log_error("Too many arguments.");
return -EINVAL;
}
return 1;
}
if (kexec_loaded())
else
if (sd_booted() > 0) {
} else {
/* Hmm, so some other init system is
* running, we need to forward this
* request to it. For now we simply
* guess that it is Upstart. */
log_error("Couldn't find an alternative telinit implementation to spawn.");
return -EIO;
}
}
}
}
_pure_ static int action_to_runlevel(void) {
static const char table[_ACTION_MAX] = {
[ACTION_HALT] = '0',
[ACTION_POWEROFF] = '0',
[ACTION_REBOOT] = '6',
[ACTION_RUNLEVEL2] = '2',
[ACTION_RUNLEVEL3] = '3',
[ACTION_RUNLEVEL4] = '4',
[ACTION_RUNLEVEL5] = '5',
[ACTION_RESCUE] = '1'
};
return table[arg_action];
}
static int talk_initctl(void) {
struct init_request request = {
.magic = INIT_MAGIC,
.sleeptime = 0,
};
char rl;
int r;
rl = action_to_runlevel();
if (!rl)
return 0;
if (fd < 0) {
return 0;
return -errno;
}
errno = 0;
if (r) {
}
return 1;
}
static const struct {
const char* verb;
const enum {
MORE,
LESS,
} argc_cmp;
const int argc;
const enum {
NOBUS = 1,
} bus;
} verbs[] = {
{}
int left;
/* Special rule: no arguments (left == 0) means "list-units" */
if (left > 0) {
log_error("This command expects one or more "
"unit names. Did you mean --help?");
return -EINVAL;
}
goto found;
return -EINVAL;
}
case EQUAL:
log_error("Invalid number of arguments.");
return -EINVAL;
}
break;
case MORE:
log_error("Too few arguments.");
return -EINVAL;
}
break;
case LESS:
log_error("Too many arguments.");
return -EINVAL;
}
break;
default:
assert_not_reached("Unknown comparison operator.");
}
/* Require a bus connection for all operations but
return -EIO;
}
} else {
if (running_in_chroot() > 0) {
log_info("Running in chroot, ignoring request.");
return 0;
}
return -EIO;
}
}
}
struct sd_shutdown_command c = {
.usec = t,
};
union sockaddr_union sockaddr = {
};
.iov_base = (char*) &c,
}};
.msg_iovlen = 1,
};
_cleanup_close_ int fd;
if (fd < 0)
return -errno;
msghdr.msg_iovlen++;
}
return -errno;
return 0;
}
if (bus) {
/* First, try systemd via D-Bus. */
return 0;
}
/* Nothing else worked, so let's try signals */
log_error("kill() failed: %m");
return -errno;
}
return 0;
}
if (bus) {
/* First, try systemd via D-Bus. */
goto done;
}
/* Nothing else worked, so let's try
if (talk_initctl() > 0)
goto done;
log_error("Failed to talk to init daemon.");
return -EIO;
done:
return 0;
}
/* Make sure C-A-D is handled by the kernel from this
* point on... */
switch (a) {
case ACTION_HALT:
log_info("Halting.");
return -errno;
case ACTION_POWEROFF:
log_info("Powering off.");
return -errno;
case ACTION_REBOOT: {
}
log_info("Rebooting.");
return -errno;
}
default:
assert_not_reached("Unknown action.");
}
}
int r;
if (r < 0)
return r;
if (geteuid() != 0) {
/* Try logind if we are a normal user and no special
* mode applies. Maybe PolicyKit allows us to shutdown
* the machine. */
if (arg_when <= 0 &&
!arg_dry &&
arg_force <= 0 &&
(arg_action == ACTION_POWEROFF ||
arg_action == ACTION_REBOOT)) {
if (r >= 0)
return r;
}
log_error("Must be root.");
return -EPERM;
}
if (arg_when > 0) {
_cleanup_free_ char *m;
if (!m)
return log_oom();
r = send_shutdownd(arg_when,
'r',
m);
if (r < 0)
else {
char date[FORMAT_TIMESTAMP_MAX];
log_info("Shutdown scheduled for %s, use 'shutdown -c' to cancel.",
return 0;
}
}
return start_with_fallback(bus);
if (!arg_no_wtmp) {
if (sd_booted() > 0)
log_debug("Not writing utmp record, assuming that systemd-update-utmp is used.");
else {
r = utmp_put_shutdown();
if (r < 0)
}
}
if (arg_dry)
return 0;
r = halt_now(arg_action);
return r;
}
static int runlevel_main(void) {
if (r < 0) {
puts("unknown");
return r;
}
printf("%c %c\n",
return 0;
}
int r;
log_open();
/* Explicitly not on_tty() to avoid setting cached value.
* This becomes relevant for piping output which might be
* ellipsized. */
if (r <= 0)
goto finish;
* let's shortcut this */
if (arg_action == ACTION_RUNLEVEL) {
r = runlevel_main();
goto finish;
}
log_info("Running in chroot, ignoring request.");
r = 0;
goto finish;
}
if (!avoid_bus())
/* systemctl_main() will print an error message for the bus
* connection, but only if it needs to */
switch (arg_action) {
case ACTION_SYSTEMCTL:
break;
case ACTION_HALT:
case ACTION_POWEROFF:
case ACTION_REBOOT:
case ACTION_KEXEC:
break;
case ACTION_RUNLEVEL2:
case ACTION_RUNLEVEL3:
case ACTION_RUNLEVEL4:
case ACTION_RUNLEVEL5:
case ACTION_RESCUE:
case ACTION_EMERGENCY:
case ACTION_DEFAULT:
r = start_with_fallback(bus);
break;
case ACTION_RELOAD:
case ACTION_REEXEC:
r = reload_with_fallback(bus);
break;
case ACTION_CANCEL_SHUTDOWN: {
_cleanup_free_ char *m = NULL;
if (arg_wall) {
if (!m) {
r = log_oom();
goto finish;
}
}
if (r < 0)
break;
}
case ACTION_RUNLEVEL:
case _ACTION_INVALID:
default:
assert_not_reached("Unknown action");
}
pager_close();
return r < 0 ? EXIT_FAILURE : r;
}