Multicast-ideas-updated » 0001-bridge-Add-export-of-multicast-database-adjacent-to-.patch
| include/linux/if_bridge.h | ||
|---|---|---|
|
#include <linux/netdevice.h>
|
||
|
#include <uapi/linux/if_bridge.h>
|
||
|
struct br_ip {
|
||
|
union {
|
||
|
__be32 ip4;
|
||
|
#if IS_ENABLED(CONFIG_IPV6)
|
||
|
struct in6_addr ip6;
|
||
|
#endif
|
||
|
} u;
|
||
|
__be16 proto;
|
||
|
};
|
||
|
struct br_ip_list {
|
||
|
struct list_head list;
|
||
|
struct br_ip addr;
|
||
|
};
|
||
|
extern void brioctl_set(int (*ioctl_hook)(struct net *, unsigned int, void __user *));
|
||
|
typedef int br_should_route_hook_t(struct sk_buff *skb);
|
||
|
extern br_should_route_hook_t __rcu *br_should_route_hook;
|
||
|
extern int br_multicast_list_adjacent(struct net_device *net_dev,
|
||
|
struct list_head *br_ip_list);
|
||
|
#endif
|
||
| net/bridge/br_multicast.c | ||
|---|---|---|
|
*/
|
||
|
#include <linux/err.h>
|
||
|
#include <linux/export.h>
|
||
|
#include <linux/if_ether.h>
|
||
|
#include <linux/igmp.h>
|
||
|
#include <linux/jhash.h>
|
||
| ... | ... | |
|
return err;
|
||
|
}
|
||
|
/**
|
||
|
* br_multicast_list_adjacent - Returns snooped multicast addresses
|
||
|
* @dev: The bridge port adjacent to which to retrieve addresses
|
||
|
* @br_ip_list: The list to store found, snooped multicast IP addresses in
|
||
|
*
|
||
|
* Creates a list of IP addresses (struct br_ip_list) sensed by the multicast
|
||
|
* snooping feature on all bridge ports of dev's bridge device, excluding
|
||
|
* the addresses from dev itself.
|
||
|
*
|
||
|
* Returns the number of items added to br_ip_list.
|
||
|
*
|
||
|
* Notes:
|
||
|
* - br_ip_list needs to be initialized by caller
|
||
|
* - br_ip_list might contain duplicates in the end
|
||
|
* (needs to be taken care of by caller)
|
||
|
* - br_ip_list needs to be freed by caller
|
||
|
*/
|
||
|
int br_multicast_list_adjacent(struct net_device *dev,
|
||
|
struct list_head *br_ip_list)
|
||
|
{
|
||
|
struct net_bridge *br;
|
||
|
struct net_bridge_port *port;
|
||
|
struct net_bridge_port_group *group;
|
||
|
struct br_ip_list *entry;
|
||
|
struct hlist_node *p;
|
||
|
int count = 0;
|
||
|
rcu_read_lock();
|
||
|
if (!br_ip_list || !br_port_exists(dev))
|
||
|
goto unlock;
|
||
|
port = br_port_get_rcu(dev);
|
||
|
if (!port || !port->br)
|
||
|
goto unlock;
|
||
|
br = port->br;
|
||
|
list_for_each_entry_rcu(port, &br->port_list, list) {
|
||
|
if (!port->dev || port->dev == dev)
|
||
|
continue;
|
||
|
hlist_for_each_entry_rcu(group, p, &port->mglist, mglist) {
|
||
|
entry = kmalloc(sizeof(struct br_ip_list), GFP_ATOMIC);
|
||
|
if (!entry)
|
||
|
goto unlock;
|
||
|
entry->addr = group->addr;
|
||
|
list_add(&entry->list, br_ip_list);
|
||
|
count++;
|
||
|
}
|
||
|
}
|
||
|
unlock:
|
||
|
rcu_read_unlock();
|
||
|
return count;
|
||
|
}
|
||
|
EXPORT_SYMBOL(br_multicast_list_adjacent);
|
||
| net/bridge/br_private.h | ||
|---|---|---|
|
unsigned char addr[6];
|
||
|
};
|
||
|
struct br_ip
|
||
|
{
|
||
|
union {
|
||
|
__be32 ip4;
|
||
|
#if IS_ENABLED(CONFIG_IPV6)
|
||
|
struct in6_addr ip6;
|
||
|
#endif
|
||
|
} u;
|
||
|
__be16 proto;
|
||
|
};
|
||
|
struct net_bridge_fdb_entry
|
||
|
{
|
||
|
struct hlist_node hlist;
|
||