123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083 |
- #include <linux/kernel.h>
- #include <linux/netdevice.h>
- #include <linux/rtnetlink.h>
- #include <linux/slab.h>
- #include <net/switchdev.h>
- #include "br_private.h"
- static inline int br_vlan_cmp(struct rhashtable_compare_arg *arg,
- const void *ptr)
- {
- const struct net_bridge_vlan *vle = ptr;
- u16 vid = *(u16 *)arg->key;
- return vle->vid != vid;
- }
- static const struct rhashtable_params br_vlan_rht_params = {
- .head_offset = offsetof(struct net_bridge_vlan, vnode),
- .key_offset = offsetof(struct net_bridge_vlan, vid),
- .key_len = sizeof(u16),
- .nelem_hint = 3,
- .locks_mul = 1,
- .max_size = VLAN_N_VID,
- .obj_cmpfn = br_vlan_cmp,
- .automatic_shrinking = true,
- };
- static struct net_bridge_vlan *br_vlan_lookup(struct rhashtable *tbl, u16 vid)
- {
- return rhashtable_lookup_fast(tbl, &vid, br_vlan_rht_params);
- }
- static void __vlan_add_pvid(struct net_bridge_vlan_group *vg, u16 vid)
- {
- if (vg->pvid == vid)
- return;
- smp_wmb();
- vg->pvid = vid;
- }
- static void __vlan_delete_pvid(struct net_bridge_vlan_group *vg, u16 vid)
- {
- if (vg->pvid != vid)
- return;
- smp_wmb();
- vg->pvid = 0;
- }
- static void __vlan_add_flags(struct net_bridge_vlan *v, u16 flags)
- {
- struct net_bridge_vlan_group *vg;
- if (br_vlan_is_master(v))
- vg = br_vlan_group(v->br);
- else
- vg = nbp_vlan_group(v->port);
- if (flags & BRIDGE_VLAN_INFO_PVID)
- __vlan_add_pvid(vg, v->vid);
- else
- __vlan_delete_pvid(vg, v->vid);
- if (flags & BRIDGE_VLAN_INFO_UNTAGGED)
- v->flags |= BRIDGE_VLAN_INFO_UNTAGGED;
- else
- v->flags &= ~BRIDGE_VLAN_INFO_UNTAGGED;
- }
- static int __vlan_vid_add(struct net_device *dev, struct net_bridge *br,
- u16 vid, u16 flags)
- {
- struct switchdev_obj_port_vlan v = {
- .obj.orig_dev = dev,
- .obj.id = SWITCHDEV_OBJ_ID_PORT_VLAN,
- .flags = flags,
- .vid_begin = vid,
- .vid_end = vid,
- };
- int err;
- /* Try switchdev op first. In case it is not supported, fallback to
- * 8021q add.
- */
- err = switchdev_port_obj_add(dev, &v.obj);
- if (err == -EOPNOTSUPP)
- return vlan_vid_add(dev, br->vlan_proto, vid);
- return err;
- }
- static void __vlan_add_list(struct net_bridge_vlan *v)
- {
- struct net_bridge_vlan_group *vg;
- struct list_head *headp, *hpos;
- struct net_bridge_vlan *vent;
- if (br_vlan_is_master(v))
- vg = br_vlan_group(v->br);
- else
- vg = nbp_vlan_group(v->port);
- headp = &vg->vlan_list;
- list_for_each_prev(hpos, headp) {
- vent = list_entry(hpos, struct net_bridge_vlan, vlist);
- if (v->vid < vent->vid)
- continue;
- else
- break;
- }
- list_add_rcu(&v->vlist, hpos);
- }
- static void __vlan_del_list(struct net_bridge_vlan *v)
- {
- list_del_rcu(&v->vlist);
- }
- static int __vlan_vid_del(struct net_device *dev, struct net_bridge *br,
- u16 vid)
- {
- struct switchdev_obj_port_vlan v = {
- .obj.orig_dev = dev,
- .obj.id = SWITCHDEV_OBJ_ID_PORT_VLAN,
- .vid_begin = vid,
- .vid_end = vid,
- };
- int err;
- /* Try switchdev op first. In case it is not supported, fallback to
- * 8021q del.
- */
- err = switchdev_port_obj_del(dev, &v.obj);
- if (err == -EOPNOTSUPP) {
- vlan_vid_del(dev, br->vlan_proto, vid);
- return 0;
- }
- return err;
- }
- /* Returns a master vlan, if it didn't exist it gets created. In all cases a
- * a reference is taken to the master vlan before returning.
- */
- static struct net_bridge_vlan *br_vlan_get_master(struct net_bridge *br, u16 vid)
- {
- struct net_bridge_vlan_group *vg;
- struct net_bridge_vlan *masterv;
- vg = br_vlan_group(br);
- masterv = br_vlan_find(vg, vid);
- if (!masterv) {
- /* missing global ctx, create it now */
- if (br_vlan_add(br, vid, 0))
- return NULL;
- masterv = br_vlan_find(vg, vid);
- if (WARN_ON(!masterv))
- return NULL;
- }
- atomic_inc(&masterv->refcnt);
- return masterv;
- }
- static void br_master_vlan_rcu_free(struct rcu_head *rcu)
- {
- struct net_bridge_vlan *v;
- v = container_of(rcu, struct net_bridge_vlan, rcu);
- WARN_ON(!br_vlan_is_master(v));
- free_percpu(v->stats);
- v->stats = NULL;
- kfree(v);
- }
- static void br_vlan_put_master(struct net_bridge_vlan *masterv)
- {
- struct net_bridge_vlan_group *vg;
- if (!br_vlan_is_master(masterv))
- return;
- vg = br_vlan_group(masterv->br);
- if (atomic_dec_and_test(&masterv->refcnt)) {
- rhashtable_remove_fast(&vg->vlan_hash,
- &masterv->vnode, br_vlan_rht_params);
- __vlan_del_list(masterv);
- call_rcu(&masterv->rcu, br_master_vlan_rcu_free);
- }
- }
- /* This is the shared VLAN add function which works for both ports and bridge
- * devices. There are four possible calls to this function in terms of the
- * vlan entry type:
- * 1. vlan is being added on a port (no master flags, global entry exists)
- * 2. vlan is being added on a bridge (both master and brentry flags)
- * 3. vlan is being added on a port, but a global entry didn't exist which
- * is being created right now (master flag set, brentry flag unset), the
- * global entry is used for global per-vlan features, but not for filtering
- * 4. same as 3 but with both master and brentry flags set so the entry
- * will be used for filtering in both the port and the bridge
- */
- static int __vlan_add(struct net_bridge_vlan *v, u16 flags)
- {
- struct net_bridge_vlan *masterv = NULL;
- struct net_bridge_port *p = NULL;
- struct net_bridge_vlan_group *vg;
- struct net_device *dev;
- struct net_bridge *br;
- int err;
- if (br_vlan_is_master(v)) {
- br = v->br;
- dev = br->dev;
- vg = br_vlan_group(br);
- } else {
- p = v->port;
- br = p->br;
- dev = p->dev;
- vg = nbp_vlan_group(p);
- }
- if (p) {
- /* Add VLAN to the device filter if it is supported.
- * This ensures tagged traffic enters the bridge when
- * promiscuous mode is disabled by br_manage_promisc().
- */
- err = __vlan_vid_add(dev, br, v->vid, flags);
- if (err)
- goto out;
- /* need to work on the master vlan too */
- if (flags & BRIDGE_VLAN_INFO_MASTER) {
- err = br_vlan_add(br, v->vid, flags |
- BRIDGE_VLAN_INFO_BRENTRY);
- if (err)
- goto out_filt;
- }
- masterv = br_vlan_get_master(br, v->vid);
- if (!masterv)
- goto out_filt;
- v->brvlan = masterv;
- v->stats = masterv->stats;
- }
- /* Add the dev mac and count the vlan only if it's usable */
- if (br_vlan_should_use(v)) {
- err = br_fdb_insert(br, p, dev->dev_addr, v->vid);
- if (err) {
- br_err(br, "failed insert local address into bridge forwarding table\n");
- goto out_filt;
- }
- vg->num_vlans++;
- }
- err = rhashtable_lookup_insert_fast(&vg->vlan_hash, &v->vnode,
- br_vlan_rht_params);
- if (err)
- goto out_fdb_insert;
- __vlan_add_list(v);
- __vlan_add_flags(v, flags);
- out:
- return err;
- out_fdb_insert:
- if (br_vlan_should_use(v)) {
- br_fdb_find_delete_local(br, p, dev->dev_addr, v->vid);
- vg->num_vlans--;
- }
- out_filt:
- if (p) {
- __vlan_vid_del(dev, br, v->vid);
- if (masterv) {
- br_vlan_put_master(masterv);
- v->brvlan = NULL;
- }
- }
- goto out;
- }
- static int __vlan_del(struct net_bridge_vlan *v)
- {
- struct net_bridge_vlan *masterv = v;
- struct net_bridge_vlan_group *vg;
- struct net_bridge_port *p = NULL;
- int err = 0;
- if (br_vlan_is_master(v)) {
- vg = br_vlan_group(v->br);
- } else {
- p = v->port;
- vg = nbp_vlan_group(v->port);
- masterv = v->brvlan;
- }
- __vlan_delete_pvid(vg, v->vid);
- if (p) {
- err = __vlan_vid_del(p->dev, p->br, v->vid);
- if (err)
- goto out;
- }
- if (br_vlan_should_use(v)) {
- v->flags &= ~BRIDGE_VLAN_INFO_BRENTRY;
- vg->num_vlans--;
- }
- if (masterv != v) {
- rhashtable_remove_fast(&vg->vlan_hash, &v->vnode,
- br_vlan_rht_params);
- __vlan_del_list(v);
- kfree_rcu(v, rcu);
- }
- br_vlan_put_master(masterv);
- out:
- return err;
- }
- static void __vlan_group_free(struct net_bridge_vlan_group *vg)
- {
- WARN_ON(!list_empty(&vg->vlan_list));
- rhashtable_destroy(&vg->vlan_hash);
- kfree(vg);
- }
- static void __vlan_flush(struct net_bridge_vlan_group *vg)
- {
- struct net_bridge_vlan *vlan, *tmp;
- __vlan_delete_pvid(vg, vg->pvid);
- list_for_each_entry_safe(vlan, tmp, &vg->vlan_list, vlist)
- __vlan_del(vlan);
- }
- struct sk_buff *br_handle_vlan(struct net_bridge *br,
- struct net_bridge_vlan_group *vg,
- struct sk_buff *skb)
- {
- struct br_vlan_stats *stats;
- struct net_bridge_vlan *v;
- u16 vid;
- /* If this packet was not filtered at input, let it pass */
- if (!BR_INPUT_SKB_CB(skb)->vlan_filtered)
- goto out;
- /* At this point, we know that the frame was filtered and contains
- * a valid vlan id. If the vlan id has untagged flag set,
- * send untagged; otherwise, send tagged.
- */
- br_vlan_get_tag(skb, &vid);
- v = br_vlan_find(vg, vid);
- /* Vlan entry must be configured at this point. The
- * only exception is the bridge is set in promisc mode and the
- * packet is destined for the bridge device. In this case
- * pass the packet as is.
- */
- if (!v || !br_vlan_should_use(v)) {
- if ((br->dev->flags & IFF_PROMISC) && skb->dev == br->dev) {
- goto out;
- } else {
- kfree_skb(skb);
- return NULL;
- }
- }
- if (br->vlan_stats_enabled) {
- stats = this_cpu_ptr(v->stats);
- u64_stats_update_begin(&stats->syncp);
- stats->tx_bytes += skb->len;
- stats->tx_packets++;
- u64_stats_update_end(&stats->syncp);
- }
- if (v->flags & BRIDGE_VLAN_INFO_UNTAGGED)
- skb->vlan_tci = 0;
- out:
- return skb;
- }
- /* Called under RCU */
- static bool __allowed_ingress(const struct net_bridge *br,
- struct net_bridge_vlan_group *vg,
- struct sk_buff *skb, u16 *vid)
- {
- struct br_vlan_stats *stats;
- struct net_bridge_vlan *v;
- bool tagged;
- BR_INPUT_SKB_CB(skb)->vlan_filtered = true;
- /* If vlan tx offload is disabled on bridge device and frame was
- * sent from vlan device on the bridge device, it does not have
- * HW accelerated vlan tag.
- */
- if (unlikely(!skb_vlan_tag_present(skb) &&
- skb->protocol == br->vlan_proto)) {
- skb = skb_vlan_untag(skb);
- if (unlikely(!skb))
- return false;
- }
- if (!br_vlan_get_tag(skb, vid)) {
- /* Tagged frame */
- if (skb->vlan_proto != br->vlan_proto) {
- /* Protocol-mismatch, empty out vlan_tci for new tag */
- skb_push(skb, ETH_HLEN);
- skb = vlan_insert_tag_set_proto(skb, skb->vlan_proto,
- skb_vlan_tag_get(skb));
- if (unlikely(!skb))
- return false;
- skb_pull(skb, ETH_HLEN);
- skb_reset_mac_len(skb);
- *vid = 0;
- tagged = false;
- } else {
- tagged = true;
- }
- } else {
- /* Untagged frame */
- tagged = false;
- }
- if (!*vid) {
- u16 pvid = br_get_pvid(vg);
- /* Frame had a tag with VID 0 or did not have a tag.
- * See if pvid is set on this port. That tells us which
- * vlan untagged or priority-tagged traffic belongs to.
- */
- if (!pvid)
- goto drop;
- /* PVID is set on this port. Any untagged or priority-tagged
- * ingress frame is considered to belong to this vlan.
- */
- *vid = pvid;
- if (likely(!tagged))
- /* Untagged Frame. */
- __vlan_hwaccel_put_tag(skb, br->vlan_proto, pvid);
- else
- /* Priority-tagged Frame.
- * At this point, We know that skb->vlan_tci had
- * VLAN_TAG_PRESENT bit and its VID field was 0x000.
- * We update only VID field and preserve PCP field.
- */
- skb->vlan_tci |= pvid;
- /* if stats are disabled we can avoid the lookup */
- if (!br->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) {
- stats = this_cpu_ptr(v->stats);
- u64_stats_update_begin(&stats->syncp);
- stats->rx_bytes += skb->len;
- stats->rx_packets++;
- u64_stats_update_end(&stats->syncp);
- }
- return true;
- drop:
- kfree_skb(skb);
- return false;
- }
- bool br_allowed_ingress(const struct net_bridge *br,
- struct net_bridge_vlan_group *vg, struct sk_buff *skb,
- u16 *vid)
- {
- /* If VLAN filtering is disabled on the bridge, all packets are
- * permitted.
- */
- if (!br->vlan_enabled) {
- BR_INPUT_SKB_CB(skb)->vlan_filtered = false;
- return true;
- }
- return __allowed_ingress(br, vg, skb, vid);
- }
- /* Called under RCU. */
- bool br_allowed_egress(struct net_bridge_vlan_group *vg,
- const struct sk_buff *skb)
- {
- const struct net_bridge_vlan *v;
- u16 vid;
- /* If this packet was not filtered at input, let it pass */
- if (!BR_INPUT_SKB_CB(skb)->vlan_filtered)
- return true;
- br_vlan_get_tag(skb, &vid);
- v = br_vlan_find(vg, vid);
- if (v && br_vlan_should_use(v))
- return true;
- return false;
- }
- /* Called under RCU */
- bool br_should_learn(struct net_bridge_port *p, struct sk_buff *skb, u16 *vid)
- {
- struct net_bridge_vlan_group *vg;
- struct net_bridge *br = p->br;
- /* If filtering was disabled at input, let it pass. */
- if (!br->vlan_enabled)
- return true;
- vg = nbp_vlan_group_rcu(p);
- if (!vg || !vg->num_vlans)
- return false;
- if (!br_vlan_get_tag(skb, vid) && skb->vlan_proto != br->vlan_proto)
- *vid = 0;
- if (!*vid) {
- *vid = br_get_pvid(vg);
- if (!*vid)
- return false;
- return true;
- }
- if (br_vlan_find(vg, *vid))
- return true;
- return false;
- }
- /* Must be protected by RTNL.
- * Must be called with vid in range from 1 to 4094 inclusive.
- */
- int br_vlan_add(struct net_bridge *br, u16 vid, u16 flags)
- {
- struct net_bridge_vlan_group *vg;
- struct net_bridge_vlan *vlan;
- int ret;
- ASSERT_RTNL();
- vg = br_vlan_group(br);
- vlan = br_vlan_find(vg, vid);
- if (vlan) {
- if (!br_vlan_is_brentry(vlan)) {
- /* Trying to change flags of non-existent bridge vlan */
- if (!(flags & BRIDGE_VLAN_INFO_BRENTRY))
- return -EINVAL;
- /* It was only kept for port vlans, now make it real */
- ret = br_fdb_insert(br, NULL, br->dev->dev_addr,
- vlan->vid);
- if (ret) {
- br_err(br, "failed insert local address into bridge forwarding table\n");
- return ret;
- }
- atomic_inc(&vlan->refcnt);
- vlan->flags |= BRIDGE_VLAN_INFO_BRENTRY;
- vg->num_vlans++;
- }
- __vlan_add_flags(vlan, flags);
- return 0;
- }
- vlan = kzalloc(sizeof(*vlan), GFP_KERNEL);
- if (!vlan)
- return -ENOMEM;
- vlan->stats = netdev_alloc_pcpu_stats(struct br_vlan_stats);
- if (!vlan->stats) {
- kfree(vlan);
- return -ENOMEM;
- }
- vlan->vid = vid;
- vlan->flags = flags | BRIDGE_VLAN_INFO_MASTER;
- vlan->flags &= ~BRIDGE_VLAN_INFO_PVID;
- vlan->br = br;
- if (flags & BRIDGE_VLAN_INFO_BRENTRY)
- atomic_set(&vlan->refcnt, 1);
- ret = __vlan_add(vlan, flags);
- if (ret) {
- free_percpu(vlan->stats);
- kfree(vlan);
- }
- return ret;
- }
- /* Must be protected by RTNL.
- * Must be called with vid in range from 1 to 4094 inclusive.
- */
- int br_vlan_delete(struct net_bridge *br, u16 vid)
- {
- struct net_bridge_vlan_group *vg;
- struct net_bridge_vlan *v;
- ASSERT_RTNL();
- vg = br_vlan_group(br);
- v = br_vlan_find(vg, vid);
- if (!v || !br_vlan_is_brentry(v))
- return -ENOENT;
- br_fdb_find_delete_local(br, NULL, br->dev->dev_addr, vid);
- br_fdb_delete_by_port(br, NULL, vid, 0);
- return __vlan_del(v);
- }
- void br_vlan_flush(struct net_bridge *br)
- {
- struct net_bridge_vlan_group *vg;
- ASSERT_RTNL();
- vg = br_vlan_group(br);
- __vlan_flush(vg);
- RCU_INIT_POINTER(br->vlgrp, NULL);
- synchronize_rcu();
- __vlan_group_free(vg);
- }
- struct net_bridge_vlan *br_vlan_find(struct net_bridge_vlan_group *vg, u16 vid)
- {
- if (!vg)
- return NULL;
- return br_vlan_lookup(&vg->vlan_hash, vid);
- }
- /* Must be protected by RTNL. */
- static void recalculate_group_addr(struct net_bridge *br)
- {
- if (br->group_addr_set)
- return;
- spin_lock_bh(&br->lock);
- if (!br->vlan_enabled || br->vlan_proto == htons(ETH_P_8021Q)) {
- /* Bridge Group Address */
- br->group_addr[5] = 0x00;
- } else { /* vlan_enabled && ETH_P_8021AD */
- /* Provider Bridge Group Address */
- br->group_addr[5] = 0x08;
- }
- spin_unlock_bh(&br->lock);
- }
- /* 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))
- br->group_fwd_mask_required = BR_GROUPFWD_DEFAULT;
- else /* vlan_enabled && ETH_P_8021AD */
- br->group_fwd_mask_required = BR_GROUPFWD_8021AD &
- ~(1u << br->group_addr[5]);
- }
- int __br_vlan_filter_toggle(struct net_bridge *br, unsigned long val)
- {
- struct switchdev_attr attr = {
- .orig_dev = br->dev,
- .id = SWITCHDEV_ATTR_ID_BRIDGE_VLAN_FILTERING,
- .flags = SWITCHDEV_F_SKIP_EOPNOTSUPP,
- .u.vlan_filtering = val,
- };
- int err;
- if (br->vlan_enabled == val)
- return 0;
- err = switchdev_port_attr_set(br->dev, &attr);
- if (err && err != -EOPNOTSUPP)
- return err;
- br->vlan_enabled = val;
- br_manage_promisc(br);
- recalculate_group_addr(br);
- br_recalculate_fwd_mask(br);
- return 0;
- }
- int br_vlan_filter_toggle(struct net_bridge *br, unsigned long val)
- {
- return __br_vlan_filter_toggle(br, val);
- }
- int __br_vlan_set_proto(struct net_bridge *br, __be16 proto)
- {
- int err = 0;
- struct net_bridge_port *p;
- struct net_bridge_vlan *vlan;
- struct net_bridge_vlan_group *vg;
- __be16 oldproto;
- if (br->vlan_proto == proto)
- return 0;
- /* Add VLANs for the new proto to the device filter. */
- list_for_each_entry(p, &br->port_list, list) {
- vg = nbp_vlan_group(p);
- list_for_each_entry(vlan, &vg->vlan_list, vlist) {
- err = vlan_vid_add(p->dev, proto, vlan->vid);
- if (err)
- goto err_filt;
- }
- }
- oldproto = br->vlan_proto;
- br->vlan_proto = proto;
- recalculate_group_addr(br);
- br_recalculate_fwd_mask(br);
- /* Delete VLANs for the old proto from the device filter. */
- list_for_each_entry(p, &br->port_list, list) {
- vg = nbp_vlan_group(p);
- list_for_each_entry(vlan, &vg->vlan_list, vlist)
- vlan_vid_del(p->dev, oldproto, vlan->vid);
- }
- return 0;
- err_filt:
- list_for_each_entry_continue_reverse(vlan, &vg->vlan_list, vlist)
- vlan_vid_del(p->dev, proto, vlan->vid);
- list_for_each_entry_continue_reverse(p, &br->port_list, list) {
- vg = nbp_vlan_group(p);
- list_for_each_entry(vlan, &vg->vlan_list, vlist)
- vlan_vid_del(p->dev, proto, vlan->vid);
- }
- return err;
- }
- int br_vlan_set_proto(struct net_bridge *br, unsigned long val)
- {
- if (val != ETH_P_8021Q && val != ETH_P_8021AD)
- return -EPROTONOSUPPORT;
- return __br_vlan_set_proto(br, htons(val));
- }
- int br_vlan_set_stats(struct net_bridge *br, unsigned long val)
- {
- switch (val) {
- case 0:
- case 1:
- br->vlan_stats_enabled = val;
- break;
- default:
- return -EINVAL;
- }
- return 0;
- }
- static bool vlan_default_pvid(struct net_bridge_vlan_group *vg, u16 vid)
- {
- struct net_bridge_vlan *v;
- if (vid != vg->pvid)
- return false;
- v = br_vlan_lookup(&vg->vlan_hash, vid);
- if (v && br_vlan_should_use(v) &&
- (v->flags & BRIDGE_VLAN_INFO_UNTAGGED))
- return true;
- return false;
- }
- static void br_vlan_disable_default_pvid(struct net_bridge *br)
- {
- struct net_bridge_port *p;
- u16 pvid = br->default_pvid;
- /* Disable default_pvid on all ports where it is still
- * configured.
- */
- if (vlan_default_pvid(br_vlan_group(br), pvid))
- br_vlan_delete(br, pvid);
- list_for_each_entry(p, &br->port_list, list) {
- if (vlan_default_pvid(nbp_vlan_group(p), pvid))
- nbp_vlan_delete(p, pvid);
- }
- br->default_pvid = 0;
- }
- int __br_vlan_set_default_pvid(struct net_bridge *br, u16 pvid)
- {
- const struct net_bridge_vlan *pvent;
- struct net_bridge_vlan_group *vg;
- struct net_bridge_port *p;
- u16 old_pvid;
- int err = 0;
- unsigned long *changed;
- if (!pvid) {
- br_vlan_disable_default_pvid(br);
- return 0;
- }
- changed = kcalloc(BITS_TO_LONGS(BR_MAX_PORTS), sizeof(unsigned long),
- GFP_KERNEL);
- if (!changed)
- return -ENOMEM;
- old_pvid = br->default_pvid;
- /* Update default_pvid config only if we do not conflict with
- * user configuration.
- */
- vg = br_vlan_group(br);
- pvent = br_vlan_find(vg, pvid);
- if ((!old_pvid || vlan_default_pvid(vg, old_pvid)) &&
- (!pvent || !br_vlan_should_use(pvent))) {
- err = br_vlan_add(br, pvid,
- BRIDGE_VLAN_INFO_PVID |
- BRIDGE_VLAN_INFO_UNTAGGED |
- BRIDGE_VLAN_INFO_BRENTRY);
- if (err)
- goto out;
- br_vlan_delete(br, old_pvid);
- set_bit(0, changed);
- }
- list_for_each_entry(p, &br->port_list, list) {
- /* Update default_pvid config only if we do not conflict with
- * user configuration.
- */
- vg = nbp_vlan_group(p);
- if ((old_pvid &&
- !vlan_default_pvid(vg, old_pvid)) ||
- br_vlan_find(vg, pvid))
- continue;
- err = nbp_vlan_add(p, pvid,
- BRIDGE_VLAN_INFO_PVID |
- BRIDGE_VLAN_INFO_UNTAGGED);
- if (err)
- goto err_port;
- nbp_vlan_delete(p, old_pvid);
- set_bit(p->port_no, changed);
- }
- br->default_pvid = pvid;
- out:
- kfree(changed);
- return err;
- err_port:
- list_for_each_entry_continue_reverse(p, &br->port_list, list) {
- if (!test_bit(p->port_no, changed))
- continue;
- if (old_pvid)
- nbp_vlan_add(p, old_pvid,
- BRIDGE_VLAN_INFO_PVID |
- BRIDGE_VLAN_INFO_UNTAGGED);
- nbp_vlan_delete(p, pvid);
- }
- if (test_bit(0, changed)) {
- if (old_pvid)
- br_vlan_add(br, old_pvid,
- BRIDGE_VLAN_INFO_PVID |
- BRIDGE_VLAN_INFO_UNTAGGED |
- BRIDGE_VLAN_INFO_BRENTRY);
- br_vlan_delete(br, pvid);
- }
- goto out;
- }
- int br_vlan_set_default_pvid(struct net_bridge *br, unsigned long val)
- {
- u16 pvid = val;
- int err = 0;
- if (val >= VLAN_VID_MASK)
- return -EINVAL;
- if (pvid == br->default_pvid)
- goto out;
- /* Only allow default pvid change when filtering is disabled */
- if (br->vlan_enabled) {
- pr_info_once("Please disable vlan filtering to change default_pvid\n");
- err = -EPERM;
- goto out;
- }
- err = __br_vlan_set_default_pvid(br, pvid);
- out:
- return err;
- }
- int br_vlan_init(struct net_bridge *br)
- {
- struct net_bridge_vlan_group *vg;
- int ret = -ENOMEM;
- vg = kzalloc(sizeof(*vg), GFP_KERNEL);
- if (!vg)
- goto out;
- ret = rhashtable_init(&vg->vlan_hash, &br_vlan_rht_params);
- if (ret)
- goto err_rhtbl;
- INIT_LIST_HEAD(&vg->vlan_list);
- br->vlan_proto = htons(ETH_P_8021Q);
- br->default_pvid = 1;
- rcu_assign_pointer(br->vlgrp, vg);
- ret = br_vlan_add(br, 1,
- BRIDGE_VLAN_INFO_PVID | BRIDGE_VLAN_INFO_UNTAGGED |
- BRIDGE_VLAN_INFO_BRENTRY);
- if (ret)
- goto err_vlan_add;
- out:
- return ret;
- err_vlan_add:
- rhashtable_destroy(&vg->vlan_hash);
- err_rhtbl:
- kfree(vg);
- goto out;
- }
- int nbp_vlan_init(struct net_bridge_port *p)
- {
- struct switchdev_attr attr = {
- .orig_dev = p->br->dev,
- .id = SWITCHDEV_ATTR_ID_BRIDGE_VLAN_FILTERING,
- .flags = SWITCHDEV_F_SKIP_EOPNOTSUPP,
- .u.vlan_filtering = p->br->vlan_enabled,
- };
- struct net_bridge_vlan_group *vg;
- int ret = -ENOMEM;
- vg = kzalloc(sizeof(struct net_bridge_vlan_group), GFP_KERNEL);
- if (!vg)
- goto out;
- ret = switchdev_port_attr_set(p->dev, &attr);
- if (ret && ret != -EOPNOTSUPP)
- goto err_vlan_enabled;
- ret = rhashtable_init(&vg->vlan_hash, &br_vlan_rht_params);
- if (ret)
- goto err_rhtbl;
- INIT_LIST_HEAD(&vg->vlan_list);
- rcu_assign_pointer(p->vlgrp, vg);
- if (p->br->default_pvid) {
- ret = nbp_vlan_add(p, p->br->default_pvid,
- BRIDGE_VLAN_INFO_PVID |
- BRIDGE_VLAN_INFO_UNTAGGED);
- if (ret)
- goto err_vlan_add;
- }
- out:
- return ret;
- err_vlan_add:
- RCU_INIT_POINTER(p->vlgrp, NULL);
- synchronize_rcu();
- rhashtable_destroy(&vg->vlan_hash);
- err_vlan_enabled:
- err_rhtbl:
- kfree(vg);
- goto out;
- }
- /* Must be protected by RTNL.
- * Must be called with vid in range from 1 to 4094 inclusive.
- */
- int nbp_vlan_add(struct net_bridge_port *port, u16 vid, u16 flags)
- {
- struct switchdev_obj_port_vlan v = {
- .obj.orig_dev = port->dev,
- .obj.id = SWITCHDEV_OBJ_ID_PORT_VLAN,
- .flags = flags,
- .vid_begin = vid,
- .vid_end = vid,
- };
- struct net_bridge_vlan *vlan;
- int ret;
- ASSERT_RTNL();
- vlan = br_vlan_find(nbp_vlan_group(port), vid);
- if (vlan) {
- /* Pass the flags to the hardware bridge */
- ret = switchdev_port_obj_add(port->dev, &v.obj);
- if (ret && ret != -EOPNOTSUPP)
- return ret;
- __vlan_add_flags(vlan, flags);
- return 0;
- }
- vlan = kzalloc(sizeof(*vlan), GFP_KERNEL);
- if (!vlan)
- return -ENOMEM;
- vlan->vid = vid;
- vlan->port = port;
- ret = __vlan_add(vlan, flags);
- if (ret)
- kfree(vlan);
- return ret;
- }
- /* Must be protected by RTNL.
- * Must be called with vid in range from 1 to 4094 inclusive.
- */
- int nbp_vlan_delete(struct net_bridge_port *port, u16 vid)
- {
- struct net_bridge_vlan *v;
- ASSERT_RTNL();
- v = br_vlan_find(nbp_vlan_group(port), vid);
- if (!v)
- return -ENOENT;
- br_fdb_find_delete_local(port->br, port, port->dev->dev_addr, vid);
- br_fdb_delete_by_port(port->br, port, vid, 0);
- return __vlan_del(v);
- }
- void nbp_vlan_flush(struct net_bridge_port *port)
- {
- struct net_bridge_vlan_group *vg;
- ASSERT_RTNL();
- vg = nbp_vlan_group(port);
- __vlan_flush(vg);
- RCU_INIT_POINTER(port->vlgrp, NULL);
- synchronize_rcu();
- __vlan_group_free(vg);
- }
- void br_vlan_get_stats(const struct net_bridge_vlan *v,
- struct br_vlan_stats *stats)
- {
- int i;
- memset(stats, 0, sizeof(*stats));
- for_each_possible_cpu(i) {
- u64 rxpackets, rxbytes, txpackets, txbytes;
- struct br_vlan_stats *cpu_stats;
- unsigned int start;
- cpu_stats = per_cpu_ptr(v->stats, i);
- do {
- start = u64_stats_fetch_begin_irq(&cpu_stats->syncp);
- rxpackets = cpu_stats->rx_packets;
- rxbytes = cpu_stats->rx_bytes;
- txbytes = cpu_stats->tx_bytes;
- txpackets = cpu_stats->tx_packets;
- } while (u64_stats_fetch_retry_irq(&cpu_stats->syncp, start));
- stats->rx_packets += rxpackets;
- stats->rx_bytes += rxbytes;
- stats->tx_bytes += txbytes;
- stats->tx_packets += txpackets;
- }
- }
|