gluon/patches/openwrt/0007-kernel-bridge-backport-mcast-router-split-and-export.patch
Linus Lüssing 69cbaba4f4 batman-adv: backport routeable mcast address support
With batman-adv 2022.0 support for directed multicast in bridged
setups for routeable multicast addresses was refined - if the
Linux kernel is 5.14 or higher.

This backports the according changes made to the Linux bridge in 5.14
and kernel exports to 5.10, to allow batman-adv to tap into and use
the Linux bridge's Multicast Router Discovery (RFC4286) based multicast
router state. Which enables the batman-adv multicast optimizations not
only for IPv6 link-local multicast addresses (ffX2::/16) but now also
IPv6 routeable multicast addresses (ffXY::/16, with Y > 2) in Gluon.

Note that since batman-adv v2019.3 with a Linux < 5.14 multicast routers
were initially "guessed" by checking for listeners on ff02::2 (all-routers).
Which could potentially overestimate - like in our case:
This approach won't work for us, as every Gluon node listens on ff02::2
on the local-node interface due to Gluon's radvd for local IPv6 prefix
assignments.

Signed-off-by: Linus Lüssing <linus.luessing@c0d3.blue>
2022-09-09 13:09:59 +02:00

1873 lines
66 KiB
Diff
Raw Blame History

This file contains ambiguous Unicode characters

This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

