2018-11-20 15:55:05 -08:00
|
|
|
// SPDX-License-Identifier: GPL-2.0
|
2017-05-31 20:19:19 +00:00
|
|
|
/*
|
|
|
|
* Microchip switch driver main logic
|
|
|
|
*
|
2019-02-22 16:36:47 -08:00
|
|
|
* Copyright (C) 2017-2019 Microchip Technology Inc.
|
2017-05-31 20:19:19 +00:00
|
|
|
*/
|
|
|
|
|
|
|
|
#include <linux/delay.h>
|
|
|
|
#include <linux/export.h>
|
2018-12-10 14:43:06 +01:00
|
|
|
#include <linux/gpio/consumer.h>
|
2017-05-31 20:19:19 +00:00
|
|
|
#include <linux/kernel.h>
|
|
|
|
#include <linux/module.h>
|
|
|
|
#include <linux/platform_data/microchip-ksz.h>
|
|
|
|
#include <linux/phy.h>
|
|
|
|
#include <linux/etherdevice.h>
|
|
|
|
#include <linux/if_bridge.h>
|
2018-11-20 15:55:09 -08:00
|
|
|
#include <linux/of_net.h>
|
2017-05-31 20:19:19 +00:00
|
|
|
#include <net/dsa.h>
|
|
|
|
#include <net/switchdev.h>
|
|
|
|
|
2019-08-06 15:06:07 +02:00
|
|
|
#include "ksz_common.h"
|
2019-02-22 16:36:51 -08:00
|
|
|
|
2018-11-20 15:55:09 -08:00
|
|
|
void ksz_update_port_member(struct ksz_device *dev, int port)
|
2017-05-31 20:19:19 +00:00
|
|
|
{
|
2021-11-26 13:39:26 +01:00
|
|
|
struct ksz_port *p = &dev->ports[port];
|
|
|
|
struct dsa_switch *ds = dev->ds;
|
|
|
|
u8 port_member = 0, cpu_port;
|
|
|
|
const struct dsa_port *dp;
|
2017-05-31 20:19:19 +00:00
|
|
|
int i;
|
|
|
|
|
2021-11-26 13:39:26 +01:00
|
|
|
if (!dsa_is_user_port(ds, port))
|
|
|
|
return;
|
|
|
|
|
|
|
|
dp = dsa_to_port(ds, port);
|
|
|
|
cpu_port = BIT(dsa_upstream_port(ds, port));
|
|
|
|
|
|
|
|
for (i = 0; i < ds->num_ports; i++) {
|
|
|
|
const struct dsa_port *other_dp = dsa_to_port(ds, i);
|
|
|
|
struct ksz_port *other_p = &dev->ports[i];
|
|
|
|
u8 val = 0;
|
|
|
|
|
|
|
|
if (!dsa_is_user_port(ds, i))
|
2018-11-20 15:55:09 -08:00
|
|
|
continue;
|
2021-11-26 13:39:26 +01:00
|
|
|
if (port == i)
|
|
|
|
continue;
|
2021-12-06 18:57:53 +02:00
|
|
|
if (!dsa_port_bridge_same(dp, other_dp))
|
2018-11-20 15:55:09 -08:00
|
|
|
continue;
|
|
|
|
|
2021-11-26 13:39:26 +01:00
|
|
|
if (other_p->stp_state == BR_STATE_FORWARDING &&
|
|
|
|
p->stp_state == BR_STATE_FORWARDING) {
|
|
|
|
val |= BIT(port);
|
|
|
|
port_member |= BIT(i);
|
|
|
|
}
|
|
|
|
|
|
|
|
dev->dev_ops->cfg_port_member(dev, i, val | cpu_port);
|
2017-05-31 20:19:19 +00:00
|
|
|
}
|
2021-11-26 13:39:26 +01:00
|
|
|
|
|
|
|
dev->dev_ops->cfg_port_member(dev, port, port_member | cpu_port);
|
2017-05-31 20:19:19 +00:00
|
|
|
}
|
2018-11-20 15:55:09 -08:00
|
|
|
EXPORT_SYMBOL_GPL(ksz_update_port_member);
|
2017-05-31 20:19:19 +00:00
|
|
|
|
2019-02-22 16:36:48 -08:00
|
|
|
static void port_r_cnt(struct ksz_device *dev, int port)
|
|
|
|
{
|
|
|
|
struct ksz_port_mib *mib = &dev->ports[port].mib;
|
|
|
|
u64 *dropped;
|
|
|
|
|
|
|
|
/* Some ports may not have MIB counters before SWITCH_COUNTER_NUM. */
|
|
|
|
while (mib->cnt_ptr < dev->reg_mib_cnt) {
|
|
|
|
dev->dev_ops->r_mib_cnt(dev, port, mib->cnt_ptr,
|
|
|
|
&mib->counters[mib->cnt_ptr]);
|
|
|
|
++mib->cnt_ptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* last one in storage */
|
|
|
|
dropped = &mib->counters[dev->mib_cnt];
|
|
|
|
|
|
|
|
/* Some ports may not have MIB counters after SWITCH_COUNTER_NUM. */
|
|
|
|
while (mib->cnt_ptr < dev->mib_cnt) {
|
|
|
|
dev->dev_ops->r_mib_pkt(dev, port, mib->cnt_ptr,
|
|
|
|
dropped, &mib->counters[mib->cnt_ptr]);
|
|
|
|
++mib->cnt_ptr;
|
|
|
|
}
|
|
|
|
mib->cnt_ptr = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void ksz_mib_read_work(struct work_struct *work)
|
|
|
|
{
|
|
|
|
struct ksz_device *dev = container_of(work, struct ksz_device,
|
2020-03-10 12:58:59 -05:00
|
|
|
mib_read.work);
|
2019-02-22 16:36:48 -08:00
|
|
|
struct ksz_port_mib *mib;
|
|
|
|
struct ksz_port *p;
|
|
|
|
int i;
|
|
|
|
|
2020-12-01 21:45:04 +01:00
|
|
|
for (i = 0; i < dev->port_cnt; i++) {
|
2019-06-12 14:33:32 -06:00
|
|
|
if (dsa_is_unused_port(dev->ds, i))
|
|
|
|
continue;
|
|
|
|
|
2019-02-22 16:36:48 -08:00
|
|
|
p = &dev->ports[i];
|
|
|
|
mib = &p->mib;
|
|
|
|
mutex_lock(&mib->cnt_mutex);
|
|
|
|
|
|
|
|
/* Only read MIB counters when the port is told to do.
|
|
|
|
* If not, read only dropped counters when link is not up.
|
|
|
|
*/
|
|
|
|
if (!p->read) {
|
|
|
|
const struct dsa_port *dp = dsa_to_port(dev->ds, i);
|
|
|
|
|
|
|
|
if (!netif_carrier_ok(dp->slave))
|
|
|
|
mib->cnt_ptr = dev->reg_mib_cnt;
|
|
|
|
}
|
|
|
|
port_r_cnt(dev, i);
|
|
|
|
p->read = false;
|
|
|
|
mutex_unlock(&mib->cnt_mutex);
|
|
|
|
}
|
|
|
|
|
2020-03-10 12:58:59 -05:00
|
|
|
schedule_delayed_work(&dev->mib_read, dev->mib_read_interval);
|
2019-02-22 16:36:48 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
void ksz_init_mib_timer(struct ksz_device *dev)
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
|
2020-03-10 12:58:59 -05:00
|
|
|
INIT_DELAYED_WORK(&dev->mib_read, ksz_mib_read_work);
|
|
|
|
|
2020-12-01 21:45:04 +01:00
|
|
|
for (i = 0; i < dev->port_cnt; i++)
|
2019-02-22 16:36:48 -08:00
|
|
|
dev->dev_ops->port_init_cnt(dev, i);
|
|
|
|
}
|
|
|
|
EXPORT_SYMBOL_GPL(ksz_init_mib_timer);
|
|
|
|
|
2018-11-20 15:55:09 -08:00
|
|
|
int ksz_phy_read16(struct dsa_switch *ds, int addr, int reg)
|
2017-05-31 20:19:19 +00:00
|
|
|
{
|
|
|
|
struct ksz_device *dev = ds->priv;
|
2018-11-20 15:55:09 -08:00
|
|
|
u16 val = 0xffff;
|
2017-05-31 20:19:19 +00:00
|
|
|
|
2018-11-20 15:55:09 -08:00
|
|
|
dev->dev_ops->r_phy(dev, addr, reg, &val);
|
2017-05-31 20:19:19 +00:00
|
|
|
|
|
|
|
return val;
|
|
|
|
}
|
2018-11-20 15:55:09 -08:00
|
|
|
EXPORT_SYMBOL_GPL(ksz_phy_read16);
|
2017-05-31 20:19:19 +00:00
|
|
|
|
2018-11-20 15:55:09 -08:00
|
|
|
int ksz_phy_write16(struct dsa_switch *ds, int addr, int reg, u16 val)
|
2017-05-31 20:19:19 +00:00
|
|
|
{
|
|
|
|
struct ksz_device *dev = ds->priv;
|
|
|
|
|
2018-11-20 15:55:09 -08:00
|
|
|
dev->dev_ops->w_phy(dev, addr, reg, val);
|
2017-05-31 20:19:19 +00:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
2018-11-20 15:55:09 -08:00
|
|
|
EXPORT_SYMBOL_GPL(ksz_phy_write16);
|
2017-05-31 20:19:19 +00:00
|
|
|
|
2020-07-02 18:17:23 +03:00
|
|
|
void ksz_mac_link_down(struct dsa_switch *ds, int port, unsigned int mode,
|
|
|
|
phy_interface_t interface)
|
2019-02-22 16:36:49 -08:00
|
|
|
{
|
|
|
|
struct ksz_device *dev = ds->priv;
|
|
|
|
struct ksz_port *p = &dev->ports[port];
|
|
|
|
|
|
|
|
/* Read all MIB counters when the link is going down. */
|
2020-07-02 18:17:23 +03:00
|
|
|
p->read = true;
|
2020-10-12 10:39:42 +02:00
|
|
|
/* timer started */
|
|
|
|
if (dev->mib_read_interval)
|
|
|
|
schedule_delayed_work(&dev->mib_read, 0);
|
2020-07-02 18:17:23 +03:00
|
|
|
}
|
|
|
|
EXPORT_SYMBOL_GPL(ksz_mac_link_down);
|
|
|
|
|
2018-11-20 15:55:09 -08:00
|
|
|
int ksz_sset_count(struct dsa_switch *ds, int port, int sset)
|
2017-05-31 20:19:19 +00:00
|
|
|
{
|
|
|
|
struct ksz_device *dev = ds->priv;
|
|
|
|
|
2018-04-25 12:12:50 -07:00
|
|
|
if (sset != ETH_SS_STATS)
|
|
|
|
return 0;
|
|
|
|
|
2018-11-20 15:55:09 -08:00
|
|
|
return dev->mib_cnt;
|
2017-05-31 20:19:19 +00:00
|
|
|
}
|
2018-11-20 15:55:09 -08:00
|
|
|
EXPORT_SYMBOL_GPL(ksz_sset_count);
|
2017-05-31 20:19:19 +00:00
|
|
|
|
2019-02-22 16:36:48 -08:00
|
|
|
void ksz_get_ethtool_stats(struct dsa_switch *ds, int port, uint64_t *buf)
|
|
|
|
{
|
|
|
|
const struct dsa_port *dp = dsa_to_port(ds, port);
|
|
|
|
struct ksz_device *dev = ds->priv;
|
|
|
|
struct ksz_port_mib *mib;
|
|
|
|
|
|
|
|
mib = &dev->ports[port].mib;
|
|
|
|
mutex_lock(&mib->cnt_mutex);
|
|
|
|
|
|
|
|
/* Only read dropped counters if no link. */
|
|
|
|
if (!netif_carrier_ok(dp->slave))
|
|
|
|
mib->cnt_ptr = dev->reg_mib_cnt;
|
|
|
|
port_r_cnt(dev, port);
|
|
|
|
memcpy(buf, mib->counters, dev->mib_cnt * sizeof(u64));
|
|
|
|
mutex_unlock(&mib->cnt_mutex);
|
|
|
|
}
|
|
|
|
EXPORT_SYMBOL_GPL(ksz_get_ethtool_stats);
|
|
|
|
|
2018-11-20 15:55:09 -08:00
|
|
|
int ksz_port_bridge_join(struct dsa_switch *ds, int port,
|
2021-12-06 18:57:57 +02:00
|
|
|
struct dsa_bridge bridge,
|
|
|
|
bool *tx_fwd_offload)
|
2017-05-31 20:19:19 +00:00
|
|
|
{
|
2018-11-20 15:55:09 -08:00
|
|
|
/* port_stp_state_set() will be called after to put the port in
|
|
|
|
* appropriate state so there is no need to do anything.
|
|
|
|
*/
|
2017-05-31 20:19:19 +00:00
|
|
|
|
2018-11-20 15:55:09 -08:00
|
|
|
return 0;
|
2017-05-31 20:19:19 +00:00
|
|
|
}
|
2018-11-20 15:55:09 -08:00
|
|
|
EXPORT_SYMBOL_GPL(ksz_port_bridge_join);
|
2017-05-31 20:19:19 +00:00
|
|
|
|
2018-11-20 15:55:09 -08:00
|
|
|
void ksz_port_bridge_leave(struct dsa_switch *ds, int port,
|
net: dsa: keep the bridge_dev and bridge_num as part of the same structure
The main desire behind this is to provide coherent bridge information to
the fast path without locking.
For example, right now we set dp->bridge_dev and dp->bridge_num from
separate code paths, it is theoretically possible for a packet
transmission to read these two port properties consecutively and find a
bridge number which does not correspond with the bridge device.
Another desire is to start passing more complex bridge information to
dsa_switch_ops functions. For example, with FDB isolation, it is
expected that drivers will need to be passed the bridge which requested
an FDB/MDB entry to be offloaded, and along with that bridge_dev, the
associated bridge_num should be passed too, in case the driver might
want to implement an isolation scheme based on that number.
We already pass the {bridge_dev, bridge_num} pair to the TX forwarding
offload switch API, however we'd like to remove that and squash it into
the basic bridge join/leave API. So that means we need to pass this
pair to the bridge join/leave API.
During dsa_port_bridge_leave, first we unset dp->bridge_dev, then we
call the driver's .port_bridge_leave with what used to be our
dp->bridge_dev, but provided as an argument.
When bridge_dev and bridge_num get folded into a single structure, we
need to preserve this behavior in dsa_port_bridge_leave: we need a copy
of what used to be in dp->bridge.
Switch drivers check bridge membership by comparing dp->bridge_dev with
the provided bridge_dev, but now, if we provide the struct dsa_bridge as
a pointer, they cannot keep comparing dp->bridge to the provided
pointer, since this only points to an on-stack copy. To make this
obvious and prevent driver writers from forgetting and doing stupid
things, in this new API, the struct dsa_bridge is provided as a full
structure (not very large, contains an int and a pointer) instead of a
pointer. An explicit comparison function needs to be used to determine
bridge membership: dsa_port_offloads_bridge().
Signed-off-by: Vladimir Oltean <vladimir.oltean@nxp.com>
Reviewed-by: Alvin Šipraga <alsi@bang-olufsen.dk>
Signed-off-by: Jakub Kicinski <kuba@kernel.org>
2021-12-06 18:57:56 +02:00
|
|
|
struct dsa_bridge bridge)
|
2017-05-31 20:19:19 +00:00
|
|
|
{
|
2018-11-20 15:55:09 -08:00
|
|
|
/* port_stp_state_set() will be called after to put the port in
|
|
|
|
* forwarding state so there is no need to do anything.
|
|
|
|
*/
|
2017-05-31 20:19:19 +00:00
|
|
|
}
|
2018-11-20 15:55:09 -08:00
|
|
|
EXPORT_SYMBOL_GPL(ksz_port_bridge_leave);
|
2017-05-31 20:19:19 +00:00
|
|
|
|
2018-11-20 15:55:09 -08:00
|
|
|
void ksz_port_fast_age(struct dsa_switch *ds, int port)
|
2017-05-31 20:19:19 +00:00
|
|
|
{
|
|
|
|
struct ksz_device *dev = ds->priv;
|
|
|
|
|
2018-11-20 15:55:09 -08:00
|
|
|
dev->dev_ops->flush_dyn_mac_table(dev, port);
|
2017-05-31 20:19:19 +00:00
|
|
|
}
|
2018-11-20 15:55:09 -08:00
|
|
|
EXPORT_SYMBOL_GPL(ksz_port_fast_age);
|
2017-05-31 20:19:19 +00:00
|
|
|
|
2018-11-20 15:55:09 -08:00
|
|
|
int ksz_port_fdb_dump(struct dsa_switch *ds, int port, dsa_fdb_dump_cb_t *cb,
|
|
|
|
void *data)
|
2017-05-31 20:19:19 +00:00
|
|
|
{
|
|
|
|
struct ksz_device *dev = ds->priv;
|
2017-08-06 16:15:40 +03:00
|
|
|
int ret = 0;
|
2018-11-20 15:55:09 -08:00
|
|
|
u16 i = 0;
|
|
|
|
u16 entries = 0;
|
|
|
|
u8 timestamp = 0;
|
|
|
|
u8 fid;
|
|
|
|
u8 member;
|
2017-05-31 20:19:19 +00:00
|
|
|
struct alu_struct alu;
|
|
|
|
|
|
|
|
do {
|
2018-11-20 15:55:09 -08:00
|
|
|
alu.is_static = false;
|
|
|
|
ret = dev->dev_ops->r_dyn_mac_table(dev, i, alu.mac, &fid,
|
|
|
|
&member, ×tamp,
|
|
|
|
&entries);
|
|
|
|
if (!ret && (member & BIT(port))) {
|
2017-08-06 16:15:49 +03:00
|
|
|
ret = cb(alu.mac, alu.fid, alu.is_static, data);
|
2017-05-31 20:19:19 +00:00
|
|
|
if (ret)
|
2018-11-20 15:55:09 -08:00
|
|
|
break;
|
2017-05-31 20:19:19 +00:00
|
|
|
}
|
2018-11-20 15:55:09 -08:00
|
|
|
i++;
|
|
|
|
} while (i < entries);
|
|
|
|
if (i >= entries)
|
|
|
|
ret = 0;
|
2017-05-31 20:19:19 +00:00
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
2018-11-20 15:55:09 -08:00
|
|
|
EXPORT_SYMBOL_GPL(ksz_port_fdb_dump);
|
2017-05-31 20:19:19 +00:00
|
|
|
|
2021-01-09 02:01:52 +02:00
|
|
|
int ksz_port_mdb_add(struct dsa_switch *ds, int port,
|
|
|
|
const struct switchdev_obj_port_mdb *mdb)
|
2017-05-31 20:19:19 +00:00
|
|
|
{
|
|
|
|
struct ksz_device *dev = ds->priv;
|
2018-11-20 15:55:09 -08:00
|
|
|
struct alu_struct alu;
|
2017-05-31 20:19:19 +00:00
|
|
|
int index;
|
2018-11-20 15:55:09 -08:00
|
|
|
int empty = 0;
|
2017-05-31 20:19:19 +00:00
|
|
|
|
2018-11-20 15:55:09 -08:00
|
|
|
alu.port_forward = 0;
|
2017-05-31 20:19:19 +00:00
|
|
|
for (index = 0; index < dev->num_statics; index++) {
|
2018-11-20 15:55:09 -08:00
|
|
|
if (!dev->dev_ops->r_sta_mac_table(dev, index, &alu)) {
|
|
|
|
/* Found one already in static MAC table. */
|
|
|
|
if (!memcmp(alu.mac, mdb->addr, ETH_ALEN) &&
|
|
|
|
alu.fid == mdb->vid)
|
2017-05-31 20:19:19 +00:00
|
|
|
break;
|
2018-11-20 15:55:09 -08:00
|
|
|
/* Remember the first empty entry. */
|
|
|
|
} else if (!empty) {
|
|
|
|
empty = index + 1;
|
2017-05-31 20:19:19 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* no available entry */
|
2018-11-20 15:55:09 -08:00
|
|
|
if (index == dev->num_statics && !empty)
|
2021-01-09 02:01:52 +02:00
|
|
|
return -ENOSPC;
|
2017-05-31 20:19:19 +00:00
|
|
|
|
|
|
|
/* add entry */
|
2018-11-20 15:55:09 -08:00
|
|
|
if (index == dev->num_statics) {
|
|
|
|
index = empty - 1;
|
|
|
|
memset(&alu, 0, sizeof(alu));
|
|
|
|
memcpy(alu.mac, mdb->addr, ETH_ALEN);
|
|
|
|
alu.is_static = true;
|
|
|
|
}
|
|
|
|
alu.port_forward |= BIT(port);
|
|
|
|
if (mdb->vid) {
|
|
|
|
alu.is_use_fid = true;
|
2017-05-31 20:19:19 +00:00
|
|
|
|
2018-11-20 15:55:09 -08:00
|
|
|
/* Need a way to map VID to FID. */
|
|
|
|
alu.fid = mdb->vid;
|
|
|
|
}
|
|
|
|
dev->dev_ops->w_sta_mac_table(dev, index, &alu);
|
2021-01-09 02:01:52 +02:00
|
|
|
|
|
|
|
return 0;
|
2017-05-31 20:19:19 +00:00
|
|
|
}
|
2018-11-20 15:55:09 -08:00
|
|
|
EXPORT_SYMBOL_GPL(ksz_port_mdb_add);
|
2017-05-31 20:19:19 +00:00
|
|
|
|
2018-11-20 15:55:09 -08:00
|
|
|
int ksz_port_mdb_del(struct dsa_switch *ds, int port,
|
|
|
|
const struct switchdev_obj_port_mdb *mdb)
|
2017-05-31 20:19:19 +00:00
|
|
|
{
|
|
|
|
struct ksz_device *dev = ds->priv;
|
2018-11-20 15:55:09 -08:00
|
|
|
struct alu_struct alu;
|
2017-05-31 20:19:19 +00:00
|
|
|
int index;
|
|
|
|
|
|
|
|
for (index = 0; index < dev->num_statics; index++) {
|
2018-11-20 15:55:09 -08:00
|
|
|
if (!dev->dev_ops->r_sta_mac_table(dev, index, &alu)) {
|
|
|
|
/* Found one already in static MAC table. */
|
|
|
|
if (!memcmp(alu.mac, mdb->addr, ETH_ALEN) &&
|
|
|
|
alu.fid == mdb->vid)
|
2017-05-31 20:19:19 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* no available entry */
|
2018-11-20 15:55:09 -08:00
|
|
|
if (index == dev->num_statics)
|
2017-05-31 20:19:19 +00:00
|
|
|
goto exit;
|
|
|
|
|
|
|
|
/* clear port */
|
2018-11-20 15:55:09 -08:00
|
|
|
alu.port_forward &= ~BIT(port);
|
|
|
|
if (!alu.port_forward)
|
|
|
|
alu.is_static = false;
|
|
|
|
dev->dev_ops->w_sta_mac_table(dev, index, &alu);
|
2017-05-31 20:19:19 +00:00
|
|
|
|
|
|
|
exit:
|
2021-12-16 09:13:39 +00:00
|
|
|
return 0;
|
2017-05-31 20:19:19 +00:00
|
|
|
}
|
2018-11-20 15:55:09 -08:00
|
|
|
EXPORT_SYMBOL_GPL(ksz_port_mdb_del);
|
2017-05-31 20:19:19 +00:00
|
|
|
|
2018-11-20 15:55:09 -08:00
|
|
|
int ksz_enable_port(struct dsa_switch *ds, int port, struct phy_device *phy)
|
2017-05-31 20:19:19 +00:00
|
|
|
{
|
|
|
|
struct ksz_device *dev = ds->priv;
|
|
|
|
|
2019-08-19 16:00:49 -04:00
|
|
|
if (!dsa_is_user_port(ds, port))
|
|
|
|
return 0;
|
|
|
|
|
2018-11-20 15:55:09 -08:00
|
|
|
/* setup slave port */
|
|
|
|
dev->dev_ops->port_setup(dev, port, false);
|
2017-05-31 20:19:19 +00:00
|
|
|
|
2018-11-20 15:55:09 -08:00
|
|
|
/* port_stp_state_set() will be called after to enable the port so
|
|
|
|
* there is no need to do anything.
|
|
|
|
*/
|
2017-05-31 20:19:19 +00:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
2018-11-20 15:55:09 -08:00
|
|
|
EXPORT_SYMBOL_GPL(ksz_enable_port);
|
2017-05-31 20:19:19 +00:00
|
|
|
|
2019-06-26 01:43:46 +02:00
|
|
|
struct ksz_device *ksz_switch_alloc(struct device *base, void *priv)
|
2017-05-31 20:19:19 +00:00
|
|
|
{
|
|
|
|
struct dsa_switch *ds;
|
|
|
|
struct ksz_device *swdev;
|
|
|
|
|
2019-10-21 16:51:30 -04:00
|
|
|
ds = devm_kzalloc(base, sizeof(*ds), GFP_KERNEL);
|
2017-05-31 20:19:19 +00:00
|
|
|
if (!ds)
|
|
|
|
return NULL;
|
|
|
|
|
2019-10-21 16:51:30 -04:00
|
|
|
ds->dev = base;
|
|
|
|
ds->num_ports = DSA_MAX_PORTS;
|
|
|
|
|
2017-05-31 20:19:19 +00:00
|
|
|
swdev = devm_kzalloc(base, sizeof(*swdev), GFP_KERNEL);
|
|
|
|
if (!swdev)
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
ds->priv = swdev;
|
|
|
|
swdev->dev = base;
|
|
|
|
|
|
|
|
swdev->ds = ds;
|
|
|
|
swdev->priv = priv;
|
|
|
|
|
|
|
|
return swdev;
|
|
|
|
}
|
|
|
|
EXPORT_SYMBOL(ksz_switch_alloc);
|
|
|
|
|
2018-11-20 15:55:09 -08:00
|
|
|
int ksz_switch_register(struct ksz_device *dev,
|
|
|
|
const struct ksz_dev_ops *ops)
|
2017-05-31 20:19:19 +00:00
|
|
|
{
|
2020-09-24 10:37:47 +02:00
|
|
|
struct device_node *port, *ports;
|
net: of_get_phy_mode: Change API to solve int/unit warnings
Before this change of_get_phy_mode() returned an enum,
phy_interface_t. On error, -ENODEV etc, is returned. If the result of
the function is stored in a variable of type phy_interface_t, and the
compiler has decided to represent this as an unsigned int, comparision
with -ENODEV etc, is a signed vs unsigned comparision.
Fix this problem by changing the API. Make the function return an
error, or 0 on success, and pass a pointer, of type phy_interface_t,
where the phy mode should be stored.
v2:
Return with *interface set to PHY_INTERFACE_MODE_NA on error.
Add error checks to all users of of_get_phy_mode()
Fixup a few reverse christmas tree errors
Fixup a few slightly malformed reverse christmas trees
v3:
Fix 0-day reported errors.
Reported-by: Dan Carpenter <dan.carpenter@oracle.com>
Signed-off-by: Andrew Lunn <andrew@lunn.ch>
Signed-off-by: David S. Miller <davem@davemloft.net>
2019-11-04 02:40:33 +01:00
|
|
|
phy_interface_t interface;
|
2020-09-08 10:01:38 +02:00
|
|
|
unsigned int port_num;
|
2017-05-31 20:19:19 +00:00
|
|
|
int ret;
|
|
|
|
|
|
|
|
if (dev->pdata)
|
|
|
|
dev->chip_id = dev->pdata->chip_id;
|
|
|
|
|
2018-12-10 14:43:06 +01:00
|
|
|
dev->reset_gpio = devm_gpiod_get_optional(dev->dev, "reset",
|
|
|
|
GPIOD_OUT_LOW);
|
|
|
|
if (IS_ERR(dev->reset_gpio))
|
|
|
|
return PTR_ERR(dev->reset_gpio);
|
|
|
|
|
|
|
|
if (dev->reset_gpio) {
|
2019-06-23 17:12:57 +02:00
|
|
|
gpiod_set_value_cansleep(dev->reset_gpio, 1);
|
2020-09-09 11:04:17 +01:00
|
|
|
usleep_range(10000, 12000);
|
2019-06-23 17:12:57 +02:00
|
|
|
gpiod_set_value_cansleep(dev->reset_gpio, 0);
|
2021-01-20 04:05:02 +01:00
|
|
|
msleep(100);
|
2018-12-10 14:43:06 +01:00
|
|
|
}
|
|
|
|
|
2019-02-22 16:36:51 -08:00
|
|
|
mutex_init(&dev->dev_mutex);
|
2019-10-16 15:33:24 +02:00
|
|
|
mutex_init(&dev->regmap_mutex);
|
2018-11-02 19:23:41 -07:00
|
|
|
mutex_init(&dev->alu_mutex);
|
|
|
|
mutex_init(&dev->vlan_mutex);
|
|
|
|
|
2018-11-20 15:55:09 -08:00
|
|
|
dev->dev_ops = ops;
|
|
|
|
|
|
|
|
if (dev->dev_ops->detect(dev))
|
2017-05-31 20:19:19 +00:00
|
|
|
return -EINVAL;
|
|
|
|
|
2018-11-20 15:55:09 -08:00
|
|
|
ret = dev->dev_ops->init(dev);
|
2017-05-31 20:19:19 +00:00
|
|
|
if (ret)
|
|
|
|
return ret;
|
|
|
|
|
2019-02-28 19:57:24 -08:00
|
|
|
/* Host port interface will be self detected, or specifically set in
|
|
|
|
* device tree.
|
|
|
|
*/
|
2020-09-08 10:01:38 +02:00
|
|
|
for (port_num = 0; port_num < dev->port_cnt; ++port_num)
|
|
|
|
dev->ports[port_num].interface = PHY_INTERFACE_MODE_NA;
|
2018-11-20 15:55:09 -08:00
|
|
|
if (dev->dev->of_node) {
|
net: of_get_phy_mode: Change API to solve int/unit warnings
Before this change of_get_phy_mode() returned an enum,
phy_interface_t. On error, -ENODEV etc, is returned. If the result of
the function is stored in a variable of type phy_interface_t, and the
compiler has decided to represent this as an unsigned int, comparision
with -ENODEV etc, is a signed vs unsigned comparision.
Fix this problem by changing the API. Make the function return an
error, or 0 on success, and pass a pointer, of type phy_interface_t,
where the phy mode should be stored.
v2:
Return with *interface set to PHY_INTERFACE_MODE_NA on error.
Add error checks to all users of of_get_phy_mode()
Fixup a few reverse christmas tree errors
Fixup a few slightly malformed reverse christmas trees
v3:
Fix 0-day reported errors.
Reported-by: Dan Carpenter <dan.carpenter@oracle.com>
Signed-off-by: Andrew Lunn <andrew@lunn.ch>
Signed-off-by: David S. Miller <davem@davemloft.net>
2019-11-04 02:40:33 +01:00
|
|
|
ret = of_get_phy_mode(dev->dev->of_node, &interface);
|
|
|
|
if (ret == 0)
|
2020-09-08 10:01:38 +02:00
|
|
|
dev->compat_interface = interface;
|
2020-11-20 12:21:05 +01:00
|
|
|
ports = of_get_child_by_name(dev->dev->of_node, "ethernet-ports");
|
|
|
|
if (!ports)
|
|
|
|
ports = of_get_child_by_name(dev->dev->of_node, "ports");
|
2020-09-24 10:37:47 +02:00
|
|
|
if (ports)
|
|
|
|
for_each_available_child_of_node(ports, port) {
|
|
|
|
if (of_property_read_u32(port, "reg",
|
|
|
|
&port_num))
|
|
|
|
continue;
|
2021-07-11 18:12:56 +02:00
|
|
|
if (!(dev->port_mask & BIT(port_num))) {
|
|
|
|
of_node_put(port);
|
2020-09-24 10:37:47 +02:00
|
|
|
return -EINVAL;
|
2021-07-11 18:12:56 +02:00
|
|
|
}
|
2020-09-24 10:37:47 +02:00
|
|
|
of_get_phy_mode(port,
|
|
|
|
&dev->ports[port_num].interface);
|
|
|
|
}
|
2019-06-12 14:49:06 -06:00
|
|
|
dev->synclko_125 = of_property_read_bool(dev->dev->of_node,
|
|
|
|
"microchip,synclko-125");
|
2018-11-20 15:55:09 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
ret = dsa_register_switch(dev->ds);
|
|
|
|
if (ret) {
|
|
|
|
dev->dev_ops->exit(dev);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2020-10-12 10:39:42 +02:00
|
|
|
/* Read MIB counters every 30 seconds to avoid overflow. */
|
|
|
|
dev->mib_read_interval = msecs_to_jiffies(30000);
|
|
|
|
|
|
|
|
/* Start the MIB timer. */
|
|
|
|
schedule_delayed_work(&dev->mib_read, 0);
|
|
|
|
|
2018-11-20 15:55:09 -08:00
|
|
|
return 0;
|
2017-05-31 20:19:19 +00:00
|
|
|
}
|
|
|
|
EXPORT_SYMBOL(ksz_switch_register);
|
|
|
|
|
|
|
|
void ksz_switch_remove(struct ksz_device *dev)
|
|
|
|
{
|
2019-02-22 16:36:48 -08:00
|
|
|
/* timer started */
|
2021-10-11 21:18:08 +05:30
|
|
|
if (dev->mib_read_interval) {
|
|
|
|
dev->mib_read_interval = 0;
|
2020-03-10 12:58:59 -05:00
|
|
|
cancel_delayed_work_sync(&dev->mib_read);
|
2021-10-11 21:18:08 +05:30
|
|
|
}
|
2019-02-22 16:36:48 -08:00
|
|
|
|
2018-11-20 15:55:09 -08:00
|
|
|
dev->dev_ops->exit(dev);
|
2017-05-31 20:19:19 +00:00
|
|
|
dsa_unregister_switch(dev->ds);
|
2018-12-10 14:43:06 +01:00
|
|
|
|
|
|
|
if (dev->reset_gpio)
|
2019-06-23 17:12:57 +02:00
|
|
|
gpiod_set_value_cansleep(dev->reset_gpio, 1);
|
2018-12-10 14:43:06 +01:00
|
|
|
|
2017-05-31 20:19:19 +00:00
|
|
|
}
|
|
|
|
EXPORT_SYMBOL(ksz_switch_remove);
|
|
|
|
|
|
|
|
MODULE_AUTHOR("Woojung Huh <Woojung.Huh@microchip.com>");
|
|
|
|
MODULE_DESCRIPTION("Microchip KSZ Series Switch DSA Driver");
|
|
|
|
MODULE_LICENSE("GPL");
|