chiark / gitweb /
backlight: instead of syspath use sysname for identifying backlight devices
[elogind.git] / src / systemctl / systemctl.c
1 /*-*- Mode: C; c-basic-offset: 8; indent-tabs-mode: nil -*-*/
2
3 /***
4   This file is part of systemd.
5
6   Copyright 2010 Lennart Poettering
7
8   systemd is free software; you can redistribute it and/or modify it
9   under the terms of the GNU Lesser General Public License as published by
10   the Free Software Foundation; either version 2.1 of the License, or
11   (at your option) any later version.
12
13   systemd is distributed in the hope that it will be useful, but
14   WITHOUT ANY WARRANTY; without even the implied warranty of
15   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
16   Lesser General Public License for more details.
17
18   You should have received a copy of the GNU Lesser General Public License
19   along with systemd; If not, see <http://www.gnu.org/licenses/>.
20 ***/
21
22 #include <sys/reboot.h>
23 #include <stdio.h>
24 #include <getopt.h>
25 #include <locale.h>
26 #include <stdbool.h>
27 #include <string.h>
28 #include <errno.h>
29 #include <sys/ioctl.h>
30 #include <termios.h>
31 #include <unistd.h>
32 #include <fcntl.h>
33 #include <sys/socket.h>
34 #include <sys/stat.h>
35 #include <stddef.h>
36 #include <sys/prctl.h>
37 #include <dbus/dbus.h>
38
39 #include <systemd/sd-daemon.h>
40 #include <systemd/sd-shutdown.h>
41 #include <systemd/sd-login.h>
42
43 #include "log.h"
44 #include "util.h"
45 #include "macro.h"
46 #include "set.h"
47 #include "utmp-wtmp.h"
48 #include "special.h"
49 #include "initreq.h"
50 #include "path-util.h"
51 #include "strv.h"
52 #include "dbus-common.h"
53 #include "cgroup-show.h"
54 #include "cgroup-util.h"
55 #include "list.h"
56 #include "path-lookup.h"
57 #include "conf-parser.h"
58 #include "exit-status.h"
59 #include "bus-errors.h"
60 #include "build.h"
61 #include "unit-name.h"
62 #include "pager.h"
63 #include "spawn-ask-password-agent.h"
64 #include "spawn-polkit-agent.h"
65 #include "install.h"
66 #include "logs-show.h"
67 #include "path-util.h"
68 #include "socket-util.h"
69 #include "fileio.h"
70
71 static char **arg_types = NULL;
72 static char **arg_states = NULL;
73 static char **arg_properties = NULL;
74 static bool arg_all = false;
75 static enum dependency {
76         DEPENDENCY_FORWARD,
77         DEPENDENCY_REVERSE,
78         DEPENDENCY_AFTER,
79         DEPENDENCY_BEFORE,
80 } arg_dependency = DEPENDENCY_FORWARD;
81 static const char *arg_job_mode = "replace";
82 static UnitFileScope arg_scope = UNIT_FILE_SYSTEM;
83 static bool arg_no_block = false;
84 static bool arg_no_legend = false;
85 static bool arg_no_pager = false;
86 static bool arg_no_wtmp = false;
87 static bool arg_no_wall = false;
88 static bool arg_no_reload = false;
89 static bool arg_show_types = false;
90 static bool arg_ignore_inhibitors = false;
91 static bool arg_dry = false;
92 static bool arg_quiet = false;
93 static bool arg_full = false;
94 static int arg_force = 0;
95 static bool arg_ask_password = true;
96 static bool arg_runtime = false;
97 static char **arg_wall = NULL;
98 static const char *arg_kill_who = NULL;
99 static int arg_signal = SIGTERM;
100 static const char *arg_root = NULL;
101 static usec_t arg_when = 0;
102 static enum action {
103         ACTION_INVALID,
104         ACTION_SYSTEMCTL,
105         ACTION_HALT,
106         ACTION_POWEROFF,
107         ACTION_REBOOT,
108         ACTION_KEXEC,
109         ACTION_EXIT,
110         ACTION_SUSPEND,
111         ACTION_HIBERNATE,
112         ACTION_HYBRID_SLEEP,
113         ACTION_RUNLEVEL2,
114         ACTION_RUNLEVEL3,
115         ACTION_RUNLEVEL4,
116         ACTION_RUNLEVEL5,
117         ACTION_RESCUE,
118         ACTION_EMERGENCY,
119         ACTION_DEFAULT,
120         ACTION_RELOAD,
121         ACTION_REEXEC,
122         ACTION_RUNLEVEL,
123         ACTION_CANCEL_SHUTDOWN,
124         _ACTION_MAX
125 } arg_action = ACTION_SYSTEMCTL;
126 static enum transport {
127         TRANSPORT_NORMAL,
128         TRANSPORT_SSH,
129         TRANSPORT_POLKIT
130 } arg_transport = TRANSPORT_NORMAL;
131 static char *arg_host = NULL;
132 static char *arg_user = NULL;
133 static unsigned arg_lines = 10;
134 static OutputMode arg_output = OUTPUT_SHORT;
135 static bool arg_plain = false;
136
137 static bool private_bus = false;
138
139 static int daemon_reload(DBusConnection *bus, char **args);
140 static void halt_now(enum action a);
141
142 static void pager_open_if_enabled(void) {
143
144         if (arg_no_pager)
145                 return;
146
147         pager_open(false);
148 }
149
150 static void ask_password_agent_open_if_enabled(void) {
151
152         /* Open the password agent as a child process if necessary */
153
154         if (!arg_ask_password)
155                 return;
156
157         if (arg_scope != UNIT_FILE_SYSTEM)
158                 return;
159
160         ask_password_agent_open();
161 }
162
163 #ifdef HAVE_LOGIND
164 static void polkit_agent_open_if_enabled(void) {
165
166         /* Open the polkit agent as a child process if necessary */
167
168         if (!arg_ask_password)
169                 return;
170
171         if (arg_scope != UNIT_FILE_SYSTEM)
172                 return;
173
174         polkit_agent_open();
175 }
176 #endif
177
178 static int translate_bus_error_to_exit_status(int r, const DBusError *error) {
179         assert(error);
180
181         if (!dbus_error_is_set(error))
182                 return r;
183
184         if (dbus_error_has_name(error, DBUS_ERROR_ACCESS_DENIED) ||
185             dbus_error_has_name(error, BUS_ERROR_ONLY_BY_DEPENDENCY) ||
186             dbus_error_has_name(error, BUS_ERROR_NO_ISOLATION) ||
187             dbus_error_has_name(error, BUS_ERROR_TRANSACTION_IS_DESTRUCTIVE))
188                 return EXIT_NOPERMISSION;
189
190         if (dbus_error_has_name(error, BUS_ERROR_NO_SUCH_UNIT))
191                 return EXIT_NOTINSTALLED;
192
193         if (dbus_error_has_name(error, BUS_ERROR_JOB_TYPE_NOT_APPLICABLE) ||
194             dbus_error_has_name(error, BUS_ERROR_NOT_SUPPORTED))
195                 return EXIT_NOTIMPLEMENTED;
196
197         if (dbus_error_has_name(error, BUS_ERROR_LOAD_FAILED))
198                 return EXIT_NOTCONFIGURED;
199
200         if (r != 0)
201                 return r;
202
203         return EXIT_FAILURE;
204 }
205
206 static void warn_wall(enum action a) {
207         static const char *table[_ACTION_MAX] = {
208                 [ACTION_HALT]            = "The system is going down for system halt NOW!",
209                 [ACTION_REBOOT]          = "The system is going down for reboot NOW!",
210                 [ACTION_POWEROFF]        = "The system is going down for power-off NOW!",
211                 [ACTION_KEXEC]           = "The system is going down for kexec reboot NOW!",
212                 [ACTION_RESCUE]          = "The system is going down to rescue mode NOW!",
213                 [ACTION_EMERGENCY]       = "The system is going down to emergency mode NOW!",
214                 [ACTION_CANCEL_SHUTDOWN] = "The system shutdown has been cancelled NOW!"
215         };
216
217         if (arg_no_wall)
218                 return;
219
220         if (arg_wall) {
221                 _cleanup_free_ char *p;
222
223                 p = strv_join(arg_wall, " ");
224                 if (!p) {
225                         log_oom();
226                         return;
227                 }
228
229                 if (*p) {
230                         utmp_wall(p, NULL);
231                         return;
232                 }
233         }
234
235         if (!table[a])
236                 return;
237
238         utmp_wall(table[a], NULL);
239 }
240
241 static bool avoid_bus(void) {
242
243         if (running_in_chroot() > 0)
244                 return true;
245
246         if (sd_booted() <= 0)
247                 return true;
248
249         if (!isempty(arg_root))
250                 return true;
251
252         if (arg_scope == UNIT_FILE_GLOBAL)
253                 return true;
254
255         return false;
256 }
257
258 static int compare_unit_info(const void *a, const void *b) {
259         const char *d1, *d2;
260         const struct unit_info *u = a, *v = b;
261
262         d1 = strrchr(u->id, '.');
263         d2 = strrchr(v->id, '.');
264
265         if (d1 && d2) {
266                 int r;
267
268                 r = strcasecmp(d1, d2);
269                 if (r != 0)
270                         return r;
271         }
272
273         return strcasecmp(u->id, v->id);
274 }
275
276 static bool output_show_unit(const struct unit_info *u) {
277         const char *dot;
278
279         if (!strv_isempty(arg_states))
280                 return strv_contains(arg_states, u->load_state) || strv_contains(arg_states, u->sub_state) || strv_contains(arg_states, u->active_state);
281
282         return (!arg_types || ((dot = strrchr(u->id, '.')) &&
283                                strv_find(arg_types, dot+1))) &&
284                 (arg_all || !(streq(u->active_state, "inactive")
285                               || u->following[0]) || u->job_id > 0);
286 }
287
288 static void output_units_list(const struct unit_info *unit_infos, unsigned c) {
289         unsigned id_len, max_id_len, active_len, sub_len, job_len, desc_len, n_shown = 0;
290         const struct unit_info *u;
291         int job_count = 0;
292
293         max_id_len = sizeof("UNIT")-1;
294         active_len = sizeof("ACTIVE")-1;
295         sub_len = sizeof("SUB")-1;
296         job_len = sizeof("JOB")-1;
297         desc_len = 0;
298
299         for (u = unit_infos; u < unit_infos + c; u++) {
300                 if (!output_show_unit(u))
301                         continue;
302
303                 max_id_len = MAX(max_id_len, strlen(u->id));
304                 active_len = MAX(active_len, strlen(u->active_state));
305                 sub_len = MAX(sub_len, strlen(u->sub_state));
306                 if (u->job_id != 0) {
307                         job_len = MAX(job_len, strlen(u->job_type));
308                         job_count++;
309                 }
310         }
311
312         if (!arg_full) {
313                 unsigned basic_len;
314                 id_len = MIN(max_id_len, 25u);
315                 basic_len = 5 + id_len + 5 + active_len + sub_len;
316                 if (job_count)
317                         basic_len += job_len + 1;
318                 if (basic_len < (unsigned) columns()) {
319                         unsigned extra_len, incr;
320                         extra_len = columns() - basic_len;
321                         /* Either UNIT already got 25, or is fully satisfied.
322                          * Grant up to 25 to DESC now. */
323                         incr = MIN(extra_len, 25u);
324                         desc_len += incr;
325                         extra_len -= incr;
326                         /* split the remaining space between UNIT and DESC,
327                          * but do not give UNIT more than it needs. */
328                         if (extra_len > 0) {
329                                 incr = MIN(extra_len / 2, max_id_len - id_len);
330                                 id_len += incr;
331                                 desc_len += extra_len - incr;
332                         }
333                 }
334         } else
335                 id_len = max_id_len;
336
337         for (u = unit_infos; u < unit_infos + c; u++) {
338                 _cleanup_free_ char *e = NULL;
339                 const char *on_loaded, *off_loaded, *on = "";
340                 const char *on_active, *off_active, *off = "";
341
342                 if (!output_show_unit(u))
343                         continue;
344
345                 if (!n_shown && !arg_no_legend) {
346                         printf("%-*s %-6s %-*s %-*s ", id_len, "UNIT", "LOAD",
347                                active_len, "ACTIVE", sub_len, "SUB");
348                         if (job_count)
349                                 printf("%-*s ", job_len, "JOB");
350                         if (!arg_full && arg_no_pager)
351                                 printf("%.*s\n", desc_len, "DESCRIPTION");
352                         else
353                                 printf("%s\n", "DESCRIPTION");
354                 }
355
356                 n_shown++;
357
358                 if (streq(u->load_state, "error") ||
359                     streq(u->load_state, "not-found")) {
360                         on_loaded = on = ansi_highlight_red();
361                         off_loaded = off = ansi_highlight_off();
362                 } else
363                         on_loaded = off_loaded = "";
364
365                 if (streq(u->active_state, "failed")) {
366                         on_active = on = ansi_highlight_red();
367                         off_active = off = ansi_highlight_off();
368                 } else
369                         on_active = off_active = "";
370
371                 e = arg_full ? NULL : ellipsize(u->id, id_len, 33);
372
373                 printf("%s%-*s%s %s%-6s%s %s%-*s %-*s%s %-*s",
374                        on, id_len, e ? e : u->id, off,
375                        on_loaded, u->load_state, off_loaded,
376                        on_active, active_len, u->active_state,
377                        sub_len, u->sub_state, off_active,
378                        job_count ? job_len + 1 : 0, u->job_id ? u->job_type : "");
379                 if (!arg_full && arg_no_pager)
380                         printf("%.*s\n", desc_len, u->description);
381                 else
382                         printf("%s\n", u->description);
383         }
384
385         if (!arg_no_legend) {
386                 const char *on, *off;
387
388                 if (n_shown) {
389                         printf("\nLOAD   = Reflects whether the unit definition was properly loaded.\n"
390                                "ACTIVE = The high-level unit activation state, i.e. generalization of SUB.\n"
391                                "SUB    = The low-level unit activation state, values depend on unit type.\n");
392                         if (job_count)
393                                 printf("JOB    = Pending job for the unit.\n");
394                         puts("");
395                         on = ansi_highlight();
396                         off = ansi_highlight_off();
397                 } else {
398                         on = ansi_highlight_red();
399                         off = ansi_highlight_off();
400                 }
401
402                 if (arg_all)
403                         printf("%s%u loaded units listed.%s\n"
404                                "To show all installed unit files use 'systemctl list-unit-files'.\n",
405                                on, n_shown, off);
406                 else
407                         printf("%s%u loaded units listed.%s Pass --all to see loaded but inactive units, too.\n"
408                                "To show all installed unit files use 'systemctl list-unit-files'.\n",
409                                on, n_shown, off);
410         }
411 }
412
413 static int get_unit_list(
414                 DBusConnection *bus,
415                 DBusMessage **reply,
416                 struct unit_info **unit_infos,
417                 unsigned *c) {
418
419         DBusMessageIter iter, sub;
420         size_t size = 0;
421         int r;
422
423         assert(bus);
424         assert(unit_infos);
425         assert(c);
426
427         r = bus_method_call_with_reply(
428                         bus,
429                         "org.freedesktop.systemd1",
430                         "/org/freedesktop/systemd1",
431                         "org.freedesktop.systemd1.Manager",
432                         "ListUnits",
433                         reply,
434                         NULL,
435                         DBUS_TYPE_INVALID);
436         if (r < 0)
437                 return r;
438
439         if (!dbus_message_iter_init(*reply, &iter) ||
440             dbus_message_iter_get_arg_type(&iter) != DBUS_TYPE_ARRAY ||
441             dbus_message_iter_get_element_type(&iter) != DBUS_TYPE_STRUCT)  {
442                 log_error("Failed to parse reply.");
443                 return -EIO;
444         }
445
446         dbus_message_iter_recurse(&iter, &sub);
447
448         while (dbus_message_iter_get_arg_type(&sub) != DBUS_TYPE_INVALID) {
449                 if (!GREEDY_REALLOC(*unit_infos, size, *c + 1))
450                         return log_oom();
451
452                 bus_parse_unit_info(&sub, *unit_infos + *c);
453                 (*c)++;
454
455                 dbus_message_iter_next(&sub);
456         }
457
458         return 0;
459 }
460
461 static int list_units(DBusConnection *bus, char **args) {
462         _cleanup_dbus_message_unref_ DBusMessage *reply = NULL;
463         _cleanup_free_ struct unit_info *unit_infos = NULL;
464         unsigned c = 0;
465         int r;
466
467         pager_open_if_enabled();
468
469         r = get_unit_list(bus, &reply, &unit_infos, &c);
470         if (r < 0)
471                 return r;
472
473         qsort(unit_infos, c, sizeof(struct unit_info), compare_unit_info);
474
475         output_units_list(unit_infos, c);
476
477         return 0;
478 }
479
480 static int get_triggered_units(
481                 DBusConnection *bus,
482                 const char* unit_path,
483                 char*** triggered) {
484
485         const char *interface = "org.freedesktop.systemd1.Unit",
486                    *triggers_property = "Triggers";
487         _cleanup_dbus_message_unref_ DBusMessage *reply = NULL;
488         DBusMessageIter iter, sub;
489         int r;
490
491         r = bus_method_call_with_reply(bus,
492                                        "org.freedesktop.systemd1",
493                                        unit_path,
494                                        "org.freedesktop.DBus.Properties",
495                                        "Get",
496                                        &reply,
497                                        NULL,
498                                        DBUS_TYPE_STRING, &interface,
499                                        DBUS_TYPE_STRING, &triggers_property,
500                                        DBUS_TYPE_INVALID);
501         if (r < 0)
502                 return r;
503
504         if (!dbus_message_iter_init(reply, &iter) ||
505             dbus_message_iter_get_arg_type(&iter) != DBUS_TYPE_VARIANT) {
506                 log_error("Failed to parse reply.");
507                 return -EBADMSG;
508         }
509
510         dbus_message_iter_recurse(&iter, &sub);
511         dbus_message_iter_recurse(&sub, &iter);
512         sub = iter;
513
514         while (dbus_message_iter_get_arg_type(&sub) != DBUS_TYPE_INVALID) {
515                 const char *unit;
516
517                 if (dbus_message_iter_get_arg_type(&sub) != DBUS_TYPE_STRING) {
518                         log_error("Failed to parse reply.");
519                         return -EBADMSG;
520                 }
521
522                 dbus_message_iter_get_basic(&sub, &unit);
523                 r = strv_extend(triggered, unit);
524                 if (r < 0)
525                         return r;
526
527                 dbus_message_iter_next(&sub);
528         }
529
530         return 0;
531 }
532
533 static int get_listening(DBusConnection *bus, const char* unit_path,
534                          char*** listen, unsigned *c)
535 {
536         const char *interface = "org.freedesktop.systemd1.Socket",
537                    *listen_property = "Listen";
538         _cleanup_dbus_message_unref_ DBusMessage *reply = NULL;
539         DBusMessageIter iter, sub;
540         int r;
541
542         r = bus_method_call_with_reply(bus,
543                                        "org.freedesktop.systemd1",
544                                        unit_path,
545                                        "org.freedesktop.DBus.Properties",
546                                        "Get",
547                                        &reply,
548                                        NULL,
549                                        DBUS_TYPE_STRING, &interface,
550                                        DBUS_TYPE_STRING, &listen_property,
551                                        DBUS_TYPE_INVALID);
552         if (r < 0)
553                 return r;
554
555         if (!dbus_message_iter_init(reply, &iter) ||
556             dbus_message_iter_get_arg_type(&iter) != DBUS_TYPE_VARIANT) {
557                 log_error("Failed to parse reply.");
558                 return -EBADMSG;
559         }
560
561         dbus_message_iter_recurse(&iter, &sub);
562         dbus_message_iter_recurse(&sub, &iter);
563         sub = iter;
564
565         while (dbus_message_iter_get_arg_type(&sub) != DBUS_TYPE_INVALID) {
566                 DBusMessageIter sub2;
567                 const char *type, *path;
568
569                 if (dbus_message_iter_get_arg_type(&sub) != DBUS_TYPE_STRUCT) {
570                         log_error("Failed to parse reply.");
571                         return -EBADMSG;
572                 }
573
574                 dbus_message_iter_recurse(&sub, &sub2);
575
576                 if (bus_iter_get_basic_and_next(&sub2, DBUS_TYPE_STRING, &type, true) >= 0 &&
577                     bus_iter_get_basic_and_next(&sub2, DBUS_TYPE_STRING, &path, false) >= 0) {
578                         r = strv_extend(listen, type);
579                         if (r < 0)
580                                 return r;
581
582                         r = strv_extend(listen, path);
583                         if (r < 0)
584                                 return r;
585
586                         (*c) ++;
587                 }
588
589                 dbus_message_iter_next(&sub);
590         }
591
592         return 0;
593 }
594
595 struct socket_info {
596         const char* id;
597
598         char* type;
599         char* path;
600
601         /* Note: triggered is a list here, although it almost certainly
602          * will always be one unit. Nevertheless, dbus API allows for multiple
603          * values, so let's follow that.*/
604         char** triggered;
605
606         /* The strv above is shared. free is set only in the first one. */
607         bool own_triggered;
608 };
609
610 static int socket_info_compare(struct socket_info *a, struct socket_info *b) {
611         int o = strcmp(a->path, b->path);
612         if (o == 0)
613                 o = strcmp(a->type, b->type);
614         return o;
615 }
616
617 static int output_sockets_list(struct socket_info *socket_infos, unsigned cs) {
618         struct socket_info *s;
619         unsigned pathlen = sizeof("LISTEN") - 1,
620                 typelen = (sizeof("TYPE") - 1) * arg_show_types,
621                 socklen = sizeof("UNIT") - 1,
622                 servlen = sizeof("ACTIVATES") - 1;
623         const char *on, *off;
624
625         for (s = socket_infos; s < socket_infos + cs; s++) {
626                 char **a;
627                 unsigned tmp = 0;
628
629                 socklen = MAX(socklen, strlen(s->id));
630                 if (arg_show_types)
631                         typelen = MAX(typelen, strlen(s->type));
632                 pathlen = MAX(pathlen, strlen(s->path));
633
634                 STRV_FOREACH(a, s->triggered)
635                         tmp += strlen(*a) + 2*(a != s->triggered);
636                 servlen = MAX(servlen, tmp);
637         }
638
639         if (cs) {
640                 if (!arg_no_legend)
641                         printf("%-*s %-*.*s%-*s %s\n",
642                                pathlen, "LISTEN",
643                                typelen + arg_show_types, typelen + arg_show_types, "TYPE ",
644                                socklen, "UNIT",
645                                "ACTIVATES");
646
647                 for (s = socket_infos; s < socket_infos + cs; s++) {
648                         char **a;
649
650                         if (arg_show_types)
651                                 printf("%-*s %-*s %-*s",
652                                        pathlen, s->path, typelen, s->type, socklen, s->id);
653                         else
654                                 printf("%-*s %-*s",
655                                        pathlen, s->path, socklen, s->id);
656                         STRV_FOREACH(a, s->triggered)
657                                 printf("%s %s",
658                                        a == s->triggered ? "" : ",", *a);
659                         printf("\n");
660                 }
661
662                 on = ansi_highlight();
663                 off = ansi_highlight_off();
664                 if (!arg_no_legend)
665                         printf("\n");
666         } else {
667                 on = ansi_highlight_red();
668                 off = ansi_highlight_off();
669         }
670
671         if (!arg_no_legend) {
672                 printf("%s%u sockets listed.%s\n", on, cs, off);
673                 if (!arg_all)
674                         printf("Pass --all to see loaded but inactive sockets, too.\n");
675         }
676
677         return 0;
678 }
679
680 static int list_sockets(DBusConnection *bus, char **args) {
681         _cleanup_dbus_message_unref_ DBusMessage *reply = NULL;
682         _cleanup_free_ struct unit_info *unit_infos = NULL;
683         struct socket_info *socket_infos = NULL;
684         const struct unit_info *u;
685         struct socket_info *s;
686         unsigned cu = 0, cs = 0;
687         size_t size = 0;
688         int r;
689
690         pager_open_if_enabled();
691
692         r = get_unit_list(bus, &reply, &unit_infos, &cu);
693         if (r < 0)
694                 return r;
695
696         for (u = unit_infos; u < unit_infos + cu; u++) {
697                 const char *dot;
698                 _cleanup_strv_free_ char **listen = NULL, **triggered = NULL;
699                 unsigned c = 0, i;
700
701                 if (!output_show_unit(u))
702                         continue;
703
704                 if ((dot = strrchr(u->id, '.')) && !streq(dot+1, "socket"))
705                         continue;
706
707                 r = get_triggered_units(bus, u->unit_path, &triggered);
708                 if (r < 0)
709                         goto cleanup;
710
711                 r = get_listening(bus, u->unit_path, &listen, &c);
712                 if (r < 0)
713                         goto cleanup;
714
715                 if (!GREEDY_REALLOC(socket_infos, size, cs + c)) {
716                         r = log_oom();
717                         goto cleanup;
718                 }
719
720                 for (i = 0; i < c; i++)
721                         socket_infos[cs + i] = (struct socket_info) {
722                                 .id = u->id,
723                                 .type = listen[i*2],
724                                 .path = listen[i*2 + 1],
725                                 .triggered = triggered,
726                                 .own_triggered = i==0,
727                         };
728
729                 /* from this point on we will cleanup those socket_infos */
730                 cs += c;
731                 free(listen);
732                 listen = triggered = NULL; /* avoid cleanup */
733         }
734
735         qsort(socket_infos, cs, sizeof(struct socket_info),
736               (__compar_fn_t) socket_info_compare);
737
738         output_sockets_list(socket_infos, cs);
739
740  cleanup:
741         assert(cs == 0 || socket_infos);
742         for (s = socket_infos; s < socket_infos + cs; s++) {
743                 free(s->type);
744                 free(s->path);
745                 if (s->own_triggered)
746                         strv_free(s->triggered);
747         }
748         free(socket_infos);
749
750         return 0;
751 }
752
753 static int compare_unit_file_list(const void *a, const void *b) {
754         const char *d1, *d2;
755         const UnitFileList *u = a, *v = b;
756
757         d1 = strrchr(u->path, '.');
758         d2 = strrchr(v->path, '.');
759
760         if (d1 && d2) {
761                 int r;
762
763                 r = strcasecmp(d1, d2);
764                 if (r != 0)
765                         return r;
766         }
767
768         return strcasecmp(path_get_file_name(u->path), path_get_file_name(v->path));
769 }
770
771 static bool output_show_unit_file(const UnitFileList *u) {
772         const char *dot;
773
774         return !arg_types || ((dot = strrchr(u->path, '.')) && strv_find(arg_types, dot+1));
775 }
776
777 static void output_unit_file_list(const UnitFileList *units, unsigned c) {
778         unsigned max_id_len, id_cols, state_cols, n_shown = 0;
779         const UnitFileList *u;
780
781         max_id_len = sizeof("UNIT FILE")-1;
782         state_cols = sizeof("STATE")-1;
783         for (u = units; u < units + c; u++) {
784                 if (!output_show_unit_file(u))
785                         continue;
786
787                 max_id_len = MAX(max_id_len, strlen(path_get_file_name(u->path)));
788                 state_cols = MAX(state_cols, strlen(unit_file_state_to_string(u->state)));
789         }
790
791         if (!arg_full) {
792                 unsigned basic_cols;
793                 id_cols = MIN(max_id_len, 25u);
794                 basic_cols = 1 + id_cols + state_cols;
795                 if (basic_cols < (unsigned) columns())
796                         id_cols += MIN(columns() - basic_cols, max_id_len - id_cols);
797         } else
798                 id_cols = max_id_len;
799
800         if (!arg_no_legend)
801                 printf("%-*s %-*s\n", id_cols, "UNIT FILE", state_cols, "STATE");
802
803         for (u = units; u < units + c; u++) {
804                 _cleanup_free_ char *e = NULL;
805                 const char *on, *off;
806                 const char *id;
807
808                 if (!output_show_unit_file(u))
809                         continue;
810
811                 n_shown++;
812
813                 if (u->state == UNIT_FILE_MASKED ||
814                     u->state == UNIT_FILE_MASKED_RUNTIME ||
815                     u->state == UNIT_FILE_DISABLED ||
816                     u->state == UNIT_FILE_INVALID) {
817                         on  = ansi_highlight_red();
818                         off = ansi_highlight_off();
819                 } else if (u->state == UNIT_FILE_ENABLED) {
820                         on  = ansi_highlight_green();
821                         off = ansi_highlight_off();
822                 } else
823                         on = off = "";
824
825                 id = path_get_file_name(u->path);
826
827                 e = arg_full ? NULL : ellipsize(id, id_cols, 33);
828
829                 printf("%-*s %s%-*s%s\n",
830                        id_cols, e ? e : id,
831                        on, state_cols, unit_file_state_to_string(u->state), off);
832         }
833
834         if (!arg_no_legend)
835                 printf("\n%u unit files listed.\n", n_shown);
836 }
837
838 static int list_unit_files(DBusConnection *bus, char **args) {
839         _cleanup_dbus_message_unref_ DBusMessage *reply = NULL;
840         _cleanup_free_ UnitFileList *units = NULL;
841         DBusMessageIter iter, sub, sub2;
842         unsigned c = 0, n_units = 0;
843         int r;
844
845         pager_open_if_enabled();
846
847         if (avoid_bus()) {
848                 Hashmap *h;
849                 UnitFileList *u;
850                 Iterator i;
851
852                 h = hashmap_new(string_hash_func, string_compare_func);
853                 if (!h)
854                         return log_oom();
855
856                 r = unit_file_get_list(arg_scope, arg_root, h);
857                 if (r < 0) {
858                         unit_file_list_free(h);
859                         log_error("Failed to get unit file list: %s", strerror(-r));
860                         return r;
861                 }
862
863                 n_units = hashmap_size(h);
864                 units = new(UnitFileList, n_units);
865                 if (!units) {
866                         unit_file_list_free(h);
867                         return log_oom();
868                 }
869
870                 HASHMAP_FOREACH(u, h, i) {
871                         memcpy(units + c++, u, sizeof(UnitFileList));
872                         free(u);
873                 }
874
875                 hashmap_free(h);
876         } else {
877                 r = bus_method_call_with_reply(
878                                 bus,
879                                 "org.freedesktop.systemd1",
880                                 "/org/freedesktop/systemd1",
881                                 "org.freedesktop.systemd1.Manager",
882                                 "ListUnitFiles",
883                                 &reply,
884                                 NULL,
885                                 DBUS_TYPE_INVALID);
886                 if (r < 0)
887                         return r;
888
889                 if (!dbus_message_iter_init(reply, &iter) ||
890                     dbus_message_iter_get_arg_type(&iter) != DBUS_TYPE_ARRAY ||
891                     dbus_message_iter_get_element_type(&iter) != DBUS_TYPE_STRUCT)  {
892                         log_error("Failed to parse reply.");
893                         return -EIO;
894                 }
895
896                 dbus_message_iter_recurse(&iter, &sub);
897
898                 while (dbus_message_iter_get_arg_type(&sub) != DBUS_TYPE_INVALID) {
899                         UnitFileList *u;
900                         const char *state;
901
902                         assert(dbus_message_iter_get_arg_type(&sub) == DBUS_TYPE_STRUCT);
903
904                         if (c >= n_units) {
905                                 UnitFileList *w;
906
907                                 n_units = MAX(2*c, 16u);
908                                 w = realloc(units, sizeof(struct UnitFileList) * n_units);
909                                 if (!w)
910                                         return log_oom();
911
912                                 units = w;
913                         }
914
915                         u = units + c;
916
917                         dbus_message_iter_recurse(&sub, &sub2);
918
919                         if (bus_iter_get_basic_and_next(&sub2, DBUS_TYPE_STRING, &u->path, true) < 0 ||
920                             bus_iter_get_basic_and_next(&sub2, DBUS_TYPE_STRING, &state, false) < 0) {
921                                 log_error("Failed to parse reply.");
922                                 return -EIO;
923                         }
924
925                         u->state = unit_file_state_from_string(state);
926
927                         dbus_message_iter_next(&sub);
928                         c++;
929                 }
930         }
931
932         if (c > 0) {
933                 qsort(units, c, sizeof(UnitFileList), compare_unit_file_list);
934                 output_unit_file_list(units, c);
935         }
936
937         return 0;
938 }
939
940 static int list_dependencies_print(const char *name, int level, unsigned int branches, bool last) {
941         int i;
942         _cleanup_free_ char *n = NULL;
943         size_t len = 0;
944         size_t max_len = MAX(columns(),20u);
945
946         if (!arg_plain) {
947                 for (i = level - 1; i >= 0; i--) {
948                         len += 2;
949                         if(len > max_len - 3 && !arg_full) {
950                                 printf("%s...\n",max_len % 2 ? "" : " ");
951                                 return 0;
952                         }
953                         printf("%s", draw_special_char(branches & (1 << i) ? DRAW_TREE_VERT : DRAW_TREE_SPACE));
954                 }
955                 len += 2;
956                 if(len > max_len - 3 && !arg_full) {
957                         printf("%s...\n",max_len % 2 ? "" : " ");
958                         return 0;
959                 }
960                 printf("%s", draw_special_char(last ? DRAW_TREE_RIGHT : DRAW_TREE_BRANCH));
961         }
962
963         if(arg_full){
964                 printf("%s\n", name);
965                 return 0;
966         }
967
968         n = ellipsize(name, max_len-len, 100);
969         if(!n)
970                 return log_oom();
971
972         printf("%s\n", n);
973         return 0;
974 }
975
976 static int list_dependencies_get_dependencies(DBusConnection *bus, const char *name, char ***deps) {
977         static const char *dependencies[] = {
978                 [DEPENDENCY_FORWARD] = "Requires\0"
979                                        "RequiresOverridable\0"
980                                        "Requisite\0"
981                                        "RequisiteOverridable\0"
982                                        "Wants\0",
983                 [DEPENDENCY_REVERSE] = "RequiredBy\0"
984                                        "RequiredByOverridable\0"
985                                        "WantedBy\0"
986                                        "PartOf\0",
987                 [DEPENDENCY_AFTER]   = "After\0",
988                 [DEPENDENCY_BEFORE]  = "Before\0",
989         };
990
991         _cleanup_free_ char *path;
992         const char *interface = "org.freedesktop.systemd1.Unit";
993
994         _cleanup_dbus_message_unref_  DBusMessage *reply = NULL;
995         DBusMessageIter iter, sub, sub2, sub3;
996
997         int r = 0;
998         char **ret = NULL;
999
1000         assert(bus);
1001         assert(name);
1002         assert(deps);
1003
1004         path = unit_dbus_path_from_name(name);
1005         if (path == NULL) {
1006                 r = -EINVAL;
1007                 goto finish;
1008         }
1009
1010         r = bus_method_call_with_reply(
1011                 bus,
1012                 "org.freedesktop.systemd1",
1013                 path,
1014                 "org.freedesktop.DBus.Properties",
1015                 "GetAll",
1016                 &reply,
1017                 NULL,
1018                 DBUS_TYPE_STRING, &interface,
1019                 DBUS_TYPE_INVALID);
1020         if (r < 0)
1021                 goto finish;
1022
1023         if (!dbus_message_iter_init(reply, &iter) ||
1024                 dbus_message_iter_get_arg_type(&iter) != DBUS_TYPE_ARRAY ||
1025                 dbus_message_iter_get_element_type(&iter) != DBUS_TYPE_DICT_ENTRY) {
1026                 log_error("Failed to parse reply.");
1027                 r = -EIO;
1028                 goto finish;
1029         }
1030
1031         dbus_message_iter_recurse(&iter, &sub);
1032
1033         while (dbus_message_iter_get_arg_type(&sub) != DBUS_TYPE_INVALID) {
1034                 const char *prop;
1035
1036                 assert(dbus_message_iter_get_arg_type(&sub) == DBUS_TYPE_DICT_ENTRY);
1037                 dbus_message_iter_recurse(&sub, &sub2);
1038
1039                 if (bus_iter_get_basic_and_next(&sub2, DBUS_TYPE_STRING, &prop, true) < 0) {
1040                         log_error("Failed to parse reply.");
1041                         r = -EIO;
1042                         goto finish;
1043                 }
1044
1045                 if (dbus_message_iter_get_arg_type(&sub2) != DBUS_TYPE_VARIANT) {
1046                         log_error("Failed to parse reply.");
1047                         r = -EIO;
1048                         goto finish;
1049                 }
1050
1051                 dbus_message_iter_recurse(&sub2, &sub3);
1052                 dbus_message_iter_next(&sub);
1053
1054                 assert(arg_dependency < ELEMENTSOF(dependencies));
1055                 if (!nulstr_contains(dependencies[arg_dependency], prop))
1056                         continue;
1057
1058                 if (dbus_message_iter_get_arg_type(&sub3) == DBUS_TYPE_ARRAY) {
1059                         if (dbus_message_iter_get_element_type(&sub3) == DBUS_TYPE_STRING) {
1060                                 DBusMessageIter sub4;
1061                                 dbus_message_iter_recurse(&sub3, &sub4);
1062
1063                                 while (dbus_message_iter_get_arg_type(&sub4) != DBUS_TYPE_INVALID) {
1064                                         const char *s;
1065
1066                                         assert(dbus_message_iter_get_arg_type(&sub4) == DBUS_TYPE_STRING);
1067                                         dbus_message_iter_get_basic(&sub4, &s);
1068
1069                                         r = strv_extend(&ret, s);
1070                                         if (r < 0) {
1071                                                 log_oom();
1072                                                 goto finish;
1073                                         }
1074
1075                                         dbus_message_iter_next(&sub4);
1076                                 }
1077                         }
1078                 }
1079         }
1080 finish:
1081         if (r < 0)
1082                 strv_free(ret);
1083         else
1084                 *deps = ret;
1085         return r;
1086 }
1087
1088 static int list_dependencies_compare(const void *_a, const void *_b) {
1089         const char **a = (const char**) _a, **b = (const char**) _b;
1090         if (unit_name_to_type(*a) == UNIT_TARGET && unit_name_to_type(*b) != UNIT_TARGET)
1091                 return 1;
1092         if (unit_name_to_type(*a) != UNIT_TARGET && unit_name_to_type(*b) == UNIT_TARGET)
1093                 return -1;
1094         return strcasecmp(*a, *b);
1095 }
1096
1097 static int list_dependencies_one(DBusConnection *bus, const char *name, int level, char ***units, unsigned int branches) {
1098         _cleanup_strv_free_ char **deps = NULL, **u;
1099         char **c;
1100         int r = 0;
1101
1102         u = strv_append(*units, name);
1103         if (!u)
1104                 return log_oom();
1105
1106         r = list_dependencies_get_dependencies(bus, name, &deps);
1107         if (r < 0)
1108                 return r;
1109
1110         qsort(deps, strv_length(deps), sizeof (char*), list_dependencies_compare);
1111
1112         STRV_FOREACH(c, deps) {
1113                 if (strv_contains(u, *c)) {
1114                         if (!arg_plain) {
1115                                 r = list_dependencies_print("...", level + 1, (branches << 1) | (c[1] == NULL ? 0 : 1), 1);
1116                                 if (r < 0)
1117                                         return r;
1118                         }
1119                         continue;
1120                 }
1121
1122                 r = list_dependencies_print(*c, level, branches, c[1] == NULL);
1123                 if (r < 0)
1124                         return r;
1125
1126                 if (arg_all || unit_name_to_type(*c) == UNIT_TARGET) {
1127                        r = list_dependencies_one(bus, *c, level + 1, &u, (branches << 1) | (c[1] == NULL ? 0 : 1));
1128                        if(r < 0)
1129                                return r;
1130                 }
1131         }
1132         if (arg_plain) {
1133                 strv_free(*units);
1134                 *units = u;
1135                 u = NULL;
1136         }
1137         return 0;
1138 }
1139
1140 static int list_dependencies(DBusConnection *bus, char **args) {
1141         _cleanup_free_ char *unit = NULL;
1142         _cleanup_strv_free_ char **units = NULL;
1143         const char *u;
1144
1145         assert(bus);
1146
1147         if (args[1]) {
1148                 unit = unit_name_mangle(args[1]);
1149                 if (!unit)
1150                         return log_oom();
1151                 u = unit;
1152         } else
1153                 u = SPECIAL_DEFAULT_TARGET;
1154
1155         pager_open_if_enabled();
1156
1157         puts(u);
1158
1159         return list_dependencies_one(bus, u, 0, &units, 0);
1160 }
1161
1162 static int get_default(DBusConnection *bus, char **args) {
1163         char *path = NULL;
1164         _cleanup_dbus_message_unref_ DBusMessage *reply = NULL;
1165         int r;
1166         _cleanup_dbus_error_free_ DBusError error;
1167
1168         dbus_error_init(&error);
1169
1170         if (!bus || avoid_bus()) {
1171                 r = unit_file_get_default(arg_scope, arg_root, &path);
1172
1173                 if (r < 0) {
1174                         log_error("Operation failed: %s", strerror(-r));
1175                         goto finish;
1176                 }
1177
1178                 r = 0;
1179         } else {
1180                 r = bus_method_call_with_reply(
1181                         bus,
1182                         "org.freedesktop.systemd1",
1183                         "/org/freedesktop/systemd1",
1184                         "org.freedesktop.systemd1.Manager",
1185                         "GetDefaultTarget",
1186                         &reply,
1187                         NULL,
1188                         DBUS_TYPE_INVALID);
1189
1190                 if (r < 0) {
1191                         log_error("Operation failed: %s", strerror(-r));
1192                         goto finish;
1193                 }
1194
1195                 if (!dbus_message_get_args(reply, &error,
1196                                    DBUS_TYPE_STRING, &path,
1197                                    DBUS_TYPE_INVALID)) {
1198                         log_error("Failed to parse reply: %s", bus_error_message(&error));
1199                         dbus_error_free(&error);
1200                         return  -EIO;
1201                 }
1202         }
1203
1204         if (path)
1205                 printf("%s\n", path);
1206
1207 finish:
1208         if ((!bus || avoid_bus()) && path)
1209                 free(path);
1210
1211         return r;
1212
1213 }
1214
1215 struct job_info {
1216         uint32_t id;
1217         char *name, *type, *state;
1218 };
1219
1220 static void list_jobs_print(struct job_info* jobs, size_t n) {
1221         size_t i;
1222         struct job_info *j;
1223         const char *on, *off;
1224         bool shorten = false;
1225
1226         assert(n == 0 || jobs);
1227
1228         if (n == 0) {
1229                 on = ansi_highlight_green();
1230                 off = ansi_highlight_off();
1231
1232                 printf("%sNo jobs running.%s\n", on, off);
1233                 return;
1234         }
1235
1236         pager_open_if_enabled();
1237
1238         {
1239                 /* JOB UNIT TYPE STATE */
1240                 unsigned l0 = 3, l1 = 4, l2 = 4, l3 = 5;
1241
1242                 for (i = 0, j = jobs; i < n; i++, j++) {
1243                         assert(j->name && j->type && j->state);
1244                         l0 = MAX(l0, DECIMAL_STR_WIDTH(j->id));
1245                         l1 = MAX(l1, strlen(j->name));
1246                         l2 = MAX(l2, strlen(j->type));
1247                         l3 = MAX(l3, strlen(j->state));
1248                 }
1249
1250                 if (!arg_full && l0 + 1 + l1 + l2 + 1 + l3 > columns()) {
1251                         l1 = MAX(33u, columns() - l0 - l2 - l3 - 3);
1252                         shorten = true;
1253                 }
1254
1255                 if (on_tty())
1256                         printf("%*s %-*s %-*s %-*s\n",
1257                                l0, "JOB",
1258                                l1, "UNIT",
1259                                l2, "TYPE",
1260                                l3, "STATE");
1261
1262                 for (i = 0, j = jobs; i < n; i++, j++) {
1263                         _cleanup_free_ char *e = NULL;
1264
1265                         if (streq(j->state, "running")) {
1266                                 on = ansi_highlight();
1267                                 off = ansi_highlight_off();
1268                         } else
1269                                 on = off = "";
1270
1271                         e = shorten ? ellipsize(j->name, l1, 33) : NULL;
1272                         printf("%*u %s%-*s%s %-*s %s%-*s%s\n",
1273                                l0, j->id,
1274                                on, l1, e ? e : j->name, off,
1275                                l2, j->type,
1276                                on, l3, j->state, off);
1277                 }
1278         }
1279
1280         on = ansi_highlight();
1281         off = ansi_highlight_off();
1282
1283         if (on_tty())
1284                 printf("\n%s%zu jobs listed%s.\n", on, n, off);
1285 }
1286
1287 static int list_jobs(DBusConnection *bus, char **args) {
1288         _cleanup_dbus_message_unref_ DBusMessage *reply = NULL;
1289         DBusMessageIter iter, sub, sub2;
1290         int r;
1291         struct job_info *jobs = NULL;
1292         size_t size = 0, used = 0;
1293
1294         r = bus_method_call_with_reply(
1295                         bus,
1296                         "org.freedesktop.systemd1",
1297                         "/org/freedesktop/systemd1",
1298                         "org.freedesktop.systemd1.Manager",
1299                         "ListJobs",
1300                         &reply,
1301                         NULL,
1302                         DBUS_TYPE_INVALID);
1303         if (r < 0)
1304                 return r;
1305
1306         if (!dbus_message_iter_init(reply, &iter) ||
1307             dbus_message_iter_get_arg_type(&iter) != DBUS_TYPE_ARRAY ||
1308             dbus_message_iter_get_element_type(&iter) != DBUS_TYPE_STRUCT)  {
1309                 log_error("Failed to parse reply.");
1310                 return -EIO;
1311         }
1312
1313         dbus_message_iter_recurse(&iter, &sub);
1314
1315         while (dbus_message_iter_get_arg_type(&sub) != DBUS_TYPE_INVALID) {
1316                 const char *name, *type, *state, *job_path, *unit_path;
1317                 uint32_t id;
1318
1319                 if (dbus_message_iter_get_arg_type(&sub) != DBUS_TYPE_STRUCT) {
1320                         log_error("Failed to parse reply.");
1321                         return -EIO;
1322                 }
1323
1324                 dbus_message_iter_recurse(&sub, &sub2);
1325
1326                 if (bus_iter_get_basic_and_next(&sub2, DBUS_TYPE_UINT32, &id, true) < 0 ||
1327                     bus_iter_get_basic_and_next(&sub2, DBUS_TYPE_STRING, &name, true) < 0 ||
1328                     bus_iter_get_basic_and_next(&sub2, DBUS_TYPE_STRING, &type, true) < 0 ||
1329                     bus_iter_get_basic_and_next(&sub2, DBUS_TYPE_STRING, &state, true) < 0 ||
1330                     bus_iter_get_basic_and_next(&sub2, DBUS_TYPE_OBJECT_PATH, &job_path, true) < 0 ||
1331                     bus_iter_get_basic_and_next(&sub2, DBUS_TYPE_OBJECT_PATH, &unit_path, false) < 0) {
1332                         log_error("Failed to parse reply.");
1333                         r = -EIO;
1334                         goto finish;
1335                 }
1336
1337                 if (!GREEDY_REALLOC(jobs, size, used + 1)) {
1338                         r = log_oom();
1339                         goto finish;
1340                 }
1341
1342                 jobs[used++] = (struct job_info) { id,
1343                                                    strdup(name),
1344                                                    strdup(type),
1345                                                    strdup(state) };
1346                 if (!jobs[used-1].name || !jobs[used-1].type || !jobs[used-1].state) {
1347                         r = log_oom();
1348                         goto finish;
1349                 }
1350
1351                 dbus_message_iter_next(&sub);
1352         }
1353
1354         list_jobs_print(jobs, used);
1355
1356  finish:
1357         while (used--) {
1358                 free(jobs[used].name);
1359                 free(jobs[used].type);
1360                 free(jobs[used].state);
1361         }
1362         free(jobs);
1363
1364         return 0;
1365 }
1366
1367 static int cancel_job(DBusConnection *bus, char **args) {
1368         char **name;
1369
1370         assert(args);
1371
1372         if (strv_length(args) <= 1)
1373                 return daemon_reload(bus, args);
1374
1375         STRV_FOREACH(name, args+1) {
1376                 uint32_t id;
1377                 int r;
1378
1379                 r = safe_atou32(*name, &id);
1380                 if (r < 0) {
1381                         log_error("Failed to parse job id: %s", strerror(-r));
1382                         return r;
1383                 }
1384
1385                 r = bus_method_call_with_reply(
1386                                 bus,
1387                                 "org.freedesktop.systemd1",
1388                                 "/org/freedesktop/systemd1",
1389                                 "org.freedesktop.systemd1.Manager",
1390                                 "CancelJob",
1391                                 NULL,
1392                                 NULL,
1393                                 DBUS_TYPE_UINT32, &id,
1394                                 DBUS_TYPE_INVALID);
1395                 if (r < 0)
1396                         return r;
1397         }
1398
1399         return 0;
1400 }
1401
1402 static int need_daemon_reload(DBusConnection *bus, const char *unit) {
1403         _cleanup_dbus_message_unref_ DBusMessage *reply = NULL;
1404         _cleanup_dbus_error_free_ DBusError error;
1405         dbus_bool_t b = FALSE;
1406         DBusMessageIter iter, sub;
1407         const char
1408                 *interface = "org.freedesktop.systemd1.Unit",
1409                 *property = "NeedDaemonReload",
1410                 *path;
1411         _cleanup_free_ char *n = NULL;
1412         int r;
1413
1414         dbus_error_init(&error);
1415
1416         /* We ignore all errors here, since this is used to show a warning only */
1417
1418         n = unit_name_mangle(unit);
1419         if (!n)
1420                 return log_oom();
1421
1422         r = bus_method_call_with_reply(
1423                         bus,
1424                         "org.freedesktop.systemd1",
1425                         "/org/freedesktop/systemd1",
1426                         "org.freedesktop.systemd1.Manager",
1427                         "GetUnit",
1428                         &reply,
1429                         &error,
1430                         DBUS_TYPE_STRING, &n,
1431                         DBUS_TYPE_INVALID);
1432         if (r < 0)
1433                 return r;
1434
1435         if (!dbus_message_get_args(reply, NULL,
1436                                    DBUS_TYPE_OBJECT_PATH, &path,
1437                                    DBUS_TYPE_INVALID))
1438                 return -EIO;
1439
1440         dbus_message_unref(reply);
1441         reply = NULL;
1442
1443         r = bus_method_call_with_reply(
1444                         bus,
1445                         "org.freedesktop.systemd1",
1446                         path,
1447                         "org.freedesktop.DBus.Properties",
1448                         "Get",
1449                         &reply,
1450                         &error,
1451                         DBUS_TYPE_STRING, &interface,
1452                         DBUS_TYPE_STRING, &property,
1453                         DBUS_TYPE_INVALID);
1454         if (r < 0)
1455                 return r;
1456
1457         if (!dbus_message_iter_init(reply, &iter) ||
1458             dbus_message_iter_get_arg_type(&iter) != DBUS_TYPE_VARIANT)
1459                 return -EIO;
1460
1461         dbus_message_iter_recurse(&iter, &sub);
1462         if (dbus_message_iter_get_arg_type(&sub) != DBUS_TYPE_BOOLEAN)
1463                 return -EIO;
1464
1465         dbus_message_iter_get_basic(&sub, &b);
1466         return b;
1467 }
1468
1469 typedef struct WaitData {
1470         Set *set;
1471
1472         char *name;
1473         char *result;
1474 } WaitData;
1475
1476 static DBusHandlerResult wait_filter(DBusConnection *connection, DBusMessage *message, void *data) {
1477         _cleanup_dbus_error_free_ DBusError error;
1478         WaitData *d = data;
1479
1480         dbus_error_init(&error);
1481
1482         assert(connection);
1483         assert(message);
1484         assert(d);
1485
1486         log_debug("Got D-Bus request: %s.%s() on %s",
1487                   dbus_message_get_interface(message),
1488                   dbus_message_get_member(message),
1489                   dbus_message_get_path(message));
1490
1491         if (dbus_message_is_signal(message, DBUS_INTERFACE_LOCAL, "Disconnected")) {
1492                 log_error("Warning! D-Bus connection terminated.");
1493                 dbus_connection_close(connection);
1494
1495         } else if (dbus_message_is_signal(message, "org.freedesktop.systemd1.Manager", "JobRemoved")) {
1496                 uint32_t id;
1497                 const char *path, *result, *unit;
1498
1499                 if (dbus_message_get_args(message, &error,
1500                                           DBUS_TYPE_UINT32, &id,
1501                                           DBUS_TYPE_OBJECT_PATH, &path,
1502                                           DBUS_TYPE_STRING, &unit,
1503                                           DBUS_TYPE_STRING, &result,
1504                                           DBUS_TYPE_INVALID)) {
1505
1506                         free(set_remove(d->set, (char*) path));
1507
1508                         if (!isempty(result))
1509                                 d->result = strdup(result);
1510
1511                         if (!isempty(unit))
1512                                 d->name = strdup(unit);
1513
1514                         return DBUS_HANDLER_RESULT_NOT_YET_HANDLED;
1515                 }
1516 #ifndef NOLEGACY
1517                 dbus_error_free(&error);
1518                 if (dbus_message_get_args(message, &error,
1519                                           DBUS_TYPE_UINT32, &id,
1520                                           DBUS_TYPE_OBJECT_PATH, &path,
1521                                           DBUS_TYPE_STRING, &result,
1522                                           DBUS_TYPE_INVALID)) {
1523                         /* Compatibility with older systemd versions <
1524                          * 183 during upgrades. This should be dropped
1525                          * one day. */
1526                         free(set_remove(d->set, (char*) path));
1527
1528                         if (*result)
1529                                 d->result = strdup(result);
1530
1531                         return DBUS_HANDLER_RESULT_NOT_YET_HANDLED;
1532                 }
1533 #endif
1534
1535                 log_error("Failed to parse message: %s", bus_error_message(&error));
1536         }
1537
1538         return DBUS_HANDLER_RESULT_NOT_YET_HANDLED;
1539 }
1540
1541 static int enable_wait_for_jobs(DBusConnection *bus) {
1542         DBusError error;
1543
1544         assert(bus);
1545
1546         if (private_bus)
1547                 return 0;
1548
1549         dbus_error_init(&error);
1550         dbus_bus_add_match(bus,
1551                            "type='signal',"
1552                            "sender='org.freedesktop.systemd1',"
1553                            "interface='org.freedesktop.systemd1.Manager',"
1554                            "member='JobRemoved',"
1555                            "path='/org/freedesktop/systemd1'",
1556                            &error);
1557
1558         if (dbus_error_is_set(&error)) {
1559                 log_error("Failed to add match: %s", bus_error_message(&error));
1560                 dbus_error_free(&error);
1561                 return -EIO;
1562         }
1563
1564         /* This is slightly dirty, since we don't undo the match registrations. */
1565         return 0;
1566 }
1567
1568 static int wait_for_jobs(DBusConnection *bus, Set *s) {
1569         int r = 0;
1570         WaitData d = { .set = s };
1571
1572         assert(bus);
1573         assert(s);
1574
1575         if (!dbus_connection_add_filter(bus, wait_filter, &d, NULL))
1576                 return log_oom();
1577
1578         while (!set_isempty(s)) {
1579
1580                 if (!dbus_connection_read_write_dispatch(bus, -1)) {
1581                         log_error("Disconnected from bus.");
1582                         return -ECONNREFUSED;
1583                 }
1584
1585                 if (!d.result)
1586                         goto free_name;
1587
1588                 if (!arg_quiet) {
1589                         if (streq(d.result, "timeout"))
1590                                 log_error("Job for %s timed out.", strna(d.name));
1591                         else if (streq(d.result, "canceled"))
1592                                 log_error("Job for %s canceled.", strna(d.name));
1593                         else if (streq(d.result, "dependency"))
1594                                 log_error("A dependency job for %s failed. See 'journalctl -xn' for details.", strna(d.name));
1595                         else if (!streq(d.result, "done") && !streq(d.result, "skipped"))
1596                                 log_error("Job for %s failed. See 'systemctl status %s' and 'journalctl -xn' for details.", strna(d.name), strna(d.name));
1597                 }
1598
1599                 if (streq_ptr(d.result, "timeout"))
1600                         r = -ETIME;
1601                 else if (streq_ptr(d.result, "canceled"))
1602                         r = -ECANCELED;
1603                 else if (!streq_ptr(d.result, "done") && !streq_ptr(d.result, "skipped"))
1604                         r = -EIO;
1605
1606                 free(d.result);
1607                 d.result = NULL;
1608
1609         free_name:
1610                 free(d.name);
1611                 d.name = NULL;
1612         }
1613
1614         dbus_connection_remove_filter(bus, wait_filter, &d);
1615         return r;
1616 }
1617
1618 static int check_one_unit(DBusConnection *bus, const char *name, char **check_states, bool quiet) {
1619         _cleanup_dbus_message_unref_ DBusMessage *reply = NULL;
1620         _cleanup_free_ char *n = NULL;
1621         DBusMessageIter iter, sub;
1622         const char
1623                 *interface = "org.freedesktop.systemd1.Unit",
1624                 *property = "ActiveState";
1625         const char *state, *path;
1626         DBusError error;
1627         int r;
1628
1629         assert(name);
1630
1631         dbus_error_init(&error);
1632
1633         n = unit_name_mangle(name);
1634         if (!n)
1635                 return log_oom();
1636
1637         r = bus_method_call_with_reply (
1638                         bus,
1639                         "org.freedesktop.systemd1",
1640                         "/org/freedesktop/systemd1",
1641                         "org.freedesktop.systemd1.Manager",
1642                         "GetUnit",
1643                         &reply,
1644                         &error,
1645                         DBUS_TYPE_STRING, &n,
1646                         DBUS_TYPE_INVALID);
1647         if (r < 0) {
1648                 dbus_error_free(&error);
1649
1650                 if (!quiet)
1651                         puts("unknown");
1652                 return 0;
1653         }
1654
1655         if (!dbus_message_get_args(reply, NULL,
1656                                    DBUS_TYPE_OBJECT_PATH, &path,
1657                                    DBUS_TYPE_INVALID)) {
1658                 log_error("Failed to parse reply.");
1659                 return -EIO;
1660         }
1661
1662         dbus_message_unref(reply);
1663         reply = NULL;
1664
1665         r = bus_method_call_with_reply(
1666                         bus,
1667                         "org.freedesktop.systemd1",
1668                         path,
1669                         "org.freedesktop.DBus.Properties",
1670                         "Get",
1671                         &reply,
1672                         NULL,
1673                         DBUS_TYPE_STRING, &interface,
1674                         DBUS_TYPE_STRING, &property,
1675                         DBUS_TYPE_INVALID);
1676         if (r < 0) {
1677                 if (!quiet)
1678                         puts("unknown");
1679                 return 0;
1680         }
1681
1682         if (!dbus_message_iter_init(reply, &iter) ||
1683             dbus_message_iter_get_arg_type(&iter) != DBUS_TYPE_VARIANT)  {
1684                 log_error("Failed to parse reply.");
1685                 return r;
1686         }
1687
1688         dbus_message_iter_recurse(&iter, &sub);
1689
1690         if (dbus_message_iter_get_arg_type(&sub) != DBUS_TYPE_STRING)  {
1691                 log_error("Failed to parse reply.");
1692                 return r;
1693         }
1694
1695         dbus_message_iter_get_basic(&sub, &state);
1696
1697         if (!quiet)
1698                 puts(state);
1699
1700         return strv_find(check_states, state) ? 1 : 0;
1701 }
1702
1703 static void check_triggering_units(
1704                 DBusConnection *bus,
1705                 const char *unit_name) {
1706
1707         _cleanup_dbus_message_unref_ DBusMessage *reply = NULL;
1708         DBusMessageIter iter, sub;
1709         const char *interface = "org.freedesktop.systemd1.Unit",
1710                    *load_state_property = "LoadState",
1711                    *triggered_by_property = "TriggeredBy",
1712                    *state;
1713         _cleanup_free_ char *unit_path = NULL, *n = NULL;
1714         bool print_warning_label = true;
1715         int r;
1716
1717         n = unit_name_mangle(unit_name);
1718         if (!n) {
1719                 log_oom();
1720                 return;
1721         }
1722
1723         unit_path = unit_dbus_path_from_name(n);
1724         if (!unit_path) {
1725                 log_oom();
1726                 return;
1727         }
1728
1729         r = bus_method_call_with_reply(
1730                         bus,
1731                         "org.freedesktop.systemd1",
1732                         unit_path,
1733                         "org.freedesktop.DBus.Properties",
1734                         "Get",
1735                         &reply,
1736                         NULL,
1737                         DBUS_TYPE_STRING, &interface,
1738                         DBUS_TYPE_STRING, &load_state_property,
1739                         DBUS_TYPE_INVALID);
1740         if (r < 0)
1741                 return;
1742
1743         if (!dbus_message_iter_init(reply, &iter) ||
1744             dbus_message_iter_get_arg_type(&iter) != DBUS_TYPE_VARIANT) {
1745                 log_error("Failed to parse reply.");
1746                 return;
1747         }
1748
1749         dbus_message_iter_recurse(&iter, &sub);
1750
1751         if (dbus_message_iter_get_arg_type(&sub) != DBUS_TYPE_STRING)  {
1752             log_error("Failed to parse reply.");
1753             return;
1754         }
1755
1756         dbus_message_iter_get_basic(&sub, &state);
1757
1758         if (streq(state, "masked"))
1759             return;
1760
1761         dbus_message_unref(reply);
1762         reply = NULL;
1763
1764         r = bus_method_call_with_reply(
1765                         bus,
1766                         "org.freedesktop.systemd1",
1767                         unit_path,
1768                         "org.freedesktop.DBus.Properties",
1769                         "Get",
1770                         &reply,
1771                         NULL,
1772                         DBUS_TYPE_STRING, &interface,
1773                         DBUS_TYPE_STRING, &triggered_by_property,
1774                         DBUS_TYPE_INVALID);
1775         if (r < 0)
1776                 return;
1777
1778         if (!dbus_message_iter_init(reply, &iter) ||
1779             dbus_message_iter_get_arg_type(&iter) != DBUS_TYPE_VARIANT) {
1780                 log_error("Failed to parse reply.");
1781                 return;
1782         }
1783
1784         dbus_message_iter_recurse(&iter, &sub);
1785         dbus_message_iter_recurse(&sub, &iter);
1786         sub = iter;
1787
1788         while (dbus_message_iter_get_arg_type(&sub) != DBUS_TYPE_INVALID) {
1789                 const char * const check_states[] = {
1790                         "active",
1791                         "reloading",
1792                         NULL
1793                 };
1794                 const char *service_trigger;
1795
1796                 if (dbus_message_iter_get_arg_type(&sub) != DBUS_TYPE_STRING) {
1797                         log_error("Failed to parse reply.");
1798                         return;
1799                 }
1800
1801                 dbus_message_iter_get_basic(&sub, &service_trigger);
1802
1803                 r = check_one_unit(bus, service_trigger, (char**) check_states, true);
1804                 if (r < 0)
1805                         return;
1806                 if (r > 0) {
1807                         if (print_warning_label) {
1808                                 log_warning("Warning: Stopping %s, but it can still be activated by:", unit_name);
1809                                 print_warning_label = false;
1810                         }
1811
1812                         log_warning("  %s", service_trigger);
1813                 }
1814
1815                 dbus_message_iter_next(&sub);
1816         }
1817 }
1818
1819 static int start_unit_one(
1820                 DBusConnection *bus,
1821                 const char *method,
1822                 const char *name,
1823                 const char *mode,
1824                 DBusError *error,
1825                 Set *s) {
1826
1827         _cleanup_dbus_message_unref_ DBusMessage *reply = NULL;
1828         _cleanup_free_ char *n;
1829         const char *path;
1830         int r;
1831
1832         assert(method);
1833         assert(name);
1834         assert(mode);
1835         assert(error);
1836
1837         n = unit_name_mangle(name);
1838         if (!n)
1839                 return log_oom();
1840
1841         r = bus_method_call_with_reply(
1842                         bus,
1843                         "org.freedesktop.systemd1",
1844                         "/org/freedesktop/systemd1",
1845                         "org.freedesktop.systemd1.Manager",
1846                         method,
1847                         &reply,
1848                         error,
1849                         DBUS_TYPE_STRING, &n,
1850                         DBUS_TYPE_STRING, &mode,
1851                         DBUS_TYPE_INVALID);
1852         if (r) {
1853                 if (r == -ENOENT && arg_action != ACTION_SYSTEMCTL)
1854                         /* There's always a fallback possible for
1855                          * legacy actions. */
1856                         r = -EADDRNOTAVAIL;
1857                 else
1858                         log_error("Failed to issue method call: %s", bus_error_message(error));
1859
1860                 return r;
1861         }
1862
1863         if (!dbus_message_get_args(reply, error,
1864                                    DBUS_TYPE_OBJECT_PATH, &path,
1865                                    DBUS_TYPE_INVALID)) {
1866                 log_error("Failed to parse reply: %s", bus_error_message(error));
1867                 return -EIO;
1868         }
1869
1870         if (need_daemon_reload(bus, n) > 0)
1871                 log_warning("Warning: Unit file of %s changed on disk, 'systemctl %sdaemon-reload' recommended.",
1872                             n, arg_scope == UNIT_FILE_SYSTEM ? "" : "--user ");
1873
1874         if (s) {
1875                 char *p;
1876
1877                 p = strdup(path);
1878                 if (!p)
1879                         return log_oom();
1880
1881                 r = set_consume(s, p);
1882                 if (r < 0) {
1883                         log_error("Failed to add path to set.");
1884                         return r;
1885                 }
1886         }
1887
1888         return 0;
1889 }
1890
1891 static const struct {
1892         const char *target;
1893         const char *verb;
1894         const char *mode;
1895 } action_table[_ACTION_MAX] = {
1896         [ACTION_HALT]         = { SPECIAL_HALT_TARGET,         "halt",         "replace-irreversibly" },
1897         [ACTION_POWEROFF]     = { SPECIAL_POWEROFF_TARGET,     "poweroff",     "replace-irreversibly" },
1898         [ACTION_REBOOT]       = { SPECIAL_REBOOT_TARGET,       "reboot",       "replace-irreversibly" },
1899         [ACTION_KEXEC]        = { SPECIAL_KEXEC_TARGET,        "kexec",        "replace-irreversibly" },
1900         [ACTION_RUNLEVEL2]    = { SPECIAL_RUNLEVEL2_TARGET,    NULL,           "isolate" },
1901         [ACTION_RUNLEVEL3]    = { SPECIAL_RUNLEVEL3_TARGET,    NULL,           "isolate" },
1902         [ACTION_RUNLEVEL4]    = { SPECIAL_RUNLEVEL4_TARGET,    NULL,           "isolate" },
1903         [ACTION_RUNLEVEL5]    = { SPECIAL_RUNLEVEL5_TARGET,    NULL,           "isolate" },
1904         [ACTION_RESCUE]       = { SPECIAL_RESCUE_TARGET,       "rescue",       "isolate" },
1905         [ACTION_EMERGENCY]    = { SPECIAL_EMERGENCY_TARGET,    "emergency",    "isolate" },
1906         [ACTION_DEFAULT]      = { SPECIAL_DEFAULT_TARGET,      "default",      "isolate" },
1907         [ACTION_EXIT]         = { SPECIAL_EXIT_TARGET,         "exit",         "replace-irreversibly" },
1908         [ACTION_SUSPEND]      = { SPECIAL_SUSPEND_TARGET,      "suspend",      "replace-irreversibly" },
1909         [ACTION_HIBERNATE]    = { SPECIAL_HIBERNATE_TARGET,    "hibernate",    "replace-irreversibly" },
1910         [ACTION_HYBRID_SLEEP] = { SPECIAL_HYBRID_SLEEP_TARGET, "hybrid-sleep", "replace-irreversibly" },
1911 };
1912
1913 static enum action verb_to_action(const char *verb) {
1914         enum action i;
1915
1916         for (i = ACTION_INVALID; i < _ACTION_MAX; i++)
1917                 if (action_table[i].verb && streq(verb, action_table[i].verb))
1918                         return i;
1919         return ACTION_INVALID;
1920 }
1921
1922 static int start_unit(DBusConnection *bus, char **args) {
1923
1924         int r, ret = 0;
1925         const char *method, *mode, *one_name;
1926         _cleanup_set_free_free_ Set *s = NULL;
1927         _cleanup_dbus_error_free_ DBusError error;
1928         char **name;
1929
1930         dbus_error_init(&error);
1931
1932         assert(bus);
1933
1934         ask_password_agent_open_if_enabled();
1935
1936         if (arg_action == ACTION_SYSTEMCTL) {
1937                 enum action action;
1938                 method =
1939                         streq(args[0], "stop") ||
1940                         streq(args[0], "condstop")              ? "StopUnit" :
1941                         streq(args[0], "reload")                ? "ReloadUnit" :
1942                         streq(args[0], "restart")               ? "RestartUnit" :
1943
1944                         streq(args[0], "try-restart")           ||
1945                         streq(args[0], "condrestart")           ? "TryRestartUnit" :
1946
1947                         streq(args[0], "reload-or-restart")     ? "ReloadOrRestartUnit" :
1948
1949                         streq(args[0], "reload-or-try-restart") ||
1950                         streq(args[0], "condreload") ||
1951
1952                         streq(args[0], "force-reload")          ? "ReloadOrTryRestartUnit" :
1953                                                                   "StartUnit";
1954                 action = verb_to_action(args[0]);
1955
1956                 mode = streq(args[0], "isolate") ? "isolate" :
1957                        action_table[action].mode ?: arg_job_mode;
1958
1959                 one_name = action_table[action].target;
1960
1961         } else {
1962                 assert(arg_action < ELEMENTSOF(action_table));
1963                 assert(action_table[arg_action].target);
1964
1965                 method = "StartUnit";
1966
1967                 mode = action_table[arg_action].mode;
1968                 one_name = action_table[arg_action].target;
1969         }
1970
1971         if (!arg_no_block) {
1972                 ret = enable_wait_for_jobs(bus);
1973                 if (ret < 0) {
1974                         log_error("Could not watch jobs: %s", strerror(-ret));
1975                         return ret;
1976                 }
1977
1978                 s = set_new(string_hash_func, string_compare_func);
1979                 if (!s)
1980                         return log_oom();
1981         }
1982
1983         if (one_name) {
1984                 ret = start_unit_one(bus, method, one_name, mode, &error, s);
1985                 if (ret < 0)
1986                         ret = translate_bus_error_to_exit_status(ret, &error);
1987         } else {
1988                 STRV_FOREACH(name, args+1) {
1989                         r = start_unit_one(bus, method, *name, mode, &error, s);
1990                         if (r < 0) {
1991                                 ret = translate_bus_error_to_exit_status(r, &error);
1992                                 dbus_error_free(&error);
1993                         }
1994                 }
1995         }
1996
1997         if (!arg_no_block) {
1998                 r = wait_for_jobs(bus, s);
1999                 if (r < 0)
2000                         return r;
2001
2002                 /* When stopping units, warn if they can still be triggered by
2003                  * another active unit (socket, path, timer) */
2004                 if (!arg_quiet && streq(method, "StopUnit")) {
2005                         if (one_name)
2006                                 check_triggering_units(bus, one_name);
2007                         else
2008                                 STRV_FOREACH(name, args+1)
2009                                         check_triggering_units(bus, *name);
2010                 }
2011         }
2012
2013         return ret;
2014 }
2015
2016 /* Ask systemd-logind, which might grant access to unprivileged users
2017  * through PolicyKit */
2018 static int reboot_with_logind(DBusConnection *bus, enum action a) {
2019 #ifdef HAVE_LOGIND
2020         const char *method;
2021         dbus_bool_t interactive = true;
2022
2023         if (!bus)
2024                 return -EIO;
2025
2026         polkit_agent_open_if_enabled();
2027
2028         switch (a) {
2029
2030         case ACTION_REBOOT:
2031                 method = "Reboot";
2032                 break;
2033
2034         case ACTION_POWEROFF:
2035                 method = "PowerOff";
2036                 break;
2037
2038         case ACTION_SUSPEND:
2039                 method = "Suspend";
2040                 break;
2041
2042         case ACTION_HIBERNATE:
2043                 method = "Hibernate";
2044                 break;
2045
2046         case ACTION_HYBRID_SLEEP:
2047                 method = "HybridSleep";
2048                 break;
2049
2050         default:
2051                 return -EINVAL;
2052         }
2053
2054         return bus_method_call_with_reply(
2055                         bus,
2056                         "org.freedesktop.login1",
2057                         "/org/freedesktop/login1",
2058                         "org.freedesktop.login1.Manager",
2059                         method,
2060                         NULL,
2061                         NULL,
2062                         DBUS_TYPE_BOOLEAN, &interactive,
2063                         DBUS_TYPE_INVALID);
2064 #else
2065         return -ENOSYS;
2066 #endif
2067 }
2068
2069 static int check_inhibitors(DBusConnection *bus, enum action a) {
2070 #ifdef HAVE_LOGIND
2071         _cleanup_dbus_message_unref_ DBusMessage *reply = NULL;
2072         DBusMessageIter iter, sub, sub2;
2073         int r;
2074         unsigned c = 0;
2075         _cleanup_strv_free_ char **sessions = NULL;
2076         char **s;
2077
2078         if (!bus)
2079                 return 0;
2080
2081         if (arg_ignore_inhibitors || arg_force > 0)
2082                 return 0;
2083
2084         if (arg_when > 0)
2085                 return 0;
2086
2087         if (geteuid() == 0)
2088                 return 0;
2089
2090         if (!on_tty())
2091                 return 0;
2092
2093         r = bus_method_call_with_reply(
2094                         bus,
2095                         "org.freedesktop.login1",
2096                         "/org/freedesktop/login1",
2097                         "org.freedesktop.login1.Manager",
2098                         "ListInhibitors",
2099                         &reply,
2100                         NULL,
2101                         DBUS_TYPE_INVALID);
2102         if (r < 0)
2103                 /* If logind is not around, then there are no inhibitors... */
2104                 return 0;
2105
2106         if (!dbus_message_iter_init(reply, &iter) ||
2107             dbus_message_iter_get_arg_type(&iter) != DBUS_TYPE_ARRAY ||
2108             dbus_message_iter_get_element_type(&iter) != DBUS_TYPE_STRUCT) {
2109                 log_error("Failed to parse reply.");
2110                 return -EIO;
2111         }
2112
2113         dbus_message_iter_recurse(&iter, &sub);
2114         while (dbus_message_iter_get_arg_type(&sub) != DBUS_TYPE_INVALID) {
2115                 const char *what, *who, *why, *mode;
2116                 uint32_t uid, pid;
2117                 _cleanup_strv_free_ char **sv = NULL;
2118                 _cleanup_free_ char *comm = NULL, *user = NULL;
2119
2120                 if (dbus_message_iter_get_arg_type(&sub) != DBUS_TYPE_STRUCT) {
2121                         log_error("Failed to parse reply.");
2122                         return -EIO;
2123                 }
2124
2125                 dbus_message_iter_recurse(&sub, &sub2);
2126
2127                 if (bus_iter_get_basic_and_next(&sub2, DBUS_TYPE_STRING, &what, true) < 0 ||
2128                     bus_iter_get_basic_and_next(&sub2, DBUS_TYPE_STRING, &who, true) < 0 ||
2129                     bus_iter_get_basic_and_next(&sub2, DBUS_TYPE_STRING, &why, true) < 0 ||
2130                     bus_iter_get_basic_and_next(&sub2, DBUS_TYPE_STRING, &mode, true) < 0 ||
2131                     bus_iter_get_basic_and_next(&sub2, DBUS_TYPE_UINT32, &uid, true) < 0 ||
2132                     bus_iter_get_basic_and_next(&sub2, DBUS_TYPE_UINT32, &pid, false) < 0) {
2133                         log_error("Failed to parse reply.");
2134                         return -EIO;
2135                 }
2136
2137                 if (!streq(mode, "block"))
2138                         goto next;
2139
2140                 sv = strv_split(what, ":");
2141                 if (!sv)
2142                         return log_oom();
2143
2144                 if (!strv_contains(sv,
2145                                   a == ACTION_HALT ||
2146                                   a == ACTION_POWEROFF ||
2147                                   a == ACTION_REBOOT ||
2148                                   a == ACTION_KEXEC ? "shutdown" : "sleep"))
2149                         goto next;
2150
2151                 get_process_comm(pid, &comm);
2152                 user = uid_to_name(uid);
2153                 log_warning("Operation inhibited by \"%s\" (PID %lu \"%s\", user %s), reason is \"%s\".",
2154                             who, (unsigned long) pid, strna(comm), strna(user), why);
2155                 c++;
2156
2157         next:
2158                 dbus_message_iter_next(&sub);
2159         }
2160
2161         dbus_message_iter_recurse(&iter, &sub);
2162
2163         /* Check for current sessions */
2164         sd_get_sessions(&sessions);
2165         STRV_FOREACH(s, sessions) {
2166                 uid_t uid;
2167                 _cleanup_free_ char *type = NULL, *tty = NULL, *seat = NULL, *user = NULL, *service = NULL, *class = NULL;
2168
2169                 if (sd_session_get_uid(*s, &uid) < 0 || uid == getuid())
2170                         continue;
2171
2172                 if (sd_session_get_class(*s, &class) < 0 || !streq(class, "user"))
2173                         continue;
2174
2175                 if (sd_session_get_type(*s, &type) < 0 || (!streq(type, "x11") && !streq(type, "tty")))
2176                         continue;
2177
2178                 sd_session_get_tty(*s, &tty);
2179                 sd_session_get_seat(*s, &seat);
2180                 sd_session_get_service(*s, &service);
2181                 user = uid_to_name(uid);
2182
2183                 log_warning("User %s is logged in on %s.", strna(user), isempty(tty) ? (isempty(seat) ? strna(service) : seat) : tty);
2184                 c++;
2185         }
2186
2187         if (c <= 0)
2188                 return 0;
2189
2190         log_error("Please retry operation after closing inhibitors and logging out other users.\nAlternatively, ignore inhibitors and users with 'systemctl %s -i'.",
2191                   action_table[a].verb);
2192
2193         return -EPERM;
2194 #else
2195         return 0;
2196 #endif
2197 }
2198
2199 static int start_special(DBusConnection *bus, char **args) {
2200         enum action a;
2201         int r;
2202
2203         assert(args);
2204
2205         a = verb_to_action(args[0]);
2206
2207         r = check_inhibitors(bus, a);
2208         if (r < 0)
2209                 return r;
2210
2211         if (arg_force >= 2 && geteuid() != 0) {
2212                 log_error("Must be root.");
2213                 return -EPERM;
2214         }
2215
2216         if (arg_force >= 2 &&
2217             (a == ACTION_HALT ||
2218              a == ACTION_POWEROFF ||
2219              a == ACTION_REBOOT))
2220                 halt_now(a);
2221
2222         if (arg_force >= 1 &&
2223             (a == ACTION_HALT ||
2224              a == ACTION_POWEROFF ||
2225              a == ACTION_REBOOT ||
2226              a == ACTION_KEXEC ||
2227              a == ACTION_EXIT))
2228                 return daemon_reload(bus, args);
2229
2230         /* first try logind, to allow authentication with polkit */
2231         if (geteuid() != 0 &&
2232             (a == ACTION_POWEROFF ||
2233              a == ACTION_REBOOT ||
2234              a == ACTION_SUSPEND ||
2235              a == ACTION_HIBERNATE ||
2236              a == ACTION_HYBRID_SLEEP)) {
2237                 r = reboot_with_logind(bus, a);
2238                 if (r >= 0)
2239                         return r;
2240         }
2241
2242         r = start_unit(bus, args);
2243         if (r == EXIT_SUCCESS)
2244                 warn_wall(a);
2245
2246         return r;
2247 }
2248
2249 static int check_unit_active(DBusConnection *bus, char **args) {
2250         const char * const check_states[] = {
2251                 "active",
2252                 "reloading",
2253                 NULL
2254         };
2255
2256         char **name;
2257         int r = 3; /* According to LSB: "program is not running" */
2258
2259         assert(bus);
2260         assert(args);
2261
2262         STRV_FOREACH(name, args+1) {
2263                 int state;
2264
2265                 state = check_one_unit(bus, *name, (char**) check_states, arg_quiet);
2266                 if (state < 0)
2267                         return state;
2268                 if (state > 0)
2269                         r = 0;
2270         }
2271
2272         return r;
2273 }
2274
2275 static int check_unit_failed(DBusConnection *bus, char **args) {
2276         const char * const check_states[] = {
2277                 "failed",
2278                 NULL
2279         };
2280
2281         char **name;
2282         int r = 1;
2283
2284         assert(bus);
2285         assert(args);
2286
2287         STRV_FOREACH(name, args+1) {
2288                 int state;
2289
2290                 state = check_one_unit(bus, *name, (char**) check_states, arg_quiet);
2291                 if (state < 0)
2292                         return state;
2293                 if (state > 0)
2294                         r = 0;
2295         }
2296
2297         return r;
2298 }
2299
2300 static int kill_unit(DBusConnection *bus, char **args) {
2301         char **name;
2302         int r = 0;
2303
2304         assert(bus);
2305         assert(args);
2306
2307         if (!arg_kill_who)
2308                 arg_kill_who = "all";
2309
2310         STRV_FOREACH(name, args+1) {
2311                 _cleanup_free_ char *n = NULL;
2312
2313                 n = unit_name_mangle(*name);
2314                 if (!n)
2315                         return log_oom();
2316
2317                 r = bus_method_call_with_reply(
2318                                 bus,
2319                                 "org.freedesktop.systemd1",
2320                                 "/org/freedesktop/systemd1",
2321                                 "org.freedesktop.systemd1.Manager",
2322                                 "KillUnit",
2323                                 NULL,
2324                                 NULL,
2325                                 DBUS_TYPE_STRING, &n,
2326                                 DBUS_TYPE_STRING, &arg_kill_who,
2327                                 DBUS_TYPE_INT32, &arg_signal,
2328                                 DBUS_TYPE_INVALID);
2329                 if (r < 0)
2330                         return r;
2331         }
2332         return 0;
2333 }
2334
2335 typedef struct ExecStatusInfo {
2336         char *name;
2337
2338         char *path;
2339         char **argv;
2340
2341         bool ignore;
2342
2343         usec_t start_timestamp;
2344         usec_t exit_timestamp;
2345         pid_t pid;
2346         int code;
2347         int status;
2348
2349         LIST_FIELDS(struct ExecStatusInfo, exec);
2350 } ExecStatusInfo;
2351
2352 static void exec_status_info_free(ExecStatusInfo *i) {
2353         assert(i);
2354
2355         free(i->name);
2356         free(i->path);
2357         strv_free(i->argv);
2358         free(i);
2359 }
2360
2361 static int exec_status_info_deserialize(DBusMessageIter *sub, ExecStatusInfo *i) {
2362         uint64_t start_timestamp, exit_timestamp, start_timestamp_monotonic, exit_timestamp_monotonic;
2363         DBusMessageIter sub2, sub3;
2364         const char*path;
2365         unsigned n;
2366         uint32_t pid;
2367         int32_t code, status;
2368         dbus_bool_t ignore;
2369
2370         assert(i);
2371         assert(i);
2372
2373         if (dbus_message_iter_get_arg_type(sub) != DBUS_TYPE_STRUCT)
2374                 return -EIO;
2375
2376         dbus_message_iter_recurse(sub, &sub2);
2377
2378         if (bus_iter_get_basic_and_next(&sub2, DBUS_TYPE_STRING, &path, true) < 0)
2379                 return -EIO;
2380
2381         i->path = strdup(path);
2382         if (!i->path)
2383                 return -ENOMEM;
2384
2385         if (dbus_message_iter_get_arg_type(&sub2) != DBUS_TYPE_ARRAY ||
2386             dbus_message_iter_get_element_type(&sub2) != DBUS_TYPE_STRING)
2387                 return -EIO;
2388
2389         n = 0;
2390         dbus_message_iter_recurse(&sub2, &sub3);
2391         while (dbus_message_iter_get_arg_type(&sub3) != DBUS_TYPE_INVALID) {
2392                 assert(dbus_message_iter_get_arg_type(&sub3) == DBUS_TYPE_STRING);
2393                 dbus_message_iter_next(&sub3);
2394                 n++;
2395         }
2396
2397         i->argv = new0(char*, n+1);
2398         if (!i->argv)
2399                 return -ENOMEM;
2400
2401         n = 0;
2402         dbus_message_iter_recurse(&sub2, &sub3);
2403         while (dbus_message_iter_get_arg_type(&sub3) != DBUS_TYPE_INVALID) {
2404                 const char *s;
2405
2406                 assert(dbus_message_iter_get_arg_type(&sub3) == DBUS_TYPE_STRING);
2407                 dbus_message_iter_get_basic(&sub3, &s);
2408                 dbus_message_iter_next(&sub3);
2409
2410                 i->argv[n] = strdup(s);
2411                 if (!i->argv[n])
2412                         return -ENOMEM;
2413
2414                 n++;
2415         }
2416
2417         if (!dbus_message_iter_next(&sub2) ||
2418             bus_iter_get_basic_and_next(&sub2, DBUS_TYPE_BOOLEAN, &ignore, true) < 0 ||
2419             bus_iter_get_basic_and_next(&sub2, DBUS_TYPE_UINT64, &start_timestamp, true) < 0 ||
2420             bus_iter_get_basic_and_next(&sub2, DBUS_TYPE_UINT64, &start_timestamp_monotonic, true) < 0 ||
2421             bus_iter_get_basic_and_next(&sub2, DBUS_TYPE_UINT64, &exit_timestamp, true) < 0 ||
2422             bus_iter_get_basic_and_next(&sub2, DBUS_TYPE_UINT64, &exit_timestamp_monotonic, true) < 0 ||
2423             bus_iter_get_basic_and_next(&sub2, DBUS_TYPE_UINT32, &pid, true) < 0 ||
2424             bus_iter_get_basic_and_next(&sub2, DBUS_TYPE_INT32, &code, true) < 0 ||
2425             bus_iter_get_basic_and_next(&sub2, DBUS_TYPE_INT32, &status, false) < 0)
2426                 return -EIO;
2427
2428         i->ignore = ignore;
2429         i->start_timestamp = (usec_t) start_timestamp;
2430         i->exit_timestamp = (usec_t) exit_timestamp;
2431         i->pid = (pid_t) pid;
2432         i->code = code;
2433         i->status = status;
2434
2435         return 0;
2436 }
2437
2438 typedef struct UnitStatusInfo {
2439         const char *id;
2440         const char *load_state;
2441         const char *active_state;
2442         const char *sub_state;
2443         const char *unit_file_state;
2444
2445         const char *description;
2446         const char *following;
2447
2448         char **documentation;
2449
2450         const char *fragment_path;
2451         const char *source_path;
2452         const char *control_group;
2453
2454         char **dropin_paths;
2455
2456         const char *load_error;
2457         const char *result;
2458
2459         usec_t inactive_exit_timestamp;
2460         usec_t inactive_exit_timestamp_monotonic;
2461         usec_t active_enter_timestamp;
2462         usec_t active_exit_timestamp;
2463         usec_t inactive_enter_timestamp;
2464
2465         bool need_daemon_reload;
2466
2467         /* Service */
2468         pid_t main_pid;
2469         pid_t control_pid;
2470         const char *status_text;
2471         const char *pid_file;
2472         bool running:1;
2473
2474         usec_t start_timestamp;
2475         usec_t exit_timestamp;
2476
2477         int exit_code, exit_status;
2478
2479         usec_t condition_timestamp;
2480         bool condition_result;
2481         bool failed_condition_trigger;
2482         bool failed_condition_negate;
2483         const char *failed_condition;
2484         const char *failed_condition_param;
2485
2486         /* Socket */
2487         unsigned n_accepted;
2488         unsigned n_connections;
2489         bool accept;
2490
2491         /* Pairs of type, path */
2492         char **listen;
2493
2494         /* Device */
2495         const char *sysfs_path;
2496
2497         /* Mount, Automount */
2498         const char *where;
2499
2500         /* Swap */
2501         const char *what;
2502
2503         LIST_HEAD(ExecStatusInfo, exec);
2504 } UnitStatusInfo;
2505
2506 static void print_status_info(UnitStatusInfo *i,
2507                               bool *ellipsized) {
2508         ExecStatusInfo *p;
2509         const char *on, *off, *ss;
2510         usec_t timestamp;
2511         char since1[FORMAT_TIMESTAMP_RELATIVE_MAX], *s1;
2512         char since2[FORMAT_TIMESTAMP_MAX], *s2;
2513         const char *path;
2514         int flags =
2515                 arg_all * OUTPUT_SHOW_ALL |
2516                 (!on_tty() || pager_have()) * OUTPUT_FULL_WIDTH |
2517                 on_tty() * OUTPUT_COLOR |
2518                 !arg_quiet * OUTPUT_WARN_CUTOFF |
2519                 arg_full * OUTPUT_FULL_WIDTH;
2520         char **t, **t2;
2521
2522         assert(i);
2523
2524         /* This shows pretty information about a unit. See
2525          * print_property() for a low-level property printer */
2526
2527         printf("%s", strna(i->id));
2528
2529         if (i->description && !streq_ptr(i->id, i->description))
2530                 printf(" - %s", i->description);
2531
2532         printf("\n");
2533
2534         if (i->following)
2535                 printf("   Follow: unit currently follows state of %s\n", i->following);
2536
2537         if (streq_ptr(i->load_state, "error")) {
2538                 on = ansi_highlight_red();
2539                 off = ansi_highlight_off();
2540         } else
2541                 on = off = "";
2542
2543         path = i->source_path ? i->source_path : i->fragment_path;
2544
2545         if (i->load_error)
2546                 printf("   Loaded: %s%s%s (Reason: %s)\n",
2547                        on, strna(i->load_state), off, i->load_error);
2548         else if (path && i->unit_file_state)
2549                 printf("   Loaded: %s%s%s (%s; %s)\n",
2550                        on, strna(i->load_state), off, path, i->unit_file_state);
2551         else if (path)
2552                 printf("   Loaded: %s%s%s (%s)\n",
2553                        on, strna(i->load_state), off, path);
2554         else
2555                 printf("   Loaded: %s%s%s\n",
2556                        on, strna(i->load_state), off);
2557
2558         if (!strv_isempty(i->dropin_paths)) {
2559                 char ** dropin;
2560                 char * dir = NULL;
2561                 bool last = false;
2562
2563                 STRV_FOREACH(dropin, i->dropin_paths) {
2564                         if (! dir || last) {
2565                                 printf(dir ? "        " : "  Drop-In: ");
2566
2567                                 free(dir);
2568
2569                                 if (path_get_parent(*dropin, &dir) < 0) {
2570                                         log_oom();
2571                                         return;
2572                                 }
2573
2574                                 printf("%s\n           %s", dir,
2575                                        draw_special_char(DRAW_TREE_RIGHT));
2576                         }
2577
2578                         last = ! (*(dropin + 1) && startswith(*(dropin + 1), dir));
2579
2580                         printf("%s%s", path_get_file_name(*dropin), last ? "\n" : ", ");
2581                 }
2582
2583                 free(dir);
2584         }
2585
2586         ss = streq_ptr(i->active_state, i->sub_state) ? NULL : i->sub_state;
2587
2588         if (streq_ptr(i->active_state, "failed")) {
2589                 on = ansi_highlight_red();
2590                 off = ansi_highlight_off();
2591         } else if (streq_ptr(i->active_state, "active") || streq_ptr(i->active_state, "reloading")) {
2592                 on = ansi_highlight_green();
2593                 off = ansi_highlight_off();
2594         } else
2595                 on = off = "";
2596
2597         if (ss)
2598                 printf("   Active: %s%s (%s)%s",
2599                        on, strna(i->active_state), ss, off);
2600         else
2601                 printf("   Active: %s%s%s",
2602                        on, strna(i->active_state), off);
2603
2604         if (!isempty(i->result) && !streq(i->result, "success"))
2605                 printf(" (Result: %s)", i->result);
2606
2607         timestamp = (streq_ptr(i->active_state, "active")      ||
2608                      streq_ptr(i->active_state, "reloading"))   ? i->active_enter_timestamp :
2609                     (streq_ptr(i->active_state, "inactive")    ||
2610                      streq_ptr(i->active_state, "failed"))      ? i->inactive_enter_timestamp :
2611                     streq_ptr(i->active_state, "activating")    ? i->inactive_exit_timestamp :
2612                                                                   i->active_exit_timestamp;
2613
2614         s1 = format_timestamp_relative(since1, sizeof(since1), timestamp);
2615         s2 = format_timestamp(since2, sizeof(since2), timestamp);
2616
2617         if (s1)
2618                 printf(" since %s; %s\n", s2, s1);
2619         else if (s2)
2620                 printf(" since %s\n", s2);
2621         else
2622                 printf("\n");
2623
2624         if (!i->condition_result && i->condition_timestamp > 0) {
2625                 s1 = format_timestamp_relative(since1, sizeof(since1), i->condition_timestamp);
2626                 s2 = format_timestamp(since2, sizeof(since2), i->condition_timestamp);
2627
2628                 printf("           start condition failed at %s%s%s\n",
2629                        s2, s1 ? "; " : "", s1 ? s1 : "");
2630                 if (i->failed_condition_trigger)
2631                         printf("           none of the trigger conditions were met\n");
2632                 else if (i->failed_condition)
2633                         printf("           %s=%s%s was not met\n",
2634                                i->failed_condition,
2635                                i->failed_condition_negate ? "!" : "",
2636                                i->failed_condition_param);
2637         }
2638
2639         if (i->sysfs_path)
2640                 printf("   Device: %s\n", i->sysfs_path);
2641         if (i->where)
2642                 printf("    Where: %s\n", i->where);
2643         if (i->what)
2644                 printf("     What: %s\n", i->what);
2645
2646         STRV_FOREACH(t, i->documentation)
2647                 printf(" %*s %s\n", 9, t == i->documentation ? "Docs:" : "", *t);
2648
2649         STRV_FOREACH_PAIR(t, t2, i->listen)
2650                 printf(" %*s %s (%s)\n", 9, t == i->listen ? "Listen:" : "", *t2, *t);
2651
2652         if (i->accept)
2653                 printf(" Accepted: %u; Connected: %u\n", i->n_accepted, i->n_connections);
2654
2655         LIST_FOREACH(exec, p, i->exec) {
2656                 _cleanup_free_ char *argv = NULL;
2657                 bool good;
2658
2659                 /* Only show exited processes here */
2660                 if (p->code == 0)
2661                         continue;
2662
2663                 argv = strv_join(p->argv, " ");
2664                 printf("  Process: %u %s=%s ", p->pid, p->name, strna(argv));
2665
2666                 good = is_clean_exit_lsb(p->code, p->status, NULL);
2667                 if (!good) {
2668                         on = ansi_highlight_red();
2669                         off = ansi_highlight_off();
2670                 } else
2671                         on = off = "";
2672
2673                 printf("%s(code=%s, ", on, sigchld_code_to_string(p->code));
2674
2675                 if (p->code == CLD_EXITED) {
2676                         const char *c;
2677
2678                         printf("status=%i", p->status);
2679
2680                         c = exit_status_to_string(p->status, EXIT_STATUS_SYSTEMD);
2681                         if (c)
2682                                 printf("/%s", c);
2683
2684                 } else
2685                         printf("signal=%s", signal_to_string(p->status));
2686
2687                 printf(")%s\n", off);
2688
2689                 if (i->main_pid == p->pid &&
2690                     i->start_timestamp == p->start_timestamp &&
2691                     i->exit_timestamp == p->start_timestamp)
2692                         /* Let's not show this twice */
2693                         i->main_pid = 0;
2694
2695                 if (p->pid == i->control_pid)
2696                         i->control_pid = 0;
2697         }
2698
2699         if (i->main_pid > 0 || i->control_pid > 0) {
2700                 if (i->main_pid > 0) {
2701                         printf(" Main PID: %u", (unsigned) i->main_pid);
2702
2703                         if (i->running) {
2704                                 _cleanup_free_ char *comm = NULL;
2705                                 get_process_comm(i->main_pid, &comm);
2706                                 if (comm)
2707                                         printf(" (%s)", comm);
2708                         } else if (i->exit_code > 0) {
2709                                 printf(" (code=%s, ", sigchld_code_to_string(i->exit_code));
2710
2711                                 if (i->exit_code == CLD_EXITED) {
2712                                         const char *c;
2713
2714                                         printf("status=%i", i->exit_status);
2715
2716                                         c = exit_status_to_string(i->exit_status, EXIT_STATUS_SYSTEMD);
2717                                         if (c)
2718                                                 printf("/%s", c);
2719
2720                                 } else
2721                                         printf("signal=%s", signal_to_string(i->exit_status));
2722                                 printf(")");
2723                         }
2724
2725                         if (i->control_pid > 0)
2726                                 printf(";");
2727                 }
2728
2729                 if (i->control_pid > 0) {
2730                         _cleanup_free_ char *c = NULL;
2731
2732                         printf(" %8s: %u", i->main_pid ? "" : " Control", (unsigned) i->control_pid);
2733
2734                         get_process_comm(i->control_pid, &c);
2735                         if (c)
2736                                 printf(" (%s)", c);
2737                 }
2738
2739                 printf("\n");
2740         }
2741
2742         if (i->status_text)
2743                 printf("   Status: \"%s\"\n", i->status_text);
2744
2745         if (i->control_group &&
2746             (i->main_pid > 0 || i->control_pid > 0 || cg_is_empty_recursive(SYSTEMD_CGROUP_CONTROLLER, i->control_group, false) == 0)) {
2747                 unsigned c;
2748
2749                 printf("   CGroup: %s\n", i->control_group);
2750
2751                 if (arg_transport != TRANSPORT_SSH) {
2752                         unsigned k = 0;
2753                         pid_t extra[2];
2754                         char prefix[] = "           ";
2755
2756                         c = columns();
2757                         if (c > sizeof(prefix) - 1)
2758                                 c -= sizeof(prefix) - 1;
2759                         else
2760                                 c = 0;
2761
2762                         if (i->main_pid > 0)
2763                                 extra[k++] = i->main_pid;
2764
2765                         if (i->control_pid > 0)
2766                                 extra[k++] = i->control_pid;
2767
2768                         show_cgroup_and_extra(SYSTEMD_CGROUP_CONTROLLER, i->control_group, prefix,
2769                                                       c, false, extra, k, flags);
2770                 }
2771         }
2772
2773         if (i->id && arg_transport != TRANSPORT_SSH) {
2774                 printf("\n");
2775                 show_journal_by_unit(stdout,
2776                                      i->id,
2777                                      arg_output,
2778                                      0,
2779                                      i->inactive_exit_timestamp_monotonic,
2780                                      arg_lines,
2781                                      getuid(),
2782                                      flags,
2783                                      arg_scope == UNIT_FILE_SYSTEM,
2784                                      ellipsized);
2785         }
2786
2787         if (i->need_daemon_reload)
2788                 printf("\n%sWarning:%s Unit file changed on disk, 'systemctl %sdaemon-reload' recommended.\n",
2789                        ansi_highlight_red(),
2790                        ansi_highlight_off(),
2791                        arg_scope == UNIT_FILE_SYSTEM ? "" : "--user ");
2792 }
2793
2794 static void show_unit_help(UnitStatusInfo *i) {
2795         char **p;
2796
2797         assert(i);
2798
2799         if (!i->documentation) {
2800                 log_info("Documentation for %s not known.", i->id);
2801                 return;
2802         }
2803
2804         STRV_FOREACH(p, i->documentation) {
2805
2806                 if (startswith(*p, "man:")) {
2807                         size_t k;
2808                         char *e = NULL;
2809                         _cleanup_free_ char *page = NULL, *section = NULL;
2810                         const char *args[4] = { "man", NULL, NULL, NULL };
2811                         pid_t pid;
2812
2813                         k = strlen(*p);
2814
2815                         if ((*p)[k-1] == ')')
2816                                 e = strrchr(*p, '(');
2817
2818                         if (e) {
2819                                 page = strndup((*p) + 4, e - *p - 4);
2820                                 section = strndup(e + 1, *p + k - e - 2);
2821                                 if (!page || !section) {
2822                                         log_oom();
2823                                         return;
2824                                 }
2825
2826                                 args[1] = section;
2827                                 args[2] = page;
2828                         } else
2829                                 args[1] = *p + 4;
2830
2831                         pid = fork();
2832                         if (pid < 0) {
2833                                 log_error("Failed to fork: %m");
2834                                 continue;
2835                         }
2836
2837                         if (pid == 0) {
2838                                 /* Child */
2839                                 execvp(args[0], (char**) args);
2840                                 log_error("Failed to execute man: %m");
2841                                 _exit(EXIT_FAILURE);
2842                         }
2843
2844                         wait_for_terminate(pid, NULL);
2845                 } else
2846                         log_info("Can't show: %s", *p);
2847         }
2848 }
2849
2850 static int status_property(const char *name, DBusMessageIter *iter, UnitStatusInfo *i) {
2851
2852         assert(name);
2853         assert(iter);
2854         assert(i);
2855
2856         switch (dbus_message_iter_get_arg_type(iter)) {
2857
2858         case DBUS_TYPE_STRING: {
2859                 const char *s;
2860
2861                 dbus_message_iter_get_basic(iter, &s);
2862
2863                 if (!isempty(s)) {
2864                         if (streq(name, "Id"))
2865                                 i->id = s;
2866                         else if (streq(name, "LoadState"))
2867                                 i->load_state = s;
2868                         else if (streq(name, "ActiveState"))
2869                                 i->active_state = s;
2870                         else if (streq(name, "SubState"))
2871                                 i->sub_state = s;
2872                         else if (streq(name, "Description"))
2873                                 i->description = s;
2874                         else if (streq(name, "FragmentPath"))
2875                                 i->fragment_path = s;
2876                         else if (streq(name, "SourcePath"))
2877                                 i->source_path = s;
2878 #ifndef NOLEGACY
2879                         else if (streq(name, "DefaultControlGroup")) {
2880                                 const char *e;
2881                                 e = startswith(s, SYSTEMD_CGROUP_CONTROLLER ":");
2882                                 if (e)
2883                                         i->control_group = e;
2884                         }
2885 #endif
2886                         else if (streq(name, "ControlGroup"))
2887                                 i->control_group = s;
2888                         else if (streq(name, "StatusText"))
2889                                 i->status_text = s;
2890                         else if (streq(name, "PIDFile"))
2891                                 i->pid_file = s;
2892                         else if (streq(name, "SysFSPath"))
2893                                 i->sysfs_path = s;
2894                         else if (streq(name, "Where"))
2895                                 i->where = s;
2896                         else if (streq(name, "What"))
2897                                 i->what = s;
2898                         else if (streq(name, "Following"))
2899                                 i->following = s;
2900                         else if (streq(name, "UnitFileState"))
2901                                 i->unit_file_state = s;
2902                         else if (streq(name, "Result"))
2903                                 i->result = s;
2904                 }
2905
2906                 break;
2907         }
2908
2909         case DBUS_TYPE_BOOLEAN: {
2910                 dbus_bool_t b;
2911
2912                 dbus_message_iter_get_basic(iter, &b);
2913
2914                 if (streq(name, "Accept"))
2915                         i->accept = b;
2916                 else if (streq(name, "NeedDaemonReload"))
2917                         i->need_daemon_reload = b;
2918                 else if (streq(name, "ConditionResult"))
2919                         i->condition_result = b;
2920
2921                 break;
2922         }
2923
2924         case DBUS_TYPE_UINT32: {
2925                 uint32_t u;
2926
2927                 dbus_message_iter_get_basic(iter, &u);
2928
2929                 if (streq(name, "MainPID")) {
2930                         if (u > 0) {
2931                                 i->main_pid = (pid_t) u;
2932                                 i->running = true;
2933                         }
2934                 } else if (streq(name, "ControlPID"))
2935                         i->control_pid = (pid_t) u;
2936                 else if (streq(name, "ExecMainPID")) {
2937                         if (u > 0)
2938                                 i->main_pid = (pid_t) u;
2939                 } else if (streq(name, "NAccepted"))
2940                         i->n_accepted = u;
2941                 else if (streq(name, "NConnections"))
2942                         i->n_connections = u;
2943
2944                 break;
2945         }
2946
2947         case DBUS_TYPE_INT32: {
2948                 int32_t j;
2949
2950                 dbus_message_iter_get_basic(iter, &j);
2951
2952                 if (streq(name, "ExecMainCode"))
2953                         i->exit_code = (int) j;
2954                 else if (streq(name, "ExecMainStatus"))
2955                         i->exit_status = (int) j;
2956
2957                 break;
2958         }
2959
2960         case DBUS_TYPE_UINT64: {
2961                 uint64_t u;
2962
2963                 dbus_message_iter_get_basic(iter, &u);
2964
2965                 if (streq(name, "ExecMainStartTimestamp"))
2966                         i->start_timestamp = (usec_t) u;
2967                 else if (streq(name, "ExecMainExitTimestamp"))
2968                         i->exit_timestamp = (usec_t) u;
2969                 else if (streq(name, "ActiveEnterTimestamp"))
2970                         i->active_enter_timestamp = (usec_t) u;
2971                 else if (streq(name, "InactiveEnterTimestamp"))
2972                         i->inactive_enter_timestamp = (usec_t) u;
2973                 else if (streq(name, "InactiveExitTimestamp"))
2974                         i->inactive_exit_timestamp = (usec_t) u;
2975                 else if (streq(name, "InactiveExitTimestampMonotonic"))
2976                         i->inactive_exit_timestamp_monotonic = (usec_t) u;
2977                 else if (streq(name, "ActiveExitTimestamp"))
2978                         i->active_exit_timestamp = (usec_t) u;
2979                 else if (streq(name, "ConditionTimestamp"))
2980                         i->condition_timestamp = (usec_t) u;
2981
2982                 break;
2983         }
2984
2985         case DBUS_TYPE_ARRAY: {
2986
2987                 if (dbus_message_iter_get_element_type(iter) == DBUS_TYPE_STRUCT &&
2988                     startswith(name, "Exec")) {
2989                         DBusMessageIter sub;
2990
2991                         dbus_message_iter_recurse(iter, &sub);
2992                         while (dbus_message_iter_get_arg_type(&sub) == DBUS_TYPE_STRUCT) {
2993                                 ExecStatusInfo *info;
2994                                 int r;
2995
2996                                 info = new0(ExecStatusInfo, 1);
2997                                 if (!info)
2998                                         return -ENOMEM;
2999
3000                                 info->name = strdup(name);
3001                                 if (!info->name) {
3002                                         free(info);
3003                                         return -ENOMEM;
3004                                 }
3005
3006                                 r = exec_status_info_deserialize(&sub, info);
3007                                 if (r < 0) {
3008                                         free(info);
3009                                         return r;
3010                                 }
3011
3012                                 LIST_PREPEND(ExecStatusInfo, exec, i->exec, info);
3013
3014                                 dbus_message_iter_next(&sub);
3015                         }
3016
3017                 } else if (dbus_message_iter_get_element_type(iter) == DBUS_TYPE_STRUCT &&
3018                            streq(name, "Listen")) {
3019                         DBusMessageIter sub, sub2;
3020
3021                         dbus_message_iter_recurse(iter, &sub);
3022                         while (dbus_message_iter_get_arg_type(&sub) == DBUS_TYPE_STRUCT) {
3023                                 const char *type, *path;
3024
3025                                 dbus_message_iter_recurse(&sub, &sub2);
3026
3027                                 if (bus_iter_get_basic_and_next(&sub2, DBUS_TYPE_STRING, &type, true) >= 0 &&
3028                                     bus_iter_get_basic_and_next(&sub2, DBUS_TYPE_STRING, &path, false) >= 0) {
3029                                         int r;
3030
3031                                         r = strv_extend(&i->listen, type);
3032                                         if (r < 0)
3033                                                 return r;
3034                                         r = strv_extend(&i->listen, path);
3035                                         if (r < 0)
3036                                                 return r;
3037                                 }
3038
3039                                 dbus_message_iter_next(&sub);
3040                         }
3041
3042                         return 0;
3043
3044                 } else if (dbus_message_iter_get_element_type(iter) == DBUS_TYPE_STRING &&
3045                            streq(name, "DropInPaths")) {
3046                         int r = bus_parse_strv_iter(iter, &i->dropin_paths);
3047                         if (r < 0)
3048                                 return r;
3049
3050                 } else if (dbus_message_iter_get_element_type(iter) == DBUS_TYPE_STRING &&
3051                            streq(name, "Documentation")) {
3052
3053                         DBusMessageIter sub;
3054
3055                         dbus_message_iter_recurse(iter, &sub);
3056                         while (dbus_message_iter_get_arg_type(&sub) == DBUS_TYPE_STRING) {
3057                                 const char *s;
3058                                 int r;
3059
3060                                 dbus_message_iter_get_basic(&sub, &s);
3061
3062                                 r = strv_extend(&i->documentation, s);
3063                                 if (r < 0)
3064                                         return r;
3065
3066                                 dbus_message_iter_next(&sub);
3067                         }
3068
3069                 } else if (dbus_message_iter_get_element_type(iter) == DBUS_TYPE_STRUCT &&
3070                            streq(name, "Conditions")) {
3071                         DBusMessageIter sub, sub2;
3072
3073                         dbus_message_iter_recurse(iter, &sub);
3074                         while (dbus_message_iter_get_arg_type(&sub) == DBUS_TYPE_STRUCT) {
3075                                 const char *cond, *param;
3076                                 dbus_bool_t trigger, negate;
3077                                 dbus_int32_t state;
3078
3079                                 dbus_message_iter_recurse(&sub, &sub2);
3080                                 log_debug("here");
3081
3082                                 if(bus_iter_get_basic_and_next(&sub2, DBUS_TYPE_STRING, &cond, true) >= 0 &&
3083                                    bus_iter_get_basic_and_next(&sub2, DBUS_TYPE_BOOLEAN, &trigger, true) >= 0 &&
3084                                    bus_iter_get_basic_and_next(&sub2, DBUS_TYPE_BOOLEAN, &negate, true) >= 0 &&
3085                                    bus_iter_get_basic_and_next(&sub2, DBUS_TYPE_STRING, &param, true) >= 0 &&
3086                                    bus_iter_get_basic_and_next(&sub2, DBUS_TYPE_INT32, &state, false) >= 0) {
3087                                         log_debug("%s %d %d %s %d", cond, trigger, negate, param, state);
3088                                         if (state < 0 && (!trigger || !i->failed_condition)) {
3089                                                 i->failed_condition = cond;
3090                                                 i->failed_condition_trigger = trigger;
3091                                                 i->failed_condition_negate = negate;
3092                                                 i->failed_condition_param = param;
3093                                         }
3094                                 }
3095
3096                                 dbus_message_iter_next(&sub);
3097                         }
3098                 }
3099
3100                 break;
3101         }
3102
3103         case DBUS_TYPE_STRUCT: {
3104
3105                 if (streq(name, "LoadError")) {
3106                         DBusMessageIter sub;
3107                         const char *n, *message;
3108                         int r;
3109
3110                         dbus_message_iter_recurse(iter, &sub);
3111
3112                         r = bus_iter_get_basic_and_next(&sub, DBUS_TYPE_STRING, &n, true);
3113                         if (r < 0)
3114                                 return r;
3115
3116                         r = bus_iter_get_basic_and_next(&sub, DBUS_TYPE_STRING, &message, false);
3117                         if (r < 0)
3118                                 return r;
3119
3120                         if (!isempty(message))
3121                                 i->load_error = message;
3122                 }
3123
3124                 break;
3125         }
3126         }
3127
3128         return 0;
3129 }
3130
3131 static int print_property(const char *name, DBusMessageIter *iter) {
3132         assert(name);
3133         assert(iter);
3134
3135         /* This is a low-level property printer, see
3136          * print_status_info() for the nicer output */
3137
3138         if (arg_properties && !strv_find(arg_properties, name))
3139                 return 0;
3140
3141         switch (dbus_message_iter_get_arg_type(iter)) {
3142
3143         case DBUS_TYPE_STRUCT: {
3144                 DBusMessageIter sub;
3145                 dbus_message_iter_recurse(iter, &sub);
3146
3147                 if (dbus_message_iter_get_arg_type(&sub) == DBUS_TYPE_UINT32 && streq(name, "Job")) {
3148                         uint32_t u;
3149
3150                         dbus_message_iter_get_basic(&sub, &u);
3151
3152                         if (u)
3153                                 printf("%s=%u\n", name, (unsigned) u);
3154                         else if (arg_all)
3155                                 printf("%s=\n", name);
3156
3157                         return 0;
3158                 } else if (dbus_message_iter_get_arg_type(&sub) == DBUS_TYPE_STRING && streq(name, "Unit")) {
3159                         const char *s;
3160
3161                         dbus_message_iter_get_basic(&sub, &s);
3162
3163                         if (arg_all || s[0])
3164                                 printf("%s=%s\n", name, s);
3165
3166                         return 0;
3167                 } else if (dbus_message_iter_get_arg_type(&sub) == DBUS_TYPE_STRING && streq(name, "LoadError")) {
3168                         const char *a = NULL, *b = NULL;
3169
3170                         if (bus_iter_get_basic_and_next(&sub, DBUS_TYPE_STRING, &a, true) >= 0)
3171                                 bus_iter_get_basic_and_next(&sub, DBUS_TYPE_STRING, &b, false);
3172
3173                         if (arg_all || !isempty(a) || !isempty(b))
3174                                 printf("%s=%s \"%s\"\n", name, strempty(a), strempty(b));
3175
3176                         return 0;
3177                 }
3178
3179                 break;
3180         }
3181
3182         case DBUS_TYPE_ARRAY:
3183
3184                 if (dbus_message_iter_get_element_type(iter) == DBUS_TYPE_STRUCT && streq(name, "EnvironmentFiles")) {
3185                         DBusMessageIter sub, sub2;
3186
3187                         dbus_message_iter_recurse(iter, &sub);
3188                         while (dbus_message_iter_get_arg_type(&sub) == DBUS_TYPE_STRUCT) {
3189                                 const char *path;
3190                                 dbus_bool_t ignore;
3191
3192                                 dbus_message_iter_recurse(&sub, &sub2);
3193
3194                                 if (bus_iter_get_basic_and_next(&sub2, DBUS_TYPE_STRING, &path, true) >= 0 &&
3195                                     bus_iter_get_basic_and_next(&sub2, DBUS_TYPE_BOOLEAN, &ignore, false) >= 0)
3196                                         printf("EnvironmentFile=%s (ignore_errors=%s)\n", path, yes_no(ignore));
3197
3198                                 dbus_message_iter_next(&sub);
3199                         }
3200
3201                         return 0;
3202
3203                 } else if (dbus_message_iter_get_element_type(iter) == DBUS_TYPE_STRUCT && streq(name, "Paths")) {
3204                         DBusMessageIter sub, sub2;
3205
3206                         dbus_message_iter_recurse(iter, &sub);
3207
3208                         while (dbus_message_iter_get_arg_type(&sub) == DBUS_TYPE_STRUCT) {
3209                                 const char *type, *path;
3210
3211                                 dbus_message_iter_recurse(&sub, &sub2);
3212
3213                                 if (bus_iter_get_basic_and_next(&sub2, DBUS_TYPE_STRING, &type, true) >= 0 &&
3214                                     bus_iter_get_basic_and_next(&sub2, DBUS_TYPE_STRING, &path, false) >= 0)
3215                                         printf("%s=%s\n", type, path);
3216
3217                                 dbus_message_iter_next(&sub);
3218                         }
3219
3220                         return 0;
3221
3222                 } else if (dbus_message_iter_get_element_type(iter) == DBUS_TYPE_STRUCT && streq(name, "Listen")) {
3223                         DBusMessageIter sub, sub2;
3224
3225                         dbus_message_iter_recurse(iter, &sub);
3226                         while (dbus_message_iter_get_arg_type(&sub) == DBUS_TYPE_STRUCT) {
3227                                 const char *type, *path;
3228
3229                                 dbus_message_iter_recurse(&sub, &sub2);
3230
3231                                 if (bus_iter_get_basic_and_next(&sub2, DBUS_TYPE_STRING, &type, true) >= 0 &&
3232                                     bus_iter_get_basic_and_next(&sub2, DBUS_TYPE_STRING, &path, false) >= 0)
3233                                         printf("Listen%s=%s\n", type, path);
3234
3235                                 dbus_message_iter_next(&sub);
3236                         }
3237
3238                         return 0;
3239
3240                 } else if (dbus_message_iter_get_element_type(iter) == DBUS_TYPE_STRUCT && streq(name, "Timers")) {
3241                         DBusMessageIter sub, sub2;
3242
3243                         dbus_message_iter_recurse(iter, &sub);
3244                         while (dbus_message_iter_get_arg_type(&sub) == DBUS_TYPE_STRUCT) {
3245                                 const char *base;
3246                                 uint64_t value, next_elapse;
3247
3248                                 dbus_message_iter_recurse(&sub, &sub2);
3249
3250                                 if (bus_iter_get_basic_and_next(&sub2, DBUS_TYPE_STRING, &base, true) >= 0 &&
3251                                     bus_iter_get_basic_and_next(&sub2, DBUS_TYPE_UINT64, &value, true) >= 0 &&
3252                                     bus_iter_get_basic_and_next(&sub2, DBUS_TYPE_UINT64, &next_elapse, false) >= 0) {
3253                                         char timespan1[FORMAT_TIMESPAN_MAX], timespan2[FORMAT_TIMESPAN_MAX];
3254
3255                                         printf("%s={ value=%s ; next_elapse=%s }\n",
3256                                                base,
3257                                                format_timespan(timespan1, sizeof(timespan1), value, 0),
3258                                                format_timespan(timespan2, sizeof(timespan2), next_elapse, 0));
3259                                 }
3260
3261                                 dbus_message_iter_next(&sub);
3262                         }
3263
3264                         return 0;
3265
3266                 } else if (dbus_message_iter_get_element_type(iter) == DBUS_TYPE_STRUCT && startswith(name, "Exec")) {
3267                         DBusMessageIter sub;
3268
3269                         dbus_message_iter_recurse(iter, &sub);
3270                         while (dbus_message_iter_get_arg_type(&sub) == DBUS_TYPE_STRUCT) {
3271                                 ExecStatusInfo info = {};
3272
3273                                 if (exec_status_info_deserialize(&sub, &info) >= 0) {
3274                                         char timestamp1[FORMAT_TIMESTAMP_MAX], timestamp2[FORMAT_TIMESTAMP_MAX];
3275                                         _cleanup_free_ char *t;
3276
3277                                         t = strv_join(info.argv, " ");
3278
3279                                         printf("%s={ path=%s ; argv[]=%s ; ignore_errors=%s ; start_time=[%s] ; stop_time=[%s] ; pid=%u ; code=%s ; status=%i%s%s }\n",
3280                                                name,
3281                                                strna(info.path),
3282                                                strna(t),
3283                                                yes_no(info.ignore),
3284                                                strna(format_timestamp(timestamp1, sizeof(timestamp1), info.start_timestamp)),
3285                                                strna(format_timestamp(timestamp2, sizeof(timestamp2), info.exit_timestamp)),
3286                                                (unsigned) info. pid,
3287                                                sigchld_code_to_string(info.code),
3288                                                info.status,
3289                                                info.code == CLD_EXITED ? "" : "/",
3290                                                strempty(info.code == CLD_EXITED ? NULL : signal_to_string(info.status)));
3291                                 }
3292
3293                                 free(info.path);
3294                                 strv_free(info.argv);
3295
3296                                 dbus_message_iter_next(&sub);
3297                         }
3298
3299                         return 0;
3300
3301                 } else if (dbus_message_iter_get_element_type(iter) == DBUS_TYPE_STRUCT && streq(name, "DeviceAllow")) {
3302                         DBusMessageIter sub, sub2;
3303
3304                         dbus_message_iter_recurse(iter, &sub);
3305                         while (dbus_message_iter_get_arg_type(&sub) == DBUS_TYPE_STRUCT) {
3306                                 const char *path, *rwm;
3307
3308                                 dbus_message_iter_recurse(&sub, &sub2);
3309
3310                                 if (bus_iter_get_basic_and_next(&sub2, DBUS_TYPE_STRING, &path, true) >= 0 &&
3311                                     bus_iter_get_basic_and_next(&sub2, DBUS_TYPE_STRING, &rwm, false) >= 0)
3312                                         printf("%s=%s %s\n", name, strna(path), strna(rwm));
3313
3314                                 dbus_message_iter_next(&sub);
3315                         }
3316                         return 0;
3317
3318                 } else if (dbus_message_iter_get_element_type(iter) == DBUS_TYPE_STRUCT && (streq(name, "BlockIOReadBandwidth") || streq(name, "BlockIOWriteBandwidth"))) {
3319                         DBusMessageIter sub, sub2;
3320
3321                         dbus_message_iter_recurse(iter, &sub);
3322                         while (dbus_message_iter_get_arg_type(&sub) == DBUS_TYPE_STRUCT) {
3323                                 const char *path;
3324                                 uint64_t bandwidth;
3325
3326                                 dbus_message_iter_recurse(&sub, &sub2);
3327
3328                                 if (bus_iter_get_basic_and_next(&sub2, DBUS_TYPE_STRING, &path, true) >= 0 &&
3329                                     bus_iter_get_basic_and_next(&sub2, DBUS_TYPE_UINT64, &bandwidth, false) >= 0)
3330                                         printf("%s=%s %" PRIu64 "\n", name, strna(path), bandwidth);
3331
3332                                 dbus_message_iter_next(&sub);
3333                         }
3334                         return 0;
3335                 }
3336
3337
3338                 break;
3339         }
3340
3341         if (generic_print_property(name, iter, arg_all) > 0)
3342                 return 0;
3343
3344         if (arg_all)
3345                 printf("%s=[unprintable]\n", name);
3346
3347         return 0;
3348 }
3349
3350 static int show_one(const char *verb,
3351                     DBusConnection *bus,
3352                     const char *path,
3353                     bool show_properties,
3354                     bool *new_line,
3355                     bool *ellipsized) {
3356         _cleanup_free_ DBusMessage *reply = NULL;
3357         const char *interface = "";
3358         int r;
3359         DBusMessageIter iter, sub, sub2, sub3;
3360         UnitStatusInfo info = {};
3361         ExecStatusInfo *p;
3362
3363         assert(path);
3364         assert(new_line);
3365
3366         r = bus_method_call_with_reply(
3367                         bus,
3368                         "org.freedesktop.systemd1",
3369                         path,
3370                         "org.freedesktop.DBus.Properties",
3371                         "GetAll",
3372                         &reply,
3373                         NULL,
3374                         DBUS_TYPE_STRING, &interface,
3375                         DBUS_TYPE_INVALID);
3376         if (r < 0)
3377                 return r;
3378
3379         if (!dbus_message_iter_init(reply, &iter) ||
3380             dbus_message_iter_get_arg_type(&iter) != DBUS_TYPE_ARRAY ||
3381             dbus_message_iter_get_element_type(&iter) != DBUS_TYPE_DICT_ENTRY)  {
3382                 log_error("Failed to parse reply.");
3383                 return -EIO;
3384         }
3385
3386         dbus_message_iter_recurse(&iter, &sub);
3387
3388         if (*new_line)
3389                 printf("\n");
3390
3391         *new_line = true;
3392
3393         while (dbus_message_iter_get_arg_type(&sub) != DBUS_TYPE_INVALID) {
3394                 const char *name;
3395
3396                 assert(dbus_message_iter_get_arg_type(&sub) == DBUS_TYPE_DICT_ENTRY);
3397                 dbus_message_iter_recurse(&sub, &sub2);
3398
3399                 if (bus_iter_get_basic_and_next(&sub2, DBUS_TYPE_STRING, &name, true) < 0 ||
3400                     dbus_message_iter_get_arg_type(&sub2) != DBUS_TYPE_VARIANT) {
3401                         log_error("Failed to parse reply.");
3402                         return -EIO;
3403                 }
3404
3405                 dbus_message_iter_recurse(&sub2, &sub3);
3406
3407                 if (show_properties)
3408                         r = print_property(name, &sub3);
3409                 else
3410                         r = status_property(name, &sub3, &info);
3411                 if (r < 0) {
3412                         log_error("Failed to parse reply.");
3413                         return -EIO;
3414                 }
3415
3416                 dbus_message_iter_next(&sub);
3417         }
3418
3419         r = 0;
3420
3421         if (!show_properties) {
3422                 if (streq(verb, "help"))
3423                         show_unit_help(&info);
3424                 else
3425                         print_status_info(&info, ellipsized);
3426         }
3427
3428         strv_free(info.documentation);
3429         strv_free(info.dropin_paths);
3430         strv_free(info.listen);
3431
3432         if (!streq_ptr(info.active_state, "active") &&
3433             !streq_ptr(info.active_state, "reloading") &&
3434             streq(verb, "status")) {
3435                 /* According to LSB: "program not running" */
3436                 /* 0: program is running or service is OK
3437                  * 1: program is dead and /var/run pid file exists
3438                  * 2: program is dead and /var/lock lock file exists
3439                  * 3: program is not running
3440                  * 4: program or service status is unknown
3441                  */
3442                 if (info.pid_file && access(info.pid_file, F_OK) == 0)
3443                         r = 1;
3444                 else
3445                         r = 3;
3446         }
3447
3448         while ((p = info.exec)) {
3449                 LIST_REMOVE(ExecStatusInfo, exec, info.exec, p);
3450                 exec_status_info_free(p);
3451         }
3452
3453         return r;
3454 }
3455
3456 static int show_one_by_pid(const char *verb,
3457                            DBusConnection *bus,
3458                            uint32_t pid,
3459                            bool *new_line,
3460                            bool *ellipsized) {
3461         _cleanup_dbus_message_unref_ DBusMessage *reply = NULL;
3462         const char *path = NULL;
3463         _cleanup_dbus_error_free_ DBusError error;
3464         int r;
3465
3466         dbus_error_init(&error);
3467
3468         r = bus_method_call_with_reply(
3469                         bus,
3470                         "org.freedesktop.systemd1",
3471                         "/org/freedesktop/systemd1",
3472                         "org.freedesktop.systemd1.Manager",
3473                         "GetUnitByPID",
3474                         &reply,
3475                         NULL,
3476                         DBUS_TYPE_UINT32, &pid,
3477                         DBUS_TYPE_INVALID);
3478         if (r < 0)
3479                 return r;
3480
3481         if (!dbus_message_get_args(reply, &error,
3482                                    DBUS_TYPE_OBJECT_PATH, &path,
3483                                    DBUS_TYPE_INVALID)) {
3484                 log_error("Failed to parse reply: %s", bus_error_message(&error));
3485                 return -EIO;
3486         }
3487
3488         r = show_one(verb, bus, path, false, new_line, ellipsized);
3489         return r;
3490 }
3491
3492 static int show_all(const char* verb,
3493                     DBusConnection *bus,
3494                     bool show_properties,
3495                     bool *new_line,
3496                     bool *ellipsized) {
3497         _cleanup_dbus_message_unref_ DBusMessage *reply = NULL;
3498         _cleanup_free_ struct unit_info *unit_infos = NULL;
3499         unsigned c = 0;
3500         const struct unit_info *u;
3501         int r;
3502
3503         r = get_unit_list(bus, &reply, &unit_infos, &c);
3504         if (r < 0)
3505                 return r;
3506
3507         qsort(unit_infos, c, sizeof(struct unit_info), compare_unit_info);
3508
3509         for (u = unit_infos; u < unit_infos + c; u++) {
3510                 _cleanup_free_ char *p = NULL;
3511
3512                 if (!output_show_unit(u))
3513                         continue;
3514
3515                 p = unit_dbus_path_from_name(u->id);
3516                 if (!p)
3517                         return log_oom();
3518
3519                 printf("%s -> '%s'\n", u->id, p);
3520
3521                 r = show_one(verb, bus, p, show_properties, new_line, ellipsized);
3522                 if (r != 0)
3523                         return r;
3524         }
3525
3526         return 0;
3527 }
3528
3529 static int show(DBusConnection *bus, char **args) {
3530         int r, ret = 0;
3531         bool show_properties, show_status, new_line = false;
3532         char **name;
3533         bool ellipsized = false;
3534
3535         assert(bus);
3536         assert(args);
3537
3538         show_properties = streq(args[0], "show");
3539         show_status = streq(args[0], "status");
3540
3541         if (show_properties)
3542                 pager_open_if_enabled();
3543
3544         /* If no argument is specified inspect the manager itself */
3545
3546         if (show_properties && strv_length(args) <= 1)
3547                 return show_one(args[0], bus, "/org/freedesktop/systemd1", show_properties, &new_line, &ellipsized);
3548
3549         if (show_status && strv_length(args) <= 1)
3550                 ret = show_all(args[0], bus, false, &new_line, &ellipsized);
3551         else
3552                 STRV_FOREACH(name, args+1) {
3553                         uint32_t id;
3554
3555                         if (safe_atou32(*name, &id) < 0) {
3556                                 _cleanup_free_ char *p = NULL, *n = NULL;
3557                                 /* Interpret as unit name */
3558
3559                                 n = unit_name_mangle(*name);
3560                                 if (!n)
3561                                         return log_oom();
3562
3563                                 p = unit_dbus_path_from_name(n);
3564                                 if (!p)
3565                                         return log_oom();
3566
3567                                 r = show_one(args[0], bus, p, show_properties, &new_line, &ellipsized);
3568                                 if (r != 0)
3569                                         ret = r;
3570
3571                         } else if (show_properties) {
3572                                 _cleanup_free_ char *p = NULL;
3573
3574                                 /* Interpret as job id */
3575                                 if (asprintf(&p, "/org/freedesktop/systemd1/job/%u", id) < 0)
3576                                         return log_oom();
3577
3578                                 r = show_one(args[0], bus, p, show_properties, &new_line, &ellipsized);
3579                                 if (r != 0)
3580                                         ret = r;
3581
3582                         } else {
3583                                 /* Interpret as PID */
3584                                 r = show_one_by_pid(args[0], bus, id, &new_line, &ellipsized);
3585                                 if (r != 0)
3586                                         ret = r;
3587                         }
3588                 }
3589
3590         if (ellipsized && !arg_quiet)
3591                 printf("Hint: Some lines were ellipsized, use -l to show in full.\n");
3592
3593         return ret;
3594 }
3595
3596 static int append_assignment(DBusMessageIter *iter, const char *assignment) {
3597         const char *eq;
3598         char *field;
3599         DBusMessageIter sub;
3600         int r;
3601
3602         assert(iter);
3603         assert(assignment);
3604
3605         eq = strchr(assignment, '=');
3606         if (!eq) {
3607                 log_error("Not an assignment: %s", assignment);
3608                 return -EINVAL;
3609         }
3610
3611         field = strndupa(assignment, eq - assignment);
3612         eq ++;
3613
3614         if (!dbus_message_iter_append_basic(iter, DBUS_TYPE_STRING, &field))
3615                 return log_oom();
3616
3617         if (streq(field, "CPUAccounting") ||
3618             streq(field, "MemoryAccounting") ||
3619             streq(field, "BlockIOAccounting")) {
3620                 dbus_bool_t b;
3621
3622                 r = parse_boolean(eq);
3623                 if (r < 0) {
3624                         log_error("Failed to parse boolean assignment %s.", assignment);
3625                         return -EINVAL;
3626                 }
3627
3628                 b = r;
3629                 if (!dbus_message_iter_open_container(iter, DBUS_TYPE_VARIANT, "b", &sub) ||
3630                     !dbus_message_iter_append_basic(&sub, DBUS_TYPE_BOOLEAN, &b))
3631                         return log_oom();
3632
3633         } else if (streq(field, "MemoryLimit") || streq(field, "MemorySoftLimit")) {
3634                 off_t bytes;
3635                 uint64_t u;
3636
3637                 r = parse_bytes(eq, &bytes);
3638                 if (r < 0) {
3639                         log_error("Failed to parse bytes specification %s", assignment);
3640                         return -EINVAL;
3641                 }
3642
3643                 u = bytes;
3644                 if (!dbus_message_iter_open_container(iter, DBUS_TYPE_VARIANT, "t", &sub) ||
3645                     !dbus_message_iter_append_basic(&sub, DBUS_TYPE_UINT64, &u))
3646                         return log_oom();
3647
3648         } else if (streq(field, "CPUShares") || streq(field, "BlockIOWeight")) {
3649                 uint64_t u;
3650
3651                 r = safe_atou64(eq, &u);
3652                 if (r < 0) {
3653                         log_error("Failed to parse %s value %s.", field, eq);
3654                         return -EINVAL;
3655                 }
3656
3657                 if (!dbus_message_iter_open_container(iter, DBUS_TYPE_VARIANT, "t", &sub) ||
3658                     !dbus_message_iter_append_basic(&sub, DBUS_TYPE_UINT64, &u))
3659                         return log_oom();
3660
3661         } else if (streq(field, "DevicePolicy")) {
3662
3663                 if (!dbus_message_iter_open_container(iter, DBUS_TYPE_VARIANT, "s", &sub) ||
3664                     !dbus_message_iter_append_basic(&sub, DBUS_TYPE_STRING, &eq))
3665                         return log_oom();
3666
3667         } else if (streq(field, "DeviceAllow")) {
3668                 DBusMessageIter sub2;
3669
3670                 if (!dbus_message_iter_open_container(iter, DBUS_TYPE_VARIANT, "a(ss)", &sub) ||
3671                     !dbus_message_iter_open_container(&sub, DBUS_TYPE_ARRAY, "(ss)", &sub2))
3672                         return log_oom();
3673
3674                 if (!isempty(eq)) {
3675                         const char *path, *rwm;
3676                         DBusMessageIter sub3;
3677                         char *e;
3678
3679                         e = strchr(eq, ' ');
3680                         if (e) {
3681                                 path = strndupa(eq, e - eq);
3682                                 rwm = e+1;
3683                         } else {
3684                                 path = eq;
3685                                 rwm = "";
3686                         }
3687
3688                         if (!dbus_message_iter_open_container(&sub2, DBUS_TYPE_STRUCT, NULL, &sub3) ||
3689                             !dbus_message_iter_append_basic(&sub3, DBUS_TYPE_STRING, &path) ||
3690                             !dbus_message_iter_append_basic(&sub3, DBUS_TYPE_STRING, &rwm) ||
3691                             !dbus_message_iter_close_container(&sub2, &sub3))
3692                                 return log_oom();
3693                 }
3694
3695                 if (!dbus_message_iter_close_container(&sub, &sub2))
3696                         return log_oom();
3697
3698         } else {
3699                 log_error("Unknown assignment %s.", assignment);
3700                 return -EINVAL;
3701         }
3702
3703         if (!dbus_message_iter_close_container(iter, &sub))
3704                 return log_oom();
3705
3706         return 0;
3707 }
3708
3709 static int set_property(DBusConnection *bus, char **args) {
3710
3711         _cleanup_free_ DBusMessage *m = NULL, *reply = NULL;
3712         DBusMessageIter iter, sub;
3713         dbus_bool_t runtime;
3714         DBusError error;
3715         char **i;
3716         int r;
3717
3718         dbus_error_init(&error);
3719
3720         m = dbus_message_new_method_call(
3721                         "org.freedesktop.systemd1",
3722                         "/org/freedesktop/systemd1",
3723                         "org.freedesktop.systemd1.Manager",
3724                         "SetUnitProperties");
3725         if (!m)
3726                 return log_oom();
3727
3728         dbus_message_iter_init_append(m, &iter);
3729
3730         runtime = arg_runtime;
3731
3732         if (!dbus_message_iter_append_basic(&iter, DBUS_TYPE_STRING, &args[1]) ||
3733             !dbus_message_iter_append_basic(&iter, DBUS_TYPE_BOOLEAN, &runtime) ||
3734             !dbus_message_iter_open_container(&iter, DBUS_TYPE_ARRAY, "(sv)", &sub))
3735                 return log_oom();
3736
3737         STRV_FOREACH(i, args + 2) {
3738                 DBusMessageIter sub2;
3739
3740                 if (!dbus_message_iter_open_container(&sub, DBUS_TYPE_STRUCT, NULL, &sub2))
3741                         return log_oom();
3742
3743                 r = append_assignment(&sub2, *i);
3744                 if (r < 0)
3745                         return r;
3746
3747                 if (!dbus_message_iter_close_container(&sub, &sub2))
3748                         return log_oom();
3749
3750         }
3751
3752         if (!dbus_message_iter_close_container(&iter, &sub))
3753                 return log_oom();
3754
3755         reply = dbus_connection_send_with_reply_and_block(bus, m, -1, &error);
3756         if (!reply) {
3757                 log_error("Failed to issue method call: %s", bus_error_message(&error));
3758                 dbus_error_free(&error);
3759                 return -EIO;
3760         }
3761
3762         return 0;
3763 }
3764
3765 static int snapshot(DBusConnection *bus, char **args) {
3766         _cleanup_dbus_message_unref_ DBusMessage *reply = NULL;
3767         DBusError error;
3768         int r;
3769         dbus_bool_t cleanup = FALSE;
3770         DBusMessageIter iter, sub;
3771         const char
3772                 *path, *id,
3773                 *interface = "org.freedesktop.systemd1.Unit",
3774                 *property = "Id";
3775         _cleanup_free_ char *n = NULL;
3776
3777         dbus_error_init(&error);
3778
3779         if (strv_length(args) > 1)
3780                 n = unit_name_mangle_with_suffix(args[1], ".snapshot");
3781         else
3782                 n = strdup("");
3783         if (!n)
3784                 return log_oom();
3785
3786         r = bus_method_call_with_reply(
3787                         bus,
3788                         "org.freedesktop.systemd1",
3789                         "/org/freedesktop/systemd1",
3790                         "org.freedesktop.systemd1.Manager",
3791                         "CreateSnapshot",
3792                         &reply,
3793                         NULL,
3794                         DBUS_TYPE_STRING, &n,
3795                         DBUS_TYPE_BOOLEAN, &cleanup,
3796                         DBUS_TYPE_INVALID);
3797         if (r < 0)
3798                 return r;
3799
3800         if (!dbus_message_get_args(reply, &error,
3801                                    DBUS_TYPE_OBJECT_PATH, &path,
3802                                    DBUS_TYPE_INVALID)) {
3803                 log_error("Failed to parse reply: %s", bus_error_message(&error));
3804                 dbus_error_free(&error);
3805                 return -EIO;
3806         }
3807
3808         dbus_message_unref(reply);
3809         reply = NULL;
3810
3811         r = bus_method_call_with_reply (
3812                         bus,
3813                         "org.freedesktop.systemd1",
3814                         path,
3815                         "org.freedesktop.DBus.Properties",
3816                         "Get",
3817                         &reply,
3818                         NULL,
3819                         DBUS_TYPE_STRING, &interface,
3820                         DBUS_TYPE_STRING, &property,
3821                         DBUS_TYPE_INVALID);
3822         if (r < 0)
3823                 return r;
3824
3825         if (!dbus_message_iter_init(reply, &iter) ||
3826             dbus_message_iter_get_arg_type(&iter) != DBUS_TYPE_VARIANT)  {
3827                 log_error("Failed to parse reply.");
3828                 return -EIO;
3829         }
3830
3831         dbus_message_iter_recurse(&iter, &sub);
3832
3833         if (dbus_message_iter_get_arg_type(&sub) != DBUS_TYPE_STRING)  {
3834                 log_error("Failed to parse reply.");
3835                 return -EIO;
3836         }
3837
3838         dbus_message_iter_get_basic(&sub, &id);
3839
3840         if (!arg_quiet)
3841                 puts(id);
3842
3843         return 0;
3844 }
3845
3846 static int delete_snapshot(DBusConnection *bus, char **args) {
3847         char **name;
3848
3849         assert(args);
3850
3851         STRV_FOREACH(name, args+1) {
3852                 _cleanup_free_ char *n = NULL;
3853                 int r;
3854
3855                 n = unit_name_mangle_with_suffix(*name, ".snapshot");
3856                 if (!n)
3857                         return log_oom();
3858
3859                 r = bus_method_call_with_reply(
3860                                 bus,
3861                                 "org.freedesktop.systemd1",
3862                                 "/org/freedesktop/systemd1",
3863                                 "org.freedesktop.systemd1.Manager",
3864                                 "RemoveSnapshot",
3865                                 NULL,
3866                                 NULL,
3867                                 DBUS_TYPE_STRING, &n,
3868                                 DBUS_TYPE_INVALID);
3869                 if (r < 0)
3870                         return r;
3871         }
3872
3873         return 0;
3874 }
3875
3876 static int daemon_reload(DBusConnection *bus, char **args) {
3877         int r;
3878         const char *method;
3879         DBusError error;
3880
3881         if (arg_action == ACTION_RELOAD)
3882                 method = "Reload";
3883         else if (arg_action == ACTION_REEXEC)
3884                 method = "Reexecute";
3885         else {
3886                 assert(arg_action == ACTION_SYSTEMCTL);
3887
3888                 method =
3889                         streq(args[0], "clear-jobs")    ||
3890                         streq(args[0], "cancel")        ? "ClearJobs" :
3891                         streq(args[0], "daemon-reexec") ? "Reexecute" :
3892                         streq(args[0], "reset-failed")  ? "ResetFailed" :
3893                         streq(args[0], "halt")          ? "Halt" :
3894                         streq(args[0], "poweroff")      ? "PowerOff" :
3895                         streq(args[0], "reboot")        ? "Reboot" :
3896                         streq(args[0], "kexec")         ? "KExec" :
3897                         streq(args[0], "exit")          ? "Exit" :
3898                                     /* "daemon-reload" */ "Reload";
3899         }
3900
3901         r = bus_method_call_with_reply(
3902                         bus,
3903                         "org.freedesktop.systemd1",
3904                         "/org/freedesktop/systemd1",
3905                         "org.freedesktop.systemd1.Manager",
3906                         method,
3907                         NULL,
3908                         &error,
3909                         DBUS_TYPE_INVALID);
3910
3911         if (r == -ENOENT && arg_action != ACTION_SYSTEMCTL)
3912                 /* There's always a fallback possible for
3913                  * legacy actions. */
3914                 r = -EADDRNOTAVAIL;
3915         else if ((r == -ETIMEDOUT || r == -ECONNRESET) && streq(method, "Reexecute"))
3916                 /* On reexecution, we expect a disconnect, not a
3917                  * reply */
3918                 r = 0;
3919         else if (r < 0)
3920                 log_error("Failed to issue method call: %s", bus_error_message(&error));
3921
3922         dbus_error_free(&error);
3923         return r;
3924 }
3925
3926 static int reset_failed(DBusConnection *bus, char **args) {
3927         int r = 0;
3928         char **name;
3929
3930         if (strv_length(args) <= 1)
3931                 return daemon_reload(bus, args);
3932
3933         STRV_FOREACH(name, args+1) {
3934                 _cleanup_free_ char *n;
3935
3936                 n = unit_name_mangle(*name);
3937                 if (!n)
3938                         return log_oom();
3939
3940                 r = bus_method_call_with_reply(
3941                                 bus,
3942                                 "org.freedesktop.systemd1",
3943                                 "/org/freedesktop/systemd1",
3944                                 "org.freedesktop.systemd1.Manager",
3945                                 "ResetFailedUnit",
3946                                 NULL,
3947                                 NULL,
3948                                 DBUS_TYPE_STRING, &n,
3949                                 DBUS_TYPE_INVALID);
3950                 if (r < 0)
3951                         return r;
3952         }
3953
3954         return 0;
3955 }
3956
3957 static int show_enviroment(DBusConnection *bus, char **args) {
3958         _cleanup_dbus_message_unref_ DBusMessage *reply = NULL;
3959         DBusMessageIter iter, sub, sub2;
3960         int r;
3961         const char
3962                 *interface = "org.freedesktop.systemd1.Manager",
3963                 *property = "Environment";
3964
3965         pager_open_if_enabled();
3966
3967         r = bus_method_call_with_reply(
3968                         bus,
3969                         "org.freedesktop.systemd1",
3970                         "/org/freedesktop/systemd1",
3971                         "org.freedesktop.DBus.Properties",
3972                         "Get",
3973                         &reply,
3974                         NULL,
3975                         DBUS_TYPE_STRING, &interface,
3976                         DBUS_TYPE_STRING, &property,
3977                         DBUS_TYPE_INVALID);
3978         if (r < 0)
3979                 return r;
3980
3981         if (!dbus_message_iter_init(reply, &iter) ||
3982             dbus_message_iter_get_arg_type(&iter) != DBUS_TYPE_VARIANT)  {
3983                 log_error("Failed to parse reply.");
3984                 return -EIO;
3985         }
3986
3987         dbus_message_iter_recurse(&iter, &sub);
3988
3989         if (dbus_message_iter_get_arg_type(&sub) != DBUS_TYPE_ARRAY ||
3990             dbus_message_iter_get_element_type(&sub) != DBUS_TYPE_STRING)  {
3991                 log_error("Failed to parse reply.");
3992                 return -EIO;
3993         }
3994
3995         dbus_message_iter_recurse(&sub, &sub2);
3996
3997         while (dbus_message_iter_get_arg_type(&sub2) != DBUS_TYPE_INVALID) {
3998                 const char *text;
3999
4000                 if (dbus_message_iter_get_arg_type(&sub2) != DBUS_TYPE_STRING) {
4001                         log_error("Failed to parse reply.");
4002                         return -EIO;
4003                 }
4004
4005                 dbus_message_iter_get_basic(&sub2, &text);
4006                 puts(text);
4007
4008                 dbus_message_iter_next(&sub2);
4009         }
4010
4011         return 0;
4012 }
4013
4014 static int switch_root(DBusConnection *bus, char **args) {
4015         unsigned l;
4016         const char *root;
4017         _cleanup_free_ char *init = NULL;
4018
4019         l = strv_length(args);
4020         if (l < 2 || l > 3) {
4021                 log_error("Wrong number of arguments.");
4022                 return -EINVAL;
4023         }
4024
4025         root = args[1];
4026
4027         if (l >= 3)
4028                 init = strdup(args[2]);
4029         else {
4030                 parse_env_file("/proc/cmdline", WHITESPACE,
4031                                "init", &init,
4032                                NULL);
4033
4034                 if (!init)
4035                         init = strdup("");
4036         }
4037         if (!init)
4038                 return log_oom();
4039
4040         log_debug("switching root - root: %s; init: %s", root, init);
4041
4042         return bus_method_call_with_reply(
4043                         bus,
4044                         "org.freedesktop.systemd1",
4045                         "/org/freedesktop/systemd1",
4046                         "org.freedesktop.systemd1.Manager",
4047                         "SwitchRoot",
4048                         NULL,
4049                         NULL,
4050                         DBUS_TYPE_STRING, &root,
4051                         DBUS_TYPE_STRING, &init,
4052                         DBUS_TYPE_INVALID);
4053 }
4054
4055 static int set_environment(DBusConnection *bus, char **args) {
4056         _cleanup_dbus_message_unref_ DBusMessage *m = NULL, *reply = NULL;
4057         DBusError error;
4058         const char *method;
4059         DBusMessageIter iter;
4060         int r;
4061
4062         assert(bus);
4063         assert(args);
4064
4065         dbus_error_init(&error);
4066
4067         method = streq(args[0], "set-environment")
4068                 ? "SetEnvironment"
4069                 : "UnsetEnvironment";
4070
4071         m = dbus_message_new_method_call(
4072                         "org.freedesktop.systemd1",
4073                         "/org/freedesktop/systemd1",
4074                         "org.freedesktop.systemd1.Manager",
4075                         method);
4076         if (!m)
4077                 return log_oom();
4078
4079         dbus_message_iter_init_append(m, &iter);
4080
4081         r = bus_append_strv_iter(&iter, args + 1);
4082         if (r < 0)
4083                 return log_oom();
4084
4085         reply = dbus_connection_send_with_reply_and_block(bus, m, -1, &error);
4086         if (!reply) {
4087                 log_error("Failed to issue method call: %s", bus_error_message(&error));
4088                 dbus_error_free(&error);
4089                 return -EIO;
4090         }
4091
4092         return 0;
4093 }
4094
4095 static int enable_sysv_units(char **args) {
4096         int r = 0;
4097
4098 #if defined(HAVE_SYSV_COMPAT) && defined(HAVE_CHKCONFIG)
4099         const char *verb = args[0];
4100         unsigned f = 1, t = 1;
4101         LookupPaths paths = {};
4102
4103         if (arg_scope != UNIT_FILE_SYSTEM)
4104                 return 0;
4105
4106         if (!streq(verb, "enable") &&
4107             !streq(verb, "disable") &&
4108             !streq(verb, "is-enabled"))
4109                 return 0;
4110
4111         /* Processes all SysV units, and reshuffles the array so that
4112          * afterwards only the native units remain */
4113
4114         r = lookup_paths_init(&paths, SYSTEMD_SYSTEM, false, NULL, NULL, NULL);
4115         if (r < 0)
4116                 return r;
4117
4118         r = 0;
4119         for (f = 1; args[f]; f++) {
4120                 const char *name;
4121                 _cleanup_free_ char *p = NULL, *q = NULL;
4122                 bool found_native = false, found_sysv;
4123                 unsigned c = 1;
4124                 const char *argv[6] = { "/sbin/chkconfig", NULL, NULL, NULL, NULL };
4125                 char **k, *l;
4126                 int j;
4127                 pid_t pid;
4128                 siginfo_t status;
4129
4130                 name = args[f];
4131
4132                 if (!endswith(name, ".service"))
4133                         continue;
4134
4135                 if (path_is_absolute(name))
4136                         continue;
4137
4138                 STRV_FOREACH(k, paths.unit_path) {
4139                         if (!isempty(arg_root))
4140                                 asprintf(&p, "%s/%s/%s", arg_root, *k, name);
4141                         else
4142                                 asprintf(&p, "%s/%s", *k, name);
4143
4144                         if (!p) {
4145                                 r = log_oom();
4146                                 goto finish;
4147                         }
4148
4149                         found_native = access(p, F_OK) >= 0;
4150                         free(p);
4151                         p = NULL;
4152
4153                         if (found_native)
4154                                 break;
4155                 }
4156
4157                 if (found_native)
4158                         continue;
4159
4160                 if (!isempty(arg_root))
4161                         asprintf(&p, "%s/" SYSTEM_SYSVINIT_PATH "/%s", arg_root, name);
4162                 else
4163                         asprintf(&p, SYSTEM_SYSVINIT_PATH "/%s", name);
4164                 if (!p) {
4165                         r = log_oom();
4166                         goto finish;
4167                 }
4168
4169                 p[strlen(p) - sizeof(".service") + 1] = 0;
4170                 found_sysv = access(p, F_OK) >= 0;
4171
4172                 if (!found_sysv)
4173                         continue;
4174
4175                 /* Mark this entry, so that we don't try enabling it as native unit */
4176                 args[f] = (char*) "";
4177
4178                 log_info("%s is not a native service, redirecting to /sbin/chkconfig.", name);
4179
4180                 if (!isempty(arg_root))
4181                         argv[c++] = q = strappend("--root=", arg_root);
4182
4183                 argv[c++] = path_get_file_name(p);
4184                 argv[c++] =
4185                         streq(verb, "enable") ? "on" :
4186                         streq(verb, "disable") ? "off" : "--level=5";
4187                 argv[c] = NULL;
4188
4189                 l = strv_join((char**)argv, " ");
4190                 if (!l) {
4191                         r = log_oom();
4192                         goto finish;
4193                 }
4194
4195                 log_info("Executing %s", l);
4196                 free(l);
4197
4198                 pid = fork();
4199                 if (pid < 0) {
4200                         log_error("Failed to fork: %m");
4201                         r = -errno;
4202                         goto finish;
4203                 } else if (pid == 0) {
4204                         /* Child */
4205
4206                         execv(argv[0], (char**) argv);
4207                         _exit(EXIT_FAILURE);
4208                 }
4209
4210                 j = wait_for_terminate(pid, &status);
4211                 if (j < 0) {
4212                         log_error("Failed to wait for child: %s", strerror(-r));
4213                         r = j;
4214                         goto finish;
4215                 }
4216
4217                 if (status.si_code == CLD_EXITED) {
4218                         if (streq(verb, "is-enabled")) {
4219                                 if (status.si_status == 0) {
4220                                         if (!arg_quiet)
4221                                                 puts("enabled");
4222                                         r = 1;
4223                                 } else {
4224                                         if (!arg_quiet)
4225                                                 puts("disabled");
4226                                 }
4227
4228                         } else if (status.si_status != 0) {
4229                                 r = -EINVAL;
4230                                 goto finish;
4231                         }
4232                 } else {
4233                         r = -EPROTO;
4234                         goto finish;
4235                 }
4236         }
4237
4238 finish:
4239         lookup_paths_free(&paths);
4240
4241         /* Drop all SysV units */
4242         for (f = 1, t = 1; args[f]; f++) {
4243
4244                 if (isempty(args[f]))
4245                         continue;
4246
4247                 args[t++] = args[f];
4248         }
4249
4250         args[t] = NULL;
4251
4252 #endif
4253         return r;
4254 }
4255
4256 static int mangle_names(char **original_names, char ***mangled_names) {
4257         char **i, **l, **name;
4258
4259         l = new(char*, strv_length(original_names) + 1);
4260         if (!l)
4261                 return log_oom();
4262
4263         i = l;
4264         STRV_FOREACH(name, original_names) {
4265
4266                 /* When enabling units qualified path names are OK,
4267                  * too, hence allow them explicitly. */
4268
4269                 if (is_path(*name))
4270                         *i = strdup(*name);
4271                 else
4272                         *i = unit_name_mangle(*name);
4273
4274                 if (!*i) {
4275                         strv_free(l);
4276                         return log_oom();
4277                 }
4278
4279                 i++;
4280         }
4281
4282         *i = NULL;
4283         *mangled_names = l;
4284
4285         return 0;
4286 }
4287
4288 static int enable_unit(DBusConnection *bus, char **args) {
4289         const char *verb = args[0];
4290         UnitFileChange *changes = NULL;
4291         unsigned n_changes = 0, i;
4292         int carries_install_info = -1;
4293         _cleanup_dbus_message_unref_ DBusMessage *m = NULL, *reply = NULL;
4294         int r;
4295         _cleanup_dbus_error_free_ DBusError error;
4296         _cleanup_strv_free_ char **mangled_names = NULL;
4297
4298         dbus_error_init(&error);
4299
4300         r = enable_sysv_units(args);
4301         if (r < 0)
4302                 return r;
4303
4304         if (!args[1])
4305                 return 0;
4306
4307         r = mangle_names(args+1, &mangled_names);
4308         if (r < 0)
4309                 goto finish;
4310
4311         if (!bus || avoid_bus()) {
4312                 if (streq(verb, "enable")) {
4313                         r = unit_file_enable(arg_scope, arg_runtime, arg_root, mangled_names, arg_force, &changes, &n_changes);
4314                         carries_install_info = r;
4315                 } else if (streq(verb, "disable"))
4316                         r = unit_file_disable(arg_scope, arg_runtime, arg_root, mangled_names, &changes, &n_changes);
4317                 else if (streq(verb, "reenable")) {
4318                         r = unit_file_reenable(arg_scope, arg_runtime, arg_root, mangled_names, arg_force, &changes, &n_changes);
4319                         carries_install_info = r;
4320                 } else if (streq(verb, "link"))
4321                         r = unit_file_link(arg_scope, arg_runtime, arg_root, mangled_names, arg_force, &changes, &n_changes);
4322                 else if (streq(verb, "preset")) {
4323                         r = unit_file_preset(arg_scope, arg_runtime, arg_root, mangled_names, arg_force, &changes, &n_changes);
4324                         carries_install_info = r;
4325                 } else if (streq(verb, "mask"))
4326                         r = unit_file_mask(arg_scope, arg_runtime, arg_root, mangled_names, arg_force, &changes, &n_changes);
4327                 else if (streq(verb, "unmask"))
4328                         r = unit_file_unmask(arg_scope, arg_runtime, arg_root, mangled_names, &changes, &n_changes);
4329                 else if (streq(verb, "set-default"))
4330                         r = unit_file_set_default(arg_scope, arg_root, args[1], &changes, &n_changes);
4331                 else
4332                         assert_not_reached("Unknown verb");
4333
4334                 if (r < 0) {
4335                         log_error("Operation failed: %s", strerror(-r));
4336                         goto finish;
4337                 }
4338
4339                 if (!arg_quiet) {
4340                         for (i = 0; i < n_changes; i++) {
4341                                 if (changes[i].type == UNIT_FILE_SYMLINK)
4342                                         log_info("ln -s '%s' '%s'", changes[i].source, changes[i].path);
4343                                 else
4344                                         log_info("rm '%s'", changes[i].path);
4345                         }
4346                 }
4347
4348                 r = 0;
4349         } else {
4350                 const char *method;
4351                 bool send_force = true, expect_carries_install_info = false;
4352                 dbus_bool_t a, b;
4353                 DBusMessageIter iter, sub, sub2;
4354
4355                 if (streq(verb, "enable")) {
4356                         method = "EnableUnitFiles";
4357                         expect_carries_install_info = true;
4358                 } else if (streq(verb, "disable")) {
4359                         method = "DisableUnitFiles";
4360                         send_force = false;
4361                 } else if (streq(verb, "reenable")) {
4362                         method = "ReenableUnitFiles";
4363                         expect_carries_install_info = true;
4364                 } else if (streq(verb, "link"))
4365                         method = "LinkUnitFiles";
4366                 else if (streq(verb, "preset")) {
4367                         method = "PresetUnitFiles";
4368                         expect_carries_install_info = true;
4369                 } else if (streq(verb, "mask"))
4370                         method = "MaskUnitFiles";
4371                 else if (streq(verb, "unmask")) {
4372                         method = "UnmaskUnitFiles";
4373                         send_force = false;
4374                 } else if (streq(verb, "set-default")) {
4375                         method = "SetDefaultTarget";
4376                 } else
4377                         assert_not_reached("Unknown verb");
4378
4379                 m = dbus_message_new_method_call(
4380                                 "org.freedesktop.systemd1",
4381                                 "/org/freedesktop/systemd1",
4382                                 "org.freedesktop.systemd1.Manager",
4383                                 method);
4384                 if (!m) {
4385                         r = log_oom();
4386                         goto finish;
4387                 }
4388
4389                 dbus_message_iter_init_append(m, &iter);
4390
4391                 r = bus_append_strv_iter(&iter, mangled_names);
4392                 if (r < 0) {
4393                         log_error("Failed to append unit files.");
4394                         goto finish;
4395                 }
4396
4397                 a = arg_runtime;
4398                 if (!dbus_message_iter_append_basic(&iter, DBUS_TYPE_BOOLEAN, &a)) {
4399                         log_error("Failed to append runtime boolean.");
4400                         r = -ENOMEM;
4401                         goto finish;
4402                 }
4403
4404                 if (send_force) {
4405                         b = arg_force;
4406
4407                         if (!dbus_message_iter_append_basic(&iter, DBUS_TYPE_BOOLEAN, &b)) {
4408                                 log_error("Failed to append force boolean.");
4409                                 r = -ENOMEM;
4410                                 goto finish;
4411                         }
4412                 }
4413
4414                 reply = dbus_connection_send_with_reply_and_block(bus, m, -1, &error);
4415                 if (!reply) {
4416                         log_error("Failed to issue method call: %s", bus_error_message(&error));
4417                         r = -EIO;
4418                         goto finish;
4419                 }
4420
4421                 if (!dbus_message_iter_init(reply, &iter)) {
4422                         log_error("Failed to initialize iterator.");
4423                         goto finish;
4424                 }
4425
4426                 if (expect_carries_install_info) {
4427                         r = bus_iter_get_basic_and_next(&iter, DBUS_TYPE_BOOLEAN, &b, true);
4428                         if (r < 0) {
4429                                 log_error("Failed to parse reply.");
4430                                 goto finish;
4431                         }
4432
4433                         carries_install_info = b;
4434                 }
4435
4436                 if (dbus_message_iter_get_arg_type(&iter) != DBUS_TYPE_ARRAY ||
4437                     dbus_message_iter_get_element_type(&iter) != DBUS_TYPE_STRUCT)  {
4438                         log_error("Failed to parse reply.");
4439                         r = -EIO;
4440                         goto finish;
4441                 }
4442
4443                 dbus_message_iter_recurse(&iter, &sub);
4444                 while (dbus_message_iter_get_arg_type(&sub) != DBUS_TYPE_INVALID) {
4445                         const char *type, *path, *source;
4446
4447                         if (dbus_message_iter_get_arg_type(&sub) != DBUS_TYPE_STRUCT) {
4448                                 log_error("Failed to parse reply.");
4449                                 r = -EIO;
4450                                 goto finish;
4451                         }
4452
4453                         dbus_message_iter_recurse(&sub, &sub2);
4454
4455                         if (bus_iter_get_basic_and_next(&sub2, DBUS_TYPE_STRING, &type, true) < 0 ||
4456                             bus_iter_get_basic_and_next(&sub2, DBUS_TYPE_STRING, &path, true) < 0 ||
4457                             bus_iter_get_basic_and_next(&sub2, DBUS_TYPE_STRING, &source, false) < 0) {
4458                                 log_error("Failed to parse reply.");
4459                                 r = -EIO;
4460                                 goto finish;
4461                         }
4462
4463                         if (!arg_quiet) {
4464                                 if (streq(type, "symlink"))
4465                                         log_info("ln -s '%s' '%s'", source, path);
4466                                 else
4467                                         log_info("rm '%s'", path);
4468                         }
4469
4470                         dbus_message_iter_next(&sub);
4471                 }
4472
4473                 /* Try to reload if enabeld */
4474                 if (!arg_no_reload)
4475                         r = daemon_reload(bus, args);
4476         }
4477
4478         if (carries_install_info == 0)
4479                 log_warning("The unit files have no [Install] section. They are not meant to be enabled\n"
4480                             "using systemctl.\n"
4481                             "Possible reasons for having this kind of units are:\n"
4482                             "1) A unit may be statically enabled by being symlinked from another unit's\n"
4483                             "   .wants/ or .requires/ directory.\n"
4484                             "2) A unit's purpose may be to act as a helper for some other unit which has\n"
4485                             "   a requirement dependency on it.\n"
4486                             "3) A unit may be started when needed via activation (socket, path, timer,\n"
4487                             "   D-Bus, udev, scripted systemctl call, ...).\n");
4488
4489 finish:
4490         unit_file_changes_free(changes, n_changes);
4491
4492         return r;
4493 }
4494
4495 static int unit_is_enabled(DBusConnection *bus, char **args) {
4496         _cleanup_dbus_error_free_ DBusError error;
4497         int r;
4498         _cleanup_dbus_message_unref_ DBusMessage *reply = NULL;
4499         bool enabled;
4500         char **name;
4501         char *n;
4502
4503         dbus_error_init(&error);
4504
4505         r = enable_sysv_units(args);
4506         if (r < 0)
4507                 return r;
4508
4509         enabled = r > 0;
4510
4511         if (!bus || avoid_bus()) {
4512
4513                 STRV_FOREACH(name, args+1) {
4514                         UnitFileState state;
4515
4516                         n = unit_name_mangle(*name);
4517                         if (!n)
4518                                 return log_oom();
4519
4520                         state = unit_file_get_state(arg_scope, arg_root, n);
4521
4522                         free(n);
4523
4524                         if (state < 0)
4525                                 return state;
4526
4527                         if (state == UNIT_FILE_ENABLED ||
4528                             state == UNIT_FILE_ENABLED_RUNTIME ||
4529                             state == UNIT_FILE_STATIC)
4530                                 enabled = true;
4531
4532                         if (!arg_quiet)
4533                                 puts(unit_file_state_to_string(state));
4534                 }
4535
4536         } else {
4537                 STRV_FOREACH(name, args+1) {
4538                         const char *s;
4539
4540                         n = unit_name_mangle(*name);
4541                         if (!n)
4542                                 return log_oom();
4543
4544                         r = bus_method_call_with_reply (
4545                                         bus,
4546                                         "org.freedesktop.systemd1",
4547                                         "/org/freedesktop/systemd1",
4548                                         "org.freedesktop.systemd1.Manager",
4549                                         "GetUnitFileState",
4550                                         &reply,
4551                                         NULL,
4552                                         DBUS_TYPE_STRING, &n,
4553                                         DBUS_TYPE_INVALID);
4554
4555                         free(n);
4556
4557                         if (r)
4558                                 return r;
4559
4560                         if (!dbus_message_get_args(reply, &error,
4561                                                    DBUS_TYPE_STRING, &s,
4562                                                    DBUS_TYPE_INVALID)) {
4563                                 log_error("Failed to parse reply: %s", bus_error_message(&error));
4564                                 return -EIO;
4565                         }
4566
4567                         dbus_message_unref(reply);
4568                         reply = NULL;
4569
4570                         if (streq(s, "enabled") ||
4571                             streq(s, "enabled-runtime") ||
4572                             streq(s, "static"))
4573                                 enabled = true;
4574
4575                         if (!arg_quiet)
4576                                 puts(s);
4577                 }
4578         }
4579
4580         return enabled ? 0 : 1;
4581 }
4582
4583 static int systemctl_help(void) {
4584
4585         pager_open_if_enabled();
4586
4587         printf("%s [OPTIONS...] {COMMAND} ...\n\n"
4588                "Query or send control commands to the systemd manager.\n\n"
4589                "  -h --help           Show this help\n"
4590                "     --version        Show package version\n"
4591                "  -t --type=TYPE      List only units of a particular type\n"
4592                "     --state=STATE    List only units with particular LOAD or SUB or ACTIVE state\n"
4593                "  -p --property=NAME  Show only properties by this name\n"
4594                "  -a --all            Show all loaded units/properties, including dead/empty\n"
4595                "                      ones. To list all units installed on the system, use\n"
4596                "                      the 'list-unit-files' command instead.\n"
4597                "     --reverse        Show reverse dependencies with 'list-dependencies'\n"
4598                "  -l --full           Don't ellipsize unit names on output\n"
4599                "     --fail           When queueing a new job, fail if conflicting jobs are\n"
4600                "                      pending\n"
4601                "     --irreversible   When queueing a new job, make sure it cannot be implicitly\n"
4602                "                      cancelled\n"
4603                "     --ignore-dependencies\n"
4604                "                      When queueing a new job, ignore all its dependencies\n"
4605                "     --show-types     When showing sockets, explicitly show their type\n"
4606                "  -i --ignore-inhibitors\n"
4607                "                      When shutting down or sleeping, ignore inhibitors\n"
4608                "     --kill-who=WHO   Who to send signal to\n"
4609                "  -s --signal=SIGNAL  Which signal to send\n"
4610                "  -H --host=[USER@]HOST\n"
4611                "                      Show information for remote host\n"
4612                "  -P --privileged     Acquire privileges before execution\n"
4613                "  -q --quiet          Suppress output\n"
4614                "     --no-block       Do not wait until operation finished\n"
4615                "     --no-wall        Don't send wall message before halt/power-off/reboot\n"
4616                "     --no-reload      When enabling/disabling unit files, don't reload daemon\n"
4617                "                      configuration\n"
4618                "     --no-legend      Do not print a legend (column headers and hints)\n"
4619                "     --no-pager       Do not pipe output into a pager\n"
4620                "     --no-ask-password\n"
4621                "                      Do not ask for system passwords\n"
4622                "     --system         Connect to system manager\n"
4623                "     --user           Connect to user service manager\n"
4624                "     --global         Enable/disable unit files globally\n"
4625                "     --runtime        Enable unit files only temporarily until next reboot\n"
4626                "  -f --force          When enabling unit files, override existing symlinks\n"
4627                "                      When shutting down, execute action immediately\n"
4628                "     --root=PATH      Enable unit files in the specified root directory\n"
4629                "  -n --lines=INTEGER  Numer of journal entries to show\n"
4630                "  -o --output=STRING  Change journal output mode (short, short-monotonic,\n"
4631                "                      verbose, export, json, json-pretty, json-sse, cat)\n\n"
4632                "Unit Commands:\n"
4633                "  list-units                      List loaded units\n"
4634                "  list-sockets                    List loaded sockets ordered by address\n"
4635                "  start [NAME...]                 Start (activate) one or more units\n"
4636                "  stop [NAME...]                  Stop (deactivate) one or more units\n"
4637                "  reload [NAME...]                Reload one or more units\n"
4638                "  restart [NAME...]               Start or restart one or more units\n"
4639                "  try-restart [NAME...]           Restart one or more units if active\n"
4640                "  reload-or-restart [NAME...]     Reload one or more units if possible,\n"
4641                "                                  otherwise start or restart\n"
4642                "  reload-or-try-restart [NAME...] Reload one or more units if possible,\n"
4643                "                                  otherwise restart if active\n"
4644                "  isolate [NAME]                  Start one unit and stop all others\n"
4645                "  kill [NAME...]                  Send signal to processes of a unit\n"
4646                "  is-active [NAME...]             Check whether units are active\n"
4647                "  is-failed [NAME...]             Check whether units are failed\n"
4648                "  status [NAME...|PID...]         Show runtime status of one or more units\n"
4649                "  show [NAME...|JOB...]           Show properties of one or more\n"
4650                "                                  units/jobs or the manager\n"
4651                "  set-property [NAME] [ASSIGNMENT...]\n"
4652                "                                  Sets one or more properties of a unit\n"
4653                "  help [NAME...|PID...]           Show manual for one or more units\n"
4654                "  reset-failed [NAME...]          Reset failed state for all, one, or more\n"
4655                "                                  units\n"
4656                "  list-dependencies [NAME]        Recursively show units which are required\n"
4657                "                                  or wanted by this unit or by which this\n"
4658                "                                  unit is required or wanted\n\n"
4659                "Unit File Commands:\n"
4660                "  list-unit-files                 List installed unit files\n"
4661                "  enable [NAME...]                Enable one or more unit files\n"
4662                "  disable [NAME...]               Disable one or more unit files\n"
4663                "  reenable [NAME...]              Reenable one or more unit files\n"
4664                "  preset [NAME...]                Enable/disable one or more unit files\n"
4665                "                                  based on preset configuration\n"
4666                "  is-enabled [NAME...]            Check whether unit files are enabled\n\n"
4667                "  mask [NAME...]                  Mask one or more units\n"
4668                "  unmask [NAME...]                Unmask one or more units\n"
4669                "  link [PATH...]                  Link one or more units files into\n"
4670                "                                  the search path\n"
4671                "  get-default                     Get the name of the default target\n"
4672                "  set-default NAME                Set the default target\n\n"
4673                "Job Commands:\n"
4674                "  list-jobs                       List jobs\n"
4675                "  cancel [JOB...]                 Cancel all, one, or more jobs\n\n"
4676                "Snapshot Commands:\n"
4677                "  snapshot [NAME]                 Create a snapshot\n"
4678                "  delete [NAME...]                Remove one or more snapshots\n\n"
4679                "Environment Commands:\n"
4680                "  show-environment                Dump environment\n"
4681                "  set-environment [NAME=VALUE...] Set one or more environment variables\n"
4682                "  unset-environment [NAME...]     Unset one or more environment variables\n\n"
4683                "Manager Lifecycle Commands:\n"
4684                "  daemon-reload                   Reload systemd manager configuration\n"
4685                "  daemon-reexec                   Reexecute systemd manager\n\n"
4686                "System Commands:\n"
4687                "  default                         Enter system default mode\n"
4688                "  rescue                          Enter system rescue mode\n"
4689                "  emergency                       Enter system emergency mode\n"
4690                "  halt                            Shut down and halt the system\n"
4691                "  poweroff                        Shut down and power-off the system\n"
4692                "  reboot                          Shut down and reboot the system\n"
4693                "  kexec                           Shut down and reboot the system with kexec\n"
4694                "  exit                            Request user instance exit\n"
4695                "  switch-root [ROOT] [INIT]       Change to a different root file system\n"
4696                "  suspend                         Suspend the system\n"
4697                "  hibernate                       Hibernate the system\n"
4698                "  hybrid-sleep                    Hibernate and suspend the system\n",
4699                program_invocation_short_name);
4700
4701         return 0;
4702 }
4703
4704 static int halt_help(void) {
4705
4706         printf("%s [OPTIONS...]\n\n"
4707                "%s the system.\n\n"
4708                "     --help      Show this help\n"
4709                "     --halt      Halt the machine\n"
4710                "  -p --poweroff  Switch off the machine\n"
4711                "     --reboot    Reboot the machine\n"
4712                "  -f --force     Force immediate halt/power-off/reboot\n"
4713                "  -w --wtmp-only Don't halt/power-off/reboot, just write wtmp record\n"
4714                "  -d --no-wtmp   Don't write wtmp record\n"
4715                "     --no-wall   Don't send wall message before halt/power-off/reboot\n",
4716                program_invocation_short_name,
4717                arg_action == ACTION_REBOOT   ? "Reboot" :
4718                arg_action == ACTION_POWEROFF ? "Power off" :
4719                                                "Halt");
4720
4721         return 0;
4722 }
4723
4724 static int shutdown_help(void) {
4725
4726         printf("%s [OPTIONS...] [TIME] [WALL...]\n\n"
4727                "Shut down the system.\n\n"
4728                "     --help      Show this help\n"
4729                "  -H --halt      Halt the machine\n"
4730                "  -P --poweroff  Power-off the machine\n"
4731                "  -r --reboot    Reboot the machine\n"
4732                "  -h             Equivalent to --poweroff, overridden by --halt\n"
4733                "  -k             Don't halt/power-off/reboot, just send warnings\n"
4734                "     --no-wall   Don't send wall message before halt/power-off/reboot\n"
4735                "  -c             Cancel a pending shutdown\n",
4736                program_invocation_short_name);
4737
4738         return 0;
4739 }
4740
4741 static int telinit_help(void) {
4742
4743         printf("%s [OPTIONS...] {COMMAND}\n\n"
4744                "Send control commands to the init daemon.\n\n"
4745                "     --help      Show this help\n"
4746                "     --no-wall   Don't send wall message before halt/power-off/reboot\n\n"
4747                "Commands:\n"
4748                "  0              Power-off the machine\n"
4749                "  6              Reboot the machine\n"
4750                "  2, 3, 4, 5     Start runlevelX.target unit\n"
4751                "  1, s, S        Enter rescue mode\n"
4752                "  q, Q           Reload init daemon configuration\n"
4753                "  u, U           Reexecute init daemon\n",
4754                program_invocation_short_name);
4755
4756         return 0;
4757 }
4758
4759 static int runlevel_help(void) {
4760
4761         printf("%s [OPTIONS...]\n\n"
4762                "Prints the previous and current runlevel of the init system.\n\n"
4763                "     --help      Show this help\n",
4764                program_invocation_short_name);
4765
4766         return 0;
4767 }
4768
4769 static int help_types(void) {
4770         int i;
4771         const char *t;
4772
4773         puts("Available unit types:");
4774         for(i = 0; i < _UNIT_TYPE_MAX; i++) {
4775                 t = unit_type_to_string(i);
4776                 if (t)
4777                         puts(t);
4778         }
4779
4780         return 0;
4781 }
4782
4783 static int systemctl_parse_argv(int argc, char *argv[]) {
4784
4785         enum {
4786                 ARG_FAIL = 0x100,
4787                 ARG_REVERSE,
4788                 ARG_AFTER,
4789                 ARG_BEFORE,
4790                 ARG_SHOW_TYPES,
4791                 ARG_IRREVERSIBLE,
4792                 ARG_IGNORE_DEPENDENCIES,
4793                 ARG_VERSION,
4794                 ARG_USER,
4795                 ARG_SYSTEM,
4796                 ARG_GLOBAL,
4797                 ARG_NO_BLOCK,
4798                 ARG_NO_LEGEND,
4799                 ARG_NO_PAGER,
4800                 ARG_NO_WALL,
4801                 ARG_ROOT,
4802                 ARG_NO_RELOAD,
4803                 ARG_KILL_WHO,
4804                 ARG_NO_ASK_PASSWORD,
4805                 ARG_FAILED,
4806                 ARG_RUNTIME,
4807                 ARG_FORCE,
4808                 ARG_PLAIN,
4809                 ARG_STATE
4810         };
4811
4812         static const struct option options[] = {
4813                 { "help",                no_argument,       NULL, 'h'                     },
4814                 { "version",             no_argument,       NULL, ARG_VERSION             },
4815                 { "type",                required_argument, NULL, 't'                     },
4816                 { "property",            required_argument, NULL, 'p'                     },
4817                 { "all",                 no_argument,       NULL, 'a'                     },
4818                 { "reverse",             no_argument,       NULL, ARG_REVERSE             },
4819                 { "after",               no_argument,       NULL, ARG_AFTER               },
4820                 { "before",              no_argument,       NULL, ARG_BEFORE              },
4821                 { "show-types",          no_argument,       NULL, ARG_SHOW_TYPES          },
4822                 { "failed",              no_argument,       NULL, ARG_FAILED              }, /* compatibility only */
4823                 { "full",                no_argument,       NULL, 'l'                     },
4824                 { "fail",                no_argument,       NULL, ARG_FAIL                },
4825                 { "irreversible",        no_argument,       NULL, ARG_IRREVERSIBLE        },
4826                 { "ignore-dependencies", no_argument,       NULL, ARG_IGNORE_DEPENDENCIES },
4827                 { "ignore-inhibitors",   no_argument,       NULL, 'i'                     },
4828                 { "user",                no_argument,       NULL, ARG_USER                },
4829                 { "system",              no_argument,       NULL, ARG_SYSTEM              },
4830                 { "global",              no_argument,       NULL, ARG_GLOBAL              },
4831                 { "no-block",            no_argument,       NULL, ARG_NO_BLOCK            },
4832                 { "no-legend",           no_argument,       NULL, ARG_NO_LEGEND           },
4833                 { "no-pager",            no_argument,       NULL, ARG_NO_PAGER            },
4834                 { "no-wall",             no_argument,       NULL, ARG_NO_WALL             },
4835                 { "quiet",               no_argument,       NULL, 'q'                     },
4836                 { "root",                required_argument, NULL, ARG_ROOT                },
4837                 { "force",               no_argument,       NULL, ARG_FORCE               },
4838                 { "no-reload",           no_argument,       NULL, ARG_NO_RELOAD           },
4839                 { "kill-who",            required_argument, NULL, ARG_KILL_WHO            },
4840                 { "signal",              required_argument, NULL, 's'                     },
4841                 { "no-ask-password",     no_argument,       NULL, ARG_NO_ASK_PASSWORD     },
4842                 { "host",                required_argument, NULL, 'H'                     },
4843                 { "privileged",          no_argument,       NULL, 'P'                     },
4844                 { "runtime",             no_argument,       NULL, ARG_RUNTIME             },
4845                 { "lines",               required_argument, NULL, 'n'                     },
4846                 { "output",              required_argument, NULL, 'o'                     },
4847                 { "plain",               no_argument,       NULL, ARG_PLAIN               },
4848                 { "state",               required_argument, NULL, ARG_STATE               },
4849                 { NULL,                  0,                 NULL, 0                       }
4850         };
4851
4852         int c;
4853
4854         assert(argc >= 0);
4855         assert(argv);
4856
4857         while ((c = getopt_long(argc, argv, "ht:p:alqfs:H:Pn:o:i", options, NULL)) >= 0) {
4858
4859                 switch (c) {
4860
4861                 case 'h':
4862                         systemctl_help();
4863                         return 0;
4864
4865                 case ARG_VERSION:
4866                         puts(PACKAGE_STRING);
4867                         puts(SYSTEMD_FEATURES);
4868                         return 0;
4869
4870                 case 't': {
4871                         char *word, *state;
4872                         size_t size;
4873
4874                         FOREACH_WORD_SEPARATOR(word, size, optarg, ",", state) {
4875                                 _cleanup_free_ char *type;
4876
4877                                 type = strndup(word, size);
4878                                 if (!type)
4879                                         return -ENOMEM;
4880
4881                                 if (streq(type, "help")) {
4882                                         help_types();
4883                                         return 0;
4884                                 }
4885
4886                                 if (unit_type_from_string(type) >= 0) {
4887                                         if (strv_push(&arg_types, type))
4888                                                 return log_oom();
4889                                         type = NULL;
4890                                         continue;
4891                                 }
4892
4893                                 /* It's much nicer to use --state= for
4894                                  * load states, but let's support this
4895                                  * in --types= too for compatibility
4896                                  * with old versions */
4897                                 if (unit_load_state_from_string(optarg) >= 0) {
4898                                         if (strv_push(&arg_states, type) < 0)
4899                                                 return log_oom();
4900                                         type = NULL;
4901                                         continue;
4902                                 }
4903
4904                                 log_error("Unknown unit type or load state '%s'.", type);
4905                                 log_info("Use -t help to see a list of allowed values.");
4906                                 return -EINVAL;
4907                         }
4908
4909                         break;
4910                 }
4911
4912                 case 'p': {
4913                         /* Make sure that if the empty property list
4914                            was specified, we won't show any properties. */
4915                         if (isempty(optarg) && !arg_properties) {
4916                                 arg_properties = strv_new(NULL, NULL);
4917                                 if (!arg_properties)
4918                                         return log_oom();
4919                         } else {
4920                                 char *word, *state;
4921                                 size_t size;
4922
4923                                 FOREACH_WORD_SEPARATOR(word, size, optarg, ",", state) {
4924                                         char *prop;
4925
4926                                         prop = strndup(word, size);
4927                                         if (!prop)
4928                                                 return log_oom();
4929
4930                                         if (strv_push(&arg_properties, prop) < 0) {
4931                                                 free(prop);
4932                                                 return log_oom();
4933                                         }
4934                                 }
4935                         }
4936
4937                         /* If the user asked for a particular
4938                          * property, show it to him, even if it is
4939                          * empty. */
4940                         arg_all = true;
4941
4942                         break;
4943                 }
4944
4945                 case 'a':
4946                         arg_all = true;
4947                         break;
4948
4949                 case ARG_REVERSE:
4950                         arg_dependency = DEPENDENCY_REVERSE;
4951                         break;
4952
4953                 case ARG_AFTER:
4954                         arg_dependency = DEPENDENCY_AFTER;
4955                         break;
4956
4957                 case ARG_BEFORE:
4958                         arg_dependency = DEPENDENCY_BEFORE;
4959                         break;
4960
4961                 case ARG_SHOW_TYPES:
4962                         arg_show_types = true;
4963                         break;
4964
4965                 case ARG_FAIL:
4966                         arg_job_mode = "fail";
4967                         break;
4968
4969                 case ARG_IRREVERSIBLE:
4970                         arg_job_mode = "replace-irreversibly";
4971                         break;
4972
4973                 case ARG_IGNORE_DEPENDENCIES:
4974                         arg_job_mode = "ignore-dependencies";
4975                         break;
4976
4977                 case ARG_USER:
4978                         arg_scope = UNIT_FILE_USER;
4979                         break;
4980
4981                 case ARG_SYSTEM:
4982                         arg_scope = UNIT_FILE_SYSTEM;
4983                         break;
4984
4985                 case ARG_GLOBAL:
4986                         arg_scope = UNIT_FILE_GLOBAL;
4987                         break;
4988
4989                 case ARG_NO_BLOCK:
4990                         arg_no_block = true;
4991                         break;
4992
4993                 case ARG_NO_LEGEND:
4994                         arg_no_legend = true;
4995                         break;
4996
4997                 case ARG_NO_PAGER:
4998                         arg_no_pager = true;
4999                         break;
5000
5001                 case ARG_NO_WALL:
5002                         arg_no_wall = true;
5003                         break;
5004
5005                 case ARG_ROOT:
5006                         arg_root = optarg;
5007                         break;
5008
5009                 case 'l':
5010                         arg_full = true;
5011                         break;
5012
5013                 case ARG_FAILED:
5014                         if (strv_extend(&arg_states, "failed") < 0)
5015                                 return log_oom();
5016
5017                         break;
5018
5019                 case 'q':
5020                         arg_quiet = true;
5021                         break;
5022
5023                 case ARG_FORCE:
5024                         arg_force ++;
5025                         break;
5026
5027                 case 'f':
5028                         arg_force ++;
5029                         break;
5030
5031                 case ARG_NO_RELOAD:
5032                         arg_no_reload = true;
5033                         break;
5034
5035                 case ARG_KILL_WHO:
5036                         arg_kill_who = optarg;
5037                         break;
5038
5039                 case 's':
5040                         if ((arg_signal = signal_from_string_try_harder(optarg)) < 0) {
5041                                 log_error("Failed to parse signal string %s.", optarg);
5042                                 return -EINVAL;
5043                         }
5044                         break;
5045
5046                 case ARG_NO_ASK_PASSWORD:
5047                         arg_ask_password = false;
5048                         break;
5049
5050                 case 'P':
5051                         arg_transport = TRANSPORT_POLKIT;
5052                         break;
5053
5054                 case 'H':
5055                         arg_transport = TRANSPORT_SSH;
5056                         parse_user_at_host(optarg, &arg_user, &arg_host);
5057                         break;
5058
5059                 case ARG_RUNTIME:
5060                         arg_runtime = true;
5061                         break;
5062
5063                 case 'n':
5064                         if (safe_atou(optarg, &arg_lines) < 0) {
5065                                 log_error("Failed to parse lines '%s'", optarg);
5066                                 return -EINVAL;
5067                         }
5068                         break;
5069
5070                 case 'o':
5071                         arg_output = output_mode_from_string(optarg);
5072                         if (arg_output < 0) {
5073                                 log_error("Unknown output '%s'.", optarg);
5074                                 return -EINVAL;
5075                         }
5076                         break;
5077
5078                 case 'i':
5079                         arg_ignore_inhibitors = true;
5080                         break;
5081
5082                 case ARG_PLAIN:
5083                         arg_plain = true;
5084                         break;
5085
5086                 case ARG_STATE: {
5087                         char *word, *state;
5088                         size_t size;
5089
5090                         FOREACH_WORD_SEPARATOR(word, size, optarg, ",", state) {
5091                                 char *s;
5092
5093                                 s = strndup(word, size);
5094                                 if (!s)
5095                                         return log_oom();
5096
5097                                 if (strv_push(&arg_states, s) < 0) {
5098                                         free(s);
5099                                         return log_oom();
5100                                 }
5101                         }
5102                         break;
5103                 }
5104
5105                 case '?':
5106                         return -EINVAL;
5107
5108                 default:
5109                         log_error("Unknown option code '%c'.", c);
5110                         return -EINVAL;
5111                 }
5112         }
5113
5114         if (arg_transport != TRANSPORT_NORMAL && arg_scope != UNIT_FILE_SYSTEM) {
5115                 log_error("Cannot access user instance remotely.");
5116                 return -EINVAL;
5117         }
5118
5119         return 1;
5120 }
5121
5122 static int halt_parse_argv(int argc, char *argv[]) {
5123
5124         enum {
5125                 ARG_HELP = 0x100,
5126                 ARG_HALT,
5127                 ARG_REBOOT,
5128                 ARG_NO_WALL
5129         };
5130
5131         static const struct option options[] = {
5132                 { "help",      no_argument,       NULL, ARG_HELP    },
5133                 { "halt",      no_argument,       NULL, ARG_HALT    },
5134                 { "poweroff",  no_argument,       NULL, 'p'         },
5135                 { "reboot",    no_argument,       NULL, ARG_REBOOT  },
5136                 { "force",     no_argument,       NULL, 'f'         },
5137                 { "wtmp-only", no_argument,       NULL, 'w'         },
5138                 { "no-wtmp",   no_argument,       NULL, 'd'         },
5139                 { "no-wall",   no_argument,       NULL, ARG_NO_WALL },
5140                 { NULL,        0,                 NULL, 0           }
5141         };
5142
5143         int c, runlevel;
5144
5145         assert(argc >= 0);
5146         assert(argv);
5147
5148         if (utmp_get_runlevel(&runlevel, NULL) >= 0)
5149                 if (runlevel == '0' || runlevel == '6')
5150                         arg_force = 2;
5151
5152         while ((c = getopt_long(argc, argv, "pfwdnih", options, NULL)) >= 0) {
5153                 switch (c) {
5154
5155                 case ARG_HELP:
5156                         halt_help();
5157                         return 0;
5158
5159                 case ARG_HALT:
5160                         arg_action = ACTION_HALT;
5161                         break;
5162
5163                 case 'p':
5164                         if (arg_action != ACTION_REBOOT)
5165                                 arg_action = ACTION_POWEROFF;
5166                         break;
5167
5168                 case ARG_REBOOT:
5169                         arg_action = ACTION_REBOOT;
5170                         break;
5171
5172                 case 'f':
5173                         arg_force = 2;
5174                         break;
5175
5176                 case 'w':
5177                         arg_dry = true;
5178                         break;
5179
5180                 case 'd':
5181                         arg_no_wtmp = true;
5182                         break;
5183
5184                 case ARG_NO_WALL:
5185                         arg_no_wall = true;
5186                         break;
5187
5188                 case 'i':
5189                 case 'h':
5190                 case 'n':
5191                         /* Compatibility nops */
5192                         break;
5193
5194                 case '?':
5195                         return -EINVAL;
5196
5197                 default:
5198                         log_error("Unknown option code '%c'.", c);
5199                         return -EINVAL;
5200                 }
5201         }
5202
5203         if (optind < argc) {
5204                 log_error("Too many arguments.");
5205                 return -EINVAL;
5206         }
5207
5208         return 1;
5209 }
5210
5211 static int parse_time_spec(const char *t, usec_t *_u) {
5212         assert(t);
5213         assert(_u);
5214
5215         if (streq(t, "now"))
5216                 *_u = 0;
5217         else if (!strchr(t, ':')) {
5218                 uint64_t u;
5219
5220                 if (safe_atou64(t, &u) < 0)
5221                         return -EINVAL;
5222
5223                 *_u = now(CLOCK_REALTIME) + USEC_PER_MINUTE * u;
5224         } else {
5225                 char *e = NULL;
5226                 long hour, minute;
5227                 struct tm tm = {};
5228                 time_t s;
5229                 usec_t n;
5230
5231                 errno = 0;
5232                 hour = strtol(t, &e, 10);
5233                 if (errno > 0 || *e != ':' || hour < 0 || hour > 23)
5234                         return -EINVAL;
5235
5236                 minute = strtol(e+1, &e, 10);
5237                 if (errno > 0 || *e != 0 || minute < 0 || minute > 59)
5238                         return -EINVAL;
5239
5240                 n = now(CLOCK_REALTIME);
5241                 s = (time_t) (n / USEC_PER_SEC);
5242
5243                 assert_se(localtime_r(&s, &tm));
5244
5245                 tm.tm_hour = (int) hour;
5246                 tm.tm_min = (int) minute;
5247                 tm.tm_sec = 0;
5248
5249                 assert_se(s = mktime(&tm));
5250
5251                 *_u = (usec_t) s * USEC_PER_SEC;
5252
5253                 while (*_u <= n)
5254                         *_u += USEC_PER_DAY;
5255         }
5256
5257         return 0;
5258 }
5259
5260 static int shutdown_parse_argv(int argc, char *argv[]) {
5261
5262         enum {
5263                 ARG_HELP = 0x100,
5264                 ARG_NO_WALL
5265         };
5266
5267         static const struct option options[] = {
5268                 { "help",      no_argument,       NULL, ARG_HELP    },
5269                 { "halt",      no_argument,       NULL, 'H'         },
5270                 { "poweroff",  no_argument,       NULL, 'P'         },
5271                 { "reboot",    no_argument,       NULL, 'r'         },
5272                 { "kexec",     no_argument,       NULL, 'K'         }, /* not documented extension */
5273                 { "no-wall",   no_argument,       NULL, ARG_NO_WALL },
5274                 { NULL,        0,                 NULL, 0           }
5275         };
5276
5277         int c, r;
5278
5279         assert(argc >= 0);
5280         assert(argv);
5281
5282         while ((c = getopt_long(argc, argv, "HPrhkt:afFc", options, NULL)) >= 0) {
5283                 switch (c) {
5284
5285                 case ARG_HELP:
5286                         shutdown_help();
5287                         return 0;
5288
5289                 case 'H':
5290                         arg_action = ACTION_HALT;
5291                         break;
5292
5293                 case 'P':
5294                         arg_action = ACTION_POWEROFF;
5295                         break;
5296
5297                 case 'r':
5298                         if (kexec_loaded())
5299                                 arg_action = ACTION_KEXEC;
5300                         else
5301                                 arg_action = ACTION_REBOOT;
5302                         break;
5303
5304                 case 'K':
5305                         arg_action = ACTION_KEXEC;
5306                         break;
5307
5308                 case 'h':
5309                         if (arg_action != ACTION_HALT)
5310                                 arg_action = ACTION_POWEROFF;
5311                         break;
5312
5313                 case 'k':
5314                         arg_dry = true;
5315                         break;
5316
5317                 case ARG_NO_WALL:
5318                         arg_no_wall = true;
5319                         break;
5320
5321                 case 't':
5322                 case 'a':
5323                         /* Compatibility nops */
5324                         break;
5325
5326                 case 'c':
5327                         arg_action = ACTION_CANCEL_SHUTDOWN;
5328                         break;
5329
5330                 case '?':
5331                         return -EINVAL;
5332
5333                 default:
5334                         log_error("Unknown option code '%c'.", c);
5335                         return -EINVAL;
5336                 }
5337         }
5338
5339         if (argc > optind && arg_action != ACTION_CANCEL_SHUTDOWN) {
5340                 r = parse_time_spec(argv[optind], &arg_when);
5341                 if (r < 0) {
5342                         log_error("Failed to parse time specification: %s", argv[optind]);
5343                         return r;
5344                 }
5345         } else
5346                 arg_when = now(CLOCK_REALTIME) + USEC_PER_MINUTE;
5347
5348         if (argc > optind && arg_action == ACTION_CANCEL_SHUTDOWN)
5349                 /* No time argument for shutdown cancel */
5350                 arg_wall = argv + optind;
5351         else if (argc > optind + 1)
5352                 /* We skip the time argument */
5353                 arg_wall = argv + optind + 1;
5354
5355         optind = argc;
5356
5357         return 1;
5358 }
5359
5360 static int telinit_parse_argv(int argc, char *argv[]) {
5361
5362         enum {
5363                 ARG_HELP = 0x100,
5364                 ARG_NO_WALL
5365         };
5366
5367         static const struct option options[] = {
5368                 { "help",      no_argument,       NULL, ARG_HELP    },
5369                 { "no-wall",   no_argument,       NULL, ARG_NO_WALL },
5370                 { NULL,        0,                 NULL, 0           }
5371         };
5372
5373         static const struct {
5374                 char from;
5375                 enum action to;
5376         } table[] = {
5377                 { '0', ACTION_POWEROFF },
5378                 { '6', ACTION_REBOOT },
5379                 { '1', ACTION_RESCUE },
5380                 { '2', ACTION_RUNLEVEL2 },
5381                 { '3', ACTION_RUNLEVEL3 },
5382                 { '4', ACTION_RUNLEVEL4 },
5383                 { '5', ACTION_RUNLEVEL5 },
5384                 { 's', ACTION_RESCUE },
5385                 { 'S', ACTION_RESCUE },
5386                 { 'q', ACTION_RELOAD },
5387                 { 'Q', ACTION_RELOAD },
5388                 { 'u', ACTION_REEXEC },
5389                 { 'U', ACTION_REEXEC }
5390         };
5391
5392         unsigned i;
5393         int c;
5394
5395         assert(argc >= 0);
5396         assert(argv);
5397
5398         while ((c = getopt_long(argc, argv, "", options, NULL)) >= 0) {
5399                 switch (c) {
5400
5401                 case ARG_HELP:
5402                         telinit_help();
5403                         return 0;
5404
5405                 case ARG_NO_WALL:
5406                         arg_no_wall = true;
5407                         break;
5408
5409                 case '?':
5410                         return -EINVAL;
5411
5412                 default:
5413                         log_error("Unknown option code '%c'.", c);
5414                         return -EINVAL;
5415                 }
5416         }
5417
5418         if (optind >= argc) {
5419                 telinit_help();
5420                 return -EINVAL;
5421         }
5422
5423         if (optind + 1 < argc) {
5424                 log_error("Too many arguments.");
5425                 return -EINVAL;
5426         }
5427
5428         if (strlen(argv[optind]) != 1) {
5429                 log_error("Expected single character argument.");
5430                 return -EINVAL;
5431         }
5432
5433         for (i = 0; i < ELEMENTSOF(table); i++)
5434                 if (table[i].from == argv[optind][0])
5435                         break;
5436
5437         if (i >= ELEMENTSOF(table)) {
5438                 log_error("Unknown command '%s'.", argv[optind]);
5439                 return -EINVAL;
5440         }
5441
5442         arg_action = table[i].to;
5443
5444         optind ++;
5445
5446         return 1;
5447 }
5448
5449 static int runlevel_parse_argv(int argc, char *argv[]) {
5450
5451         enum {
5452                 ARG_HELP = 0x100,
5453         };
5454
5455         static const struct option options[] = {
5456                 { "help",      no_argument,       NULL, ARG_HELP    },
5457                 { NULL,        0,                 NULL, 0           }
5458         };
5459
5460         int c;
5461
5462         assert(argc >= 0);
5463         assert(argv);
5464
5465         while ((c = getopt_long(argc, argv, "", options, NULL)) >= 0) {
5466                 switch (c) {
5467
5468                 case ARG_HELP:
5469                         runlevel_help();
5470                         return 0;
5471
5472                 case '?':
5473                         return -EINVAL;
5474
5475                 default:
5476                         log_error("Unknown option code '%c'.", c);
5477                         return -EINVAL;
5478                 }
5479         }
5480
5481         if (optind < argc) {
5482                 log_error("Too many arguments.");
5483                 return -EINVAL;
5484         }
5485
5486         return 1;
5487 }
5488
5489 static int parse_argv(int argc, char *argv[]) {
5490         assert(argc >= 0);
5491         assert(argv);
5492
5493         if (program_invocation_short_name) {
5494
5495                 if (strstr(program_invocation_short_name, "halt")) {
5496                         arg_action = ACTION_HALT;
5497                         return halt_parse_argv(argc, argv);
5498                 } else if (strstr(program_invocation_short_name, "poweroff")) {
5499                         arg_action = ACTION_POWEROFF;
5500                         return halt_parse_argv(argc, argv);
5501                 } else if (strstr(program_invocation_short_name, "reboot")) {
5502                         if (kexec_loaded())
5503                                 arg_action = ACTION_KEXEC;
5504                         else
5505                                 arg_action = ACTION_REBOOT;
5506                         return halt_parse_argv(argc, argv);
5507                 } else if (strstr(program_invocation_short_name, "shutdown")) {
5508                         arg_action = ACTION_POWEROFF;
5509                         return shutdown_parse_argv(argc, argv);
5510                 } else if (strstr(program_invocation_short_name, "init")) {
5511
5512                         if (sd_booted() > 0) {
5513                                 arg_action = ACTION_INVALID;
5514                                 return telinit_parse_argv(argc, argv);
5515                         } else {
5516                                 /* Hmm, so some other init system is
5517                                  * running, we need to forward this
5518                                  * request to it. For now we simply
5519                                  * guess that it is Upstart. */
5520
5521                                 execv(TELINIT, argv);
5522
5523                                 log_error("Couldn't find an alternative telinit implementation to spawn.");
5524                                 return -EIO;
5525                         }
5526
5527                 } else if (strstr(program_invocation_short_name, "runlevel")) {
5528                         arg_action = ACTION_RUNLEVEL;
5529                         return runlevel_parse_argv(argc, argv);
5530                 }
5531         }
5532
5533         arg_action = ACTION_SYSTEMCTL;
5534         return systemctl_parse_argv(argc, argv);
5535 }
5536
5537 _pure_ static int action_to_runlevel(void) {
5538
5539         static const char table[_ACTION_MAX] = {
5540                 [ACTION_HALT] =      '0',
5541                 [ACTION_POWEROFF] =  '0',
5542                 [ACTION_REBOOT] =    '6',
5543                 [ACTION_RUNLEVEL2] = '2',
5544                 [ACTION_RUNLEVEL3] = '3',
5545                 [ACTION_RUNLEVEL4] = '4',
5546                 [ACTION_RUNLEVEL5] = '5',
5547                 [ACTION_RESCUE] =    '1'
5548         };
5549
5550         assert(arg_action < _ACTION_MAX);
5551
5552         return table[arg_action];
5553 }
5554
5555 static int talk_upstart(void) {
5556         _cleanup_dbus_message_unref_ DBusMessage *m = NULL, *reply = NULL;
5557         _cleanup_dbus_error_free_ DBusError error;
5558         int previous, rl, r;
5559         char
5560                 env1_buf[] = "RUNLEVEL=X",
5561                 env2_buf[] = "PREVLEVEL=X";
5562         char *env1 = env1_buf, *env2 = env2_buf;
5563         const char *emit = "runlevel";
5564         dbus_bool_t b_false = FALSE;
5565         DBusMessageIter iter, sub;
5566         DBusConnection *bus;
5567
5568         dbus_error_init(&error);
5569
5570         if (!(rl = action_to_runlevel()))
5571                 return 0;
5572
5573         if (utmp_get_runlevel(&previous, NULL) < 0)
5574                 previous = 'N';
5575
5576         if (!(bus = dbus_connection_open_private("unix:abstract=/com/ubuntu/upstart", &error))) {
5577                 if (dbus_error_has_name(&error, DBUS_ERROR_NO_SERVER)) {
5578                         r = 0;
5579                         goto finish;
5580                 }
5581
5582                 log_error("Failed to connect to Upstart bus: %s", bus_error_message(&error));
5583                 r = -EIO;
5584                 goto finish;
5585         }
5586
5587         if ((r = bus_check_peercred(bus)) < 0) {
5588                 log_error("Failed to verify owner of bus.");
5589                 goto finish;
5590         }
5591
5592         if (!(m = dbus_message_new_method_call(
5593                               "com.ubuntu.Upstart",
5594                               "/com/ubuntu/Upstart",
5595                               "com.ubuntu.Upstart0_6",
5596                               "EmitEvent"))) {
5597
5598                 log_error("Could not allocate message.");
5599                 r = -ENOMEM;
5600                 goto finish;
5601         }
5602
5603         dbus_message_iter_init_append(m, &iter);
5604
5605         env1_buf[sizeof(env1_buf)-2] = rl;
5606         env2_buf[sizeof(env2_buf)-2] = previous;
5607
5608         if (!dbus_message_iter_append_basic(&iter, DBUS_TYPE_STRING, &emit) ||
5609             !dbus_message_iter_open_container(&iter, DBUS_TYPE_ARRAY, "s", &sub) ||
5610             !dbus_message_iter_append_basic(&sub, DBUS_TYPE_STRING, &env1) ||
5611             !dbus_message_iter_append_basic(&sub, DBUS_TYPE_STRING, &env2) ||
5612             !dbus_message_iter_close_container(&iter, &sub) ||
5613             !dbus_message_iter_append_basic(&iter, DBUS_TYPE_BOOLEAN, &b_false)) {
5614                 log_error("Could not append arguments to message.");
5615                 r = -ENOMEM;
5616                 goto finish;
5617         }
5618
5619         if (!(reply = dbus_connection_send_with_reply_and_block(bus, m, -1, &error))) {
5620
5621                 if (bus_error_is_no_service(&error)) {
5622                         r = -EADDRNOTAVAIL;
5623                         goto finish;
5624                 }
5625
5626                 log_error("Failed to issue method call: %s", bus_error_message(&error));
5627                 r = -EIO;
5628                 goto finish;
5629         }
5630
5631         r = 1;
5632
5633 finish:
5634         if (bus) {
5635                 dbus_connection_flush(bus);
5636                 dbus_connection_close(bus);
5637                 dbus_connection_unref(bus);
5638         }
5639
5640         return r;
5641 }
5642
5643 static int talk_initctl(void) {
5644         struct init_request request = {};
5645         int r;
5646         _cleanup_close_ int fd = -1;
5647         char rl;
5648
5649         rl = action_to_runlevel();
5650         if (!rl)
5651                 return 0;
5652
5653         request.magic = INIT_MAGIC;
5654         request.sleeptime = 0;
5655         request.cmd = INIT_CMD_RUNLVL;
5656         request.runlevel = rl;
5657
5658         fd = open(INIT_FIFO, O_WRONLY|O_NDELAY|O_CLOEXEC|O_NOCTTY);
5659         if (fd < 0) {
5660                 if (errno == ENOENT)
5661                         return 0;
5662
5663                 log_error("Failed to open "INIT_FIFO": %m");
5664                 return -errno;
5665         }
5666
5667         errno = 0;
5668         r = loop_write(fd, &request, sizeof(request), false) != sizeof(request);
5669         if (r) {
5670                 log_error("Failed to write to "INIT_FIFO": %m");
5671                 return errno > 0 ? -errno : -EIO;
5672         }
5673
5674         return 1;
5675 }
5676
5677 static int systemctl_main(DBusConnection *bus, int argc, char *argv[], DBusError *error) {
5678
5679         static const struct {
5680                 const char* verb;
5681                 const enum {
5682                         MORE,
5683                         LESS,
5684                         EQUAL
5685                 } argc_cmp;
5686                 const int argc;
5687                 int (* const dispatch)(DBusConnection *bus, char **args);
5688         } verbs[] = {
5689                 { "list-units",            LESS,  1, list_units        },
5690                 { "list-unit-files",       EQUAL, 1, list_unit_files   },
5691                 { "list-sockets",          LESS,  1, list_sockets      },
5692                 { "list-jobs",             EQUAL, 1, list_jobs         },
5693                 { "clear-jobs",            EQUAL, 1, daemon_reload     },
5694                 { "cancel",                MORE,  2, cancel_job        },
5695                 { "start",                 MORE,  2, start_unit        },
5696                 { "stop",                  MORE,  2, start_unit        },
5697                 { "condstop",              MORE,  2, start_unit        }, /* For compatibility with ALTLinux */
5698                 { "reload",                MORE,  2, start_unit        },
5699                 { "restart",               MORE,  2, start_unit        },
5700                 { "try-restart",           MORE,  2, start_unit        },
5701                 { "reload-or-restart",     MORE,  2, start_unit        },
5702                 { "reload-or-try-restart", MORE,  2, start_unit        },
5703                 { "force-reload",          MORE,  2, start_unit        }, /* For compatibility with SysV */
5704                 { "condreload",            MORE,  2, start_unit        }, /* For compatibility with ALTLinux */
5705                 { "condrestart",           MORE,  2, start_unit        }, /* For compatibility with RH */
5706                 { "isolate",               EQUAL, 2, start_unit        },
5707                 { "kill",                  MORE,  2, kill_unit         },
5708                 { "is-active",             MORE,  2, check_unit_active },
5709                 { "check",                 MORE,  2, check_unit_active },
5710                 { "is-failed",             MORE,  2, check_unit_failed },
5711                 { "show",                  MORE,  1, show              },
5712                 { "status",                MORE,  1, show              },
5713                 { "help",                  MORE,  2, show              },
5714                 { "snapshot",              LESS,  2, snapshot          },
5715                 { "delete",                MORE,  2, delete_snapshot   },
5716                 { "daemon-reload",         EQUAL, 1, daemon_reload     },
5717                 { "daemon-reexec",         EQUAL, 1, daemon_reload     },
5718                 { "show-environment",      EQUAL, 1, show_enviroment   },
5719                 { "set-environment",       MORE,  2, set_environment   },
5720                 { "unset-environment",     MORE,  2, set_environment   },
5721                 { "halt",                  EQUAL, 1, start_special     },
5722                 { "poweroff",              EQUAL, 1, start_special     },
5723                 { "reboot",                EQUAL, 1, start_special     },
5724                 { "kexec",                 EQUAL, 1, start_special     },
5725                 { "suspend",               EQUAL, 1, start_special     },
5726                 { "hibernate",             EQUAL, 1, start_special     },
5727                 { "hybrid-sleep",          EQUAL, 1, start_special     },
5728                 { "default",               EQUAL, 1, start_special     },
5729                 { "rescue",                EQUAL, 1, start_special     },
5730                 { "emergency",             EQUAL, 1, start_special     },
5731                 { "exit",                  EQUAL, 1, start_special     },
5732                 { "reset-failed",          MORE,  1, reset_failed      },
5733                 { "enable",                MORE,  2, enable_unit       },
5734                 { "disable",               MORE,  2, enable_unit       },
5735                 { "is-enabled",            MORE,  2, unit_is_enabled   },
5736                 { "reenable",              MORE,  2, enable_unit       },
5737                 { "preset",                MORE,  2, enable_unit       },
5738                 { "mask",                  MORE,  2, enable_unit       },
5739                 { "unmask",                MORE,  2, enable_unit       },
5740                 { "link",                  MORE,  2, enable_unit       },
5741                 { "switch-root",           MORE,  2, switch_root       },
5742                 { "list-dependencies",     LESS,  2, list_dependencies },
5743                 { "set-default",           EQUAL, 2, enable_unit       },
5744                 { "get-default",           LESS,  1, get_default       },
5745                 { "set-property",          MORE,  3, set_property      },
5746         };
5747
5748         int left;
5749         unsigned i;
5750
5751         assert(argc >= 0);
5752         assert(argv);
5753         assert(error);
5754
5755         left = argc - optind;
5756
5757         if (left <= 0)
5758                 /* Special rule: no arguments means "list-units" */
5759                 i = 0;
5760         else {
5761                 if (streq(argv[optind], "help") && !argv[optind+1]) {
5762                         log_error("This command expects one or more "
5763                                   "unit names. Did you mean --help?");
5764                         return -EINVAL;
5765                 }
5766
5767                 for (i = 0; i < ELEMENTSOF(verbs); i++)
5768                         if (streq(argv[optind], verbs[i].verb))
5769                                 break;
5770
5771                 if (i >= ELEMENTSOF(verbs)) {
5772                         log_error("Unknown operation '%s'.", argv[optind]);
5773                         return -EINVAL;
5774                 }
5775         }
5776
5777         switch (verbs[i].argc_cmp) {
5778
5779         case EQUAL:
5780                 if (left != verbs[i].argc) {
5781                         log_error("Invalid number of arguments.");
5782                         return -EINVAL;
5783                 }
5784
5785                 break;
5786
5787         case MORE:
5788                 if (left < verbs[i].argc) {
5789                         log_error("Too few arguments.");
5790                         return -EINVAL;
5791                 }
5792
5793                 break;
5794
5795         case LESS:
5796                 if (left > verbs[i].argc) {
5797                         log_error("Too many arguments.");
5798                         return -EINVAL;
5799                 }
5800
5801                 break;
5802
5803         default:
5804                 assert_not_reached("Unknown comparison operator.");
5805         }
5806
5807         /* Require a bus connection for all operations but
5808          * enable/disable */
5809         if (!streq(verbs[i].verb, "enable") &&
5810             !streq(verbs[i].verb, "disable") &&
5811             !streq(verbs[i].verb, "is-enabled") &&
5812             !streq(verbs[i].verb, "list-unit-files") &&
5813             !streq(verbs[i].verb, "reenable") &&
5814             !streq(verbs[i].verb, "preset") &&
5815             !streq(verbs[i].verb, "mask") &&
5816             !streq(verbs[i].verb, "unmask") &&
5817             !streq(verbs[i].verb, "link") &&
5818             !streq(verbs[i].verb, "set-default") &&
5819             !streq(verbs[i].verb, "get-default")) {
5820
5821                 if (running_in_chroot() > 0) {
5822                         log_info("Running in chroot, ignoring request.");
5823                         return 0;
5824                 }
5825
5826                 if (((!streq(verbs[i].verb, "reboot") &&
5827                       !streq(verbs[i].verb, "halt") &&
5828                       !streq(verbs[i].verb, "poweroff")) || arg_force <= 0) && !bus) {
5829                         log_error("Failed to get D-Bus connection: %s",
5830                                   dbus_error_is_set(error) ? error->message : "No connection to service manager.");
5831                         return -EIO;
5832                 }
5833
5834         } else {
5835
5836                 if (!bus && !avoid_bus()) {
5837                         log_error("Failed to get D-Bus connection: %s",
5838                                   dbus_error_is_set(error) ? error->message : "No connection to service manager.");
5839                         return -EIO;
5840                 }
5841         }
5842
5843         return verbs[i].dispatch(bus, argv + optind);
5844 }
5845
5846 static int send_shutdownd(usec_t t, char mode, bool dry_run, bool warn, const char *message) {
5847         _cleanup_close_ int fd;
5848         struct sd_shutdown_command c = {
5849                 .usec = t,
5850                 .mode = mode,
5851                 .dry_run = dry_run,
5852                 .warn_wall = warn,
5853         };
5854         union sockaddr_union sockaddr = {
5855                 .un.sun_family = AF_UNIX,
5856                 .un.sun_path = "/run/systemd/shutdownd",
5857         };
5858         struct iovec iovec[2] = {
5859                 {.iov_base = (char*) &c,
5860                  .iov_len = offsetof(struct sd_shutdown_command, wall_message),
5861                 }
5862         };
5863         struct msghdr msghdr = {
5864                 .msg_name = &sockaddr,
5865                 .msg_namelen = offsetof(struct sockaddr_un, sun_path)
5866                                + sizeof("/run/systemd/shutdownd") - 1,
5867                 .msg_iov = iovec,
5868                 .msg_iovlen = 1,
5869         };
5870
5871         fd = socket(AF_UNIX, SOCK_DGRAM|SOCK_CLOEXEC, 0);
5872         if (fd < 0)
5873                 return -errno;
5874
5875         if (!isempty(message)) {
5876                 iovec[1].iov_base = (char*) message;
5877                 iovec[1].iov_len = strlen(message);
5878                 msghdr.msg_iovlen++;
5879         }
5880
5881         if (sendmsg(fd, &msghdr, MSG_NOSIGNAL) < 0)
5882                 return -errno;
5883
5884         return 0;
5885 }
5886
5887 static int reload_with_fallback(DBusConnection *bus) {
5888
5889         if (bus) {
5890                 /* First, try systemd via D-Bus. */
5891                 if (daemon_reload(bus, NULL) >= 0)
5892                         return 0;
5893         }
5894
5895         /* Nothing else worked, so let's try signals */
5896         assert(arg_action == ACTION_RELOAD || arg_action == ACTION_REEXEC);
5897
5898         if (kill(1, arg_action == ACTION_RELOAD ? SIGHUP : SIGTERM) < 0) {
5899                 log_error("kill() failed: %m");
5900                 return -errno;
5901         }
5902
5903         return 0;
5904 }
5905
5906 static int start_with_fallback(DBusConnection *bus) {
5907
5908         if (bus) {
5909                 /* First, try systemd via D-Bus. */
5910                 if (start_unit(bus, NULL) >= 0)
5911                         goto done;
5912         }
5913
5914         /* Hmm, talking to systemd via D-Bus didn't work. Then
5915          * let's try to talk to Upstart via D-Bus. */
5916         if (talk_upstart() > 0)
5917                 goto done;
5918
5919         /* Nothing else worked, so let's try
5920          * /dev/initctl */
5921         if (talk_initctl() > 0)
5922                 goto done;
5923
5924         log_error("Failed to talk to init daemon.");
5925         return -EIO;
5926
5927 done:
5928         warn_wall(arg_action);
5929         return 0;
5930 }
5931
5932 static _noreturn_ void halt_now(enum action a) {
5933
5934        /* Make sure C-A-D is handled by the kernel from this
5935          * point on... */
5936         reboot(RB_ENABLE_CAD);
5937
5938         switch (a) {
5939
5940         case ACTION_HALT:
5941                 log_info("Halting.");
5942                 reboot(RB_HALT_SYSTEM);
5943                 break;
5944
5945         case ACTION_POWEROFF:
5946                 log_info("Powering off.");
5947                 reboot(RB_POWER_OFF);
5948                 break;
5949
5950         case ACTION_REBOOT:
5951                 log_info("Rebooting.");
5952                 reboot(RB_AUTOBOOT);
5953                 break;
5954
5955         default:
5956                 assert_not_reached("Unknown halt action.");
5957         }
5958
5959         assert_not_reached("Uh? This shouldn't happen.");
5960 }
5961
5962 static int halt_main(DBusConnection *bus) {
5963         int r;
5964
5965         r = check_inhibitors(bus, arg_action);
5966         if (r < 0)
5967                 return r;
5968
5969         if (geteuid() != 0) {
5970                 /* Try logind if we are a normal user and no special
5971                  * mode applies. Maybe PolicyKit allows us to shutdown
5972                  * the machine. */
5973
5974                 if (arg_when <= 0 &&
5975                     !arg_dry &&
5976                     arg_force <= 0 &&
5977                     (arg_action == ACTION_POWEROFF ||
5978                      arg_action == ACTION_REBOOT)) {
5979                         r = reboot_with_logind(bus, arg_action);
5980                         if (r >= 0)
5981                                 return r;
5982                 }
5983
5984                 log_error("Must be root.");
5985                 return -EPERM;
5986         }
5987
5988         if (arg_when > 0) {
5989                 _cleanup_free_ char *m;
5990
5991                 m = strv_join(arg_wall, " ");
5992                 r = send_shutdownd(arg_when,
5993                                    arg_action == ACTION_HALT     ? 'H' :
5994                                    arg_action == ACTION_POWEROFF ? 'P' :
5995                                    arg_action == ACTION_KEXEC    ? 'K' :
5996                                                                    'r',
5997                                    arg_dry,
5998                                    !arg_no_wall,
5999                                    m);
6000
6001                 if (r < 0)
6002                         log_warning("Failed to talk to shutdownd, proceeding with immediate shutdown: %s", strerror(-r));
6003                 else {
6004                         char date[FORMAT_TIMESTAMP_MAX];
6005
6006                         log_info("Shutdown scheduled for %s, use 'shutdown -c' to cancel.",
6007                                  format_timestamp(date, sizeof(date), arg_when));
6008                         return 0;
6009                 }
6010         }
6011
6012         if (!arg_dry && !arg_force)
6013                 return start_with_fallback(bus);
6014
6015         if (!arg_no_wtmp) {
6016                 if (sd_booted() > 0)
6017                         log_debug("Not writing utmp record, assuming that systemd-update-utmp is used.");
6018                 else {
6019                         r = utmp_put_shutdown();
6020                         if (r < 0)
6021                                 log_warning("Failed to write utmp record: %s", strerror(-r));
6022                 }
6023         }
6024
6025         if (arg_dry)
6026                 return 0;
6027
6028         halt_now(arg_action);
6029         /* We should never reach this. */
6030         return -ENOSYS;
6031 }
6032
6033 static int runlevel_main(void) {
6034         int r, runlevel, previous;
6035
6036         r = utmp_get_runlevel(&runlevel, &previous);
6037         if (r < 0) {
6038                 puts("unknown");
6039                 return r;
6040         }
6041
6042         printf("%c %c\n",
6043                previous <= 0 ? 'N' : previous,
6044                runlevel <= 0 ? 'N' : runlevel);
6045
6046         return 0;
6047 }
6048
6049 int main(int argc, char*argv[]) {
6050         int r, retval = EXIT_FAILURE;
6051         DBusConnection *bus = NULL;
6052         _cleanup_dbus_error_free_ DBusError error;
6053
6054         dbus_error_init(&error);
6055
6056         setlocale(LC_ALL, "");
6057         log_parse_environment();
6058         log_open();
6059
6060         r = parse_argv(argc, argv);
6061         if (r < 0)
6062                 goto finish;
6063         else if (r == 0) {
6064                 retval = EXIT_SUCCESS;
6065                 goto finish;
6066         }
6067
6068         /* /sbin/runlevel doesn't need to communicate via D-Bus, so
6069          * let's shortcut this */
6070         if (arg_action == ACTION_RUNLEVEL) {
6071                 r = runlevel_main();
6072                 retval = r < 0 ? EXIT_FAILURE : r;
6073                 goto finish;
6074         }
6075
6076         if (running_in_chroot() > 0 && arg_action != ACTION_SYSTEMCTL) {
6077                 log_info("Running in chroot, ignoring request.");
6078                 retval = 0;
6079                 goto finish;
6080         }
6081
6082         if (!avoid_bus()) {
6083                 if (arg_transport == TRANSPORT_NORMAL)
6084                         bus_connect(arg_scope == UNIT_FILE_SYSTEM ? DBUS_BUS_SYSTEM : DBUS_BUS_SESSION, &bus, &private_bus, &error);
6085                 else if (arg_transport == TRANSPORT_POLKIT) {
6086                         bus_connect_system_polkit(&bus, &error);
6087                         private_bus = false;
6088                 } else if (arg_transport == TRANSPORT_SSH) {
6089                         bus_connect_system_ssh(arg_user, arg_host, &bus, &error);
6090                         private_bus = false;
6091                 } else
6092                         assert_not_reached("Uh, invalid transport...");
6093         }
6094
6095         switch (arg_action) {
6096
6097         case ACTION_SYSTEMCTL:
6098                 r = systemctl_main(bus, argc, argv, &error);
6099                 break;
6100
6101         case ACTION_HALT:
6102         case ACTION_POWEROFF:
6103         case ACTION_REBOOT:
6104         case ACTION_KEXEC:
6105                 r = halt_main(bus);
6106                 break;
6107
6108         case ACTION_RUNLEVEL2:
6109         case ACTION_RUNLEVEL3:
6110         case ACTION_RUNLEVEL4:
6111         case ACTION_RUNLEVEL5:
6112         case ACTION_RESCUE:
6113         case ACTION_EMERGENCY:
6114         case ACTION_DEFAULT:
6115                 r = start_with_fallback(bus);
6116                 break;
6117
6118         case ACTION_RELOAD:
6119         case ACTION_REEXEC:
6120                 r = reload_with_fallback(bus);
6121                 break;
6122
6123         case ACTION_CANCEL_SHUTDOWN: {
6124                 char *m = NULL;
6125
6126                 if (arg_wall) {
6127                         m = strv_join(arg_wall, " ");
6128                         if (!m) {
6129                                 retval = EXIT_FAILURE;
6130                                 goto finish;
6131                         }
6132                 }
6133                 r = send_shutdownd(arg_when, SD_SHUTDOWN_NONE, false, !arg_no_wall, m);
6134                 if (r < 0)
6135                         log_warning("Failed to talk to shutdownd, shutdown hasn't been cancelled: %s", strerror(-r));
6136                 free(m);
6137                 break;
6138         }
6139
6140         case ACTION_INVALID:
6141         case ACTION_RUNLEVEL:
6142         default:
6143                 assert_not_reached("Unknown action");
6144         }
6145
6146         retval = r < 0 ? EXIT_FAILURE : r;
6147
6148 finish:
6149         if (bus) {
6150                 dbus_connection_flush(bus);
6151                 dbus_connection_close(bus);
6152                 dbus_connection_unref(bus);
6153         }
6154
6155         dbus_shutdown();
6156
6157         strv_free(arg_types);
6158         strv_free(arg_states);
6159         strv_free(arg_properties);
6160
6161         pager_close();
6162         ask_password_agent_close();
6163         polkit_agent_close();
6164
6165         return retval;
6166 }