systemctl.c revision 422fa6500a44a222b655cd218a658f8a4f34a768
/*-*- Mode: C; c-basic-offset: 8; indent-tabs-mode: nil -*-*/
/***
This file is part of systemd.
Copyright 2010 Lennart Poettering
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 <systemd/sd-daemon.h>
#include <systemd/sd-shutdown.h>
#include <systemd/sd-login.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 "dbus-common.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 "path-util.h"
#include "socket-util.h"
#include "fileio.h"
static char **arg_load_states = NULL;
static char **arg_properties = NULL;
static bool arg_all = false;
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_failed = false;
static bool arg_runtime = false;
static const char *arg_kill_who = NULL;
static int arg_signal = SIGTERM;
static enum action {
static enum transport {
static unsigned arg_lines = 10;
static bool arg_plain = false;
static bool private_bus = false;
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;
}
#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;
}
#endif
static const char *ansi_highlight(bool b) {
if (!on_tty())
return "";
return b ? ANSI_HIGHLIGHT_ON : ANSI_HIGHLIGHT_OFF;
}
static const char *ansi_highlight_red(bool b) {
if (!on_tty())
return "";
return b ? ANSI_HIGHLIGHT_RED_ON : ANSI_HIGHLIGHT_OFF;
}
static const char *ansi_highlight_green(bool b) {
if (!on_tty())
return "";
return b ? ANSI_HIGHLIGHT_GREEN_ON : ANSI_HIGHLIGHT_OFF;
}
if (!dbus_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 struct unit_info *u = a, *v = b;
int r;
if (r != 0)
return r;
}
}
static bool output_show_unit(const struct unit_info *u) {
const char *dot;
if (arg_failed)
}
const struct unit_info *u;
int job_count = 0;
desc_len = 0;
for (u = unit_infos; u < unit_infos + c; u++) {
if (!output_show_unit(u))
continue;
if (u->job_id != 0) {
job_count++;
}
}
if (!arg_full) {
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 (!output_show_unit(u))
continue;
if (!n_shown && !arg_no_legend) {
if (job_count)
if (!arg_full && arg_no_pager)
else
}
n_shown++;
} else
} else
printf("%s%-*s%s %s%-6s%s %s%-*s %-*s%s %-*s",
if (!arg_full && arg_no_pager)
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(true);
off = ansi_highlight(false);
} else {
on = ansi_highlight_red(true);
off = ansi_highlight_red(false);
}
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",
}
}
struct unit_info **unit_infos, unsigned *c) {
int r;
assert(c);
bus,
"org.freedesktop.systemd1",
"/org/freedesktop/systemd1",
"org.freedesktop.systemd1.Manager",
"ListUnits",
NULL,
if (r < 0)
return r;
log_error("Failed to parse reply.");
return -EIO;
}
return log_oom();
(*c)++;
}
return 0;
}
unsigned c = 0;
int r;
if (r < 0)
return r;
return 0;
}
char*** triggered)
{
const char *interface = "org.freedesktop.systemd1.Unit",
*triggers_property = "Triggers";
int r;
"org.freedesktop.systemd1",
"org.freedesktop.DBus.Properties",
"Get",
&reply,
NULL,
if (r < 0)
return r;
log_error("Failed to parse reply.");
return -EBADMSG;
}
const char *unit;
log_error("Failed to parse reply.");
return -EBADMSG;
}
if (r < 0)
return r;
}
return 0;
}
char*** listen, unsigned *c)
{
const char *interface = "org.freedesktop.systemd1.Socket",
*listen_property = "Listen";
int r;
"org.freedesktop.systemd1",
"org.freedesktop.DBus.Properties",
"Get",
&reply,
NULL,
if (r < 0)
return r;
log_error("Failed to parse reply.");
return -EBADMSG;
}
log_error("Failed to parse reply.");
return -EBADMSG;
}
if (r < 0)
return r;
if (r < 0)
return r;
(*c) ++;
}
}
return 0;
}
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;
};
if (o == 0)
return o;
}
struct socket_info *s;
char **a;
unsigned tmp = 0;
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(true);
off = ansi_highlight(false);
if (!arg_no_legend)
printf("\n");
} else {
on = ansi_highlight_red(true);
off = ansi_highlight_red(false);
}
if (!arg_no_legend) {
if (!arg_all)
printf("Pass --all to see loaded but inactive sockets, too.\n");
}
return 0;
}
const struct unit_info *u;
struct socket_info *s;
int r;
if (r < 0)
return r;
const char *dot;
unsigned c = 0, i;
if (!output_show_unit(u))
continue;
continue;
if (r < 0)
goto cleanup;
if (r < 0)
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 0;
}
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;
}
}
static bool output_show_unit_file(const UnitFileList *u) {
const char *dot;
}
const UnitFileList *u;
if (!output_show_unit_file(u))
continue;
}
if (!arg_full) {
unsigned basic_cols;
if (basic_cols < (unsigned) columns())
} else
if (!arg_no_legend)
_cleanup_free_ char *e = NULL;
const char *id;
if (!output_show_unit_file(u))
continue;
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(true);
off = ansi_highlight_red(false);
} else if (u->state == UNIT_FILE_ENABLED) {
on = ansi_highlight_green(true);
off = ansi_highlight_green(false);
} else
printf("%-*s %s%-*s%s\n",
}
if (!arg_no_legend)
}
unsigned c = 0, n_units = 0;
int r;
if (avoid_bus()) {
Hashmap *h;
UnitFileList *u;
Iterator i;
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) {
free(u);
}
hashmap_free(h);
} else {
bus,
"org.freedesktop.systemd1",
"/org/freedesktop/systemd1",
"org.freedesktop.systemd1.Manager",
"ListUnitFiles",
&reply,
NULL,
if (r < 0)
return r;
log_error("Failed to parse reply.");
return -EIO;
}
UnitFileList *u;
const char *state;
if (c >= n_units) {
UnitFileList *w;
if (!w)
return log_oom();
units = w;
}
u = units + c;
log_error("Failed to parse reply.");
return -EIO;
}
c++;
}
}
if (c > 0) {
}
return 0;
}
int i;
_cleanup_free_ char *n = NULL;
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 int list_dependencies_get_dependencies(DBusConnection *bus, const char *name, char ***deps) {
static const char *dependencies[] = {
[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",
};
_cleanup_free_ char *path;
const char *interface = "org.freedesktop.systemd1.Unit";
int r = 0;
r = -EINVAL;
goto finish;
}
bus,
"org.freedesktop.systemd1",
path,
"org.freedesktop.DBus.Properties",
"GetAll",
&reply,
NULL,
if (r < 0)
goto finish;
log_error("Failed to parse reply.");
r = -EIO;
goto finish;
}
const char *prop;
log_error("Failed to parse reply.");
r = -EIO;
goto finish;
}
log_error("Failed to parse reply.");
r = -EIO;
goto finish;
}
continue;
const char *s;
dbus_message_iter_get_basic(&sub4, &s);
r = strv_extend(&ret, s);
if (r < 0) {
log_oom();
goto finish;
}
}
}
}
}
if (r < 0)
else
return r;
}
return 1;
return -1;
return strcasecmp(*a, *b);
}
static int list_dependencies_one(DBusConnection *bus, 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) {
if (strv_contains(u, *c)) {
if (!arg_plain) {
if (r < 0)
return r;
}
continue;
}
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);
}
int r;
if (r < 0) {
goto finish;
}
r = 0;
} else {
bus,
"org.freedesktop.systemd1",
"/org/freedesktop/systemd1",
"org.freedesktop.systemd1.Manager",
"GetDefaultTarget",
&reply,
NULL,
if (r < 0) {
goto finish;
}
return -EIO;
}
}
if (path)
return r;
}
struct job_info {
};
size_t i;
struct job_info *j;
bool shorten = false;
if (n == 0) {
on = ansi_highlight_green(true);
off = ansi_highlight_green(false);
return;
}
{
/* JOB UNIT TYPE STATE */
for (i = 0, j = jobs; i < n; i++, j++) {
}
shorten = true;
}
if (on_tty())
printf("%*s %-*s %-*s %-*s\n",
l0, "JOB",
l1, "UNIT",
l2, "TYPE",
l3, "STATE");
for (i = 0, j = jobs; i < n; i++, j++) {
_cleanup_free_ char *e = NULL;
on = ansi_highlight(true);
off = ansi_highlight(false);
} else
printf("%*u %s%-*s%s %-*s %s%-*s%s\n",
}
}
on = ansi_highlight(true);
off = ansi_highlight(false);
if (on_tty())
}
int r;
bus,
"org.freedesktop.systemd1",
"/org/freedesktop/systemd1",
"org.freedesktop.systemd1.Manager",
"ListJobs",
&reply,
NULL,
if (r < 0)
return r;
log_error("Failed to parse reply.");
return -EIO;
}
log_error("Failed to parse reply.");
return -EIO;
}
log_error("Failed to parse reply.");
r = -EIO;
goto finish;
}
r = log_oom();
goto finish;
}
r = log_oom();
goto finish;
}
}
while (used--) {
}
return 0;
}
char **name;
_cleanup_free_ char *n = NULL;
int r;
n = unit_name_mangle(*name);
if (!n)
return log_oom();
bus,
"org.freedesktop.systemd1",
"/org/freedesktop/systemd1",
"org.freedesktop.systemd1.Manager",
"LoadUnit",
NULL,
NULL,
DBUS_TYPE_STRING, &n,
if (r < 0)
return r;
}
return 0;
}
char **name;
int r;
if (r < 0) {
return r;
}
bus,
"org.freedesktop.systemd1",
"/org/freedesktop/systemd1",
"org.freedesktop.systemd1.Manager",
"CancelJob",
NULL,
NULL,
if (r < 0)
return r;
}
return 0;
}
dbus_bool_t b = FALSE;
const char
*interface = "org.freedesktop.systemd1.Unit",
*property = "NeedDaemonReload",
*path;
_cleanup_free_ char *n = NULL;
int r;
/* We ignore all errors here, since this is used to show a warning only */
n = unit_name_mangle(unit);
if (!n)
return log_oom();
bus,
"org.freedesktop.systemd1",
"/org/freedesktop/systemd1",
"org.freedesktop.systemd1.Manager",
"GetUnit",
&reply,
NULL,
DBUS_TYPE_STRING, &n,
if (r < 0)
return r;
return -EIO;
bus,
"org.freedesktop.systemd1",
path,
"org.freedesktop.DBus.Properties",
"Get",
&reply,
NULL,
if (r < 0)
return r;
return -EIO;
return -EIO;
dbus_message_iter_get_basic(&sub, &b);
return b;
}
typedef struct WaitData {
char *name;
char *result;
} WaitData;
static DBusHandlerResult wait_filter(DBusConnection *connection, DBusMessage *message, void *data) {
assert(d);
log_debug("Got D-Bus request: %s.%s() on %s",
log_error("Warning! D-Bus connection terminated.");
}
#ifndef LEGACY
/* Compatibility with older systemd versions <
* 183 during upgrades. This should be dropped
* one day. */
if (*result)
}
#endif
}
}
if (private_bus)
return 0;
"type='signal',"
"sender='org.freedesktop.systemd1',"
"interface='org.freedesktop.systemd1.Manager',"
"member='JobRemoved',"
"path='/org/freedesktop/systemd1'",
&error);
if (dbus_error_is_set(&error)) {
return -EIO;
}
/* This is slightly dirty, since we don't undo the match registrations. */
return 0;
}
int r = 0;
assert(s);
return log_oom();
while (!set_isempty(s)) {
log_error("Disconnected from bus.");
return -ECONNREFUSED;
}
if (!d.result)
goto free_name;
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;
}
return r;
}
_cleanup_free_ char *n = NULL;
const char
*interface = "org.freedesktop.systemd1.Unit",
*property = "ActiveState";
int r;
n = unit_name_mangle(name);
if (!n)
return log_oom();
bus,
"org.freedesktop.systemd1",
"/org/freedesktop/systemd1",
"org.freedesktop.systemd1.Manager",
"GetUnit",
&reply,
&error,
DBUS_TYPE_STRING, &n,
if (r < 0) {
if (!quiet)
puts("unknown");
return 0;
}
log_error("Failed to parse reply.");
return -EIO;
}
bus,
"org.freedesktop.systemd1",
path,
"org.freedesktop.DBus.Properties",
"Get",
&reply,
NULL,
if (r < 0) {
if (!quiet)
puts("unknown");
return 0;
}
log_error("Failed to parse reply.");
return r;
}
log_error("Failed to parse reply.");
return r;
}
if (!quiet)
}
static void check_triggering_units(
const char *unit_name) {
const char *interface = "org.freedesktop.systemd1.Unit",
*load_state_property = "LoadState",
*triggered_by_property = "TriggeredBy",
*state;
bool print_warning_label = true;
int r;
n = unit_name_mangle(unit_name);
if (!n) {
log_oom();
return;
}
if (!unit_path) {
log_oom();
return;
}
bus,
"org.freedesktop.systemd1",
"org.freedesktop.DBus.Properties",
"Get",
&reply,
NULL,
if (r < 0)
return;
log_error("Failed to parse reply.");
return;
}
log_error("Failed to parse reply.");
return;
}
return;
bus,
"org.freedesktop.systemd1",
"org.freedesktop.DBus.Properties",
"Get",
&reply,
NULL,
if (r < 0)
return;
log_error("Failed to parse reply.");
return;
}
const char * const check_states[] = {
"active",
"reloading",
};
const char *service_trigger;
log_error("Failed to parse reply.");
return;
}
if (r < 0)
return;
if (r > 0) {
if (print_warning_label) {
print_warning_label = false;
}
}
}
}
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();
bus,
"org.freedesktop.systemd1",
"/org/freedesktop/systemd1",
"org.freedesktop.systemd1.Manager",
&reply,
DBUS_TYPE_STRING, &n,
if (r) {
/* There's always a fallback possible for
* legacy actions. */
r = -EADDRNOTAVAIL;
else
return r;
}
return -EIO;
}
if (need_daemon_reload(bus, n))
log_warning("Warning: Unit file of %s changed on disk, 'systemctl %sdaemon-reload' recommended.",
if (s) {
char *p;
if (!p)
return log_oom();
r = set_consume(s, p);
if (r < 0) {
log_error("Failed to add path to set.");
return r;
}
}
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;
}
int r, ret = 0;
char **name;
if (arg_action == ACTION_SYSTEMCTL) {
method =
"StartUnit";
} else {
method = "StartUnit";
}
if (!arg_no_block) {
if (ret < 0) {
return ret;
}
if (!s)
return log_oom();
}
if (one_name) {
if (ret < 0)
} else {
if (r < 0) {
}
}
}
if (!arg_no_block) {
r = wait_for_jobs(bus, s);
if (r < 0)
return r;
/* When stopping units, warn if they can still be triggered by
* another active unit (socket, path, timer) */
if (one_name)
else
}
}
return ret;
}
/* Ask systemd-logind, which might grant access to unprivileged users
* through PolicyKit */
#ifdef HAVE_LOGIND
const char *method;
dbus_bool_t interactive = true;
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;
}
return bus_method_call_with_reply(
bus,
"org.freedesktop.login1",
"/org/freedesktop/login1",
"org.freedesktop.login1.Manager",
NULL,
NULL,
#else
return -ENOSYS;
#endif
}
#ifdef HAVE_LOGIND
int r;
unsigned c = 0;
char **s;
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;
bus,
"org.freedesktop.login1",
"/org/freedesktop/login1",
"org.freedesktop.login1.Manager",
"ListInhibitors",
&reply,
NULL,
if (r < 0)
/* If logind is not around, then there are no inhibitors... */
return 0;
log_error("Failed to parse reply.");
return -EIO;
}
log_error("Failed to parse reply.");
return -EIO;
}
log_error("Failed to parse reply.");
return -EIO;
}
goto next;
if (!sv)
return log_oom();
if (!strv_contains(sv,
a == ACTION_HALT ||
a == ACTION_POWEROFF ||
a == ACTION_REBOOT ||
goto next;
log_warning("Operation inhibited by \"%s\" (PID %lu \"%s\", user %s), reason is \"%s\".",
c++;
next:
}
/* 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))
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;
}
const char * const check_states[] = {
"active",
"reloading",
};
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;
}
const char * const check_states[] = {
"failed",
};
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();
bus,
"org.freedesktop.systemd1",
"/org/freedesktop/systemd1",
"org.freedesktop.systemd1.Manager",
"KillUnit",
NULL,
NULL,
DBUS_TYPE_STRING, &n,
if (r < 0)
return r;
}
return 0;
}
_cleanup_free_ char *n = NULL;
char **argument;
int r;
method =
: "UnsetUnitControlGroupAttribute";
if (!n)
return log_oom();
bus,
"org.freedesktop.systemd1",
"/org/freedesktop/systemd1",
"org.freedesktop.systemd1.Manager",
NULL,
NULL,
DBUS_TYPE_STRING, &n,
if (r < 0)
return r;
}
return 0;
}
_cleanup_free_ char *n = NULL;
const char *runtime;
int r;
if (!n)
return log_oom();
"org.freedesktop.systemd1",
"/org/freedesktop/systemd1",
"org.freedesktop.systemd1.Manager",
"SetUnitControlGroupAttribute");
if (!m)
return log_oom();
return log_oom();
if (r < 0)
return log_oom();
return log_oom();
if (!reply) {
return -EIO;
}
return 0;
}
_cleanup_free_ char *n = NULL;
char **argument;
int r;
if (!n)
return log_oom();
char **a;
bus,
"org.freedesktop.systemd1",
"/org/freedesktop/systemd1",
"org.freedesktop.systemd1.Manager",
"GetUnitControlGroupAttribute",
&reply,
NULL,
DBUS_TYPE_STRING, &n,
if (r < 0)
return r;
log_error("Failed to initialize iterator.");
return -EIO;
}
if (r < 0) {
log_error("Failed to parse value list.");
return r;
}
STRV_FOREACH(a, list) {
if (endswith(*a, "\n"))
else
puts(*a);
}
}
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;
unsigned n;
assert(i);
assert(i);
return -EIO;
return -EIO;
if (!i->path)
return -ENOMEM;
return -EIO;
n = 0;
n++;
}
if (!i->argv)
return -ENOMEM;
n = 0;
const char *s;
dbus_message_iter_get_basic(&sub3, &s);
if (!i->argv[n])
return -ENOMEM;
n++;
}
if (!dbus_message_iter_next(&sub2) ||
return -EIO;
return 0;
}
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 *default_control_group;
char **dropin_paths;
const char *load_error;
const char *result;
bool need_daemon_reload;
/* Service */
const char *status_text;
bool running:1;
int exit_code, exit_status;
bool condition_result;
/* 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) {
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(true);
off = ansi_highlight_red(false);
} 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)) {
char ** dropin;
bool last = false;
log_oom();
return;
}
}
}
}
on = ansi_highlight_red(true);
off = ansi_highlight_red(false);
on = ansi_highlight_green(true);
off = ansi_highlight_green(false);
} 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) {
if (s1)
else if (s2)
}
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(true);
off = ansi_highlight_red(false);
} 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->default_control_group &&
(i->main_pid > 0 || i->control_pid > 0 || cg_is_empty_by_spec(i->default_control_group, false) == 0)) {
unsigned c;
if (arg_transport != TRANSPORT_SSH) {
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",
ansi_highlight_red(true),
ansi_highlight_red(false),
}
static void show_unit_help(UnitStatusInfo *i) {
char **p;
assert(i);
if (!i->documentation) {
return;
}
STRV_FOREACH(p, i->documentation) {
if (startswith(*p, "man:")) {
size_t k;
char *e = NULL;
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);
}
}
assert(i);
switch (dbus_message_iter_get_arg_type(iter)) {
case DBUS_TYPE_STRING: {
const char *s;
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;
i->default_control_group = s;
i->status_text = s;
i->sysfs_path = s;
i->where = s;
i->what = s;
i->following = s;
i->unit_file_state = s;
i->result = s;
}
break;
}
case DBUS_TYPE_BOOLEAN: {
dbus_bool_t b;
i->accept = b;
i->need_daemon_reload = b;
i->condition_result = b;
break;
}
case DBUS_TYPE_UINT32: {
uint32_t u;
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 DBUS_TYPE_INT32: {
int32_t j;
i->exit_code = (int) j;
i->exit_status = (int) j;
break;
}
case DBUS_TYPE_UINT64: {
uint64_t u;
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 DBUS_TYPE_ARRAY: {
int r;
return -ENOMEM;
return -ENOMEM;
}
return r;
}
}
int r;
if (r < 0)
return r;
if (r < 0)
return r;
}
}
return 0;
} else if (dbus_message_iter_get_element_type(iter) == DBUS_TYPE_STRING && streq(name, "DropInPaths")) {
if (r < 0)
return r;
const char *s;
int r;
dbus_message_iter_get_basic(&sub, &s);
r = strv_extend(&i->documentation, s);
if (r < 0)
return r;
}
}
break;
}
case DBUS_TYPE_STRUCT: {
const char *n, *message;
int r;
if (r < 0)
return r;
if (r < 0)
return r;
i->load_error = message;
}
break;
}
}
return 0;
}
/* This is a low-level property printer, see
* print_status_info() for the nicer output */
return 0;
switch (dbus_message_iter_get_arg_type(iter)) {
case DBUS_TYPE_STRUCT: {
uint32_t u;
dbus_message_iter_get_basic(&sub, &u);
if (u)
else if (arg_all)
return 0;
const char *s;
dbus_message_iter_get_basic(&sub, &s);
if (arg_all || s[0])
return 0;
return 0;
}
break;
}
case DBUS_TYPE_ARRAY:
if (dbus_message_iter_get_element_type(iter) == DBUS_TYPE_STRUCT && streq(name, "EnvironmentFiles")) {
const char *path;
}
return 0;
}
return 0;
}
return 0;
const char *base;
printf("%s={ value=%s ; next_elapse=%s }\n",
base,
}
}
return 0;
} else if (dbus_message_iter_get_element_type(iter) == DBUS_TYPE_STRUCT && streq(name, "ControlGroupAttributes")) {
printf("ControlGroupAttributes={ controller=%s ; attribute=%s ; value=\"%s\" }\n",
attr,
value);
}
}
return 0;
} else if (dbus_message_iter_get_element_type(iter) == DBUS_TYPE_STRUCT && startswith(name, "Exec")) {
ExecStatusInfo info = {};
_cleanup_free_ char *t;
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,
strna(t),
}
}
return 0;
}
break;
}
return 0;
if (arg_all)
return 0;
}
static int show_one(const char *verb, DBusConnection *bus, const char *path, bool show_properties, bool *new_line) {
const char *interface = "";
int r;
UnitStatusInfo info = {};
ExecStatusInfo *p;
bus,
"org.freedesktop.systemd1",
path,
"org.freedesktop.DBus.Properties",
"GetAll",
&reply,
NULL,
if (r < 0)
return r;
log_error("Failed to parse reply.");
return -EIO;
}
if (*new_line)
printf("\n");
*new_line = true;
const char *name;
log_error("Failed to parse reply.");
return -EIO;
}
if (show_properties)
else
if (r < 0) {
log_error("Failed to parse reply.");
return -EIO;
}
}
r = 0;
if (!show_properties) {
else
}
/* According to LSB: "program not running" */
r = 3;
}
return r;
}
int r;
bus,
"org.freedesktop.systemd1",
"/org/freedesktop/systemd1",
"org.freedesktop.systemd1.Manager",
"GetUnitByPID",
&reply,
NULL,
if (r < 0)
return r;
return -EIO;
}
return r;
}
unsigned c = 0;
const struct unit_info *u;
int r;
if (r < 0)
return r;
for (u = unit_infos; u < unit_infos + c; u++) {
_cleanup_free_ char *p = NULL;
if (!output_show_unit(u))
continue;
p = unit_dbus_path_from_name(u->id);
if (!p)
return log_oom();
if (r != 0)
return r;
}
return 0;
}
int r, ret = 0;
char **name;
if (show_properties)
/* If no argument is specified inspect the manager itself */
/* Interpret as unit name */
n = unit_name_mangle(*name);
if (!n)
return log_oom();
p = unit_dbus_path_from_name(n);
if (!p)
return log_oom();
if (r != 0)
ret = r;
} else if (show_properties) {
_cleanup_free_ char *p = NULL;
/* Interpret as job id */
return log_oom();
if (r != 0)
ret = r;
} else {
/* Interpret as PID */
if (r != 0)
ret = r;
}
}
return ret;
}
int r;
const char *text;
bus,
"org.freedesktop.systemd1",
"/org/freedesktop/systemd1",
"org.freedesktop.systemd1.Manager",
"Dump",
&reply,
NULL,
if (r < 0)
return r;
return -EIO;
}
return 0;
}
int r;
const char
*interface = "org.freedesktop.systemd1.Unit",
*property = "Id";
_cleanup_free_ char *n = NULL;
else
n = strdup("");
if (!n)
return log_oom();
bus,
"org.freedesktop.systemd1",
"/org/freedesktop/systemd1",
"org.freedesktop.systemd1.Manager",
"CreateSnapshot",
&reply,
NULL,
DBUS_TYPE_STRING, &n,
if (r < 0)
return r;
return -EIO;
}
bus,
"org.freedesktop.systemd1",
path,
"org.freedesktop.DBus.Properties",
"Get",
&reply,
NULL,
if (r < 0)
return r;
log_error("Failed to parse reply.");
return -EIO;
}
log_error("Failed to parse reply.");
return -EIO;
}
if (!arg_quiet)
return 0;
}
char **name;
_cleanup_free_ char *n = NULL;
int r;
if (!n)
return log_oom();
bus,
"org.freedesktop.systemd1",
"/org/freedesktop/systemd1",
"org.freedesktop.systemd1.Manager",
"RemoveSnapshot",
NULL,
NULL,
DBUS_TYPE_STRING, &n,
if (r < 0)
return r;
}
return 0;
}
int r;
const char *method;
if (arg_action == ACTION_RELOAD)
method = "Reload";
else if (arg_action == ACTION_REEXEC)
method = "Reexecute";
else {
method =
/* "daemon-reload" */ "Reload";
}
bus,
"org.freedesktop.systemd1",
"/org/freedesktop/systemd1",
"org.freedesktop.systemd1.Manager",
NULL,
&error,
/* 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;
}
int r = 0;
char **name;
_cleanup_free_ char *n;
n = unit_name_mangle(*name);
if (!n)
return log_oom();
bus,
"org.freedesktop.systemd1",
"/org/freedesktop/systemd1",
"org.freedesktop.systemd1.Manager",
"ResetFailedUnit",
NULL,
NULL,
DBUS_TYPE_STRING, &n,
if (r < 0)
return r;
}
return 0;
}
int r;
const char
*interface = "org.freedesktop.systemd1.Manager",
*property = "Environment";
bus,
"org.freedesktop.systemd1",
"/org/freedesktop/systemd1",
"org.freedesktop.DBus.Properties",
"Get",
&reply,
NULL,
if (r < 0)
return r;
log_error("Failed to parse reply.");
return -EIO;
}
log_error("Failed to parse reply.");
return -EIO;
}
const char *text;
log_error("Failed to parse reply.");
return -EIO;
}
}
return 0;
}
unsigned l;
const char *root;
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();
return bus_method_call_with_reply(
bus,
"org.freedesktop.systemd1",
"/org/freedesktop/systemd1",
"org.freedesktop.systemd1.Manager",
"SwitchRoot",
NULL,
NULL,
}
const char *method;
int r;
? "SetEnvironment"
: "UnsetEnvironment";
"org.freedesktop.systemd1",
"/org/freedesktop/systemd1",
"org.freedesktop.systemd1.Manager",
method);
if (!m)
return log_oom();
if (r < 0)
return log_oom();
if (!reply) {
return -EIO;
}
return 0;
}
static int enable_sysv_units(char **args) {
int r = 0;
#if defined(HAVE_SYSV_COMPAT) && defined(HAVE_CHKCONFIG)
unsigned f = 1, t = 1;
LookupPaths paths = {};
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 = 1; 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++] = path_get_file_name(p);
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 */
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, i;
int carries_install_info = -1;
int r;
r = enable_sysv_units(args);
if (r < 0)
return r;
if (!args[1])
return 0;
if (r < 0)
goto finish;
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) {
for (i = 0; i < n_changes; i++) {
else
}
}
r = 0;
} else {
const char *method;
bool send_force = true, expect_carries_install_info = false;
dbus_bool_t a, b;
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;
method = "SetDefaultTarget";
} else
assert_not_reached("Unknown verb");
"org.freedesktop.systemd1",
"/org/freedesktop/systemd1",
"org.freedesktop.systemd1.Manager",
method);
if (!m) {
r = log_oom();
goto finish;
}
if (r < 0) {
log_error("Failed to append unit files.");
goto finish;
}
a = arg_runtime;
log_error("Failed to append runtime boolean.");
r = -ENOMEM;
goto finish;
}
if (send_force) {
b = arg_force;
log_error("Failed to append force boolean.");
r = -ENOMEM;
goto finish;
}
}
if (!reply) {
r = -EIO;
goto finish;
}
log_error("Failed to initialize iterator.");
goto finish;
}
if (expect_carries_install_info) {
if (r < 0) {
log_error("Failed to parse reply.");
goto finish;
}
carries_install_info = b;
}
log_error("Failed to parse reply.");
r = -EIO;
goto finish;
}
log_error("Failed to parse reply.");
r = -EIO;
goto finish;
}
log_error("Failed to parse reply.");
r = -EIO;
goto finish;
}
if (!arg_quiet) {
else
}
}
/* Try to reload if enabeld */
if (!arg_no_reload)
}
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;
}
const char* property = "LogLevel";
const char* interface = "org.freedesktop.systemd1.Manager";
const char* value;
m = dbus_message_new_method_call("org.freedesktop.systemd1",
"/org/freedesktop/systemd1",
"org.freedesktop.DBus.Properties",
"Set");
if (!m)
return log_oom();
return log_oom();
return log_oom();
}
return log_oom();
if (!reply) {
return -EIO;
}
return 0;
}
int r;
bool enabled;
char **name;
char *n;
r = enable_sysv_units(args);
if (r < 0)
return r;
enabled = r > 0;
n = unit_name_mangle(*name);
if (!n)
return log_oom();
free(n);
if (state < 0)
return state;
if (state == UNIT_FILE_ENABLED ||
enabled = true;
if (!arg_quiet)
}
} else {
const char *s;
n = unit_name_mangle(*name);
if (!n)
return log_oom();
bus,
"org.freedesktop.systemd1",
"/org/freedesktop/systemd1",
"org.freedesktop.systemd1.Manager",
"GetUnitFileState",
&reply,
NULL,
DBUS_TYPE_STRING, &n,
free(n);
if (r)
return r;
DBUS_TYPE_STRING, &s,
return -EIO;
}
if (streq(s, "enabled") ||
streq(s, "enabled-runtime") ||
streq(s, "static"))
enabled = true;
if (!arg_quiet)
puts(s);
}
}
return enabled ? 0 : 1;
}
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"
" -t --type=TYPE List only units of a particular type\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"
" --reverse Show reverse dependencies with 'list-dependencies'\n"
" --failed Show only failed units\n"
" -l --full Don't ellipsize unit names on output\n"
" --fail When queueing a new job, fail if conflicting jobs are\n"
" pending\n"
" --irreversible Create jobs which cannot be implicitly cancelled\n"
" --show-types When showing sockets, explicitly show their type\n"
" --ignore-dependencies\n"
" When queueing a new job, ignore all its dependencies\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"
" -H --host=[USER@]HOST\n"
" Show information for remote host\n"
" -P --privileged Acquire privileges before execution\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"
" --system Connect to system manager\n"
" --user Connect to user service manager\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"
" --runtime Enable unit files only temporarily until next reboot\n"
" -n --lines=INTEGER 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\n"
"Unit Commands:\n"
" list-units List loaded units\n"
" list-sockets List loaded sockets ordered by address\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"
" 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"
" get-cgroup-attr [NAME] [ATTR] ...\n"
" Get control group attrubute\n"
" set-cgroup-attr [NAME] [ATTR] [VALUE] ...\n"
" Set control group attribute\n"
" unset-cgroup-attr [NAME] [ATTR...]\n"
" Unset control group attribute\n"
" set-cgroup [NAME] [CGROUP...] Add unit to a control group\n"
" unset-cgroup [NAME] [CGROUP...] Remove unit from a control group\n"
" load [NAME...] Load one or more 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 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"
" 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"
" is-enabled [NAME...] Check whether unit files are enabled\n\n"
"Job Commands:\n"
" list-jobs List jobs\n"
" cancel [JOB...] Cancel all, one, or more jobs\n\n"
"Status Commands:\n"
" dump Dump server status\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"
" set-log-level LEVEL Set logging threshold for systemd\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 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...]\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);
}
puts("\nAvailable unit load states: ");
for(i = 0; i < _UNIT_LOAD_STATE_MAX; i++) {
t = unit_load_state_to_string(i);
if (t)
puts(t);
}
return 0;
}
enum {
ARG_FAIL = 0x100,
};
};
int c;
switch (c) {
case 'h':
return 0;
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;
}
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_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:
arg_failed = true;
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 'P':
break;
case 'H':
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 '?':
return -EINVAL;
default:
log_error("Unknown option code '%c'.", c);
return -EINVAL;
}
}
log_error("Cannot access user instance remotely.");
return -EINVAL;
}
return 1;
}
enum {
ARG_HELP = 0x100,
};
};
int c, runlevel;
arg_force = 2;
switch (c) {
case ARG_HELP:
halt_help();
return 0;
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:
log_error("Unknown option code '%c'.", c);
return -EINVAL;
}
}
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 0;
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:
log_error("Unknown option code '%c'.", c);
return -EINVAL;
}
}
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:
telinit_help();
return 0;
case ARG_NO_WALL:
arg_no_wall = true;
break;
case '?':
return -EINVAL;
default:
log_error("Unknown option code '%c'.", c);
return -EINVAL;
}
}
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 0;
case '?':
return -EINVAL;
default:
log_error("Unknown option code '%c'.", c);
return -EINVAL;
}
}
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_upstart(void) {
char
env1_buf[] = "RUNLEVEL=X",
env2_buf[] = "PREVLEVEL=X";
const char *emit = "runlevel";
if (!(rl = action_to_runlevel()))
return 0;
previous = 'N';
r = 0;
goto finish;
}
r = -EIO;
goto finish;
}
if ((r = bus_check_peercred(bus)) < 0) {
log_error("Failed to verify owner of bus.");
goto finish;
}
if (!(m = dbus_message_new_method_call(
"com.ubuntu.Upstart",
"com.ubuntu.Upstart0_6",
"EmitEvent"))) {
log_error("Could not allocate message.");
r = -ENOMEM;
goto finish;
}
log_error("Could not append arguments to message.");
r = -ENOMEM;
goto finish;
}
if (bus_error_is_no_service(&error)) {
r = -EADDRNOTAVAIL;
goto finish;
}
r = -EIO;
goto finish;
}
r = 1;
if (bus) {
}
return r;
}
static int talk_initctl(void) {
struct init_request request = {};
int r;
char rl;
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;
} verbs[] = {
};
int left;
unsigned i;
if (left <= 0)
/* Special rule: no arguments means "list-units" */
i = 0;
else {
log_error("This command expects one or more "
"unit names. Did you mean --help?");
return -EINVAL;
}
for (i = 0; i < ELEMENTSOF(verbs); i++)
break;
if (i >= ELEMENTSOF(verbs)) {
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
if (running_in_chroot() > 0) {
log_info("Running in chroot, ignoring request.");
return 0;
}
log_error("Failed to get D-Bus connection: %s",
return -EIO;
}
} else {
log_error("Failed to get D-Bus connection: %s",
return -EIO;
}
}
}
_cleanup_close_ int fd;
struct sd_shutdown_command c = {
.usec = t,
};
union sockaddr_union sockaddr = {
};
{.iov_base = (char*) &c,
}
};
.msg_iovlen = 1,
};
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;
}
/* Hmm, talking to systemd via D-Bus didn't work. Then
* let's try to talk to Upstart via D-Bus. */
if (talk_upstart() > 0)
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.");
break;
case ACTION_POWEROFF:
log_info("Powering off.");
break;
case ACTION_REBOOT:
log_info("Rebooting.");
break;
default:
assert_not_reached("Unknown halt action.");
}
assert_not_reached("Uh? This shouldn't happen.");
}
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;
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;
/* We should never reach this. */
return -ENOSYS;
}
static int runlevel_main(void) {
if (r < 0) {
puts("unknown");
return r;
}
printf("%c %c\n",
return 0;
}
int r, retval = EXIT_FAILURE;
log_open();
if (r < 0)
goto finish;
else if (r == 0) {
goto finish;
}
* let's shortcut this */
if (arg_action == ACTION_RUNLEVEL) {
r = runlevel_main();
retval = r < 0 ? EXIT_FAILURE : r;
goto finish;
}
log_info("Running in chroot, ignoring request.");
retval = 0;
goto finish;
}
if (!avoid_bus()) {
if (arg_transport == TRANSPORT_NORMAL)
bus_connect(arg_scope == UNIT_FILE_SYSTEM ? DBUS_BUS_SYSTEM : DBUS_BUS_SESSION, &bus, &private_bus, &error);
else if (arg_transport == TRANSPORT_POLKIT) {
private_bus = false;
} else if (arg_transport == TRANSPORT_SSH) {
private_bus = false;
} else
assert_not_reached("Uh, invalid transport...");
}
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: {
char *m = NULL;
if (arg_wall) {
if (!m) {
goto finish;
}
}
if (r < 0)
free(m);
break;
}
case ACTION_INVALID:
case ACTION_RUNLEVEL:
default:
assert_not_reached("Unknown action");
}
retval = r < 0 ? EXIT_FAILURE : r;
if (bus) {
}
pager_close();
return retval;
}