Commit bb18ef8e authored by Nikolay Aleksandrov's avatar Nikolay Aleksandrov Committed by David S. Miller

net: bridge: mcast: record querier port device ifindex instead of pointer

Currently when a querier port is detected its net_bridge_port pointer is
recorded, but it's used only for comparisons so it's fine to have stale
pointer, in order to dereference and use the port pointer a proper
accounting of its usage must be implemented adding unnecessary
complexity. To solve the problem we can just store the netdevice ifindex
instead of the port pointer and retrieve the bridge port. It is a best
effort and the device needs to be validated that is still part of that
bridge before use, but that is small price to pay for avoiding querier
reference counting for each port/vlan.
Signed-off-by: default avatarNikolay Aleksandrov <nikolay@nvidia.com>
Signed-off-by: default avatarDavid S. Miller <davem@davemloft.net>
parent 2fa16787
...@@ -2850,7 +2850,8 @@ static bool br_ip4_multicast_select_querier(struct net_bridge_mcast *brmctx, ...@@ -2850,7 +2850,8 @@ static bool br_ip4_multicast_select_querier(struct net_bridge_mcast *brmctx,
brmctx->ip4_querier.addr.src.ip4 = saddr; brmctx->ip4_querier.addr.src.ip4 = saddr;
/* update protected by general multicast_lock by caller */ /* update protected by general multicast_lock by caller */
rcu_assign_pointer(brmctx->ip4_querier.port, port); if (port)
brmctx->ip4_querier.port_ifidx = port->dev->ifindex;
return true; return true;
} }
...@@ -2875,7 +2876,8 @@ static bool br_ip6_multicast_select_querier(struct net_bridge_mcast *brmctx, ...@@ -2875,7 +2876,8 @@ static bool br_ip6_multicast_select_querier(struct net_bridge_mcast *brmctx,
brmctx->ip6_querier.addr.src.ip6 = *saddr; brmctx->ip6_querier.addr.src.ip6 = *saddr;
/* update protected by general multicast_lock by caller */ /* update protected by general multicast_lock by caller */
rcu_assign_pointer(brmctx->ip6_querier.port, port); if (port)
brmctx->ip6_querier.port_ifidx = port->dev->ifindex;
return true; return true;
} }
...@@ -3675,7 +3677,7 @@ static void br_multicast_query_expired(struct net_bridge_mcast *brmctx, ...@@ -3675,7 +3677,7 @@ static void br_multicast_query_expired(struct net_bridge_mcast *brmctx,
if (query->startup_sent < brmctx->multicast_startup_query_count) if (query->startup_sent < brmctx->multicast_startup_query_count)
query->startup_sent++; query->startup_sent++;
RCU_INIT_POINTER(querier->port, NULL); querier->port_ifidx = 0;
br_multicast_send_query(brmctx, NULL, query); br_multicast_send_query(brmctx, NULL, query);
out: out:
spin_unlock(&brmctx->br->multicast_lock); spin_unlock(&brmctx->br->multicast_lock);
...@@ -3732,12 +3734,12 @@ void br_multicast_ctx_init(struct net_bridge *br, ...@@ -3732,12 +3734,12 @@ void br_multicast_ctx_init(struct net_bridge *br,
brmctx->multicast_membership_interval = 260 * HZ; brmctx->multicast_membership_interval = 260 * HZ;
brmctx->ip4_other_query.delay_time = 0; brmctx->ip4_other_query.delay_time = 0;
brmctx->ip4_querier.port = NULL; brmctx->ip4_querier.port_ifidx = 0;
brmctx->multicast_igmp_version = 2; brmctx->multicast_igmp_version = 2;
#if IS_ENABLED(CONFIG_IPV6) #if IS_ENABLED(CONFIG_IPV6)
brmctx->multicast_mld_version = 1; brmctx->multicast_mld_version = 1;
brmctx->ip6_other_query.delay_time = 0; brmctx->ip6_other_query.delay_time = 0;
brmctx->ip6_querier.port = NULL; brmctx->ip6_querier.port_ifidx = 0;
#endif #endif
timer_setup(&brmctx->ip4_mc_router_timer, timer_setup(&brmctx->ip4_mc_router_timer,
...@@ -4479,6 +4481,7 @@ bool br_multicast_has_querier_adjacent(struct net_device *dev, int proto) ...@@ -4479,6 +4481,7 @@ bool br_multicast_has_querier_adjacent(struct net_device *dev, int proto)
struct net_bridge *br; struct net_bridge *br;
struct net_bridge_port *port; struct net_bridge_port *port;
bool ret = false; bool ret = false;
int port_ifidx;
rcu_read_lock(); rcu_read_lock();
if (!netif_is_bridge_port(dev)) if (!netif_is_bridge_port(dev))
...@@ -4493,14 +4496,16 @@ bool br_multicast_has_querier_adjacent(struct net_device *dev, int proto) ...@@ -4493,14 +4496,16 @@ bool br_multicast_has_querier_adjacent(struct net_device *dev, int proto)
switch (proto) { switch (proto) {
case ETH_P_IP: case ETH_P_IP:
port_ifidx = brmctx->ip4_querier.port_ifidx;
if (!timer_pending(&brmctx->ip4_other_query.timer) || if (!timer_pending(&brmctx->ip4_other_query.timer) ||
rcu_dereference(brmctx->ip4_querier.port) == port) port_ifidx == port->dev->ifindex)
goto unlock; goto unlock;
break; break;
#if IS_ENABLED(CONFIG_IPV6) #if IS_ENABLED(CONFIG_IPV6)
case ETH_P_IPV6: case ETH_P_IPV6:
port_ifidx = brmctx->ip6_querier.port_ifidx;
if (!timer_pending(&brmctx->ip6_other_query.timer) || if (!timer_pending(&brmctx->ip6_other_query.timer) ||
rcu_dereference(brmctx->ip6_querier.port) == port) port_ifidx == port->dev->ifindex)
goto unlock; goto unlock;
break; break;
#endif #endif
......
...@@ -81,7 +81,7 @@ struct bridge_mcast_other_query { ...@@ -81,7 +81,7 @@ struct bridge_mcast_other_query {
/* selected querier */ /* selected querier */
struct bridge_mcast_querier { struct bridge_mcast_querier {
struct br_ip addr; struct br_ip addr;
struct net_bridge_port __rcu *port; int port_ifidx;
}; };
/* IGMP/MLD statistics */ /* IGMP/MLD statistics */
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment