]> git.ipfire.org Git - thirdparty/systemd.git/commitdiff
wait-online: support $REQUIRED_OPER_STATE_FOR_ONLINE= in state file
authorYu Watanabe <watanabe.yu+github@gmail.com>
Wed, 6 Mar 2019 14:46:33 +0000 (23:46 +0900)
committerYu Watanabe <watanabe.yu+github@gmail.com>
Wed, 13 Mar 2019 05:29:03 +0000 (14:29 +0900)
This also make wait-online optionally take operstate in -i option,
and adds '--operstate' option to wait-online which also takes operstate.

Also, this contains huge refactoring for wait-online.

src/network/wait-online/link.c
src/network/wait-online/link.h
src/network/wait-online/manager.c
src/network/wait-online/manager.h
src/network/wait-online/wait-online.c

index 87d54fe6544cb3bea26eebf5d972bd91a40ebebf..201be477c20cec0a0430138c6f21ef08df43c2a9 100644 (file)
@@ -10,6 +10,7 @@
 
 int link_new(Manager *m, Link **ret, int ifindex, const char *ifname) {
         _cleanup_(link_freep) Link *l = NULL;
+        _cleanup_free_ char *n = NULL;
         int r;
 
         assert(m);
@@ -23,30 +24,33 @@ int link_new(Manager *m, Link **ret, int ifindex, const char *ifname) {
         if (r < 0)
                 return r;
 
-        l = new0(Link, 1);
-        if (!l)
+        n = strdup(ifname);
+        if (!n)
                 return -ENOMEM;
 
-        l->manager = m;
-
-        l->ifname = strdup(ifname);
-        if (!l->ifname)
+        l = new(Link, 1);
+        if (!l)
                 return -ENOMEM;
 
+        *l = (Link) {
+                .manager = m,
+                .ifname = TAKE_PTR(n),
+                .ifindex = ifindex,
+                .required_operstate = LINK_OPERSTATE_DEGRADED,
+        };
+
         r = hashmap_put(m->links_by_name, l->ifname, l);
         if (r < 0)
                 return r;
 
-        l->ifindex = ifindex;
-
         r = hashmap_put(m->links, INT_TO_PTR(ifindex), l);
         if (r < 0)
                 return r;
 
         if (ret)
                 *ret = l;
-        l = NULL;
 
+        TAKE_PTR(l);
         return 0;
 }
 
@@ -60,6 +64,7 @@ Link *link_free(Link *l) {
                 hashmap_remove(l->manager->links_by_name, l->ifname);
         }
 
+        free(l->state);
         free(l->ifname);
         return mfree(l);
  }
