[RESEND PATCH v2 5/5] net: dwc_eth_qos: Add glue driver for Intel MAC
Marek Vasut
marex at denx.de
Sun Jun 30 07:38:43 CEST 2024
On 6/24/24 10:34 AM, Philip Oberfichtner wrote:
> +++ b/drivers/net/dwc_eth_qos_intel.c
> @@ -0,0 +1,446 @@
> +// SPDX-License-Identifier: GPL-2.0
> +/*
> + * Copyright (c) 2023 DENX Software Engineering GmbH
> + * Philip Oberfichtner <pro at denx.de>
> + *
> + * Based on linux v6.1.38, especially drivers/net/ethernet/stmicro/stmmac
It is 2024 now , also you might want to update this to match Linux 6.6.y
which is the current LTS .
> + */
> +
> +#include <asm/io.h>
> +#include <dm.h>
> +#include <linux/delay.h>
> +#include <miiphy.h>
> +#include <net.h>
> +#include <pci.h>
> +
> +#include "dwc_eth_qos.h"
> +#include "dwc_eth_qos_intel.h"
> +
> +static struct pci_device_id intel_pci_ids[] = {
> + { PCI_DEVICE(PCI_VENDOR_ID_INTEL, PCI_DEVICE_ID_INTEL_EHL_RGMII1G) },
> + { PCI_DEVICE(PCI_VENDOR_ID_INTEL, PCI_DEVICE_ID_INTEL_EHL_SGMII1) },
> + { PCI_DEVICE(PCI_VENDOR_ID_INTEL, PCI_DEVICE_ID_INTEL_EHL_SGMII2G5) },
> + { PCI_DEVICE(PCI_VENDOR_ID_INTEL, PCI_DEVICE_ID_INTEL_EHL_PSE0_RGMII1G) },
> + { PCI_DEVICE(PCI_VENDOR_ID_INTEL, PCI_DEVICE_ID_INTEL_EHL_PSE0_SGMII1G) },
> + { PCI_DEVICE(PCI_VENDOR_ID_INTEL, PCI_DEVICE_ID_INTEL_EHL_PSE0_SGMII2G5) },
> + { PCI_DEVICE(PCI_VENDOR_ID_INTEL, PCI_DEVICE_ID_INTEL_EHL_PSE1_RGMII1G) },
> + { PCI_DEVICE(PCI_VENDOR_ID_INTEL, PCI_DEVICE_ID_INTEL_EHL_PSE1_SGMII1G) },
> + { PCI_DEVICE(PCI_VENDOR_ID_INTEL, PCI_DEVICE_ID_INTEL_EHL_PSE1_SGMII2G5) },
> + {}
> +};
> +
> +static int __pci_config(struct udevice *dev)
> +{
> + u32 val;
> +
> + /* Try to enable I/O accesses and bus-mastering */
> + dm_pci_read_config32(dev, PCI_COMMAND, &val);
> + val |= PCI_COMMAND_MEMORY | PCI_COMMAND_MASTER;
> + dm_pci_write_config32(dev, PCI_COMMAND, val);
> +
> + /* Make sure it worked */
> + dm_pci_read_config32(dev, PCI_COMMAND, &val);
> + if (!(val & PCI_COMMAND_MEMORY)) {
> + pr_err("%s: Can't enable I/O memory\n", __func__);
dev_err() please
> + return -ENOSPC;
> + }
> +
> + if (!(val & PCI_COMMAND_MASTER)) {
> + pr_err("%s: Can't enable bus-mastering\n", __func__);
dev_err() , fix globally.
> + return -EPERM;
> + }
> +
> + return 0;
> +}
> +
> +static void __limit_fifo_size(struct udevice *dev)
Please drop the leading __ from function names .
> +{
> + /*
> + * As described in Intel Erratum EHL22, Document Number: 636674-2.1,
> + * the PSE GbE Controllers advertise a wrong RX and TX fifo size.
> + * Software should limit this value to 64KB.
> + */
> + struct eqos_priv *eqos = dev_get_priv(dev);
> +
> + eqos->tx_fifo_sz = 0x8000;
> + eqos->rx_fifo_sz = 0x8000;
> +}
> +
> +static int __serdes_status_poll(struct udevice *dev,
> + unsigned char phyaddr, unsigned char phyreg,
> + unsigned short mask, unsigned short val)
> +{
> + struct eqos_priv *eqos = dev_get_priv(dev);
> + unsigned int retries = 10;
> + unsigned short val_rd;
> +
> + do {
> + miiphy_read(eqos->mii->name, phyaddr, phyreg, &val_rd);
> + if ((val_rd & mask) == (val & mask))
> + return 0;
> + udelay(POLL_DELAY_US);
> + } while (--retries);
Is this some implementation of phy_read_mmd_poll_timeout() ?
> + return -ETIMEDOUT;
> +}
> +
> + /* Returns -ve if MAC is unknown and 0 on success */
> +static int __mac_check_pse(const struct udevice *dev, bool *is_pse)
> +{
> + struct pci_child_plat *plat = dev_get_parent_plat(dev);
> +
> + if (!plat || plat->vendor != PCI_VENDOR_ID_INTEL)
> + return -ENXIO;
> +
> + switch (plat->device) {
> + case PCI_DEVICE_ID_INTEL_EHL_PSE0_RGMII1G:
> + case PCI_DEVICE_ID_INTEL_EHL_PSE1_RGMII1G:
> + case PCI_DEVICE_ID_INTEL_EHL_PSE0_SGMII1G:
> + case PCI_DEVICE_ID_INTEL_EHL_PSE1_SGMII1G:
> + case PCI_DEVICE_ID_INTEL_EHL_PSE0_SGMII2G5:
> + case PCI_DEVICE_ID_INTEL_EHL_PSE1_SGMII2G5:
> + *is_pse = 1;
> + return 0;
> +
> + case PCI_DEVICE_ID_INTEL_EHL_RGMII1G:
> + case PCI_DEVICE_ID_INTEL_EHL_SGMII1:
> + case PCI_DEVICE_ID_INTEL_EHL_SGMII2G5:
> + *is_pse = 0;
> + return 0;
> + };
> +
> + return -ENXIO;
> +}
> +
> +/* Check if we're in 2G5 mode */
> +static bool __serdes_link_mode_2500(struct udevice *dev)
> +{
> + const unsigned char phyad = INTEL_MGBE_ADHOC_ADDR;
> + struct eqos_priv *eqos = dev_get_priv(dev);
> + unsigned short data;
> +
> + miiphy_read(eqos->mii->name, phyad, SERDES_GCR, &data);
> + if (((data & SERDES_LINK_MODE_MASK) >> SERDES_LINK_MODE_SHIFT) ==
> + SERDES_LINK_MODE_2G5)
Use FIELD_PREP() and FIELD_GET() to simplify this bitfield extraction.
> + return true;
> +
> + return false;
> +}
> +
> +static int __serdes_powerup(struct udevice *dev)
> +{
> + /* Based on linux/drivers/net/ethernet/stmicro/stmmac/dwmac-intel.c */
> +
> + const unsigned char phyad = INTEL_MGBE_ADHOC_ADDR;
> + struct eqos_priv *eqos = dev_get_priv(dev);
> + unsigned short data;
> + int ret;
> + bool is_pse;
> +
> + /* Set the serdes rate and the PCLK rate */
> + miiphy_read(eqos->mii->name, phyad, SERDES_GCR0, &data);
> +
> + data &= ~SERDES_RATE_MASK;
> + data &= ~SERDES_PCLK_MASK;
> +
> + if (__serdes_link_mode_2500(dev))
> + data |= SERDES_RATE_PCIE_GEN2 << SERDES_RATE_PCIE_SHIFT |
> + SERDES_PCLK_37p5MHZ << SERDES_PCLK_SHIFT;
> + else
> + data |= SERDES_RATE_PCIE_GEN1 << SERDES_RATE_PCIE_SHIFT |
> + SERDES_PCLK_70MHZ << SERDES_PCLK_SHIFT;
> +
> + miiphy_write(eqos->mii->name, phyad, SERDES_GCR0, data);
> +
> + /* assert clk_req */
> + miiphy_read(eqos->mii->name, phyad, SERDES_GCR0, &data);
> + data |= SERDES_PLL_CLK;
> + miiphy_write(eqos->mii->name, phyad, SERDES_GCR0, data);
Could this use phy_modify() ?
[...]
More information about the U-Boot
mailing list