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

net: bridge: mcast: install S,G entries automatically based on reports

This patch adds support for automatic install of S,G mdb entries based
on the port group's source list and the source entry's timer.
Once installed the S,G will be used when forwarding packets if the
approprate multicast/mld versions are set. A new source flag called
BR_SGRP_F_INSTALLED denotes if the source has a forwarding mdb entry
installed.
Signed-off-by: default avatarNikolay Aleksandrov <nikolay@nvidia.com>
Signed-off-by: default avatarDavid S. Miller <davem@davemloft.net>
parent 085b53c8
...@@ -66,6 +66,13 @@ static void br_ip6_multicast_leave_group(struct net_bridge *br, ...@@ -66,6 +66,13 @@ static void br_ip6_multicast_leave_group(struct net_bridge *br,
const struct in6_addr *group, const struct in6_addr *group,
__u16 vid, const unsigned char *src); __u16 vid, const unsigned char *src);
#endif #endif
static struct net_bridge_port_group *
__br_multicast_add_group(struct net_bridge *br,
struct net_bridge_port *port,
struct br_ip *group,
const unsigned char *src,
u8 filter_mode,
bool igmpv2_mldv1);
static struct net_bridge_port_group * static struct net_bridge_port_group *
br_sg_port_find(struct net_bridge *br, br_sg_port_find(struct net_bridge *br,
...@@ -175,6 +182,81 @@ struct net_bridge_mdb_entry *br_mdb_get(struct net_bridge *br, ...@@ -175,6 +182,81 @@ struct net_bridge_mdb_entry *br_mdb_get(struct net_bridge *br,
return br_mdb_ip_get_rcu(br, &ip); return br_mdb_ip_get_rcu(br, &ip);
} }
static bool br_port_group_equal(struct net_bridge_port_group *p,
struct net_bridge_port *port,
const unsigned char *src)
{
if (p->key.port != port)
return false;
if (!(port->flags & BR_MULTICAST_TO_UNICAST))
return true;
return ether_addr_equal(src, p->eth_addr);
}
static void br_multicast_fwd_src_add(struct net_bridge_group_src *src)
{
struct net_bridge_port_group *sg;
struct br_ip sg_ip;
if (src->flags & BR_SGRP_F_INSTALLED)
return;
memset(&sg_ip, 0, sizeof(sg_ip));
sg_ip = src->pg->key.addr;
sg_ip.src = src->addr.src;
sg = __br_multicast_add_group(src->br, src->pg->key.port, &sg_ip,
src->pg->eth_addr, MCAST_INCLUDE, false);
if (IS_ERR_OR_NULL(sg))
return;
src->flags |= BR_SGRP_F_INSTALLED;
/* if it was added by user-space as perm we can skip next steps */
if (sg->rt_protocol != RTPROT_KERNEL &&
(sg->flags & MDB_PG_FLAGS_PERMANENT))
return;
/* the kernel is now responsible for removing this S,G */
del_timer(&sg->timer);
}
static void br_multicast_fwd_src_remove(struct net_bridge_group_src *src)
{
struct net_bridge_port_group *p, *pg = src->pg;
struct net_bridge_port_group __rcu **pp;
struct net_bridge_mdb_entry *mp;
struct br_ip sg_ip;
memset(&sg_ip, 0, sizeof(sg_ip));
sg_ip = pg->key.addr;
sg_ip.src = src->addr.src;
mp = br_mdb_ip_get(src->br, &sg_ip);
if (!mp)
return;
for (pp = &mp->ports;
(p = mlock_dereference(*pp, src->br)) != NULL;
pp = &p->next) {
if (!br_port_group_equal(p, pg->key.port, pg->eth_addr))
continue;
if (p->rt_protocol != RTPROT_KERNEL &&
(p->flags & MDB_PG_FLAGS_PERMANENT))
break;
br_multicast_del_pg(mp, p, pp);
break;
}
src->flags &= ~BR_SGRP_F_INSTALLED;
}
static void br_multicast_fwd_src_handle(struct net_bridge_group_src *src)
{
br_multicast_fwd_src_add(src);
}
static void br_multicast_destroy_mdb_entry(struct net_bridge_mcast_gc *gc) static void br_multicast_destroy_mdb_entry(struct net_bridge_mcast_gc *gc)
{ {
struct net_bridge_mdb_entry *mp; struct net_bridge_mdb_entry *mp;
...@@ -204,7 +286,8 @@ static void br_multicast_group_expired(struct timer_list *t) ...@@ -204,7 +286,8 @@ static void br_multicast_group_expired(struct timer_list *t)
struct net_bridge *br = mp->br; struct net_bridge *br = mp->br;
spin_lock(&br->multicast_lock); spin_lock(&br->multicast_lock);
if (!netif_running(br->dev) || timer_pending(&mp->timer)) if (hlist_unhashed(&mp->mdb_node) || !netif_running(br->dev) ||
timer_pending(&mp->timer))
goto out; goto out;
br_multicast_host_leave(mp, true); br_multicast_host_leave(mp, true);
...@@ -231,6 +314,7 @@ static void br_multicast_del_group_src(struct net_bridge_group_src *src) ...@@ -231,6 +314,7 @@ static void br_multicast_del_group_src(struct net_bridge_group_src *src)
{ {
struct net_bridge *br = src->pg->key.port->br; struct net_bridge *br = src->pg->key.port->br;
br_multicast_fwd_src_remove(src);
hlist_del_init_rcu(&src->node); hlist_del_init_rcu(&src->node);
src->pg->src_ents--; src->pg->src_ents--;
hlist_add_head(&src->mcast_gc.gc_node, &br->mcast_gc_list); hlist_add_head(&src->mcast_gc.gc_node, &br->mcast_gc_list);
...@@ -851,19 +935,6 @@ struct net_bridge_port_group *br_multicast_new_port_group( ...@@ -851,19 +935,6 @@ struct net_bridge_port_group *br_multicast_new_port_group(
return p; return p;
} }
static bool br_port_group_equal(struct net_bridge_port_group *p,
struct net_bridge_port *port,
const unsigned char *src)
{
if (p->key.port != port)
return false;
if (!(port->flags & BR_MULTICAST_TO_UNICAST))
return true;
return ether_addr_equal(src, p->eth_addr);
}
void br_multicast_host_join(struct net_bridge_mdb_entry *mp, bool notify) void br_multicast_host_join(struct net_bridge_mdb_entry *mp, bool notify)
{ {
if (!mp->host_joined) { if (!mp->host_joined) {
...@@ -884,28 +955,26 @@ void br_multicast_host_leave(struct net_bridge_mdb_entry *mp, bool notify) ...@@ -884,28 +955,26 @@ void br_multicast_host_leave(struct net_bridge_mdb_entry *mp, bool notify)
br_mdb_notify(mp->br->dev, mp, NULL, RTM_DELMDB); br_mdb_notify(mp->br->dev, mp, NULL, RTM_DELMDB);
} }
static int br_multicast_add_group(struct net_bridge *br, static struct net_bridge_port_group *
struct net_bridge_port *port, __br_multicast_add_group(struct net_bridge *br,
struct br_ip *group, struct net_bridge_port *port,
const unsigned char *src, struct br_ip *group,
u8 filter_mode, const unsigned char *src,
bool igmpv2_mldv1) u8 filter_mode,
bool igmpv2_mldv1)
{ {
struct net_bridge_port_group __rcu **pp; struct net_bridge_port_group __rcu **pp;
struct net_bridge_port_group *p; struct net_bridge_port_group *p = NULL;
struct net_bridge_mdb_entry *mp; struct net_bridge_mdb_entry *mp;
unsigned long now = jiffies; unsigned long now = jiffies;
int err;
spin_lock(&br->multicast_lock);
if (!netif_running(br->dev) || if (!netif_running(br->dev) ||
(port && port->state == BR_STATE_DISABLED)) (port && port->state == BR_STATE_DISABLED))
goto out; goto out;
mp = br_multicast_new_group(br, group); mp = br_multicast_new_group(br, group);
err = PTR_ERR(mp);
if (IS_ERR(mp)) if (IS_ERR(mp))
goto err; return ERR_PTR(PTR_ERR(mp));
if (!port) { if (!port) {
br_multicast_host_join(mp, true); br_multicast_host_join(mp, true);
...@@ -923,8 +992,10 @@ static int br_multicast_add_group(struct net_bridge *br, ...@@ -923,8 +992,10 @@ static int br_multicast_add_group(struct net_bridge *br,
p = br_multicast_new_port_group(port, group, *pp, 0, src, filter_mode, p = br_multicast_new_port_group(port, group, *pp, 0, src, filter_mode,
RTPROT_KERNEL); RTPROT_KERNEL);
if (unlikely(!p)) if (unlikely(!p)) {
goto err; p = ERR_PTR(-ENOMEM);
goto out;
}
rcu_assign_pointer(*pp, p); rcu_assign_pointer(*pp, p);
br_mdb_notify(br->dev, mp, p, RTM_NEWMDB); br_mdb_notify(br->dev, mp, p, RTM_NEWMDB);
...@@ -933,10 +1004,26 @@ static int br_multicast_add_group(struct net_bridge *br, ...@@ -933,10 +1004,26 @@ static int br_multicast_add_group(struct net_bridge *br,
mod_timer(&p->timer, now + br->multicast_membership_interval); mod_timer(&p->timer, now + br->multicast_membership_interval);
out: out:
err = 0; return p;
}
static int br_multicast_add_group(struct net_bridge *br,
struct net_bridge_port *port,
struct br_ip *group,
const unsigned char *src,
u8 filter_mode,
bool igmpv2_mldv1)
{
struct net_bridge_port_group *pg;
int err;
err: spin_lock(&br->multicast_lock);
pg = __br_multicast_add_group(br, port, group, src, filter_mode,
igmpv2_mldv1);
/* NULL is considered valid for host joined groups */
err = IS_ERR(pg) ? PTR_ERR(pg) : 0;
spin_unlock(&br->multicast_lock); spin_unlock(&br->multicast_lock);
return err; return err;
} }
...@@ -1349,6 +1436,13 @@ static int __grp_src_delete_marked(struct net_bridge_port_group *pg) ...@@ -1349,6 +1436,13 @@ static int __grp_src_delete_marked(struct net_bridge_port_group *pg)
return deleted; return deleted;
} }
static void __grp_src_mod_timer(struct net_bridge_group_src *src,
unsigned long expires)
{
mod_timer(&src->timer, expires);
br_multicast_fwd_src_handle(src);
}
static void __grp_src_query_marked_and_rexmit(struct net_bridge_port_group *pg) static void __grp_src_query_marked_and_rexmit(struct net_bridge_port_group *pg)
{ {
struct bridge_mcast_other_query *other_query = NULL; struct bridge_mcast_other_query *other_query = NULL;
...@@ -1377,7 +1471,7 @@ static void __grp_src_query_marked_and_rexmit(struct net_bridge_port_group *pg) ...@@ -1377,7 +1471,7 @@ static void __grp_src_query_marked_and_rexmit(struct net_bridge_port_group *pg)
other_query && other_query &&
!timer_pending(&other_query->timer)) !timer_pending(&other_query->timer))
ent->src_query_rexmit_cnt = lmqc; ent->src_query_rexmit_cnt = lmqc;
mod_timer(&ent->timer, lmqt); __grp_src_mod_timer(ent, lmqt);
} }
} }
} }
...@@ -1456,7 +1550,7 @@ static bool br_multicast_isinc_allow(struct net_bridge_port_group *pg, ...@@ -1456,7 +1550,7 @@ static bool br_multicast_isinc_allow(struct net_bridge_port_group *pg,
} }
if (ent) if (ent)
mod_timer(&ent->timer, now + br_multicast_gmi(br)); __grp_src_mod_timer(ent, now + br_multicast_gmi(br));
srcs += src_size; srcs += src_size;
} }
...@@ -1486,7 +1580,9 @@ static void __grp_src_isexc_incl(struct net_bridge_port_group *pg, ...@@ -1486,7 +1580,9 @@ static void __grp_src_isexc_incl(struct net_bridge_port_group *pg,
if (ent) if (ent)
ent->flags &= ~BR_SGRP_F_DELETE; ent->flags &= ~BR_SGRP_F_DELETE;
else else
br_multicast_new_group_src(pg, &src_ip); ent = br_multicast_new_group_src(pg, &src_ip);
if (ent)
br_multicast_fwd_src_handle(ent);
srcs += src_size; srcs += src_size;
} }
...@@ -1522,8 +1618,8 @@ static bool __grp_src_isexc_excl(struct net_bridge_port_group *pg, ...@@ -1522,8 +1618,8 @@ static bool __grp_src_isexc_excl(struct net_bridge_port_group *pg,
} else { } else {
ent = br_multicast_new_group_src(pg, &src_ip); ent = br_multicast_new_group_src(pg, &src_ip);
if (ent) { if (ent) {
mod_timer(&ent->timer, __grp_src_mod_timer(ent,
now + br_multicast_gmi(br)); now + br_multicast_gmi(br));
changed = true; changed = true;
} }
} }
...@@ -1589,7 +1685,7 @@ static bool __grp_src_toin_incl(struct net_bridge_port_group *pg, ...@@ -1589,7 +1685,7 @@ static bool __grp_src_toin_incl(struct net_bridge_port_group *pg,
changed = true; changed = true;
} }
if (ent) if (ent)
mod_timer(&ent->timer, now + br_multicast_gmi(br)); __grp_src_mod_timer(ent, now + br_multicast_gmi(br));
srcs += src_size; srcs += src_size;
} }
...@@ -1634,7 +1730,7 @@ static bool __grp_src_toin_excl(struct net_bridge_port_group *pg, ...@@ -1634,7 +1730,7 @@ static bool __grp_src_toin_excl(struct net_bridge_port_group *pg,
changed = true; changed = true;
} }
if (ent) if (ent)
mod_timer(&ent->timer, now + br_multicast_gmi(br)); __grp_src_mod_timer(ent, now + br_multicast_gmi(br));
srcs += src_size; srcs += src_size;
} }
...@@ -1689,8 +1785,10 @@ static void __grp_src_toex_incl(struct net_bridge_port_group *pg, ...@@ -1689,8 +1785,10 @@ static void __grp_src_toex_incl(struct net_bridge_port_group *pg,
BR_SGRP_F_SEND; BR_SGRP_F_SEND;
to_send++; to_send++;
} else { } else {
br_multicast_new_group_src(pg, &src_ip); ent = br_multicast_new_group_src(pg, &src_ip);
} }
if (ent)
br_multicast_fwd_src_handle(ent);
srcs += src_size; srcs += src_size;
} }
...@@ -1727,7 +1825,7 @@ static bool __grp_src_toex_excl(struct net_bridge_port_group *pg, ...@@ -1727,7 +1825,7 @@ static bool __grp_src_toex_excl(struct net_bridge_port_group *pg,
} else { } else {
ent = br_multicast_new_group_src(pg, &src_ip); ent = br_multicast_new_group_src(pg, &src_ip);
if (ent) { if (ent) {
mod_timer(&ent->timer, pg->timer.expires); __grp_src_mod_timer(ent, pg->timer.expires);
changed = true; changed = true;
} }
} }
...@@ -1823,7 +1921,7 @@ static bool __grp_src_block_excl(struct net_bridge_port_group *pg, ...@@ -1823,7 +1921,7 @@ static bool __grp_src_block_excl(struct net_bridge_port_group *pg,
if (!ent) { if (!ent) {
ent = br_multicast_new_group_src(pg, &src_ip); ent = br_multicast_new_group_src(pg, &src_ip);
if (ent) { if (ent) {
mod_timer(&ent->timer, pg->timer.expires); __grp_src_mod_timer(ent, pg->timer.expires);
changed = true; changed = true;
} }
} }
......
...@@ -218,6 +218,7 @@ struct net_bridge_fdb_entry { ...@@ -218,6 +218,7 @@ struct net_bridge_fdb_entry {
#define BR_SGRP_F_DELETE BIT(0) #define BR_SGRP_F_DELETE BIT(0)
#define BR_SGRP_F_SEND BIT(1) #define BR_SGRP_F_SEND BIT(1)
#define BR_SGRP_F_INSTALLED BIT(2)
struct net_bridge_mcast_gc { struct net_bridge_mcast_gc {
struct hlist_node gc_node; struct hlist_node gc_node;
......
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