@@ -87,9 +92,8 @@ int link_update_rtnl(Link *l, sd_netlink_message *m) {
                 if (!new_ifname)
                         return -ENOMEM;
 
-                hashmap_remove(l->manager->links_by_name, l->ifname);
-                free(l->ifname);
-                l->ifname = new_ifname;
+                assert_se(hashmap_remove(l->manager->links_by_name, l->ifname) == l);
+                free_and_replace(l->ifname, new_ifname);
 
                 r = hashmap_put(l->manager->links_by_name, l->ifname, l);
                 if (r < 0)
@@ -100,17 +104,55 @@ int link_update_rtnl(Link *l, sd_netlink_message *m) {
 }
 
 int link_update_monitor(Link *l) {
-        assert(l);
+        _cleanup_free_ char *operstate = NULL, *required_operstate = NULL, *state = NULL;
+        LinkOperationalState s;
+        int r, ret = 0;
 
-        l->required_for_online = sd_network_link_get_required_for_online(l->ifindex) != 0;
+        assert(l);
+        assert(l->ifname);
 
-        l->operational_state = mfree(l->operational_state);
+        r = sd_network_link_get_required_for_online(l->ifindex);
+        if (r < 0)
+                ret = log_debug_errno(r,
+                                      "Failed to get whether link %s is required for online or not, "
+                                      "ignoring: %m", l->ifname);
+        else
+                l->required_for_online = r > 0;
 
-        sd_network_link_get_operational_state(l->ifindex, &l->operational_state);
+        r = sd_network_link_get_required_operstate_for_online(l->ifindex, &required_operstate);
+        if (r < 0)
+                ret = log_debug_errno(r, "Failed to get required operational state for link %s to be online, "
+                                      "ignoring: %m", l->ifname);
+        else {
+                s = link_operstate_from_string(required_operstate);
+                if (s < 0)
+                        ret = log_debug_errno(SYNTHETIC_ERRNO(EINVAL),
+                                              "Failed to parse required operational state for link %s to be online, "
+                                              "ignoring: %m", l->ifname);
+                else
+                        l->required_operstate = s;
+        }
 
-        l->state = mfree(l->state);
+        r = sd_network_link_get_operational_state(l->ifindex, &operstate);
+        if (r < 0)
+                ret = log_debug_errno(r, "Failed to get operational state of link %s, ignoring: %m",
+                                      l->ifname);
+        else {
+                s = link_operstate_from_string(operstate);
+                if (s < 0)
+                        ret = log_debug_errno(SYNTHETIC_ERRNO(EINVAL),
+                                              "Failed to parse operational state of link %s, ignoring: %m",
+                                              l->ifname);
+                else
+                        l->operational_state = s;
+        }
 
-        sd_network_link_get_setup_state(l->ifindex, &l->state);
+        r = sd_network_link_get_setup_state(l->ifindex, &state);
+        if (r < 0)
+                ret = log_debug_errno(r, "Failed to get setup state of link %s, ignoring: %m",
+                                      l->ifname);
+        else
+                free_and_replace(l->state, state);
 
-        return 0;
+        return ret;
 }
index 57e8f12e034ea544c1db23290fe1982599780104..07c2499e1613c708493b69907be7819c83a4b3ce 100644 (file)
@@ -3,6 +3,8 @@
 
 #include "sd-netlink.h"
 
+#include "network-util.h"
+
 typedef struct Link Link;
 typedef struct Manager Manager;
 
@@ -14,7 +16,8 @@ struct Link {
         unsigned flags;
 
         bool required_for_online;
-        char *operational_state;
+        LinkOperationalState required_operstate;
+        LinkOperationalState operational_state;
         char *state;
 };
 
index 67218b6db3635d596fe49744a76ccb6dbe2c288b..48f197e74f934b5c9a0b2f15c2bf1460a439780e 100644 (file)
@@ -13,7 +13,7 @@
 #include "time-util.h"
 #include "util.h"
 
-bool manager_ignore_link(Manager *m, Link *link) {
+static bool manager_ignore_link(Manager *m, Link *link) {
         assert(m);
         assert(link);
 
@@ -22,7 +22,7 @@ bool manager_ignore_link(Manager *m, Link *link) {
                 return true;
 
         /* if interfaces are given on the command line, ignore all others */
-        if (m->interfaces && !strv_contains(m->interfaces, link->ifname))
+        if (m->interfaces && !hashmap_contains(m->interfaces, link->ifname))
                 return true;
 
         if (!link->required_for_online)
@@ -32,45 +32,76 @@ bool manager_ignore_link(Manager *m, Link *link) {
         return strv_fnmatch(m->ignore, link->ifname, 0);
 }
 
+static int manager_link_is_online(Manager *m, Link *l, LinkOperationalState s) {
+        /* This returns the following:
+         * -EAGAIN: not processed by udev or networkd
+         *       0: operstate is not enough
+         *       1: online */
+
+        if (!l->state) {
+                log_debug("link %s has not yet been processed by udev",
+                          l->ifname);
+                return -EAGAIN;
+        }
+
+        if (STR_IN_SET(l->state, "configuring", "pending")) {
+                log_debug("link %s is being processed by networkd",
+                          l->ifname);
+                return -EAGAIN;
+        }
+
+        if (s < 0)
+                s = m->required_operstate >= 0 ? m->required_operstate : l->required_operstate;
+
+        if (l->operational_state < s) {
+                log_debug("Operational state of link %s does not reach to %s",
+                          l->ifname, link_operstate_to_string(s));
+                return 0;
+        }
+
+        return 1;
+}
+
 bool manager_all_configured(Manager *m) {
+        bool one_ready = false;
         Iterator i;
+        const char *ifname;
+        void *p;
         Link *l;
-        char **ifname;
-        bool one_ready = false;
+        int r;
 
         /* wait for all the links given on the command line to appear */
-        STRV_FOREACH(ifname, m->interfaces) {
-                l = hashmap_get(m->links_by_name, *ifname);
+        HASHMAP_FOREACH_KEY(p, ifname, m->interfaces, i) {
+                LinkOperationalState s = PTR_TO_INT(p);
+
+                l = hashmap_get(m->links_by_name, ifname);
                 if (!l) {
-                        log_debug("still waiting for %s", *ifname);
+                        log_debug("still waiting for %s", ifname);
                         return false;
                 }
+
+                if (manager_link_is_online(m, l, s) <= 0)
+                        return false;
         }
 
+        if (!hashmap_isempty(m->interfaces))
+                /* all interfaces given by the command line are online. */
+                return true;
+
         /* wait for all links networkd manages to be in admin state 'configured'
-           and at least one link to gain a carrier */
+         * and at least one link to gain a carrier */
         HASHMAP_FOREACH(l, m->links, i) {
                 if (manager_ignore_link(m, l)) {
                         log_info("ignoring: %s", l->ifname);
                         continue;
                 }
 
-                if (!l->state) {
-                        log_debug("link %s has not yet been processed by udev",
-                                  l->ifname);
-                        return false;
-                }
-
-                if (STR_IN_SET(l->state, "configuring", "pending")) {
-                        log_debug("link %s is being processed by networkd",
-                                  l->ifname);
+                r = manager_link_is_online(m, l, _LINK_OPERSTATE_INVALID);
+                if (r < 0)
                         return false;
-                }
-
-                if (l->operational_state &&
-                    STR_IN_SET(l->operational_state, "degraded", "routable"))
+                if (r > 0)
                         /* we wait for at least one link to be ready,
-                           regardless of who manages it */
+                         * regardless of who manages it */
                         one_ready = true;
         }
 
@@ -120,15 +151,15 @@ static int manager_process_link(sd_netlink *rtnl, sd_netlink_message *mm, void *
                         r = link_new(m, &l, ifindex, ifname);
                         if (r < 0)
                                 return log_error_errno(r, "Failed to create link object: %m");
-
-                        r = link_update_monitor(l);
-                        if (r < 0)
-                                return log_error_errno(r, "Failed to initialize link object: %m");
                 }
 
                 r = link_update_rtnl(l, mm);
                 if (r < 0)
-                        return log_warning_errno(r, "Failed to process RTNL link message: %m");;
+                        log_warning_errno(r, "Failed to process RTNL link message, ignoring: %m");
+
+                r = link_update_monitor(l);
+                if (r < 0)
+                        log_warning_errno(r, "Failed to initialize link object, ignoring: %m");
 
                 break;
 
@@ -251,18 +282,22 @@ static int manager_network_monitor_listen(Manager *m) {
         return 0;
 }
 
-int manager_new(Manager **ret, char **interfaces, char **ignore, usec_t timeout) {
+int manager_new(Manager **ret, Hashmap *interfaces, char **ignore,
+                LinkOperationalState required_operstate, usec_t timeout) {
         _cleanup_(manager_freep) Manager *m = NULL;
         int r;
 
         assert(ret);
 
-        m = new0(Manager, 1);
+        m = new(Manager, 1);
         if (!m)
                 return -ENOMEM;
 
-        m->interfaces = interfaces;
-        m->ignore = ignore;
+        *m = (Manager) {
+                .interfaces = interfaces,
+                .ignore = ignore,
+                .required_operstate = required_operstate,
+        };
 
         r = sd_event_default(&m->event);
         if (r < 0)
index 0fde272ddbe763f8ea85a5780b90f4945efe414f..eb81e106e11669a6bb00937abd16ac41a3b0ba39 100644 (file)
@@ -6,6 +6,7 @@
 #include "sd-network.h"
 
 #include "hashmap.h"
+#include "network-util.h"
 
 typedef struct Manager Manager;
 typedef struct Link Link;
@@ -14,9 +15,12 @@ struct Manager {
         Hashmap *links;
         Hashmap *links_by_name;
 
-        char **interfaces;
+        /* Do not free the two members below. */
+        Hashmap *interfaces;
         char **ignore;
 
+        LinkOperationalState required_operstate;
+
         sd_netlink *rtnl;
         sd_event_source *rtnl_event_source;
 
@@ -27,9 +31,9 @@ struct Manager {
 };
 
 void manager_free(Manager *m);
-int manager_new(Manager **ret, char **interfaces, char **ignore, usec_t timeout);
+int manager_new(Manager **ret, Hashmap *interfaces, char **ignore,
+                LinkOperationalState required_operstate, usec_t timeout);
 
 DEFINE_TRIVIAL_CLEANUP_FUNC(Manager*, manager_free);
 
 bool manager_all_configured(Manager *m);
-bool manager_ignore_link(Manager *m, Link *link);
index 71b6cf6b87774b83f118725b6cc89563b04bc749..a5eddf3ca2689debbbb66ef005b1b3ccf9057d80 100644 (file)
@@ -9,14 +9,16 @@
 #include "manager.h"
 #include "pretty-print.h"
 #include "signal-util.h"
+#include "socket-util.h"
 #include "strv.h"
 
 static bool arg_quiet = false;
 static usec_t arg_timeout = 120 * USEC_PER_SEC;
-static char **arg_interfaces = NULL;
+static Hashmap *arg_interfaces = NULL;
 static char **arg_ignore = NULL;
+static LinkOperationalState arg_required_operstate = _LINK_OPERSTATE_INVALID;
 
-STATIC_DESTRUCTOR_REGISTER(arg_interfaces, strv_freep);
+STATIC_DESTRUCTOR_REGISTER(arg_interfaces, hashmap_free_free_keyp);
 STATIC_DESTRUCTOR_REGISTER(arg_ignore, strv_freep);
 
 static int help(void) {
@@ -32,8 +34,11 @@ static int help(void) {
                "  -h --help                 Show this help\n"
                "     --version              Print version string\n"
                "  -q --quiet                Do not show status information\n"
-               "  -i --interface=INTERFACE  Block until at least these interfaces have appeared\n"
+               "  -i --interface=INTERFACE[:OPERSTATE]\n"
+               "                            Block until at least these interfaces have appeared\n"
                "     --ignore=INTERFACE     Don't take these interfaces into account\n"
+               "  -o --operational-state=OPERSTATE\n"
+               "                            Required operational state\n"
                "     --timeout=SECS         Maximum time to wait for network connectivity\n"
                "\nSee the %s for details.\n"
                , program_invocation_short_name
@@ -43,6 +48,52 @@ static int help(void) {
         return 0;
 }
 
+static int parse_interface_with_operstate(const char *str) {
+        _cleanup_free_ char *ifname = NULL;
+        LinkOperationalState s;
+        const char *p;
+        int r;
+
+        assert(str);
+
+        p = strchr(str, ':');
+        if (p) {
+                if (isempty(p + 1))
+                        return log_error_errno(SYNTHETIC_ERRNO(EINVAL),
+                                               "Operational state is empty.");
+
+                s = link_operstate_from_string(p + 1);
+                if (s < 0)
+                        return log_error_errno(SYNTHETIC_ERRNO(EINVAL),
+                                               "Invalid operational state '%s'", p + 1);
+
+                ifname = strndup(optarg, p - optarg);
+        } else {
+                s = _LINK_OPERSTATE_INVALID;
+                ifname = strdup(str);
+        }
+        if (!ifname)
+                return log_oom();
+
+        if (!ifname_valid(ifname))
+                return log_error_errno(SYNTHETIC_ERRNO(EINVAL),
+                                       "Invalid interface name '%s'", ifname);
+
+        r = hashmap_ensure_allocated(&arg_interfaces, &string_hash_ops);
+        if (r < 0)
+                return log_oom();
+
+        r = hashmap_put(arg_interfaces, ifname, INT_TO_PTR(s));
+        if (r < 0)
+                return log_error_errno(r, "Failed to store interface name: %m");
+        if (r == 0)
+                return log_error_errno(SYNTHETIC_ERRNO(EINVAL),
+                                       "Interface name %s is already specified", ifname);
+
+        TAKE_PTR(ifname);
+        return 0;
+}
+
 static int parse_argv(int argc, char *argv[]) {
 
         enum {
@@ -52,12 +103,13 @@ static int parse_argv(int argc, char *argv[]) {
         };
 
         static const struct option options[] = {
-                { "help",            no_argument,       NULL, 'h'         },
-                { "version",         no_argument,       NULL, ARG_VERSION },
-                { "quiet",           no_argument,       NULL, 'q'         },
-                { "interface",       required_argument, NULL, 'i'         },
-                { "ignore",          required_argument, NULL, ARG_IGNORE  },
-                { "timeout",         required_argument, NULL, ARG_TIMEOUT  },
+                { "help",              no_argument,       NULL, 'h'         },
+                { "version",           no_argument,       NULL, ARG_VERSION },
+                { "quiet",             no_argument,       NULL, 'q'         },
+                { "interface",         required_argument, NULL, 'i'         },
+                { "ignore",            required_argument, NULL, ARG_IGNORE  },
+                { "operational-state", required_argument, NULL, 'o'         },
+                { "timeout",           required_argument, NULL, ARG_TIMEOUT },
                 {}
         };
 
@@ -66,7 +118,7 @@ static int parse_argv(int argc, char *argv[]) {
         assert(argc >= 0);
         assert(argv);
 
-        while ((c = getopt_long(argc, argv, "+hi:q", options, NULL)) >= 0)
+        while ((c = getopt_long(argc, argv, "hi:qo:", options, NULL)) >= 0)
 
                 switch (c) {
 
@@ -82,9 +134,9 @@ static int parse_argv(int argc, char *argv[]) {
                         return version();
 
                 case 'i':
-                        if (strv_extend(&arg_interfaces, optarg) < 0)
-                                return log_oom();
-
+                        r = parse_interface_with_operstate(optarg);
+                        if (r < 0)
+                                return r;
                         break;
 
                 case ARG_IGNORE:
@@ -93,11 +145,21 @@ static int parse_argv(int argc, char *argv[]) {
 
                         break;
 
+                case 'o': {
+                        LinkOperationalState s;
+
+                        s = link_operstate_from_string(optarg);
+                        if (s < 0)
+                                return log_error_errno(SYNTHETIC_ERRNO(EINVAL),
+                                                       "Invalid operational state '%s'", optarg);
+
+                        arg_required_operstate = s;
+                        break;
+                }
                 case ARG_TIMEOUT:
                         r = parse_sec(optarg, &arg_timeout);
                         if (r < 0)
                                 return r;
-
                         break;
 
                 case '?':
@@ -124,11 +186,11 @@ static int run(int argc, char *argv[]) {
                 return r;
 
         if (arg_quiet)
-                log_set_max_level(LOG_WARNING);
+                log_set_max_level(LOG_ERR);
 
         assert_se(sigprocmask_many(SIG_BLOCK, NULL, SIGTERM, SIGINT, -1) >= 0);
 
-        r = manager_new(&m, arg_interfaces, arg_ignore, arg_timeout);
+        r = manager_new(&m, arg_interfaces, arg_ignore, arg_required_operstate, arg_timeout);
         if (r < 0)
                 return log_error_errno(r, "Could not create manager: %m");