net: bridge: add bitfield for options and convert vlan opts
Bridge options have usually been added as separate fields all over the net_bridge struct taking up space and ending up in different cache lines. Let's move them to a single bitfield to save up space and speedup lookups. This patch adds a simple API for option modifying and retrieving using bitops and converts the first user of the API - the bridge vlan options (vlan_enabled and vlan_stats_enabled). Signed-off-by: Nikolay Aleksandrov <nikolay@cumulusnetworks.com> Reviewed-by: Stephen Hemminger <stephen@networkplumber.org> Signed-off-by: David S. Miller <davem@davemloft.net>
This commit is contained in:
parent
1c1cb6d032
commit
ae75767ec2
|
@ -175,6 +175,22 @@ static struct notifier_block br_switchdev_notifier = {
|
|||
.notifier_call = br_switchdev_event,
|
||||
};
|
||||
|
||||
void br_opt_toggle(struct net_bridge *br, enum net_bridge_opts opt, bool on)
|
||||
{
|
||||
bool cur = !!br_opt_get(br, opt);
|
||||
|
||||
br_debug(br, "toggle option: %d state: %d -> %d\n",
|
||||
opt, cur, on);
|
||||
|
||||
if (cur == on)
|
||||
return;
|
||||
|
||||
if (on)
|
||||
set_bit(opt, &br->options);
|
||||
else
|
||||
clear_bit(opt, &br->options);
|
||||
}
|
||||
|
||||
static void __net_exit br_net_exit(struct net *net)
|
||||
{
|
||||
struct net_device *dev;
|
||||
|
|
|
@ -1416,7 +1416,8 @@ static int br_fill_info(struct sk_buff *skb, const struct net_device *brdev)
|
|||
#ifdef CONFIG_BRIDGE_VLAN_FILTERING
|
||||
if (nla_put_be16(skb, IFLA_BR_VLAN_PROTOCOL, br->vlan_proto) ||
|
||||
nla_put_u16(skb, IFLA_BR_VLAN_DEFAULT_PVID, br->default_pvid) ||
|
||||
nla_put_u8(skb, IFLA_BR_VLAN_STATS_ENABLED, br->vlan_stats_enabled))
|
||||
nla_put_u8(skb, IFLA_BR_VLAN_STATS_ENABLED,
|
||||
br_opt_get(br, BROPT_VLAN_STATS_ENABLED)))
|
||||
return -EMSGSIZE;
|
||||
#endif
|
||||
#ifdef CONFIG_BRIDGE_IGMP_SNOOPING
|
||||
|
|
|
@ -306,16 +306,20 @@ static inline struct net_bridge_port *br_port_get_rtnl_rcu(const struct net_devi
|
|||
rcu_dereference_rtnl(dev->rx_handler_data) : NULL;
|
||||
}
|
||||
|
||||
enum net_bridge_opts {
|
||||
BROPT_VLAN_ENABLED,
|
||||
BROPT_VLAN_STATS_ENABLED,
|
||||
};
|
||||
|
||||
struct net_bridge {
|
||||
spinlock_t lock;
|
||||
spinlock_t hash_lock;
|
||||
struct list_head port_list;
|
||||
struct net_device *dev;
|
||||
struct pcpu_sw_netstats __percpu *stats;
|
||||
unsigned long options;
|
||||
/* These fields are accessed on each packet */
|
||||
#ifdef CONFIG_BRIDGE_VLAN_FILTERING
|
||||
u8 vlan_enabled;
|
||||
u8 vlan_stats_enabled;
|
||||
__be16 vlan_proto;
|
||||
u16 default_pvid;
|
||||
struct net_bridge_vlan_group __rcu *vlgrp;
|
||||
|
@ -489,6 +493,14 @@ static inline bool br_vlan_should_use(const struct net_bridge_vlan *v)
|
|||
return true;
|
||||
}
|
||||
|
||||
static inline int br_opt_get(const struct net_bridge *br,
|
||||
enum net_bridge_opts opt)
|
||||
{
|
||||
return test_bit(opt, &br->options);
|
||||
}
|
||||
|
||||
void br_opt_toggle(struct net_bridge *br, enum net_bridge_opts opt, bool on);
|
||||
|
||||
/* br_device.c */
|
||||
void br_dev_setup(struct net_device *dev);
|
||||
void br_dev_delete(struct net_device *dev, struct list_head *list);
|
||||
|
|
|
@ -743,7 +743,7 @@ static ssize_t vlan_filtering_show(struct device *d,
|
|||
char *buf)
|
||||
{
|
||||
struct net_bridge *br = to_bridge(d);
|
||||
return sprintf(buf, "%d\n", br->vlan_enabled);
|
||||
return sprintf(buf, "%d\n", br_opt_get(br, BROPT_VLAN_ENABLED));
|
||||
}
|
||||
|
||||
static ssize_t vlan_filtering_store(struct device *d,
|
||||
|
@ -791,7 +791,7 @@ static ssize_t vlan_stats_enabled_show(struct device *d,
|
|||
char *buf)
|
||||
{
|
||||
struct net_bridge *br = to_bridge(d);
|
||||
return sprintf(buf, "%u\n", br->vlan_stats_enabled);
|
||||
return sprintf(buf, "%u\n", br_opt_get(br, BROPT_VLAN_STATS_ENABLED));
|
||||
}
|
||||
|
||||
static ssize_t vlan_stats_enabled_store(struct device *d,
|
||||
|
|
|
@ -386,7 +386,7 @@ struct sk_buff *br_handle_vlan(struct net_bridge *br,
|
|||
return NULL;
|
||||
}
|
||||
}
|
||||
if (br->vlan_stats_enabled) {
|
||||
if (br_opt_get(br, BROPT_VLAN_STATS_ENABLED)) {
|
||||
stats = this_cpu_ptr(v->stats);
|
||||
u64_stats_update_begin(&stats->syncp);
|
||||
stats->tx_bytes += skb->len;
|
||||
|
@ -475,14 +475,14 @@ static bool __allowed_ingress(const struct net_bridge *br,
|
|||
skb->vlan_tci |= pvid;
|
||||
|
||||
/* if stats are disabled we can avoid the lookup */
|
||||
if (!br->vlan_stats_enabled)
|
||||
if (!br_opt_get(br, BROPT_VLAN_STATS_ENABLED))
|
||||
return true;
|
||||
}
|
||||
v = br_vlan_find(vg, *vid);
|
||||
if (!v || !br_vlan_should_use(v))
|
||||
goto drop;
|
||||
|
||||
if (br->vlan_stats_enabled) {
|
||||
if (br_opt_get(br, BROPT_VLAN_STATS_ENABLED)) {
|
||||
stats = this_cpu_ptr(v->stats);
|
||||
u64_stats_update_begin(&stats->syncp);
|
||||
stats->rx_bytes += skb->len;
|
||||
|
@ -504,7 +504,7 @@ bool br_allowed_ingress(const struct net_bridge *br,
|
|||
/* If VLAN filtering is disabled on the bridge, all packets are
|
||||
* permitted.
|
||||
*/
|
||||
if (!br->vlan_enabled) {
|
||||
if (!br_opt_get(br, BROPT_VLAN_ENABLED)) {
|
||||
BR_INPUT_SKB_CB(skb)->vlan_filtered = false;
|
||||
return true;
|
||||
}
|
||||
|
@ -538,7 +538,7 @@ bool br_should_learn(struct net_bridge_port *p, struct sk_buff *skb, u16 *vid)
|
|||
struct net_bridge *br = p->br;
|
||||
|
||||
/* If filtering was disabled at input, let it pass. */
|
||||
if (!br->vlan_enabled)
|
||||
if (!br_opt_get(br, BROPT_VLAN_ENABLED))
|
||||
return true;
|
||||
|
||||
vg = nbp_vlan_group_rcu(p);
|
||||
|
@ -699,7 +699,8 @@ static void recalculate_group_addr(struct net_bridge *br)
|
|||
return;
|
||||
|
||||
spin_lock_bh(&br->lock);
|
||||
if (!br->vlan_enabled || br->vlan_proto == htons(ETH_P_8021Q)) {
|
||||
if (!br_opt_get(br, BROPT_VLAN_ENABLED) ||
|
||||
br->vlan_proto == htons(ETH_P_8021Q)) {
|
||||
/* Bridge Group Address */
|
||||
br->group_addr[5] = 0x00;
|
||||
} else { /* vlan_enabled && ETH_P_8021AD */
|
||||
|
@ -712,7 +713,8 @@ static void recalculate_group_addr(struct net_bridge *br)
|
|||
/* Must be protected by RTNL. */
|
||||
void br_recalculate_fwd_mask(struct net_bridge *br)
|
||||
{
|
||||
if (!br->vlan_enabled || br->vlan_proto == htons(ETH_P_8021Q))
|
||||
if (!br_opt_get(br, BROPT_VLAN_ENABLED) ||
|
||||
br->vlan_proto == htons(ETH_P_8021Q))
|
||||
br->group_fwd_mask_required = BR_GROUPFWD_DEFAULT;
|
||||
else /* vlan_enabled && ETH_P_8021AD */
|
||||
br->group_fwd_mask_required = BR_GROUPFWD_8021AD &
|
||||
|
@ -729,14 +731,14 @@ int __br_vlan_filter_toggle(struct net_bridge *br, unsigned long val)
|
|||
};
|
||||
int err;
|
||||
|
||||
if (br->vlan_enabled == val)
|
||||
if (br_opt_get(br, BROPT_VLAN_ENABLED) == !!val)
|
||||
return 0;
|
||||
|
||||
err = switchdev_port_attr_set(br->dev, &attr);
|
||||
if (err && err != -EOPNOTSUPP)
|
||||
return err;
|
||||
|
||||
br->vlan_enabled = val;
|
||||
br_opt_toggle(br, BROPT_VLAN_ENABLED, !!val);
|
||||
br_manage_promisc(br);
|
||||
recalculate_group_addr(br);
|
||||
br_recalculate_fwd_mask(br);
|
||||
|
@ -753,7 +755,7 @@ bool br_vlan_enabled(const struct net_device *dev)
|
|||
{
|
||||
struct net_bridge *br = netdev_priv(dev);
|
||||
|
||||
return !!br->vlan_enabled;
|
||||
return br_opt_get(br, BROPT_VLAN_ENABLED);
|
||||
}
|
||||
EXPORT_SYMBOL_GPL(br_vlan_enabled);
|
||||
|
||||
|
@ -819,7 +821,7 @@ int br_vlan_set_stats(struct net_bridge *br, unsigned long val)
|
|||
switch (val) {
|
||||
case 0:
|
||||
case 1:
|
||||
br->vlan_stats_enabled = val;
|
||||
br_opt_toggle(br, BROPT_VLAN_STATS_ENABLED, !!val);
|
||||
break;
|
||||
default:
|
||||
return -EINVAL;
|
||||
|
@ -964,7 +966,7 @@ int br_vlan_set_default_pvid(struct net_bridge *br, unsigned long val)
|
|||
goto out;
|
||||
|
||||
/* Only allow default pvid change when filtering is disabled */
|
||||
if (br->vlan_enabled) {
|
||||
if (br_opt_get(br, BROPT_VLAN_ENABLED)) {
|
||||
pr_info_once("Please disable vlan filtering to change default_pvid\n");
|
||||
err = -EPERM;
|
||||
goto out;
|
||||
|
@ -1018,7 +1020,7 @@ int nbp_vlan_init(struct net_bridge_port *p)
|
|||
.orig_dev = p->br->dev,
|
||||
.id = SWITCHDEV_ATTR_ID_BRIDGE_VLAN_FILTERING,
|
||||
.flags = SWITCHDEV_F_SKIP_EOPNOTSUPP,
|
||||
.u.vlan_filtering = p->br->vlan_enabled,
|
||||
.u.vlan_filtering = br_opt_get(p->br, BROPT_VLAN_ENABLED),
|
||||
};
|
||||
struct net_bridge_vlan_group *vg;
|
||||
int ret = -ENOMEM;
|
||||
|
|
Loading…
Reference in New Issue