Message ID | 20191105181826.25114-2-dmurphy@ti.com |
---|---|
State | New |
Headers | show |
Series | [1/2] dt-bindings: net: dp83869: Add TI dp83869 phy | expand |
On 05.11.2019 19:18, Dan Murphy wrote: > Add support for the TI DP83869 Gigabit ethernet phy > device. > > The DP83869 is a robust, low power, fully featured > Physical Layer transceiver with integrated PMD > sublayers to support 10BASE-T, 100BASE-TX and > 1000BASE-T Ethernet protocols. > > Signed-off-by: Dan Murphy <dmurphy@ti.com> > --- > drivers/net/phy/Kconfig | 6 + > drivers/net/phy/Makefile | 1 + > drivers/net/phy/dp83869.c | 439 +++++++++++++++++++++++++++ > include/dt-bindings/net/ti-dp83869.h | 43 +++ > 4 files changed, 489 insertions(+) > create mode 100644 drivers/net/phy/dp83869.c > create mode 100644 include/dt-bindings/net/ti-dp83869.h > > diff --git a/drivers/net/phy/Kconfig b/drivers/net/phy/Kconfig > index fe602648b99f..4f8db1bb72a8 100644 > --- a/drivers/net/phy/Kconfig > +++ b/drivers/net/phy/Kconfig > @@ -364,6 +364,12 @@ config DP83867_PHY > ---help--- > Currently supports the DP83867 PHY. > > +config DP83869_PHY > + tristate "Texas Instruments DP83869 Gigabit PHY" > + ---help--- > + Currently supports the DP83869 PHY. This PHY supports copper and > + fiber connections. > + > config FIXED_PHY > tristate "MDIO Bus/PHY emulation with fixed speed/link PHYs" > depends on PHYLIB > diff --git a/drivers/net/phy/Makefile b/drivers/net/phy/Makefile > index a03437e091f3..b433ec3bf9a6 100644 > --- a/drivers/net/phy/Makefile > +++ b/drivers/net/phy/Makefile > @@ -70,6 +70,7 @@ obj-$(CONFIG_DP83822_PHY) += dp83822.o > obj-$(CONFIG_DP83TC811_PHY) += dp83tc811.o > obj-$(CONFIG_DP83848_PHY) += dp83848.o > obj-$(CONFIG_DP83867_PHY) += dp83867.o > +obj-$(CONFIG_DP83869_PHY) += dp83869.o > obj-$(CONFIG_FIXED_PHY) += fixed_phy.o > obj-$(CONFIG_ICPLUS_PHY) += icplus.o > obj-$(CONFIG_INTEL_XWAY_PHY) += intel-xway.o > diff --git a/drivers/net/phy/dp83869.c b/drivers/net/phy/dp83869.c > new file mode 100644 > index 000000000000..eecc9b3129ca > --- /dev/null > +++ b/drivers/net/phy/dp83869.c > @@ -0,0 +1,439 @@ > +// SPDX-License-Identifier: GPL-2.0 > +/* Driver for the Texas Instruments DP83869 PHY > + * Copyright (C) 2019 Texas Instruments Inc. > + */ > + > +#include <linux/ethtool.h> > +#include <linux/kernel.h> > +#include <linux/mii.h> > +#include <linux/module.h> > +#include <linux/of.h> > +#include <linux/phy.h> > +#include <linux/delay.h> > + > +#include <dt-bindings/net/ti-dp83869.h> > + > +#define DP83869_PHY_ID 0x2000a0f1 > +#define DP83869_DEVADDR 0x1f > + > +#define MII_DP83869_CFG1 0x09 Is this different from standard register MII_CTRL1000 ? > +#define MII_DP83869_PHYCTRL 0x10 > +#define MII_DP83869_MICR 0x12 > +#define MII_DP83869_ISR 0x13 > +#define DP83869_CTRL 0x1f > +#define DP83869_CFG4 0x1e > + > +/* Extended Registers */ > +#define DP83869_GEN_CFG3 0x0031 > +#define DP83869_RGMIICTL 0x0032 > +#define DP83869_STRAP_STS1 0x006e > +#define DP83869_RGMIIDCTL 0x0086 > +#define DP83869_IO_MUX_CFG 0x0170 > +#define DP83869_OP_MODE 0x01df > +#define DP83869_FX_CTRL 0x0c00 > + > +#define DP83869_SW_RESET BIT(15) > +#define DP83869_SW_RESTART BIT(14) > + > +/* MICR Interrupt bits */ > +#define MII_DP83869_MICR_AN_ERR_INT_EN BIT(15) > +#define MII_DP83869_MICR_SPEED_CHNG_INT_EN BIT(14) > +#define MII_DP83869_MICR_DUP_MODE_CHNG_INT_EN BIT(13) > +#define MII_DP83869_MICR_PAGE_RXD_INT_EN BIT(12) > +#define MII_DP83869_MICR_AUTONEG_COMP_INT_EN BIT(11) > +#define MII_DP83869_MICR_LINK_STS_CHNG_INT_EN BIT(10) > +#define MII_DP83869_MICR_FALSE_CARRIER_INT_EN BIT(8) > +#define MII_DP83869_MICR_SLEEP_MODE_CHNG_INT_EN BIT(4) > +#define MII_DP83869_MICR_WOL_INT_EN BIT(3) > +#define MII_DP83869_MICR_XGMII_ERR_INT_EN BIT(2) > +#define MII_DP83869_MICR_POL_CHNG_INT_EN BIT(1) > +#define MII_DP83869_MICR_JABBER_INT_EN BIT(0) > + > +#define MII_DP83869_BMCR_DEFAULT (BMCR_ANENABLE | \ > + BMCR_FULLDPLX | \ > + BMCR_SPEED1000) > + > +/* This is the same bit mask as the BMCR so re-use the BMCR default */ > +#define DP83869_FX_CTRL_DEFAULT MII_DP83869_BMCR_DEFAULT > + > +/* CFG1 bits */ > +#define DP83869_1000BT_HALF_DUP BIT(8) > +#define DP83869_1000BT_FULL_DUP BIT(9) This seems to duplicate ADVERTISE_1000FULL et al > +#define DP83869_MAN_CFG_MASTER_SLAVE BIT(10) > + Is this really different from standard bits CTL1000_AS_MASTER and CTL1000_ENABLE_MASTER ? > +#define DP83869_CFG1_DEFAULT (DP83869_1000BT_HALF_DUP | \ > + DP83869_1000BT_HALF_DUP | \ > + DP83869_MAN_CFG_MASTER_SLAVE) > + > +/* RGMIICTL bits */ > +#define DP83869_RGMII_TX_CLK_DELAY_EN BIT(1) > +#define DP83869_RGMII_RX_CLK_DELAY_EN BIT(0) > + > +/* STRAP_STS1 bits */ > +#define DP83869_STRAP_STS1_RESERVED BIT(11) > + > +/* PHY CTRL bits */ > +/* PHYCTRL bits */ > +#define DP83869_RX_FIFO_SHIFT 12 > +#define DP83869_TX_FIFO_SHIFT 14 > + > +/* PHY_CTRL lower bytes 0x48 are declared as reserved */ > +#define DP83869_PHY_CTRL_DEFAULT 0x48 > +#define DP83869_PHYCR_FIFO_DEPTH_MASK GENMASK(15, 12) > +#define DP83869_PHYCR_RESERVED_MASK BIT(11) > + > +/* RGMIIDCTL bits */ > +#define DP83869_RGMII_TX_CLK_DELAY_SHIFT 4 > + > +/* IO_MUX_CFG bits */ > +#define DP83869_IO_MUX_CFG_IO_IMPEDANCE_CTRL 0x1f > + > +#define DP83869_IO_MUX_CFG_IO_IMPEDANCE_MAX 0x0 > +#define DP83869_IO_MUX_CFG_IO_IMPEDANCE_MIN 0x1f > +#define DP83869_IO_MUX_CFG_CLK_O_SEL_MASK (0x1f << 8) > +#define DP83869_IO_MUX_CFG_CLK_O_SEL_SHIFT 8 > + > +/* CFG3 bits */ > +#define DP83869_CFG3_PORT_MIRROR_EN BIT(0) > + > +/* OP MODE */ > +#define DP83869_OP_MODE_MII BIT(5) > +#define DP83869_SGMII_RGMII_BRIDGE BIT(6) > + > +enum { > + DP83869_PORT_MIRROING_KEEP, > + DP83869_PORT_MIRROING_EN, > + DP83869_PORT_MIRROING_DIS, MIRRORING misses an R > +}; > + > +struct dp83869_private { > + int tx_fifo_depth; > + int rx_fifo_depth; > + int io_impedance; > + int port_mirroring; > + bool rxctrl_strap_quirk; > + int clk_output_sel; > + int mode; > +}; > + > +static int op_mode; > + > +module_param(op_mode, int, 0644); > +MODULE_PARM_DESC(op_mode, "The operational mode of the PHY"); > + A module parameter isn't the preferred option here. You could have more than one such PHY in different configurations. Other drivers like the Marvell one use the interface mode to check for the desired mode. Or you could read it from DT. > +static int dp83869_ack_interrupt(struct phy_device *phydev) > +{ > + int err = phy_read(phydev, MII_DP83869_ISR); > + > + if (err < 0) > + return err; > + > + return 0; > +} > + > +static int dp83869_config_intr(struct phy_device *phydev) > +{ > + int micr_status; > + > + if (phydev->interrupts == PHY_INTERRUPT_ENABLED) { > + micr_status = phy_read(phydev, MII_DP83869_MICR); > + if (micr_status < 0) > + return micr_status; > + > + micr_status |= > + (MII_DP83869_MICR_AN_ERR_INT_EN | > + MII_DP83869_MICR_SPEED_CHNG_INT_EN | > + MII_DP83869_MICR_AUTONEG_COMP_INT_EN | > + MII_DP83869_MICR_LINK_STS_CHNG_INT_EN | > + MII_DP83869_MICR_DUP_MODE_CHNG_INT_EN | > + MII_DP83869_MICR_SLEEP_MODE_CHNG_INT_EN); > + > + return phy_write(phydev, MII_DP83869_MICR, micr_status); > + } > + > + micr_status = 0x0; This isn't really needed here, you can use 0 directly. > + return phy_write(phydev, MII_DP83869_MICR, micr_status); > +} > + > +static int dp83869_config_port_mirroring(struct phy_device *phydev) > +{ > + struct dp83869_private *dp83869 = > + (struct dp83869_private *)phydev->priv; > + The cast isn't needed. > + if (dp83869->port_mirroring == DP83869_PORT_MIRROING_EN) > + phy_set_bits_mmd(phydev, DP83869_DEVADDR, DP83869_GEN_CFG3, > + DP83869_CFG3_PORT_MIRROR_EN); > + else > + phy_clear_bits_mmd(phydev, DP83869_DEVADDR, DP83869_GEN_CFG3, > + DP83869_CFG3_PORT_MIRROR_EN); > + > + return 0; > +} > + > +#ifdef CONFIG_OF_MDIO > +static int dp83869_of_init(struct phy_device *phydev) > +{ > + struct dp83869_private *dp83869 = phydev->priv; > + struct device *dev = &phydev->mdio.dev; > + struct device_node *of_node = dev->of_node; > + int ret; > + > + if (!of_node) > + return -ENODEV; > + > + dp83869->io_impedance = -EINVAL; > + > + /* Optional configuration */ > + ret = of_property_read_u32(of_node, "ti,clk-output-sel", > + &dp83869->clk_output_sel); > + if (ret || dp83869->clk_output_sel > DP83869_CLK_O_SEL_REF_CLK) > + dp83869->clk_output_sel = DP83869_CLK_O_SEL_REF_CLK; > + > + ret = of_property_read_u32(of_node, "ti,op-mode", &dp83869->mode); > + if (ret == 0) { > + if (dp83869->mode < DP83869_RGMII_COPPER_ETHERNET || > + dp83869->mode > DP83869_SGMII_COPPER_ETHERNET) > + return -EINVAL; > + } > + > + op_mode = dp83869->mode; > + > + if (of_property_read_bool(of_node, "ti,max-output-impedance")) > + dp83869->io_impedance = DP83869_IO_MUX_CFG_IO_IMPEDANCE_MAX; > + else if (of_property_read_bool(of_node, "ti,min-output-impedance")) > + dp83869->io_impedance = DP83869_IO_MUX_CFG_IO_IMPEDANCE_MIN; > + > + if (of_property_read_bool(of_node, "enet-phy-lane-swap")) > + dp83869->port_mirroring = DP83869_PORT_MIRROING_EN; > + else > + dp83869->port_mirroring = DP83869_PORT_MIRROING_DIS; > + > + if (of_property_read_u32(of_node, "rx-fifo-depth", > + &dp83869->rx_fifo_depth)) > + dp83869->rx_fifo_depth = DP83869_PHYCR_FIFO_DEPTH_4_B_NIB; > + > + if (of_property_read_u32(of_node, "tx-fifo-depth", > + &dp83869->tx_fifo_depth)) > + dp83869->tx_fifo_depth = DP83869_PHYCR_FIFO_DEPTH_4_B_NIB; > + > + return 0; > +} > +#else > +static int dp83869_of_init(struct phy_device *phydev) > +{ > + return 0; > +} > +#endif /* CONFIG_OF_MDIO */ > + > +static int dp83869_configure_rgmii(struct phy_device *phydev, > + struct dp83869_private *dp83869) > +{ > + int ret, val; > + > + if (phy_interface_is_rgmii(phydev)) { > + val = phy_read(phydev, MII_DP83869_PHYCTRL); > + if (val < 0) > + return val; > + val &= ~DP83869_PHYCR_FIFO_DEPTH_MASK; > + val |= (dp83869->tx_fifo_depth << DP83869_TX_FIFO_SHIFT); > + val |= (dp83869->rx_fifo_depth << DP83869_RX_FIFO_SHIFT); > + > + ret = phy_write(phydev, MII_DP83869_PHYCTRL, val); > + if (ret) > + return ret; > + val = phy_read(phydev, MII_DP83869_PHYCTRL); Why do you read back the register? > + } > + > + if (dp83869->io_impedance >= 0) > + phy_modify_mmd(phydev, DP83869_DEVADDR, > + DP83869_IO_MUX_CFG, > + DP83869_IO_MUX_CFG_IO_IMPEDANCE_CTRL, > + dp83869->io_impedance & > + DP83869_IO_MUX_CFG_IO_IMPEDANCE_CTRL); > + > + return ret; > +} > + > +static int dp83869_configure_mode(struct phy_device *phydev, > + struct dp83869_private *dp83869) > +{ > + int phy_ctrl_val; > + int ret; > + > + if (op_mode < DP83869_RGMII_COPPER_ETHERNET || > + op_mode > DP83869_SGMII_COPPER_ETHERNET) > + return -EINVAL; > + > + if (dp83869->mode != op_mode) > + dp83869->mode = op_mode; > + > + /* Below init sequence for each operational mode is defined in > + * section 9.4.8 of the datasheet. > + */ > + ret = phy_write_mmd(phydev, DP83869_DEVADDR, DP83869_OP_MODE, > + dp83869->mode); > + if (ret) > + return ret; > + > + ret = phy_write(phydev, MII_BMCR, MII_DP83869_BMCR_DEFAULT); > + if (ret) > + return ret; > + > + phy_ctrl_val = (dp83869->rx_fifo_depth << DP83869_RX_FIFO_SHIFT | > + dp83869->tx_fifo_depth << DP83869_TX_FIFO_SHIFT | > + DP83869_PHY_CTRL_DEFAULT); > + > + switch (dp83869->mode) { > + case DP83869_RGMII_COPPER_ETHERNET: > + ret = phy_write(phydev, MII_DP83869_PHYCTRL, > + phy_ctrl_val); > + if (ret) > + return ret; > + > + ret = phy_write(phydev, MII_DP83869_CFG1, DP83869_CFG1_DEFAULT); > + if (ret) > + return ret; > + > + ret = dp83869_configure_rgmii(phydev, dp83869); > + if (ret) > + return ret; > + break; > + case DP83869_RGMII_SGMII_BRIDGE: > + phy_modify_mmd(phydev, DP83869_DEVADDR, DP83869_OP_MODE, > + DP83869_SGMII_RGMII_BRIDGE, > + DP83869_SGMII_RGMII_BRIDGE); > + > + ret = phy_write_mmd(phydev, DP83869_DEVADDR, > + DP83869_FX_CTRL, DP83869_FX_CTRL_DEFAULT); > + if (ret) > + return ret; > + > + break; > + case DP83869_1000M_MEDIA_CONVERT: > + ret = phy_write(phydev, MII_DP83869_PHYCTRL, > + phy_ctrl_val); > + if (ret) > + return ret; > + > + ret = phy_write_mmd(phydev, DP83869_DEVADDR, > + DP83869_FX_CTRL, DP83869_FX_CTRL_DEFAULT); > + if (ret) > + return ret; > + break; > + case DP83869_100M_MEDIA_CONVERT: > + ret = phy_write(phydev, MII_DP83869_PHYCTRL, > + phy_ctrl_val); > + if (ret) > + return ret; > + break; > + case DP83869_SGMII_COPPER_ETHERNET: > + ret = phy_write(phydev, MII_DP83869_PHYCTRL, > + phy_ctrl_val); > + if (ret) > + return ret; > + > + ret = phy_write(phydev, MII_DP83869_CFG1, DP83869_CFG1_DEFAULT); > + if (ret) > + return ret; > + > + ret = phy_write_mmd(phydev, DP83869_DEVADDR, > + DP83869_FX_CTRL, DP83869_FX_CTRL_DEFAULT); > + if (ret) > + return ret; > + > + break; > + case DP83869_RGMII_1000_BASE: > + break; > + default: > + return -EINVAL; > + }; > + > + return 0; > +} > + > +static int dp83869_config_init(struct phy_device *phydev) > +{ > + struct dp83869_private *dp83869; > + int ret, val; > + > + if (!phydev->priv) { > + dp83869 = devm_kzalloc(&phydev->mdio.dev, sizeof(*dp83869), > + GFP_KERNEL); This belongs into the probe callback. > + if (!dp83869) > + return -ENOMEM; > + > + phydev->priv = dp83869; > + ret = dp83869_of_init(phydev); > + if (ret) > + return ret; > + } else { > + dp83869 = (struct dp83869_private *)phydev->priv; > + } > + > + ret = dp83869_configure_mode(phydev, dp83869); > + if (ret) > + return ret; > + > + /* Enable Interrupt output INT_OE in CFG4 register */ > + if (phy_interrupt_is_valid(phydev)) { > + val = phy_read(phydev, DP83869_CFG4); > + val |= BIT(7); Better add a constant for this bit. > + phy_write(phydev, DP83869_CFG4, val); > + } > + > + if (dp83869->port_mirroring != DP83869_PORT_MIRROING_KEEP) > + dp83869_config_port_mirroring(phydev); > + > + /* Clock output selection if muxing property is set */ > + if (dp83869->clk_output_sel != DP83869_CLK_O_SEL_REF_CLK) > + phy_modify_mmd(phydev, DP83869_DEVADDR, DP83869_IO_MUX_CFG, > + DP83869_IO_MUX_CFG_CLK_O_SEL_MASK, > + dp83869->clk_output_sel << > + DP83869_IO_MUX_CFG_CLK_O_SEL_SHIFT); > + > + return 0; > +} > + > +static int dp83869_phy_reset(struct phy_device *phydev) > +{ > + int err; > + > + err = phy_write(phydev, DP83869_CTRL, DP83869_SW_RESET); > + if (err < 0) > + return err; > + > + usleep_range(10, 20); > + > + return dp83869_config_init(phydev); If you really need to re-configure the PHY here then it's questionnable whether this is still a "soft" reset. > +} > + > +static struct phy_driver dp83869_driver[] = { > + { > + .phy_id = DP83869_PHY_ID, > + .phy_id_mask = 0xfffffff0, You can use PHY_ID_MATCH_MODEL here. > + .name = "TI DP83869", > + .features = PHY_GBIT_FIBRE_FEATURES, The use of features is deprecated. Check whether feature autoprobing works properly for this PHY, else implement the get_features callback. > + > + .config_init = dp83869_config_init, > + .soft_reset = dp83869_phy_reset, > + > + /* IRQ related */ > + .ack_interrupt = dp83869_ack_interrupt, > + .config_intr = dp83869_config_intr, > + > + .suspend = genphy_suspend, > + .resume = genphy_resume, > + }, > +}; > +module_phy_driver(dp83869_driver); > + > +static struct mdio_device_id __maybe_unused dp83869_tbl[] = { > + { DP83869_PHY_ID, 0xfffffff0 }, You can use PHY_ID_MATCH_MODEL here. > + { } > +}; > + > +MODULE_DEVICE_TABLE(mdio, dp83869_tbl); > + > +MODULE_DESCRIPTION("Texas Instruments DP83869 PHY driver"); > +MODULE_AUTHOR("Dan Murphy <dmurphy@ti.com"); > +MODULE_LICENSE("GPL v2"); > diff --git a/include/dt-bindings/net/ti-dp83869.h b/include/dt-bindings/net/ti-dp83869.h > new file mode 100644 > index 000000000000..863074be52c7 > --- /dev/null > +++ b/include/dt-bindings/net/ti-dp83869.h > @@ -0,0 +1,43 @@ > +/* SPDX-License-Identifier: GPL-2.0-only */ > +/* > + * Device Tree constants for the Texas Instruments DP83869 PHY > + * > + * Author: Dan Murphy <dmurphy@ti.com> > + * > + * Copyright: (C) 2019 Texas Instruments, Inc. > + */ > + > +#ifndef _DT_BINDINGS_TI_DP83869_H > +#define _DT_BINDINGS_TI_DP83869_H > + > +/* PHY CTRL bits */ > +#define DP83869_PHYCR_FIFO_DEPTH_3_B_NIB 0x00 > +#define DP83869_PHYCR_FIFO_DEPTH_4_B_NIB 0x01 > +#define DP83869_PHYCR_FIFO_DEPTH_6_B_NIB 0x02 > +#define DP83869_PHYCR_FIFO_DEPTH_8_B_NIB 0x03 > + > +/* IO_MUX_CFG - Clock output selection */ > +#define DP83869_CLK_O_SEL_CHN_A_RCLK 0x0 > +#define DP83869_CLK_O_SEL_CHN_B_RCLK 0x1 > +#define DP83869_CLK_O_SEL_CHN_C_RCLK 0x2 > +#define DP83869_CLK_O_SEL_CHN_D_RCLK 0x3 > +#define DP83869_CLK_O_SEL_CHN_A_RCLK_DIV5 0x4 > +#define DP83869_CLK_O_SEL_CHN_B_RCLK_DIV5 0x5 > +#define DP83869_CLK_O_SEL_CHN_C_RCLK_DIV5 0x6 > +#define DP83869_CLK_O_SEL_CHN_D_RCLK_DIV5 0x7 > +#define DP83869_CLK_O_SEL_CHN_A_TCLK 0x8 > +#define DP83869_CLK_O_SEL_CHN_B_TCLK 0x9 > +#define DP83869_CLK_O_SEL_CHN_C_TCLK 0xa > +#define DP83869_CLK_O_SEL_CHN_D_TCLK 0xb > +#define DP83869_CLK_O_SEL_REF_CLK 0xc > + > +#define DP83869_RGMII_COPPER_ETHERNET 0x00 > +#define DP83869_RGMII_1000_BASE 0x01 > +#define DP83869_RGMII_100_BASE 0x02 > +#define DP83869_RGMII_SGMII_BRIDGE 0x03 > +#define DP83869_1000M_MEDIA_CONVERT 0x04 > +#define DP83869_100M_MEDIA_CONVERT 0x05 > +#define DP83869_SGMII_COPPER_ETHERNET 0x06 > + > +#endif > + >
Heiner On 11/5/19 1:55 PM, Heiner Kallweit wrote: > On 05.11.2019 19:18, Dan Murphy wrote: >> Add support for the TI DP83869 Gigabit ethernet phy >> device. >> >> The DP83869 is a robust, low power, fully featured >> Physical Layer transceiver with integrated PMD >> sublayers to support 10BASE-T, 100BASE-TX and >> 1000BASE-T Ethernet protocols. >> >> Signed-off-by: Dan Murphy <dmurphy@ti.com> >> --- >> drivers/net/phy/Kconfig | 6 + >> drivers/net/phy/Makefile | 1 + >> drivers/net/phy/dp83869.c | 439 +++++++++++++++++++++++++++ >> include/dt-bindings/net/ti-dp83869.h | 43 +++ >> 4 files changed, 489 insertions(+) >> create mode 100644 drivers/net/phy/dp83869.c >> create mode 100644 include/dt-bindings/net/ti-dp83869.h >> >> diff --git a/drivers/net/phy/Kconfig b/drivers/net/phy/Kconfig >> index fe602648b99f..4f8db1bb72a8 100644 >> --- a/drivers/net/phy/Kconfig >> +++ b/drivers/net/phy/Kconfig >> @@ -364,6 +364,12 @@ config DP83867_PHY >> ---help--- >> Currently supports the DP83867 PHY. >> >> +config DP83869_PHY >> + tristate "Texas Instruments DP83869 Gigabit PHY" >> + ---help--- >> + Currently supports the DP83869 PHY. This PHY supports copper and >> + fiber connections. >> + >> config FIXED_PHY >> tristate "MDIO Bus/PHY emulation with fixed speed/link PHYs" >> depends on PHYLIB >> diff --git a/drivers/net/phy/Makefile b/drivers/net/phy/Makefile >> index a03437e091f3..b433ec3bf9a6 100644 >> --- a/drivers/net/phy/Makefile >> +++ b/drivers/net/phy/Makefile >> @@ -70,6 +70,7 @@ obj-$(CONFIG_DP83822_PHY) += dp83822.o >> obj-$(CONFIG_DP83TC811_PHY) += dp83tc811.o >> obj-$(CONFIG_DP83848_PHY) += dp83848.o >> obj-$(CONFIG_DP83867_PHY) += dp83867.o >> +obj-$(CONFIG_DP83869_PHY) += dp83869.o >> obj-$(CONFIG_FIXED_PHY) += fixed_phy.o >> obj-$(CONFIG_ICPLUS_PHY) += icplus.o >> obj-$(CONFIG_INTEL_XWAY_PHY) += intel-xway.o >> diff --git a/drivers/net/phy/dp83869.c b/drivers/net/phy/dp83869.c >> new file mode 100644 >> index 000000000000..eecc9b3129ca >> --- /dev/null >> +++ b/drivers/net/phy/dp83869.c >> @@ -0,0 +1,439 @@ >> +// SPDX-License-Identifier: GPL-2.0 >> +/* Driver for the Texas Instruments DP83869 PHY >> + * Copyright (C) 2019 Texas Instruments Inc. >> + */ >> + >> +#include <linux/ethtool.h> >> +#include <linux/kernel.h> >> +#include <linux/mii.h> >> +#include <linux/module.h> >> +#include <linux/of.h> >> +#include <linux/phy.h> >> +#include <linux/delay.h> >> + >> +#include <dt-bindings/net/ti-dp83869.h> >> + >> +#define DP83869_PHY_ID 0x2000a0f1 >> +#define DP83869_DEVADDR 0x1f >> + >> +#define MII_DP83869_CFG1 0x09 > Is this different from standard register MII_CTRL1000 ? No it is the same I will change >> +#define MII_DP83869_PHYCTRL 0x10 >> +#define MII_DP83869_MICR 0x12 >> +#define MII_DP83869_ISR 0x13 >> +#define DP83869_CTRL 0x1f >> +#define DP83869_CFG4 0x1e >> + >> +/* Extended Registers */ >> +#define DP83869_GEN_CFG3 0x0031 >> +#define DP83869_RGMIICTL 0x0032 >> +#define DP83869_STRAP_STS1 0x006e >> +#define DP83869_RGMIIDCTL 0x0086 >> +#define DP83869_IO_MUX_CFG 0x0170 >> +#define DP83869_OP_MODE 0x01df >> +#define DP83869_FX_CTRL 0x0c00 >> + >> +#define DP83869_SW_RESET BIT(15) >> +#define DP83869_SW_RESTART BIT(14) >> + >> +/* MICR Interrupt bits */ >> +#define MII_DP83869_MICR_AN_ERR_INT_EN BIT(15) >> +#define MII_DP83869_MICR_SPEED_CHNG_INT_EN BIT(14) >> +#define MII_DP83869_MICR_DUP_MODE_CHNG_INT_EN BIT(13) >> +#define MII_DP83869_MICR_PAGE_RXD_INT_EN BIT(12) >> +#define MII_DP83869_MICR_AUTONEG_COMP_INT_EN BIT(11) >> +#define MII_DP83869_MICR_LINK_STS_CHNG_INT_EN BIT(10) >> +#define MII_DP83869_MICR_FALSE_CARRIER_INT_EN BIT(8) >> +#define MII_DP83869_MICR_SLEEP_MODE_CHNG_INT_EN BIT(4) >> +#define MII_DP83869_MICR_WOL_INT_EN BIT(3) >> +#define MII_DP83869_MICR_XGMII_ERR_INT_EN BIT(2) >> +#define MII_DP83869_MICR_POL_CHNG_INT_EN BIT(1) >> +#define MII_DP83869_MICR_JABBER_INT_EN BIT(0) >> + >> +#define MII_DP83869_BMCR_DEFAULT (BMCR_ANENABLE | \ >> + BMCR_FULLDPLX | \ >> + BMCR_SPEED1000) >> + >> +/* This is the same bit mask as the BMCR so re-use the BMCR default */ >> +#define DP83869_FX_CTRL_DEFAULT MII_DP83869_BMCR_DEFAULT >> + >> +/* CFG1 bits */ >> +#define DP83869_1000BT_HALF_DUP BIT(8) >> +#define DP83869_1000BT_FULL_DUP BIT(9) > This seems to duplicate ADVERTISE_1000FULL et al No it is the same I will change >> +#define DP83869_MAN_CFG_MASTER_SLAVE BIT(10) >> + > Is this really different from standard bits CTL1000_AS_MASTER > and CTL1000_ENABLE_MASTER ? No it is the same I will change >> +#define DP83869_CFG1_DEFAULT (DP83869_1000BT_HALF_DUP | \ >> + DP83869_1000BT_HALF_DUP | \ >> + DP83869_MAN_CFG_MASTER_SLAVE) >> + >> +/* RGMIICTL bits */ >> +#define DP83869_RGMII_TX_CLK_DELAY_EN BIT(1) >> +#define DP83869_RGMII_RX_CLK_DELAY_EN BIT(0) >> + >> +/* STRAP_STS1 bits */ >> +#define DP83869_STRAP_STS1_RESERVED BIT(11) >> + >> +/* PHY CTRL bits */ >> +/* PHYCTRL bits */ >> +#define DP83869_RX_FIFO_SHIFT 12 >> +#define DP83869_TX_FIFO_SHIFT 14 >> + >> +/* PHY_CTRL lower bytes 0x48 are declared as reserved */ >> +#define DP83869_PHY_CTRL_DEFAULT 0x48 >> +#define DP83869_PHYCR_FIFO_DEPTH_MASK GENMASK(15, 12) >> +#define DP83869_PHYCR_RESERVED_MASK BIT(11) >> + >> +/* RGMIIDCTL bits */ >> +#define DP83869_RGMII_TX_CLK_DELAY_SHIFT 4 >> + >> +/* IO_MUX_CFG bits */ >> +#define DP83869_IO_MUX_CFG_IO_IMPEDANCE_CTRL 0x1f >> + >> +#define DP83869_IO_MUX_CFG_IO_IMPEDANCE_MAX 0x0 >> +#define DP83869_IO_MUX_CFG_IO_IMPEDANCE_MIN 0x1f >> +#define DP83869_IO_MUX_CFG_CLK_O_SEL_MASK (0x1f << 8) >> +#define DP83869_IO_MUX_CFG_CLK_O_SEL_SHIFT 8 >> + >> +/* CFG3 bits */ >> +#define DP83869_CFG3_PORT_MIRROR_EN BIT(0) >> + >> +/* OP MODE */ >> +#define DP83869_OP_MODE_MII BIT(5) >> +#define DP83869_SGMII_RGMII_BRIDGE BIT(6) >> + >> +enum { >> + DP83869_PORT_MIRROING_KEEP, >> + DP83869_PORT_MIRROING_EN, >> + DP83869_PORT_MIRROING_DIS, > MIRRORING misses an R ack > >> +}; >> + >> +struct dp83869_private { >> + int tx_fifo_depth; >> + int rx_fifo_depth; >> + int io_impedance; >> + int port_mirroring; >> + bool rxctrl_strap_quirk; >> + int clk_output_sel; >> + int mode; >> +}; >> + >> +static int op_mode; >> + >> +module_param(op_mode, int, 0644); >> +MODULE_PARM_DESC(op_mode, "The operational mode of the PHY"); >> + > A module parameter isn't the preferred option here. > You could have more than one such PHY in different configurations. > Other drivers like the Marvell one use the interface mode to > check for the desired mode. Or you could read it from DT. > We do read the initial mode from the DT but there was a request to be able to change the mode during runtime. >> +static int dp83869_ack_interrupt(struct phy_device *phydev) >> +{ >> + int err = phy_read(phydev, MII_DP83869_ISR); >> + >> + if (err < 0) >> + return err; >> + >> + return 0; >> +} >> + >> +static int dp83869_config_intr(struct phy_device *phydev) >> +{ >> + int micr_status; >> + >> + if (phydev->interrupts == PHY_INTERRUPT_ENABLED) { >> + micr_status = phy_read(phydev, MII_DP83869_MICR); >> + if (micr_status < 0) >> + return micr_status; >> + >> + micr_status |= >> + (MII_DP83869_MICR_AN_ERR_INT_EN | >> + MII_DP83869_MICR_SPEED_CHNG_INT_EN | >> + MII_DP83869_MICR_AUTONEG_COMP_INT_EN | >> + MII_DP83869_MICR_LINK_STS_CHNG_INT_EN | >> + MII_DP83869_MICR_DUP_MODE_CHNG_INT_EN | >> + MII_DP83869_MICR_SLEEP_MODE_CHNG_INT_EN); >> + >> + return phy_write(phydev, MII_DP83869_MICR, micr_status); >> + } >> + >> + micr_status = 0x0; > This isn't really needed here, you can use 0 directly. ack > >> + return phy_write(phydev, MII_DP83869_MICR, micr_status); >> +} >> + >> +static int dp83869_config_port_mirroring(struct phy_device *phydev) >> +{ >> + struct dp83869_private *dp83869 = >> + (struct dp83869_private *)phydev->priv; >> + > The cast isn't needed. ack > >> + if (dp83869->port_mirroring == DP83869_PORT_MIRROING_EN) >> + phy_set_bits_mmd(phydev, DP83869_DEVADDR, DP83869_GEN_CFG3, >> + DP83869_CFG3_PORT_MIRROR_EN); >> + else >> + phy_clear_bits_mmd(phydev, DP83869_DEVADDR, DP83869_GEN_CFG3, >> + DP83869_CFG3_PORT_MIRROR_EN); >> + >> + return 0; >> +} >> + >> +#ifdef CONFIG_OF_MDIO >> +static int dp83869_of_init(struct phy_device *phydev) >> +{ >> + struct dp83869_private *dp83869 = phydev->priv; >> + struct device *dev = &phydev->mdio.dev; >> + struct device_node *of_node = dev->of_node; >> + int ret; >> + >> + if (!of_node) >> + return -ENODEV; >> + >> + dp83869->io_impedance = -EINVAL; >> + >> + /* Optional configuration */ >> + ret = of_property_read_u32(of_node, "ti,clk-output-sel", >> + &dp83869->clk_output_sel); >> + if (ret || dp83869->clk_output_sel > DP83869_CLK_O_SEL_REF_CLK) >> + dp83869->clk_output_sel = DP83869_CLK_O_SEL_REF_CLK; >> + >> + ret = of_property_read_u32(of_node, "ti,op-mode", &dp83869->mode); >> + if (ret == 0) { >> + if (dp83869->mode < DP83869_RGMII_COPPER_ETHERNET || >> + dp83869->mode > DP83869_SGMII_COPPER_ETHERNET) >> + return -EINVAL; >> + } >> + >> + op_mode = dp83869->mode; >> + >> + if (of_property_read_bool(of_node, "ti,max-output-impedance")) >> + dp83869->io_impedance = DP83869_IO_MUX_CFG_IO_IMPEDANCE_MAX; >> + else if (of_property_read_bool(of_node, "ti,min-output-impedance")) >> + dp83869->io_impedance = DP83869_IO_MUX_CFG_IO_IMPEDANCE_MIN; >> + >> + if (of_property_read_bool(of_node, "enet-phy-lane-swap")) >> + dp83869->port_mirroring = DP83869_PORT_MIRROING_EN; >> + else >> + dp83869->port_mirroring = DP83869_PORT_MIRROING_DIS; >> + >> + if (of_property_read_u32(of_node, "rx-fifo-depth", >> + &dp83869->rx_fifo_depth)) >> + dp83869->rx_fifo_depth = DP83869_PHYCR_FIFO_DEPTH_4_B_NIB; >> + >> + if (of_property_read_u32(of_node, "tx-fifo-depth", >> + &dp83869->tx_fifo_depth)) >> + dp83869->tx_fifo_depth = DP83869_PHYCR_FIFO_DEPTH_4_B_NIB; >> + >> + return 0; >> +} >> +#else >> +static int dp83869_of_init(struct phy_device *phydev) >> +{ >> + return 0; >> +} >> +#endif /* CONFIG_OF_MDIO */ >> + >> +static int dp83869_configure_rgmii(struct phy_device *phydev, >> + struct dp83869_private *dp83869) >> +{ >> + int ret, val; >> + >> + if (phy_interface_is_rgmii(phydev)) { >> + val = phy_read(phydev, MII_DP83869_PHYCTRL); >> + if (val < 0) >> + return val; >> + val &= ~DP83869_PHYCR_FIFO_DEPTH_MASK; >> + val |= (dp83869->tx_fifo_depth << DP83869_TX_FIFO_SHIFT); >> + val |= (dp83869->rx_fifo_depth << DP83869_RX_FIFO_SHIFT); >> + >> + ret = phy_write(phydev, MII_DP83869_PHYCTRL, val); >> + if (ret) >> + return ret; >> + val = phy_read(phydev, MII_DP83869_PHYCTRL); > Why do you read back the register? Debug artifact will remove > >> + } >> + >> + if (dp83869->io_impedance >= 0) >> + phy_modify_mmd(phydev, DP83869_DEVADDR, >> + DP83869_IO_MUX_CFG, >> + DP83869_IO_MUX_CFG_IO_IMPEDANCE_CTRL, >> + dp83869->io_impedance & >> + DP83869_IO_MUX_CFG_IO_IMPEDANCE_CTRL); >> + >> + return ret; >> +} >> + >> +static int dp83869_configure_mode(struct phy_device *phydev, >> + struct dp83869_private *dp83869) >> +{ >> + int phy_ctrl_val; >> + int ret; >> + >> + if (op_mode < DP83869_RGMII_COPPER_ETHERNET || >> + op_mode > DP83869_SGMII_COPPER_ETHERNET) >> + return -EINVAL; >> + >> + if (dp83869->mode != op_mode) >> + dp83869->mode = op_mode; >> + >> + /* Below init sequence for each operational mode is defined in >> + * section 9.4.8 of the datasheet. >> + */ >> + ret = phy_write_mmd(phydev, DP83869_DEVADDR, DP83869_OP_MODE, >> + dp83869->mode); >> + if (ret) >> + return ret; >> + >> + ret = phy_write(phydev, MII_BMCR, MII_DP83869_BMCR_DEFAULT); >> + if (ret) >> + return ret; >> + >> + phy_ctrl_val = (dp83869->rx_fifo_depth << DP83869_RX_FIFO_SHIFT | >> + dp83869->tx_fifo_depth << DP83869_TX_FIFO_SHIFT | >> + DP83869_PHY_CTRL_DEFAULT); >> + >> + switch (dp83869->mode) { >> + case DP83869_RGMII_COPPER_ETHERNET: >> + ret = phy_write(phydev, MII_DP83869_PHYCTRL, >> + phy_ctrl_val); >> + if (ret) >> + return ret; >> + >> + ret = phy_write(phydev, MII_DP83869_CFG1, DP83869_CFG1_DEFAULT); >> + if (ret) >> + return ret; >> + >> + ret = dp83869_configure_rgmii(phydev, dp83869); >> + if (ret) >> + return ret; >> + break; >> + case DP83869_RGMII_SGMII_BRIDGE: >> + phy_modify_mmd(phydev, DP83869_DEVADDR, DP83869_OP_MODE, >> + DP83869_SGMII_RGMII_BRIDGE, >> + DP83869_SGMII_RGMII_BRIDGE); >> + >> + ret = phy_write_mmd(phydev, DP83869_DEVADDR, >> + DP83869_FX_CTRL, DP83869_FX_CTRL_DEFAULT); >> + if (ret) >> + return ret; >> + >> + break; >> + case DP83869_1000M_MEDIA_CONVERT: >> + ret = phy_write(phydev, MII_DP83869_PHYCTRL, >> + phy_ctrl_val); >> + if (ret) >> + return ret; >> + >> + ret = phy_write_mmd(phydev, DP83869_DEVADDR, >> + DP83869_FX_CTRL, DP83869_FX_CTRL_DEFAULT); >> + if (ret) >> + return ret; >> + break; >> + case DP83869_100M_MEDIA_CONVERT: >> + ret = phy_write(phydev, MII_DP83869_PHYCTRL, >> + phy_ctrl_val); >> + if (ret) >> + return ret; >> + break; >> + case DP83869_SGMII_COPPER_ETHERNET: >> + ret = phy_write(phydev, MII_DP83869_PHYCTRL, >> + phy_ctrl_val); >> + if (ret) >> + return ret; >> + >> + ret = phy_write(phydev, MII_DP83869_CFG1, DP83869_CFG1_DEFAULT); >> + if (ret) >> + return ret; >> + >> + ret = phy_write_mmd(phydev, DP83869_DEVADDR, >> + DP83869_FX_CTRL, DP83869_FX_CTRL_DEFAULT); >> + if (ret) >> + return ret; >> + >> + break; >> + case DP83869_RGMII_1000_BASE: >> + break; >> + default: >> + return -EINVAL; >> + }; >> + >> + return 0; >> +} >> + >> +static int dp83869_config_init(struct phy_device *phydev) >> +{ >> + struct dp83869_private *dp83869; >> + int ret, val; >> + >> + if (!phydev->priv) { >> + dp83869 = devm_kzalloc(&phydev->mdio.dev, sizeof(*dp83869), >> + GFP_KERNEL); > This belongs into the probe callback. probe callback? Why do I need a probe function? > >> + if (!dp83869) >> + return -ENOMEM; >> + >> + phydev->priv = dp83869; >> + ret = dp83869_of_init(phydev); >> + if (ret) >> + return ret; >> + } else { >> + dp83869 = (struct dp83869_private *)phydev->priv; >> + } >> + >> + ret = dp83869_configure_mode(phydev, dp83869); >> + if (ret) >> + return ret; >> + >> + /* Enable Interrupt output INT_OE in CFG4 register */ >> + if (phy_interrupt_is_valid(phydev)) { >> + val = phy_read(phydev, DP83869_CFG4); >> + val |= BIT(7); > Better add a constant for this bit. ack > >> + phy_write(phydev, DP83869_CFG4, val); >> + } >> + >> + if (dp83869->port_mirroring != DP83869_PORT_MIRROING_KEEP) >> + dp83869_config_port_mirroring(phydev); >> + >> + /* Clock output selection if muxing property is set */ >> + if (dp83869->clk_output_sel != DP83869_CLK_O_SEL_REF_CLK) >> + phy_modify_mmd(phydev, DP83869_DEVADDR, DP83869_IO_MUX_CFG, >> + DP83869_IO_MUX_CFG_CLK_O_SEL_MASK, >> + dp83869->clk_output_sel << >> + DP83869_IO_MUX_CFG_CLK_O_SEL_SHIFT); >> + >> + return 0; >> +} >> + >> +static int dp83869_phy_reset(struct phy_device *phydev) >> +{ >> + int err; >> + >> + err = phy_write(phydev, DP83869_CTRL, DP83869_SW_RESET); >> + if (err < 0) >> + return err; >> + >> + usleep_range(10, 20); >> + >> + return dp83869_config_init(phydev); > If you really need to re-configure the PHY here then it's > questionnable whether this is still a "soft" reset. I will need to look at this again to see how the registers are affected during a SW reset. I always wrote the config. > >> +} >> + >> +static struct phy_driver dp83869_driver[] = { >> + { >> + .phy_id = DP83869_PHY_ID, >> + .phy_id_mask = 0xfffffff0, > You can use PHY_ID_MATCH_MODEL here. Ack. > >> + .name = "TI DP83869", >> + .features = PHY_GBIT_FIBRE_FEATURES, > The use of features is deprecated. Check whether feature > autoprobing works properly for this PHY, else implement the > get_features callback. Ack >> + >> + .config_init = dp83869_config_init, >> + .soft_reset = dp83869_phy_reset, >> + >> + /* IRQ related */ >> + .ack_interrupt = dp83869_ack_interrupt, >> + .config_intr = dp83869_config_intr, >> + >> + .suspend = genphy_suspend, >> + .resume = genphy_resume, >> + }, >> +}; >> +module_phy_driver(dp83869_driver); >> + >> +static struct mdio_device_id __maybe_unused dp83869_tbl[] = { >> + { DP83869_PHY_ID, 0xfffffff0 }, > You can use PHY_ID_MATCH_MODEL here. Ack Dan
On 11/5/19 12:02 PM, Dan Murphy wrote: int ret, val; >>> + >>> + if (!phydev->priv) { >>> + dp83869 = devm_kzalloc(&phydev->mdio.dev, sizeof(*dp83869), >>> + GFP_KERNEL); >> This belongs into the probe callback. > probe callback? Why do I need a probe function? To allocate your driver private memory and do that just once, config_init() can be called multiple times through the lifetime of your PHY device driver, including but not limited to: phy_connect(), phy_init_hw(), phy_resume() etc. etc. -- Florian
Florian On 11/5/19 2:05 PM, Florian Fainelli wrote: > On 11/5/19 12:02 PM, Dan Murphy wrote: > int ret, val; >>>> + >>>> + if (!phydev->priv) { >>>> + dp83869 = devm_kzalloc(&phydev->mdio.dev, sizeof(*dp83869), >>>> + GFP_KERNEL); >>> This belongs into the probe callback. >> probe callback? Why do I need a probe function? > To allocate your driver private memory and do that just once, > config_init() can be called multiple times through the lifetime of your > PHY device driver, including but not limited to: phy_connect(), > phy_init_hw(), phy_resume() etc. etc. Ah OK. Makes sense will fix it. Dan
On 05.11.2019 21:02, Dan Murphy wrote: > Heiner > > On 11/5/19 1:55 PM, Heiner Kallweit wrote: >> On 05.11.2019 19:18, Dan Murphy wrote: >>> Add support for the TI DP83869 Gigabit ethernet phy >>> device. >>> >>> The DP83869 is a robust, low power, fully featured >>> Physical Layer transceiver with integrated PMD >>> sublayers to support 10BASE-T, 100BASE-TX and >>> 1000BASE-T Ethernet protocols. >>> >>> Signed-off-by: Dan Murphy <dmurphy@ti.com> >>> --- >>> drivers/net/phy/Kconfig | 6 + >>> drivers/net/phy/Makefile | 1 + >>> drivers/net/phy/dp83869.c | 439 +++++++++++++++++++++++++++ >>> include/dt-bindings/net/ti-dp83869.h | 43 +++ >>> 4 files changed, 489 insertions(+) >>> create mode 100644 drivers/net/phy/dp83869.c >>> create mode 100644 include/dt-bindings/net/ti-dp83869.h [...] >>> +static int op_mode; >>> + >>> +module_param(op_mode, int, 0644); >>> +MODULE_PARM_DESC(op_mode, "The operational mode of the PHY"); >>> + >> A module parameter isn't the preferred option here. >> You could have more than one such PHY in different configurations. >> Other drivers like the Marvell one use the interface mode to >> check for the desired mode. Or you could read it from DT. >> > We do read the initial mode from the DT but there was a request to be able to change the mode during runtime. Maybe we need to understand the use case better to be able to advise. Will this be needed in production? Or was it requested as debug feature? There's the option to set PHY registers from userspace, e.g. with phytool. This could be used for reconfiguring the PHY. Heiner [...]
Heiner On 11/5/19 2:20 PM, Heiner Kallweit wrote: > On 05.11.2019 21:02, Dan Murphy wrote: >> Heiner >> >> On 11/5/19 1:55 PM, Heiner Kallweit wrote: >>> On 05.11.2019 19:18, Dan Murphy wrote: >>>> Add support for the TI DP83869 Gigabit ethernet phy >>>> device. >>>> >>>> The DP83869 is a robust, low power, fully featured >>>> Physical Layer transceiver with integrated PMD >>>> sublayers to support 10BASE-T, 100BASE-TX and >>>> 1000BASE-T Ethernet protocols. >>>> >>>> Signed-off-by: Dan Murphy <dmurphy@ti.com> >>>> --- >>>> drivers/net/phy/Kconfig | 6 + >>>> drivers/net/phy/Makefile | 1 + >>>> drivers/net/phy/dp83869.c | 439 +++++++++++++++++++++++++++ >>>> include/dt-bindings/net/ti-dp83869.h | 43 +++ >>>> 4 files changed, 489 insertions(+) >>>> create mode 100644 drivers/net/phy/dp83869.c >>>> create mode 100644 include/dt-bindings/net/ti-dp83869.h > [...] > >>>> +static int op_mode; >>>> + >>>> +module_param(op_mode, int, 0644); >>>> +MODULE_PARM_DESC(op_mode, "The operational mode of the PHY"); >>>> + >>> A module parameter isn't the preferred option here. >>> You could have more than one such PHY in different configurations. >>> Other drivers like the Marvell one use the interface mode to >>> check for the desired mode. Or you could read it from DT. >>> >> We do read the initial mode from the DT but there was a request to be able to change the mode during runtime. > Maybe we need to understand the use case better to be able to advise. > Will this be needed in production? Or was it requested as debug feature? > There's the option to set PHY registers from userspace, e.g. with phytool. > This could be used for reconfiguring the PHY. This was a customer request that they be able to modify the op_mode from user space. This was all I was given for a requirement. The customers use case was proprietary. Dan > Heiner > > > > [...]
On 05.11.2019 21:26, Dan Murphy wrote: > Heiner > > On 11/5/19 2:20 PM, Heiner Kallweit wrote: >> On 05.11.2019 21:02, Dan Murphy wrote: >>> Heiner >>> >>> On 11/5/19 1:55 PM, Heiner Kallweit wrote: >>>> On 05.11.2019 19:18, Dan Murphy wrote: >>>>> Add support for the TI DP83869 Gigabit ethernet phy >>>>> device. >>>>> >>>>> The DP83869 is a robust, low power, fully featured >>>>> Physical Layer transceiver with integrated PMD >>>>> sublayers to support 10BASE-T, 100BASE-TX and >>>>> 1000BASE-T Ethernet protocols. >>>>> >>>>> Signed-off-by: Dan Murphy <dmurphy@ti.com> >>>>> --- >>>>> drivers/net/phy/Kconfig | 6 + >>>>> drivers/net/phy/Makefile | 1 + >>>>> drivers/net/phy/dp83869.c | 439 +++++++++++++++++++++++++++ >>>>> include/dt-bindings/net/ti-dp83869.h | 43 +++ >>>>> 4 files changed, 489 insertions(+) >>>>> create mode 100644 drivers/net/phy/dp83869.c >>>>> create mode 100644 include/dt-bindings/net/ti-dp83869.h >> [...] >> >>>>> +static int op_mode; >>>>> + >>>>> +module_param(op_mode, int, 0644); >>>>> +MODULE_PARM_DESC(op_mode, "The operational mode of the PHY"); >>>>> + >>>> A module parameter isn't the preferred option here. >>>> You could have more than one such PHY in different configurations. >>>> Other drivers like the Marvell one use the interface mode to >>>> check for the desired mode. Or you could read it from DT. >>>> >>> We do read the initial mode from the DT but there was a request to be able to change the mode during runtime. >> Maybe we need to understand the use case better to be able to advise. >> Will this be needed in production? Or was it requested as debug feature? >> There's the option to set PHY registers from userspace, e.g. with phytool. >> This could be used for reconfiguring the PHY. > > This was a customer request that they be able to modify the op_mode from user space. > > This was all I was given for a requirement. The customers use case was proprietary. > If the module parameter serves just a (unknown) proprietary use case, then it should not be in mainline. You could provide a patch with this parameter separately to the client. > Dan > > >> Heiner >> >> >> >> [...] >
diff --git a/drivers/net/phy/Kconfig b/drivers/net/phy/Kconfig index fe602648b99f..4f8db1bb72a8 100644 --- a/drivers/net/phy/Kconfig +++ b/drivers/net/phy/Kconfig @@ -364,6 +364,12 @@ config DP83867_PHY ---help--- Currently supports the DP83867 PHY. +config DP83869_PHY + tristate "Texas Instruments DP83869 Gigabit PHY" + ---help--- + Currently supports the DP83869 PHY. This PHY supports copper and + fiber connections. + config FIXED_PHY tristate "MDIO Bus/PHY emulation with fixed speed/link PHYs" depends on PHYLIB diff --git a/drivers/net/phy/Makefile b/drivers/net/phy/Makefile index a03437e091f3..b433ec3bf9a6 100644 --- a/drivers/net/phy/Makefile +++ b/drivers/net/phy/Makefile @@ -70,6 +70,7 @@ obj-$(CONFIG_DP83822_PHY) += dp83822.o obj-$(CONFIG_DP83TC811_PHY) += dp83tc811.o obj-$(CONFIG_DP83848_PHY) += dp83848.o obj-$(CONFIG_DP83867_PHY) += dp83867.o +obj-$(CONFIG_DP83869_PHY) += dp83869.o obj-$(CONFIG_FIXED_PHY) += fixed_phy.o obj-$(CONFIG_ICPLUS_PHY) += icplus.o obj-$(CONFIG_INTEL_XWAY_PHY) += intel-xway.o diff --git a/drivers/net/phy/dp83869.c b/drivers/net/phy/dp83869.c new file mode 100644 index 000000000000..eecc9b3129ca --- /dev/null +++ b/drivers/net/phy/dp83869.c @@ -0,0 +1,439 @@ +// SPDX-License-Identifier: GPL-2.0 +/* Driver for the Texas Instruments DP83869 PHY + * Copyright (C) 2019 Texas Instruments Inc. + */ + +#include <linux/ethtool.h> +#include <linux/kernel.h> +#include <linux/mii.h> +#include <linux/module.h> +#include <linux/of.h> +#include <linux/phy.h> +#include <linux/delay.h> + +#include <dt-bindings/net/ti-dp83869.h> + +#define DP83869_PHY_ID 0x2000a0f1 +#define DP83869_DEVADDR 0x1f + +#define MII_DP83869_CFG1 0x09 +#define MII_DP83869_PHYCTRL 0x10 +#define MII_DP83869_MICR 0x12 +#define MII_DP83869_ISR 0x13 +#define DP83869_CTRL 0x1f +#define DP83869_CFG4 0x1e + +/* Extended Registers */ +#define DP83869_GEN_CFG3 0x0031 +#define DP83869_RGMIICTL 0x0032 +#define DP83869_STRAP_STS1 0x006e +#define DP83869_RGMIIDCTL 0x0086 +#define DP83869_IO_MUX_CFG 0x0170 +#define DP83869_OP_MODE 0x01df +#define DP83869_FX_CTRL 0x0c00 + +#define DP83869_SW_RESET BIT(15) +#define DP83869_SW_RESTART BIT(14) + +/* MICR Interrupt bits */ +#define MII_DP83869_MICR_AN_ERR_INT_EN BIT(15) +#define MII_DP83869_MICR_SPEED_CHNG_INT_EN BIT(14) +#define MII_DP83869_MICR_DUP_MODE_CHNG_INT_EN BIT(13) +#define MII_DP83869_MICR_PAGE_RXD_INT_EN BIT(12) +#define MII_DP83869_MICR_AUTONEG_COMP_INT_EN BIT(11) +#define MII_DP83869_MICR_LINK_STS_CHNG_INT_EN BIT(10) +#define MII_DP83869_MICR_FALSE_CARRIER_INT_EN BIT(8) +#define MII_DP83869_MICR_SLEEP_MODE_CHNG_INT_EN BIT(4) +#define MII_DP83869_MICR_WOL_INT_EN BIT(3) +#define MII_DP83869_MICR_XGMII_ERR_INT_EN BIT(2) +#define MII_DP83869_MICR_POL_CHNG_INT_EN BIT(1) +#define MII_DP83869_MICR_JABBER_INT_EN BIT(0) + +#define MII_DP83869_BMCR_DEFAULT (BMCR_ANENABLE | \ + BMCR_FULLDPLX | \ + BMCR_SPEED1000) + +/* This is the same bit mask as the BMCR so re-use the BMCR default */ +#define DP83869_FX_CTRL_DEFAULT MII_DP83869_BMCR_DEFAULT + +/* CFG1 bits */ +#define DP83869_1000BT_HALF_DUP BIT(8) +#define DP83869_1000BT_FULL_DUP BIT(9) +#define DP83869_MAN_CFG_MASTER_SLAVE BIT(10) + +#define DP83869_CFG1_DEFAULT (DP83869_1000BT_HALF_DUP | \ + DP83869_1000BT_HALF_DUP | \ + DP83869_MAN_CFG_MASTER_SLAVE) + +/* RGMIICTL bits */ +#define DP83869_RGMII_TX_CLK_DELAY_EN BIT(1) +#define DP83869_RGMII_RX_CLK_DELAY_EN BIT(0) + +/* STRAP_STS1 bits */ +#define DP83869_STRAP_STS1_RESERVED BIT(11) + +/* PHY CTRL bits */ +/* PHYCTRL bits */ +#define DP83869_RX_FIFO_SHIFT 12 +#define DP83869_TX_FIFO_SHIFT 14 + +/* PHY_CTRL lower bytes 0x48 are declared as reserved */ +#define DP83869_PHY_CTRL_DEFAULT 0x48 +#define DP83869_PHYCR_FIFO_DEPTH_MASK GENMASK(15, 12) +#define DP83869_PHYCR_RESERVED_MASK BIT(11) + +/* RGMIIDCTL bits */ +#define DP83869_RGMII_TX_CLK_DELAY_SHIFT 4 + +/* IO_MUX_CFG bits */ +#define DP83869_IO_MUX_CFG_IO_IMPEDANCE_CTRL 0x1f + +#define DP83869_IO_MUX_CFG_IO_IMPEDANCE_MAX 0x0 +#define DP83869_IO_MUX_CFG_IO_IMPEDANCE_MIN 0x1f +#define DP83869_IO_MUX_CFG_CLK_O_SEL_MASK (0x1f << 8) +#define DP83869_IO_MUX_CFG_CLK_O_SEL_SHIFT 8 + +/* CFG3 bits */ +#define DP83869_CFG3_PORT_MIRROR_EN BIT(0) + +/* OP MODE */ +#define DP83869_OP_MODE_MII BIT(5) +#define DP83869_SGMII_RGMII_BRIDGE BIT(6) + +enum { + DP83869_PORT_MIRROING_KEEP, + DP83869_PORT_MIRROING_EN, + DP83869_PORT_MIRROING_DIS, +}; + +struct dp83869_private { + int tx_fifo_depth; + int rx_fifo_depth; + int io_impedance; + int port_mirroring; + bool rxctrl_strap_quirk; + int clk_output_sel; + int mode; +}; + +static int op_mode; + +module_param(op_mode, int, 0644); +MODULE_PARM_DESC(op_mode, "The operational mode of the PHY"); + +static int dp83869_ack_interrupt(struct phy_device *phydev) +{ + int err = phy_read(phydev, MII_DP83869_ISR); + + if (err < 0) + return err; + + return 0; +} + +static int dp83869_config_intr(struct phy_device *phydev) +{ + int micr_status; + + if (phydev->interrupts == PHY_INTERRUPT_ENABLED) { + micr_status = phy_read(phydev, MII_DP83869_MICR); + if (micr_status < 0) + return micr_status; + + micr_status |= + (MII_DP83869_MICR_AN_ERR_INT_EN | + MII_DP83869_MICR_SPEED_CHNG_INT_EN | + MII_DP83869_MICR_AUTONEG_COMP_INT_EN | + MII_DP83869_MICR_LINK_STS_CHNG_INT_EN | + MII_DP83869_MICR_DUP_MODE_CHNG_INT_EN | + MII_DP83869_MICR_SLEEP_MODE_CHNG_INT_EN); + + return phy_write(phydev, MII_DP83869_MICR, micr_status); + } + + micr_status = 0x0; + return phy_write(phydev, MII_DP83869_MICR, micr_status); +} + +static int dp83869_config_port_mirroring(struct phy_device *phydev) +{ + struct dp83869_private *dp83869 = + (struct dp83869_private *)phydev->priv; + + if (dp83869->port_mirroring == DP83869_PORT_MIRROING_EN) + phy_set_bits_mmd(phydev, DP83869_DEVADDR, DP83869_GEN_CFG3, + DP83869_CFG3_PORT_MIRROR_EN); + else + phy_clear_bits_mmd(phydev, DP83869_DEVADDR, DP83869_GEN_CFG3, + DP83869_CFG3_PORT_MIRROR_EN); + + return 0; +} + +#ifdef CONFIG_OF_MDIO +static int dp83869_of_init(struct phy_device *phydev) +{ + struct dp83869_private *dp83869 = phydev->priv; + struct device *dev = &phydev->mdio.dev; + struct device_node *of_node = dev->of_node; + int ret; + + if (!of_node) + return -ENODEV; + + dp83869->io_impedance = -EINVAL; + + /* Optional configuration */ + ret = of_property_read_u32(of_node, "ti,clk-output-sel", + &dp83869->clk_output_sel); + if (ret || dp83869->clk_output_sel > DP83869_CLK_O_SEL_REF_CLK) + dp83869->clk_output_sel = DP83869_CLK_O_SEL_REF_CLK; + + ret = of_property_read_u32(of_node, "ti,op-mode", &dp83869->mode); + if (ret == 0) { + if (dp83869->mode < DP83869_RGMII_COPPER_ETHERNET || + dp83869->mode > DP83869_SGMII_COPPER_ETHERNET) + return -EINVAL; + } + + op_mode = dp83869->mode; + + if (of_property_read_bool(of_node, "ti,max-output-impedance")) + dp83869->io_impedance = DP83869_IO_MUX_CFG_IO_IMPEDANCE_MAX; + else if (of_property_read_bool(of_node, "ti,min-output-impedance")) + dp83869->io_impedance = DP83869_IO_MUX_CFG_IO_IMPEDANCE_MIN; + + if (of_property_read_bool(of_node, "enet-phy-lane-swap")) + dp83869->port_mirroring = DP83869_PORT_MIRROING_EN; + else + dp83869->port_mirroring = DP83869_PORT_MIRROING_DIS; + + if (of_property_read_u32(of_node, "rx-fifo-depth", + &dp83869->rx_fifo_depth)) + dp83869->rx_fifo_depth = DP83869_PHYCR_FIFO_DEPTH_4_B_NIB; + + if (of_property_read_u32(of_node, "tx-fifo-depth", + &dp83869->tx_fifo_depth)) + dp83869->tx_fifo_depth = DP83869_PHYCR_FIFO_DEPTH_4_B_NIB; + + return 0; +} +#else +static int dp83869_of_init(struct phy_device *phydev) +{ + return 0; +} +#endif /* CONFIG_OF_MDIO */ + +static int dp83869_configure_rgmii(struct phy_device *phydev, + struct dp83869_private *dp83869) +{ + int ret, val; + + if (phy_interface_is_rgmii(phydev)) { + val = phy_read(phydev, MII_DP83869_PHYCTRL); + if (val < 0) + return val; + val &= ~DP83869_PHYCR_FIFO_DEPTH_MASK; + val |= (dp83869->tx_fifo_depth << DP83869_TX_FIFO_SHIFT); + val |= (dp83869->rx_fifo_depth << DP83869_RX_FIFO_SHIFT); + + ret = phy_write(phydev, MII_DP83869_PHYCTRL, val); + if (ret) + return ret; + val = phy_read(phydev, MII_DP83869_PHYCTRL); + } + + if (dp83869->io_impedance >= 0) + phy_modify_mmd(phydev, DP83869_DEVADDR, + DP83869_IO_MUX_CFG, + DP83869_IO_MUX_CFG_IO_IMPEDANCE_CTRL, + dp83869->io_impedance & + DP83869_IO_MUX_CFG_IO_IMPEDANCE_CTRL); + + return ret; +} + +static int dp83869_configure_mode(struct phy_device *phydev, + struct dp83869_private *dp83869) +{ + int phy_ctrl_val; + int ret; + + if (op_mode < DP83869_RGMII_COPPER_ETHERNET || + op_mode > DP83869_SGMII_COPPER_ETHERNET) + return -EINVAL; + + if (dp83869->mode != op_mode) + dp83869->mode = op_mode; + + /* Below init sequence for each operational mode is defined in + * section 9.4.8 of the datasheet. + */ + ret = phy_write_mmd(phydev, DP83869_DEVADDR, DP83869_OP_MODE, + dp83869->mode); + if (ret) + return ret; + + ret = phy_write(phydev, MII_BMCR, MII_DP83869_BMCR_DEFAULT); + if (ret) + return ret; + + phy_ctrl_val = (dp83869->rx_fifo_depth << DP83869_RX_FIFO_SHIFT | + dp83869->tx_fifo_depth << DP83869_TX_FIFO_SHIFT | + DP83869_PHY_CTRL_DEFAULT); + + switch (dp83869->mode) { + case DP83869_RGMII_COPPER_ETHERNET: + ret = phy_write(phydev, MII_DP83869_PHYCTRL, + phy_ctrl_val); + if (ret) + return ret; + + ret = phy_write(phydev, MII_DP83869_CFG1, DP83869_CFG1_DEFAULT); + if (ret) + return ret; + + ret = dp83869_configure_rgmii(phydev, dp83869); + if (ret) + return ret; + break; + case DP83869_RGMII_SGMII_BRIDGE: + phy_modify_mmd(phydev, DP83869_DEVADDR, DP83869_OP_MODE, + DP83869_SGMII_RGMII_BRIDGE, + DP83869_SGMII_RGMII_BRIDGE); + + ret = phy_write_mmd(phydev, DP83869_DEVADDR, + DP83869_FX_CTRL, DP83869_FX_CTRL_DEFAULT); + if (ret) + return ret; + + break; + case DP83869_1000M_MEDIA_CONVERT: + ret = phy_write(phydev, MII_DP83869_PHYCTRL, + phy_ctrl_val); + if (ret) + return ret; + + ret = phy_write_mmd(phydev, DP83869_DEVADDR, + DP83869_FX_CTRL, DP83869_FX_CTRL_DEFAULT); + if (ret) + return ret; + break; + case DP83869_100M_MEDIA_CONVERT: + ret = phy_write(phydev, MII_DP83869_PHYCTRL, + phy_ctrl_val); + if (ret) + return ret; + break; + case DP83869_SGMII_COPPER_ETHERNET: + ret = phy_write(phydev, MII_DP83869_PHYCTRL, + phy_ctrl_val); + if (ret) + return ret; + + ret = phy_write(phydev, MII_DP83869_CFG1, DP83869_CFG1_DEFAULT); + if (ret) + return ret; + + ret = phy_write_mmd(phydev, DP83869_DEVADDR, + DP83869_FX_CTRL, DP83869_FX_CTRL_DEFAULT); + if (ret) + return ret; + + break; + case DP83869_RGMII_1000_BASE: + break; + default: + return -EINVAL; + }; + + return 0; +} + +static int dp83869_config_init(struct phy_device *phydev) +{ + struct dp83869_private *dp83869; + int ret, val; + + if (!phydev->priv) { + dp83869 = devm_kzalloc(&phydev->mdio.dev, sizeof(*dp83869), + GFP_KERNEL); + if (!dp83869) + return -ENOMEM; + + phydev->priv = dp83869; + ret = dp83869_of_init(phydev); + if (ret) + return ret; + } else { + dp83869 = (struct dp83869_private *)phydev->priv; + } + + ret = dp83869_configure_mode(phydev, dp83869); + if (ret) + return ret; + + /* Enable Interrupt output INT_OE in CFG4 register */ + if (phy_interrupt_is_valid(phydev)) { + val = phy_read(phydev, DP83869_CFG4); + val |= BIT(7); + phy_write(phydev, DP83869_CFG4, val); + } + + if (dp83869->port_mirroring != DP83869_PORT_MIRROING_KEEP) + dp83869_config_port_mirroring(phydev); + + /* Clock output selection if muxing property is set */ + if (dp83869->clk_output_sel != DP83869_CLK_O_SEL_REF_CLK) + phy_modify_mmd(phydev, DP83869_DEVADDR, DP83869_IO_MUX_CFG, + DP83869_IO_MUX_CFG_CLK_O_SEL_MASK, + dp83869->clk_output_sel << + DP83869_IO_MUX_CFG_CLK_O_SEL_SHIFT); + + return 0; +} + +static int dp83869_phy_reset(struct phy_device *phydev) +{ + int err; + + err = phy_write(phydev, DP83869_CTRL, DP83869_SW_RESET); + if (err < 0) + return err; + + usleep_range(10, 20); + + return dp83869_config_init(phydev); +} + +static struct phy_driver dp83869_driver[] = { + { + .phy_id = DP83869_PHY_ID, + .phy_id_mask = 0xfffffff0, + .name = "TI DP83869", + .features = PHY_GBIT_FIBRE_FEATURES, + + .config_init = dp83869_config_init, + .soft_reset = dp83869_phy_reset, + + /* IRQ related */ + .ack_interrupt = dp83869_ack_interrupt, + .config_intr = dp83869_config_intr, + + .suspend = genphy_suspend, + .resume = genphy_resume, + }, +}; +module_phy_driver(dp83869_driver); + +static struct mdio_device_id __maybe_unused dp83869_tbl[] = { + { DP83869_PHY_ID, 0xfffffff0 }, + { } +}; + +MODULE_DEVICE_TABLE(mdio, dp83869_tbl); + +MODULE_DESCRIPTION("Texas Instruments DP83869 PHY driver"); +MODULE_AUTHOR("Dan Murphy <dmurphy@ti.com"); +MODULE_LICENSE("GPL v2"); diff --git a/include/dt-bindings/net/ti-dp83869.h b/include/dt-bindings/net/ti-dp83869.h new file mode 100644 index 000000000000..863074be52c7 --- /dev/null +++ b/include/dt-bindings/net/ti-dp83869.h @@ -0,0 +1,43 @@ +/* SPDX-License-Identifier: GPL-2.0-only */ +/* + * Device Tree constants for the Texas Instruments DP83869 PHY + * + * Author: Dan Murphy <dmurphy@ti.com> + * + * Copyright: (C) 2019 Texas Instruments, Inc. + */ + +#ifndef _DT_BINDINGS_TI_DP83869_H +#define _DT_BINDINGS_TI_DP83869_H + +/* PHY CTRL bits */ +#define DP83869_PHYCR_FIFO_DEPTH_3_B_NIB 0x00 +#define DP83869_PHYCR_FIFO_DEPTH_4_B_NIB 0x01 +#define DP83869_PHYCR_FIFO_DEPTH_6_B_NIB 0x02 +#define DP83869_PHYCR_FIFO_DEPTH_8_B_NIB 0x03 + +/* IO_MUX_CFG - Clock output selection */ +#define DP83869_CLK_O_SEL_CHN_A_RCLK 0x0 +#define DP83869_CLK_O_SEL_CHN_B_RCLK 0x1 +#define DP83869_CLK_O_SEL_CHN_C_RCLK 0x2 +#define DP83869_CLK_O_SEL_CHN_D_RCLK 0x3 +#define DP83869_CLK_O_SEL_CHN_A_RCLK_DIV5 0x4 +#define DP83869_CLK_O_SEL_CHN_B_RCLK_DIV5 0x5 +#define DP83869_CLK_O_SEL_CHN_C_RCLK_DIV5 0x6 +#define DP83869_CLK_O_SEL_CHN_D_RCLK_DIV5 0x7 +#define DP83869_CLK_O_SEL_CHN_A_TCLK 0x8 +#define DP83869_CLK_O_SEL_CHN_B_TCLK 0x9 +#define DP83869_CLK_O_SEL_CHN_C_TCLK 0xa +#define DP83869_CLK_O_SEL_CHN_D_TCLK 0xb +#define DP83869_CLK_O_SEL_REF_CLK 0xc + +#define DP83869_RGMII_COPPER_ETHERNET 0x00 +#define DP83869_RGMII_1000_BASE 0x01 +#define DP83869_RGMII_100_BASE 0x02 +#define DP83869_RGMII_SGMII_BRIDGE 0x03 +#define DP83869_1000M_MEDIA_CONVERT 0x04 +#define DP83869_100M_MEDIA_CONVERT 0x05 +#define DP83869_SGMII_COPPER_ETHERNET 0x06 + +#endif +
Add support for the TI DP83869 Gigabit ethernet phy device. The DP83869 is a robust, low power, fully featured Physical Layer transceiver with integrated PMD sublayers to support 10BASE-T, 100BASE-TX and 1000BASE-T Ethernet protocols. Signed-off-by: Dan Murphy <dmurphy@ti.com> --- drivers/net/phy/Kconfig | 6 + drivers/net/phy/Makefile | 1 + drivers/net/phy/dp83869.c | 439 +++++++++++++++++++++++++++ include/dt-bindings/net/ti-dp83869.h | 43 +++ 4 files changed, 489 insertions(+) create mode 100644 drivers/net/phy/dp83869.c create mode 100644 include/dt-bindings/net/ti-dp83869.h -- 2.22.0.214.g8dca754b1e