Merge branch 'sf2_hwbridge'
Florian Fainelli says: ==================== net: dsa: integration with SWITCHDEV for HW bridging This patch set provides the DSA and SWITCHDEV integration bits together and modifies the bcm_sf2 driver accordingly such that it works properly with HW bridging. Changes in v3: - add back the null pointer check in dsa_slave_br_port_mask from Guenter - slightly rework patch 1 commit message not to mention the function name we add in patch 2 Changes in v2: - avoid a race condition in how DSA network devices are created, patch from Guenter Roeck - provide a consistent and work STP state once a port leaves the bridge - retain a bridge device pointer to properly flag port/bridge membership - properly flush the ARL (Address Resolution Logic) in bcm_sf2.c - properly retain port membership when individually bringing devices up/down while they are members of a bridge We discussed on the mailing-list the possibility of standardizing a "fdb_flush" operation for DSA switch drivers, looking at the Marvell and Broadcom switches, I am not convinced this is practical or diserable as the terminologies vary here, but there is nothing preventing us from doing it later. Many thanks to Guenter and Andrew for both testing and providing feedback. ==================== Signed-off-by: David S. Miller <davem@davemloft.net>
This commit is contained in:
commit
bb66be1c54
@ -23,6 +23,7 @@
|
|||||||
#include <linux/of_address.h>
|
#include <linux/of_address.h>
|
||||||
#include <net/dsa.h>
|
#include <net/dsa.h>
|
||||||
#include <linux/ethtool.h>
|
#include <linux/ethtool.h>
|
||||||
|
#include <linux/if_bridge.h>
|
||||||
|
|
||||||
#include "bcm_sf2.h"
|
#include "bcm_sf2.h"
|
||||||
#include "bcm_sf2_regs.h"
|
#include "bcm_sf2_regs.h"
|
||||||
@ -299,10 +300,14 @@ static int bcm_sf2_port_setup(struct dsa_switch *ds, int port,
|
|||||||
if (port == 7)
|
if (port == 7)
|
||||||
intrl2_1_mask_clear(priv, P_IRQ_MASK(P7_IRQ_OFF));
|
intrl2_1_mask_clear(priv, P_IRQ_MASK(P7_IRQ_OFF));
|
||||||
|
|
||||||
/* Set this port, and only this one to be in the default VLAN */
|
/* Set this port, and only this one to be in the default VLAN,
|
||||||
|
* if member of a bridge, restore its membership prior to
|
||||||
|
* bringing down this port.
|
||||||
|
*/
|
||||||
reg = core_readl(priv, CORE_PORT_VLAN_CTL_PORT(port));
|
reg = core_readl(priv, CORE_PORT_VLAN_CTL_PORT(port));
|
||||||
reg &= ~PORT_VLAN_CTRL_MASK;
|
reg &= ~PORT_VLAN_CTRL_MASK;
|
||||||
reg |= (1 << port);
|
reg |= (1 << port);
|
||||||
|
reg |= priv->port_sts[port].vlan_ctl_mask;
|
||||||
core_writel(priv, reg, CORE_PORT_VLAN_CTL_PORT(port));
|
core_writel(priv, reg, CORE_PORT_VLAN_CTL_PORT(port));
|
||||||
|
|
||||||
bcm_sf2_imp_vlan_setup(ds, cpu_port);
|
bcm_sf2_imp_vlan_setup(ds, cpu_port);
|
||||||
@ -400,6 +405,151 @@ static int bcm_sf2_sw_set_eee(struct dsa_switch *ds, int port,
|
|||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* Fast-ageing of ARL entries for a given port, equivalent to an ARL
|
||||||
|
* flush for that port.
|
||||||
|
*/
|
||||||
|
static int bcm_sf2_sw_fast_age_port(struct dsa_switch *ds, int port)
|
||||||
|
{
|
||||||
|
struct bcm_sf2_priv *priv = ds_to_priv(ds);
|
||||||
|
unsigned int timeout = 1000;
|
||||||
|
u32 reg;
|
||||||
|
|
||||||
|
core_writel(priv, port, CORE_FAST_AGE_PORT);
|
||||||
|
|
||||||
|
reg = core_readl(priv, CORE_FAST_AGE_CTRL);
|
||||||
|
reg |= EN_AGE_PORT | FAST_AGE_STR_DONE;
|
||||||
|
core_writel(priv, reg, CORE_FAST_AGE_CTRL);
|
||||||
|
|
||||||
|
do {
|
||||||
|
reg = core_readl(priv, CORE_FAST_AGE_CTRL);
|
||||||
|
if (!(reg & FAST_AGE_STR_DONE))
|
||||||
|
break;
|
||||||
|
|
||||||
|
cpu_relax();
|
||||||
|
} while (timeout--);
|
||||||
|
|
||||||
|
if (!timeout)
|
||||||
|
return -ETIMEDOUT;
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int bcm_sf2_sw_br_join(struct dsa_switch *ds, int port,
|
||||||
|
u32 br_port_mask)
|
||||||
|
{
|
||||||
|
struct bcm_sf2_priv *priv = ds_to_priv(ds);
|
||||||
|
unsigned int i;
|
||||||
|
u32 reg, p_ctl;
|
||||||
|
|
||||||
|
p_ctl = core_readl(priv, CORE_PORT_VLAN_CTL_PORT(port));
|
||||||
|
|
||||||
|
for (i = 0; i < priv->hw_params.num_ports; i++) {
|
||||||
|
if (!((1 << i) & br_port_mask))
|
||||||
|
continue;
|
||||||
|
|
||||||
|
/* Add this local port to the remote port VLAN control
|
||||||
|
* membership and update the remote port bitmask
|
||||||
|
*/
|
||||||
|
reg = core_readl(priv, CORE_PORT_VLAN_CTL_PORT(i));
|
||||||
|
reg |= 1 << port;
|
||||||
|
core_writel(priv, reg, CORE_PORT_VLAN_CTL_PORT(i));
|
||||||
|
priv->port_sts[i].vlan_ctl_mask = reg;
|
||||||
|
|
||||||
|
p_ctl |= 1 << i;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Configure the local port VLAN control membership to include
|
||||||
|
* remote ports and update the local port bitmask
|
||||||
|
*/
|
||||||
|
core_writel(priv, p_ctl, CORE_PORT_VLAN_CTL_PORT(port));
|
||||||
|
priv->port_sts[port].vlan_ctl_mask = p_ctl;
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int bcm_sf2_sw_br_leave(struct dsa_switch *ds, int port,
|
||||||
|
u32 br_port_mask)
|
||||||
|
{
|
||||||
|
struct bcm_sf2_priv *priv = ds_to_priv(ds);
|
||||||
|
unsigned int i;
|
||||||
|
u32 reg, p_ctl;
|
||||||
|
|
||||||
|
p_ctl = core_readl(priv, CORE_PORT_VLAN_CTL_PORT(port));
|
||||||
|
|
||||||
|
for (i = 0; i < priv->hw_params.num_ports; i++) {
|
||||||
|
/* Don't touch the remaining ports */
|
||||||
|
if (!((1 << i) & br_port_mask))
|
||||||
|
continue;
|
||||||
|
|
||||||
|
reg = core_readl(priv, CORE_PORT_VLAN_CTL_PORT(i));
|
||||||
|
reg &= ~(1 << port);
|
||||||
|
core_writel(priv, reg, CORE_PORT_VLAN_CTL_PORT(i));
|
||||||
|
priv->port_sts[port].vlan_ctl_mask = reg;
|
||||||
|
|
||||||
|
/* Prevent self removal to preserve isolation */
|
||||||
|
if (port != i)
|
||||||
|
p_ctl &= ~(1 << i);
|
||||||
|
}
|
||||||
|
|
||||||
|
core_writel(priv, p_ctl, CORE_PORT_VLAN_CTL_PORT(port));
|
||||||
|
priv->port_sts[port].vlan_ctl_mask = p_ctl;
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int bcm_sf2_sw_br_set_stp_state(struct dsa_switch *ds, int port,
|
||||||
|
u8 state)
|
||||||
|
{
|
||||||
|
struct bcm_sf2_priv *priv = ds_to_priv(ds);
|
||||||
|
u8 hw_state, cur_hw_state;
|
||||||
|
int ret = 0;
|
||||||
|
u32 reg;
|
||||||
|
|
||||||
|
reg = core_readl(priv, CORE_G_PCTL_PORT(port));
|
||||||
|
cur_hw_state = reg >> G_MISTP_STATE_SHIFT;
|
||||||
|
|
||||||
|
switch (state) {
|
||||||
|
case BR_STATE_DISABLED:
|
||||||
|
hw_state = G_MISTP_DIS_STATE;
|
||||||
|
break;
|
||||||
|
case BR_STATE_LISTENING:
|
||||||
|
hw_state = G_MISTP_LISTEN_STATE;
|
||||||
|
break;
|
||||||
|
case BR_STATE_LEARNING:
|
||||||
|
hw_state = G_MISTP_LEARN_STATE;
|
||||||
|
break;
|
||||||
|
case BR_STATE_FORWARDING:
|
||||||
|
hw_state = G_MISTP_FWD_STATE;
|
||||||
|
break;
|
||||||
|
case BR_STATE_BLOCKING:
|
||||||
|
hw_state = G_MISTP_BLOCK_STATE;
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
pr_err("%s: invalid STP state: %d\n", __func__, state);
|
||||||
|
return -EINVAL;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Fast-age ARL entries if we are moving a port from Learning or
|
||||||
|
* Forwarding state to Disabled, Blocking or Listening state
|
||||||
|
*/
|
||||||
|
if (cur_hw_state != hw_state) {
|
||||||
|
if (cur_hw_state & 4 && !(hw_state & 4)) {
|
||||||
|
ret = bcm_sf2_sw_fast_age_port(ds, port);
|
||||||
|
if (ret) {
|
||||||
|
pr_err("%s: fast-ageing failed\n", __func__);
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
reg = core_readl(priv, CORE_G_PCTL_PORT(port));
|
||||||
|
reg &= ~(G_MISTP_STATE_MASK << G_MISTP_STATE_SHIFT);
|
||||||
|
reg |= hw_state;
|
||||||
|
core_writel(priv, reg, CORE_G_PCTL_PORT(port));
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
static irqreturn_t bcm_sf2_switch_0_isr(int irq, void *dev_id)
|
static irqreturn_t bcm_sf2_switch_0_isr(int irq, void *dev_id)
|
||||||
{
|
{
|
||||||
struct bcm_sf2_priv *priv = dev_id;
|
struct bcm_sf2_priv *priv = dev_id;
|
||||||
@ -916,6 +1066,9 @@ static struct dsa_switch_driver bcm_sf2_switch_driver = {
|
|||||||
.port_disable = bcm_sf2_port_disable,
|
.port_disable = bcm_sf2_port_disable,
|
||||||
.get_eee = bcm_sf2_sw_get_eee,
|
.get_eee = bcm_sf2_sw_get_eee,
|
||||||
.set_eee = bcm_sf2_sw_set_eee,
|
.set_eee = bcm_sf2_sw_set_eee,
|
||||||
|
.port_join_bridge = bcm_sf2_sw_br_join,
|
||||||
|
.port_leave_bridge = bcm_sf2_sw_br_leave,
|
||||||
|
.port_stp_update = bcm_sf2_sw_br_set_stp_state,
|
||||||
};
|
};
|
||||||
|
|
||||||
static int __init bcm_sf2_init(void)
|
static int __init bcm_sf2_init(void)
|
||||||
|
@ -46,6 +46,8 @@ struct bcm_sf2_port_status {
|
|||||||
unsigned int link;
|
unsigned int link;
|
||||||
|
|
||||||
struct ethtool_eee eee;
|
struct ethtool_eee eee;
|
||||||
|
|
||||||
|
u32 vlan_ctl_mask;
|
||||||
};
|
};
|
||||||
|
|
||||||
struct bcm_sf2_priv {
|
struct bcm_sf2_priv {
|
||||||
|
@ -163,6 +163,21 @@
|
|||||||
#define EN_CHIP_RST (1 << 6)
|
#define EN_CHIP_RST (1 << 6)
|
||||||
#define EN_SW_RESET (1 << 4)
|
#define EN_SW_RESET (1 << 4)
|
||||||
|
|
||||||
|
#define CORE_FAST_AGE_CTRL 0x00220
|
||||||
|
#define EN_FAST_AGE_STATIC (1 << 0)
|
||||||
|
#define EN_AGE_DYNAMIC (1 << 1)
|
||||||
|
#define EN_AGE_PORT (1 << 2)
|
||||||
|
#define EN_AGE_VLAN (1 << 3)
|
||||||
|
#define EN_AGE_SPT (1 << 4)
|
||||||
|
#define EN_AGE_MCAST (1 << 5)
|
||||||
|
#define FAST_AGE_STR_DONE (1 << 7)
|
||||||
|
|
||||||
|
#define CORE_FAST_AGE_PORT 0x00224
|
||||||
|
#define AGE_PORT_MASK 0xf
|
||||||
|
|
||||||
|
#define CORE_FAST_AGE_VID 0x00228
|
||||||
|
#define AGE_VID_MASK 0x3fff
|
||||||
|
|
||||||
#define CORE_LNKSTS 0x00400
|
#define CORE_LNKSTS 0x00400
|
||||||
#define LNK_STS_MASK 0x1ff
|
#define LNK_STS_MASK 0x1ff
|
||||||
|
|
||||||
|
@ -275,6 +275,16 @@ struct dsa_switch_driver {
|
|||||||
int (*get_regs_len)(struct dsa_switch *ds, int port);
|
int (*get_regs_len)(struct dsa_switch *ds, int port);
|
||||||
void (*get_regs)(struct dsa_switch *ds, int port,
|
void (*get_regs)(struct dsa_switch *ds, int port,
|
||||||
struct ethtool_regs *regs, void *p);
|
struct ethtool_regs *regs, void *p);
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Bridge integration
|
||||||
|
*/
|
||||||
|
int (*port_join_bridge)(struct dsa_switch *ds, int port,
|
||||||
|
u32 br_port_mask);
|
||||||
|
int (*port_leave_bridge)(struct dsa_switch *ds, int port,
|
||||||
|
u32 br_port_mask);
|
||||||
|
int (*port_stp_update)(struct dsa_switch *ds, int port,
|
||||||
|
u8 state);
|
||||||
};
|
};
|
||||||
|
|
||||||
void register_switch_driver(struct dsa_switch_driver *type);
|
void register_switch_driver(struct dsa_switch_driver *type);
|
||||||
|
@ -8,6 +8,7 @@ config NET_DSA
|
|||||||
tristate
|
tristate
|
||||||
depends on HAVE_NET_DSA
|
depends on HAVE_NET_DSA
|
||||||
select PHYLIB
|
select PHYLIB
|
||||||
|
select NET_SWITCHDEV
|
||||||
|
|
||||||
if NET_DSA
|
if NET_DSA
|
||||||
|
|
||||||
|
@ -314,19 +314,15 @@ dsa_switch_setup(struct dsa_switch_tree *dst, int index,
|
|||||||
* Create network devices for physical switch ports.
|
* Create network devices for physical switch ports.
|
||||||
*/
|
*/
|
||||||
for (i = 0; i < DSA_MAX_PORTS; i++) {
|
for (i = 0; i < DSA_MAX_PORTS; i++) {
|
||||||
struct net_device *slave_dev;
|
|
||||||
|
|
||||||
if (!(ds->phys_port_mask & (1 << i)))
|
if (!(ds->phys_port_mask & (1 << i)))
|
||||||
continue;
|
continue;
|
||||||
|
|
||||||
slave_dev = dsa_slave_create(ds, parent, i, pd->port_names[i]);
|
ret = dsa_slave_create(ds, parent, i, pd->port_names[i]);
|
||||||
if (slave_dev == NULL) {
|
if (ret < 0) {
|
||||||
netdev_err(dst->master_netdev, "[%d]: can't create dsa slave device for port %d(%s)\n",
|
netdev_err(dst->master_netdev, "[%d]: can't create dsa slave device for port %d(%s)\n",
|
||||||
index, i, pd->port_names[i]);
|
index, i, pd->port_names[i]);
|
||||||
continue;
|
ret = 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
ds->ports[i] = slave_dev;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
#ifdef CONFIG_NET_DSA_HWMON
|
#ifdef CONFIG_NET_DSA_HWMON
|
||||||
@ -830,6 +826,10 @@ static struct packet_type dsa_pack_type __read_mostly = {
|
|||||||
.func = dsa_switch_rcv,
|
.func = dsa_switch_rcv,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
static struct notifier_block dsa_netdevice_nb __read_mostly = {
|
||||||
|
.notifier_call = dsa_slave_netdevice_event,
|
||||||
|
};
|
||||||
|
|
||||||
#ifdef CONFIG_PM_SLEEP
|
#ifdef CONFIG_PM_SLEEP
|
||||||
static int dsa_suspend(struct device *d)
|
static int dsa_suspend(struct device *d)
|
||||||
{
|
{
|
||||||
@ -888,6 +888,8 @@ static int __init dsa_init_module(void)
|
|||||||
{
|
{
|
||||||
int rc;
|
int rc;
|
||||||
|
|
||||||
|
register_netdevice_notifier(&dsa_netdevice_nb);
|
||||||
|
|
||||||
rc = platform_driver_register(&dsa_driver);
|
rc = platform_driver_register(&dsa_driver);
|
||||||
if (rc)
|
if (rc)
|
||||||
return rc;
|
return rc;
|
||||||
@ -900,6 +902,7 @@ module_init(dsa_init_module);
|
|||||||
|
|
||||||
static void __exit dsa_cleanup_module(void)
|
static void __exit dsa_cleanup_module(void)
|
||||||
{
|
{
|
||||||
|
unregister_netdevice_notifier(&dsa_netdevice_nb);
|
||||||
dev_remove_pack(&dsa_pack_type);
|
dev_remove_pack(&dsa_pack_type);
|
||||||
platform_driver_unregister(&dsa_driver);
|
platform_driver_unregister(&dsa_driver);
|
||||||
}
|
}
|
||||||
|
@ -45,6 +45,8 @@ struct dsa_slave_priv {
|
|||||||
int old_link;
|
int old_link;
|
||||||
int old_pause;
|
int old_pause;
|
||||||
int old_duplex;
|
int old_duplex;
|
||||||
|
|
||||||
|
struct net_device *bridge_dev;
|
||||||
};
|
};
|
||||||
|
|
||||||
/* dsa.c */
|
/* dsa.c */
|
||||||
@ -53,11 +55,12 @@ extern char dsa_driver_version[];
|
|||||||
/* slave.c */
|
/* slave.c */
|
||||||
extern const struct dsa_device_ops notag_netdev_ops;
|
extern const struct dsa_device_ops notag_netdev_ops;
|
||||||
void dsa_slave_mii_bus_init(struct dsa_switch *ds);
|
void dsa_slave_mii_bus_init(struct dsa_switch *ds);
|
||||||
struct net_device *dsa_slave_create(struct dsa_switch *ds,
|
int dsa_slave_create(struct dsa_switch *ds, struct device *parent,
|
||||||
struct device *parent,
|
int port, char *name);
|
||||||
int port, char *name);
|
|
||||||
int dsa_slave_suspend(struct net_device *slave_dev);
|
int dsa_slave_suspend(struct net_device *slave_dev);
|
||||||
int dsa_slave_resume(struct net_device *slave_dev);
|
int dsa_slave_resume(struct net_device *slave_dev);
|
||||||
|
int dsa_slave_netdevice_event(struct notifier_block *unused,
|
||||||
|
unsigned long event, void *ptr);
|
||||||
|
|
||||||
/* tag_dsa.c */
|
/* tag_dsa.c */
|
||||||
extern const struct dsa_device_ops dsa_netdev_ops;
|
extern const struct dsa_device_ops dsa_netdev_ops;
|
||||||
|
164
net/dsa/slave.c
164
net/dsa/slave.c
@ -10,10 +10,13 @@
|
|||||||
|
|
||||||
#include <linux/list.h>
|
#include <linux/list.h>
|
||||||
#include <linux/etherdevice.h>
|
#include <linux/etherdevice.h>
|
||||||
|
#include <linux/netdevice.h>
|
||||||
#include <linux/phy.h>
|
#include <linux/phy.h>
|
||||||
#include <linux/phy_fixed.h>
|
#include <linux/phy_fixed.h>
|
||||||
#include <linux/of_net.h>
|
#include <linux/of_net.h>
|
||||||
#include <linux/of_mdio.h>
|
#include <linux/of_mdio.h>
|
||||||
|
#include <net/rtnetlink.h>
|
||||||
|
#include <linux/if_bridge.h>
|
||||||
#include "dsa_priv.h"
|
#include "dsa_priv.h"
|
||||||
|
|
||||||
/* slave mii_bus handling ***************************************************/
|
/* slave mii_bus handling ***************************************************/
|
||||||
@ -60,11 +63,18 @@ static int dsa_slave_init(struct net_device *dev)
|
|||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static inline bool dsa_port_is_bridged(struct dsa_slave_priv *p)
|
||||||
|
{
|
||||||
|
return !!p->bridge_dev;
|
||||||
|
}
|
||||||
|
|
||||||
static int dsa_slave_open(struct net_device *dev)
|
static int dsa_slave_open(struct net_device *dev)
|
||||||
{
|
{
|
||||||
struct dsa_slave_priv *p = netdev_priv(dev);
|
struct dsa_slave_priv *p = netdev_priv(dev);
|
||||||
struct net_device *master = p->parent->dst->master_netdev;
|
struct net_device *master = p->parent->dst->master_netdev;
|
||||||
struct dsa_switch *ds = p->parent;
|
struct dsa_switch *ds = p->parent;
|
||||||
|
u8 stp_state = dsa_port_is_bridged(p) ?
|
||||||
|
BR_STATE_BLOCKING : BR_STATE_FORWARDING;
|
||||||
int err;
|
int err;
|
||||||
|
|
||||||
if (!(master->flags & IFF_UP))
|
if (!(master->flags & IFF_UP))
|
||||||
@ -93,6 +103,9 @@ static int dsa_slave_open(struct net_device *dev)
|
|||||||
goto clear_promisc;
|
goto clear_promisc;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (ds->drv->port_stp_update)
|
||||||
|
ds->drv->port_stp_update(ds, p->port, stp_state);
|
||||||
|
|
||||||
if (p->phy)
|
if (p->phy)
|
||||||
phy_start(p->phy);
|
phy_start(p->phy);
|
||||||
|
|
||||||
@ -133,6 +146,9 @@ static int dsa_slave_close(struct net_device *dev)
|
|||||||
if (ds->drv->port_disable)
|
if (ds->drv->port_disable)
|
||||||
ds->drv->port_disable(ds, p->port, p->phy);
|
ds->drv->port_disable(ds, p->port, p->phy);
|
||||||
|
|
||||||
|
if (ds->drv->port_stp_update)
|
||||||
|
ds->drv->port_stp_update(ds, p->port, BR_STATE_DISABLED);
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -194,6 +210,95 @@ static int dsa_slave_ioctl(struct net_device *dev, struct ifreq *ifr, int cmd)
|
|||||||
return -EOPNOTSUPP;
|
return -EOPNOTSUPP;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* Return a bitmask of all ports being currently bridged within a given bridge
|
||||||
|
* device. Note that on leave, the mask will still return the bitmask of ports
|
||||||
|
* currently bridged, prior to port removal, and this is exactly what we want.
|
||||||
|
*/
|
||||||
|
static u32 dsa_slave_br_port_mask(struct dsa_switch *ds,
|
||||||
|
struct net_device *bridge)
|
||||||
|
{
|
||||||
|
struct dsa_slave_priv *p;
|
||||||
|
unsigned int port;
|
||||||
|
u32 mask = 0;
|
||||||
|
|
||||||
|
for (port = 0; port < DSA_MAX_PORTS; port++) {
|
||||||
|
if (!((1 << port) & ds->phys_port_mask))
|
||||||
|
continue;
|
||||||
|
|
||||||
|
if (!ds->ports[port])
|
||||||
|
continue;
|
||||||
|
|
||||||
|
p = netdev_priv(ds->ports[port]);
|
||||||
|
|
||||||
|
if (ds->ports[port]->priv_flags & IFF_BRIDGE_PORT &&
|
||||||
|
p->bridge_dev == bridge)
|
||||||
|
mask |= 1 << port;
|
||||||
|
}
|
||||||
|
|
||||||
|
return mask;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int dsa_slave_stp_update(struct net_device *dev, u8 state)
|
||||||
|
{
|
||||||
|
struct dsa_slave_priv *p = netdev_priv(dev);
|
||||||
|
struct dsa_switch *ds = p->parent;
|
||||||
|
int ret = -EOPNOTSUPP;
|
||||||
|
|
||||||
|
if (ds->drv->port_stp_update)
|
||||||
|
ret = ds->drv->port_stp_update(ds, p->port, state);
|
||||||
|
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int dsa_slave_bridge_port_join(struct net_device *dev,
|
||||||
|
struct net_device *br)
|
||||||
|
{
|
||||||
|
struct dsa_slave_priv *p = netdev_priv(dev);
|
||||||
|
struct dsa_switch *ds = p->parent;
|
||||||
|
int ret = -EOPNOTSUPP;
|
||||||
|
|
||||||
|
p->bridge_dev = br;
|
||||||
|
|
||||||
|
if (ds->drv->port_join_bridge)
|
||||||
|
ret = ds->drv->port_join_bridge(ds, p->port,
|
||||||
|
dsa_slave_br_port_mask(ds, br));
|
||||||
|
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int dsa_slave_bridge_port_leave(struct net_device *dev)
|
||||||
|
{
|
||||||
|
struct dsa_slave_priv *p = netdev_priv(dev);
|
||||||
|
struct dsa_switch *ds = p->parent;
|
||||||
|
int ret = -EOPNOTSUPP;
|
||||||
|
|
||||||
|
|
||||||
|
if (ds->drv->port_leave_bridge)
|
||||||
|
ret = ds->drv->port_leave_bridge(ds, p->port,
|
||||||
|
dsa_slave_br_port_mask(ds, p->bridge_dev));
|
||||||
|
|
||||||
|
p->bridge_dev = NULL;
|
||||||
|
|
||||||
|
/* Port left the bridge, put in BR_STATE_DISABLED by the bridge layer,
|
||||||
|
* so allow it to be in BR_STATE_FORWARDING to be kept functional
|
||||||
|
*/
|
||||||
|
dsa_slave_stp_update(dev, BR_STATE_FORWARDING);
|
||||||
|
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int dsa_slave_parent_id_get(struct net_device *dev,
|
||||||
|
struct netdev_phys_item_id *psid)
|
||||||
|
{
|
||||||
|
struct dsa_slave_priv *p = netdev_priv(dev);
|
||||||
|
struct dsa_switch *ds = p->parent;
|
||||||
|
|
||||||
|
psid->id_len = sizeof(ds->index);
|
||||||
|
memcpy(&psid->id, &ds->index, psid->id_len);
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
static netdev_tx_t dsa_slave_xmit(struct sk_buff *skb, struct net_device *dev)
|
static netdev_tx_t dsa_slave_xmit(struct sk_buff *skb, struct net_device *dev)
|
||||||
{
|
{
|
||||||
struct dsa_slave_priv *p = netdev_priv(dev);
|
struct dsa_slave_priv *p = netdev_priv(dev);
|
||||||
@ -470,6 +575,8 @@ static const struct net_device_ops dsa_slave_netdev_ops = {
|
|||||||
.ndo_set_rx_mode = dsa_slave_set_rx_mode,
|
.ndo_set_rx_mode = dsa_slave_set_rx_mode,
|
||||||
.ndo_set_mac_address = dsa_slave_set_mac_address,
|
.ndo_set_mac_address = dsa_slave_set_mac_address,
|
||||||
.ndo_do_ioctl = dsa_slave_ioctl,
|
.ndo_do_ioctl = dsa_slave_ioctl,
|
||||||
|
.ndo_switch_parent_id_get = dsa_slave_parent_id_get,
|
||||||
|
.ndo_switch_port_stp_update = dsa_slave_stp_update,
|
||||||
};
|
};
|
||||||
|
|
||||||
static void dsa_slave_adjust_link(struct net_device *dev)
|
static void dsa_slave_adjust_link(struct net_device *dev)
|
||||||
@ -605,9 +712,8 @@ int dsa_slave_resume(struct net_device *slave_dev)
|
|||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
struct net_device *
|
int dsa_slave_create(struct dsa_switch *ds, struct device *parent,
|
||||||
dsa_slave_create(struct dsa_switch *ds, struct device *parent,
|
int port, char *name)
|
||||||
int port, char *name)
|
|
||||||
{
|
{
|
||||||
struct net_device *master = ds->dst->master_netdev;
|
struct net_device *master = ds->dst->master_netdev;
|
||||||
struct net_device *slave_dev;
|
struct net_device *slave_dev;
|
||||||
@ -617,7 +723,7 @@ dsa_slave_create(struct dsa_switch *ds, struct device *parent,
|
|||||||
slave_dev = alloc_netdev(sizeof(struct dsa_slave_priv), name,
|
slave_dev = alloc_netdev(sizeof(struct dsa_slave_priv), name,
|
||||||
NET_NAME_UNKNOWN, ether_setup);
|
NET_NAME_UNKNOWN, ether_setup);
|
||||||
if (slave_dev == NULL)
|
if (slave_dev == NULL)
|
||||||
return slave_dev;
|
return -ENOMEM;
|
||||||
|
|
||||||
slave_dev->features = master->vlan_features;
|
slave_dev->features = master->vlan_features;
|
||||||
slave_dev->ethtool_ops = &dsa_slave_ethtool_ops;
|
slave_dev->ethtool_ops = &dsa_slave_ethtool_ops;
|
||||||
@ -667,19 +773,63 @@ dsa_slave_create(struct dsa_switch *ds, struct device *parent,
|
|||||||
ret = dsa_slave_phy_setup(p, slave_dev);
|
ret = dsa_slave_phy_setup(p, slave_dev);
|
||||||
if (ret) {
|
if (ret) {
|
||||||
free_netdev(slave_dev);
|
free_netdev(slave_dev);
|
||||||
return NULL;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
ds->ports[port] = slave_dev;
|
||||||
ret = register_netdev(slave_dev);
|
ret = register_netdev(slave_dev);
|
||||||
if (ret) {
|
if (ret) {
|
||||||
netdev_err(master, "error %d registering interface %s\n",
|
netdev_err(master, "error %d registering interface %s\n",
|
||||||
ret, slave_dev->name);
|
ret, slave_dev->name);
|
||||||
phy_disconnect(p->phy);
|
phy_disconnect(p->phy);
|
||||||
|
ds->ports[port] = NULL;
|
||||||
free_netdev(slave_dev);
|
free_netdev(slave_dev);
|
||||||
return NULL;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
netif_carrier_off(slave_dev);
|
netif_carrier_off(slave_dev);
|
||||||
|
|
||||||
return slave_dev;
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static bool dsa_slave_dev_check(struct net_device *dev)
|
||||||
|
{
|
||||||
|
return dev->netdev_ops == &dsa_slave_netdev_ops;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int dsa_slave_master_changed(struct net_device *dev)
|
||||||
|
{
|
||||||
|
struct net_device *master = netdev_master_upper_dev_get(dev);
|
||||||
|
int err = 0;
|
||||||
|
|
||||||
|
if (master && master->rtnl_link_ops &&
|
||||||
|
!strcmp(master->rtnl_link_ops->kind, "bridge"))
|
||||||
|
err = dsa_slave_bridge_port_join(dev, master);
|
||||||
|
else
|
||||||
|
err = dsa_slave_bridge_port_leave(dev);
|
||||||
|
|
||||||
|
return err;
|
||||||
|
}
|
||||||
|
|
||||||
|
int dsa_slave_netdevice_event(struct notifier_block *unused,
|
||||||
|
unsigned long event, void *ptr)
|
||||||
|
{
|
||||||
|
struct net_device *dev;
|
||||||
|
int err = 0;
|
||||||
|
|
||||||
|
switch (event) {
|
||||||
|
case NETDEV_CHANGEUPPER:
|
||||||
|
dev = netdev_notifier_info_to_dev(ptr);
|
||||||
|
if (!dsa_slave_dev_check(dev))
|
||||||
|
goto out;
|
||||||
|
|
||||||
|
err = dsa_slave_master_changed(dev);
|
||||||
|
if (err)
|
||||||
|
netdev_warn(dev, "failed to reflect master change\n");
|
||||||
|
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
out:
|
||||||
|
return NOTIFY_DONE;
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user