Attention is currently required from: Lean Sheng Tan, Werner Zeh.
Mario Scheithauer has uploaded this change for review. ( https://review.coreboot.org/c/coreboot/+/69383 )
Change subject: soc/intel/ehl: Add EHL MDIO operation ......................................................................
soc/intel/ehl: Add EHL MDIO operation
Previously, MDIO read/write access was only necessary for internal GbE TSN controller. For external MDIO access, it is necessary to provide a more general solution. This MDIO operation provides a read and write function which can be also used on mainboard level.
Change-Id: I5d1b9dd2f2ba8c18291fff314c13f0c3851784aa Signed-off-by: Mario Scheithauer mario.scheithauer@siemens.com --- M src/soc/intel/elkhartlake/Makefile.inc A src/soc/intel/elkhartlake/include/soc/mdio.h M src/soc/intel/elkhartlake/include/soc/tsn_gbe.h A src/soc/intel/elkhartlake/mdio.c M src/soc/intel/elkhartlake/tsn_gbe.c 5 files changed, 157 insertions(+), 83 deletions(-)
git pull ssh://review.coreboot.org:29418/coreboot refs/changes/83/69383/1
diff --git a/src/soc/intel/elkhartlake/Makefile.inc b/src/soc/intel/elkhartlake/Makefile.inc index 1ed11b4..6c8207f 100644 --- a/src/soc/intel/elkhartlake/Makefile.inc +++ b/src/soc/intel/elkhartlake/Makefile.inc @@ -32,6 +32,7 @@ ramstage-y += fsp_params.c ramstage-y += gpio.c ramstage-y += lockdown.c +ramstage-y += mdio.c ramstage-y += p2sb.c ramstage-y += pmc.c ramstage-y += reset.c diff --git a/src/soc/intel/elkhartlake/include/soc/mdio.h b/src/soc/intel/elkhartlake/include/soc/mdio.h new file mode 100644 index 0000000..c61a342 --- /dev/null +++ b/src/soc/intel/elkhartlake/include/soc/mdio.h @@ -0,0 +1,25 @@ +/* SPDX-License-Identifier: GPL-2.0-only */ + +#ifndef _SOC_ELKHARTLAKE_MDIO_H_ +#define _SOC_ELKHARTLAKE_MDIO_H_ + +#define GMII_TIMEOUT_MS 20 + +#define MAC_MDIO_ADR 0x200 /* MAC MDIO Address register */ +#define MAC_MDIO_ADR_MASK 0x03FF7F0E +#define MAC_PHYAD(pa) (pa << 21) /* Physical Layer Address */ +#define MAC_REGAD(rda) (rda << 16) /* Register/Device Address */ +#define MAC_CLK_TRAIL_4 (4 << 12) /* 4 Trailing Clocks */ +#define MAC_CSR_CLK_DIV_102 (1 << 10) /* 0100: CSR=150-250 MHz; CSR/102 */ +#define MAC_OP_CMD_WRITE (1 << 2) /* GMII Operation Command Write */ +#define MAC_OP_CMD_READ (3 << 2) /* GMII Operation Command Read */ +#define MAC_GMII_BUSY (1 << 0) /* GMII Busy bit */ + +#define MAC_MDIO_DATA 0x204 /* MAC MDIO Data register */ + +/* + * Set mdio ops of the device to mdio dev ops. + */ +void mdio_dev_enable(struct device *dev); + +#endif /* _SOC_ELKHARTLAKE_MDIO_H_ */ diff --git a/src/soc/intel/elkhartlake/include/soc/tsn_gbe.h b/src/soc/intel/elkhartlake/include/soc/tsn_gbe.h index cd9f12e..ba037ed 100644 --- a/src/soc/intel/elkhartlake/include/soc/tsn_gbe.h +++ b/src/soc/intel/elkhartlake/include/soc/tsn_gbe.h @@ -8,18 +8,6 @@ #define TSN_MAC_ADD0_HIGH 0x300 /* MAC Address0 High register */ #define TSN_MAC_ADD0_LOW 0x304 /* MAC Address0 Low register */
-#define TSN_GMII_TIMEOUT_MS 20 - -#define TSN_MAC_MDIO_ADR 0x200 /* MAC MDIO Address register */ -#define TSN_MAC_MDIO_ADR_MASK 0x03FF7F0E -#define TSN_MAC_PHYAD(pa) (pa << 21) /* Physical Layer Address */ -#define TSN_MAC_REGAD(rda) (rda << 16) /* Register/Device Address */ -#define TSN_MAC_CLK_TRAIL_4 (4 << 12) /* 4 Trailing Clocks */ -#define TSN_MAC_CSR_CLK_DIV_102 (1 << 10) /* 0100: CSR=150-250 MHz; CSR/102 */ -#define TSN_MAC_OP_CMD_WRITE (1 << 2) /* GMII Operation Command Write */ -#define TSN_MAC_OP_CMD_READ (3 << 2) /* GMII Operation Command Read */ -#define TSN_MAC_GMII_BUSY (1 << 0) /* GMII Busy bit */ - /* MDIO - Adhoc PHY Sublayer Register */ #define TSN_MAC_MDIO_ADHOC_ADR 0x15 /* Global Configuration Register */ @@ -27,14 +15,8 @@ /* PHY to MAC Interrupt Polarity bit */ #define TSN_MAC_PHY2MAC_INTR_POL (1 << 6)
-#define TSN_MAC_MDIO_DATA 0x204 /* MAC MDIO Data register */ - /* We need one function we can call to get a MAC address to use. */ /* This function can be coded somewhere else but must exist. */ enum cb_err mainboard_get_mac_address(struct device *dev, uint8_t mac[MAC_ADDR_LEN]);
-enum cb_err phy_gmii_ready(void *base); -uint16_t tsn_mdio_read(void *base, uint8_t phy_adr, uint8_t reg_adr); -void tsn_mdio_write(void *base, uint8_t phy_adr, uint8_t reg_adr, uint16_t data); - #endif /* _SOC_ELKHARTLAKE_TSN_GBE_H_ */ diff --git a/src/soc/intel/elkhartlake/mdio.c b/src/soc/intel/elkhartlake/mdio.c new file mode 100644 index 0000000..7a7adbc --- /dev/null +++ b/src/soc/intel/elkhartlake/mdio.c @@ -0,0 +1,98 @@ +/* SPDX-License-Identifier: GPL-2.0-or-later */ + +#define __SIMPLE_DEVICE__ + +#include <console/console.h> +#include <device/mdio.h> +#include <device/pci.h> +#include <device/pci_ids.h> +#include <device/pci_ops.h> +#include <soc/mdio.h> +#include <timer.h> + +static enum cb_err phy_gmii_ready(void *base) +{ + struct stopwatch sw; + + stopwatch_init_msecs_expire(&sw, GMII_TIMEOUT_MS); + do { + if (!(read32((base + MAC_MDIO_ADR)) & MAC_GMII_BUSY)) + return CB_SUCCESS; + + } while (!stopwatch_expired(&sw)); + + printk(BIOS_ERR, "%s Timeout after %lld msec\n", __func__, + stopwatch_duration_msecs(&sw)); + return CB_ERR; +} + +static uint16_t mdio_read(struct device *dev, uint8_t phy_adr, uint8_t reg_adr) +{ + uint16_t data = 0; + enum cb_err status; + /* Get the base address of the I/O registers in memory space */ + struct resource *gbe_tsn_res = find_resource(dev, PCI_BASE_ADDRESS_0); + void *io_mem_base = (void *)(uintptr_t)gbe_tsn_res->base; + + if (!io_mem_base) + return data; + + clrsetbits32(io_mem_base + MAC_MDIO_ADR, MAC_MDIO_ADR_MASK, + MAC_PHYAD(phy_adr) | MAC_REGAD(reg_adr) + | MAC_CLK_TRAIL_4 | MAC_CSR_CLK_DIV_102 + | MAC_OP_CMD_READ | MAC_GMII_BUSY); + + /* Wait for MDIO frame transfer to complete before reading MDIO DATA register */ + status = phy_gmii_ready(io_mem_base); + if (status == CB_ERR) { + printk(BIOS_ERR, "%s TSN GMII busy. PHY Adr: 0x%x, Reg 0x%x\n", + __func__, phy_adr, reg_adr); + } else { + data = read16(io_mem_base + MAC_MDIO_DATA); + printk(BIOS_DEBUG, "%s PHY Adr: 0x%x, Reg: 0x%x , Data: 0x%x\n", + __func__, phy_adr, reg_adr, data); + } + return data; +} + +static void mdio_write(struct device *dev, uint8_t phy_adr, uint8_t reg_adr, uint16_t data) +{ + enum cb_err status; + /* Get the base address of the I/O registers in memory space */ + struct resource *gbe_tsn_res = find_resource(dev, PCI_BASE_ADDRESS_0); + void *io_mem_base = (void *)(uintptr_t)gbe_tsn_res->base; + + if (!io_mem_base) + return; + + write16(io_mem_base + MAC_MDIO_DATA, data); + clrsetbits32(io_mem_base + MAC_MDIO_ADR, MAC_MDIO_ADR_MASK, + MAC_PHYAD(phy_adr) | MAC_REGAD(reg_adr) + | MAC_CLK_TRAIL_4 | MAC_CSR_CLK_DIV_102 + | MAC_OP_CMD_WRITE | MAC_GMII_BUSY); + + /* Wait for MDIO frame transfer to complete before do next */ + status = phy_gmii_ready(io_mem_base); + if (status == CB_ERR) + printk(BIOS_ERR, "%s TSN GMII busy. PHY Adr: 0x%x, Reg 0x%x\n", + __func__, phy_adr, reg_adr); + else + printk(BIOS_DEBUG, "%s PHY Adr: 0x%x, Reg: 0x%x , Data: 0x%x\n", + __func__, phy_adr, reg_adr, data); +} + +static struct mdio_operations mdio_ops = { + .read = mdio_read, + .write = mdio_write, +}; + +struct device_operations mdio_dev_ops = { + .read_resources = noop_read_resources, + .set_resources = noop_set_resources, + .ops_mdio = &mdio_ops, +}; + +void mdio_dev_enable(struct device *dev) +{ + dev->ops = &mdio_dev_ops; +} diff --git a/src/soc/intel/elkhartlake/tsn_gbe.c b/src/soc/intel/elkhartlake/tsn_gbe.c index 2a1468c..28ed0a4 100644 --- a/src/soc/intel/elkhartlake/tsn_gbe.c +++ b/src/soc/intel/elkhartlake/tsn_gbe.c @@ -1,12 +1,13 @@ /* SPDX-License-Identifier: GPL-2.0-only */
#include <console/console.h> +#include <device/mdio.h> #include <device/pci.h> #include <device/pci_ids.h> #include <intelblocks/lpss.h> +#include <soc/mdio.h> #include <soc/soc_chip.h> #include <soc/tsn_gbe.h> -#include <timer.h>
static void program_mac_address(struct device *dev, void *base) { @@ -29,74 +30,23 @@ (mac[3] << 24) | (mac[2] << 16) | (mac[1] << 8) | mac[0]); }
-enum cb_err phy_gmii_ready(void *base) -{ - struct stopwatch sw;
- stopwatch_init_msecs_expire(&sw, TSN_GMII_TIMEOUT_MS); - do { - if (!(read32((base + TSN_MAC_MDIO_ADR)) & TSN_MAC_GMII_BUSY)) - return CB_SUCCESS; - - } while (!stopwatch_expired(&sw)); - - printk(BIOS_ERR, "%s Timeout after %lld msec\n", __func__, - stopwatch_duration_msecs(&sw)); - return CB_ERR; -} - -uint16_t tsn_mdio_read(void *base, uint8_t phy_adr, uint8_t reg_adr) -{ - uint16_t data = 0; - enum cb_err status; - - clrsetbits32(base + TSN_MAC_MDIO_ADR, TSN_MAC_MDIO_ADR_MASK, - TSN_MAC_PHYAD(phy_adr) | TSN_MAC_REGAD(reg_adr) - | TSN_MAC_CLK_TRAIL_4 | TSN_MAC_CSR_CLK_DIV_102 - | TSN_MAC_OP_CMD_READ | TSN_MAC_GMII_BUSY); - - /* Wait for MDIO frame transfer to complete before reading MDIO DATA register */ - status = phy_gmii_ready(base); - if (status == CB_ERR) { - printk(BIOS_ERR, "%s TSN GMII busy. PHY Adr: 0x%x, Reg 0x%x\n", - __func__, phy_adr, reg_adr); - } else { - data = read16(base + TSN_MAC_MDIO_DATA); - printk(BIOS_DEBUG, "%s PHY Adr: 0x%x, Reg: 0x%x , Data: 0x%x\n", - __func__, phy_adr, reg_adr, data); - } - return data; -} - -void tsn_mdio_write(void *base, uint8_t phy_adr, uint8_t reg_adr, uint16_t data) -{ - enum cb_err status; - - write16(base + TSN_MAC_MDIO_DATA, data); - clrsetbits32(base + TSN_MAC_MDIO_ADR, TSN_MAC_MDIO_ADR_MASK, - TSN_MAC_PHYAD(phy_adr) | TSN_MAC_REGAD(reg_adr) - | TSN_MAC_CLK_TRAIL_4 | TSN_MAC_CSR_CLK_DIV_102 - | TSN_MAC_OP_CMD_WRITE | TSN_MAC_GMII_BUSY); - - /* Wait for MDIO frame transfer to complete before do next */ - status = phy_gmii_ready(base); - if (status == CB_ERR) - printk(BIOS_ERR, "%s TSN GMII busy. PHY Adr: 0x%x, Reg 0x%x\n", - __func__, phy_adr, reg_adr); - else - printk(BIOS_DEBUG, "%s PHY Adr: 0x%x, Reg: 0x%x , Data: 0x%x\n", - __func__, phy_adr, reg_adr, data); -} - -static void tsn_set_phy2mac_irq_polarity(void *base, enum tsn_phy_irq_polarity pol) +static void tsn_set_phy2mac_irq_polarity(struct device *dev, enum tsn_phy_irq_polarity pol) { uint16_t gcr_reg; + const struct mdio_operations *mdio_ops; + + mdio_ops = dev_get_mdio_ops(dev); + if (!mdio_ops) { + printk(BIOS_WARNING, "MDIO: specified mdio device is missing mdio ops!\n"); + return; + }
if (pol == RISING_EDGE) { /* Read TSN adhoc PHY sublayer register - global configuration register */ - gcr_reg = tsn_mdio_read(base, TSN_MAC_MDIO_ADHOC_ADR, TSN_MAC_MDIO_GCR); + gcr_reg = mdio_ops->read(dev, TSN_MAC_MDIO_ADHOC_ADR, TSN_MAC_MDIO_GCR); gcr_reg |= TSN_MAC_PHY2MAC_INTR_POL; - tsn_mdio_write(base, TSN_MAC_MDIO_ADHOC_ADR, TSN_MAC_MDIO_GCR, gcr_reg); + mdio_ops->write(dev, TSN_MAC_MDIO_ADHOC_ADR, TSN_MAC_MDIO_GCR, gcr_reg); } }
@@ -114,19 +64,22 @@ void *io_mem_base = (void *)(uintptr_t)gbe_tsn_res->base; config_t *config = config_of(dev);
+ /* Enable MDIO interface */ + mdio_dev_enable(dev); + /* Program MAC address */ program_mac_address(dev, io_mem_base);
/* Set PHY-to-MAC IRQ polarity according to devicetree */ switch (dev->path.pci.devfn) { case PCH_DEVFN_GBE: - tsn_set_phy2mac_irq_polarity(io_mem_base, config->pch_tsn_phy_irq_edge); + tsn_set_phy2mac_irq_polarity(dev, config->pch_tsn_phy_irq_edge); break; case PCH_DEVFN_PSEGBE0: - tsn_set_phy2mac_irq_polarity(io_mem_base, config->pse_tsn_phy_irq_edge[0]); + tsn_set_phy2mac_irq_polarity(dev, config->pse_tsn_phy_irq_edge[0]); break; case PCH_DEVFN_PSEGBE1: - tsn_set_phy2mac_irq_polarity(io_mem_base, config->pse_tsn_phy_irq_edge[1]); + tsn_set_phy2mac_irq_polarity(dev, config->pse_tsn_phy_irq_edge[1]); break; } }