diff mbox series

[2/2] net: phy: dp83869: Add TI dp83869 phy

Message ID 20191105181826.25114-2-dmurphy@ti.com
State New
Headers show
Series [1/2] dt-bindings: net: dp83869: Add TI dp83869 phy | expand

Commit Message

Dan Murphy Nov. 5, 2019, 6:18 p.m. UTC
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

Comments

Heiner Kallweit Nov. 5, 2019, 7:55 p.m. UTC | #1
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

> +

>
Dan Murphy Nov. 5, 2019, 8:02 p.m. UTC | #2
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
Florian Fainelli Nov. 5, 2019, 8:05 p.m. UTC | #3
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
Dan Murphy Nov. 5, 2019, 8:08 p.m. UTC | #4
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
Heiner Kallweit Nov. 5, 2019, 8:20 p.m. UTC | #5
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



[...]
Dan Murphy Nov. 5, 2019, 8:26 p.m. UTC | #6
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

>

>

>

> [...]
Heiner Kallweit Nov. 5, 2019, 8:48 p.m. UTC | #7
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 mbox series

Patch

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
+