networkctl.c revision ab1525bc2d92780b4dc64bc5b9cdb52594e2df7d
ee8c45689526ca973407cbb77bce7b96a062c40bLennart Poettering/*-*- Mode: C; c-basic-offset: 8; indent-tabs-mode: nil -*-*/
ee8c45689526ca973407cbb77bce7b96a062c40bLennart Poettering This file is part of systemd.
ee8c45689526ca973407cbb77bce7b96a062c40bLennart Poettering Copyright 2014 Lennart Poettering
ee8c45689526ca973407cbb77bce7b96a062c40bLennart Poettering systemd is free software; you can redistribute it and/or modify it
ee8c45689526ca973407cbb77bce7b96a062c40bLennart Poettering under the terms of the GNU Lesser General Public License as published by
ee8c45689526ca973407cbb77bce7b96a062c40bLennart Poettering the Free Software Foundation; either version 2.1 of the License, or
ee8c45689526ca973407cbb77bce7b96a062c40bLennart Poettering (at your option) any later version.
ee8c45689526ca973407cbb77bce7b96a062c40bLennart Poettering systemd is distributed in the hope that it will be useful, but
ee8c45689526ca973407cbb77bce7b96a062c40bLennart Poettering WITHOUT ANY WARRANTY; without even the implied warranty of
ee8c45689526ca973407cbb77bce7b96a062c40bLennart Poettering MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
ee8c45689526ca973407cbb77bce7b96a062c40bLennart Poettering Lesser General Public License for more details.
ee8c45689526ca973407cbb77bce7b96a062c40bLennart Poettering You should have received a copy of the GNU Lesser General Public License
ee8c45689526ca973407cbb77bce7b96a062c40bLennart Poettering along with systemd; If not, see <http://www.gnu.org/licenses/>.
ee8c45689526ca973407cbb77bce7b96a062c40bLennart Poetteringstatic bool arg_no_pager = false;
ee8c45689526ca973407cbb77bce7b96a062c40bLennart Poetteringstatic bool arg_legend = true;
9085f64a6694f2928c79fcce365edb1dca6937d4Lennart Poetteringstatic bool arg_all = false;
ee8c45689526ca973407cbb77bce7b96a062c40bLennart Poetteringstatic void pager_open_if_enabled(void) {
ee8c45689526ca973407cbb77bce7b96a062c40bLennart Poetteringstatic int link_get_type_string(int iftype, struct udev_device *d, char **ret) {
ee8c45689526ca973407cbb77bce7b96a062c40bLennart Poettering const char *t;
ee8c45689526ca973407cbb77bce7b96a062c40bLennart Poettering /* WLANs have iftype ARPHRD_ETHER, but we want
ee8c45689526ca973407cbb77bce7b96a062c40bLennart Poettering * to show a more useful type string for
6d0c65ffb4f82e8c6dceb453919b3db54343fc27Lennart Poetteringtypedef struct LinkInfo {
6d0c65ffb4f82e8c6dceb453919b3db54343fc27Lennart Poetteringstatic int link_info_compare(const void *a, const void *b) {
6d0c65ffb4f82e8c6dceb453919b3db54343fc27Lennart Poettering const LinkInfo *x = a, *y = b;
6d0c65ffb4f82e8c6dceb453919b3db54343fc27Lennart Poetteringstatic int decode_and_sort_links(sd_rtnl_message *m, LinkInfo **ret) {
6d0c65ffb4f82e8c6dceb453919b3db54343fc27Lennart Poettering for (i = m; i; i = sd_rtnl_message_next(i)) {
6d0c65ffb4f82e8c6dceb453919b3db54343fc27Lennart Poettering r = sd_rtnl_message_link_get_ifindex(i, &ifindex);
6d0c65ffb4f82e8c6dceb453919b3db54343fc27Lennart Poettering r = sd_rtnl_message_read_string(i, IFLA_IFNAME, &name);
6d0c65ffb4f82e8c6dceb453919b3db54343fc27Lennart Poettering r = sd_rtnl_message_link_get_type(i, &iftype);
6d0c65ffb4f82e8c6dceb453919b3db54343fc27Lennart Poettering qsort(links, c, sizeof(LinkInfo), link_info_compare);
6d0c65ffb4f82e8c6dceb453919b3db54343fc27Lennart Poettering return (int) c;
ee8c45689526ca973407cbb77bce7b96a062c40bLennart Poetteringstatic int list_links(char **args, unsigned n) {
ee8c45689526ca973407cbb77bce7b96a062c40bLennart Poettering _cleanup_rtnl_message_unref_ sd_rtnl_message *req = NULL, *reply = NULL;
ee8c45689526ca973407cbb77bce7b96a062c40bLennart Poettering _cleanup_udev_unref_ struct udev *udev = NULL;
ee8c45689526ca973407cbb77bce7b96a062c40bLennart Poettering _cleanup_rtnl_unref_ sd_rtnl *rtnl = NULL;
ee8c45689526ca973407cbb77bce7b96a062c40bLennart Poettering log_error("Failed to connect to netlink: %s", strerror(-r));
ee8c45689526ca973407cbb77bce7b96a062c40bLennart Poettering log_error("Failed to connect to udev: %m");
ee8c45689526ca973407cbb77bce7b96a062c40bLennart Poettering r = sd_rtnl_message_new_link(rtnl, &req, RTM_GETLINK, 0);
ee8c45689526ca973407cbb77bce7b96a062c40bLennart Poettering r = sd_rtnl_message_request_dump(req, true);
ee8c45689526ca973407cbb77bce7b96a062c40bLennart Poettering log_error("Failed to enumerate links: %s", strerror(-r));
2368512a31f55e107f0cfb6e68cd4057648fd2deLennart Poettering printf("%3s %-16s %-10s %-11s %-10s\n", "IDX", "LINK", "TYPE", "SETUP", "OPERATIONAL");
6d0c65ffb4f82e8c6dceb453919b3db54343fc27Lennart Poettering c = decode_and_sort_links(reply, &links);
6d0c65ffb4f82e8c6dceb453919b3db54343fc27Lennart Poettering for (i = 0; i < c; i++) {
ab1525bc2d92780b4dc64bc5b9cdb52594e2df7dLennart Poettering _cleanup_free_ char *setup_state = NULL, *operational_state = NULL;
ee8c45689526ca973407cbb77bce7b96a062c40bLennart Poettering _cleanup_udev_device_unref_ struct udev_device *d = NULL;
636db02d54b64fb5bbad86c8015a9ef51299693bTom Gundersen const char *on_color_oper = "", *off_color_oper = "",
ab1525bc2d92780b4dc64bc5b9cdb52594e2df7dLennart Poettering sd_network_link_get_setup_state(links[i].ifindex, &setup_state);
d6731e4c7964ee2860d4f5abdb0b52acd7a66960Tom Gundersen sd_network_link_get_operational_state(links[i].ifindex, &operational_state);
ee8c45689526ca973407cbb77bce7b96a062c40bLennart Poettering d = udev_device_new_from_device_id(udev, devid);
6d0c65ffb4f82e8c6dceb453919b3db54343fc27Lennart Poettering link_get_type_string(links[i].iftype, d, &t);
df3fb561b2df486a495a5f0bcc83168bd1860533Lennart Poettering if (streq_ptr(operational_state, "routable")) {
636db02d54b64fb5bbad86c8015a9ef51299693bTom Gundersen } else if (streq_ptr(operational_state, "degraded")) {
ab1525bc2d92780b4dc64bc5b9cdb52594e2df7dLennart Poettering if (streq_ptr(setup_state, "configured")) {
ab1525bc2d92780b4dc64bc5b9cdb52594e2df7dLennart Poettering } else if (streq_ptr(setup_state, "configuring")) {
ab1525bc2d92780b4dc64bc5b9cdb52594e2df7dLennart Poettering } else if (streq_ptr(setup_state, "failed") ||
636db02d54b64fb5bbad86c8015a9ef51299693bTom Gundersen printf("%3i %-16s %-10s %s%-11s%s %s%-10s%s\n", links[i].ifindex,
ab1525bc2d92780b4dc64bc5b9cdb52594e2df7dLennart Poettering links[i].name, strna(t), on_color, strna(setup_state), off_color,
636db02d54b64fb5bbad86c8015a9ef51299693bTom Gundersen on_color_oper, strna(operational_state), off_color_oper);
ee8c45689526ca973407cbb77bce7b96a062c40bLennart Poetteringstatic int dump_addresses(sd_rtnl *rtnl, const char *prefix, int ifindex) {
ee8c45689526ca973407cbb77bce7b96a062c40bLennart Poettering _cleanup_free_ struct local_address *local = NULL;
ee8c45689526ca973407cbb77bce7b96a062c40bLennart Poettering n = local_addresses(rtnl, ifindex, &local);
ee8c45689526ca973407cbb77bce7b96a062c40bLennart Poettering for (i = 0; i < n; i++) {
ee8c45689526ca973407cbb77bce7b96a062c40bLennart Poettering r = in_addr_to_string(local[i].family, &local[i].address, &pretty);
ee8c45689526ca973407cbb77bce7b96a062c40bLennart Poetteringstatic void dump_list(const char *prefix, char **l) {
9085f64a6694f2928c79fcce365edb1dca6937d4Lennart Poetteringstatic int link_status_one(sd_rtnl *rtnl, struct udev *udev, const char *name) {
9085f64a6694f2928c79fcce365edb1dca6937d4Lennart Poettering _cleanup_strv_free_ char **dns = NULL, **ntp = NULL;
ab1525bc2d92780b4dc64bc5b9cdb52594e2df7dLennart Poettering _cleanup_free_ char *setup_state = NULL, *operational_state = NULL;
9085f64a6694f2928c79fcce365edb1dca6937d4Lennart Poettering _cleanup_rtnl_message_unref_ sd_rtnl_message *req = NULL, *reply = NULL;
9085f64a6694f2928c79fcce365edb1dca6937d4Lennart Poettering _cleanup_udev_device_unref_ struct udev_device *d = NULL;
9085f64a6694f2928c79fcce365edb1dca6937d4Lennart Poettering const char *driver = NULL, *path = NULL, *vendor = NULL, *model = NULL;
df3fb561b2df486a495a5f0bcc83168bd1860533Lennart Poettering const char *on_color = "", *off_color = "";
9085f64a6694f2928c79fcce365edb1dca6937d4Lennart Poettering if (safe_atoi(name, &ifindex) >= 0 && ifindex > 0)
9085f64a6694f2928c79fcce365edb1dca6937d4Lennart Poettering r = sd_rtnl_message_new_link(rtnl, &req, RTM_GETLINK, ifindex);
9085f64a6694f2928c79fcce365edb1dca6937d4Lennart Poettering r = sd_rtnl_message_new_link(rtnl, &req, RTM_GETLINK, 0);
9085f64a6694f2928c79fcce365edb1dca6937d4Lennart Poettering r = sd_rtnl_message_append_string(req, IFLA_IFNAME, name);
9085f64a6694f2928c79fcce365edb1dca6937d4Lennart Poettering log_error("Failed to query link: %s", strerror(-r));
9085f64a6694f2928c79fcce365edb1dca6937d4Lennart Poettering r = sd_rtnl_message_link_get_ifindex(reply, &ifindex);
9085f64a6694f2928c79fcce365edb1dca6937d4Lennart Poettering r = sd_rtnl_message_read_string(reply, IFLA_IFNAME, &name);
9085f64a6694f2928c79fcce365edb1dca6937d4Lennart Poettering r = sd_rtnl_message_link_get_type(reply, &iftype);
9085f64a6694f2928c79fcce365edb1dca6937d4Lennart Poettering have_mac = sd_rtnl_message_read_ether_addr(reply, IFLA_ADDRESS, &e) >= 0;
9085f64a6694f2928c79fcce365edb1dca6937d4Lennart Poettering for (p = (uint8_t*) &e; p < (uint8_t*) &e + sizeof(e); p++)
9085f64a6694f2928c79fcce365edb1dca6937d4Lennart Poettering if (*p != 0) {
9085f64a6694f2928c79fcce365edb1dca6937d4Lennart Poettering sd_rtnl_message_read_u32(reply, IFLA_MTU, &mtu);
ab1525bc2d92780b4dc64bc5b9cdb52594e2df7dLennart Poettering sd_network_link_get_setup_state(ifindex, &setup_state);
d6731e4c7964ee2860d4f5abdb0b52acd7a66960Tom Gundersen sd_network_link_get_operational_state(ifindex, &operational_state);
9085f64a6694f2928c79fcce365edb1dca6937d4Lennart Poettering d = udev_device_new_from_device_id(udev, devid);
9085f64a6694f2928c79fcce365edb1dca6937d4Lennart Poettering driver = udev_device_get_property_value(d, "ID_NET_DRIVER");
9085f64a6694f2928c79fcce365edb1dca6937d4Lennart Poettering path = udev_device_get_property_value(d, "ID_PATH");
9085f64a6694f2928c79fcce365edb1dca6937d4Lennart Poettering vendor = udev_device_get_property_value(d, "ID_VENDOR_FROM_DATABASE");
9085f64a6694f2928c79fcce365edb1dca6937d4Lennart Poettering vendor = udev_device_get_property_value(d, "ID_VENDOR");
9085f64a6694f2928c79fcce365edb1dca6937d4Lennart Poettering model = udev_device_get_property_value(d, "ID_MODEL_FROM_DATABASE");
9085f64a6694f2928c79fcce365edb1dca6937d4Lennart Poettering model = udev_device_get_property_value(d, "ID_MODEL");
df3fb561b2df486a495a5f0bcc83168bd1860533Lennart Poettering if (streq_ptr(operational_state, "routable")) {
df3fb561b2df486a495a5f0bcc83168bd1860533Lennart Poettering } else if (streq_ptr(operational_state, "degraded")) {
df3fb561b2df486a495a5f0bcc83168bd1860533Lennart Poettering printf("%s%s%s %i: %s\n", on_color, draw_special_char(DRAW_BLACK_CIRCLE), off_color, ifindex, name);
df3fb561b2df486a495a5f0bcc83168bd1860533Lennart Poettering " State: %s%s%s (%s)\n",
df3fb561b2df486a495a5f0bcc83168bd1860533Lennart Poettering on_color, strna(operational_state), off_color,
9085f64a6694f2928c79fcce365edb1dca6937d4Lennart Poettering printf(" HW Address: %s\n", ether_ntoa(&e));
9085f64a6694f2928c79fcce365edb1dca6937d4Lennart Poettering dump_addresses(rtnl, " Address: ", ifindex);
ee8c45689526ca973407cbb77bce7b96a062c40bLennart Poetteringstatic int link_status(char **args, unsigned n) {
ee8c45689526ca973407cbb77bce7b96a062c40bLennart Poettering _cleanup_udev_unref_ struct udev *udev = NULL;
ee8c45689526ca973407cbb77bce7b96a062c40bLennart Poettering _cleanup_rtnl_unref_ sd_rtnl *rtnl = NULL;
f7d68aa8c9430dfddee5f913f93b405a5d75723aLennart Poettering log_error("Failed to connect to netlink: %s", strerror(-r));
f7d68aa8c9430dfddee5f913f93b405a5d75723aLennart Poettering log_error("Failed to connect to udev: %m");
ee8c45689526ca973407cbb77bce7b96a062c40bLennart Poettering _cleanup_free_ char *operational_state = NULL;
03cc0fd1431b82e59c11ae12a274c1f2df23169dLennart Poettering _cleanup_strv_free_ char **dns = NULL, **ntp = NULL;
f7d68aa8c9430dfddee5f913f93b405a5d75723aLennart Poettering _cleanup_free_ struct local_address *addresses = NULL;
5323ead145304785121b72dcbc1c244303de7575Daniel Buch const char *on_color_oper = "", *off_color_oper = "";
03cc0fd1431b82e59c11ae12a274c1f2df23169dLennart Poettering sd_network_get_operational_state(&operational_state);
5323ead145304785121b72dcbc1c244303de7575Daniel Buch } else if (streq_ptr(operational_state, "degraded")) {
5323ead145304785121b72dcbc1c244303de7575Daniel Buch printf(" State: %s%s%s\n", on_color_oper, strna(operational_state), off_color_oper);
f7d68aa8c9430dfddee5f913f93b405a5d75723aLennart Poettering c = local_addresses(rtnl, 0, &addresses);
f7d68aa8c9430dfddee5f913f93b405a5d75723aLennart Poettering for (i = 0; i < c; i++) {
f7d68aa8c9430dfddee5f913f93b405a5d75723aLennart Poettering r = in_addr_to_string(addresses[i].family, &addresses[i].address, &pretty);
ee8c45689526ca973407cbb77bce7b96a062c40bLennart Poettering _cleanup_rtnl_message_unref_ sd_rtnl_message *req = NULL, *reply = NULL;
9085f64a6694f2928c79fcce365edb1dca6937d4Lennart Poettering r = sd_rtnl_message_new_link(rtnl, &req, RTM_GETLINK, 0);
9085f64a6694f2928c79fcce365edb1dca6937d4Lennart Poettering r = sd_rtnl_message_request_dump(req, true);
9085f64a6694f2928c79fcce365edb1dca6937d4Lennart Poettering log_error("Failed to enumerate links: %s", strerror(-r));
6d0c65ffb4f82e8c6dceb453919b3db54343fc27Lennart Poettering c = decode_and_sort_links(reply, &links);
6d0c65ffb4f82e8c6dceb453919b3db54343fc27Lennart Poettering for (i = 0; i < c; i++) {
6d0c65ffb4f82e8c6dceb453919b3db54343fc27Lennart Poettering link_status_one(rtnl, udev, links[i].name);
ee8c45689526ca973407cbb77bce7b96a062c40bLennart Poetteringstatic void help(void) {
ee8c45689526ca973407cbb77bce7b96a062c40bLennart Poettering "Query and control the networking subsystem.\n\n"
ee8c45689526ca973407cbb77bce7b96a062c40bLennart Poettering " -h --help Show this help\n"
9085f64a6694f2928c79fcce365edb1dca6937d4Lennart Poettering " --version Show package version\n"
9085f64a6694f2928c79fcce365edb1dca6937d4Lennart Poettering " --no-pager Do not pipe output into a pager\n"
9085f64a6694f2928c79fcce365edb1dca6937d4Lennart Poettering " --no-legend Do not show the headers and footers\n"
9085f64a6694f2928c79fcce365edb1dca6937d4Lennart Poettering " -a --all Show status for all links\n\n"
ee8c45689526ca973407cbb77bce7b96a062c40bLennart Poettering " list List links\n"
ee8c45689526ca973407cbb77bce7b96a062c40bLennart Poettering " status LINK Show link status\n"
ee8c45689526ca973407cbb77bce7b96a062c40bLennart Poetteringstatic int parse_argv(int argc, char *argv[]) {
ee8c45689526ca973407cbb77bce7b96a062c40bLennart Poettering { "version", no_argument, NULL, ARG_VERSION },
ee8c45689526ca973407cbb77bce7b96a062c40bLennart Poettering { "no-pager", no_argument, NULL, ARG_NO_PAGER },
ee8c45689526ca973407cbb77bce7b96a062c40bLennart Poettering { "no-legend", no_argument, NULL, ARG_NO_LEGEND },
9085f64a6694f2928c79fcce365edb1dca6937d4Lennart Poettering while ((c = getopt_long(argc, argv, "ha", options, NULL)) >= 0) {
ee8c45689526ca973407cbb77bce7b96a062c40bLennart Poetteringstatic int networkctl_main(int argc, char *argv[]) {
ee8c45689526ca973407cbb77bce7b96a062c40bLennart Poettering static const struct {
ee8c45689526ca973407cbb77bce7b96a062c40bLennart Poettering int (* const dispatch)(char **args, unsigned n);
ee8c45689526ca973407cbb77bce7b96a062c40bLennart Poettering /* Special rule: no arguments means "list" */
ee8c45689526ca973407cbb77bce7b96a062c40bLennart Poettering log_error("Unknown operation %s", argv[optind]);
ee8c45689526ca973407cbb77bce7b96a062c40bLennart Poettering log_error("Invalid number of arguments.");
ee8c45689526ca973407cbb77bce7b96a062c40bLennart Poettering assert_not_reached("Unknown comparison operator.");
ee8c45689526ca973407cbb77bce7b96a062c40bLennart Poettering return verbs[i].dispatch(argv + optind, left);