1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030 |
- // SPDX-License-Identifier: GPL-2.0-or-later
- /*
- * Handling of a single switch chip, part of a switch fabric
- *
- * Copyright (c) 2017 Savoir-faire Linux Inc.
- * Vivien Didelot <[email protected]>
- */
- #include <linux/if_bridge.h>
- #include <linux/netdevice.h>
- #include <linux/notifier.h>
- #include <linux/if_vlan.h>
- #include <net/switchdev.h>
- #include "dsa_priv.h"
- static unsigned int dsa_switch_fastest_ageing_time(struct dsa_switch *ds,
- unsigned int ageing_time)
- {
- struct dsa_port *dp;
- dsa_switch_for_each_port(dp, ds)
- if (dp->ageing_time && dp->ageing_time < ageing_time)
- ageing_time = dp->ageing_time;
- return ageing_time;
- }
- static int dsa_switch_ageing_time(struct dsa_switch *ds,
- struct dsa_notifier_ageing_time_info *info)
- {
- unsigned int ageing_time = info->ageing_time;
- if (ds->ageing_time_min && ageing_time < ds->ageing_time_min)
- return -ERANGE;
- if (ds->ageing_time_max && ageing_time > ds->ageing_time_max)
- return -ERANGE;
- /* Program the fastest ageing time in case of multiple bridges */
- ageing_time = dsa_switch_fastest_ageing_time(ds, ageing_time);
- if (ds->ops->set_ageing_time)
- return ds->ops->set_ageing_time(ds, ageing_time);
- return 0;
- }
- static bool dsa_port_mtu_match(struct dsa_port *dp,
- struct dsa_notifier_mtu_info *info)
- {
- return dp == info->dp || dsa_port_is_dsa(dp) || dsa_port_is_cpu(dp);
- }
- static int dsa_switch_mtu(struct dsa_switch *ds,
- struct dsa_notifier_mtu_info *info)
- {
- struct dsa_port *dp;
- int ret;
- if (!ds->ops->port_change_mtu)
- return -EOPNOTSUPP;
- dsa_switch_for_each_port(dp, ds) {
- if (dsa_port_mtu_match(dp, info)) {
- ret = ds->ops->port_change_mtu(ds, dp->index,
- info->mtu);
- if (ret)
- return ret;
- }
- }
- return 0;
- }
- static int dsa_switch_bridge_join(struct dsa_switch *ds,
- struct dsa_notifier_bridge_info *info)
- {
- int err;
- if (info->dp->ds == ds) {
- if (!ds->ops->port_bridge_join)
- return -EOPNOTSUPP;
- err = ds->ops->port_bridge_join(ds, info->dp->index,
- info->bridge,
- &info->tx_fwd_offload,
- info->extack);
- if (err)
- return err;
- }
- if (info->dp->ds != ds && ds->ops->crosschip_bridge_join) {
- err = ds->ops->crosschip_bridge_join(ds,
- info->dp->ds->dst->index,
- info->dp->ds->index,
- info->dp->index,
- info->bridge,
- info->extack);
- if (err)
- return err;
- }
- return 0;
- }
- static int dsa_switch_bridge_leave(struct dsa_switch *ds,
- struct dsa_notifier_bridge_info *info)
- {
- if (info->dp->ds == ds && ds->ops->port_bridge_leave)
- ds->ops->port_bridge_leave(ds, info->dp->index, info->bridge);
- if (info->dp->ds != ds && ds->ops->crosschip_bridge_leave)
- ds->ops->crosschip_bridge_leave(ds, info->dp->ds->dst->index,
- info->dp->ds->index,
- info->dp->index,
- info->bridge);
- return 0;
- }
- /* Matches for all upstream-facing ports (the CPU port and all upstream-facing
- * DSA links) that sit between the targeted port on which the notifier was
- * emitted and its dedicated CPU port.
- */
- static bool dsa_port_host_address_match(struct dsa_port *dp,
- const struct dsa_port *targeted_dp)
- {
- struct dsa_port *cpu_dp = targeted_dp->cpu_dp;
- if (dsa_switch_is_upstream_of(dp->ds, targeted_dp->ds))
- return dp->index == dsa_towards_port(dp->ds, cpu_dp->ds->index,
- cpu_dp->index);
- return false;
- }
- static struct dsa_mac_addr *dsa_mac_addr_find(struct list_head *addr_list,
- const unsigned char *addr, u16 vid,
- struct dsa_db db)
- {
- struct dsa_mac_addr *a;
- list_for_each_entry(a, addr_list, list)
- if (ether_addr_equal(a->addr, addr) && a->vid == vid &&
- dsa_db_equal(&a->db, &db))
- return a;
- return NULL;
- }
- static int dsa_port_do_mdb_add(struct dsa_port *dp,
- const struct switchdev_obj_port_mdb *mdb,
- struct dsa_db db)
- {
- struct dsa_switch *ds = dp->ds;
- struct dsa_mac_addr *a;
- int port = dp->index;
- int err = 0;
- /* No need to bother with refcounting for user ports */
- if (!(dsa_port_is_cpu(dp) || dsa_port_is_dsa(dp)))
- return ds->ops->port_mdb_add(ds, port, mdb, db);
- mutex_lock(&dp->addr_lists_lock);
- a = dsa_mac_addr_find(&dp->mdbs, mdb->addr, mdb->vid, db);
- if (a) {
- refcount_inc(&a->refcount);
- goto out;
- }
- a = kzalloc(sizeof(*a), GFP_KERNEL);
- if (!a) {
- err = -ENOMEM;
- goto out;
- }
- err = ds->ops->port_mdb_add(ds, port, mdb, db);
- if (err) {
- kfree(a);
- goto out;
- }
- ether_addr_copy(a->addr, mdb->addr);
- a->vid = mdb->vid;
- a->db = db;
- refcount_set(&a->refcount, 1);
- list_add_tail(&a->list, &dp->mdbs);
- out:
- mutex_unlock(&dp->addr_lists_lock);
- return err;
- }
- static int dsa_port_do_mdb_del(struct dsa_port *dp,
- const struct switchdev_obj_port_mdb *mdb,
- struct dsa_db db)
- {
- struct dsa_switch *ds = dp->ds;
- struct dsa_mac_addr *a;
- int port = dp->index;
- int err = 0;
- /* No need to bother with refcounting for user ports */
- if (!(dsa_port_is_cpu(dp) || dsa_port_is_dsa(dp)))
- return ds->ops->port_mdb_del(ds, port, mdb, db);
- mutex_lock(&dp->addr_lists_lock);
- a = dsa_mac_addr_find(&dp->mdbs, mdb->addr, mdb->vid, db);
- if (!a) {
- err = -ENOENT;
- goto out;
- }
- if (!refcount_dec_and_test(&a->refcount))
- goto out;
- err = ds->ops->port_mdb_del(ds, port, mdb, db);
- if (err) {
- refcount_set(&a->refcount, 1);
- goto out;
- }
- list_del(&a->list);
- kfree(a);
- out:
- mutex_unlock(&dp->addr_lists_lock);
- return err;
- }
- static int dsa_port_do_fdb_add(struct dsa_port *dp, const unsigned char *addr,
- u16 vid, struct dsa_db db)
- {
- struct dsa_switch *ds = dp->ds;
- struct dsa_mac_addr *a;
- int port = dp->index;
- int err = 0;
- /* No need to bother with refcounting for user ports */
- if (!(dsa_port_is_cpu(dp) || dsa_port_is_dsa(dp)))
- return ds->ops->port_fdb_add(ds, port, addr, vid, db);
- mutex_lock(&dp->addr_lists_lock);
- a = dsa_mac_addr_find(&dp->fdbs, addr, vid, db);
- if (a) {
- refcount_inc(&a->refcount);
- goto out;
- }
- a = kzalloc(sizeof(*a), GFP_KERNEL);
- if (!a) {
- err = -ENOMEM;
- goto out;
- }
- err = ds->ops->port_fdb_add(ds, port, addr, vid, db);
- if (err) {
- kfree(a);
- goto out;
- }
- ether_addr_copy(a->addr, addr);
- a->vid = vid;
- a->db = db;
- refcount_set(&a->refcount, 1);
- list_add_tail(&a->list, &dp->fdbs);
- out:
- mutex_unlock(&dp->addr_lists_lock);
- return err;
- }
- static int dsa_port_do_fdb_del(struct dsa_port *dp, const unsigned char *addr,
- u16 vid, struct dsa_db db)
- {
- struct dsa_switch *ds = dp->ds;
- struct dsa_mac_addr *a;
- int port = dp->index;
- int err = 0;
- /* No need to bother with refcounting for user ports */
- if (!(dsa_port_is_cpu(dp) || dsa_port_is_dsa(dp)))
- return ds->ops->port_fdb_del(ds, port, addr, vid, db);
- mutex_lock(&dp->addr_lists_lock);
- a = dsa_mac_addr_find(&dp->fdbs, addr, vid, db);
- if (!a) {
- err = -ENOENT;
- goto out;
- }
- if (!refcount_dec_and_test(&a->refcount))
- goto out;
- err = ds->ops->port_fdb_del(ds, port, addr, vid, db);
- if (err) {
- refcount_set(&a->refcount, 1);
- goto out;
- }
- list_del(&a->list);
- kfree(a);
- out:
- mutex_unlock(&dp->addr_lists_lock);
- return err;
- }
- static int dsa_switch_do_lag_fdb_add(struct dsa_switch *ds, struct dsa_lag *lag,
- const unsigned char *addr, u16 vid,
- struct dsa_db db)
- {
- struct dsa_mac_addr *a;
- int err = 0;
- mutex_lock(&lag->fdb_lock);
- a = dsa_mac_addr_find(&lag->fdbs, addr, vid, db);
- if (a) {
- refcount_inc(&a->refcount);
- goto out;
- }
- a = kzalloc(sizeof(*a), GFP_KERNEL);
- if (!a) {
- err = -ENOMEM;
- goto out;
- }
- err = ds->ops->lag_fdb_add(ds, *lag, addr, vid, db);
- if (err) {
- kfree(a);
- goto out;
- }
- ether_addr_copy(a->addr, addr);
- a->vid = vid;
- a->db = db;
- refcount_set(&a->refcount, 1);
- list_add_tail(&a->list, &lag->fdbs);
- out:
- mutex_unlock(&lag->fdb_lock);
- return err;
- }
- static int dsa_switch_do_lag_fdb_del(struct dsa_switch *ds, struct dsa_lag *lag,
- const unsigned char *addr, u16 vid,
- struct dsa_db db)
- {
- struct dsa_mac_addr *a;
- int err = 0;
- mutex_lock(&lag->fdb_lock);
- a = dsa_mac_addr_find(&lag->fdbs, addr, vid, db);
- if (!a) {
- err = -ENOENT;
- goto out;
- }
- if (!refcount_dec_and_test(&a->refcount))
- goto out;
- err = ds->ops->lag_fdb_del(ds, *lag, addr, vid, db);
- if (err) {
- refcount_set(&a->refcount, 1);
- goto out;
- }
- list_del(&a->list);
- kfree(a);
- out:
- mutex_unlock(&lag->fdb_lock);
- return err;
- }
- static int dsa_switch_host_fdb_add(struct dsa_switch *ds,
- struct dsa_notifier_fdb_info *info)
- {
- struct dsa_port *dp;
- int err = 0;
- if (!ds->ops->port_fdb_add)
- return -EOPNOTSUPP;
- dsa_switch_for_each_port(dp, ds) {
- if (dsa_port_host_address_match(dp, info->dp)) {
- if (dsa_port_is_cpu(dp) && info->dp->cpu_port_in_lag) {
- err = dsa_switch_do_lag_fdb_add(ds, dp->lag,
- info->addr,
- info->vid,
- info->db);
- } else {
- err = dsa_port_do_fdb_add(dp, info->addr,
- info->vid, info->db);
- }
- if (err)
- break;
- }
- }
- return err;
- }
- static int dsa_switch_host_fdb_del(struct dsa_switch *ds,
- struct dsa_notifier_fdb_info *info)
- {
- struct dsa_port *dp;
- int err = 0;
- if (!ds->ops->port_fdb_del)
- return -EOPNOTSUPP;
- dsa_switch_for_each_port(dp, ds) {
- if (dsa_port_host_address_match(dp, info->dp)) {
- if (dsa_port_is_cpu(dp) && info->dp->cpu_port_in_lag) {
- err = dsa_switch_do_lag_fdb_del(ds, dp->lag,
- info->addr,
- info->vid,
- info->db);
- } else {
- err = dsa_port_do_fdb_del(dp, info->addr,
- info->vid, info->db);
- }
- if (err)
- break;
- }
- }
- return err;
- }
- static int dsa_switch_fdb_add(struct dsa_switch *ds,
- struct dsa_notifier_fdb_info *info)
- {
- int port = dsa_towards_port(ds, info->dp->ds->index, info->dp->index);
- struct dsa_port *dp = dsa_to_port(ds, port);
- if (!ds->ops->port_fdb_add)
- return -EOPNOTSUPP;
- return dsa_port_do_fdb_add(dp, info->addr, info->vid, info->db);
- }
- static int dsa_switch_fdb_del(struct dsa_switch *ds,
- struct dsa_notifier_fdb_info *info)
- {
- int port = dsa_towards_port(ds, info->dp->ds->index, info->dp->index);
- struct dsa_port *dp = dsa_to_port(ds, port);
- if (!ds->ops->port_fdb_del)
- return -EOPNOTSUPP;
- return dsa_port_do_fdb_del(dp, info->addr, info->vid, info->db);
- }
- static int dsa_switch_lag_fdb_add(struct dsa_switch *ds,
- struct dsa_notifier_lag_fdb_info *info)
- {
- struct dsa_port *dp;
- if (!ds->ops->lag_fdb_add)
- return -EOPNOTSUPP;
- /* Notify switch only if it has a port in this LAG */
- dsa_switch_for_each_port(dp, ds)
- if (dsa_port_offloads_lag(dp, info->lag))
- return dsa_switch_do_lag_fdb_add(ds, info->lag,
- info->addr, info->vid,
- info->db);
- return 0;
- }
- static int dsa_switch_lag_fdb_del(struct dsa_switch *ds,
- struct dsa_notifier_lag_fdb_info *info)
- {
- struct dsa_port *dp;
- if (!ds->ops->lag_fdb_del)
- return -EOPNOTSUPP;
- /* Notify switch only if it has a port in this LAG */
- dsa_switch_for_each_port(dp, ds)
- if (dsa_port_offloads_lag(dp, info->lag))
- return dsa_switch_do_lag_fdb_del(ds, info->lag,
- info->addr, info->vid,
- info->db);
- return 0;
- }
- static int dsa_switch_lag_change(struct dsa_switch *ds,
- struct dsa_notifier_lag_info *info)
- {
- if (info->dp->ds == ds && ds->ops->port_lag_change)
- return ds->ops->port_lag_change(ds, info->dp->index);
- if (info->dp->ds != ds && ds->ops->crosschip_lag_change)
- return ds->ops->crosschip_lag_change(ds, info->dp->ds->index,
- info->dp->index);
- return 0;
- }
- static int dsa_switch_lag_join(struct dsa_switch *ds,
- struct dsa_notifier_lag_info *info)
- {
- if (info->dp->ds == ds && ds->ops->port_lag_join)
- return ds->ops->port_lag_join(ds, info->dp->index, info->lag,
- info->info, info->extack);
- if (info->dp->ds != ds && ds->ops->crosschip_lag_join)
- return ds->ops->crosschip_lag_join(ds, info->dp->ds->index,
- info->dp->index, info->lag,
- info->info, info->extack);
- return -EOPNOTSUPP;
- }
- static int dsa_switch_lag_leave(struct dsa_switch *ds,
- struct dsa_notifier_lag_info *info)
- {
- if (info->dp->ds == ds && ds->ops->port_lag_leave)
- return ds->ops->port_lag_leave(ds, info->dp->index, info->lag);
- if (info->dp->ds != ds && ds->ops->crosschip_lag_leave)
- return ds->ops->crosschip_lag_leave(ds, info->dp->ds->index,
- info->dp->index, info->lag);
- return -EOPNOTSUPP;
- }
- static int dsa_switch_mdb_add(struct dsa_switch *ds,
- struct dsa_notifier_mdb_info *info)
- {
- int port = dsa_towards_port(ds, info->dp->ds->index, info->dp->index);
- struct dsa_port *dp = dsa_to_port(ds, port);
- if (!ds->ops->port_mdb_add)
- return -EOPNOTSUPP;
- return dsa_port_do_mdb_add(dp, info->mdb, info->db);
- }
- static int dsa_switch_mdb_del(struct dsa_switch *ds,
- struct dsa_notifier_mdb_info *info)
- {
- int port = dsa_towards_port(ds, info->dp->ds->index, info->dp->index);
- struct dsa_port *dp = dsa_to_port(ds, port);
- if (!ds->ops->port_mdb_del)
- return -EOPNOTSUPP;
- return dsa_port_do_mdb_del(dp, info->mdb, info->db);
- }
- static int dsa_switch_host_mdb_add(struct dsa_switch *ds,
- struct dsa_notifier_mdb_info *info)
- {
- struct dsa_port *dp;
- int err = 0;
- if (!ds->ops->port_mdb_add)
- return -EOPNOTSUPP;
- dsa_switch_for_each_port(dp, ds) {
- if (dsa_port_host_address_match(dp, info->dp)) {
- err = dsa_port_do_mdb_add(dp, info->mdb, info->db);
- if (err)
- break;
- }
- }
- return err;
- }
- static int dsa_switch_host_mdb_del(struct dsa_switch *ds,
- struct dsa_notifier_mdb_info *info)
- {
- struct dsa_port *dp;
- int err = 0;
- if (!ds->ops->port_mdb_del)
- return -EOPNOTSUPP;
- dsa_switch_for_each_port(dp, ds) {
- if (dsa_port_host_address_match(dp, info->dp)) {
- err = dsa_port_do_mdb_del(dp, info->mdb, info->db);
- if (err)
- break;
- }
- }
- return err;
- }
- /* Port VLANs match on the targeted port and on all DSA ports */
- static bool dsa_port_vlan_match(struct dsa_port *dp,
- struct dsa_notifier_vlan_info *info)
- {
- return dsa_port_is_dsa(dp) || dp == info->dp;
- }
- /* Host VLANs match on the targeted port's CPU port, and on all DSA ports
- * (upstream and downstream) of that switch and its upstream switches.
- */
- static bool dsa_port_host_vlan_match(struct dsa_port *dp,
- const struct dsa_port *targeted_dp)
- {
- struct dsa_port *cpu_dp = targeted_dp->cpu_dp;
- if (dsa_switch_is_upstream_of(dp->ds, targeted_dp->ds))
- return dsa_port_is_dsa(dp) || dp == cpu_dp;
- return false;
- }
- static struct dsa_vlan *dsa_vlan_find(struct list_head *vlan_list,
- const struct switchdev_obj_port_vlan *vlan)
- {
- struct dsa_vlan *v;
- list_for_each_entry(v, vlan_list, list)
- if (v->vid == vlan->vid)
- return v;
- return NULL;
- }
- static int dsa_port_do_vlan_add(struct dsa_port *dp,
- const struct switchdev_obj_port_vlan *vlan,
- struct netlink_ext_ack *extack)
- {
- struct dsa_switch *ds = dp->ds;
- int port = dp->index;
- struct dsa_vlan *v;
- int err = 0;
- /* No need to bother with refcounting for user ports. */
- if (!(dsa_port_is_cpu(dp) || dsa_port_is_dsa(dp)))
- return ds->ops->port_vlan_add(ds, port, vlan, extack);
- /* No need to propagate on shared ports the existing VLANs that were
- * re-notified after just the flags have changed. This would cause a
- * refcount bump which we need to avoid, since it unbalances the
- * additions with the deletions.
- */
- if (vlan->changed)
- return 0;
- mutex_lock(&dp->vlans_lock);
- v = dsa_vlan_find(&dp->vlans, vlan);
- if (v) {
- refcount_inc(&v->refcount);
- goto out;
- }
- v = kzalloc(sizeof(*v), GFP_KERNEL);
- if (!v) {
- err = -ENOMEM;
- goto out;
- }
- err = ds->ops->port_vlan_add(ds, port, vlan, extack);
- if (err) {
- kfree(v);
- goto out;
- }
- v->vid = vlan->vid;
- refcount_set(&v->refcount, 1);
- list_add_tail(&v->list, &dp->vlans);
- out:
- mutex_unlock(&dp->vlans_lock);
- return err;
- }
- static int dsa_port_do_vlan_del(struct dsa_port *dp,
- const struct switchdev_obj_port_vlan *vlan)
- {
- struct dsa_switch *ds = dp->ds;
- int port = dp->index;
- struct dsa_vlan *v;
- int err = 0;
- /* No need to bother with refcounting for user ports */
- if (!(dsa_port_is_cpu(dp) || dsa_port_is_dsa(dp)))
- return ds->ops->port_vlan_del(ds, port, vlan);
- mutex_lock(&dp->vlans_lock);
- v = dsa_vlan_find(&dp->vlans, vlan);
- if (!v) {
- err = -ENOENT;
- goto out;
- }
- if (!refcount_dec_and_test(&v->refcount))
- goto out;
- err = ds->ops->port_vlan_del(ds, port, vlan);
- if (err) {
- refcount_set(&v->refcount, 1);
- goto out;
- }
- list_del(&v->list);
- kfree(v);
- out:
- mutex_unlock(&dp->vlans_lock);
- return err;
- }
- static int dsa_switch_vlan_add(struct dsa_switch *ds,
- struct dsa_notifier_vlan_info *info)
- {
- struct dsa_port *dp;
- int err;
- if (!ds->ops->port_vlan_add)
- return -EOPNOTSUPP;
- dsa_switch_for_each_port(dp, ds) {
- if (dsa_port_vlan_match(dp, info)) {
- err = dsa_port_do_vlan_add(dp, info->vlan,
- info->extack);
- if (err)
- return err;
- }
- }
- return 0;
- }
- static int dsa_switch_vlan_del(struct dsa_switch *ds,
- struct dsa_notifier_vlan_info *info)
- {
- struct dsa_port *dp;
- int err;
- if (!ds->ops->port_vlan_del)
- return -EOPNOTSUPP;
- dsa_switch_for_each_port(dp, ds) {
- if (dsa_port_vlan_match(dp, info)) {
- err = dsa_port_do_vlan_del(dp, info->vlan);
- if (err)
- return err;
- }
- }
- return 0;
- }
- static int dsa_switch_host_vlan_add(struct dsa_switch *ds,
- struct dsa_notifier_vlan_info *info)
- {
- struct dsa_port *dp;
- int err;
- if (!ds->ops->port_vlan_add)
- return -EOPNOTSUPP;
- dsa_switch_for_each_port(dp, ds) {
- if (dsa_port_host_vlan_match(dp, info->dp)) {
- err = dsa_port_do_vlan_add(dp, info->vlan,
- info->extack);
- if (err)
- return err;
- }
- }
- return 0;
- }
- static int dsa_switch_host_vlan_del(struct dsa_switch *ds,
- struct dsa_notifier_vlan_info *info)
- {
- struct dsa_port *dp;
- int err;
- if (!ds->ops->port_vlan_del)
- return -EOPNOTSUPP;
- dsa_switch_for_each_port(dp, ds) {
- if (dsa_port_host_vlan_match(dp, info->dp)) {
- err = dsa_port_do_vlan_del(dp, info->vlan);
- if (err)
- return err;
- }
- }
- return 0;
- }
- static int dsa_switch_change_tag_proto(struct dsa_switch *ds,
- struct dsa_notifier_tag_proto_info *info)
- {
- const struct dsa_device_ops *tag_ops = info->tag_ops;
- struct dsa_port *dp, *cpu_dp;
- int err;
- if (!ds->ops->change_tag_protocol)
- return -EOPNOTSUPP;
- ASSERT_RTNL();
- err = ds->ops->change_tag_protocol(ds, tag_ops->proto);
- if (err)
- return err;
- dsa_switch_for_each_cpu_port(cpu_dp, ds)
- dsa_port_set_tag_protocol(cpu_dp, tag_ops);
- /* Now that changing the tag protocol can no longer fail, let's update
- * the remaining bits which are "duplicated for faster access", and the
- * bits that depend on the tagger, such as the MTU.
- */
- dsa_switch_for_each_user_port(dp, ds) {
- struct net_device *slave = dp->slave;
- dsa_slave_setup_tagger(slave);
- /* rtnl_mutex is held in dsa_tree_change_tag_proto */
- dsa_slave_change_mtu(slave, slave->mtu);
- }
- return 0;
- }
- /* We use the same cross-chip notifiers to inform both the tagger side, as well
- * as the switch side, of connection and disconnection events.
- * Since ds->tagger_data is owned by the tagger, it isn't a hard error if the
- * switch side doesn't support connecting to this tagger, and therefore, the
- * fact that we don't disconnect the tagger side doesn't constitute a memory
- * leak: the tagger will still operate with persistent per-switch memory, just
- * with the switch side unconnected to it. What does constitute a hard error is
- * when the switch side supports connecting but fails.
- */
- static int
- dsa_switch_connect_tag_proto(struct dsa_switch *ds,
- struct dsa_notifier_tag_proto_info *info)
- {
- const struct dsa_device_ops *tag_ops = info->tag_ops;
- int err;
- /* Notify the new tagger about the connection to this switch */
- if (tag_ops->connect) {
- err = tag_ops->connect(ds);
- if (err)
- return err;
- }
- if (!ds->ops->connect_tag_protocol)
- return -EOPNOTSUPP;
- /* Notify the switch about the connection to the new tagger */
- err = ds->ops->connect_tag_protocol(ds, tag_ops->proto);
- if (err) {
- /* Revert the new tagger's connection to this tree */
- if (tag_ops->disconnect)
- tag_ops->disconnect(ds);
- return err;
- }
- return 0;
- }
- static int
- dsa_switch_disconnect_tag_proto(struct dsa_switch *ds,
- struct dsa_notifier_tag_proto_info *info)
- {
- const struct dsa_device_ops *tag_ops = info->tag_ops;
- /* Notify the tagger about the disconnection from this switch */
- if (tag_ops->disconnect && ds->tagger_data)
- tag_ops->disconnect(ds);
- /* No need to notify the switch, since it shouldn't have any
- * resources to tear down
- */
- return 0;
- }
- static int
- dsa_switch_master_state_change(struct dsa_switch *ds,
- struct dsa_notifier_master_state_info *info)
- {
- if (!ds->ops->master_state_change)
- return 0;
- ds->ops->master_state_change(ds, info->master, info->operational);
- return 0;
- }
- static int dsa_switch_event(struct notifier_block *nb,
- unsigned long event, void *info)
- {
- struct dsa_switch *ds = container_of(nb, struct dsa_switch, nb);
- int err;
- switch (event) {
- case DSA_NOTIFIER_AGEING_TIME:
- err = dsa_switch_ageing_time(ds, info);
- break;
- case DSA_NOTIFIER_BRIDGE_JOIN:
- err = dsa_switch_bridge_join(ds, info);
- break;
- case DSA_NOTIFIER_BRIDGE_LEAVE:
- err = dsa_switch_bridge_leave(ds, info);
- break;
- case DSA_NOTIFIER_FDB_ADD:
- err = dsa_switch_fdb_add(ds, info);
- break;
- case DSA_NOTIFIER_FDB_DEL:
- err = dsa_switch_fdb_del(ds, info);
- break;
- case DSA_NOTIFIER_HOST_FDB_ADD:
- err = dsa_switch_host_fdb_add(ds, info);
- break;
- case DSA_NOTIFIER_HOST_FDB_DEL:
- err = dsa_switch_host_fdb_del(ds, info);
- break;
- case DSA_NOTIFIER_LAG_FDB_ADD:
- err = dsa_switch_lag_fdb_add(ds, info);
- break;
- case DSA_NOTIFIER_LAG_FDB_DEL:
- err = dsa_switch_lag_fdb_del(ds, info);
- break;
- case DSA_NOTIFIER_LAG_CHANGE:
- err = dsa_switch_lag_change(ds, info);
- break;
- case DSA_NOTIFIER_LAG_JOIN:
- err = dsa_switch_lag_join(ds, info);
- break;
- case DSA_NOTIFIER_LAG_LEAVE:
- err = dsa_switch_lag_leave(ds, info);
- break;
- case DSA_NOTIFIER_MDB_ADD:
- err = dsa_switch_mdb_add(ds, info);
- break;
- case DSA_NOTIFIER_MDB_DEL:
- err = dsa_switch_mdb_del(ds, info);
- break;
- case DSA_NOTIFIER_HOST_MDB_ADD:
- err = dsa_switch_host_mdb_add(ds, info);
- break;
- case DSA_NOTIFIER_HOST_MDB_DEL:
- err = dsa_switch_host_mdb_del(ds, info);
- break;
- case DSA_NOTIFIER_VLAN_ADD:
- err = dsa_switch_vlan_add(ds, info);
- break;
- case DSA_NOTIFIER_VLAN_DEL:
- err = dsa_switch_vlan_del(ds, info);
- break;
- case DSA_NOTIFIER_HOST_VLAN_ADD:
- err = dsa_switch_host_vlan_add(ds, info);
- break;
- case DSA_NOTIFIER_HOST_VLAN_DEL:
- err = dsa_switch_host_vlan_del(ds, info);
- break;
- case DSA_NOTIFIER_MTU:
- err = dsa_switch_mtu(ds, info);
- break;
- case DSA_NOTIFIER_TAG_PROTO:
- err = dsa_switch_change_tag_proto(ds, info);
- break;
- case DSA_NOTIFIER_TAG_PROTO_CONNECT:
- err = dsa_switch_connect_tag_proto(ds, info);
- break;
- case DSA_NOTIFIER_TAG_PROTO_DISCONNECT:
- err = dsa_switch_disconnect_tag_proto(ds, info);
- break;
- case DSA_NOTIFIER_TAG_8021Q_VLAN_ADD:
- err = dsa_switch_tag_8021q_vlan_add(ds, info);
- break;
- case DSA_NOTIFIER_TAG_8021Q_VLAN_DEL:
- err = dsa_switch_tag_8021q_vlan_del(ds, info);
- break;
- case DSA_NOTIFIER_MASTER_STATE_CHANGE:
- err = dsa_switch_master_state_change(ds, info);
- break;
- default:
- err = -EOPNOTSUPP;
- break;
- }
- if (err)
- dev_dbg(ds->dev, "breaking chain for DSA event %lu (%d)\n",
- event, err);
- return notifier_from_errno(err);
- }
- int dsa_switch_register_notifier(struct dsa_switch *ds)
- {
- ds->nb.notifier_call = dsa_switch_event;
- return raw_notifier_chain_register(&ds->dst->nh, &ds->nb);
- }
- void dsa_switch_unregister_notifier(struct dsa_switch *ds)
- {
- int err;
- err = raw_notifier_chain_unregister(&ds->dst->nh, &ds->nb);
- if (err)
- dev_err(ds->dev, "failed to unregister notifier (%d)\n", err);
- }
|