if (!a) {
if (link)
- r = link_get_address(link, t->family, &t->local, 0, &a);
+ r = link_get_address(link, t->family, &t->local, &a);
else
- r = manager_get_address(netdev->manager, t->family, &t->local, 0, &a);
+ r = manager_get_address(netdev->manager, t->family, &t->local, &a);
if (r < 0)
return r;
return 0;
}
-int link_get_address(Link *link, int family, const union in_addr_union *address, unsigned char prefixlen, Address **ret) {
+int link_get_address_full(
+ Link *link,
+ int family,
+ const union in_addr_union *address,
+ const union in_addr_union *peer, /* optional, can be NULL */
+ unsigned char prefixlen, /* optional, can be 0 */
+ Address **ret) {
+
Address *a;
int r;
assert(IN_SET(family, AF_INET, AF_INET6));
assert(address);
- /* This find an Address object on the link which matches the given address and prefix length
- * and does not have peer address. When the prefixlen is zero, then an Address object with an
- * arbitrary prefixlen will be returned. */
+ /* This finds an Address object on the link which matches the given address, peer, and prefix length.
+ * If the prefixlen is zero, then an Address object with an arbitrary prefixlen will be returned.
+ * If the peer is NULL, then an Address object with an arbitrary peer will be returned. */
- if (family == AF_INET6 || prefixlen != 0) {
+ if (family == AF_INET6 || (prefixlen != 0 && peer)) {
_cleanup_(address_unrefp) Address *tmp = NULL;
/* In this case, we can use address_get(). */
tmp->family = family;
tmp->in_addr = *address;
+ if (peer)
+ tmp->in_addr_peer = *peer;
tmp->prefixlen = prefixlen;
r = address_get(link, tmp, &a);
if (family == AF_INET6) {
/* IPv6 addresses are managed without peer address and prefix length. Hence, we need
* to check them explicitly. */
- if (in_addr_is_set(family, &a->in_addr_peer))
+ if (peer && !in_addr_equal(family, &a->in_addr_peer, peer))
return -ENOENT;
if (prefixlen != 0 && a->prefixlen != prefixlen)
return -ENOENT;
if (!in_addr_equal(family, &a->in_addr, address))
continue;
- if (in_addr_is_set(family, &a->in_addr_peer))
+ if (peer && !in_addr_equal(family, &a->in_addr_peer, peer))
+ continue;
+
+ if (prefixlen != 0 && a->prefixlen != prefixlen)
continue;
if (ret)
return -ENOENT;
}
-int manager_get_address(Manager *manager, int family, const union in_addr_union *address, unsigned char prefixlen, Address **ret) {
+int manager_get_address_full(
+ Manager *manager,
+ int family,
+ const union in_addr_union *address,
+ const union in_addr_union *peer,
+ unsigned char prefixlen,
+ Address **ret) {
+
Link *link;
assert(manager);
if (!IN_SET(link->state, LINK_STATE_CONFIGURING, LINK_STATE_CONFIGURED))
continue;
- if (link_get_address(link, family, address, prefixlen, ret) >= 0)
+ if (link_get_address_full(link, family, address, peer, prefixlen, ret) >= 0)
return 0;
}
return -ENOENT;
}
-bool manager_has_address(Manager *manager, int family, const union in_addr_union *address) {
- Address *a;
-
- assert(manager);
- assert(IN_SET(family, AF_INET, AF_INET6));
- assert(address);
-
- if (manager_get_address(manager, family, address, 0, &a) < 0)
- return false;
-
- return address_is_ready(a);
-}
-
const char* format_lifetime(char *buf, size_t l, usec_t lifetime_usec) {
assert(buf);
assert(l > 4);
int link_drop_ipv6ll_addresses(Link *link);
void link_foreignize_addresses(Link *link);
bool link_address_is_dynamic(const Link *link, const Address *address);
-int link_get_address(Link *link, int family, const union in_addr_union *address, unsigned char prefixlen, Address **ret);
-static inline int link_get_ipv6_address(Link *link, const struct in6_addr *address, unsigned char prefixlen, Address **ret) {
+
+int link_get_address_full(
+ Link *link,
+ int family,
+ const union in_addr_union *address,
+ const union in_addr_union *peer, /* optional, can be NULL */
+ unsigned char prefixlen, /* optional, can be 0 */
+ Address **ret);
+static inline int link_get_address(Link *link, int family, const union in_addr_union *address, Address **ret) {
+ return link_get_address_full(link, family, address, NULL, 0, ret);
+}
+static inline int link_get_ipv6_address(Link *link, const struct in6_addr *address, Address **ret) {
assert(address);
- return link_get_address(link, AF_INET6, &(union in_addr_union) { .in6 = *address }, prefixlen, ret);
+ return link_get_address(link, AF_INET6, &(union in_addr_union) { .in6 = *address }, ret);
+}
+int manager_get_address_full(
+ Manager *manager,
+ int family,
+ const union in_addr_union *address,
+ const union in_addr_union *peer,
+ unsigned char prefixlen,
+ Address **ret);
+static inline int manager_get_address(Manager *manager, int family, const union in_addr_union *address, Address **ret) {
+ return manager_get_address_full(manager, family, address, NULL, 0, ret);
}
-int manager_get_address(Manager *manager, int family, const union in_addr_union *address, unsigned char prefixlen, Address **ret);
-bool manager_has_address(Manager *manager, int family, const union in_addr_union *address);
int link_request_address(
Link *link,
if (r < 0)
return log_link_warning_errno(link, r, "Failed to get gateway address from RA: %m");
- if (link_get_ipv6_address(link, &gateway, 0, NULL) >= 0) {
+ if (link_get_ipv6_address(link, &gateway, NULL) >= 0) {
if (DEBUG_LOGGING)
log_link_debug(link, "Advertised route gateway %s is local to the link, ignoring route",
IN6_ADDR_TO_STRING(&gateway));
return 0;
}
- r = link_get_address(link, route->family, &route->prefsrc, 0, &a);
+ r = link_get_address(link, route->family, &route->prefsrc, &a);
if (r < 0)
return r;
if (r < 0)
return r;
- r = link_get_address(link, found->family, &found->prefsrc, 0, &a);
+ r = link_get_address(link, found->family, &found->prefsrc, &a);
if (r < 0)
return r;
}
static int route_is_ready_to_configure(const Route *route, Link *link) {
- int r;
-
assert(route);
assert(link);
return false;
if (in_addr_is_set(route->family, &route->prefsrc) > 0) {
- r = manager_has_address(link->manager, route->family, &route->prefsrc);
- if (r <= 0)
- return r;
+ Address *a;
+
+ if (manager_get_address(link->manager, route->family, &route->prefsrc, &a) < 0)
+ return false;
+
+ if (!address_is_ready(a))
+ return false;
}
return route_nexthops_is_ready_to_configure(route, link->manager);