mirror of https://gitee.com/openkylin/linux.git
Merge branch 'net-bridge-mcast-initial-IGMPv3-MLDv2-support-part-1'
Nikolay Aleksandrov says: ==================== net: bridge: mcast: initial IGMPv3/MLDv2 support (part 1) This patch-set implements the control plane for initial IGMPv3/MLDv2 support which takes care of include/exclude sets and state transitions based on the different report types. Patch 01 arranges the structure better by moving the frequently used fields together, patch 02 factors out the port group deletion code which is used in a few places. Patches 03 and 04 add support for source lists and group modes per port group which are dumped. Patch 05 adds support for group-and-source specific queries required for IGMPv3/MLDv2. Then patch 06 adds support for group and group-and-source query retransmissions via a new rexmit timer. Patches 07 and 08 make use of the already present mdb fill functions when sending notifications so we can have the full mdb entries' state filled in (with sources, mode etc). Patch 09 takes care of port group expiration, it switches the group mode to include and deletes it if there are no sources with active timers. Patches 10-13 are the core changes which add support for IGMPv3/MLDv2 reports and handle the source list set operations as per RFCs 3376 and 3810, all IGMPv3/MLDv2 report types with their transitions should be supported after these patches. I've used RFCs 3376, 3810 and FRR as a reference implementation. The source lists are capped at 32 entries, we can remove that limitation at a later point which would require a better data structure to hold them. IGMPv3 processing is hidden behind the bridge's multicast_igmp_version option which must be set to 3 in order to enable it. MLDv2 processing is hidden behind the bridge's multicast_mld_version which must be set to 2 in order to enable it. Patch 14 improves other querier processing a bit (more about this below). And finally patch 15 transforms the src gc so it can be used with all mcast objects since now we have multiple timers that can be running and we need to make sure they have all finished before freeing the objects. This is part 1, it only adds control plane support and doesn't change the fast path. A following patch-set will take care of that. Here're the sets that will come next (in order): - Fast path patch-set which adds support for (S, G) mdb entries needed for IGMPv3/MLDv2 forwarding, entry add source (kernel, user-space etc) needed for IGMPv3/MLDv2 entry management, entry block mode needed for IGMPv3/MLDv2 exclude mode. This set will also add iproute2 support for manipulating and showing all the new state. - Selftests patches which will verify all state transitions and forwarding - Explicit host tracking patch-set, needed for proper fast leave and with it fast leave will be enabled for IGMPv3/MLDv2 Not implemented yet: - Host IGMPv3/MLDv2 filter support (currently we handle only join/leave as before) - Proper other querier source timer and value updates - IGMPv3/v2 MLDv2/v1 compat (I have a few rough patches for this one) v4: move old patch 05 to 02 (group del patch), before src lists patch 02: set pg's fast leave flag when deleting due to fast leave patch 03: now can use the new port del function add igmpv2/mldv1 bool which are set when the entry is added in those modes (later will be passed as update_timer) patch 10: rename update_timer to igmpv2_mldv1 and use the passed value from br_multicast_add_group's callers v3: add IPv6/MLDv2 support, most patches are changed v2: patches 03-04: make src lists RCU friendly so they can be traversed when dumping, reduce limit to a more conservative 32 src group entries for a start patches 11-13: remove helper and directly do bitops patch 15: force mcast gc on bridge port del to make sure port group timers have finished before freeing the port ==================== Signed-off-by: Jakub Kicinski <kuba@kernel.org>
This commit is contained in:
commit
6af52ae2ed
|
@ -455,10 +455,31 @@ enum {
|
|||
enum {
|
||||
MDBA_MDB_EATTR_UNSPEC,
|
||||
MDBA_MDB_EATTR_TIMER,
|
||||
MDBA_MDB_EATTR_SRC_LIST,
|
||||
MDBA_MDB_EATTR_GROUP_MODE,
|
||||
__MDBA_MDB_EATTR_MAX
|
||||
};
|
||||
#define MDBA_MDB_EATTR_MAX (__MDBA_MDB_EATTR_MAX - 1)
|
||||
|
||||
/* per mdb entry source */
|
||||
enum {
|
||||
MDBA_MDB_SRCLIST_UNSPEC,
|
||||
MDBA_MDB_SRCLIST_ENTRY,
|
||||
__MDBA_MDB_SRCLIST_MAX
|
||||
};
|
||||
#define MDBA_MDB_SRCLIST_MAX (__MDBA_MDB_SRCLIST_MAX - 1)
|
||||
|
||||
/* per mdb entry per source attributes
|
||||
* these are embedded in MDBA_MDB_SRCLIST_ENTRY
|
||||
*/
|
||||
enum {
|
||||
MDBA_MDB_SRCATTR_UNSPEC,
|
||||
MDBA_MDB_SRCATTR_ADDRESS,
|
||||
MDBA_MDB_SRCATTR_TIMER,
|
||||
__MDBA_MDB_SRCATTR_MAX
|
||||
};
|
||||
#define MDBA_MDB_SRCATTR_MAX (__MDBA_MDB_SRCATTR_MAX - 1)
|
||||
|
||||
/* multicast router types */
|
||||
enum {
|
||||
MDB_RTR_TYPE_DISABLED,
|
||||
|
|
|
@ -77,10 +77,67 @@ static void __mdb_entry_to_br_ip(struct br_mdb_entry *entry, struct br_ip *ip)
|
|||
#endif
|
||||
}
|
||||
|
||||
static int __mdb_fill_srcs(struct sk_buff *skb,
|
||||
struct net_bridge_port_group *p)
|
||||
{
|
||||
struct net_bridge_group_src *ent;
|
||||
struct nlattr *nest, *nest_ent;
|
||||
|
||||
if (hlist_empty(&p->src_list))
|
||||
return 0;
|
||||
|
||||
nest = nla_nest_start(skb, MDBA_MDB_EATTR_SRC_LIST);
|
||||
if (!nest)
|
||||
return -EMSGSIZE;
|
||||
|
||||
hlist_for_each_entry_rcu(ent, &p->src_list, node,
|
||||
lockdep_is_held(&p->port->br->multicast_lock)) {
|
||||
nest_ent = nla_nest_start(skb, MDBA_MDB_SRCLIST_ENTRY);
|
||||
if (!nest_ent)
|
||||
goto out_cancel_err;
|
||||
switch (ent->addr.proto) {
|
||||
case htons(ETH_P_IP):
|
||||
if (nla_put_in_addr(skb, MDBA_MDB_SRCATTR_ADDRESS,
|
||||
ent->addr.u.ip4)) {
|
||||
nla_nest_cancel(skb, nest_ent);
|
||||
goto out_cancel_err;
|
||||
}
|
||||
break;
|
||||
#if IS_ENABLED(CONFIG_IPV6)
|
||||
case htons(ETH_P_IPV6):
|
||||
if (nla_put_in6_addr(skb, MDBA_MDB_SRCATTR_ADDRESS,
|
||||
&ent->addr.u.ip6)) {
|
||||
nla_nest_cancel(skb, nest_ent);
|
||||
goto out_cancel_err;
|
||||
}
|
||||
break;
|
||||
#endif
|
||||
default:
|
||||
nla_nest_cancel(skb, nest_ent);
|
||||
continue;
|
||||
}
|
||||
if (nla_put_u32(skb, MDBA_MDB_SRCATTR_TIMER,
|
||||
br_timer_value(&ent->timer))) {
|
||||
nla_nest_cancel(skb, nest_ent);
|
||||
goto out_cancel_err;
|
||||
}
|
||||
nla_nest_end(skb, nest_ent);
|
||||
}
|
||||
|
||||
nla_nest_end(skb, nest);
|
||||
|
||||
return 0;
|
||||
|
||||
out_cancel_err:
|
||||
nla_nest_cancel(skb, nest);
|
||||
return -EMSGSIZE;
|
||||
}
|
||||
|
||||
static int __mdb_fill_info(struct sk_buff *skb,
|
||||
struct net_bridge_mdb_entry *mp,
|
||||
struct net_bridge_port_group *p)
|
||||
{
|
||||
bool dump_srcs_mode = false;
|
||||
struct timer_list *mtimer;
|
||||
struct nlattr *nest_ent;
|
||||
struct br_mdb_entry e;
|
||||
|
@ -119,6 +176,23 @@ static int __mdb_fill_info(struct sk_buff *skb,
|
|||
nla_nest_cancel(skb, nest_ent);
|
||||
return -EMSGSIZE;
|
||||
}
|
||||
switch (mp->addr.proto) {
|
||||
case htons(ETH_P_IP):
|
||||
dump_srcs_mode = !!(p && mp->br->multicast_igmp_version == 3);
|
||||
break;
|
||||
#if IS_ENABLED(CONFIG_IPV6)
|
||||
case htons(ETH_P_IPV6):
|
||||
dump_srcs_mode = !!(p && mp->br->multicast_mld_version == 2);
|
||||
break;
|
||||
#endif
|
||||
}
|
||||
if (dump_srcs_mode &&
|
||||
(__mdb_fill_srcs(skb, p) ||
|
||||
nla_put_u8(skb, MDBA_MDB_EATTR_GROUP_MODE, p->filter_mode))) {
|
||||
nla_nest_cancel(skb, nest_ent);
|
||||
return -EMSGSIZE;
|
||||
}
|
||||
|
||||
nla_nest_end(skb, nest_ent);
|
||||
|
||||
return 0;
|
||||
|
@ -127,7 +201,7 @@ static int __mdb_fill_info(struct sk_buff *skb,
|
|||
static int br_mdb_fill_info(struct sk_buff *skb, struct netlink_callback *cb,
|
||||
struct net_device *dev)
|
||||
{
|
||||
int idx = 0, s_idx = cb->args[1], err = 0;
|
||||
int idx = 0, s_idx = cb->args[1], err = 0, pidx = 0, s_pidx = cb->args[2];
|
||||
struct net_bridge *br = netdev_priv(dev);
|
||||
struct net_bridge_mdb_entry *mp;
|
||||
struct nlattr *nest, *nest2;
|
||||
|
@ -152,7 +226,7 @@ static int br_mdb_fill_info(struct sk_buff *skb, struct netlink_callback *cb,
|
|||
break;
|
||||
}
|
||||
|
||||
if (mp->host_joined) {
|
||||
if (!s_pidx && mp->host_joined) {
|
||||
err = __mdb_fill_info(skb, mp, NULL);
|
||||
if (err) {
|
||||
nla_nest_cancel(skb, nest2);
|
||||
|
@ -164,13 +238,19 @@ static int br_mdb_fill_info(struct sk_buff *skb, struct netlink_callback *cb,
|
|||
pp = &p->next) {
|
||||
if (!p->port)
|
||||
continue;
|
||||
if (pidx < s_pidx)
|
||||
goto skip_pg;
|
||||
|
||||
err = __mdb_fill_info(skb, mp, p);
|
||||
if (err) {
|
||||
nla_nest_cancel(skb, nest2);
|
||||
goto out;
|
||||
}
|
||||
skip_pg:
|
||||
pidx++;
|
||||
}
|
||||
pidx = 0;
|
||||
s_pidx = 0;
|
||||
nla_nest_end(skb, nest2);
|
||||
skip:
|
||||
idx++;
|
||||
|
@ -178,6 +258,7 @@ static int br_mdb_fill_info(struct sk_buff *skb, struct netlink_callback *cb,
|
|||
|
||||
out:
|
||||
cb->args[1] = idx;
|
||||
cb->args[2] = pidx;
|
||||
nla_nest_end(skb, nest);
|
||||
return err;
|
||||
}
|
||||
|
@ -263,14 +344,15 @@ static int br_mdb_dump(struct sk_buff *skb, struct netlink_callback *cb)
|
|||
|
||||
static int nlmsg_populate_mdb_fill(struct sk_buff *skb,
|
||||
struct net_device *dev,
|
||||
struct br_mdb_entry *entry, u32 pid,
|
||||
u32 seq, int type, unsigned int flags)
|
||||
struct net_bridge_mdb_entry *mp,
|
||||
struct net_bridge_port_group *pg,
|
||||
int type)
|
||||
{
|
||||
struct nlmsghdr *nlh;
|
||||
struct br_port_msg *bpm;
|
||||
struct nlattr *nest, *nest2;
|
||||
|
||||
nlh = nlmsg_put(skb, pid, seq, type, sizeof(*bpm), 0);
|
||||
nlh = nlmsg_put(skb, 0, 0, type, sizeof(*bpm), 0);
|
||||
if (!nlh)
|
||||
return -EMSGSIZE;
|
||||
|
||||
|
@ -285,7 +367,7 @@ static int nlmsg_populate_mdb_fill(struct sk_buff *skb,
|
|||
if (nest2 == NULL)
|
||||
goto end;
|
||||
|
||||
if (nla_put(skb, MDBA_MDB_ENTRY_INFO, sizeof(*entry), entry))
|
||||
if (__mdb_fill_info(skb, mp, pg))
|
||||
goto end;
|
||||
|
||||
nla_nest_end(skb, nest2);
|
||||
|
@ -300,10 +382,49 @@ static int nlmsg_populate_mdb_fill(struct sk_buff *skb,
|
|||
return -EMSGSIZE;
|
||||
}
|
||||
|
||||
static inline size_t rtnl_mdb_nlmsg_size(void)
|
||||
static size_t rtnl_mdb_nlmsg_size(struct net_bridge_port_group *pg)
|
||||
{
|
||||
return NLMSG_ALIGN(sizeof(struct br_port_msg))
|
||||
+ nla_total_size(sizeof(struct br_mdb_entry));
|
||||
size_t nlmsg_size = NLMSG_ALIGN(sizeof(struct br_port_msg)) +
|
||||
nla_total_size(sizeof(struct br_mdb_entry)) +
|
||||
nla_total_size(sizeof(u32));
|
||||
struct net_bridge_group_src *ent;
|
||||
size_t addr_size = 0;
|
||||
|
||||
if (!pg)
|
||||
goto out;
|
||||
|
||||
switch (pg->addr.proto) {
|
||||
case htons(ETH_P_IP):
|
||||
if (pg->port->br->multicast_igmp_version == 2)
|
||||
goto out;
|
||||
addr_size = sizeof(__be32);
|
||||
break;
|
||||
#if IS_ENABLED(CONFIG_IPV6)
|
||||
case htons(ETH_P_IPV6):
|
||||
if (pg->port->br->multicast_mld_version == 1)
|
||||
goto out;
|
||||
addr_size = sizeof(struct in6_addr);
|
||||
break;
|
||||
#endif
|
||||
}
|
||||
|
||||
/* MDBA_MDB_EATTR_GROUP_MODE */
|
||||
nlmsg_size += nla_total_size(sizeof(u8));
|
||||
|
||||
/* MDBA_MDB_EATTR_SRC_LIST nested attr */
|
||||
if (!hlist_empty(&pg->src_list))
|
||||
nlmsg_size += nla_total_size(0);
|
||||
|
||||
hlist_for_each_entry(ent, &pg->src_list, node) {
|
||||
/* MDBA_MDB_SRCLIST_ENTRY nested attr +
|
||||
* MDBA_MDB_SRCATTR_ADDRESS + MDBA_MDB_SRCATTR_TIMER
|
||||
*/
|
||||
nlmsg_size += nla_total_size(0) +
|
||||
nla_total_size(addr_size) +
|
||||
nla_total_size(sizeof(u32));
|
||||
}
|
||||
out:
|
||||
return nlmsg_size;
|
||||
}
|
||||
|
||||
struct br_mdb_complete_info {
|
||||
|
@ -341,21 +462,22 @@ static void br_mdb_complete(struct net_device *dev, int err, void *priv)
|
|||
|
||||
static void br_mdb_switchdev_host_port(struct net_device *dev,
|
||||
struct net_device *lower_dev,
|
||||
struct br_mdb_entry *entry, int type)
|
||||
struct net_bridge_mdb_entry *mp,
|
||||
int type)
|
||||
{
|
||||
struct switchdev_obj_port_mdb mdb = {
|
||||
.obj = {
|
||||
.id = SWITCHDEV_OBJ_ID_HOST_MDB,
|
||||
.flags = SWITCHDEV_F_DEFER,
|
||||
},
|
||||
.vid = entry->vid,
|
||||
.vid = mp->addr.vid,
|
||||
};
|
||||
|
||||
if (entry->addr.proto == htons(ETH_P_IP))
|
||||
ip_eth_mc_map(entry->addr.u.ip4, mdb.addr);
|
||||
if (mp->addr.proto == htons(ETH_P_IP))
|
||||
ip_eth_mc_map(mp->addr.u.ip4, mdb.addr);
|
||||
#if IS_ENABLED(CONFIG_IPV6)
|
||||
else
|
||||
ipv6_eth_mc_map(&entry->addr.u.ip6, mdb.addr);
|
||||
ipv6_eth_mc_map(&mp->addr.u.ip6, mdb.addr);
|
||||
#endif
|
||||
|
||||
mdb.obj.orig_dev = dev;
|
||||
|
@ -370,17 +492,19 @@ static void br_mdb_switchdev_host_port(struct net_device *dev,
|
|||
}
|
||||
|
||||
static void br_mdb_switchdev_host(struct net_device *dev,
|
||||
struct br_mdb_entry *entry, int type)
|
||||
struct net_bridge_mdb_entry *mp, int type)
|
||||
{
|
||||
struct net_device *lower_dev;
|
||||
struct list_head *iter;
|
||||
|
||||
netdev_for_each_lower_dev(dev, lower_dev, iter)
|
||||
br_mdb_switchdev_host_port(dev, lower_dev, entry, type);
|
||||
br_mdb_switchdev_host_port(dev, lower_dev, mp, type);
|
||||
}
|
||||
|
||||
static void __br_mdb_notify(struct net_device *dev, struct net_bridge_port *p,
|
||||
struct br_mdb_entry *entry, int type)
|
||||
void br_mdb_notify(struct net_device *dev,
|
||||
struct net_bridge_mdb_entry *mp,
|
||||
struct net_bridge_port_group *pg,
|
||||
int type)
|
||||
{
|
||||
struct br_mdb_complete_info *complete_info;
|
||||
struct switchdev_obj_port_mdb mdb = {
|
||||
|
@ -388,44 +512,45 @@ static void __br_mdb_notify(struct net_device *dev, struct net_bridge_port *p,
|
|||
.id = SWITCHDEV_OBJ_ID_PORT_MDB,
|
||||
.flags = SWITCHDEV_F_DEFER,
|
||||
},
|
||||
.vid = entry->vid,
|
||||
.vid = mp->addr.vid,
|
||||
};
|
||||
struct net_device *port_dev;
|
||||
struct net *net = dev_net(dev);
|
||||
struct sk_buff *skb;
|
||||
int err = -ENOBUFS;
|
||||
|
||||
port_dev = __dev_get_by_index(net, entry->ifindex);
|
||||
if (entry->addr.proto == htons(ETH_P_IP))
|
||||
ip_eth_mc_map(entry->addr.u.ip4, mdb.addr);
|
||||
if (pg) {
|
||||
if (mp->addr.proto == htons(ETH_P_IP))
|
||||
ip_eth_mc_map(mp->addr.u.ip4, mdb.addr);
|
||||
#if IS_ENABLED(CONFIG_IPV6)
|
||||
else
|
||||
ipv6_eth_mc_map(&entry->addr.u.ip6, mdb.addr);
|
||||
else
|
||||
ipv6_eth_mc_map(&mp->addr.u.ip6, mdb.addr);
|
||||
#endif
|
||||
|
||||
mdb.obj.orig_dev = port_dev;
|
||||
if (p && port_dev && type == RTM_NEWMDB) {
|
||||
complete_info = kmalloc(sizeof(*complete_info), GFP_ATOMIC);
|
||||
if (complete_info) {
|
||||
complete_info->port = p;
|
||||
__mdb_entry_to_br_ip(entry, &complete_info->ip);
|
||||
mdb.obj.orig_dev = pg->port->dev;
|
||||
switch (type) {
|
||||
case RTM_NEWMDB:
|
||||
complete_info = kmalloc(sizeof(*complete_info), GFP_ATOMIC);
|
||||
if (!complete_info)
|
||||
break;
|
||||
complete_info->port = pg->port;
|
||||
complete_info->ip = mp->addr;
|
||||
mdb.obj.complete_priv = complete_info;
|
||||
mdb.obj.complete = br_mdb_complete;
|
||||
if (switchdev_port_obj_add(port_dev, &mdb.obj, NULL))
|
||||
if (switchdev_port_obj_add(pg->port->dev, &mdb.obj, NULL))
|
||||
kfree(complete_info);
|
||||
break;
|
||||
case RTM_DELMDB:
|
||||
switchdev_port_obj_del(pg->port->dev, &mdb.obj);
|
||||
break;
|
||||
}
|
||||
} else if (p && port_dev && type == RTM_DELMDB) {
|
||||
switchdev_port_obj_del(port_dev, &mdb.obj);
|
||||
} else {
|
||||
br_mdb_switchdev_host(dev, mp, type);
|
||||
}
|
||||
|
||||
if (!p)
|
||||
br_mdb_switchdev_host(dev, entry, type);
|
||||
|
||||
skb = nlmsg_new(rtnl_mdb_nlmsg_size(), GFP_ATOMIC);
|
||||
skb = nlmsg_new(rtnl_mdb_nlmsg_size(pg), GFP_ATOMIC);
|
||||
if (!skb)
|
||||
goto errout;
|
||||
|
||||
err = nlmsg_populate_mdb_fill(skb, dev, entry, 0, 0, type, NTF_SELF);
|
||||
err = nlmsg_populate_mdb_fill(skb, dev, mp, pg, type);
|
||||
if (err < 0) {
|
||||
kfree_skb(skb);
|
||||
goto errout;
|
||||
|
@ -437,26 +562,6 @@ static void __br_mdb_notify(struct net_device *dev, struct net_bridge_port *p,
|
|||
rtnl_set_sk_err(net, RTNLGRP_MDB, err);
|
||||
}
|
||||
|
||||
void br_mdb_notify(struct net_device *dev, struct net_bridge_port *port,
|
||||
struct br_ip *group, int type, u8 flags)
|
||||
{
|
||||
struct br_mdb_entry entry;
|
||||
|
||||
memset(&entry, 0, sizeof(entry));
|
||||
if (port)
|
||||
entry.ifindex = port->dev->ifindex;
|
||||
else
|
||||
entry.ifindex = dev->ifindex;
|
||||
entry.addr.proto = group->proto;
|
||||
entry.addr.u.ip4 = group->u.ip4;
|
||||
#if IS_ENABLED(CONFIG_IPV6)
|
||||
entry.addr.u.ip6 = group->u.ip6;
|
||||
#endif
|
||||
entry.vid = group->vid;
|
||||
__mdb_entry_fill_flags(&entry, flags);
|
||||
__br_mdb_notify(dev, port, &entry, type);
|
||||
}
|
||||
|
||||
static int nlmsg_populate_rtr_fill(struct sk_buff *skb,
|
||||
struct net_device *dev,
|
||||
int ifindex, u32 pid,
|
||||
|
@ -600,7 +705,7 @@ static int br_mdb_parse(struct sk_buff *skb, struct nlmsghdr *nlh,
|
|||
}
|
||||
|
||||
static int br_mdb_add_group(struct net_bridge *br, struct net_bridge_port *port,
|
||||
struct br_ip *group, unsigned char state)
|
||||
struct br_ip *group, struct br_mdb_entry *entry)
|
||||
{
|
||||
struct net_bridge_mdb_entry *mp;
|
||||
struct net_bridge_port_group *p;
|
||||
|
@ -619,12 +724,13 @@ static int br_mdb_add_group(struct net_bridge *br, struct net_bridge_port *port,
|
|||
/* host join */
|
||||
if (!port) {
|
||||
/* don't allow any flags for host-joined groups */
|
||||
if (state)
|
||||
if (entry->state)
|
||||
return -EINVAL;
|
||||
if (mp->host_joined)
|
||||
return -EEXIST;
|
||||
|
||||
br_multicast_host_join(mp, false);
|
||||
br_mdb_notify(br->dev, mp, NULL, RTM_NEWMDB);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
@ -638,12 +744,14 @@ static int br_mdb_add_group(struct net_bridge *br, struct net_bridge_port *port,
|
|||
break;
|
||||
}
|
||||
|
||||
p = br_multicast_new_port_group(port, group, *pp, state, NULL);
|
||||
p = br_multicast_new_port_group(port, group, *pp, entry->state, NULL,
|
||||
MCAST_EXCLUDE);
|
||||
if (unlikely(!p))
|
||||
return -ENOMEM;
|
||||
rcu_assign_pointer(*pp, p);
|
||||
if (state == MDB_TEMPORARY)
|
||||
if (entry->state == MDB_TEMPORARY)
|
||||
mod_timer(&p->timer, now + br->multicast_membership_interval);
|
||||
br_mdb_notify(br->dev, mp, p, RTM_NEWMDB);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
@ -672,7 +780,7 @@ static int __br_mdb_add(struct net *net, struct net_bridge *br,
|
|||
__mdb_entry_to_br_ip(entry, &ip);
|
||||
|
||||
spin_lock_bh(&br->multicast_lock);
|
||||
ret = br_mdb_add_group(br, p, &ip, entry->state);
|
||||
ret = br_mdb_add_group(br, p, &ip, entry);
|
||||
spin_unlock_bh(&br->multicast_lock);
|
||||
return ret;
|
||||
}
|
||||
|
@ -717,12 +825,9 @@ static int br_mdb_add(struct sk_buff *skb, struct nlmsghdr *nlh,
|
|||
err = __br_mdb_add(net, br, entry);
|
||||
if (err)
|
||||
break;
|
||||
__br_mdb_notify(dev, p, entry, RTM_NEWMDB);
|
||||
}
|
||||
} else {
|
||||
err = __br_mdb_add(net, br, entry);
|
||||
if (!err)
|
||||
__br_mdb_notify(dev, p, entry, RTM_NEWMDB);
|
||||
}
|
||||
|
||||
return err;
|
||||
|
@ -750,6 +855,7 @@ static int __br_mdb_del(struct net_bridge *br, struct br_mdb_entry *entry)
|
|||
if (entry->ifindex == mp->br->dev->ifindex && mp->host_joined) {
|
||||
br_multicast_host_leave(mp, false);
|
||||
err = 0;
|
||||
br_mdb_notify(br->dev, mp, NULL, RTM_DELMDB);
|
||||
if (!mp->ports && netif_running(br->dev))
|
||||
mod_timer(&mp->timer, jiffies);
|
||||
goto unlock;
|
||||
|
@ -764,16 +870,8 @@ static int __br_mdb_del(struct net_bridge *br, struct br_mdb_entry *entry)
|
|||
if (p->port->state == BR_STATE_DISABLED)
|
||||
goto unlock;
|
||||
|
||||
__mdb_entry_fill_flags(entry, p->flags);
|
||||
rcu_assign_pointer(*pp, p->next);
|
||||
hlist_del_init(&p->mglist);
|
||||
del_timer(&p->timer);
|
||||
kfree_rcu(p, rcu);
|
||||
br_multicast_del_pg(mp, p, pp);
|
||||
err = 0;
|
||||
|
||||
if (!mp->ports && !mp->host_joined &&
|
||||
netif_running(br->dev))
|
||||
mod_timer(&mp->timer, jiffies);
|
||||
break;
|
||||
}
|
||||
|
||||
|
@ -820,13 +918,9 @@ static int br_mdb_del(struct sk_buff *skb, struct nlmsghdr *nlh,
|
|||
list_for_each_entry(v, &vg->vlan_list, vlist) {
|
||||
entry->vid = v->vid;
|
||||
err = __br_mdb_del(br, entry);
|
||||
if (!err)
|
||||
__br_mdb_notify(dev, p, entry, RTM_DELMDB);
|
||||
}
|
||||
} else {
|
||||
err = __br_mdb_del(br, entry);
|
||||
if (!err)
|
||||
__br_mdb_notify(dev, p, entry, RTM_DELMDB);
|
||||
}
|
||||
|
||||
return err;
|
||||
|
|
File diff suppressed because it is too large
Load Diff
|
@ -214,26 +214,61 @@ struct net_bridge_fdb_entry {
|
|||
#define MDB_PG_FLAGS_OFFLOAD BIT(1)
|
||||
#define MDB_PG_FLAGS_FAST_LEAVE BIT(2)
|
||||
|
||||
#define PG_SRC_ENT_LIMIT 32
|
||||
|
||||
#define BR_SGRP_F_DELETE BIT(0)
|
||||
#define BR_SGRP_F_SEND BIT(1)
|
||||
|
||||
struct net_bridge_mcast_gc {
|
||||
struct hlist_node gc_node;
|
||||
void (*destroy)(struct net_bridge_mcast_gc *gc);
|
||||
};
|
||||
|
||||
struct net_bridge_group_src {
|
||||
struct hlist_node node;
|
||||
|
||||
struct br_ip addr;
|
||||
struct net_bridge_port_group *pg;
|
||||
u8 flags;
|
||||
u8 src_query_rexmit_cnt;
|
||||
struct timer_list timer;
|
||||
|
||||
struct net_bridge *br;
|
||||
struct net_bridge_mcast_gc mcast_gc;
|
||||
struct rcu_head rcu;
|
||||
};
|
||||
|
||||
struct net_bridge_port_group {
|
||||
struct net_bridge_port *port;
|
||||
struct net_bridge_port_group __rcu *next;
|
||||
struct hlist_node mglist;
|
||||
struct rcu_head rcu;
|
||||
struct timer_list timer;
|
||||
struct br_ip addr;
|
||||
unsigned char eth_addr[ETH_ALEN] __aligned(2);
|
||||
unsigned char flags;
|
||||
unsigned char filter_mode;
|
||||
unsigned char grp_query_rexmit_cnt;
|
||||
|
||||
struct hlist_head src_list;
|
||||
unsigned int src_ents;
|
||||
struct timer_list timer;
|
||||
struct timer_list rexmit_timer;
|
||||
struct hlist_node mglist;
|
||||
|
||||
struct net_bridge_mcast_gc mcast_gc;
|
||||
struct rcu_head rcu;
|
||||
};
|
||||
|
||||
struct net_bridge_mdb_entry {
|
||||
struct rhash_head rhnode;
|
||||
struct net_bridge *br;
|
||||
struct net_bridge_port_group __rcu *ports;
|
||||
struct rcu_head rcu;
|
||||
struct timer_list timer;
|
||||
struct br_ip addr;
|
||||
bool host_joined;
|
||||
|
||||
struct timer_list timer;
|
||||
struct hlist_node mdb_node;
|
||||
|
||||
struct net_bridge_mcast_gc mcast_gc;
|
||||
struct rcu_head rcu;
|
||||
};
|
||||
|
||||
struct net_bridge_port {
|
||||
|
@ -406,6 +441,7 @@ struct net_bridge {
|
|||
|
||||
struct rhashtable mdb_hash_tbl;
|
||||
|
||||
struct hlist_head mcast_gc_list;
|
||||
struct hlist_head mdb_list;
|
||||
struct hlist_head router_list;
|
||||
|
||||
|
@ -419,6 +455,7 @@ struct net_bridge {
|
|||
struct bridge_mcast_own_query ip6_own_query;
|
||||
struct bridge_mcast_querier ip6_querier;
|
||||
#endif /* IS_ENABLED(CONFIG_IPV6) */
|
||||
struct work_struct mcast_gc_work;
|
||||
#endif
|
||||
|
||||
struct timer_list hello_timer;
|
||||
|
@ -766,13 +803,17 @@ br_multicast_new_group(struct net_bridge *br, struct br_ip *group);
|
|||
struct net_bridge_port_group *
|
||||
br_multicast_new_port_group(struct net_bridge_port *port, struct br_ip *group,
|
||||
struct net_bridge_port_group __rcu *next,
|
||||
unsigned char flags, const unsigned char *src);
|
||||
unsigned char flags, const unsigned char *src,
|
||||
u8 filter_mode);
|
||||
int br_mdb_hash_init(struct net_bridge *br);
|
||||
void br_mdb_hash_fini(struct net_bridge *br);
|
||||
void br_mdb_notify(struct net_device *dev, struct net_bridge_port *port,
|
||||
struct br_ip *group, int type, u8 flags);
|
||||
void br_mdb_notify(struct net_device *dev, struct net_bridge_mdb_entry *mp,
|
||||
struct net_bridge_port_group *pg, int type);
|
||||
void br_rtr_notify(struct net_device *dev, struct net_bridge_port *port,
|
||||
int type);
|
||||
void br_multicast_del_pg(struct net_bridge_mdb_entry *mp,
|
||||
struct net_bridge_port_group *pg,
|
||||
struct net_bridge_port_group __rcu **pp);
|
||||
void br_multicast_count(struct net_bridge *br, const struct net_bridge_port *p,
|
||||
const struct sk_buff *skb, u8 type, u8 dir);
|
||||
int br_multicast_init_stats(struct net_bridge *br);
|
||||
|
@ -836,6 +877,19 @@ static inline int br_multicast_igmp_type(const struct sk_buff *skb)
|
|||
{
|
||||
return BR_INPUT_SKB_CB(skb)->igmp;
|
||||
}
|
||||
|
||||
static inline unsigned long br_multicast_lmqt(const struct net_bridge *br)
|
||||
{
|
||||
return br->multicast_last_member_interval *
|
||||
br->multicast_last_member_count;
|
||||
}
|
||||
|
||||
static inline unsigned long br_multicast_gmi(const struct net_bridge *br)
|
||||
{
|
||||
/* use the RFC default of 2 for QRV */
|
||||
return 2 * br->multicast_query_interval +
|
||||
br->multicast_query_response_interval;
|
||||
}
|
||||
#else
|
||||
static inline int br_multicast_rcv(struct net_bridge *br,
|
||||
struct net_bridge_port *port,
|
||||
|
|
Loading…
Reference in New Issue