net: dsa: support switchdev ageing time attr
Add a new function for DSA drivers to handle the switchdev SWITCHDEV_ATTR_ID_BRIDGE_AGEING_TIME attribute. The ageing time is passed as milliseconds. Also because we can have multiple logical bridges on top of a physical switch and ageing time are switch-wide, call the driver function with the fastest ageing time in use on the chip instead of the requested one. Signed-off-by: Vivien Didelot <vivien.didelot@savoirfairelinux.com> Reviewed-by: Andrew Lunn <andrew@lunn.ch> Signed-off-by: David S. Miller <davem@davemloft.net>
This commit is contained in:
parent
8ec61c7f7c
commit
34a79f63bb
@ -141,6 +141,7 @@ struct dsa_switch_tree {
|
|||||||
struct dsa_port {
|
struct dsa_port {
|
||||||
struct net_device *netdev;
|
struct net_device *netdev;
|
||||||
struct device_node *dn;
|
struct device_node *dn;
|
||||||
|
unsigned int ageing_time;
|
||||||
};
|
};
|
||||||
|
|
||||||
struct dsa_switch {
|
struct dsa_switch {
|
||||||
@ -329,6 +330,7 @@ struct dsa_switch_driver {
|
|||||||
/*
|
/*
|
||||||
* Bridge integration
|
* Bridge integration
|
||||||
*/
|
*/
|
||||||
|
int (*set_ageing_time)(struct dsa_switch *ds, unsigned int msecs);
|
||||||
int (*port_bridge_join)(struct dsa_switch *ds, int port,
|
int (*port_bridge_join)(struct dsa_switch *ds, int port,
|
||||||
struct net_device *bridge);
|
struct net_device *bridge);
|
||||||
void (*port_bridge_leave)(struct dsa_switch *ds, int port);
|
void (*port_bridge_leave)(struct dsa_switch *ds, int port);
|
||||||
|
@ -333,6 +333,44 @@ static int dsa_slave_vlan_filtering(struct net_device *dev,
|
|||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static int dsa_fastest_ageing_time(struct dsa_switch *ds,
|
||||||
|
unsigned int ageing_time)
|
||||||
|
{
|
||||||
|
int i;
|
||||||
|
|
||||||
|
for (i = 0; i < DSA_MAX_PORTS; ++i) {
|
||||||
|
struct dsa_port *dp = &ds->ports[i];
|
||||||
|
|
||||||
|
if (dp && dp->ageing_time && dp->ageing_time < ageing_time)
|
||||||
|
ageing_time = dp->ageing_time;
|
||||||
|
}
|
||||||
|
|
||||||
|
return ageing_time;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int dsa_slave_ageing_time(struct net_device *dev,
|
||||||
|
const struct switchdev_attr *attr,
|
||||||
|
struct switchdev_trans *trans)
|
||||||
|
{
|
||||||
|
struct dsa_slave_priv *p = netdev_priv(dev);
|
||||||
|
struct dsa_switch *ds = p->parent;
|
||||||
|
unsigned long ageing_jiffies = clock_t_to_jiffies(attr->u.ageing_time);
|
||||||
|
unsigned int ageing_time = jiffies_to_msecs(ageing_jiffies);
|
||||||
|
|
||||||
|
/* bridge skips -EOPNOTSUPP, so skip the prepare phase */
|
||||||
|
if (switchdev_trans_ph_prepare(trans))
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
/* Keep the fastest ageing time in case of multiple bridges */
|
||||||
|
ds->ports[p->port].ageing_time = ageing_time;
|
||||||
|
ageing_time = dsa_fastest_ageing_time(ds, ageing_time);
|
||||||
|
|
||||||
|
if (ds->drv->set_ageing_time)
|
||||||
|
return ds->drv->set_ageing_time(ds, ageing_time);
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
static int dsa_slave_port_attr_set(struct net_device *dev,
|
static int dsa_slave_port_attr_set(struct net_device *dev,
|
||||||
const struct switchdev_attr *attr,
|
const struct switchdev_attr *attr,
|
||||||
struct switchdev_trans *trans)
|
struct switchdev_trans *trans)
|
||||||
@ -346,6 +384,9 @@ static int dsa_slave_port_attr_set(struct net_device *dev,
|
|||||||
case SWITCHDEV_ATTR_ID_BRIDGE_VLAN_FILTERING:
|
case SWITCHDEV_ATTR_ID_BRIDGE_VLAN_FILTERING:
|
||||||
ret = dsa_slave_vlan_filtering(dev, attr, trans);
|
ret = dsa_slave_vlan_filtering(dev, attr, trans);
|
||||||
break;
|
break;
|
||||||
|
case SWITCHDEV_ATTR_ID_BRIDGE_AGEING_TIME:
|
||||||
|
ret = dsa_slave_ageing_time(dev, attr, trans);
|
||||||
|
break;
|
||||||
default:
|
default:
|
||||||
ret = -EOPNOTSUPP;
|
ret = -EOPNOTSUPP;
|
||||||
break;
|
break;
|
||||||
|
Loading…
Reference in New Issue
Block a user