From: Linus Lüssing <linus.luessing@c0d3.blue>
Date: Fri, 19 Aug 2022 11:30:49 +0200
Subject: kernel: bridge: backport mcast router split and export
This backports patches from Linux 5.14 which added support for
more fine grained, individual IPv4/IPv6 multicast router
snooping, which was tracked combined for both protocol families
before. As well as a patch to export this now split multicast
router state to other kernel modules.
These are needed for batman-adv 2021.2 and newer to support not only
link-local multicast addresses in bridged scenarios but routeable ones,
too.
Signed-off-by: Linus Lüssing <linus.luessing@c0d3.blue>
diff --git a/target/linux/generic/backport-5.10/610-v5.14-00-net-bridge-mcast-rename-multicast-router-lists-and-t.patch b/target/linux/generic/backport-5.10/610-v5.14-00-net-bridge-mcast-rename-multicast-router-lists-and-t.patch
new file mode 100644
index 0000000000000000000000000000000000000000..49b590191a4bc796a2c4e6951ab09e7e7ccca1a5
--- /dev/null
+++ b/target/linux/generic/backport-5.10/610-v5.14-00-net-bridge-mcast-rename-multicast-router-lists-and-t.patch
@@ -0,0 +1,291 @@
+From 9533309bfab9b3c03db120984ffdecbf33dee24b Mon Sep 17 00:00:00 2001
+From: =?UTF-8?q?Linus=20L=C3=BCssing?= <linus.luessing@c0d3.blue>
+Date: Thu, 13 May 2021 15:20:43 +0200
+Subject: [PATCH 01/13] net: bridge: mcast: rename multicast router lists and
+ timers
+MIME-Version: 1.0
+Content-Type: text/plain; charset=UTF-8
+Content-Transfer-Encoding: 8bit
+
+In preparation for the upcoming split of multicast router state into
+their IPv4 and IPv6 variants, rename the affected variable to the IPv4
+version first to avoid some renames in later commits.
+
+Signed-off-by: Linus Lüssing <linus.luessing@c0d3.blue>
+Signed-off-by: David S. Miller <davem@davemloft.net>
+---
+ net/bridge/br_forward.c | 4 ++--
+ net/bridge/br_mdb.c | 6 ++---
+ net/bridge/br_multicast.c | 48 +++++++++++++++++++--------------------
+ net/bridge/br_private.h | 10 ++++----
+ 4 files changed, 34 insertions(+), 34 deletions(-)
+
+--- a/net/bridge/br_forward.c
++++ b/net/bridge/br_forward.c
+@@ -277,7 +277,7 @@ void br_multicast_flood(struct net_bridg
+ bool allow_mode_include = true;
+ struct hlist_node *rp;
+
+- rp = rcu_dereference(hlist_first_rcu(&br->router_list));
++ rp = rcu_dereference(hlist_first_rcu(&br->ip4_mc_router_list));
+ if (mdst) {
+ p = rcu_dereference(mdst->ports);
+ if (br_multicast_should_handle_mode(br, mdst->addr.proto) &&
+@@ -291,7 +291,7 @@ void br_multicast_flood(struct net_bridg
+ struct net_bridge_port *port, *lport, *rport;
+
+ lport = p ? p->key.port : NULL;
+- rport = hlist_entry_safe(rp, struct net_bridge_port, rlist);
++ rport = hlist_entry_safe(rp, struct net_bridge_port, ip4_rlist);
+
+ if ((unsigned long)lport > (unsigned long)rport) {
+ port = lport;
+--- a/net/bridge/br_mdb.c
++++ b/net/bridge/br_mdb.c
+@@ -23,14 +23,14 @@ static int br_rports_fill_info(struct sk
+ struct net_bridge_port *p;
+ struct nlattr *nest, *port_nest;
+
+- if (!br->multicast_router || hlist_empty(&br->router_list))
++ if (!br->multicast_router || hlist_empty(&br->ip4_mc_router_list))
+ return 0;
+
+ nest = nla_nest_start_noflag(skb, MDBA_ROUTER);
+ if (nest == NULL)
+ return -EMSGSIZE;
+
+- hlist_for_each_entry_rcu(p, &br->router_list, rlist) {
++ hlist_for_each_entry_rcu(p, &br->ip4_mc_router_list, ip4_rlist) {
+ if (!p)
+ continue;
+ port_nest = nla_nest_start_noflag(skb, MDBA_ROUTER_PORT);
+@@ -38,7 +38,7 @@ static int br_rports_fill_info(struct sk
+ goto fail;
+ if (nla_put_nohdr(skb, sizeof(u32), &p->dev->ifindex) ||
+ nla_put_u32(skb, MDBA_ROUTER_PATTR_TIMER,
+- br_timer_value(&p->multicast_router_timer)) ||
++ br_timer_value(&p->ip4_mc_router_timer)) ||
+ nla_put_u8(skb, MDBA_ROUTER_PATTR_TYPE,
+ p->multicast_router)) {
+ nla_nest_cancel(skb, port_nest);
+--- a/net/bridge/br_multicast.c
++++ b/net/bridge/br_multicast.c
+@@ -1344,13 +1344,13 @@ static int br_ip6_multicast_add_group(st
+ static void br_multicast_router_expired(struct timer_list *t)
+ {
+ struct net_bridge_port *port =
+- from_timer(port, t, multicast_router_timer);
++ from_timer(port, t, ip4_mc_router_timer);
+ struct net_bridge *br = port->br;
+
+ spin_lock(&br->multicast_lock);
+ if (port->multicast_router == MDB_RTR_TYPE_DISABLED ||
+ port->multicast_router == MDB_RTR_TYPE_PERM ||
+- timer_pending(&port->multicast_router_timer))
++ timer_pending(&port->ip4_mc_router_timer))
+ goto out;
+
+ __del_port_router(port);
+@@ -1373,12 +1373,12 @@ static void br_mc_router_state_change(st
+
+ static void br_multicast_local_router_expired(struct timer_list *t)
+ {
+- struct net_bridge *br = from_timer(br, t, multicast_router_timer);
++ struct net_bridge *br = from_timer(br, t, ip4_mc_router_timer);
+
+ spin_lock(&br->multicast_lock);
+ if (br->multicast_router == MDB_RTR_TYPE_DISABLED ||
+ br->multicast_router == MDB_RTR_TYPE_PERM ||
+- timer_pending(&br->multicast_router_timer))
++ timer_pending(&br->ip4_mc_router_timer))
+ goto out;
+
+ br_mc_router_state_change(br, false);
+@@ -1596,7 +1596,7 @@ int br_multicast_add_port(struct net_bri
+ {
+ port->multicast_router = MDB_RTR_TYPE_TEMP_QUERY;
+
+- timer_setup(&port->multicast_router_timer,
++ timer_setup(&port->ip4_mc_router_timer,
+ br_multicast_router_expired, 0);
+ timer_setup(&port->ip4_own_query.timer,
+ br_ip4_multicast_port_query_expired, 0);
+@@ -1628,7 +1628,7 @@ void br_multicast_del_port(struct net_br
+ hlist_move_list(&br->mcast_gc_list, &deleted_head);
+ spin_unlock_bh(&br->multicast_lock);
+ br_multicast_gc(&deleted_head);
+- del_timer_sync(&port->multicast_router_timer);
++ del_timer_sync(&port->ip4_mc_router_timer);
+ free_percpu(port->mcast_stats);
+ }
+
+@@ -1653,7 +1653,7 @@ static void __br_multicast_enable_port(s
+ br_multicast_enable(&port->ip6_own_query);
+ #endif
+ if (port->multicast_router == MDB_RTR_TYPE_PERM &&
+- hlist_unhashed(&port->rlist))
++ hlist_unhashed(&port->ip4_rlist))
+ br_multicast_add_router(br, port);
+ }
+
+@@ -1679,7 +1679,7 @@ void br_multicast_disable_port(struct ne
+
+ __del_port_router(port);
+
+- del_timer(&port->multicast_router_timer);
++ del_timer(&port->ip4_mc_router_timer);
+ del_timer(&port->ip4_own_query.timer);
+ #if IS_ENABLED(CONFIG_IPV6)
+ del_timer(&port->ip6_own_query.timer);
+@@ -2577,19 +2577,19 @@ static void br_multicast_add_router(stru
+ struct net_bridge_port *p;
+ struct hlist_node *slot = NULL;
+
+- if (!hlist_unhashed(&port->rlist))
++ if (!hlist_unhashed(&port->ip4_rlist))
+ return;
+
+- hlist_for_each_entry(p, &br->router_list, rlist) {
++ hlist_for_each_entry(p, &br->ip4_mc_router_list, ip4_rlist) {
+ if ((unsigned long) port >= (unsigned long) p)
+ break;
+- slot = &p->rlist;
++ slot = &p->ip4_rlist;
+ }
+
+ if (slot)
+- hlist_add_behind_rcu(&port->rlist, slot);
++ hlist_add_behind_rcu(&port->ip4_rlist, slot);
+ else
+- hlist_add_head_rcu(&port->rlist, &br->router_list);
++ hlist_add_head_rcu(&port->ip4_rlist, &br->ip4_mc_router_list);
+ br_rtr_notify(br->dev, port, RTM_NEWMDB);
+ br_port_mc_router_state_change(port, true);
+ }
+@@ -2601,9 +2601,9 @@ static void br_multicast_mark_router(str
+
+ if (!port) {
+ if (br->multicast_router == MDB_RTR_TYPE_TEMP_QUERY) {
+- if (!timer_pending(&br->multicast_router_timer))
++ if (!timer_pending(&br->ip4_mc_router_timer))
+ br_mc_router_state_change(br, true);
+- mod_timer(&br->multicast_router_timer,
++ mod_timer(&br->ip4_mc_router_timer,
+ now + br->multicast_querier_interval);
+ }
+ return;
+@@ -2615,7 +2615,7 @@ static void br_multicast_mark_router(str
+
+ br_multicast_add_router(br, port);
+
+- mod_timer(&port->multicast_router_timer,
++ mod_timer(&port->ip4_mc_router_timer,
+ now + br->multicast_querier_interval);
+ }
+
+@@ -3233,7 +3233,7 @@ void br_multicast_init(struct net_bridge
+ br_opt_toggle(br, BROPT_HAS_IPV6_ADDR, true);
+
+ spin_lock_init(&br->multicast_lock);
+- timer_setup(&br->multicast_router_timer,
++ timer_setup(&br->ip4_mc_router_timer,
+ br_multicast_local_router_expired, 0);
+ timer_setup(&br->ip4_other_query.timer,
+ br_ip4_multicast_querier_expired, 0);
+@@ -3333,7 +3333,7 @@ void br_multicast_open(struct net_bridge
+
+ void br_multicast_stop(struct net_bridge *br)
+ {
+- del_timer_sync(&br->multicast_router_timer);
++ del_timer_sync(&br->ip4_mc_router_timer);
+ del_timer_sync(&br->ip4_other_query.timer);
+ del_timer_sync(&br->ip4_own_query.timer);
+ #if IS_ENABLED(CONFIG_IPV6)
+@@ -3370,7 +3370,7 @@ int br_multicast_set_router(struct net_b
+ case MDB_RTR_TYPE_DISABLED:
+ case MDB_RTR_TYPE_PERM:
+ br_mc_router_state_change(br, val == MDB_RTR_TYPE_PERM);
+- del_timer(&br->multicast_router_timer);
++ del_timer(&br->ip4_mc_router_timer);
+ br->multicast_router = val;
+ err = 0;
+ break;
+@@ -3389,9 +3389,9 @@ int br_multicast_set_router(struct net_b
+
+ static void __del_port_router(struct net_bridge_port *p)
+ {
+- if (hlist_unhashed(&p->rlist))
++ if (hlist_unhashed(&p->ip4_rlist))
+ return;
+- hlist_del_init_rcu(&p->rlist);
++ hlist_del_init_rcu(&p->ip4_rlist);
+ br_rtr_notify(p->br->dev, p, RTM_DELMDB);
+ br_port_mc_router_state_change(p, false);
+
+@@ -3410,7 +3410,7 @@ int br_multicast_set_port_router(struct
+ if (p->multicast_router == val) {
+ /* Refresh the temp router port timer */
+ if (p->multicast_router == MDB_RTR_TYPE_TEMP)
+- mod_timer(&p->multicast_router_timer,
++ mod_timer(&p->ip4_mc_router_timer,
+ now + br->multicast_querier_interval);
+ err = 0;
+ goto unlock;
+@@ -3419,7 +3419,7 @@ int br_multicast_set_port_router(struct
+ case MDB_RTR_TYPE_DISABLED:
+ p->multicast_router = MDB_RTR_TYPE_DISABLED;
+ __del_port_router(p);
+- del_timer(&p->multicast_router_timer);
++ del_timer(&p->ip4_mc_router_timer);
+ break;
+ case MDB_RTR_TYPE_TEMP_QUERY:
+ p->multicast_router = MDB_RTR_TYPE_TEMP_QUERY;
+@@ -3427,7 +3427,7 @@ int br_multicast_set_port_router(struct
+ break;
+ case MDB_RTR_TYPE_PERM:
+ p->multicast_router = MDB_RTR_TYPE_PERM;
+- del_timer(&p->multicast_router_timer);
++ del_timer(&p->ip4_mc_router_timer);
+ br_multicast_add_router(br, p);
+ break;
+ case MDB_RTR_TYPE_TEMP:
+--- a/net/bridge/br_private.h
++++ b/net/bridge/br_private.h
+@@ -313,14 +313,14 @@ struct net_bridge_port {
+
+ #ifdef CONFIG_BRIDGE_IGMP_SNOOPING
+ struct bridge_mcast_own_query ip4_own_query;
++ struct timer_list ip4_mc_router_timer;
++ struct hlist_node ip4_rlist;
+ #if IS_ENABLED(CONFIG_IPV6)
+ struct bridge_mcast_own_query ip6_own_query;
+ #endif /* IS_ENABLED(CONFIG_IPV6) */
+ unsigned char multicast_router;
+ struct bridge_mcast_stats __percpu *mcast_stats;
+- struct timer_list multicast_router_timer;
+ struct hlist_head mglist;
+- struct hlist_node rlist;
+ #endif
+
+ #ifdef CONFIG_SYSFS
+@@ -453,9 +453,9 @@ struct net_bridge {
+
+ struct hlist_head mcast_gc_list;
+ struct hlist_head mdb_list;
+- struct hlist_head router_list;
+
+- struct timer_list multicast_router_timer;
++ struct hlist_head ip4_mc_router_list;
++ struct timer_list ip4_mc_router_timer;
+ struct bridge_mcast_other_query ip4_other_query;
+ struct bridge_mcast_own_query ip4_own_query;
+ struct bridge_mcast_querier ip4_querier;
+@@ -849,7 +849,7 @@ static inline bool br_multicast_is_route
+ {
+ return br->multicast_router == 2 ||
+ (br->multicast_router == 1 &&
+- timer_pending(&br->multicast_router_timer));
++ timer_pending(&br->ip4_mc_router_timer));
+ }
+
+ static inline bool
diff --git a/target/linux/generic/backport-5.10/610-v5.14-01-net-bridge-mcast-add-wrappers-for-router-node-retrie.patch b/target/linux/generic/backport-5.10/610-v5.14-01-net-bridge-mcast-add-wrappers-for-router-node-retrie.patch
new file mode 100644
index 0000000000000000000000000000000000000000..a330765b3ef5a7bbea3eb00004a08d2b10a7457c
--- /dev/null
+++ b/target/linux/generic/backport-5.10/610-v5.14-01-net-bridge-mcast-add-wrappers-for-router-node-retrie.patch
@@ -0,0 +1,61 @@
+From 9aa73cb262cb647e0a42606980c6fdf585308615 Mon Sep 17 00:00:00 2001
+From: =?UTF-8?q?Linus=20L=C3=BCssing?= <linus.luessing@c0d3.blue>
+Date: Thu, 13 May 2021 15:20:44 +0200
+Subject: [PATCH 02/13] net: bridge: mcast: add wrappers for router node
+ retrieval
+MIME-Version: 1.0
+Content-Type: text/plain; charset=UTF-8
+Content-Transfer-Encoding: 8bit
+
+In preparation for the upcoming split of multicast router state into
+their IPv4 and IPv6 variants and to avoid IPv6 #ifdef clutter later add
+two wrapper functions for router node retrieval in the payload
+forwarding code.
+
+Signed-off-by: Linus Lüssing <linus.luessing@c0d3.blue>
+Signed-off-by: David S. Miller <davem@davemloft.net>
+---
+ net/bridge/br_forward.c | 5 +++--
+ net/bridge/br_private.h | 10 ++++++++++
+ 2 files changed, 13 insertions(+), 2 deletions(-)
+
+--- a/net/bridge/br_forward.c
++++ b/net/bridge/br_forward.c
+@@ -277,7 +277,8 @@ void br_multicast_flood(struct net_bridg
+ bool allow_mode_include = true;
+ struct hlist_node *rp;
+
+- rp = rcu_dereference(hlist_first_rcu(&br->ip4_mc_router_list));
++ rp = br_multicast_get_first_rport_node(br, skb);
++
+ if (mdst) {
+ p = rcu_dereference(mdst->ports);
+ if (br_multicast_should_handle_mode(br, mdst->addr.proto) &&
+@@ -291,7 +292,7 @@ void br_multicast_flood(struct net_bridg
+ struct net_bridge_port *port, *lport, *rport;
+
+ lport = p ? p->key.port : NULL;
+- rport = hlist_entry_safe(rp, struct net_bridge_port, ip4_rlist);
++ rport = br_multicast_rport_from_node_skb(rp, skb);
+
+ if ((unsigned long)lport > (unsigned long)rport) {
+ port = lport;
+--- a/net/bridge/br_private.h
++++ b/net/bridge/br_private.h
+@@ -845,6 +845,16 @@ void br_multicast_sg_add_exclude_ports(s
+ #define mlock_dereference(X, br) \
+ rcu_dereference_protected(X, lockdep_is_held(&br->multicast_lock))
+
++static inline struct hlist_node *
++br_multicast_get_first_rport_node(struct net_bridge *b, struct sk_buff *skb) {
++ return rcu_dereference(hlist_first_rcu(&b->ip4_mc_router_list));
++}
++
++static inline struct net_bridge_port *
++br_multicast_rport_from_node_skb(struct hlist_node *rp, struct sk_buff *skb) {
++ return hlist_entry_safe(rp, struct net_bridge_port, ip4_rlist);
++}
++
+ static inline bool br_multicast_is_router(struct net_bridge *br)
+ {
+ return br->multicast_router == 2 ||
diff --git a/target/linux/generic/backport-5.10/610-v5.14-02-net-bridge-mcast-prepare-mdb-netlink-for-mcast-route.patch b/target/linux/generic/backport-5.10/610-v5.14-02-net-bridge-mcast-prepare-mdb-netlink-for-mcast-route.patch
new file mode 100644
index 0000000000000000000000000000000000000000..22bf0cf43d4f587a4b0e9cd70f1e04c698737b6f
--- /dev/null
+++ b/target/linux/generic/backport-5.10/610-v5.14-02-net-bridge-mcast-prepare-mdb-netlink-for-mcast-route.patch
@@ -0,0 +1,88 @@
+From cc19b1f59994df56e8e3748e496cc1a03b8c4a06 Mon Sep 17 00:00:00 2001
+From: =?UTF-8?q?Linus=20L=C3=BCssing?= <linus.luessing@c0d3.blue>
+Date: Thu, 13 May 2021 15:20:45 +0200
+Subject: [PATCH 03/13] net: bridge: mcast: prepare mdb netlink for mcast
+ router split
+MIME-Version: 1.0
+Content-Type: text/plain; charset=UTF-8
+Content-Transfer-Encoding: 8bit
+
+In preparation for the upcoming split of multicast router state into
+their IPv4 and IPv6 variants and to avoid IPv6 #ifdef clutter later add
+some inline functions for the protocol specific parts in the mdb router
+netlink code. Also the we need iterate over the port instead of router
+list to be able put one router port entry with both the IPv4 and IPv6
+multicast router info later.
+
+Signed-off-by: Linus Lüssing <linus.luessing@c0d3.blue>
+Signed-off-by: David S. Miller <davem@davemloft.net>
+---
+ net/bridge/br_mdb.c | 39 ++++++++++++++++++++++++++++++++++-----
+ 1 file changed, 34 insertions(+), 5 deletions(-)
+
+--- a/net/bridge/br_mdb.c
++++ b/net/bridge/br_mdb.c
+@@ -16,29 +16,58 @@
+
+ #include "br_private.h"
+
++static bool br_rports_have_mc_router(struct net_bridge *br)
++{
++ return !hlist_empty(&br->ip4_mc_router_list);
++}
++
++static bool
++br_ip4_rports_get_timer(struct net_bridge_port *port, unsigned long *timer)
++{
++ *timer = br_timer_value(&port->ip4_mc_router_timer);
++ return !hlist_unhashed(&port->ip4_rlist);
++}
++
++static bool
++br_ip6_rports_get_timer(struct net_bridge_port *port, unsigned long *timer)
++{
++ *timer = 0;
++ return false;
++}
++
+ static int br_rports_fill_info(struct sk_buff *skb, struct netlink_callback *cb,
+ struct net_device *dev)
+ {
+ struct net_bridge *br = netdev_priv(dev);
+- struct net_bridge_port *p;
++ bool have_ip4_mc_rtr, have_ip6_mc_rtr;
++ unsigned long ip4_timer, ip6_timer;
+ struct nlattr *nest, *port_nest;
++ struct net_bridge_port *p;
++
++ if (!br->multicast_router)
++ return 0;
+
+- if (!br->multicast_router || hlist_empty(&br->ip4_mc_router_list))
++ if (!br_rports_have_mc_router(br))
+ return 0;
+
+ nest = nla_nest_start_noflag(skb, MDBA_ROUTER);
+ if (nest == NULL)
+ return -EMSGSIZE;
+
+- hlist_for_each_entry_rcu(p, &br->ip4_mc_router_list, ip4_rlist) {
+- if (!p)
++ list_for_each_entry_rcu(p, &br->port_list, list) {
++ have_ip4_mc_rtr = br_ip4_rports_get_timer(p, &ip4_timer);
++ have_ip6_mc_rtr = br_ip6_rports_get_timer(p, &ip6_timer);
++
++ if (!have_ip4_mc_rtr && !have_ip6_mc_rtr)
+ continue;
++
+ port_nest = nla_nest_start_noflag(skb, MDBA_ROUTER_PORT);
+ if (!port_nest)
+ goto fail;
++
+ if (nla_put_nohdr(skb, sizeof(u32), &p->dev->ifindex) ||
+ nla_put_u32(skb, MDBA_ROUTER_PATTR_TIMER,
+- br_timer_value(&p->ip4_mc_router_timer)) ||
++ max(ip4_timer, ip6_timer)) ||
+ nla_put_u8(skb, MDBA_ROUTER_PATTR_TYPE,
+ p->multicast_router)) {
+ nla_nest_cancel(skb, port_nest);
diff --git a/target/linux/generic/backport-5.10/610-v5.14-03-net-bridge-mcast-prepare-query-reception-for-mcast-r.patch b/target/linux/generic/backport-5.10/610-v5.14-03-net-bridge-mcast-prepare-query-reception-for-mcast-r.patch
new file mode 100644
index 0000000000000000000000000000000000000000..ea6852b794d7653cd30e6853c5335c1b76e10ca3
--- /dev/null
+++ b/target/linux/generic/backport-5.10/610-v5.14-03-net-bridge-mcast-prepare-query-reception-for-mcast-r.patch
@@ -0,0 +1,110 @@
+From cf580b34cb6709d3efc31fe54d5eb35744025f1b Mon Sep 17 00:00:00 2001
+From: =?UTF-8?q?Linus=20L=C3=BCssing?= <linus.luessing@c0d3.blue>
+Date: Thu, 13 May 2021 15:20:46 +0200
+Subject: [PATCH 04/13] net: bridge: mcast: prepare query reception for mcast
+ router split
+MIME-Version: 1.0
+Content-Type: text/plain; charset=UTF-8
+Content-Transfer-Encoding: 8bit
+
+In preparation for the upcoming split of multicast router state into
+their IPv4 and IPv6 variants and as the br_multicast_mark_router() will
+be split for that remove the select querier wrapper and instead add
+ip4 and ip6 variants for br_multicast_query_received().
+
+Signed-off-by: Linus Lüssing <linus.luessing@c0d3.blue>
+Signed-off-by: David S. Miller <davem@davemloft.net>
+---
+ net/bridge/br_multicast.c | 53 ++++++++++++++++++++-------------------
+ 1 file changed, 27 insertions(+), 26 deletions(-)
+
+--- a/net/bridge/br_multicast.c
++++ b/net/bridge/br_multicast.c
+@@ -2526,22 +2526,6 @@ update:
+ }
+ #endif
+
+-static bool br_multicast_select_querier(struct net_bridge *br,
+- struct net_bridge_port *port,
+- struct br_ip *saddr)
+-{
+- switch (saddr->proto) {
+- case htons(ETH_P_IP):
+- return br_ip4_multicast_select_querier(br, port, saddr->src.ip4);
+-#if IS_ENABLED(CONFIG_IPV6)
+- case htons(ETH_P_IPV6):
+- return br_ip6_multicast_select_querier(br, port, &saddr->src.ip6);
+-#endif
+- }
+-
+- return false;
+-}
+-
+ static void
+ br_multicast_update_query_timer(struct net_bridge *br,
+ struct bridge_mcast_other_query *query,
+@@ -2619,19 +2603,36 @@ static void br_multicast_mark_router(str
+ now + br->multicast_querier_interval);
+ }
+
+-static void br_multicast_query_received(struct net_bridge *br,
+- struct net_bridge_port *port,
+- struct bridge_mcast_other_query *query,
+- struct br_ip *saddr,
+- unsigned long max_delay)
++static void
++br_ip4_multicast_query_received(struct net_bridge *br,
++ struct net_bridge_port *port,
++ struct bridge_mcast_other_query *query,
++ struct br_ip *saddr,
++ unsigned long max_delay)
+ {
+- if (!br_multicast_select_querier(br, port, saddr))
++ if (!br_ip4_multicast_select_querier(br, port, saddr->src.ip4))
+ return;
+
+ br_multicast_update_query_timer(br, query, max_delay);
+ br_multicast_mark_router(br, port);
+ }
+
++#if IS_ENABLED(CONFIG_IPV6)
++static void
++br_ip6_multicast_query_received(struct net_bridge *br,
++ struct net_bridge_port *port,
++ struct bridge_mcast_other_query *query,
++ struct br_ip *saddr,
++ unsigned long max_delay)
++{
++ if (!br_ip6_multicast_select_querier(br, port, &saddr->src.ip6))
++ return;
++
++ br_multicast_update_query_timer(br, query, max_delay);
++ br_multicast_mark_router(br, port);
++}
++#endif
++
+ static void br_ip4_multicast_query(struct net_bridge *br,
+ struct net_bridge_port *port,
+ struct sk_buff *skb,
+@@ -2679,8 +2680,8 @@ static void br_ip4_multicast_query(struc
+ saddr.proto = htons(ETH_P_IP);
+ saddr.src.ip4 = iph->saddr;
+
+- br_multicast_query_received(br, port, &br->ip4_other_query,
+- &saddr, max_delay);
++ br_ip4_multicast_query_received(br, port, &br->ip4_other_query,
++ &saddr, max_delay);
+ goto out;
+ }
+
+@@ -2767,8 +2768,8 @@ static int br_ip6_multicast_query(struct
+ saddr.proto = htons(ETH_P_IPV6);
+ saddr.src.ip6 = ipv6_hdr(skb)->saddr;
+
+- br_multicast_query_received(br, port, &br->ip6_other_query,
+- &saddr, max_delay);
++ br_ip6_multicast_query_received(br, port, &br->ip6_other_query,
++ &saddr, max_delay);
+ goto out;
+ } else if (!group) {
+ goto out;
diff --git a/target/linux/generic/backport-5.10/610-v5.14-04-net-bridge-mcast-prepare-is-router-function-for-mcas.patch b/target/linux/generic/backport-5.10/610-v5.14-04-net-bridge-mcast-prepare-is-router-function-for-mcas.patch
new file mode 100644
index 0000000000000000000000000000000000000000..d4baadb0ca9702cd5c6a639f8f7dc17f72e42fef
--- /dev/null
+++ b/target/linux/generic/backport-5.10/610-v5.14-04-net-bridge-mcast-prepare-is-router-function-for-mcas.patch
@@ -0,0 +1,108 @@
+From ea978567bf6e94b8ef5539ae68d1fe6b216b1647 Mon Sep 17 00:00:00 2001
+From: =?UTF-8?q?Linus=20L=C3=BCssing?= <linus.luessing@c0d3.blue>
+Date: Thu, 13 May 2021 15:20:47 +0200
+Subject: [PATCH 05/13] net: bridge: mcast: prepare is-router function for
+ mcast router split
+MIME-Version: 1.0
+Content-Type: text/plain; charset=UTF-8
+Content-Transfer-Encoding: 8bit
+
+In preparation for the upcoming split of multicast router state into
+their IPv4 and IPv6 variants make br_multicast_is_router() protocol
+family aware.
+
+Note that for now br_ip6_multicast_is_router() uses the currently still
+common ip4_mc_router_timer for now. It will be renamed to
+ip6_mc_router_timer later when the split is performed.
+
+While at it also renames the "1" and "2" constants in
+br_multicast_is_router() to the MDB_RTR_TYPE_TEMP_QUERY and
+MDB_RTR_TYPE_PERM enums.
+
+Signed-off-by: Linus Lüssing <linus.luessing@c0d3.blue>
+Signed-off-by: David S. Miller <davem@davemloft.net>
+---
+ net/bridge/br_input.c | 2 +-
+ net/bridge/br_multicast.c | 5 +++--
+ net/bridge/br_private.h | 37 +++++++++++++++++++++++++++++++++----
+ 3 files changed, 37 insertions(+), 7 deletions(-)
+
+--- a/net/bridge/br_input.c
++++ b/net/bridge/br_input.c
+@@ -143,7 +143,7 @@ int br_handle_frame_finish(struct net *n
+ if ((mdst || BR_INPUT_SKB_CB_MROUTERS_ONLY(skb)) &&
+ br_multicast_querier_exists(br, eth_hdr(skb))) {
+ if ((mdst && mdst->host_joined) ||
+- br_multicast_is_router(br)) {
++ br_multicast_is_router(br, skb)) {
+ local_rcv = true;
+ br->dev->stats.multicast++;
+ }
+--- a/net/bridge/br_multicast.c
++++ b/net/bridge/br_multicast.c
+@@ -1378,7 +1378,8 @@ static void br_multicast_local_router_ex
+ spin_lock(&br->multicast_lock);
+ if (br->multicast_router == MDB_RTR_TYPE_DISABLED ||
+ br->multicast_router == MDB_RTR_TYPE_PERM ||
+- timer_pending(&br->ip4_mc_router_timer))
++ br_ip4_multicast_is_router(br) ||
++ br_ip6_multicast_is_router(br))
+ goto out;
+
+ br_mc_router_state_change(br, false);
+@@ -3532,7 +3533,7 @@ bool br_multicast_router(const struct ne
+ bool is_router;
+
+ spin_lock_bh(&br->multicast_lock);
+- is_router = br_multicast_is_router(br);
++ is_router = br_multicast_is_router(br, NULL);
+ spin_unlock_bh(&br->multicast_lock);
+ return is_router;
+ }
+--- a/net/bridge/br_private.h
++++ b/net/bridge/br_private.h
+@@ -855,11 +855,40 @@ br_multicast_rport_from_node_skb(struct
+ return hlist_entry_safe(rp, struct net_bridge_port, ip4_rlist);
+ }
+
+-static inline bool br_multicast_is_router(struct net_bridge *br)
++static inline bool br_ip4_multicast_is_router(struct net_bridge *br)
+ {
+- return br->multicast_router == 2 ||
+- (br->multicast_router == 1 &&
+- timer_pending(&br->ip4_mc_router_timer));
++ return timer_pending(&br->ip4_mc_router_timer);
++}
++
++static inline bool br_ip6_multicast_is_router(struct net_bridge *br)
++{
++#if IS_ENABLED(CONFIG_IPV6)
++ return timer_pending(&br->ip4_mc_router_timer);
++#else
++ return false;
++#endif
++}
++
++static inline bool
++br_multicast_is_router(struct net_bridge *br, struct sk_buff *skb)
++{
++ switch (br->multicast_router) {
++ case MDB_RTR_TYPE_PERM:
++ return true;
++ case MDB_RTR_TYPE_TEMP_QUERY:
++ if (skb) {
++ if (skb->protocol == htons(ETH_P_IP))
++ return br_ip4_multicast_is_router(br);
++ else if (skb->protocol == htons(ETH_P_IPV6))
++ return br_ip6_multicast_is_router(br);
++ } else {
++ return br_ip4_multicast_is_router(br) ||
++ br_ip6_multicast_is_router(br);
++ }
++ fallthrough;
++ default:
++ return false;
++ }
+ }
+
+ static inline bool
diff --git a/target/linux/generic/backport-5.10/610-v5.14-05-net-bridge-mcast-prepare-expiry-functions-for-mcast-.patch b/target/linux/generic/backport-5.10/610-v5.14-05-net-bridge-mcast-prepare-expiry-functions-for-mcast-.patch
new file mode 100644
index 0000000000000000000000000000000000000000..0f934ce85d3edf60fb47a880263b1002d74d69cf
--- /dev/null
+++ b/target/linux/generic/backport-5.10/610-v5.14-05-net-bridge-mcast-prepare-expiry-functions-for-mcast-.patch
@@ -0,0 +1,101 @@
+From 09b719247b153c9e1c8377f19ffa1ca66204b75d Mon Sep 17 00:00:00 2001
+From: =?UTF-8?q?Linus=20L=C3=BCssing?= <linus.luessing@c0d3.blue>
+Date: Thu, 13 May 2021 15:20:48 +0200
+Subject: [PATCH 06/13] net: bridge: mcast: prepare expiry functions for mcast
+ router split
+MIME-Version: 1.0
+Content-Type: text/plain; charset=UTF-8
+Content-Transfer-Encoding: 8bit
+
+In preparation for the upcoming split of multicast router state into
+their IPv4 and IPv6 variants move the protocol specific timer access to
+an ip4 wrapper function.
+
+Signed-off-by: Linus Lüssing <linus.luessing@c0d3.blue>
+Signed-off-by: David S. Miller <davem@davemloft.net>
+---
+ net/bridge/br_multicast.c | 31 ++++++++++++++++++++++---------
+ 1 file changed, 22 insertions(+), 9 deletions(-)
+
+--- a/net/bridge/br_multicast.c
++++ b/net/bridge/br_multicast.c
+@@ -1341,16 +1341,16 @@ static int br_ip6_multicast_add_group(st
+ }
+ #endif
+
+-static void br_multicast_router_expired(struct timer_list *t)
++static void br_multicast_router_expired(struct net_bridge_port *port,
++ struct timer_list *t,
++ struct hlist_node *rlist)
+ {
+- struct net_bridge_port *port =
+- from_timer(port, t, ip4_mc_router_timer);
+ struct net_bridge *br = port->br;
+
+ spin_lock(&br->multicast_lock);
+ if (port->multicast_router == MDB_RTR_TYPE_DISABLED ||
+ port->multicast_router == MDB_RTR_TYPE_PERM ||
+- timer_pending(&port->ip4_mc_router_timer))
++ timer_pending(t))
+ goto out;
+
+ __del_port_router(port);
+@@ -1358,6 +1358,13 @@ out:
+ spin_unlock(&br->multicast_lock);
+ }
+
++static void br_ip4_multicast_router_expired(struct timer_list *t)
++{
++ struct net_bridge_port *port = from_timer(port, t, ip4_mc_router_timer);
++
++ br_multicast_router_expired(port, t, &port->ip4_rlist);
++}
++
+ static void br_mc_router_state_change(struct net_bridge *p,
+ bool is_mc_router)
+ {
+@@ -1371,10 +1378,9 @@ static void br_mc_router_state_change(st
+ switchdev_port_attr_set(p->dev, &attr);
+ }
+
+-static void br_multicast_local_router_expired(struct timer_list *t)
++static void br_multicast_local_router_expired(struct net_bridge *br,
++ struct timer_list *timer)
+ {
+- struct net_bridge *br = from_timer(br, t, ip4_mc_router_timer);
+-
+ spin_lock(&br->multicast_lock);
+ if (br->multicast_router == MDB_RTR_TYPE_DISABLED ||
+ br->multicast_router == MDB_RTR_TYPE_PERM ||
+@@ -1387,6 +1393,13 @@ out:
+ spin_unlock(&br->multicast_lock);
+ }
+
++static void br_ip4_multicast_local_router_expired(struct timer_list *t)
++{
++ struct net_bridge *br = from_timer(br, t, ip4_mc_router_timer);
++
++ br_multicast_local_router_expired(br, t);
++}
++
+ static void br_multicast_querier_expired(struct net_bridge *br,
+ struct bridge_mcast_own_query *query)
+ {
+@@ -1598,7 +1611,7 @@ int br_multicast_add_port(struct net_bri
+ port->multicast_router = MDB_RTR_TYPE_TEMP_QUERY;
+
+ timer_setup(&port->ip4_mc_router_timer,
+- br_multicast_router_expired, 0);
++ br_ip4_multicast_router_expired, 0);
+ timer_setup(&port->ip4_own_query.timer,
+ br_ip4_multicast_port_query_expired, 0);
+ #if IS_ENABLED(CONFIG_IPV6)
+@@ -3236,7 +3249,7 @@ void br_multicast_init(struct net_bridge
+
+ spin_lock_init(&br->multicast_lock);
+ timer_setup(&br->ip4_mc_router_timer,
+- br_multicast_local_router_expired, 0);
++ br_ip4_multicast_local_router_expired, 0);
+ timer_setup(&br->ip4_other_query.timer,
+ br_ip4_multicast_querier_expired, 0);
+ timer_setup(&br->ip4_own_query.timer,
diff --git a/target/linux/generic/backport-5.10/610-v5.14-06-net-bridge-mcast-prepare-add-router-function-for-mca.patch b/target/linux/generic/backport-5.10/610-v5.14-06-net-bridge-mcast-prepare-add-router-function-for-mca.patch
new file mode 100644
index 0000000000000000000000000000000000000000..63c9783559c3cdc5298cb28078b7b26496669801
--- /dev/null
+++ b/target/linux/generic/backport-5.10/610-v5.14-06-net-bridge-mcast-prepare-add-router-function-for-mca.patch
@@ -0,0 +1,234 @@
+From 7396aa0c21f93427c5298f342dafccfef8f0d5db Mon Sep 17 00:00:00 2001
+From: =?UTF-8?q?Linus=20L=C3=BCssing?= <linus.luessing@c0d3.blue>
+Date: Thu, 13 May 2021 15:20:49 +0200
+Subject: [PATCH 07/13] net: bridge: mcast: prepare add-router function for
+ mcast router split
+MIME-Version: 1.0
+Content-Type: text/plain; charset=UTF-8
+Content-Transfer-Encoding: 8bit
+
+In preparation for the upcoming split of multicast router state into
+their IPv4 and IPv6 variants move the protocol specific router list
+and timer access to ip4 wrapper functions.
+
+Signed-off-by: Linus Lüssing <linus.luessing@c0d3.blue>
+Signed-off-by: David S. Miller <davem@davemloft.net>
+---
+ net/bridge/br_multicast.c | 121 +++++++++++++++++++++++++++-----------
+ 1 file changed, 87 insertions(+), 34 deletions(-)
+
+--- a/net/bridge/br_multicast.c
++++ b/net/bridge/br_multicast.c
+@@ -50,8 +50,8 @@ static const struct rhashtable_params br
+
+ static void br_multicast_start_querier(struct net_bridge *br,
+ struct bridge_mcast_own_query *query);
+-static void br_multicast_add_router(struct net_bridge *br,
+- struct net_bridge_port *port);
++static void br_ip4_multicast_add_router(struct net_bridge *br,
++ struct net_bridge_port *port);
+ static void br_ip4_multicast_leave_group(struct net_bridge *br,
+ struct net_bridge_port *port,
+ __be32 group,
+@@ -1666,9 +1666,8 @@ static void __br_multicast_enable_port(s
+ #if IS_ENABLED(CONFIG_IPV6)
+ br_multicast_enable(&port->ip6_own_query);
+ #endif
+- if (port->multicast_router == MDB_RTR_TYPE_PERM &&
+- hlist_unhashed(&port->ip4_rlist))
+- br_multicast_add_router(br, port);
++ if (port->multicast_router == MDB_RTR_TYPE_PERM)
++ br_ip4_multicast_add_router(br, port);
+ }
+
+ void br_multicast_enable_port(struct net_bridge_port *port)
+@@ -2564,45 +2563,86 @@ static void br_port_mc_router_state_chan
+ switchdev_port_attr_set(p->dev, &attr);
+ }
+
+-/*
+- * Add port to router_list
++static struct net_bridge_port *
++br_multicast_rport_from_node(struct net_bridge *br,
++ struct hlist_head *mc_router_list,
++ struct hlist_node *rlist)
++{
++ return hlist_entry(rlist, struct net_bridge_port, ip4_rlist);
++}
++
++static struct hlist_node *
++br_multicast_get_rport_slot(struct net_bridge *br,
++ struct net_bridge_port *port,
++ struct hlist_head *mc_router_list)
++
++{
++ struct hlist_node *slot = NULL;
++ struct net_bridge_port *p;
++ struct hlist_node *rlist;
++
++ hlist_for_each(rlist, mc_router_list) {
++ p = br_multicast_rport_from_node(br, mc_router_list, rlist);
++
++ if ((unsigned long)port >= (unsigned long)p)
++ break;
++
++ slot = rlist;
++ }
++
++ return slot;
++}
++
++/* Add port to router_list
+ * list is maintained ordered by pointer value
+ * and locked by br->multicast_lock and RCU
+ */
+ static void br_multicast_add_router(struct net_bridge *br,
+- struct net_bridge_port *port)
++ struct net_bridge_port *port,
++ struct hlist_node *rlist,
++ struct hlist_head *mc_router_list)
+ {
+- struct net_bridge_port *p;
+- struct hlist_node *slot = NULL;
++ struct hlist_node *slot;
+
+- if (!hlist_unhashed(&port->ip4_rlist))
++ if (!hlist_unhashed(rlist))
+ return;
+
+- hlist_for_each_entry(p, &br->ip4_mc_router_list, ip4_rlist) {
+- if ((unsigned long) port >= (unsigned long) p)
+- break;
+- slot = &p->ip4_rlist;
+- }
++ slot = br_multicast_get_rport_slot(br, port, mc_router_list);
+
+ if (slot)
+- hlist_add_behind_rcu(&port->ip4_rlist, slot);
++ hlist_add_behind_rcu(rlist, slot);
+ else
+- hlist_add_head_rcu(&port->ip4_rlist, &br->ip4_mc_router_list);
++ hlist_add_head_rcu(rlist, mc_router_list);
++
+ br_rtr_notify(br->dev, port, RTM_NEWMDB);
+ br_port_mc_router_state_change(port, true);
+ }
+
++/* Add port to router_list
++ * list is maintained ordered by pointer value
++ * and locked by br->multicast_lock and RCU
++ */
++static void br_ip4_multicast_add_router(struct net_bridge *br,
++ struct net_bridge_port *port)
++{
++ br_multicast_add_router(br, port, &port->ip4_rlist,
++ &br->ip4_mc_router_list);
++}
++
+ static void br_multicast_mark_router(struct net_bridge *br,
+- struct net_bridge_port *port)
++ struct net_bridge_port *port,
++ struct timer_list *timer,
++ struct hlist_node *rlist,
++ struct hlist_head *mc_router_list)
+ {
+ unsigned long now = jiffies;
+
+ if (!port) {
+ if (br->multicast_router == MDB_RTR_TYPE_TEMP_QUERY) {
+- if (!timer_pending(&br->ip4_mc_router_timer))
++ if (!br_ip4_multicast_is_router(br) &&
++ !br_ip6_multicast_is_router(br))
+ br_mc_router_state_change(br, true);
+- mod_timer(&br->ip4_mc_router_timer,
+- now + br->multicast_querier_interval);
++ mod_timer(timer, now + br->multicast_querier_interval);
+ }
+ return;
+ }
+@@ -2611,10 +2651,23 @@ static void br_multicast_mark_router(str
+ port->multicast_router == MDB_RTR_TYPE_PERM)
+ return;
+
+- br_multicast_add_router(br, port);
++ br_multicast_add_router(br, port, rlist, mc_router_list);
++ mod_timer(timer, now + br->multicast_querier_interval);
++}
++
++static void br_ip4_multicast_mark_router(struct net_bridge *br,
++ struct net_bridge_port *port)
++{
++ struct timer_list *timer = &br->ip4_mc_router_timer;
++ struct hlist_node *rlist = NULL;
++
++ if (port) {
++ timer = &port->ip4_mc_router_timer;
++ rlist = &port->ip4_rlist;
++ }
+
+- mod_timer(&port->ip4_mc_router_timer,
+- now + br->multicast_querier_interval);
++ br_multicast_mark_router(br, port, timer, rlist,
++ &br->ip4_mc_router_list);
+ }
+
+ static void
+@@ -2628,7 +2681,7 @@ br_ip4_multicast_query_received(struct n
+ return;
+
+ br_multicast_update_query_timer(br, query, max_delay);
+- br_multicast_mark_router(br, port);
++ br_ip4_multicast_mark_router(br, port);
+ }
+
+ #if IS_ENABLED(CONFIG_IPV6)
+@@ -2643,7 +2696,7 @@ br_ip6_multicast_query_received(struct n
+ return;
+
+ br_multicast_update_query_timer(br, query, max_delay);
+- br_multicast_mark_router(br, port);
++ br_ip4_multicast_mark_router(br, port);
+ }
+ #endif
+
+@@ -3014,7 +3067,7 @@ static void br_multicast_pim(struct net_
+ return;
+
+ spin_lock(&br->multicast_lock);
+- br_multicast_mark_router(br, port);
++ br_ip4_multicast_mark_router(br, port);
+ spin_unlock(&br->multicast_lock);
+ }
+
+@@ -3027,7 +3080,7 @@ static int br_ip4_multicast_mrd_rcv(stru
+ return -ENOMSG;
+
+ spin_lock(&br->multicast_lock);
+- br_multicast_mark_router(br, port);
++ br_ip4_multicast_mark_router(br, port);
+ spin_unlock(&br->multicast_lock);
+
+ return 0;
+@@ -3097,7 +3150,7 @@ static void br_ip6_multicast_mrd_rcv(str
+ return;
+
+ spin_lock(&br->multicast_lock);
+- br_multicast_mark_router(br, port);
++ br_ip4_multicast_mark_router(br, port);
+ spin_unlock(&br->multicast_lock);
+ }
+
+@@ -3443,11 +3496,11 @@ int br_multicast_set_port_router(struct
+ case MDB_RTR_TYPE_PERM:
+ p->multicast_router = MDB_RTR_TYPE_PERM;
+ del_timer(&p->ip4_mc_router_timer);
+- br_multicast_add_router(br, p);
++ br_ip4_multicast_add_router(br, p);
+ break;
+ case MDB_RTR_TYPE_TEMP:
+ p->multicast_router = MDB_RTR_TYPE_TEMP;
+- br_multicast_mark_router(br, p);
++ br_ip4_multicast_mark_router(br, p);
+ break;
+ default:
+ goto unlock;
diff --git a/target/linux/generic/backport-5.10/610-v5.14-07-net-bridge-mcast-split-router-port-del-notify-for-mc.patch b/target/linux/generic/backport-5.10/610-v5.14-07-net-bridge-mcast-split-router-port-del-notify-for-mc.patch
new file mode 100644
index 0000000000000000000000000000000000000000..ea0895b16fc3648393a16b7219af1af7a6bd77a1
--- /dev/null
+++ b/target/linux/generic/backport-5.10/610-v5.14-07-net-bridge-mcast-split-router-port-del-notify-for-mc.patch
@@ -0,0 +1,135 @@
+From 047d1e34ade29e4d91797e5510f24aae75783895 Mon Sep 17 00:00:00 2001
+From: =?UTF-8?q?Linus=20L=C3=BCssing?= <linus.luessing@c0d3.blue>
+Date: Thu, 13 May 2021 15:20:50 +0200
+Subject: [PATCH 08/13] net: bridge: mcast: split router port del+notify for
+ mcast router split
+MIME-Version: 1.0
+Content-Type: text/plain; charset=UTF-8
+Content-Transfer-Encoding: 8bit
+
+In preparation for the upcoming split of multicast router state into
+their IPv4 and IPv6 variants split router port deletion and notification
+into two functions. When we disable a port for instance later we want to
+only send one notification to switchdev and netlink for compatibility
+and want to avoid sending one for IPv4 and one for IPv6. For that the
+split is needed.
+
+Signed-off-by: Linus Lüssing <linus.luessing@c0d3.blue>
+Signed-off-by: David S. Miller <davem@davemloft.net>
+---
+ net/bridge/br_multicast.c | 40 ++++++++++++++++++++++++++++++---------
+ 1 file changed, 31 insertions(+), 9 deletions(-)
+
+--- a/net/bridge/br_multicast.c
++++ b/net/bridge/br_multicast.c
+@@ -59,7 +59,8 @@ static void br_ip4_multicast_leave_group
+ const unsigned char *src);
+ static void br_multicast_port_group_rexmit(struct timer_list *t);
+
+-static void __del_port_router(struct net_bridge_port *p);
++static void
++br_multicast_rport_del_notify(struct net_bridge_port *p, bool deleted);
+ #if IS_ENABLED(CONFIG_IPV6)
+ static void br_ip6_multicast_leave_group(struct net_bridge *br,
+ struct net_bridge_port *port,
+@@ -1341,11 +1342,26 @@ static int br_ip6_multicast_add_group(st
+ }
+ #endif
+
++static bool br_multicast_rport_del(struct hlist_node *rlist)
++{
++ if (hlist_unhashed(rlist))
++ return false;
++
++ hlist_del_init_rcu(rlist);
++ return true;
++}
++
++static bool br_ip4_multicast_rport_del(struct net_bridge_port *p)
++{
++ return br_multicast_rport_del(&p->ip4_rlist);
++}
++
+ static void br_multicast_router_expired(struct net_bridge_port *port,
+ struct timer_list *t,
+ struct hlist_node *rlist)
+ {
+ struct net_bridge *br = port->br;
++ bool del;
+
+ spin_lock(&br->multicast_lock);
+ if (port->multicast_router == MDB_RTR_TYPE_DISABLED ||
+@@ -1353,7 +1369,8 @@ static void br_multicast_router_expired(
+ timer_pending(t))
+ goto out;
+
+- __del_port_router(port);
++ del = br_multicast_rport_del(rlist);
++ br_multicast_rport_del_notify(port, del);
+ out:
+ spin_unlock(&br->multicast_lock);
+ }
+@@ -1684,19 +1701,20 @@ void br_multicast_disable_port(struct ne
+ struct net_bridge *br = port->br;
+ struct net_bridge_port_group *pg;
+ struct hlist_node *n;
++ bool del = false;
+
+ spin_lock(&br->multicast_lock);
+ hlist_for_each_entry_safe(pg, n, &port->mglist, mglist)
+ if (!(pg->flags & MDB_PG_FLAGS_PERMANENT))
+ br_multicast_find_del_pg(br, pg);
+
+- __del_port_router(port);
+-
++ del |= br_ip4_multicast_rport_del(port);
+ del_timer(&port->ip4_mc_router_timer);
+ del_timer(&port->ip4_own_query.timer);
+ #if IS_ENABLED(CONFIG_IPV6)
+ del_timer(&port->ip6_own_query.timer);
+ #endif
++ br_multicast_rport_del_notify(port, del);
+ spin_unlock(&br->multicast_lock);
+ }
+
+@@ -3455,11 +3473,12 @@ int br_multicast_set_router(struct net_b
+ return err;
+ }
+
+-static void __del_port_router(struct net_bridge_port *p)
++static void
++br_multicast_rport_del_notify(struct net_bridge_port *p, bool deleted)
+ {
+- if (hlist_unhashed(&p->ip4_rlist))
++ if (!deleted)
+ return;
+- hlist_del_init_rcu(&p->ip4_rlist);
++
+ br_rtr_notify(p->br->dev, p, RTM_DELMDB);
+ br_port_mc_router_state_change(p, false);
+
+@@ -3473,6 +3492,7 @@ int br_multicast_set_port_router(struct
+ struct net_bridge *br = p->br;
+ unsigned long now = jiffies;
+ int err = -EINVAL;
++ bool del = false;
+
+ spin_lock(&br->multicast_lock);
+ if (p->multicast_router == val) {
+@@ -3486,12 +3506,14 @@ int br_multicast_set_port_router(struct
+ switch (val) {
+ case MDB_RTR_TYPE_DISABLED:
+ p->multicast_router = MDB_RTR_TYPE_DISABLED;
+- __del_port_router(p);
++ del |= br_ip4_multicast_rport_del(p);
+ del_timer(&p->ip4_mc_router_timer);
++ br_multicast_rport_del_notify(p, del);
+ break;
+ case MDB_RTR_TYPE_TEMP_QUERY:
+ p->multicast_router = MDB_RTR_TYPE_TEMP_QUERY;
+- __del_port_router(p);
++ del |= br_ip4_multicast_rport_del(p);
++ br_multicast_rport_del_notify(p, del);
+ break;
+ case MDB_RTR_TYPE_PERM:
+ p->multicast_router = MDB_RTR_TYPE_PERM;
diff --git a/target/linux/generic/backport-5.10/610-v5.14-08-net-bridge-mcast-split-multicast-router-state-for-IP.patch b/target/linux/generic/backport-5.10/610-v5.14-08-net-bridge-mcast-split-multicast-router-state-for-IP.patch
new file mode 100644
index 0000000000000000000000000000000000000000..8c8ba065a322b7bd6c55ba6f51bd3b7941a8a584
--- /dev/null
+++ b/target/linux/generic/backport-5.10/610-v5.14-08-net-bridge-mcast-split-multicast-router-state-for-IP.patch
@@ -0,0 +1,410 @@
+From 2542514507c9c942f157c01882a380e0cbc0e6ea Mon Sep 17 00:00:00 2001
+From: =?UTF-8?q?Linus=20L=C3=BCssing?= <linus.luessing@c0d3.blue>
+Date: Thu, 13 May 2021 15:20:51 +0200
+Subject: [PATCH 09/13] net: bridge: mcast: split multicast router state for
+ IPv4 and IPv6
+MIME-Version: 1.0
+Content-Type: text/plain; charset=UTF-8
+Content-Transfer-Encoding: 8bit
+
+A multicast router for IPv4 does not imply that the same host also is a
+multicast router for IPv6 and vice versa.
+
+To reduce multicast traffic when a host is only a multicast router for
+one of these two protocol families, keep router state for IPv4 and IPv6
+separately. Similar to how querier state is kept separately.
+
+For backwards compatibility for netlink and switchdev notifications
+these two will still only notify if a port switched from either no
+IPv4/IPv6 multicast router to any IPv4/IPv6 multicast router or the
+other way round. However a full netlink MDB router dump will now also
+include a multicast router timeout for both IPv4 and IPv6.
+
+Signed-off-by: Linus Lüssing <linus.luessing@c0d3.blue>
+Signed-off-by: David S. Miller <davem@davemloft.net>
+---
+ net/bridge/br_mdb.c | 10 +++
+ net/bridge/br_multicast.c | 134 ++++++++++++++++++++++++++++++++++++--
+ net/bridge/br_private.h | 14 +++-
+ 3 files changed, 151 insertions(+), 7 deletions(-)
+
+--- a/net/bridge/br_mdb.c
++++ b/net/bridge/br_mdb.c
+@@ -18,7 +18,12 @@
+
+ static bool br_rports_have_mc_router(struct net_bridge *br)
+ {
++#if IS_ENABLED(CONFIG_IPV6)
++ return !hlist_empty(&br->ip4_mc_router_list) ||
++ !hlist_empty(&br->ip6_mc_router_list);
++#else
+ return !hlist_empty(&br->ip4_mc_router_list);
++#endif
+ }
+
+ static bool
+@@ -31,8 +36,13 @@ br_ip4_rports_get_timer(struct net_bridg
+ static bool
+ br_ip6_rports_get_timer(struct net_bridge_port *port, unsigned long *timer)
+ {
++#if IS_ENABLED(CONFIG_IPV6)
++ *timer = br_timer_value(&port->ip6_mc_router_timer);
++ return !hlist_unhashed(&port->ip6_rlist);
++#else
+ *timer = 0;
+ return false;
++#endif
+ }
+
+ static int br_rports_fill_info(struct sk_buff *skb, struct netlink_callback *cb,
+--- a/net/bridge/br_multicast.c
++++ b/net/bridge/br_multicast.c
+@@ -62,6 +62,8 @@ static void br_multicast_port_group_rexm
+ static void
+ br_multicast_rport_del_notify(struct net_bridge_port *p, bool deleted);
+ #if IS_ENABLED(CONFIG_IPV6)
++static void br_ip6_multicast_add_router(struct net_bridge *br,
++ struct net_bridge_port *port);
+ static void br_ip6_multicast_leave_group(struct net_bridge *br,
+ struct net_bridge_port *port,
+ const struct in6_addr *group,
+@@ -1356,6 +1358,15 @@ static bool br_ip4_multicast_rport_del(s
+ return br_multicast_rport_del(&p->ip4_rlist);
+ }
+
++static bool br_ip6_multicast_rport_del(struct net_bridge_port *p)
++{
++#if IS_ENABLED(CONFIG_IPV6)
++ return br_multicast_rport_del(&p->ip6_rlist);
++#else
++ return false;
++#endif
++}
++
+ static void br_multicast_router_expired(struct net_bridge_port *port,
+ struct timer_list *t,
+ struct hlist_node *rlist)
+@@ -1382,6 +1393,15 @@ static void br_ip4_multicast_router_expi
+ br_multicast_router_expired(port, t, &port->ip4_rlist);
+ }
+
++#if IS_ENABLED(CONFIG_IPV6)
++static void br_ip6_multicast_router_expired(struct timer_list *t)
++{
++ struct net_bridge_port *port = from_timer(port, t, ip6_mc_router_timer);
++
++ br_multicast_router_expired(port, t, &port->ip6_rlist);
++}
++#endif
++
+ static void br_mc_router_state_change(struct net_bridge *p,
+ bool is_mc_router)
+ {
+@@ -1417,6 +1437,15 @@ static void br_ip4_multicast_local_route
+ br_multicast_local_router_expired(br, t);
+ }
+
++#if IS_ENABLED(CONFIG_IPV6)
++static void br_ip6_multicast_local_router_expired(struct timer_list *t)
++{
++ struct net_bridge *br = from_timer(br, t, ip6_mc_router_timer);
++
++ br_multicast_local_router_expired(br, t);
++}
++#endif
++
+ static void br_multicast_querier_expired(struct net_bridge *br,
+ struct bridge_mcast_own_query *query)
+ {
+@@ -1632,6 +1661,8 @@ int br_multicast_add_port(struct net_bri
+ timer_setup(&port->ip4_own_query.timer,
+ br_ip4_multicast_port_query_expired, 0);
+ #if IS_ENABLED(CONFIG_IPV6)
++ timer_setup(&port->ip6_mc_router_timer,
++ br_ip6_multicast_router_expired, 0);
+ timer_setup(&port->ip6_own_query.timer,
+ br_ip6_multicast_port_query_expired, 0);
+ #endif
+@@ -1660,6 +1691,9 @@ void br_multicast_del_port(struct net_br
+ spin_unlock_bh(&br->multicast_lock);
+ br_multicast_gc(&deleted_head);
+ del_timer_sync(&port->ip4_mc_router_timer);
++#if IS_ENABLED(CONFIG_IPV6)
++ del_timer_sync(&port->ip6_mc_router_timer);
++#endif
+ free_percpu(port->mcast_stats);
+ }
+
+@@ -1683,8 +1717,10 @@ static void __br_multicast_enable_port(s
+ #if IS_ENABLED(CONFIG_IPV6)
+ br_multicast_enable(&port->ip6_own_query);
+ #endif
+- if (port->multicast_router == MDB_RTR_TYPE_PERM)
++ if (port->multicast_router == MDB_RTR_TYPE_PERM) {
+ br_ip4_multicast_add_router(br, port);
++ br_ip6_multicast_add_router(br, port);
++ }
+ }
+
+ void br_multicast_enable_port(struct net_bridge_port *port)
+@@ -1711,7 +1747,9 @@ void br_multicast_disable_port(struct ne
+ del |= br_ip4_multicast_rport_del(port);
+ del_timer(&port->ip4_mc_router_timer);
+ del_timer(&port->ip4_own_query.timer);
++ del |= br_ip6_multicast_rport_del(port);
+ #if IS_ENABLED(CONFIG_IPV6)
++ del_timer(&port->ip6_mc_router_timer);
+ del_timer(&port->ip6_own_query.timer);
+ #endif
+ br_multicast_rport_del_notify(port, del);
+@@ -2586,6 +2624,10 @@ br_multicast_rport_from_node(struct net_
+ struct hlist_head *mc_router_list,
+ struct hlist_node *rlist)
+ {
++#if IS_ENABLED(CONFIG_IPV6)
++ if (mc_router_list == &br->ip6_mc_router_list)
++ return hlist_entry(rlist, struct net_bridge_port, ip6_rlist);
++#endif
+ return hlist_entry(rlist, struct net_bridge_port, ip4_rlist);
+ }
+
+@@ -2611,6 +2653,19 @@ br_multicast_get_rport_slot(struct net_b
+ return slot;
+ }
+
++static bool br_multicast_no_router_otherpf(struct net_bridge_port *port,
++ struct hlist_node *rnode)
++{
++#if IS_ENABLED(CONFIG_IPV6)
++ if (rnode != &port->ip6_rlist)
++ return hlist_unhashed(&port->ip6_rlist);
++ else
++ return hlist_unhashed(&port->ip4_rlist);
++#else
++ return true;
++#endif
++}
++
+ /* Add port to router_list
+ * list is maintained ordered by pointer value
+ * and locked by br->multicast_lock and RCU
+@@ -2632,8 +2687,14 @@ static void br_multicast_add_router(stru
+ else
+ hlist_add_head_rcu(rlist, mc_router_list);
+
+- br_rtr_notify(br->dev, port, RTM_NEWMDB);
+- br_port_mc_router_state_change(port, true);
++ /* For backwards compatibility for now, only notify if we
++ * switched from no IPv4/IPv6 multicast router to a new
++ * IPv4 or IPv6 multicast router.
++ */
++ if (br_multicast_no_router_otherpf(port, rlist)) {
++ br_rtr_notify(br->dev, port, RTM_NEWMDB);
++ br_port_mc_router_state_change(port, true);
++ }
+ }
+
+ /* Add port to router_list
+@@ -2647,6 +2708,19 @@ static void br_ip4_multicast_add_router(
+ &br->ip4_mc_router_list);
+ }
+
++/* Add port to router_list
++ * list is maintained ordered by pointer value
++ * and locked by br->multicast_lock and RCU
++ */
++static void br_ip6_multicast_add_router(struct net_bridge *br,
++ struct net_bridge_port *port)
++{
++#if IS_ENABLED(CONFIG_IPV6)
++ br_multicast_add_router(br, port, &port->ip6_rlist,
++ &br->ip6_mc_router_list);
++#endif
++}
++
+ static void br_multicast_mark_router(struct net_bridge *br,
+ struct net_bridge_port *port,
+ struct timer_list *timer,
+@@ -2688,6 +2762,23 @@ static void br_ip4_multicast_mark_router
+ &br->ip4_mc_router_list);
+ }
+
++static void br_ip6_multicast_mark_router(struct net_bridge *br,
++ struct net_bridge_port *port)
++{
++#if IS_ENABLED(CONFIG_IPV6)
++ struct timer_list *timer = &br->ip6_mc_router_timer;
++ struct hlist_node *rlist = NULL;
++
++ if (port) {
++ timer = &port->ip6_mc_router_timer;
++ rlist = &port->ip6_rlist;
++ }
++
++ br_multicast_mark_router(br, port, timer, rlist,
++ &br->ip6_mc_router_list);
++#endif
++}
++
+ static void
+ br_ip4_multicast_query_received(struct net_bridge *br,
+ struct net_bridge_port *port,
+@@ -2714,7 +2805,7 @@ br_ip6_multicast_query_received(struct n
+ return;
+
+ br_multicast_update_query_timer(br, query, max_delay);
+- br_ip4_multicast_mark_router(br, port);
++ br_ip6_multicast_mark_router(br, port);
+ }
+ #endif
+
+@@ -3168,7 +3259,7 @@ static void br_ip6_multicast_mrd_rcv(str
+ return;
+
+ spin_lock(&br->multicast_lock);
+- br_ip4_multicast_mark_router(br, port);
++ br_ip6_multicast_mark_router(br, port);
+ spin_unlock(&br->multicast_lock);
+ }
+
+@@ -3326,6 +3417,8 @@ void br_multicast_init(struct net_bridge
+ timer_setup(&br->ip4_own_query.timer,
+ br_ip4_multicast_query_expired, 0);
+ #if IS_ENABLED(CONFIG_IPV6)
++ timer_setup(&br->ip6_mc_router_timer,
++ br_ip6_multicast_local_router_expired, 0);
+ timer_setup(&br->ip6_other_query.timer,
+ br_ip6_multicast_querier_expired, 0);
+ timer_setup(&br->ip6_own_query.timer,
+@@ -3423,6 +3516,7 @@ void br_multicast_stop(struct net_bridge
+ del_timer_sync(&br->ip4_other_query.timer);
+ del_timer_sync(&br->ip4_own_query.timer);
+ #if IS_ENABLED(CONFIG_IPV6)
++ del_timer_sync(&br->ip6_mc_router_timer);
+ del_timer_sync(&br->ip6_other_query.timer);
+ del_timer_sync(&br->ip6_own_query.timer);
+ #endif
+@@ -3457,6 +3551,9 @@ int br_multicast_set_router(struct net_b
+ case MDB_RTR_TYPE_PERM:
+ br_mc_router_state_change(br, val == MDB_RTR_TYPE_PERM);
+ del_timer(&br->ip4_mc_router_timer);
++#if IS_ENABLED(CONFIG_IPV6)
++ del_timer(&br->ip6_mc_router_timer);
++#endif
+ br->multicast_router = val;
+ err = 0;
+ break;
+@@ -3479,6 +3576,16 @@ br_multicast_rport_del_notify(struct net
+ if (!deleted)
+ return;
+
++ /* For backwards compatibility for now, only notify if there is
++ * no multicast router anymore for both IPv4 and IPv6.
++ */
++ if (!hlist_unhashed(&p->ip4_rlist))
++ return;
++#if IS_ENABLED(CONFIG_IPV6)
++ if (!hlist_unhashed(&p->ip6_rlist))
++ return;
++#endif
++
+ br_rtr_notify(p->br->dev, p, RTM_DELMDB);
+ br_port_mc_router_state_change(p, false);
+
+@@ -3497,9 +3604,14 @@ int br_multicast_set_port_router(struct
+ spin_lock(&br->multicast_lock);
+ if (p->multicast_router == val) {
+ /* Refresh the temp router port timer */
+- if (p->multicast_router == MDB_RTR_TYPE_TEMP)
++ if (p->multicast_router == MDB_RTR_TYPE_TEMP) {
+ mod_timer(&p->ip4_mc_router_timer,
+ now + br->multicast_querier_interval);
++#if IS_ENABLED(CONFIG_IPV6)
++ mod_timer(&p->ip6_mc_router_timer,
++ now + br->multicast_querier_interval);
++#endif
++ }
+ err = 0;
+ goto unlock;
+ }
+@@ -3508,21 +3620,31 @@ int br_multicast_set_port_router(struct
+ p->multicast_router = MDB_RTR_TYPE_DISABLED;
+ del |= br_ip4_multicast_rport_del(p);
+ del_timer(&p->ip4_mc_router_timer);
++ del |= br_ip6_multicast_rport_del(p);
++#if IS_ENABLED(CONFIG_IPV6)
++ del_timer(&p->ip6_mc_router_timer);
++#endif
+ br_multicast_rport_del_notify(p, del);
+ break;
+ case MDB_RTR_TYPE_TEMP_QUERY:
+ p->multicast_router = MDB_RTR_TYPE_TEMP_QUERY;
+ del |= br_ip4_multicast_rport_del(p);
++ del |= br_ip6_multicast_rport_del(p);
+ br_multicast_rport_del_notify(p, del);
+ break;
+ case MDB_RTR_TYPE_PERM:
+ p->multicast_router = MDB_RTR_TYPE_PERM;
+ del_timer(&p->ip4_mc_router_timer);
+ br_ip4_multicast_add_router(br, p);
++#if IS_ENABLED(CONFIG_IPV6)
++ del_timer(&p->ip6_mc_router_timer);
++#endif
++ br_ip6_multicast_add_router(br, p);
+ break;
+ case MDB_RTR_TYPE_TEMP:
+ p->multicast_router = MDB_RTR_TYPE_TEMP;
+ br_ip4_multicast_mark_router(br, p);
++ br_ip6_multicast_mark_router(br, p);
+ break;
+ default:
+ goto unlock;
+--- a/net/bridge/br_private.h
++++ b/net/bridge/br_private.h
+@@ -317,6 +317,8 @@ struct net_bridge_port {
+ struct hlist_node ip4_rlist;
+ #if IS_ENABLED(CONFIG_IPV6)
+ struct bridge_mcast_own_query ip6_own_query;
++ struct timer_list ip6_mc_router_timer;
++ struct hlist_node ip6_rlist;
+ #endif /* IS_ENABLED(CONFIG_IPV6) */
+ unsigned char multicast_router;
+ struct bridge_mcast_stats __percpu *mcast_stats;
+@@ -461,6 +463,8 @@ struct net_bridge {
+ struct bridge_mcast_querier ip4_querier;
+ struct bridge_mcast_stats __percpu *mcast_stats;
+ #if IS_ENABLED(CONFIG_IPV6)
++ struct hlist_head ip6_mc_router_list;
++ struct timer_list ip6_mc_router_timer;
+ struct bridge_mcast_other_query ip6_other_query;
+ struct bridge_mcast_own_query ip6_own_query;
+ struct bridge_mcast_querier ip6_querier;
+@@ -847,11 +851,19 @@ void br_multicast_sg_add_exclude_ports(s
+
+ static inline struct hlist_node *
+ br_multicast_get_first_rport_node(struct net_bridge *b, struct sk_buff *skb) {
++#if IS_ENABLED(CONFIG_IPV6)
++ if (skb->protocol == htons(ETH_P_IPV6))
++ return rcu_dereference(hlist_first_rcu(&b->ip6_mc_router_list));
++#endif
+ return rcu_dereference(hlist_first_rcu(&b->ip4_mc_router_list));
+ }
+
+ static inline struct net_bridge_port *
+ br_multicast_rport_from_node_skb(struct hlist_node *rp, struct sk_buff *skb) {
++#if IS_ENABLED(CONFIG_IPV6)
++ if (skb->protocol == htons(ETH_P_IPV6))
++ return hlist_entry_safe(rp, struct net_bridge_port, ip6_rlist);
++#endif
+ return hlist_entry_safe(rp, struct net_bridge_port, ip4_rlist);
+ }
+
+@@ -863,7 +875,7 @@ static inline bool br_ip4_multicast_is_r
+ static inline bool br_ip6_multicast_is_router(struct net_bridge *br)
+ {
+ #if IS_ENABLED(CONFIG_IPV6)
+- return timer_pending(&br->ip4_mc_router_timer);
++ return timer_pending(&br->ip6_mc_router_timer);
+ #else
+ return false;
+ #endif
diff --git a/target/linux/generic/backport-5.10/610-v5.14-09-net-bridge-mcast-add-ip4-ip6-mcast-router-timers-to-.patch b/target/linux/generic/backport-5.10/610-v5.14-09-net-bridge-mcast-add-ip4-ip6-mcast-router-timers-to-.patch
new file mode 100644
index 0000000000000000000000000000000000000000..3aeacab6fd29c024865dc60912cfbb92275a3164
--- /dev/null
+++ b/target/linux/generic/backport-5.10/610-v5.14-09-net-bridge-mcast-add-ip4-ip6-mcast-router-timers-to-.patch
@@ -0,0 +1,48 @@
+From 46b9de28078dd3f35ce41ad72fa29e8cbc944e0b Mon Sep 17 00:00:00 2001
+From: =?UTF-8?q?Linus=20L=C3=BCssing?= <linus.luessing@c0d3.blue>
+Date: Thu, 13 May 2021 15:20:52 +0200
+Subject: [PATCH 10/13] net: bridge: mcast: add ip4+ip6 mcast router timers to
+ mdb netlink
+MIME-Version: 1.0
+Content-Type: text/plain; charset=UTF-8
+Content-Transfer-Encoding: 8bit
+
+Now that we have split the multicast router state into two, one for IPv4
+and one for IPv6, also add individual timers to the mdb netlink router
+port dump. Leaving the old timer attribute for backwards compatibility.
+
+Signed-off-by: Linus Lüssing <linus.luessing@c0d3.blue>
+Signed-off-by: David S. Miller <davem@davemloft.net>
+---
+ include/uapi/linux/if_bridge.h | 2 ++
+ net/bridge/br_mdb.c | 8 +++++++-
+ 2 files changed, 9 insertions(+), 1 deletion(-)
+
+--- a/include/uapi/linux/if_bridge.h
++++ b/include/uapi/linux/if_bridge.h
+@@ -502,6 +502,8 @@ enum {
+ MDBA_ROUTER_PATTR_UNSPEC,
+ MDBA_ROUTER_PATTR_TIMER,
+ MDBA_ROUTER_PATTR_TYPE,
++ MDBA_ROUTER_PATTR_INET_TIMER,
++ MDBA_ROUTER_PATTR_INET6_TIMER,
+ __MDBA_ROUTER_PATTR_MAX
+ };
+ #define MDBA_ROUTER_PATTR_MAX (__MDBA_ROUTER_PATTR_MAX - 1)
+--- a/net/bridge/br_mdb.c
++++ b/net/bridge/br_mdb.c
+@@ -79,7 +79,13 @@ static int br_rports_fill_info(struct sk
+ nla_put_u32(skb, MDBA_ROUTER_PATTR_TIMER,
+ max(ip4_timer, ip6_timer)) ||
+ nla_put_u8(skb, MDBA_ROUTER_PATTR_TYPE,
+- p->multicast_router)) {
++ p->multicast_router) ||
++ (have_ip4_mc_rtr &&
++ nla_put_u32(skb, MDBA_ROUTER_PATTR_INET_TIMER,
++ ip4_timer)) ||
++ (have_ip6_mc_rtr &&
++ nla_put_u32(skb, MDBA_ROUTER_PATTR_INET6_TIMER,
++ ip6_timer))) {
+ nla_nest_cancel(skb, port_nest);
+ goto fail;
+ }
diff --git a/target/linux/generic/backport-5.10/610-v5.14-10-net-bridge-mcast-export-multicast-router-presence-ad.patch b/target/linux/generic/backport-5.10/610-v5.14-10-net-bridge-mcast-export-multicast-router-presence-ad.patch
new file mode 100644
index 0000000000000000000000000000000000000000..284550befaf7423e5c682fb7193985206901513e
--- /dev/null
+++ b/target/linux/generic/backport-5.10/610-v5.14-10-net-bridge-mcast-export-multicast-router-presence-ad.patch
@@ -0,0 +1,112 @@
+From 2e1580e0ea307cec3eca1896d53353b7b7d50ef3 Mon Sep 17 00:00:00 2001
+From: =?UTF-8?q?Linus=20L=C3=BCssing?= <linus.luessing@c0d3.blue>
+Date: Thu, 13 May 2021 15:20:53 +0200
+Subject: [PATCH 11/13] net: bridge: mcast: export multicast router presence
+ adjacent to a port
+MIME-Version: 1.0
+Content-Type: text/plain; charset=UTF-8
+Content-Transfer-Encoding: 8bit
+
+To properly support routable multicast addresses in batman-adv in a
+group-aware way, a batman-adv node needs to know if it serves multicast
+routers.
+
+This adds a function to the bridge to export this so that batman-adv
+can then make full use of the Multicast Router Discovery capability of
+the bridge.
+
+Signed-off-by: Linus Lüssing <linus.luessing@c0d3.blue>
+Signed-off-by: David S. Miller <davem@davemloft.net>
+---
+ include/linux/if_bridge.h | 8 ++++++
+ net/bridge/br_multicast.c | 55 +++++++++++++++++++++++++++++++++++++++
+ 2 files changed, 63 insertions(+)
+
+--- a/include/linux/if_bridge.h
++++ b/include/linux/if_bridge.h
+@@ -66,6 +66,7 @@ int br_multicast_list_adjacent(struct ne
+ struct list_head *br_ip_list);
+ bool br_multicast_has_querier_anywhere(struct net_device *dev, int proto);
+ bool br_multicast_has_querier_adjacent(struct net_device *dev, int proto);
++bool br_multicast_has_router_adjacent(struct net_device *dev, int proto);
+ bool br_multicast_enabled(const struct net_device *dev);
+ bool br_multicast_router(const struct net_device *dev);
+ #else
+@@ -84,6 +85,13 @@ static inline bool br_multicast_has_quer
+ {
+ return false;
+ }
++
++static inline bool br_multicast_has_router_adjacent(struct net_device *dev,
++ int proto)
++{
++ return true;
++}
++
+ static inline bool br_multicast_enabled(const struct net_device *dev)
+ {
+ return false;
+--- a/net/bridge/br_multicast.c
++++ b/net/bridge/br_multicast.c
+@@ -3964,6 +3964,61 @@ unlock:
+ }
+ EXPORT_SYMBOL_GPL(br_multicast_has_querier_adjacent);
+
++/**
++ * br_multicast_has_router_adjacent - Checks for a router behind a bridge port
++ * @dev: The bridge port adjacent to which to check for a multicast router
++ * @proto: The protocol family to check for: IGMP -> ETH_P_IP, MLD -> ETH_P_IPV6
++ *
++ * Checks whether the given interface has a bridge on top and if so returns
++ * true if a multicast router is behind one of the other ports of this
++ * bridge. Otherwise returns false.
++ */
++bool br_multicast_has_router_adjacent(struct net_device *dev, int proto)
++{
++ struct net_bridge_port *port, *p;
++ bool ret = false;
++
++ rcu_read_lock();
++ port = br_port_get_check_rcu(dev);
++ if (!port)
++ goto unlock;
++
++ switch (proto) {
++ case ETH_P_IP:
++ hlist_for_each_entry_rcu(p, &port->br->ip4_mc_router_list,
++ ip4_rlist) {
++ if (p == port)
++ continue;
++
++ ret = true;
++ goto unlock;
++ }
++ break;
++#if IS_ENABLED(CONFIG_IPV6)
++ case ETH_P_IPV6:
++ hlist_for_each_entry_rcu(p, &port->br->ip6_mc_router_list,
++ ip6_rlist) {
++ if (p == port)
++ continue;
++
++ ret = true;
++ goto unlock;
++ }
++ break;
++#endif
++ default:
++ /* when compiled without IPv6 support, be conservative and
++ * always assume presence of an IPv6 multicast router
++ */
++ ret = true;
++ }
++
++unlock:
++ rcu_read_unlock();
++ return ret;
++}
++EXPORT_SYMBOL_GPL(br_multicast_has_router_adjacent);
++
+ static void br_mcast_stats_add(struct bridge_mcast_stats __percpu *stats,
+ const struct sk_buff *skb, u8 type, u8 dir)
+ {
diff --git a/target/linux/generic/backport-5.10/610-v5.14-11-net-bridge-fix-br_multicast_is_router-stub-when-igmp.patch b/target/linux/generic/backport-5.10/610-v5.14-11-net-bridge-fix-br_multicast_is_router-stub-when-igmp.patch
new file mode 100644
index 0000000000000000000000000000000000000000..c178ed3829e1c4795411476d1dec7ea382fe166f
--- /dev/null
+++ b/target/linux/generic/backport-5.10/610-v5.14-11-net-bridge-fix-br_multicast_is_router-stub-when-igmp.patch
@@ -0,0 +1,32 @@
+From 8577c2385c98490cb1c7c4821954c2e36511d733 Mon Sep 17 00:00:00 2001
+From: Nikolay Aleksandrov <nikolay@nvidia.com>
+Date: Fri, 14 May 2021 10:32:33 +0300
+Subject: [PATCH 12/13] net: bridge: fix br_multicast_is_router stub when igmp
+ is disabled
+MIME-Version: 1.0
+Content-Type: text/plain; charset=UTF-8
+Content-Transfer-Encoding: 8bit
+
+br_multicast_is_router takes two arguments when bridge IGMP is enabled
+and just one when it's disabled, fix the stub to take two as well.
+
+Fixes: 1a3065a26807 ("net: bridge: mcast: prepare is-router function for mcast router split")
+Signed-off-by: Nikolay Aleksandrov <nikolay@nvidia.com>
+Acked-by: Linus Lüssing <linus.luessing@c0d3.blue>
+Signed-off-by: David S. Miller <davem@davemloft.net>
+---
+ net/bridge/br_private.h | 3 ++-
+ 1 file changed, 2 insertions(+), 1 deletion(-)
+
+--- a/net/bridge/br_private.h
++++ b/net/bridge/br_private.h
+@@ -1046,7 +1046,8 @@ static inline void br_multicast_flood(st
+ {
+ }
+
+-static inline bool br_multicast_is_router(struct net_bridge *br)
++static inline bool br_multicast_is_router(struct net_bridge *br,
++ struct sk_buff *skb)
+ {
+ return false;
+ }
diff --git a/target/linux/generic/backport-5.10/610-v5.14-12-net-bridge-fix-build-when-IPv6-is-disabled.patch b/target/linux/generic/backport-5.10/610-v5.14-12-net-bridge-fix-build-when-IPv6-is-disabled.patch
new file mode 100644
index 0000000000000000000000000000000000000000..7716079dd490ca9f83db3489f45f4e6f99d1f5bb
--- /dev/null
+++ b/target/linux/generic/backport-5.10/610-v5.14-12-net-bridge-fix-build-when-IPv6-is-disabled.patch
@@ -0,0 +1,48 @@
+From ca129ff130fa72e2e4d3ec4edd5a694165796e56 Mon Sep 17 00:00:00 2001
+From: Matteo Croce <mcroce@microsoft.com>
+Date: Fri, 14 May 2021 03:53:48 +0200
+Subject: [PATCH 13/13] net: bridge: fix build when IPv6 is disabled
+MIME-Version: 1.0
+Content-Type: text/plain; charset=UTF-8
+Content-Transfer-Encoding: 8bit
+
+The br_ip6_multicast_add_router() prototype is defined only when
+CONFIG_IPV6 is enabled, but the function is always referenced, so there
+is this build error with CONFIG_IPV6 not defined:
+
+net/bridge/br_multicast.c: In function __br_multicast_enable_port:
+net/bridge/br_multicast.c:1743:3: error: implicit declaration of function br_ip6_multicast_add_router; did you mean br_ip4_multicast_add_router? [-Werror=implicit-function-declaration]
+ 1743 | br_ip6_multicast_add_router(br, port);
+ | ^~~~~~~~~~~~~~~~~~~~~~~~~~~
+ | br_ip4_multicast_add_router
+net/bridge/br_multicast.c: At top level:
+net/bridge/br_multicast.c:2804:13: warning: conflicting types for br_ip6_multicast_add_router
+ 2804 | static void br_ip6_multicast_add_router(struct net_bridge *br,
+ | ^~~~~~~~~~~~~~~~~~~~~~~~~~~
+net/bridge/br_multicast.c:2804:13: error: static declaration of br_ip6_multicast_add_router follows non-static declaration
+net/bridge/br_multicast.c:1743:3: note: previous implicit declaration of br_ip6_multicast_add_router was here
+ 1743 | br_ip6_multicast_add_router(br, port);
+ | ^~~~~~~~~~~~~~~~~~~~~~~~~~~
+
+Fix this build error by moving the definition out of the #ifdef.
+
+Fixes: a3c02e769efe ("net: bridge: mcast: split multicast router state for IPv4 and IPv6")
+Signed-off-by: Matteo Croce <mcroce@microsoft.com>
+Signed-off-by: David S. Miller <davem@davemloft.net>
+---
+ net/bridge/br_multicast.c | 2 +-
+ 1 file changed, 1 insertion(+), 1 deletion(-)
+
+--- a/net/bridge/br_multicast.c
++++ b/net/bridge/br_multicast.c
+@@ -61,9 +61,9 @@ static void br_multicast_port_group_rexm
+
+ static void
+ br_multicast_rport_del_notify(struct net_bridge_port *p, bool deleted);
+-#if IS_ENABLED(CONFIG_IPV6)
+ static void br_ip6_multicast_add_router(struct net_bridge *br,
+ struct net_bridge_port *port);
++#if IS_ENABLED(CONFIG_IPV6)
+ static void br_ip6_multicast_leave_group(struct net_bridge *br,
+ struct net_bridge_port *port,
+ const struct in6_addr *group,