[<prev] [next>] [<thread-prev] [thread-next>] [day] [month] [year] [list]
Message-ID: <68b9c003-4fb3-b854-695a-fa1c6e08f518@gmail.com>
Date: Tue, 5 Nov 2019 20:55:47 +0100
From: Heiner Kallweit <hkallweit1@...il.com>
To: Dan Murphy <dmurphy@...com>, andrew@...n.ch, f.fainelli@...il.com,
davem@...emloft.net
Cc: netdev@...r.kernel.org, linux-kernel@...r.kernel.org
Subject: Re: [PATCH 2/2] net: phy: dp83869: Add TI dp83869 phy
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@...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@...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@...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
> +
>
Powered by blists - more mailing lists