Hi all,

while working on IPv6 vlxan driver I figured out that with recent version
of iproute2 it is no longer possible to configure an IPv6 vxlan device without
endpoint info (local ip, remote ip or group ip) and later add entries in
the vxlan fdb. This issue can be triggered with the following reproducer:

$ip -6 link add vxlan0 type vxlan id 42 dev enp0s2 \
    proxy nolearning l2miss l3miss
$bridge fdb add 46:47:1f:a7:1c:25 dev vxlan1 dst 2000::2
RTNETLINK answers: Address family not supported by protocol

Starting from commit 1e9b8072de2c ("iplink_vxlan: Get rid of inet_get_addr()")
the preferred_family is no longer taken into account if neither saddr or daddr
are provided and vxlan kernel module will use IPv4 as default remote inet
family neglecting the one provided by userspace.
I guess we can fix that issue in two ways:

1- add a new netlink attribute to vxlan driver in order to specify the
preferred_family to use

2- restore the behaviour introduced in commit 97d564b90ccb ("vxlan: use
preferred address family when neither group or remote is specified") with
the following patch

-- >8 --
diff --git a/ip/iplink_vxlan.c b/ip/iplink_vxlan.c
index 2bc253fc..831f39a2 100644
--- a/ip/iplink_vxlan.c
+++ b/ip/iplink_vxlan.c
@@ -82,6 +82,7 @@ static int vxlan_parse_opt(struct link_util *lu, int argc, 
char **argv,
        __u64 attrs = 0;
        bool set_op = (n->nlmsg_type == RTM_NEWLINK &&
                       !(n->nlmsg_flags & NLM_F_CREATE));
+       bool selected_family = false;
 
        saddr.family = daddr.family = AF_UNSPEC;
 
@@ -356,12 +357,26 @@ static int vxlan_parse_opt(struct link_util *lu, int 
argc, char **argv,
                int type = (saddr.family == AF_INET) ? IFLA_VXLAN_LOCAL
                                                     : IFLA_VXLAN_LOCAL6;
                addattr_l(n, 1024, type, saddr.data, saddr.bytelen);
+               selected_family = true;
        }
 
        if (is_addrtype_inet(&daddr)) {
                int type = (daddr.family == AF_INET) ? IFLA_VXLAN_GROUP
                                                     : IFLA_VXLAN_GROUP6;
                addattr_l(n, 1024, type, daddr.data, daddr.bytelen);
+               selected_family = true;
+       }
+
+       if (!selected_family) {
+               if (preferred_family == AF_INET) {
+                       get_addr(&daddr, "default", AF_INET);
+                       addattr_l(n, 1024, IFLA_VXLAN_GROUP,
+                                 daddr.data, daddr.bytelen);
+               } else if (preferred_family == AF_INET6) {
+                       get_addr(&daddr, "default", AF_INET6);
+                       addattr_l(n, 1024, IFLA_VXLAN_GROUP6,
+                                 daddr.data, daddr.bytelen);
+               }
        }
 
        if (!set_op || VXLAN_ATTRSET(attrs, IFLA_VXLAN_LEARNING))
-- 8< --

What is the best way to proceed?

Regards,
Lorenzo

Reply via email to