diff options
Diffstat (limited to 'net/switchdev/switchdev.c')
| -rw-r--r-- | net/switchdev/switchdev.c | 73 | 
1 files changed, 73 insertions, 0 deletions
| diff --git a/net/switchdev/switchdev.c b/net/switchdev/switchdev.c index 5b045284849e..c9189a970eec 100644 --- a/net/switchdev/switchdev.c +++ b/net/switchdev/switchdev.c @@ -19,6 +19,35 @@  #include <linux/rtnetlink.h>  #include <net/switchdev.h> +static bool switchdev_obj_eq(const struct switchdev_obj *a, +			     const struct switchdev_obj *b) +{ +	const struct switchdev_obj_port_vlan *va, *vb; +	const struct switchdev_obj_port_mdb *ma, *mb; + +	if (a->id != b->id || a->orig_dev != b->orig_dev) +		return false; + +	switch (a->id) { +	case SWITCHDEV_OBJ_ID_PORT_VLAN: +		va = SWITCHDEV_OBJ_PORT_VLAN(a); +		vb = SWITCHDEV_OBJ_PORT_VLAN(b); +		return va->flags == vb->flags && +			va->vid == vb->vid && +			va->changed == vb->changed; +	case SWITCHDEV_OBJ_ID_PORT_MDB: +	case SWITCHDEV_OBJ_ID_HOST_MDB: +		ma = SWITCHDEV_OBJ_PORT_MDB(a); +		mb = SWITCHDEV_OBJ_PORT_MDB(b); +		return ma->vid == mb->vid && +			ether_addr_equal(ma->addr, mb->addr); +	default: +		break; +	} + +	BUG(); +} +  static LIST_HEAD(deferred);  static DEFINE_SPINLOCK(deferred_lock); @@ -307,6 +336,50 @@ int switchdev_port_obj_del(struct net_device *dev,  }  EXPORT_SYMBOL_GPL(switchdev_port_obj_del); +/** + *	switchdev_port_obj_act_is_deferred - Is object action pending? + * + *	@dev: port device + *	@nt: type of action; add or delete + *	@obj: object to test + * + *	Returns true if a deferred item is pending, which is + *	equivalent to the action @nt on an object @obj. + * + *	rtnl_lock must be held. + */ +bool switchdev_port_obj_act_is_deferred(struct net_device *dev, +					enum switchdev_notifier_type nt, +					const struct switchdev_obj *obj) +{ +	struct switchdev_deferred_item *dfitem; +	bool found = false; + +	ASSERT_RTNL(); + +	spin_lock_bh(&deferred_lock); + +	list_for_each_entry(dfitem, &deferred, list) { +		if (dfitem->dev != dev) +			continue; + +		if ((dfitem->func == switchdev_port_obj_add_deferred && +		     nt == SWITCHDEV_PORT_OBJ_ADD) || +		    (dfitem->func == switchdev_port_obj_del_deferred && +		     nt == SWITCHDEV_PORT_OBJ_DEL)) { +			if (switchdev_obj_eq((const void *)dfitem->data, obj)) { +				found = true; +				break; +			} +		} +	} + +	spin_unlock_bh(&deferred_lock); + +	return found; +} +EXPORT_SYMBOL_GPL(switchdev_port_obj_act_is_deferred); +  static ATOMIC_NOTIFIER_HEAD(switchdev_notif_chain);  static BLOCKING_NOTIFIER_HEAD(switchdev_blocking_notif_chain); | 
