Commit 7049f9b5 authored by Tristram Ha's avatar Tristram Ha Committed by David S. Miller

net: dsa: microchip: add port_cleanup function

Add port_cleanup function to reset some device variables when the port is
disabled.  Add a mutex to make sure changing those variables is
thread-safe.
Signed-off-by: default avatarTristram Ha <Tristram.Ha@microchip.com>
Reviewed-by: default avatarFlorian Fainelli <f.fainelli@gmail.com>
Signed-off-by: default avatarDavid S. Miller <davem@davemloft.net>
parent 6ca50815
...@@ -450,12 +450,14 @@ static void ksz9477_port_stp_state_set(struct dsa_switch *ds, int port, ...@@ -450,12 +450,14 @@ static void ksz9477_port_stp_state_set(struct dsa_switch *ds, int port,
break; break;
member = dev->host_mask | p->vid_member; member = dev->host_mask | p->vid_member;
mutex_lock(&dev->dev_mutex);
/* Port is a member of a bridge. */ /* Port is a member of a bridge. */
if (dev->br_member & (1 << port)) { if (dev->br_member & (1 << port)) {
dev->member |= (1 << port); dev->member |= (1 << port);
member = dev->member; member = dev->member;
} }
mutex_unlock(&dev->dev_mutex);
break; break;
case BR_STATE_BLOCKING: case BR_STATE_BLOCKING:
data |= PORT_LEARN_DISABLE; data |= PORT_LEARN_DISABLE;
...@@ -470,6 +472,7 @@ static void ksz9477_port_stp_state_set(struct dsa_switch *ds, int port, ...@@ -470,6 +472,7 @@ static void ksz9477_port_stp_state_set(struct dsa_switch *ds, int port,
ksz_pwrite8(dev, port, P_STP_CTRL, data); ksz_pwrite8(dev, port, P_STP_CTRL, data);
p->stp_state = state; p->stp_state = state;
mutex_lock(&dev->dev_mutex);
if (data & PORT_RX_ENABLE) if (data & PORT_RX_ENABLE)
dev->rx_ports |= (1 << port); dev->rx_ports |= (1 << port);
else else
...@@ -494,6 +497,7 @@ static void ksz9477_port_stp_state_set(struct dsa_switch *ds, int port, ...@@ -494,6 +497,7 @@ static void ksz9477_port_stp_state_set(struct dsa_switch *ds, int port,
*/ */
if (forward != dev->member) if (forward != dev->member)
ksz_update_port_member(dev, port); ksz_update_port_member(dev, port);
mutex_unlock(&dev->dev_mutex);
} }
static void ksz9477_flush_dyn_mac_table(struct ksz_device *dev, int port) static void ksz9477_flush_dyn_mac_table(struct ksz_device *dev, int port)
...@@ -1080,6 +1084,7 @@ static void ksz9477_port_setup(struct ksz_device *dev, int port, bool cpu_port) ...@@ -1080,6 +1084,7 @@ static void ksz9477_port_setup(struct ksz_device *dev, int port, bool cpu_port)
ksz_pwrite8(dev, port, REG_PORT_XMII_CTRL_1, data8); ksz_pwrite8(dev, port, REG_PORT_XMII_CTRL_1, data8);
p->phydev.duplex = 1; p->phydev.duplex = 1;
} }
mutex_lock(&dev->dev_mutex);
if (cpu_port) { if (cpu_port) {
member = dev->port_mask; member = dev->port_mask;
dev->on_ports = dev->host_mask; dev->on_ports = dev->host_mask;
...@@ -1092,6 +1097,7 @@ static void ksz9477_port_setup(struct ksz_device *dev, int port, bool cpu_port) ...@@ -1092,6 +1097,7 @@ static void ksz9477_port_setup(struct ksz_device *dev, int port, bool cpu_port)
if (p->phydev.link) if (p->phydev.link)
dev->live_ports |= (1 << port); dev->live_ports |= (1 << port);
} }
mutex_unlock(&dev->dev_mutex);
ksz9477_cfg_port_member(dev, port, member); ksz9477_cfg_port_member(dev, port, member);
/* clear pending interrupts */ /* clear pending interrupts */
......
...@@ -20,6 +20,16 @@ ...@@ -20,6 +20,16 @@
#include "ksz_priv.h" #include "ksz_priv.h"
void ksz_port_cleanup(struct ksz_device *dev, int port)
{
/* Common code for port cleanup. */
mutex_lock(&dev->dev_mutex);
dev->on_ports &= ~(1 << port);
dev->live_ports &= ~(1 << port);
mutex_unlock(&dev->dev_mutex);
}
EXPORT_SYMBOL_GPL(ksz_port_cleanup);
void ksz_update_port_member(struct ksz_device *dev, int port) void ksz_update_port_member(struct ksz_device *dev, int port)
{ {
struct ksz_port *p; struct ksz_port *p;
...@@ -151,6 +161,13 @@ void ksz_adjust_link(struct dsa_switch *ds, int port, ...@@ -151,6 +161,13 @@ void ksz_adjust_link(struct dsa_switch *ds, int port,
p->read = true; p->read = true;
schedule_work(&dev->mib_read); schedule_work(&dev->mib_read);
} }
mutex_lock(&dev->dev_mutex);
if (!phydev->link)
dev->live_ports &= ~(1 << port);
else
/* Remember which port is connected and active. */
dev->live_ports |= (1 << port) & dev->on_ports;
mutex_unlock(&dev->dev_mutex);
} }
EXPORT_SYMBOL_GPL(ksz_adjust_link); EXPORT_SYMBOL_GPL(ksz_adjust_link);
...@@ -188,7 +205,9 @@ int ksz_port_bridge_join(struct dsa_switch *ds, int port, ...@@ -188,7 +205,9 @@ int ksz_port_bridge_join(struct dsa_switch *ds, int port,
{ {
struct ksz_device *dev = ds->priv; struct ksz_device *dev = ds->priv;
mutex_lock(&dev->dev_mutex);
dev->br_member |= (1 << port); dev->br_member |= (1 << port);
mutex_unlock(&dev->dev_mutex);
/* port_stp_state_set() will be called after to put the port in /* port_stp_state_set() will be called after to put the port in
* appropriate state so there is no need to do anything. * appropriate state so there is no need to do anything.
...@@ -203,8 +222,10 @@ void ksz_port_bridge_leave(struct dsa_switch *ds, int port, ...@@ -203,8 +222,10 @@ void ksz_port_bridge_leave(struct dsa_switch *ds, int port,
{ {
struct ksz_device *dev = ds->priv; struct ksz_device *dev = ds->priv;
mutex_lock(&dev->dev_mutex);
dev->br_member &= ~(1 << port); dev->br_member &= ~(1 << port);
dev->member &= ~(1 << port); dev->member &= ~(1 << port);
mutex_unlock(&dev->dev_mutex);
/* port_stp_state_set() will be called after to put the port in /* port_stp_state_set() will be called after to put the port in
* forwarding state so there is no need to do anything. * forwarding state so there is no need to do anything.
...@@ -417,6 +438,7 @@ int ksz_switch_register(struct ksz_device *dev, ...@@ -417,6 +438,7 @@ int ksz_switch_register(struct ksz_device *dev,
gpiod_set_value(dev->reset_gpio, 0); gpiod_set_value(dev->reset_gpio, 0);
} }
mutex_init(&dev->dev_mutex);
mutex_init(&dev->reg_mutex); mutex_init(&dev->reg_mutex);
mutex_init(&dev->stats_mutex); mutex_init(&dev->stats_mutex);
mutex_init(&dev->alu_mutex); mutex_init(&dev->alu_mutex);
......
...@@ -7,6 +7,7 @@ ...@@ -7,6 +7,7 @@
#ifndef __KSZ_COMMON_H #ifndef __KSZ_COMMON_H
#define __KSZ_COMMON_H #define __KSZ_COMMON_H
void ksz_port_cleanup(struct ksz_device *dev, int port);
void ksz_update_port_member(struct ksz_device *dev, int port); void ksz_update_port_member(struct ksz_device *dev, int port);
void ksz_init_mib_timer(struct ksz_device *dev); void ksz_init_mib_timer(struct ksz_device *dev);
......
...@@ -48,6 +48,7 @@ struct ksz_device { ...@@ -48,6 +48,7 @@ struct ksz_device {
struct ksz_platform_data *pdata; struct ksz_platform_data *pdata;
const char *name; const char *name;
struct mutex dev_mutex; /* device access */
struct mutex reg_mutex; /* register access */ struct mutex reg_mutex; /* register access */
struct mutex stats_mutex; /* status access */ struct mutex stats_mutex; /* status access */
struct mutex alu_mutex; /* ALU access */ struct mutex alu_mutex; /* ALU access */
...@@ -137,6 +138,7 @@ struct ksz_dev_ops { ...@@ -137,6 +138,7 @@ struct ksz_dev_ops {
void (*flush_dyn_mac_table)(struct ksz_device *dev, int port); void (*flush_dyn_mac_table)(struct ksz_device *dev, int port);
void (*phy_setup)(struct ksz_device *dev, int port, void (*phy_setup)(struct ksz_device *dev, int port,
struct phy_device *phy); struct phy_device *phy);
void (*port_cleanup)(struct ksz_device *dev, int port);
void (*port_setup)(struct ksz_device *dev, int port, bool cpu_port); void (*port_setup)(struct ksz_device *dev, int port, bool cpu_port);
void (*r_phy)(struct ksz_device *dev, u16 phy, u16 reg, u16 *val); void (*r_phy)(struct ksz_device *dev, u16 phy, u16 reg, u16 *val);
void (*w_phy)(struct ksz_device *dev, u16 phy, u16 reg, u16 val); void (*w_phy)(struct ksz_device *dev, u16 phy, u16 reg, u16 val);
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment