machinectl.c revision c19de71113f956809995fc68817e055e9f61f607
1ee306e1248866617c96ed9f4263f375588ad838Lennart Poettering/*-*- Mode: C; c-basic-offset: 8; indent-tabs-mode: nil -*-*/
1ee306e1248866617c96ed9f4263f375588ad838Lennart Poettering This file is part of systemd.
1ee306e1248866617c96ed9f4263f375588ad838Lennart Poettering Copyright 2013 Lennart Poettering
1ee306e1248866617c96ed9f4263f375588ad838Lennart Poettering systemd is free software; you can redistribute it and/or modify it
1ee306e1248866617c96ed9f4263f375588ad838Lennart Poettering under the terms of the GNU Lesser General Public License as published by
1ee306e1248866617c96ed9f4263f375588ad838Lennart Poettering the Free Software Foundation; either version 2.1 of the License, or
1ee306e1248866617c96ed9f4263f375588ad838Lennart Poettering (at your option) any later version.
1ee306e1248866617c96ed9f4263f375588ad838Lennart Poettering systemd is distributed in the hope that it will be useful, but
1ee306e1248866617c96ed9f4263f375588ad838Lennart Poettering WITHOUT ANY WARRANTY; without even the implied warranty of
1ee306e1248866617c96ed9f4263f375588ad838Lennart Poettering MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
1ee306e1248866617c96ed9f4263f375588ad838Lennart Poettering Lesser General Public License for more details.
1ee306e1248866617c96ed9f4263f375588ad838Lennart Poettering You should have received a copy of the GNU Lesser General Public License
1ee306e1248866617c96ed9f4263f375588ad838Lennart Poettering along with systemd; If not, see <http://www.gnu.org/licenses/>.
1ee306e1248866617c96ed9f4263f375588ad838Lennart Poetteringstatic bool arg_all = false;
e56056e93d33619a3acf13e483900b4f8938228fThomas Hindoe Paaboel Andersenstatic bool arg_full = false;
1ee306e1248866617c96ed9f4263f375588ad838Lennart Poetteringstatic bool arg_no_pager = false;
1ee306e1248866617c96ed9f4263f375588ad838Lennart Poetteringstatic bool arg_legend = true;
785890acf6d629ff881a1f065f431df1b7fc8c7aLennart Poetteringstatic BusTransport arg_transport = BUS_TRANSPORT_LOCAL;
1ee306e1248866617c96ed9f4263f375588ad838Lennart Poetteringstatic bool arg_read_only = false;
1ee306e1248866617c96ed9f4263f375588ad838Lennart Poetteringstatic bool arg_mkdir = false;
1ee306e1248866617c96ed9f4263f375588ad838Lennart Poetteringstatic bool arg_quiet = false;
1ee306e1248866617c96ed9f4263f375588ad838Lennart Poetteringstatic bool arg_ask_password = true;
1ee306e1248866617c96ed9f4263f375588ad838Lennart Poetteringstatic OutputMode arg_output = OUTPUT_SHORT;
1ee306e1248866617c96ed9f4263f375588ad838Lennart Poetteringstatic void pager_open_if_enabled(void) {
56159e0d918e9a9be07988133bb2847779325de0Lennart Poetteringstatic void polkit_agent_open_if_enabled(void) {
1ee306e1248866617c96ed9f4263f375588ad838Lennart Poettering /* Open the polkit agent as a child process if necessary */
1ee306e1248866617c96ed9f4263f375588ad838Lennart Poettering if (arg_transport != BUS_TRANSPORT_LOCAL)
a1da85830bfaa77b9eb9c54693e5573559c97e50Tom Gundersen (!on_tty() || pager_have()) * OUTPUT_FULL_WIDTH |
1ee306e1248866617c96ed9f4263f375588ad838Lennart Poetteringstatic int compare_machine_info(const void *a, const void *b) {
a1da85830bfaa77b9eb9c54693e5573559c97e50Tom Gundersen const MachineInfo *x = a, *y = b;
a1da85830bfaa77b9eb9c54693e5573559c97e50Tom Gundersenstatic int list_machines(int argc, char *argv[], void *userdata) {
a1da85830bfaa77b9eb9c54693e5573559c97e50Tom Gundersen size_t max_name = strlen("MACHINE"), max_class = strlen("CLASS"), max_service = strlen("SERVICE");
a1da85830bfaa77b9eb9c54693e5573559c97e50Tom Gundersen _cleanup_bus_error_free_ sd_bus_error error = SD_BUS_ERROR_NULL;
a1da85830bfaa77b9eb9c54693e5573559c97e50Tom Gundersen _cleanup_bus_message_unref_ sd_bus_message *reply = NULL;
5b30bef856e89a571df57b7b953e9a1409d9acedLennart Poettering _cleanup_free_ MachineInfo *machines = NULL;
1ee306e1248866617c96ed9f4263f375588ad838Lennart Poettering const char *name, *class, *service, *object;
e56056e93d33619a3acf13e483900b4f8938228fThomas Hindoe Paaboel Andersen size_t n_machines = 0, n_allocated = 0, j;
10f9c75519671e7c7ab8993b54fe22da7c2d0c38Lennart Poettering "org.freedesktop.machine1",
cd61c3bfd718fb398cc53ced906266a9297782c9Lennart Poettering "org.freedesktop.machine1.Manager",
cd61c3bfd718fb398cc53ced906266a9297782c9Lennart Poettering "ListMachines",
cd61c3bfd718fb398cc53ced906266a9297782c9Lennart Poettering log_error("Could not get machines: %s", bus_error_message(&error, -r));
cd61c3bfd718fb398cc53ced906266a9297782c9Lennart Poettering r = sd_bus_message_enter_container(reply, SD_BUS_TYPE_ARRAY, "(ssso)");
cd61c3bfd718fb398cc53ced906266a9297782c9Lennart Poettering while ((r = sd_bus_message_read(reply, "(ssso)", &name, &class, &service, &object)) > 0) {
10f9c75519671e7c7ab8993b54fe22da7c2d0c38Lennart Poettering if (!GREEDY_REALLOC(machines, n_allocated, n_machines + 1))
cd61c3bfd718fb398cc53ced906266a9297782c9Lennart Poettering r = sd_bus_message_exit_container(reply);
cd61c3bfd718fb398cc53ced906266a9297782c9Lennart Poettering qsort_safe(machines, n_machines, sizeof(MachineInfo), compare_machine_info);
cd61c3bfd718fb398cc53ced906266a9297782c9Lennart Poettering for (j = 0; j < n_machines; j++)
10f9c75519671e7c7ab8993b54fe22da7c2d0c38Lennart Poettering printf("\n%zu machines listed.\n", n_machines);
cd61c3bfd718fb398cc53ced906266a9297782c9Lennart Poetteringtypedef struct ImageInfo {
10f9c75519671e7c7ab8993b54fe22da7c2d0c38Lennart Poetteringstatic int compare_image_info(const void *a, const void *b) {
10f9c75519671e7c7ab8993b54fe22da7c2d0c38Lennart Poettering const ImageInfo *x = a, *y = b;
cd61c3bfd718fb398cc53ced906266a9297782c9Lennart Poetteringstatic int list_images(int argc, char *argv[], void *userdata) {
cd61c3bfd718fb398cc53ced906266a9297782c9Lennart Poettering _cleanup_bus_error_free_ sd_bus_error error = SD_BUS_ERROR_NULL;
cd61c3bfd718fb398cc53ced906266a9297782c9Lennart Poettering size_t max_name = strlen("NAME"), max_type = strlen("TYPE"), max_size = strlen("USAGE"), max_crtime = strlen("CREATED"), max_mtime = strlen("MODIFIED");
cd61c3bfd718fb398cc53ced906266a9297782c9Lennart Poettering _cleanup_bus_message_unref_ sd_bus_message *reply = NULL;
10f9c75519671e7c7ab8993b54fe22da7c2d0c38Lennart Poettering "org.freedesktop.machine1",
8937e7b68940d0fa0d0aab90eb7425fa7dccebc9Lennart Poettering "org.freedesktop.machine1.Manager",
cd61c3bfd718fb398cc53ced906266a9297782c9Lennart Poettering log_error("Could not get images: %s", bus_error_message(&error, -r));
cd61c3bfd718fb398cc53ced906266a9297782c9Lennart Poettering r = sd_bus_message_enter_container(reply, SD_BUS_TYPE_ARRAY, "(ssbttto)");
cd61c3bfd718fb398cc53ced906266a9297782c9Lennart Poettering while ((r = sd_bus_message_read(reply, "(ssbttto)", &name, &type, &read_only, &crtime, &mtime, &size, &object)) > 0) {
cd61c3bfd718fb398cc53ced906266a9297782c9Lennart Poettering char buf[MAX(FORMAT_TIMESTAMP_MAX, FORMAT_BYTES_MAX)];
aa1936ea1a89c2bb968ba33e3274898a4eeae771Lennart Poettering if (!GREEDY_REALLOC(images, n_allocated, n_images + 1))
89f7c8465cd1ab37347dd0c15920bce31e8225dfLennart Poettering l = strlen(strna(format_timestamp(buf, sizeof(buf), crtime)));
a1da85830bfaa77b9eb9c54693e5573559c97e50Tom Gundersen l = strlen(strna(format_timestamp(buf, sizeof(buf), mtime)));
5b30bef856e89a571df57b7b953e9a1409d9acedLennart Poettering l = strlen(strna(format_bytes(buf, sizeof(buf), size)));
aa1936ea1a89c2bb968ba33e3274898a4eeae771Lennart Poettering r = sd_bus_message_exit_container(reply);
aa1936ea1a89c2bb968ba33e3274898a4eeae771Lennart Poettering qsort_safe(images, n_images, sizeof(ImageInfo), compare_image_info);
aa1936ea1a89c2bb968ba33e3274898a4eeae771Lennart Poettering printf("%-*s %-*s %-3s %-*s %-*s %-*s\n",
878cd7e95ca303f9851d227a22d2022bd49944b0Lennart Poettering for (j = 0; j < n_images; j++) {
878cd7e95ca303f9851d227a22d2022bd49944b0Lennart Poettering char crtime_buf[FORMAT_TIMESTAMP_MAX], mtime_buf[FORMAT_TIMESTAMP_MAX], size_buf[FORMAT_BYTES_MAX];
878cd7e95ca303f9851d227a22d2022bd49944b0Lennart Poettering printf("%-*s %-*s %s%-3s%s %-*s %-*s %-*s\n",
878cd7e95ca303f9851d227a22d2022bd49944b0Lennart Poettering images[j].read_only ? ansi_highlight_red() : "", yes_no(images[j].read_only), images[j].read_only ? ansi_highlight_off() : "",
878cd7e95ca303f9851d227a22d2022bd49944b0Lennart Poettering (int) max_size, strna(format_bytes(size_buf, sizeof(size_buf), images[j].size)),
878cd7e95ca303f9851d227a22d2022bd49944b0Lennart Poettering (int) max_crtime, strna(format_timestamp(crtime_buf, sizeof(crtime_buf), images[j].crtime)),
878cd7e95ca303f9851d227a22d2022bd49944b0Lennart Poettering (int) max_mtime, strna(format_timestamp(mtime_buf, sizeof(mtime_buf), images[j].mtime)));
878cd7e95ca303f9851d227a22d2022bd49944b0Lennart Poettering printf("\n%zu images listed.\n", n_images);
878cd7e95ca303f9851d227a22d2022bd49944b0Lennart Poetteringstatic int show_unit_cgroup(sd_bus *bus, const char *unit, pid_t leader) {
878cd7e95ca303f9851d227a22d2022bd49944b0Lennart Poettering _cleanup_bus_message_unref_ sd_bus_message *reply = NULL;
878cd7e95ca303f9851d227a22d2022bd49944b0Lennart Poettering _cleanup_bus_error_free_ sd_bus_error error = SD_BUS_ERROR_NULL;
878cd7e95ca303f9851d227a22d2022bd49944b0Lennart Poettering if (arg_transport == BUS_TRANSPORT_REMOTE)
878cd7e95ca303f9851d227a22d2022bd49944b0Lennart Poettering endswith(unit, ".scope") ? "org.freedesktop.systemd1.Scope" : "org.freedesktop.systemd1.Service",
878cd7e95ca303f9851d227a22d2022bd49944b0Lennart Poettering "ControlGroup",
878cd7e95ca303f9851d227a22d2022bd49944b0Lennart Poettering log_error("Failed to query ControlGroup: %s", bus_error_message(&error, -r));
878cd7e95ca303f9851d227a22d2022bd49944b0Lennart Poettering r = sd_bus_message_read(reply, "s", &cgroup);
878cd7e95ca303f9851d227a22d2022bd49944b0Lennart Poettering if (cg_is_empty_recursive(SYSTEMD_CGROUP_CONTROLLER, cgroup, false) != 0 && leader <= 0)
717603e391b52983ca1fd218e7333a1b9dfc5c05Lennart Poettering show_cgroup_and_extra(SYSTEMD_CGROUP_CONTROLLER, cgroup, "\t\t ", c, false, &leader, leader > 0, get_output_flags());
717603e391b52983ca1fd218e7333a1b9dfc5c05Lennart Poetteringstatic int print_addresses(sd_bus *bus, const char *name, int ifi, const char *prefix, const char *prefix2) {
717603e391b52983ca1fd218e7333a1b9dfc5c05Lennart Poettering _cleanup_bus_message_unref_ sd_bus_message *reply = NULL;
717603e391b52983ca1fd218e7333a1b9dfc5c05Lennart Poettering "org.freedesktop.machine1",
717603e391b52983ca1fd218e7333a1b9dfc5c05Lennart Poettering "org.freedesktop.machine1.Manager",
717603e391b52983ca1fd218e7333a1b9dfc5c05Lennart Poettering "GetMachineAddresses",
717603e391b52983ca1fd218e7333a1b9dfc5c05Lennart Poettering r = sd_bus_message_enter_container(reply, 'a', "(iay)");
717603e391b52983ca1fd218e7333a1b9dfc5c05Lennart Poettering while ((r = sd_bus_message_enter_container(reply, 'r', "iay")) > 0) {
717603e391b52983ca1fd218e7333a1b9dfc5c05Lennart Poettering const void *a;
717603e391b52983ca1fd218e7333a1b9dfc5c05Lennart Poettering char buffer[MAX(INET6_ADDRSTRLEN, INET_ADDRSTRLEN)];
1ee306e1248866617c96ed9f4263f375588ad838Lennart Poettering r = sd_bus_message_read(reply, "i", &family);
9f6eb1cd58f2ddf2eb6ba0e4de056e13d938af75Kay Sievers r = sd_bus_message_read_array(reply, 'y', &a, &sz);
f48e75cb9a8112d35855c44a156934f2ee0edb2eLennart Poettering fputs(inet_ntop(family, a, buffer, sizeof(buffer)), stdout);
1ee306e1248866617c96ed9f4263f375588ad838Lennart Poettering r = sd_bus_message_exit_container(reply);
1ee306e1248866617c96ed9f4263f375588ad838Lennart Poettering r = sd_bus_message_exit_container(reply);
1ee306e1248866617c96ed9f4263f375588ad838Lennart Poetteringstatic int print_os_release(sd_bus *bus, const char *name, const char *prefix) {
1ee306e1248866617c96ed9f4263f375588ad838Lennart Poettering _cleanup_bus_message_unref_ sd_bus_message *reply = NULL;
1ee306e1248866617c96ed9f4263f375588ad838Lennart Poettering "org.freedesktop.machine1",
1ee306e1248866617c96ed9f4263f375588ad838Lennart Poettering "org.freedesktop.machine1.Manager",
1ee306e1248866617c96ed9f4263f375588ad838Lennart Poettering "GetMachineOSRelease",
1ee306e1248866617c96ed9f4263f375588ad838Lennart Poettering r = sd_bus_message_enter_container(reply, 'a', "{ss}");
1ee306e1248866617c96ed9f4263f375588ad838Lennart Poettering while ((r = sd_bus_message_read(reply, "{ss}", &k, &v)) > 0) {
f48e75cb9a8112d35855c44a156934f2ee0edb2eLennart Poettering r = sd_bus_message_exit_container(reply);
878cd7e95ca303f9851d227a22d2022bd49944b0Lennart Poetteringstatic void print_machine_status_info(sd_bus *bus, MachineStatusInfo *i) {
878cd7e95ca303f9851d227a22d2022bd49944b0Lennart Poettering char since1[FORMAT_TIMESTAMP_RELATIVE_MAX], *s1;
f48e75cb9a8112d35855c44a156934f2ee0edb2eLennart Poettering if (!sd_id128_equal(i->id, SD_ID128_NULL))
f48e75cb9a8112d35855c44a156934f2ee0edb2eLennart Poettering printf("(" SD_ID128_FORMAT_STR ")\n", SD_ID128_FORMAT_VAL(i->id));
f48e75cb9a8112d35855c44a156934f2ee0edb2eLennart Poettering s1 = format_timestamp_relative(since1, sizeof(since1), i->timestamp.realtime);
f48e75cb9a8112d35855c44a156934f2ee0edb2eLennart Poettering s2 = format_timestamp(since2, sizeof(since2), i->timestamp.realtime);
f48e75cb9a8112d35855c44a156934f2ee0edb2eLennart Poettering printf("\t Leader: %u", (unsigned) i->leader);
f48e75cb9a8112d35855c44a156934f2ee0edb2eLennart Poettering } else if (i->class)
a6c616024db23fef34152c1432892824a07799ccLennart Poettering printf("\t Root: %s\n", i->root_directory);
a1da85830bfaa77b9eb9c54693e5573559c97e50Tom Gundersen if (i->n_netif > 0) {
1ee306e1248866617c96ed9f4263f375588ad838Lennart Poettering for (c = 0; c < i->n_netif; c++) {
9f6eb1cd58f2ddf2eb6ba0e4de056e13d938af75Kay Sievers "\t Address: ",
56159e0d918e9a9be07988133bb2847779325de0Lennart Poettering if (arg_transport == BUS_TRANSPORT_LOCAL) {
a1da85830bfaa77b9eb9c54693e5573559c97e50Tom Gundersenstatic int map_netif(sd_bus *bus, const char *member, sd_bus_message *m, sd_bus_error *error, void *userdata) {
56159e0d918e9a9be07988133bb2847779325de0Lennart Poettering const void *v;
1ee306e1248866617c96ed9f4263f375588ad838Lennart Poettering assert_cc(sizeof(int32_t) == sizeof(int));
1ee306e1248866617c96ed9f4263f375588ad838Lennart Poettering r = sd_bus_message_read_array(m, SD_BUS_TYPE_INT32, &v, &l);
56159e0d918e9a9be07988133bb2847779325de0Lennart Poetteringstatic int show_machine_info(const char *verb, sd_bus *bus, const char *path, bool *new_line) {
1ee306e1248866617c96ed9f4263f375588ad838Lennart Poettering static const struct bus_properties_map map[] = {
a1da85830bfaa77b9eb9c54693e5573559c97e50Tom Gundersen { "Name", "s", NULL, offsetof(MachineStatusInfo, name) },
a1da85830bfaa77b9eb9c54693e5573559c97e50Tom Gundersen { "Class", "s", NULL, offsetof(MachineStatusInfo, class) },
a1da85830bfaa77b9eb9c54693e5573559c97e50Tom Gundersen { "Service", "s", NULL, offsetof(MachineStatusInfo, service) },
a1da85830bfaa77b9eb9c54693e5573559c97e50Tom Gundersen { "Unit", "s", NULL, offsetof(MachineStatusInfo, unit) },
a1da85830bfaa77b9eb9c54693e5573559c97e50Tom Gundersen { "RootDirectory", "s", NULL, offsetof(MachineStatusInfo, root_directory) },
a1da85830bfaa77b9eb9c54693e5573559c97e50Tom Gundersen { "Leader", "u", NULL, offsetof(MachineStatusInfo, leader) },
a1da85830bfaa77b9eb9c54693e5573559c97e50Tom Gundersen { "Timestamp", "t", NULL, offsetof(MachineStatusInfo, timestamp.realtime) },
a1da85830bfaa77b9eb9c54693e5573559c97e50Tom Gundersen { "TimestampMonotonic", "t", NULL, offsetof(MachineStatusInfo, timestamp.monotonic) },
56159e0d918e9a9be07988133bb2847779325de0Lennart Poettering { "Id", "ay", bus_map_id128, offsetof(MachineStatusInfo, id) },
fefdc04b38725457a91651218feb7000f6ccc1f4Lennart Poettering "org.freedesktop.machine1",
fefdc04b38725457a91651218feb7000f6ccc1f4Lennart Poettering return log_error_errno(r, "Could not get properties: %m");
fefdc04b38725457a91651218feb7000f6ccc1f4Lennart Poetteringstatic int show_machine_properties(sd_bus *bus, const char *path, bool *new_line) {
fefdc04b38725457a91651218feb7000f6ccc1f4Lennart Poettering r = bus_print_all_properties(bus, "org.freedesktop.machine1", path, arg_property, arg_all);
fefdc04b38725457a91651218feb7000f6ccc1f4Lennart Poettering log_error_errno(r, "Could not get properties: %m");
fefdc04b38725457a91651218feb7000f6ccc1f4Lennart Poetteringstatic int show_machine(int argc, char *argv[], void *userdata) {
fefdc04b38725457a91651218feb7000f6ccc1f4Lennart Poettering _cleanup_bus_error_free_ sd_bus_error error = SD_BUS_ERROR_NULL;
fefdc04b38725457a91651218feb7000f6ccc1f4Lennart Poettering _cleanup_bus_message_unref_ sd_bus_message *reply = NULL;
fefdc04b38725457a91651218feb7000f6ccc1f4Lennart Poettering /* If no argument is specified, inspect the manager
fefdc04b38725457a91651218feb7000f6ccc1f4Lennart Poettering r = show_machine_properties(bus, "/org/freedesktop/machine1", &new_line);
fefdc04b38725457a91651218feb7000f6ccc1f4Lennart Poettering "org.freedesktop.machine1",
fefdc04b38725457a91651218feb7000f6ccc1f4Lennart Poettering "org.freedesktop.machine1.Manager",
fefdc04b38725457a91651218feb7000f6ccc1f4Lennart Poettering log_error("Could not get path to machine: %s", bus_error_message(&error, -r));
fefdc04b38725457a91651218feb7000f6ccc1f4Lennart Poettering r = sd_bus_message_read(reply, "o", &path);
fefdc04b38725457a91651218feb7000f6ccc1f4Lennart Poettering r = show_machine_properties(bus, path, &new_line);
fefdc04b38725457a91651218feb7000f6ccc1f4Lennart Poettering r = show_machine_info(argv[0], bus, path, &new_line);
fefdc04b38725457a91651218feb7000f6ccc1f4Lennart Poetteringstatic void print_image_status_info(sd_bus *bus, ImageStatusInfo *i) {
fefdc04b38725457a91651218feb7000f6ccc1f4Lennart Poettering char ts_relative[FORMAT_TIMESTAMP_RELATIVE_MAX], *s1;
fefdc04b38725457a91651218feb7000f6ccc1f4Lennart Poettering char ts_absolute[FORMAT_TIMESTAMP_MAX], *s2;
fefdc04b38725457a91651218feb7000f6ccc1f4Lennart Poettering char bs_exclusive[FORMAT_BYTES_MAX], *s4;
fefdc04b38725457a91651218feb7000f6ccc1f4Lennart Poettering i->read_only ? ansi_highlight_red() : "",
fefdc04b38725457a91651218feb7000f6ccc1f4Lennart Poettering i->read_only ? ansi_highlight_off() : "");
fefdc04b38725457a91651218feb7000f6ccc1f4Lennart Poettering s1 = format_timestamp_relative(ts_relative, sizeof(ts_relative), i->crtime);
fefdc04b38725457a91651218feb7000f6ccc1f4Lennart Poettering s2 = format_timestamp(ts_absolute, sizeof(ts_absolute), i->crtime);
fefdc04b38725457a91651218feb7000f6ccc1f4Lennart Poettering s1 = format_timestamp_relative(ts_relative, sizeof(ts_relative), i->mtime);
fefdc04b38725457a91651218feb7000f6ccc1f4Lennart Poettering s2 = format_timestamp(ts_absolute, sizeof(ts_absolute), i->mtime);
fefdc04b38725457a91651218feb7000f6ccc1f4Lennart Poettering s3 = format_bytes(bs, sizeof(bs), i->usage);
fefdc04b38725457a91651218feb7000f6ccc1f4Lennart Poettering s4 = i->usage_exclusive != i->usage ? format_bytes(bs_exclusive, sizeof(bs_exclusive), i->usage_exclusive) : NULL;
fefdc04b38725457a91651218feb7000f6ccc1f4Lennart Poettering printf("\t Usage: %s (exclusive: %s)\n", s3, s4);
fefdc04b38725457a91651218feb7000f6ccc1f4Lennart Poettering s3 = format_bytes(bs, sizeof(bs), i->limit);
fefdc04b38725457a91651218feb7000f6ccc1f4Lennart Poettering s4 = i->limit_exclusive != i->limit ? format_bytes(bs_exclusive, sizeof(bs_exclusive), i->limit_exclusive) : NULL;
fefdc04b38725457a91651218feb7000f6ccc1f4Lennart Poettering printf("\t Limit: %s (exclusive: %s)\n", s3, s4);
9f6eb1cd58f2ddf2eb6ba0e4de056e13d938af75Kay Sieversstatic int show_image_info(const char *verb, sd_bus *bus, const char *path, bool *new_line) {
1ee306e1248866617c96ed9f4263f375588ad838Lennart Poettering static const struct bus_properties_map map[] = {
56159e0d918e9a9be07988133bb2847779325de0Lennart Poettering { "Name", "s", NULL, offsetof(ImageStatusInfo, name) },
a1da85830bfaa77b9eb9c54693e5573559c97e50Tom Gundersen { "Path", "s", NULL, offsetof(ImageStatusInfo, path) },
56159e0d918e9a9be07988133bb2847779325de0Lennart Poettering { "Type", "s", NULL, offsetof(ImageStatusInfo, type) },
56159e0d918e9a9be07988133bb2847779325de0Lennart Poettering { "ReadOnly", "b", NULL, offsetof(ImageStatusInfo, read_only) },
1ee306e1248866617c96ed9f4263f375588ad838Lennart Poettering { "CreationTimestamp", "t", NULL, offsetof(ImageStatusInfo, crtime) },
56159e0d918e9a9be07988133bb2847779325de0Lennart Poettering { "ModificationTimestamp", "t", NULL, offsetof(ImageStatusInfo, mtime) },
1ee306e1248866617c96ed9f4263f375588ad838Lennart Poettering { "Usage", "t", NULL, offsetof(ImageStatusInfo, usage) },
1ee306e1248866617c96ed9f4263f375588ad838Lennart Poettering { "Limit", "t", NULL, offsetof(ImageStatusInfo, limit) },
1ee306e1248866617c96ed9f4263f375588ad838Lennart Poettering { "UsageExclusive", "t", NULL, offsetof(ImageStatusInfo, usage_exclusive) },
1ee306e1248866617c96ed9f4263f375588ad838Lennart Poettering { "LimitExclusive", "t", NULL, offsetof(ImageStatusInfo, limit_exclusive) },
a1da85830bfaa77b9eb9c54693e5573559c97e50Tom Gundersen "org.freedesktop.machine1",
1ee306e1248866617c96ed9f4263f375588ad838Lennart Poettering return log_error_errno(r, "Could not get properties: %m");
56159e0d918e9a9be07988133bb2847779325de0Lennart Poetteringstatic int show_image_properties(sd_bus *bus, const char *path, bool *new_line) {
923d8fd381bced1c2d90ca53d18629d61a0f454aLennart Poettering r = bus_print_all_properties(bus, "org.freedesktop.machine1", path, arg_property, arg_all);
923d8fd381bced1c2d90ca53d18629d61a0f454aLennart Poettering log_error_errno(r, "Could not get properties: %m");
923d8fd381bced1c2d90ca53d18629d61a0f454aLennart Poetteringstatic int show_image(int argc, char *argv[], void *userdata) {
56159e0d918e9a9be07988133bb2847779325de0Lennart Poettering _cleanup_bus_error_free_ sd_bus_error error = SD_BUS_ERROR_NULL;
923d8fd381bced1c2d90ca53d18629d61a0f454aLennart Poettering _cleanup_bus_message_unref_ sd_bus_message *reply = NULL;
785890acf6d629ff881a1f065f431df1b7fc8c7aLennart Poettering /* If no argument is specified, inspect the manager
785890acf6d629ff881a1f065f431df1b7fc8c7aLennart Poettering r = show_image_properties(bus, "/org/freedesktop/machine1", &new_line);
785890acf6d629ff881a1f065f431df1b7fc8c7aLennart Poettering "org.freedesktop.machine1",
785890acf6d629ff881a1f065f431df1b7fc8c7aLennart Poettering "org.freedesktop.machine1.Manager",
785890acf6d629ff881a1f065f431df1b7fc8c7aLennart Poettering log_error("Could not get path to image: %s", bus_error_message(&error, -r));
785890acf6d629ff881a1f065f431df1b7fc8c7aLennart Poettering r = sd_bus_message_read(reply, "o", &path);
785890acf6d629ff881a1f065f431df1b7fc8c7aLennart Poettering r = show_image_properties(bus, path, &new_line);
785890acf6d629ff881a1f065f431df1b7fc8c7aLennart Poettering r = show_image_info(argv[0], bus, path, &new_line);
785890acf6d629ff881a1f065f431df1b7fc8c7aLennart Poetteringstatic int kill_machine(int argc, char *argv[], void *userdata) {
785890acf6d629ff881a1f065f431df1b7fc8c7aLennart Poettering _cleanup_bus_error_free_ sd_bus_error error = SD_BUS_ERROR_NULL;
56159e0d918e9a9be07988133bb2847779325de0Lennart Poettering "org.freedesktop.machine1",
56159e0d918e9a9be07988133bb2847779325de0Lennart Poettering "org.freedesktop.machine1.Manager",
56159e0d918e9a9be07988133bb2847779325de0Lennart Poettering "KillMachine",
f2cbe59e113f08549949a76ac5b9b3972df4cc30Lennart Poettering "ssi", argv[i], arg_kill_who, arg_signal);
f2cbe59e113f08549949a76ac5b9b3972df4cc30Lennart Poettering log_error("Could not kill machine: %s", bus_error_message(&error, -r));
f2cbe59e113f08549949a76ac5b9b3972df4cc30Lennart Poetteringstatic int reboot_machine(int argc, char *argv[], void *userdata) {
f2cbe59e113f08549949a76ac5b9b3972df4cc30Lennart Poettering arg_signal = SIGINT; /* sysvinit + systemd */
56159e0d918e9a9be07988133bb2847779325de0Lennart Poettering return kill_machine(argc, argv, userdata);
f2cbe59e113f08549949a76ac5b9b3972df4cc30Lennart Poetteringstatic int poweroff_machine(int argc, char *argv[], void *userdata) {
f2cbe59e113f08549949a76ac5b9b3972df4cc30Lennart Poettering arg_signal = SIGRTMIN+4; /* only systemd */
f2cbe59e113f08549949a76ac5b9b3972df4cc30Lennart Poettering return kill_machine(argc, argv, userdata);
f2cbe59e113f08549949a76ac5b9b3972df4cc30Lennart Poetteringstatic int terminate_machine(int argc, char *argv[], void *userdata) {
f2cbe59e113f08549949a76ac5b9b3972df4cc30Lennart Poettering _cleanup_bus_error_free_ sd_bus_error error = SD_BUS_ERROR_NULL;
f2cbe59e113f08549949a76ac5b9b3972df4cc30Lennart Poettering "org.freedesktop.machine1",
f2cbe59e113f08549949a76ac5b9b3972df4cc30Lennart Poettering "org.freedesktop.machine1.Manager",
f2cbe59e113f08549949a76ac5b9b3972df4cc30Lennart Poettering "TerminateMachine",
f2cbe59e113f08549949a76ac5b9b3972df4cc30Lennart Poettering log_error("Could not terminate machine: %s", bus_error_message(&error, -r));
f2cbe59e113f08549949a76ac5b9b3972df4cc30Lennart Poetteringstatic int machine_get_leader(sd_bus *bus, const char *name, pid_t *ret) {
f2cbe59e113f08549949a76ac5b9b3972df4cc30Lennart Poettering _cleanup_bus_error_free_ sd_bus_error error = SD_BUS_ERROR_NULL;
f2cbe59e113f08549949a76ac5b9b3972df4cc30Lennart Poettering _cleanup_bus_message_unref_ sd_bus_message *reply = NULL, *reply2 = NULL;
f2cbe59e113f08549949a76ac5b9b3972df4cc30Lennart Poettering "org.freedesktop.machine1",
f2cbe59e113f08549949a76ac5b9b3972df4cc30Lennart Poettering "org.freedesktop.machine1.Manager",
785890acf6d629ff881a1f065f431df1b7fc8c7aLennart Poettering log_error("Could not get path to machine: %s", bus_error_message(&error, -r));
785890acf6d629ff881a1f065f431df1b7fc8c7aLennart Poettering r = sd_bus_message_read(reply, "o", &object);
56159e0d918e9a9be07988133bb2847779325de0Lennart Poettering "org.freedesktop.machine1",
785890acf6d629ff881a1f065f431df1b7fc8c7aLennart Poettering "org.freedesktop.machine1.Machine",
785890acf6d629ff881a1f065f431df1b7fc8c7aLennart Poettering return log_error_errno(r, "Failed to retrieve PID of leader: %m");
785890acf6d629ff881a1f065f431df1b7fc8c7aLennart Poettering r = sd_bus_message_read(reply2, "u", &leader);
56159e0d918e9a9be07988133bb2847779325de0Lennart Poetteringstatic int copy_files(int argc, char *argv[], void *userdata) {
785890acf6d629ff881a1f065f431df1b7fc8c7aLennart Poettering char *dest, *host_path, *container_path, *host_dirname, *host_basename, *container_dirname, *container_basename, *t;
785890acf6d629ff881a1f065f431df1b7fc8c7aLennart Poettering host_path = strdupa(copy_from ? dest : argv[2]);
785890acf6d629ff881a1f065f431df1b7fc8c7aLennart Poettering container_path = strdupa(copy_from ? argv[2] : dest);
785890acf6d629ff881a1f065f431df1b7fc8c7aLennart Poettering log_error("Container path not absolute.");
785890acf6d629ff881a1f065f431df1b7fc8c7aLennart Poettering container_dirname = dirname(container_path);
785890acf6d629ff881a1f065f431df1b7fc8c7aLennart Poettering r = machine_get_leader(bus, argv[1], &leader);
785890acf6d629ff881a1f065f431df1b7fc8c7aLennart Poettering hostfd = open(host_dirname, O_CLOEXEC|O_RDONLY|O_NOCTTY|O_DIRECTORY);
785890acf6d629ff881a1f065f431df1b7fc8c7aLennart Poettering return log_error_errno(errno, "Failed to open source directory: %m");
785890acf6d629ff881a1f065f431df1b7fc8c7aLennart Poettering return log_error_errno(errno, "Failed to fork(): %m");
785890acf6d629ff881a1f065f431df1b7fc8c7aLennart Poettering const char *q;
785890acf6d629ff881a1f065f431df1b7fc8c7aLennart Poettering q = procfs_file_alloca(leader, "ns/mnt");
785890acf6d629ff881a1f065f431df1b7fc8c7aLennart Poettering mntfd = open(q, O_RDONLY|O_NOCTTY|O_CLOEXEC);
785890acf6d629ff881a1f065f431df1b7fc8c7aLennart Poettering log_error_errno(errno, "Failed to open mount namespace of leader: %m");
785890acf6d629ff881a1f065f431df1b7fc8c7aLennart Poettering log_error_errno(errno, "Failed to join namespace of leader: %m");
56159e0d918e9a9be07988133bb2847779325de0Lennart Poettering containerfd = open(container_dirname, O_CLOEXEC|O_RDONLY|O_NOCTTY|O_DIRECTORY);
785890acf6d629ff881a1f065f431df1b7fc8c7aLennart Poettering log_error_errno(errno, "Failed top open destination directory: %m");
785890acf6d629ff881a1f065f431df1b7fc8c7aLennart Poettering r = copy_tree_at(containerfd, container_basename, hostfd, host_basename, true);
785890acf6d629ff881a1f065f431df1b7fc8c7aLennart Poettering r = copy_tree_at(hostfd, host_basename, containerfd, container_basename, true);
785890acf6d629ff881a1f065f431df1b7fc8c7aLennart Poettering log_error_errno(errno, "Failed to copy tree: %m");
785890acf6d629ff881a1f065f431df1b7fc8c7aLennart Poettering return log_error_errno(r, "Failed to wait for client: %m");
785890acf6d629ff881a1f065f431df1b7fc8c7aLennart Poetteringstatic int bind_mount(int argc, char *argv[], void *userdata) {
785890acf6d629ff881a1f065f431df1b7fc8c7aLennart Poettering char mount_slave[] = "/tmp/propagate.XXXXXX", *mount_tmp, *mount_outside, *p;
785890acf6d629ff881a1f065f431df1b7fc8c7aLennart Poettering bool mount_slave_created = false, mount_slave_mounted = false,
785890acf6d629ff881a1f065f431df1b7fc8c7aLennart Poettering mount_tmp_created = false, mount_tmp_mounted = false,
785890acf6d629ff881a1f065f431df1b7fc8c7aLennart Poettering mount_outside_created = false, mount_outside_mounted = false;
785890acf6d629ff881a1f065f431df1b7fc8c7aLennart Poettering /* One day, when bind mounting /proc/self/fd/n works across
785890acf6d629ff881a1f065f431df1b7fc8c7aLennart Poettering * namespace boundaries we should rework this logic to make
785890acf6d629ff881a1f065f431df1b7fc8c7aLennart Poettering * use of it... */
785890acf6d629ff881a1f065f431df1b7fc8c7aLennart Poettering log_error("Destination path not absolute.");
785890acf6d629ff881a1f065f431df1b7fc8c7aLennart Poettering p = strappenda("/run/systemd/nspawn/propagate/", argv[1], "/");
785890acf6d629ff881a1f065f431df1b7fc8c7aLennart Poettering log_error("Container does not allow propagation of mount points.");
785890acf6d629ff881a1f065f431df1b7fc8c7aLennart Poettering r = machine_get_leader(bus, argv[1], &leader);
785890acf6d629ff881a1f065f431df1b7fc8c7aLennart Poettering /* Our goal is to install a new bind mount into the container,
785890acf6d629ff881a1f065f431df1b7fc8c7aLennart Poettering possibly read-only. This is irritatingly complex
785890acf6d629ff881a1f065f431df1b7fc8c7aLennart Poettering unfortunately, currently.
785890acf6d629ff881a1f065f431df1b7fc8c7aLennart Poettering First, we start by creating a private playground in /tmp,
785890acf6d629ff881a1f065f431df1b7fc8c7aLennart Poettering that we can mount MS_SLAVE. (Which is necessary, since
785890acf6d629ff881a1f065f431df1b7fc8c7aLennart Poettering MS_MOUNT cannot be applied to mounts with MS_SHARED parent
785890acf6d629ff881a1f065f431df1b7fc8c7aLennart Poettering return log_error_errno(errno, "Failed to create playground: %m");
785890acf6d629ff881a1f065f431df1b7fc8c7aLennart Poettering if (mount(mount_slave, mount_slave, NULL, MS_BIND, NULL) < 0) {
785890acf6d629ff881a1f065f431df1b7fc8c7aLennart Poettering r = log_error_errno(errno, "Failed to make bind mount: %m");
785890acf6d629ff881a1f065f431df1b7fc8c7aLennart Poettering if (mount(NULL, mount_slave, NULL, MS_SLAVE, NULL) < 0) {
785890acf6d629ff881a1f065f431df1b7fc8c7aLennart Poettering r = log_error_errno(errno, "Failed to remount slave: %m");
785890acf6d629ff881a1f065f431df1b7fc8c7aLennart Poettering /* Second, we mount the source directory to a directory inside
785890acf6d629ff881a1f065f431df1b7fc8c7aLennart Poettering of our MS_SLAVE playground. */
785890acf6d629ff881a1f065f431df1b7fc8c7aLennart Poettering mount_tmp = strappenda(mount_slave, "/mount");
785890acf6d629ff881a1f065f431df1b7fc8c7aLennart Poettering r = log_error_errno(errno, "Failed to create temporary mount: %m");
023fb90b83871a15ef7f57e8cd126e3426f99b9eLennart Poettering if (mount(argv[2], mount_tmp, NULL, MS_BIND, NULL) < 0) {
023fb90b83871a15ef7f57e8cd126e3426f99b9eLennart Poettering r = log_error_errno(errno, "Failed to overmount: %m");
04d39279245834494baccfdb9349db8bf80abd13Lennart Poettering /* Third, we remount the new bind mount read-only if requested. */
04d39279245834494baccfdb9349db8bf80abd13Lennart Poettering if (mount(NULL, mount_tmp, NULL, MS_BIND|MS_REMOUNT|MS_RDONLY, NULL) < 0) {
bf441e3d9371a7e5aa1def66cfc40f0118884644Lennart Poettering r = log_error_errno(errno, "Failed to mark read-only: %m");
04d39279245834494baccfdb9349db8bf80abd13Lennart Poettering /* Fourth, we move the new bind mount into the propagation
04d39279245834494baccfdb9349db8bf80abd13Lennart Poettering * directory. This way it will appear there read-only
023fb90b83871a15ef7f57e8cd126e3426f99b9eLennart Poettering * right-away. */
f647962d64e844689f3e2acfce6102fc47e76df2Michal Schmidt mount_outside = strappenda("/run/systemd/nspawn/propagate/", argv[1], "/XXXXXX");
023fb90b83871a15ef7f57e8cd126e3426f99b9eLennart Poettering r = log_error_errno(errno, "Cannot create propagation directory: %m");
d04c1fb8e215600b4950c6778c6c16ddafc14024Lennart Poettering if (mount(mount_tmp, mount_outside, NULL, MS_MOVE, NULL) < 0) {
d04c1fb8e215600b4950c6778c6c16ddafc14024Lennart Poettering r = log_error_errno(errno, "Failed to move: %m");
40205d706e1210763ff4c98a317556375bd04bcdLennart Poettering r = log_error_errno(errno, "Failed to fork(): %m");
04d39279245834494baccfdb9349db8bf80abd13Lennart Poettering const char *q;
04d39279245834494baccfdb9349db8bf80abd13Lennart Poettering q = procfs_file_alloca(leader, "ns/mnt");
56159e0d918e9a9be07988133bb2847779325de0Lennart Poettering mntfd = open(q, O_RDONLY|O_NOCTTY|O_CLOEXEC);
023fb90b83871a15ef7f57e8cd126e3426f99b9eLennart Poettering log_error_errno(errno, "Failed to open mount namespace of leader: %m");
f647962d64e844689f3e2acfce6102fc47e76df2Michal Schmidt log_error_errno(errno, "Failed to join namespace of leader: %m");
c7b7d4493aa03e9ef5fb1e670b8969a48aa494ddLennart Poettering /* Fifth, move the mount to the right place inside */
023fb90b83871a15ef7f57e8cd126e3426f99b9eLennart Poettering mount_inside = strappenda("/run/systemd/nspawn/incoming/", basename(mount_outside));
023fb90b83871a15ef7f57e8cd126e3426f99b9eLennart Poettering if (mount(mount_inside, dest, NULL, MS_MOVE, NULL) < 0) {
c7b7d4493aa03e9ef5fb1e670b8969a48aa494ddLennart Poettering log_error_errno(errno, "Failed to mount: %m");
086821244b5113f00a0ef993b78dc56aae2a8f6cLennart Poettering log_error_errno(r, "Failed to wait for client: %m");
f2cbe59e113f08549949a76ac5b9b3972df4cc30Lennart Poetteringstatic int on_machine_removed(sd_bus *bus, sd_bus_message *m, void *userdata, sd_bus_error *ret_error) {
f2cbe59e113f08549949a76ac5b9b3972df4cc30Lennart Poettering PTYForward ** forward = (PTYForward**) userdata;
f2cbe59e113f08549949a76ac5b9b3972df4cc30Lennart Poettering /* If the forwarder is already initialized, tell it to
f2cbe59e113f08549949a76ac5b9b3972df4cc30Lennart Poettering * exit on the next vhangup(), so that we still flush
f2cbe59e113f08549949a76ac5b9b3972df4cc30Lennart Poettering * out what might be queued and exit then. */
f2cbe59e113f08549949a76ac5b9b3972df4cc30Lennart Poettering r = pty_forward_set_ignore_vhangup(*forward, false);
f2cbe59e113f08549949a76ac5b9b3972df4cc30Lennart Poettering log_error_errno(r, "Failed to set ignore_vhangup flag: %m");
f2cbe59e113f08549949a76ac5b9b3972df4cc30Lennart Poettering /* On error, or when the forwarder is not initialized yet, quit immediately */
f2cbe59e113f08549949a76ac5b9b3972df4cc30Lennart Poettering sd_event_exit(sd_bus_get_event(bus), EXIT_FAILURE);
fefdc04b38725457a91651218feb7000f6ccc1f4Lennart Poetteringstatic int login_machine(int argc, char *argv[], void *userdata) {
fefdc04b38725457a91651218feb7000f6ccc1f4Lennart Poettering _cleanup_bus_error_free_ sd_bus_error error = SD_BUS_ERROR_NULL;
086821244b5113f00a0ef993b78dc56aae2a8f6cLennart Poettering _cleanup_bus_message_unref_ sd_bus_message *m = NULL, *reply = NULL;
086821244b5113f00a0ef993b78dc56aae2a8f6cLennart Poettering _cleanup_bus_slot_unref_ sd_bus_slot *slot = NULL;
1ee306e1248866617c96ed9f4263f375588ad838Lennart Poettering _cleanup_(pty_forward_freep) PTYForward *forward = NULL;
56159e0d918e9a9be07988133bb2847779325de0Lennart Poettering _cleanup_event_unref_ sd_event *event = NULL;
1ee306e1248866617c96ed9f4263f375588ad838Lennart Poettering if (arg_transport != BUS_TRANSPORT_LOCAL &&
785890acf6d629ff881a1f065f431df1b7fc8c7aLennart Poettering arg_transport != BUS_TRANSPORT_MACHINE) {
785890acf6d629ff881a1f065f431df1b7fc8c7aLennart Poettering log_error("Login only supported on local machines.");
1ee306e1248866617c96ed9f4263f375588ad838Lennart Poettering return log_error_errno(r, "Failed to get event loop: %m");
e56056e93d33619a3acf13e483900b4f8938228fThomas Hindoe Paaboel Andersen r = sd_bus_attach_event(bus, event, 0);
1ee306e1248866617c96ed9f4263f375588ad838Lennart Poettering return log_error_errno(r, "Failed to attach bus to event loop: %m");
785890acf6d629ff881a1f065f431df1b7fc8c7aLennart Poettering "sender='org.freedesktop.machine1',"
eb9da376d76b48585b3b63b4f91903b54f7abd36Lennart Poettering "interface='org.freedesktop.machine1.Manager',"
1ee306e1248866617c96ed9f4263f375588ad838Lennart Poettering "member='MachineRemoved',"
1ee306e1248866617c96ed9f4263f375588ad838Lennart Poettering r = sd_bus_add_match(bus, &slot, match, on_machine_removed, &forward);
601185b43da638b1c74153deae01dbd518680889Zbigniew Jędrzejewski-Szmek return log_error_errno(r, "Failed to add machine removal match: %m");
1ee306e1248866617c96ed9f4263f375588ad838Lennart Poettering "org.freedesktop.machine1",
1ee306e1248866617c96ed9f4263f375588ad838Lennart Poettering "org.freedesktop.machine1.Manager",
1ee306e1248866617c96ed9f4263f375588ad838Lennart Poettering "OpenMachineLogin");
1ee306e1248866617c96ed9f4263f375588ad838Lennart Poettering r = sd_bus_message_set_allow_interactive_authorization(m, arg_ask_password);
a7893c6b28772edbc7e1fea3c209caa54d465648Lennart Poettering r = sd_bus_message_append(m, "s", argv[1]);
1ee306e1248866617c96ed9f4263f375588ad838Lennart Poettering r = sd_bus_call(bus, m, 0, &error, &reply);
1ee306e1248866617c96ed9f4263f375588ad838Lennart Poettering log_error("Failed to get machine PTY: %s", bus_error_message(&error, -r));
1ee306e1248866617c96ed9f4263f375588ad838Lennart Poettering r = sd_bus_message_read(reply, "hs", &master, &pty);
1ee306e1248866617c96ed9f4263f375588ad838Lennart Poettering sigset_add_many(&mask, SIGWINCH, SIGTERM, SIGINT, -1);
1ee306e1248866617c96ed9f4263f375588ad838Lennart Poettering assert_se(sigprocmask(SIG_BLOCK, &mask, NULL) == 0);
1ee306e1248866617c96ed9f4263f375588ad838Lennart Poettering log_info("Connected to machine %s. Press ^] three times within 1s to exit session.", argv[1]);
e56056e93d33619a3acf13e483900b4f8938228fThomas Hindoe Paaboel Andersen sd_event_add_signal(event, NULL, SIGINT, NULL, NULL);
e56056e93d33619a3acf13e483900b4f8938228fThomas Hindoe Paaboel Andersen sd_event_add_signal(event, NULL, SIGTERM, NULL, NULL);
e56056e93d33619a3acf13e483900b4f8938228fThomas Hindoe Paaboel Andersen r = pty_forward_new(event, master, true, &forward);
1ee306e1248866617c96ed9f4263f375588ad838Lennart Poettering return log_error_errno(r, "Failed to create PTY forwarder: %m");
1ee306e1248866617c96ed9f4263f375588ad838Lennart Poettering return log_error_errno(r, "Failed to run event loop: %m");
1ee306e1248866617c96ed9f4263f375588ad838Lennart Poettering pty_forward_get_last_char(forward, &last_char);
1ee306e1248866617c96ed9f4263f375588ad838Lennart Poettering machine_died = pty_forward_get_ignore_vhangup(forward) == 0;
a7893c6b28772edbc7e1fea3c209caa54d465648Lennart Poettering log_info("Machine %s terminated.", argv[1]);
de33fc625725d199629ed074d6278504deb23debLennart Poettering log_info("Connection to machine %s terminated.", argv[1]);
785890acf6d629ff881a1f065f431df1b7fc8c7aLennart Poetteringstatic int remove_image(int argc, char *argv[], void *userdata) {
785890acf6d629ff881a1f065f431df1b7fc8c7aLennart Poettering _cleanup_bus_error_free_ sd_bus_error error = SD_BUS_ERROR_NULL;
1ee306e1248866617c96ed9f4263f375588ad838Lennart Poettering "org.freedesktop.machine1",
1ee306e1248866617c96ed9f4263f375588ad838Lennart Poettering "org.freedesktop.machine1.Manager",
1ee306e1248866617c96ed9f4263f375588ad838Lennart Poettering "RemoveImage",
56159e0d918e9a9be07988133bb2847779325de0Lennart Poettering log_error("Could not remove image: %s", bus_error_message(&error, -r));
56159e0d918e9a9be07988133bb2847779325de0Lennart Poetteringstatic int rename_image(int argc, char *argv[], void *userdata) {
56159e0d918e9a9be07988133bb2847779325de0Lennart Poettering _cleanup_bus_error_free_ sd_bus_error error = SD_BUS_ERROR_NULL;
1ee306e1248866617c96ed9f4263f375588ad838Lennart Poettering "org.freedesktop.machine1",
1ee306e1248866617c96ed9f4263f375588ad838Lennart Poettering "org.freedesktop.machine1.Manager",
1ee306e1248866617c96ed9f4263f375588ad838Lennart Poettering "RenameImage",
1ee306e1248866617c96ed9f4263f375588ad838Lennart Poettering log_error("Could not rename image: %s", bus_error_message(&error, -r));
1ee306e1248866617c96ed9f4263f375588ad838Lennart Poetteringstatic int clone_image(int argc, char *argv[], void *userdata) {
d21ed1ead18d16d35c30299a69d3366847f8a039Lennart Poettering _cleanup_bus_error_free_ sd_bus_error error = SD_BUS_ERROR_NULL;
1ee306e1248866617c96ed9f4263f375588ad838Lennart Poettering "org.freedesktop.machine1",
1ee306e1248866617c96ed9f4263f375588ad838Lennart Poettering "org.freedesktop.machine1.Manager",
return -EINVAL;
r = sd_bus_call_method(
bus,
"/org/freedesktop/machine1",
&error,
NULL,
return log_oom();
const char *object;
return -EINVAL;
return log_oom();
if (!unit)
return log_oom();
bus,
"org.freedesktop.systemd1",
"/org/freedesktop/systemd1",
"org.freedesktop.systemd1.Manager",
return bus_log_create_error(r);
return bus_log_create_error(r);
return bus_log_create_error(r);
return bus_log_parse_error(r);
return log_oom();
int carries_install_info = 0;
bus,
"org.freedesktop.systemd1",
"/org/freedesktop/systemd1",
"org.freedesktop.systemd1.Manager",
method);
return bus_log_create_error(r);
return bus_log_create_error(r);
return bus_log_create_error(r);
return -EINVAL;
return log_oom();
if (!unit)
return log_oom();
return bus_log_create_error(r);
r = sd_bus_message_close_container(m);
return bus_log_create_error(r);
return bus_log_create_error(r);
return bus_log_parse_error(r);
m = sd_bus_message_unref(m);
bus,
"org.freedesktop.systemd1",
"/org/freedesktop/systemd1",
"org.freedesktop.systemd1.Manager",
return bus_log_create_error(r);
return bus_log_create_error(r);
" show NAME... Show properties of one or more VMs/containers\n"
" terminate NAME... Terminate one or more VMs/containers\n"
case ARG_VERSION:
return log_oom();
arg_all = true;
arg_all = true;
arg_full = true;
return -EINVAL;
if (arg_output < 0) {
return -EINVAL;
case ARG_NO_PAGER:
arg_no_pager = true;
case ARG_NO_LEGEND:
arg_legend = false;
case ARG_KILL_WHO:
if (arg_signal < 0) {
return -EINVAL;
case ARG_NO_ASK_PASSWORD:
arg_ask_password = false;
case ARG_READ_ONLY:
arg_read_only = true;
case ARG_MKDIR:
arg_mkdir = true;
arg_quiet = true;
return -EINVAL;
log_open();
goto finish;
goto finish;
pager_close();