From patchwork Fri May 29 01:42:16 2015 Content-Type: text/plain; charset="utf-8" MIME-Version: 1.0 Content-Transfer-Encoding: 7bit Subject: [1/7] net: dsa: add new driver for ar8xxx family From: Mathieu Olivari X-Patchwork-Id: 477523 X-Patchwork-Delegate: davem@davemloft.net Message-Id: <1432863742-18427-2-git-send-email-mathieu@codeaurora.org> To: robh+dt@kernel.org, pawel.moll@arm.com, mark.rutland@arm.com, ijc+devicetree@hellion.org.uk, galak@codeaurora.org, davem@davemloft.net, mathieu@codeaurora.org, andrew@lunn.ch, f.fainelli@gmail.com, linux@roeck-us.net, gang.chen.5i5j@gmail.com, jiri@resnulli.us, leitec@staticky.com, fabf@skynet.be, alexander.h.duyck@intel.com, pavel.nakonechny@skitlab.ru, joe@perches.com, sfeldma@gmail.com, nbd@openwrt.org, juhosg@openwrt.org Cc: devicetree@vger.kernel.org, linux-kernel@vger.kernel.org, netdev@vger.kernel.org Date: Thu, 28 May 2015 18:42:16 -0700 This patch contains initial init & registration code for QCA8337. It will detect a QCA8337 switch, if present and declared in DT/platform. Each port will be represented through a standalone net_device interface, as for other DSA switches. CPU can communicate with any of the ports by setting an IP@ on ethN interface. Ports cannot communicate with each other just yet. Link status will be reported through polling, and we don't use any encapsulation. Signed-off-by: Mathieu Olivari --- drivers/net/dsa/Kconfig | 7 ++ drivers/net/dsa/Makefile | 1 + drivers/net/dsa/ar8xxx.c | 303 +++++++++++++++++++++++++++++++++++++++++++++++ drivers/net/dsa/ar8xxx.h | 82 +++++++++++++ net/dsa/dsa.c | 1 + 5 files changed, 394 insertions(+) create mode 100644 drivers/net/dsa/ar8xxx.c create mode 100644 drivers/net/dsa/ar8xxx.h diff --git a/drivers/net/dsa/Kconfig b/drivers/net/dsa/Kconfig index 7ad0a4d..2aae541 100644 --- a/drivers/net/dsa/Kconfig +++ b/drivers/net/dsa/Kconfig @@ -65,4 +65,11 @@ config NET_DSA_BCM_SF2 This enables support for the Broadcom Starfighter 2 Ethernet switch chips. +config NET_DSA_AR8XXX + tristate "Qualcomm Atheros AR8XXX Ethernet switch family support" + depends on NET_DSA + ---help--- + This enables support for the Qualcomm Atheros AR8XXX Ethernet + switch chips. + endmenu diff --git a/drivers/net/dsa/Makefile b/drivers/net/dsa/Makefile index e2d51c4..7647687 100644 --- a/drivers/net/dsa/Makefile +++ b/drivers/net/dsa/Makefile @@ -14,3 +14,4 @@ ifdef CONFIG_NET_DSA_MV88E6171 mv88e6xxx_drv-y += mv88e6171.o endif obj-$(CONFIG_NET_DSA_BCM_SF2) += bcm_sf2.o +obj-$(CONFIG_NET_DSA_AR8XXX) += ar8xxx.o diff --git a/drivers/net/dsa/ar8xxx.c b/drivers/net/dsa/ar8xxx.c new file mode 100644 index 0000000..4ce3ffc --- /dev/null +++ b/drivers/net/dsa/ar8xxx.c @@ -0,0 +1,303 @@ +/* + * Copyright (C) 2009 Felix Fietkau + * Copyright (C) 2011-2012 Gabor Juhos + * Copyright (c) 2015, The Linux Foundation. All rights reserved. + * + * This program is free software; you can redistribute it and/or modify + * it under the terms of the GNU General Public License version 2 and + * only version 2 as published by the Free Software Foundation. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + */ + +#include +#include +#include +#include +#include +#include + +#include "ar8xxx.h" + +u32 +ar8xxx_mii_read32(struct mii_bus *bus, int phy_id, int regnum) +{ + u16 lo, hi; + + lo = bus->read(bus, phy_id, regnum); + hi = bus->read(bus, phy_id, regnum + 1); + + return (hi << 16) | lo; +} + +void +ar8xxx_mii_write32(struct mii_bus *bus, int phy_id, int regnum, u32 val) +{ + u16 lo, hi; + + lo = val & 0xffff; + hi = (u16)(val >> 16); + + bus->write(bus, phy_id, regnum, lo); + bus->write(bus, phy_id, regnum + 1, hi); +} + +u32 ar8xxx_read(struct dsa_switch *ds, int reg) +{ + struct mii_bus *bus = dsa_host_dev_to_mii_bus(ds->master_dev); + u16 r1, r2, page; + u32 val; + + split_addr((u32)reg, &r1, &r2, &page); + + mutex_lock(&bus->mdio_lock); + + bus->write(bus, 0x18, 0, page); + wait_for_page_switch(); + val = ar8xxx_mii_read32(bus, 0x10 | r2, r1); + + mutex_unlock(&bus->mdio_lock); + + return val; +} + +void ar8xxx_write(struct dsa_switch *ds, int reg, u32 val) +{ + struct mii_bus *bus = dsa_host_dev_to_mii_bus(ds->master_dev); + u16 r1, r2, page; + + split_addr((u32)reg, &r1, &r2, &page); + + mutex_lock(&bus->mdio_lock); + + bus->write(bus, 0x18, 0, page); + wait_for_page_switch(); + ar8xxx_mii_write32(bus, 0x10 | r2, r1, val); + + mutex_unlock(&bus->mdio_lock); +} + +u32 +ar8xxx_rmw(struct dsa_switch *ds, int reg, u32 mask, u32 val) +{ + struct mii_bus *bus = dsa_host_dev_to_mii_bus(ds->master_dev); + u16 r1, r2, page; + u32 ret; + + split_addr((u32)reg, &r1, &r2, &page); + + mutex_lock(&bus->mdio_lock); + + bus->write(bus, 0x18, 0, page); + wait_for_page_switch(); + + ret = ar8xxx_mii_read32(bus, 0x10 | r2, r1); + ret &= ~mask; + ret |= val; + ar8xxx_mii_write32(bus, 0x10 | r2, r1, ret); + + mutex_unlock(&bus->mdio_lock); + + return ret; +} + +static char *ar8xxx_probe(struct device *host_dev, int sw_addr) +{ + struct mii_bus *bus = dsa_host_dev_to_mii_bus(host_dev); + u32 phy_id; + + if (!bus) + return NULL; + + /* sw_addr is irrelevant as the switch occupies the MDIO bus from + * addresses 0 to 4 (PHYs) and 16-23 (for MDIO 32bits protocol). So + * we'll probe address 0 to see if we see the right switch family. + */ + phy_id = mdiobus_read(bus, 0, MII_PHYSID1) << 16; + phy_id |= mdiobus_read(bus, 0, MII_PHYSID2); + + switch (phy_id) { + case PHY_ID_QCA8337: + return "QCA8337"; + default: + return NULL; + } +} + +static int ar8xxx_set_pad_ctrl(struct dsa_switch *ds, int port, int mode) +{ + int reg; + + switch (port) { + case 0: + reg = AR8327_REG_PORT0_PAD_CTRL; + break; + case 6: + reg = AR8327_REG_PORT6_PAD_CTRL; + break; + default: + pr_err("Can't set PAD_CTRL on port %d\n", port); + return -EINVAL; + } + + /* DSA only supports 1 CPU port for now, so we'll take the assumption + * that P0 is connected to the CPU master_dev. + */ + switch (mode) { + case PHY_INTERFACE_MODE_RGMII: + ar8xxx_write(ds, reg, + AR8327_PORT_PAD_RGMII_EN | + AR8327_PORT_PAD_RGMII_TX_DELAY(3) | + AR8327_PORT_PAD_RGMII_RX_DELAY(3)); + + /* According to the datasheet, RGMII delay is enabled through + * PORT5_PAD_CTRL for all ports, rather than individual port + * registers + */ + ar8xxx_write(ds, AR8327_REG_PORT5_PAD_CTRL, + AR8327_PORT_PAD_RGMII_RX_DELAY_EN); + break; + default: + pr_err("xMII mode %d not supported\n", mode); + return -EINVAL; + } + + return 0; +} + +static int ar8xxx_setup(struct dsa_switch *ds) +{ + struct net_device *netdev = ds->dst->pd->of_netdev; + int ret, i, phy_mode; + + /* Initialize CPU port pad mode (xMII type, delays...) */ + phy_mode = of_get_phy_mode(netdev->dev.parent->of_node); + if (phy_mode < 0) { + pr_err("Can't find phy-mode for master device\n"); + return phy_mode; + } + + ret = ar8xxx_set_pad_ctrl(ds, 0, phy_mode); + if (ret < 0) + return ret; + + /* Disable forwarding by default on all ports */ + for (i = 0; i < AR8327_NUM_PORTS; i++) + ar8xxx_rmw(ds, AR8327_PORT_LOOKUP_CTRL(i), + AR8327_PORT_LOOKUP_MEMBER, 0); + + /* Setup connection between CPU ports & PHYs */ + for (i = 0; i < DSA_MAX_PORTS; i++) { + /* CPU port gets connected to all PHYs in the switch */ + if (dsa_is_cpu_port(ds, i)) { + ar8xxx_rmw(ds, AR8327_PORT_LOOKUP_CTRL(0), + AR8327_PORT_LOOKUP_MEMBER, + ds->phys_port_mask << 1); + } + + /* Invividual PHYs gets connected to CPU port only */ + if (ds->phys_port_mask & BIT(i)) { + ar8xxx_rmw(ds, AR8327_PORT_LOOKUP_CTRL(phy_to_port(i)), + AR8327_PORT_LOOKUP_MEMBER, BIT(0)); + } + } + + return 0; +} + +static int ar8xxx_set_addr(struct dsa_switch *ds, u8 *addr) +{ + return 0; +} + +static int ar8xxx_phy_read(struct dsa_switch *ds, int phy, int regnum) +{ + struct mii_bus *bus = dsa_host_dev_to_mii_bus(ds->master_dev); + + return mdiobus_read(bus, phy, regnum); +} + +static int +ar8xxx_phy_write(struct dsa_switch *ds, int phy, int regnum, u16 val) +{ + struct mii_bus *bus = dsa_host_dev_to_mii_bus(ds->master_dev); + + return mdiobus_write(bus, phy, regnum, val); +} + +static void ar8xxx_poll_link(struct dsa_switch *ds) +{ + int i = 0; + struct net_device *dev; + + while ((dev = ds->ports[i++]) != NULL) { + u32 status; + int link; + int speed; + int duplex; + + status = ar8xxx_read(ds, AR8327_REG_PORT_STATUS(i)); + link = !!(status & AR8XXX_PORT_STATUS_LINK_UP); + duplex = !!(status & AR8XXX_PORT_STATUS_DUPLEX); + + switch (status & AR8XXX_PORT_STATUS_SPEED) { + case AR8XXX_PORT_SPEED_10M: + speed = 10; + break; + case AR8XXX_PORT_SPEED_100M: + speed = 100; + break; + case AR8XXX_PORT_SPEED_1000M: + speed = 1000; + break; + default: + speed = 0; + } + + if (!link) { + /* This poll happens every ~1s, so we don't want to + * print the status every time. Only when the device + * transitions from Link UP to Link DOWN + */ + if (netif_carrier_ok(dev)) + netif_carrier_off(dev); + continue; + } else { + /* Same thing here. But we detect a Link UP event */ + if (!netif_carrier_ok(dev)) + netif_carrier_on(dev); + continue; + } + } +} + +static struct dsa_switch_driver ar8xxx_switch_driver = { + .tag_protocol = DSA_TAG_PROTO_NONE, + .probe = ar8xxx_probe, + .setup = ar8xxx_setup, + .set_addr = ar8xxx_set_addr, + .poll_link = ar8xxx_poll_link, + .phy_read = ar8xxx_phy_read, + .phy_write = ar8xxx_phy_write, +}; + +static int __init ar8xxx_init(void) +{ + register_switch_driver(&ar8xxx_switch_driver); + return 0; +} +module_init(ar8xxx_init); + +static void __exit ar8xxx_cleanup(void) +{ + unregister_switch_driver(&ar8xxx_switch_driver); +} +module_exit(ar8xxx_cleanup); + +MODULE_AUTHOR("Mathieu Olivari "); +MODULE_DESCRIPTION("Driver for AR8XXX ethernet switch family"); +MODULE_LICENSE("GPL"); +MODULE_ALIAS("platform:ar8xxx"); diff --git a/drivers/net/dsa/ar8xxx.h b/drivers/net/dsa/ar8xxx.h new file mode 100644 index 0000000..a29b6d3 --- /dev/null +++ b/drivers/net/dsa/ar8xxx.h @@ -0,0 +1,82 @@ +/* + * Copyright (C) 2009 Felix Fietkau + * Copyright (C) 2011-2012 Gabor Juhos + * Copyright (c) 2015, The Linux Foundation. All rights reserved. + * + * This program is free software; you can redistribute it and/or modify + * it under the terms of the GNU General Public License version 2 and + * only version 2 as published by the Free Software Foundation. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + */ + +#ifndef __AR8XXX_H +#define __AR8XXX_H + +#include + +#define AR8327_NUM_PORTS 7 + +#define PHY_ID_QCA8337 0x004dd036 + +#define AR8327_REG_PORT0_PAD_CTRL 0x004 +#define AR8327_REG_PORT5_PAD_CTRL 0x008 +#define AR8327_REG_PORT6_PAD_CTRL 0x00c +#define AR8327_PORT_PAD_RGMII_EN BIT(26) +#define AR8327_PORT_PAD_RGMII_TX_DELAY(x) ((0x8 + (x & 0x3)) << 22) +#define AR8327_PORT_PAD_RGMII_RX_DELAY(x) ((0x10 + (x & 0x3)) << 20) +#define AR8327_PORT_PAD_RGMII_RX_DELAY_EN BIT(24) +#define AR8327_PORT_PAD_SGMII_EN BIT(7) + +#define AR8327_REG_PORT_STATUS(_i) (0x07c + (_i) * 4) +#define AR8XXX_PORT_STATUS_SPEED GENMASK(2, 0) +#define AR8XXX_PORT_STATUS_SPEED_S 0 +#define AR8XXX_PORT_STATUS_TXMAC BIT(2) +#define AR8XXX_PORT_STATUS_RXMAC BIT(3) +#define AR8XXX_PORT_STATUS_TXFLOW BIT(4) +#define AR8XXX_PORT_STATUS_RXFLOW BIT(5) +#define AR8XXX_PORT_STATUS_DUPLEX BIT(6) +#define AR8XXX_PORT_STATUS_LINK_UP BIT(8) +#define AR8XXX_PORT_STATUS_LINK_AUTO BIT(9) +#define AR8XXX_PORT_STATUS_LINK_PAUSE BIT(10) + +#define AR8327_PORT_LOOKUP_CTRL(_i) (0x660 + (_i) * 0xc) +#define AR8327_PORT_LOOKUP_MEMBER GENMASK(6, 0) +#define AR8327_PORT_LOOKUP_IN_MODE GENMASK(9, 8) +#define AR8327_PORT_LOOKUP_IN_MODE_S 8 +#define AR8327_PORT_LOOKUP_STATE GENMASK(18, 16) +#define AR8327_PORT_LOOKUP_STATE_S 16 +#define AR8327_PORT_LOOKUP_LEARN BIT(20) +#define AR8327_PORT_LOOKUP_ING_MIRROR_EN BIT(25) + +/* port speed */ +enum { + AR8XXX_PORT_SPEED_10M = 0, + AR8XXX_PORT_SPEED_100M = 1, + AR8XXX_PORT_SPEED_1000M = 2, + AR8XXX_PORT_SPEED_ERR = 3, +}; + +static inline void +split_addr(u32 regaddr, u16 *r1, u16 *r2, u16 *page) +{ + regaddr >>= 1; + *r1 = regaddr & 0x1e; + + regaddr >>= 5; + *r2 = regaddr & 0x7; + + regaddr >>= 3; + *page = regaddr & 0x1ff; +} + +static inline void +wait_for_page_switch(void) +{ + udelay(5); +} + +#endif /* __AR8XXX_H */ diff --git a/net/dsa/dsa.c b/net/dsa/dsa.c index e6f6cc3..fffb9aa 100644 --- a/net/dsa/dsa.c +++ b/net/dsa/dsa.c @@ -893,6 +893,7 @@ static SIMPLE_DEV_PM_OPS(dsa_pm_ops, dsa_suspend, dsa_resume); static const struct of_device_id dsa_of_match_table[] = { { .compatible = "brcm,bcm7445-switch-v4.0" }, + { .compatible = "qca,ar8xxx", }, { .compatible = "marvell,dsa", }, {} }; From patchwork Fri May 29 01:42:17 2015 Content-Type: text/plain; charset="utf-8" MIME-Version: 1.0 Content-Transfer-Encoding: 7bit Subject: [2/7] net: dsa: ar8xxx: add ethtool hw statistics support From: Mathieu Olivari X-Patchwork-Id: 477524 X-Patchwork-Delegate: davem@davemloft.net Message-Id: <1432863742-18427-3-git-send-email-mathieu@codeaurora.org> To: robh+dt@kernel.org, pawel.moll@arm.com, mark.rutland@arm.com, ijc+devicetree@hellion.org.uk, galak@codeaurora.org, davem@davemloft.net, mathieu@codeaurora.org, andrew@lunn.ch, f.fainelli@gmail.com, linux@roeck-us.net, gang.chen.5i5j@gmail.com, jiri@resnulli.us, leitec@staticky.com, fabf@skynet.be, alexander.h.duyck@intel.com, pavel.nakonechny@skitlab.ru, joe@perches.com, sfeldma@gmail.com, nbd@openwrt.org, juhosg@openwrt.org Cc: devicetree@vger.kernel.org, linux-kernel@vger.kernel.org, netdev@vger.kernel.org Date: Thu, 28 May 2015 18:42:17 -0700 MIB counters can now be reported through each switch port by using "ethtool -S". Signed-off-by: Mathieu Olivari --- drivers/net/dsa/ar8xxx.c | 106 +++++++++++++++++++++++++++++++++++++++++++---- drivers/net/dsa/ar8xxx.h | 47 +++++++++++++++++++++ 2 files changed, 146 insertions(+), 7 deletions(-) diff --git a/drivers/net/dsa/ar8xxx.c b/drivers/net/dsa/ar8xxx.c index 4ce3ffc..2f0fa4d 100644 --- a/drivers/net/dsa/ar8xxx.c +++ b/drivers/net/dsa/ar8xxx.c @@ -22,6 +22,55 @@ #include "ar8xxx.h" +#define MIB_DESC(_s, _o, _n) \ + { \ + .size = (_s), \ + .offset = (_o), \ + .name = (_n), \ + } + +static const struct ar8xxx_mib_desc ar8327_mib[] = { + MIB_DESC(1, 0x00, "RxBroad"), + MIB_DESC(1, 0x04, "RxPause"), + MIB_DESC(1, 0x08, "RxMulti"), + MIB_DESC(1, 0x0c, "RxFcsErr"), + MIB_DESC(1, 0x10, "RxAlignErr"), + MIB_DESC(1, 0x14, "RxRunt"), + MIB_DESC(1, 0x18, "RxFragment"), + MIB_DESC(1, 0x1c, "Rx64Byte"), + MIB_DESC(1, 0x20, "Rx128Byte"), + MIB_DESC(1, 0x24, "Rx256Byte"), + MIB_DESC(1, 0x28, "Rx512Byte"), + MIB_DESC(1, 0x2c, "Rx1024Byte"), + MIB_DESC(1, 0x30, "Rx1518Byte"), + MIB_DESC(1, 0x34, "RxMaxByte"), + MIB_DESC(1, 0x38, "RxTooLong"), + MIB_DESC(2, 0x3c, "RxGoodByte"), + MIB_DESC(2, 0x44, "RxBadByte"), + MIB_DESC(1, 0x4c, "RxOverFlow"), + MIB_DESC(1, 0x50, "Filtered"), + MIB_DESC(1, 0x54, "TxBroad"), + MIB_DESC(1, 0x58, "TxPause"), + MIB_DESC(1, 0x5c, "TxMulti"), + MIB_DESC(1, 0x60, "TxUnderRun"), + MIB_DESC(1, 0x64, "Tx64Byte"), + MIB_DESC(1, 0x68, "Tx128Byte"), + MIB_DESC(1, 0x6c, "Tx256Byte"), + MIB_DESC(1, 0x70, "Tx512Byte"), + MIB_DESC(1, 0x74, "Tx1024Byte"), + MIB_DESC(1, 0x78, "Tx1518Byte"), + MIB_DESC(1, 0x7c, "TxMaxByte"), + MIB_DESC(1, 0x80, "TxOverSize"), + MIB_DESC(2, 0x84, "TxByte"), + MIB_DESC(1, 0x8c, "TxCollision"), + MIB_DESC(1, 0x90, "TxAbortCol"), + MIB_DESC(1, 0x94, "TxMultiCol"), + MIB_DESC(1, 0x98, "TxSingleCol"), + MIB_DESC(1, 0x9c, "TxExcDefer"), + MIB_DESC(1, 0xa0, "TxDefer"), + MIB_DESC(1, 0xa4, "TxLateCol"), +}; + u32 ar8xxx_mii_read32(struct mii_bus *bus, int phy_id, int regnum) { @@ -184,6 +233,10 @@ static int ar8xxx_setup(struct dsa_switch *ds) if (ret < 0) return ret; + /* Enable MIB counters */ + ar8xxx_reg_set(ds, AR8327_REG_MIB, AR8327_MIB_CPU_KEEP); + ar8xxx_write(ds, AR8327_REG_MODULE_EN, AR8327_MODULE_EN_MIB); + /* Disable forwarding by default on all ports */ for (i = 0; i < AR8327_NUM_PORTS; i++) ar8xxx_rmw(ds, AR8327_PORT_LOOKUP_CTRL(i), @@ -228,6 +281,42 @@ ar8xxx_phy_write(struct dsa_switch *ds, int phy, int regnum, u16 val) return mdiobus_write(bus, phy, regnum, val); } +static void ar8xxx_get_strings(struct dsa_switch *ds, int phy, uint8_t *data) +{ + int i; + + for (i = 0; i < ARRAY_SIZE(ar8327_mib); i++) { + strncpy(data + i * ETH_GSTRING_LEN, ar8327_mib[i].name, + ETH_GSTRING_LEN); + } +} + +static void ar8xxx_get_ethtool_stats(struct dsa_switch *ds, int phy, + uint64_t *data) +{ + const struct ar8xxx_mib_desc *mib; + uint32_t reg, i, port; + u64 hi; + + port = phy_to_port(phy); + + for (i = 0; i < ARRAY_SIZE(ar8327_mib); i++) { + mib = &ar8327_mib[i]; + reg = AR8327_PORT_MIB_COUNTER(port) + mib->offset; + + data[i] = ar8xxx_read(ds, reg); + if (mib->size == 2) { + hi = ar8xxx_read(ds, reg + 4); + data[i] |= hi << 32; + } + } +} + +static int ar8xxx_get_sset_count(struct dsa_switch *ds) +{ + return ARRAY_SIZE(ar8327_mib); +} + static void ar8xxx_poll_link(struct dsa_switch *ds) { int i = 0; @@ -275,13 +364,16 @@ static void ar8xxx_poll_link(struct dsa_switch *ds) } static struct dsa_switch_driver ar8xxx_switch_driver = { - .tag_protocol = DSA_TAG_PROTO_NONE, - .probe = ar8xxx_probe, - .setup = ar8xxx_setup, - .set_addr = ar8xxx_set_addr, - .poll_link = ar8xxx_poll_link, - .phy_read = ar8xxx_phy_read, - .phy_write = ar8xxx_phy_write, + .tag_protocol = DSA_TAG_PROTO_NONE, + .probe = ar8xxx_probe, + .setup = ar8xxx_setup, + .set_addr = ar8xxx_set_addr, + .poll_link = ar8xxx_poll_link, + .phy_read = ar8xxx_phy_read, + .phy_write = ar8xxx_phy_write, + .get_strings = ar8xxx_get_strings, + .get_ethtool_stats = ar8xxx_get_ethtool_stats, + .get_sset_count = ar8xxx_get_sset_count, }; static int __init ar8xxx_init(void) diff --git a/drivers/net/dsa/ar8xxx.h b/drivers/net/dsa/ar8xxx.h index a29b6d3..7c7a125 100644 --- a/drivers/net/dsa/ar8xxx.h +++ b/drivers/net/dsa/ar8xxx.h @@ -18,6 +18,12 @@ #include +struct ar8xxx_mib_desc { + unsigned int size; + unsigned int offset; + const char *name; +}; + #define AR8327_NUM_PORTS 7 #define PHY_ID_QCA8337 0x004dd036 @@ -31,6 +37,14 @@ #define AR8327_PORT_PAD_RGMII_RX_DELAY_EN BIT(24) #define AR8327_PORT_PAD_SGMII_EN BIT(7) +#define AR8327_REG_MODULE_EN 0x030 +#define AR8327_MODULE_EN_MIB BIT(0) +#define AR8327_MODULE_EN_ACL BIT(1) +#define AR8327_MODULE_EN_L3 BIT(2) + +#define AR8327_REG_MIB 0x034 +#define AR8327_MIB_CPU_KEEP BIT(20) + #define AR8327_REG_PORT_STATUS(_i) (0x07c + (_i) * 4) #define AR8XXX_PORT_STATUS_SPEED GENMASK(2, 0) #define AR8XXX_PORT_STATUS_SPEED_S 0 @@ -52,6 +66,8 @@ #define AR8327_PORT_LOOKUP_LEARN BIT(20) #define AR8327_PORT_LOOKUP_ING_MIRROR_EN BIT(25) +#define AR8327_PORT_MIB_COUNTER(_i) (0x1000 + (_i) * 0x100) + /* port speed */ enum { AR8XXX_PORT_SPEED_10M = 0, @@ -60,6 +76,25 @@ enum { AR8XXX_PORT_SPEED_ERR = 3, }; +static inline int port_to_phy(int port) +{ + if (port >= 1 && port <= 6) + return port - 1; + + return -1; +} + +static inline int phy_to_port(int phy) +{ + if (phy < 5) + return phy + 1; + + return -1; +} + +u32 +ar8xxx_rmw(struct dsa_switch *ds, int reg, u32 mask, u32 val); + static inline void split_addr(u32 regaddr, u16 *r1, u16 *r2, u16 *page) { @@ -79,4 +114,16 @@ wait_for_page_switch(void) udelay(5); } +static inline void +ar8xxx_reg_set(struct dsa_switch *ds, int reg, u32 val) +{ + ar8xxx_rmw(ds, reg, 0, val); +} + +static inline void +ar8xxx_reg_clear(struct dsa_switch *ds, int reg, u32 val) +{ + ar8xxx_rmw(ds, reg, val, 0); +} + #endif /* __AR8XXX_H */ From patchwork Fri May 29 01:42:18 2015 Content-Type: text/plain; charset="utf-8" MIME-Version: 1.0 Content-Transfer-Encoding: 7bit Subject: [3/7] net: dsa: ar8xxx: add regmap support From: Mathieu Olivari X-Patchwork-Id: 477522 X-Patchwork-Delegate: davem@davemloft.net Message-Id: <1432863742-18427-4-git-send-email-mathieu@codeaurora.org> To: robh+dt@kernel.org, pawel.moll@arm.com, mark.rutland@arm.com, ijc+devicetree@hellion.org.uk, galak@codeaurora.org, davem@davemloft.net, mathieu@codeaurora.org, andrew@lunn.ch, f.fainelli@gmail.com, linux@roeck-us.net, gang.chen.5i5j@gmail.com, jiri@resnulli.us, leitec@staticky.com, fabf@skynet.be, alexander.h.duyck@intel.com, pavel.nakonechny@skitlab.ru, joe@perches.com, sfeldma@gmail.com, nbd@openwrt.org, juhosg@openwrt.org Cc: devicetree@vger.kernel.org, linux-kernel@vger.kernel.org, netdev@vger.kernel.org Date: Thu, 28 May 2015 18:42:18 -0700 All switch registers can now be dumped using regmap/debugfs. \# cat /sys/kernel/debug/regmap//registers 0000: 00001302 0004: ... ... Signed-off-by: Mathieu Olivari --- drivers/net/dsa/Kconfig | 1 + drivers/net/dsa/ar8xxx.c | 60 ++++++++++++++++++++++++++++++++++++++++++++++++ drivers/net/dsa/ar8xxx.h | 5 ++++ 3 files changed, 66 insertions(+) diff --git a/drivers/net/dsa/Kconfig b/drivers/net/dsa/Kconfig index 2aae541..17fb296 100644 --- a/drivers/net/dsa/Kconfig +++ b/drivers/net/dsa/Kconfig @@ -68,6 +68,7 @@ config NET_DSA_BCM_SF2 config NET_DSA_AR8XXX tristate "Qualcomm Atheros AR8XXX Ethernet switch family support" depends on NET_DSA + select REGMAP ---help--- This enables support for the Qualcomm Atheros AR8XXX Ethernet switch chips. diff --git a/drivers/net/dsa/ar8xxx.c b/drivers/net/dsa/ar8xxx.c index 2f0fa4d..327abd4 100644 --- a/drivers/net/dsa/ar8xxx.c +++ b/drivers/net/dsa/ar8xxx.c @@ -176,6 +176,57 @@ static char *ar8xxx_probe(struct device *host_dev, int sw_addr) } } +static int ar8xxx_regmap_read(void *ctx, uint32_t reg, uint32_t *val) +{ + struct dsa_switch *ds = (struct dsa_switch *)ctx; + + *val = ar8xxx_read(ds, reg); + + return 0; +} + +static int ar8xxx_regmap_write(void *ctx, uint32_t reg, uint32_t val) +{ + struct dsa_switch *ds = (struct dsa_switch *)ctx; + + ar8xxx_write(ds, reg, val); + + return 0; +} + +static const struct regmap_range ar8xxx_readable_ranges[] = { + regmap_reg_range(0x0000, 0x00e4), /* Global control */ + regmap_reg_range(0x0100, 0x0168), /* EEE control */ + regmap_reg_range(0x0200, 0x0270), /* Parser control */ + regmap_reg_range(0x0400, 0x0454), /* ACL */ + regmap_reg_range(0x0600, 0x0718), /* Lookup */ + regmap_reg_range(0x0800, 0x0b70), /* QM */ + regmap_reg_range(0x0C00, 0x0c80), /* PKT */ + regmap_reg_range(0x1000, 0x10ac), /* MIB - Port0 */ + regmap_reg_range(0x1100, 0x11ac), /* MIB - Port1 */ + regmap_reg_range(0x1200, 0x12ac), /* MIB - Port2 */ + regmap_reg_range(0x1300, 0x13ac), /* MIB - Port3 */ + regmap_reg_range(0x1400, 0x14ac), /* MIB - Port4 */ + regmap_reg_range(0x1500, 0x15ac), /* MIB - Port5 */ + regmap_reg_range(0x1600, 0x16ac), /* MIB - Port6 */ + +}; + +static struct regmap_access_table ar8xxx_readable_table = { + .yes_ranges = ar8xxx_readable_ranges, + .n_yes_ranges = ARRAY_SIZE(ar8xxx_readable_ranges), +}; + +struct regmap_config ar8xxx_regmap_config = { + .reg_bits = 16, + .val_bits = 32, + .reg_stride = 4, + .max_register = 0x16ac, /* end MIB - Port6 range */ + .reg_read = ar8xxx_regmap_read, + .reg_write = ar8xxx_regmap_write, + .rd_table = &ar8xxx_readable_table, +}; + static int ar8xxx_set_pad_ctrl(struct dsa_switch *ds, int port, int mode) { int reg; @@ -219,9 +270,17 @@ static int ar8xxx_set_pad_ctrl(struct dsa_switch *ds, int port, int mode) static int ar8xxx_setup(struct dsa_switch *ds) { + struct ar8xxx_priv *priv = ds_to_priv(ds); struct net_device *netdev = ds->dst->pd->of_netdev; int ret, i, phy_mode; + /* Start by setting up the register mapping */ + priv->regmap = devm_regmap_init(ds->master_dev, NULL, ds, + &ar8xxx_regmap_config); + + if (IS_ERR(priv->regmap)) + pr_warn("regmap initialization failed"); + /* Initialize CPU port pad mode (xMII type, delays...) */ phy_mode = of_get_phy_mode(netdev->dev.parent->of_node); if (phy_mode < 0) { @@ -365,6 +424,7 @@ static void ar8xxx_poll_link(struct dsa_switch *ds) static struct dsa_switch_driver ar8xxx_switch_driver = { .tag_protocol = DSA_TAG_PROTO_NONE, + .priv_size = sizeof(struct ar8xxx_priv), .probe = ar8xxx_probe, .setup = ar8xxx_setup, .set_addr = ar8xxx_set_addr, diff --git a/drivers/net/dsa/ar8xxx.h b/drivers/net/dsa/ar8xxx.h index 7c7a125..98cc7ed 100644 --- a/drivers/net/dsa/ar8xxx.h +++ b/drivers/net/dsa/ar8xxx.h @@ -17,6 +17,11 @@ #define __AR8XXX_H #include +#include + +struct ar8xxx_priv { + struct regmap *regmap; +}; struct ar8xxx_mib_desc { unsigned int size; From patchwork Fri May 29 01:42:19 2015 Content-Type: text/plain; charset="utf-8" MIME-Version: 1.0 Content-Transfer-Encoding: 7bit Subject: [4/7] net: dsa: add QCA tag support From: Mathieu Olivari X-Patchwork-Id: 477521 X-Patchwork-Delegate: davem@davemloft.net Message-Id: <1432863742-18427-5-git-send-email-mathieu@codeaurora.org> To: robh+dt@kernel.org, pawel.moll@arm.com, mark.rutland@arm.com, ijc+devicetree@hellion.org.uk, galak@codeaurora.org, davem@davemloft.net, mathieu@codeaurora.org, andrew@lunn.ch, f.fainelli@gmail.com, linux@roeck-us.net, gang.chen.5i5j@gmail.com, jiri@resnulli.us, leitec@staticky.com, fabf@skynet.be, alexander.h.duyck@intel.com, pavel.nakonechny@skitlab.ru, joe@perches.com, sfeldma@gmail.com, nbd@openwrt.org, juhosg@openwrt.org Cc: devicetree@vger.kernel.org, linux-kernel@vger.kernel.org, netdev@vger.kernel.org Date: Thu, 28 May 2015 18:42:19 -0700 QCA tags are used on QCA ar8xxx switch family. This change adds support for encap/decap using 2 bytes header mode. Signed-off-by: Mathieu Olivari --- include/net/dsa.h | 1 + net/dsa/Kconfig | 3 + net/dsa/Makefile | 1 + net/dsa/dsa.c | 5 ++ net/dsa/dsa_priv.h | 2 + net/dsa/slave.c | 5 ++ net/dsa/tag_qca.c | 158 +++++++++++++++++++++++++++++++++++++++++++++++++++++ 7 files changed, 175 insertions(+) create mode 100644 net/dsa/tag_qca.c diff --git a/include/net/dsa.h b/include/net/dsa.h index fbca63b..64ddf6f 100644 --- a/include/net/dsa.h +++ b/include/net/dsa.h @@ -26,6 +26,7 @@ enum dsa_tag_protocol { DSA_TAG_PROTO_TRAILER, DSA_TAG_PROTO_EDSA, DSA_TAG_PROTO_BRCM, + DSA_TAG_PROTO_QCA, }; #define DSA_MAX_SWITCHES 4 diff --git a/net/dsa/Kconfig b/net/dsa/Kconfig index ff7736f..4f3cce1 100644 --- a/net/dsa/Kconfig +++ b/net/dsa/Kconfig @@ -26,6 +26,9 @@ config NET_DSA_HWMON via the hwmon sysfs interface and exposes the onboard sensors. # tagging formats +config NET_DSA_TAG_QCA + bool + config NET_DSA_TAG_BRCM bool diff --git a/net/dsa/Makefile b/net/dsa/Makefile index da06ed1..9feb86c 100644 --- a/net/dsa/Makefile +++ b/net/dsa/Makefile @@ -3,6 +3,7 @@ obj-$(CONFIG_NET_DSA) += dsa_core.o dsa_core-y += dsa.o slave.o # tagging formats +dsa_core-$(CONFIG_NET_DSA_TAG_QCA) += tag_qca.o dsa_core-$(CONFIG_NET_DSA_TAG_BRCM) += tag_brcm.o dsa_core-$(CONFIG_NET_DSA_TAG_DSA) += tag_dsa.o dsa_core-$(CONFIG_NET_DSA_TAG_EDSA) += tag_edsa.o diff --git a/net/dsa/dsa.c b/net/dsa/dsa.c index fffb9aa..6010a7d 100644 --- a/net/dsa/dsa.c +++ b/net/dsa/dsa.c @@ -249,6 +249,11 @@ static int dsa_switch_setup_one(struct dsa_switch *ds, struct device *parent) dst->rcv = brcm_netdev_ops.rcv; break; #endif +#ifdef CONFIG_NET_DSA_TAG_QCA + case DSA_TAG_PROTO_QCA: + dst->rcv = qca_netdev_ops.rcv; + break; +#endif case DSA_TAG_PROTO_NONE: break; default: diff --git a/net/dsa/dsa_priv.h b/net/dsa/dsa_priv.h index d5f1f9b..350c94b 100644 --- a/net/dsa/dsa_priv.h +++ b/net/dsa/dsa_priv.h @@ -74,5 +74,7 @@ extern const struct dsa_device_ops trailer_netdev_ops; /* tag_brcm.c */ extern const struct dsa_device_ops brcm_netdev_ops; +/* tag_qca.c */ +extern const struct dsa_device_ops qca_netdev_ops; #endif diff --git a/net/dsa/slave.c b/net/dsa/slave.c index 04ffad3..cd8f552 100644 --- a/net/dsa/slave.c +++ b/net/dsa/slave.c @@ -925,6 +925,11 @@ int dsa_slave_create(struct dsa_switch *ds, struct device *parent, p->xmit = brcm_netdev_ops.xmit; break; #endif +#ifdef CONFIG_NET_DSA_TAG_QCA + case DSA_TAG_PROTO_QCA: + p->xmit = qca_netdev_ops.xmit; + break; +#endif default: p->xmit = dsa_slave_notag_xmit; break; diff --git a/net/dsa/tag_qca.c b/net/dsa/tag_qca.c new file mode 100644 index 0000000..8f02196 --- /dev/null +++ b/net/dsa/tag_qca.c @@ -0,0 +1,158 @@ +/* + * Copyright (c) 2015, The Linux Foundation. All rights reserved. + * + * This program is free software; you can redistribute it and/or modify + * it under the terms of the GNU General Public License version 2 and + * only version 2 as published by the Free Software Foundation. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + */ + +#include +#include "dsa_priv.h" + +#define QCA_HDR_LEN 2 +#define QCA_HDR_VERSION 0x2 + +#define QCA_HDR_RECV_VERSION_MASK GENMASK(15, 14) +#define QCA_HDR_RECV_VERSION_S 14 +#define QCA_HDR_RECV_PRIORITY_MASK GENMASK(13, 11) +#define QCA_HDR_RECV_PRIORITY_S 11 +#define QCA_HDR_RECV_TYPE_MASK GENMASK(10, 6) +#define QCA_HDR_RECV_TYPE_S 6 +#define QCA_HDR_RECV_FRAME_IS_TAGGED BIT(3) +#define QCA_HDR_RECV_SOURCE_PORT_MASK GENMASK(2, 0) + +#define QCA_HDR_XMIT_VERSION_MASK GENMASK(15, 14) +#define QCA_HDR_XMIT_VERSION_S 14 +#define QCA_HDR_XMIT_PRIORITY_MASK GENMASK(13, 11) +#define QCA_HDR_XMIT_PRIORITY_S 11 +#define QCA_HDR_XMIT_CONTROL_MASK GENMASK(10, 8) +#define QCA_HDR_XMIT_CONTROL_S 8 +#define QCA_HDR_XMIT_FROM_CPU BIT(7) +#define QCA_HDR_XMIT_DP_BIT_MASK GENMASK(6, 0) + +static inline int reg_to_port(int reg) +{ + if (reg < 5) + return reg + 1; + + return -1; +} + +static inline int port_to_reg(int port) +{ + if (port >= 1 && port <= 6) + return port - 1; + + return -1; +} + +static netdev_tx_t qca_tag_xmit(struct sk_buff *skb, struct net_device *dev) +{ + struct dsa_slave_priv *p = netdev_priv(dev); + u16 *phdr, hdr; + + dev->stats.tx_packets++; + dev->stats.tx_bytes += skb->len; + + if (skb_cow_head(skb, 0) < 0) + goto out_free; + + skb_push(skb, QCA_HDR_LEN); + + memmove(skb->data, skb->data + QCA_HDR_LEN, 2 * ETH_ALEN); + phdr = (u16 *)(skb->data + 2 * ETH_ALEN); + + /* Set the version field, and set destination port information */ + hdr = QCA_HDR_VERSION << QCA_HDR_XMIT_VERSION_S | + QCA_HDR_XMIT_FROM_CPU | + 1 << reg_to_port(p->port); + + *phdr = htons(hdr); + + skb->dev = p->parent->dst->master_netdev; + dev_queue_xmit(skb); + + return NETDEV_TX_OK; + +out_free: + kfree_skb(skb); + return NETDEV_TX_OK; +} + +static int qca_tag_rcv(struct sk_buff *skb, struct net_device *dev, + struct packet_type *pt, struct net_device *orig_dev) +{ + struct dsa_switch_tree *dst = dev->dsa_ptr; + struct dsa_switch *ds; + u8 ver; + int port, phy; + __be16 *phdr, hdr; + + if (unlikely(!dst)) + goto out_drop; + + skb = skb_unshare(skb, GFP_ATOMIC); + if (!skb) + goto out; + + if (unlikely(!pskb_may_pull(skb, QCA_HDR_LEN))) + goto out_drop; + + /* Ethernet is added by the switch between src addr and Ethertype + * At this point, skb->data points to ethertype so header should be + * right before + */ + phdr = (__be16 *)(skb->data - 2); + hdr = ntohs(*phdr); + + /* Make sure the version is correct */ + ver = (hdr & QCA_HDR_RECV_VERSION_MASK) >> QCA_HDR_RECV_VERSION_S; + if (unlikely(ver != QCA_HDR_VERSION)) + goto out_drop; + + /* Remove QCA tag and recalculate checksum */ + skb_pull_rcsum(skb, QCA_HDR_LEN); + memmove(skb->data - ETH_HLEN, skb->data - ETH_HLEN - QCA_HDR_LEN, + ETH_HLEN - QCA_HDR_LEN); + + /* This protocol doesn't support cascading multiple switches so it's + * safe to assume the switch is first in the tree + */ + ds = dst->ds[0]; + if (!ds) + goto out_drop; + + /* Get source port information */ + port = (hdr & QCA_HDR_RECV_SOURCE_PORT_MASK); + phy = port_to_reg(port); + if (unlikely(phy < 0) || !ds->ports[phy]) + goto out_drop; + + /* Update skb & forward the frame accordingly */ + skb_push(skb, ETH_HLEN); + skb->pkt_type = PACKET_HOST; + skb->dev = ds->ports[phy]; + skb->protocol = eth_type_trans(skb, skb->dev); + + skb->dev->stats.rx_packets++; + skb->dev->stats.rx_bytes += skb->len; + + netif_receive_skb(skb); + + return 0; + +out_drop: + kfree_skb(skb); +out: + return 0; +} + +const struct dsa_device_ops qca_netdev_ops = { + .xmit = qca_tag_xmit, + .rcv = qca_tag_rcv, +}; From patchwork Fri May 29 01:42:20 2015 Content-Type: text/plain; charset="utf-8" MIME-Version: 1.0 Content-Transfer-Encoding: 7bit Subject: [5/7] net: dsa: ar8xxx: enable QCA header support on AR8xxx From: Mathieu Olivari X-Patchwork-Id: 477527 X-Patchwork-Delegate: davem@davemloft.net Message-Id: <1432863742-18427-6-git-send-email-mathieu@codeaurora.org> To: robh+dt@kernel.org, pawel.moll@arm.com, mark.rutland@arm.com, ijc+devicetree@hellion.org.uk, galak@codeaurora.org, davem@davemloft.net, mathieu@codeaurora.org, andrew@lunn.ch, f.fainelli@gmail.com, linux@roeck-us.net, gang.chen.5i5j@gmail.com, jiri@resnulli.us, leitec@staticky.com, fabf@skynet.be, alexander.h.duyck@intel.com, pavel.nakonechny@skitlab.ru, joe@perches.com, sfeldma@gmail.com, nbd@openwrt.org, juhosg@openwrt.org Cc: devicetree@vger.kernel.org, linux-kernel@vger.kernel.org, netdev@vger.kernel.org Date: Thu, 28 May 2015 18:42:20 -0700 This change enable support for the QCA headers in QCA83337 driver. A 2 bytes header will be added by the switch on every incoming packet to identify the ingress port, and the DSA tagging code will add a similar 2 bytes header to control which port is used to send a particular packet. Signed-off-by: Mathieu Olivari --- drivers/net/dsa/Kconfig | 1 + drivers/net/dsa/ar8xxx.c | 28 ++++++++++++++++++++++++++-- drivers/net/dsa/ar8xxx.h | 22 ++++++++++++++++++++++ 3 files changed, 49 insertions(+), 2 deletions(-) diff --git a/drivers/net/dsa/Kconfig b/drivers/net/dsa/Kconfig index 17fb296..fa8b484 100644 --- a/drivers/net/dsa/Kconfig +++ b/drivers/net/dsa/Kconfig @@ -68,6 +68,7 @@ config NET_DSA_BCM_SF2 config NET_DSA_AR8XXX tristate "Qualcomm Atheros AR8XXX Ethernet switch family support" depends on NET_DSA + select NET_DSA_TAG_QCA select REGMAP ---help--- This enables support for the Qualcomm Atheros AR8XXX Ethernet diff --git a/drivers/net/dsa/ar8xxx.c b/drivers/net/dsa/ar8xxx.c index 327abd4..4044614 100644 --- a/drivers/net/dsa/ar8xxx.c +++ b/drivers/net/dsa/ar8xxx.c @@ -292,15 +292,31 @@ static int ar8xxx_setup(struct dsa_switch *ds) if (ret < 0) return ret; + /* Enable CPU Port */ + ar8xxx_reg_set(ds, AR8327_REG_GLOBAL_FW_CTRL0, + AR8327_GLOBAL_FW_CTRL0_CPU_PORT_EN); + /* Enable MIB counters */ ar8xxx_reg_set(ds, AR8327_REG_MIB, AR8327_MIB_CPU_KEEP); ar8xxx_write(ds, AR8327_REG_MODULE_EN, AR8327_MODULE_EN_MIB); + /* Enable QCA header mode on Port 0 */ + ar8xxx_write(ds, AR8327_REG_PORT_HDR_CTRL(0), + AR8327_PORT_HDR_CTRL_ALL << AR8327_PORT_HDR_CTRL_TX_S | + AR8327_PORT_HDR_CTRL_ALL << AR8327_PORT_HDR_CTRL_RX_S); + /* Disable forwarding by default on all ports */ for (i = 0; i < AR8327_NUM_PORTS; i++) ar8xxx_rmw(ds, AR8327_PORT_LOOKUP_CTRL(i), AR8327_PORT_LOOKUP_MEMBER, 0); + /* Forward all unknown frames to CPU port for Linux processing */ + ar8xxx_write(ds, AR8327_REG_GLOBAL_FW_CTRL1, + BIT(0) << AR8327_GLOBAL_FW_CTRL1_IGMP_DP_S | + BIT(0) << AR8327_GLOBAL_FW_CTRL1_BC_DP_S | + BIT(0) << AR8327_GLOBAL_FW_CTRL1_MC_DP_S | + BIT(0) << AR8327_GLOBAL_FW_CTRL1_UC_DP_S); + /* Setup connection between CPU ports & PHYs */ for (i = 0; i < DSA_MAX_PORTS; i++) { /* CPU port gets connected to all PHYs in the switch */ @@ -312,8 +328,16 @@ static int ar8xxx_setup(struct dsa_switch *ds) /* Invividual PHYs gets connected to CPU port only */ if (ds->phys_port_mask & BIT(i)) { - ar8xxx_rmw(ds, AR8327_PORT_LOOKUP_CTRL(phy_to_port(i)), + int phy = phy_to_port(i); + + ar8xxx_rmw(ds, AR8327_PORT_LOOKUP_CTRL(phy), AR8327_PORT_LOOKUP_MEMBER, BIT(0)); + + /* Disable Auto-learning by default so the switch + * doesn't try to forward the frame to another port + */ + ar8xxx_reg_clear(ds, AR8327_PORT_LOOKUP_CTRL(phy), + AR8327_PORT_LOOKUP_LEARN); } } @@ -423,7 +447,7 @@ static void ar8xxx_poll_link(struct dsa_switch *ds) } static struct dsa_switch_driver ar8xxx_switch_driver = { - .tag_protocol = DSA_TAG_PROTO_NONE, + .tag_protocol = DSA_TAG_PROTO_QCA, .priv_size = sizeof(struct ar8xxx_priv), .probe = ar8xxx_probe, .setup = ar8xxx_setup, diff --git a/drivers/net/dsa/ar8xxx.h b/drivers/net/dsa/ar8xxx.h index 98cc7ed..e68b92a 100644 --- a/drivers/net/dsa/ar8xxx.h +++ b/drivers/net/dsa/ar8xxx.h @@ -62,6 +62,28 @@ struct ar8xxx_mib_desc { #define AR8XXX_PORT_STATUS_LINK_AUTO BIT(9) #define AR8XXX_PORT_STATUS_LINK_PAUSE BIT(10) +#define AR8327_REG_PORT_HDR_CTRL(_i) (0x9c + (_i * 4)) +#define AR8327_PORT_HDR_CTRL_RX_MASK GENMASK(3, 2) +#define AR8327_PORT_HDR_CTRL_RX_S 2 +#define AR8327_PORT_HDR_CTRL_TX_MASK GENMASK(1, 0) +#define AR8327_PORT_HDR_CTRL_TX_S 0 +#define AR8327_PORT_HDR_CTRL_ALL 2 +#define AR8327_PORT_HDR_CTRL_MGMT 1 +#define AR8327_PORT_HDR_CTRL_NONE 0 + +#define AR8327_REG_GLOBAL_FW_CTRL0 0x620 +#define AR8327_GLOBAL_FW_CTRL0_CPU_PORT_EN BIT(10) + +#define AR8327_REG_GLOBAL_FW_CTRL1 0x624 +#define AR8327_GLOBAL_FW_CTRL1_IGMP_DP_MASK GENMASK(30, 24) +#define AR8327_GLOBAL_FW_CTRL1_IGMP_DP_S 24 +#define AR8327_GLOBAL_FW_CTRL1_BC_DP_MASK GENMASK(22, 16) +#define AR8327_GLOBAL_FW_CTRL1_BC_DP_S 16 +#define AR8327_GLOBAL_FW_CTRL1_MC_DP_MASK GENMASK(14, 8) +#define AR8327_GLOBAL_FW_CTRL1_MC_DP_S 8 +#define AR8327_GLOBAL_FW_CTRL1_UC_DP_MASK GENMASK(6, 0) +#define AR8327_GLOBAL_FW_CTRL1_UC_DP_S 0 + #define AR8327_PORT_LOOKUP_CTRL(_i) (0x660 + (_i) * 0xc) #define AR8327_PORT_LOOKUP_MEMBER GENMASK(6, 0) #define AR8327_PORT_LOOKUP_IN_MODE GENMASK(9, 8) From patchwork Fri May 29 01:42:21 2015 Content-Type: text/plain; charset="utf-8" MIME-Version: 1.0 Content-Transfer-Encoding: 7bit Subject: [6/7] net: dsa: ar8xxx: add support for second xMII interfaces through DT From: Mathieu Olivari X-Patchwork-Id: 477525 X-Patchwork-Delegate: davem@davemloft.net Message-Id: <1432863742-18427-7-git-send-email-mathieu@codeaurora.org> To: robh+dt@kernel.org, pawel.moll@arm.com, mark.rutland@arm.com, ijc+devicetree@hellion.org.uk, galak@codeaurora.org, davem@davemloft.net, mathieu@codeaurora.org, andrew@lunn.ch, f.fainelli@gmail.com, linux@roeck-us.net, gang.chen.5i5j@gmail.com, jiri@resnulli.us, leitec@staticky.com, fabf@skynet.be, alexander.h.duyck@intel.com, pavel.nakonechny@skitlab.ru, joe@perches.com, sfeldma@gmail.com, nbd@openwrt.org, juhosg@openwrt.org Cc: devicetree@vger.kernel.org, linux-kernel@vger.kernel.org, netdev@vger.kernel.org Date: Thu, 28 May 2015 18:42:21 -0700 This patch is adding support for port6 specific options to device tree. They can be used to setup the second xMII interface, and connect it to one of the switch port. Signed-off-by: Mathieu Olivari --- drivers/net/dsa/ar8xxx.c | 50 ++++++++++++++++++++++++++++++++++++++++++++++++ 1 file changed, 50 insertions(+) diff --git a/drivers/net/dsa/ar8xxx.c b/drivers/net/dsa/ar8xxx.c index 4044614..7559249 100644 --- a/drivers/net/dsa/ar8xxx.c +++ b/drivers/net/dsa/ar8xxx.c @@ -19,6 +19,7 @@ #include #include #include +#include #include "ar8xxx.h" @@ -260,6 +261,9 @@ static int ar8xxx_set_pad_ctrl(struct dsa_switch *ds, int port, int mode) ar8xxx_write(ds, AR8327_REG_PORT5_PAD_CTRL, AR8327_PORT_PAD_RGMII_RX_DELAY_EN); break; + case PHY_INTERFACE_MODE_SGMII: + ar8xxx_write(ds, reg, AR8327_PORT_PAD_SGMII_EN); + break; default: pr_err("xMII mode %d not supported\n", mode); return -EINVAL; @@ -268,6 +272,48 @@ static int ar8xxx_set_pad_ctrl(struct dsa_switch *ds, int port, int mode) return 0; } +static int ar8xxx_of_setup(struct dsa_switch *ds) +{ + struct device_node *dn = ds->pd->of_node; + const char *s_phymode; + int ret, mode; + u32 phy_id, ctrl; + + /* If port6-phy-mode property exists, configure it accordingly */ + if (!of_property_read_string(dn, "qca,port6-phy-mode", &s_phymode)) { + for (mode = 0; mode < PHY_INTERFACE_MODE_MAX; mode++) + if (!strcasecmp(s_phymode, phy_modes(mode))) + break; + + if (mode == PHY_INTERFACE_MODE_MAX) + pr_err("Unknown phy-mode: \"%s\"\n", s_phymode); + + ret = ar8xxx_set_pad_ctrl(ds, 6, mode); + if (ret < 0) + return ret; + } + + /* If a phy ID is specified for PORT6 mac, connect them together */ + if (!of_property_read_u32(dn, "qca,port6-phy-id", &phy_id)) { + ar8xxx_rmw(ds, AR8327_PORT_LOOKUP_CTRL(6), + AR8327_PORT_LOOKUP_MEMBER, BIT(phy_to_port(phy_id))); + ar8xxx_rmw(ds, AR8327_PORT_LOOKUP_CTRL(phy_to_port(phy_id)), + AR8327_PORT_LOOKUP_MEMBER, BIT(6)); + + /* We want the switch to be pass-through and act like a PHY on + * these ports. So BC/MC/UC & IGMP frames need to be accepted + */ + ctrl = BIT(phy_to_port(phy_id)) | BIT(6); + ar8xxx_reg_set(ds, AR8327_REG_GLOBAL_FW_CTRL1, + ctrl << AR8327_GLOBAL_FW_CTRL1_IGMP_DP_S | + ctrl << AR8327_GLOBAL_FW_CTRL1_BC_DP_S | + ctrl << AR8327_GLOBAL_FW_CTRL1_MC_DP_S | + ctrl << AR8327_GLOBAL_FW_CTRL1_UC_DP_S); + } + + return 0; +} + static int ar8xxx_setup(struct dsa_switch *ds) { struct ar8xxx_priv *priv = ds_to_priv(ds); @@ -341,6 +387,10 @@ static int ar8xxx_setup(struct dsa_switch *ds) } } + ret = ar8xxx_of_setup(ds); + if (ret < 0) + return ret; + return 0; } From patchwork Fri May 29 01:42:22 2015 Content-Type: text/plain; charset="utf-8" MIME-Version: 1.0 Content-Transfer-Encoding: 7bit Subject: [7/7] Documentation: devicetree: add ar8xxx binding From: Mathieu Olivari X-Patchwork-Id: 477528 X-Patchwork-Delegate: davem@davemloft.net Message-Id: <1432863742-18427-8-git-send-email-mathieu@codeaurora.org> To: robh+dt@kernel.org, pawel.moll@arm.com, mark.rutland@arm.com, ijc+devicetree@hellion.org.uk, galak@codeaurora.org, davem@davemloft.net, mathieu@codeaurora.org, andrew@lunn.ch, f.fainelli@gmail.com, linux@roeck-us.net, gang.chen.5i5j@gmail.com, jiri@resnulli.us, leitec@staticky.com, fabf@skynet.be, alexander.h.duyck@intel.com, pavel.nakonechny@skitlab.ru, joe@perches.com, sfeldma@gmail.com, nbd@openwrt.org, juhosg@openwrt.org Cc: devicetree@vger.kernel.org, linux-kernel@vger.kernel.org, netdev@vger.kernel.org Date: Thu, 28 May 2015 18:42:22 -0700 Add device-tree binding for ar8xxx switch families. Signed-off-by: Mathieu Olivari --- .../devicetree/bindings/net/dsa/qca-ar8xxx.txt | 70 ++++++++++++++++++++++ 1 file changed, 70 insertions(+) create mode 100644 Documentation/devicetree/bindings/net/dsa/qca-ar8xxx.txt diff --git a/Documentation/devicetree/bindings/net/dsa/qca-ar8xxx.txt b/Documentation/devicetree/bindings/net/dsa/qca-ar8xxx.txt new file mode 100644 index 0000000..f4fd3f1 --- /dev/null +++ b/Documentation/devicetree/bindings/net/dsa/qca-ar8xxx.txt @@ -0,0 +1,70 @@ +* Qualcomm Atheros AR8xxx switch family + +Required properties: + +- compatible: should be "qca,ar8xxx" +- dsa,mii-bus: phandle to the MDIO bus controller, see dsa/dsa.txt +- dsa,ethernet: phandle to the CPU network interface controller, see dsa/dsa.txt +- #size-cells: must be 0 +- #address-cells: must be 2, see dsa/dsa.txt + +Subnodes: + +The integrated switch subnode should be specified according to the binding +described in dsa/dsa.txt. + +Optional properties: + +- qca,port6-phy-mode: if specified, the driver will configure Port 6 in the + given phy-mode. See Documentation/devicetree/bindings/net/ethernet.txt for + the list of valid phy-mode. + +- qca,port6-phy-id: if specified, the driver will connect Port 6 to the PHY + given as a parameter. In this case, Port6 and the corresponding PHY will be + isolated from the rest of the switch. From a system perspective, they will + act as a regular PHY. + +Example: + + dsa@0 { + compatible = "qca,ar8xxx"; + #address-cells = <2>; + #size-cells = <0>; + + dsa,ethernet = <ðernet0>; + dsa,mii-bus = <&mii_bus0>; + + switch@0 { + #address-cells = <1>; + #size-cells = <0>; + reg = <0 0>; /* MDIO address 0, switch 0 in tree */ + + qca,port6-phy-mode = "sgmii"; + qca,port6-phy-id = <4>; + + port@0 { + reg = <11>; + label = "cpu"; + }; + + port@1 { + reg = <0>; + label = "lan1"; + }; + + port@2 { + reg = <1>; + label = "lan2"; + }; + + port@3 { + reg = <2>; + label = "lan3"; + }; + + port@4 { + reg = <3>; + label = "lan4"; + }; + }; + };