mirror of
				https://source.denx.de/u-boot/u-boot.git
				synced 2025-11-04 10:21:25 +01:00 
			
		
		
		
	The bcmgenet and sun8i_emac drivers call phy_connect(), which finds / creates the PHY and also connects it to the eth device via phy_connect_dev(), then set some phydev members (bcmgenet only), and then call phy_connect_dev() explicitly again. Drop the second phy_connect_dev(), since it is unnecesary. Signed-off-by: Marek Behún <marek.behun@nic.cz> Reviewed-by: Ramon Fried <rfried.dev@gmail.com>
		
			
				
	
	
		
			731 lines
		
	
	
		
			21 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			731 lines
		
	
	
		
			21 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
// SPDX-License-Identifier: GPL-2.0+
 | 
						|
/*
 | 
						|
 * Copyright (C) 2019 Amit Singh Tomar <amittomer25@gmail.com>
 | 
						|
 *
 | 
						|
 * Driver for Broadcom GENETv5 Ethernet controller (as found on the RPi4)
 | 
						|
 * This driver is based on the Linux driver:
 | 
						|
 *      drivers/net/ethernet/broadcom/genet/bcmgenet.c
 | 
						|
 *      which is: Copyright (c) 2014-2017 Broadcom
 | 
						|
 *
 | 
						|
 * The hardware supports multiple queues (16 priority queues and one
 | 
						|
 * default queue), both for RX and TX. There are 256 DMA descriptors (both
 | 
						|
 * for TX and RX), and they live in MMIO registers. The hardware allows
 | 
						|
 * assigning descriptor ranges to queues, but we choose the most simple setup:
 | 
						|
 * All 256 descriptors are assigned to the default queue (#16).
 | 
						|
 * Also the Linux driver supports multiple generations of the MAC, whereas
 | 
						|
 * we only support v5, as used in the Raspberry Pi 4.
 | 
						|
 */
 | 
						|
 | 
						|
#include <log.h>
 | 
						|
#include <asm/cache.h>
 | 
						|
#include <asm/io.h>
 | 
						|
#include <clk.h>
 | 
						|
#include <cpu_func.h>
 | 
						|
#include <dm.h>
 | 
						|
#include <fdt_support.h>
 | 
						|
#include <linux/bitops.h>
 | 
						|
#include <linux/delay.h>
 | 
						|
#include <linux/err.h>
 | 
						|
#include <malloc.h>
 | 
						|
#include <miiphy.h>
 | 
						|
#include <net.h>
 | 
						|
#include <dm/of_access.h>
 | 
						|
#include <dm/ofnode.h>
 | 
						|
#include <linux/iopoll.h>
 | 
						|
#include <linux/sizes.h>
 | 
						|
#include <asm/dma-mapping.h>
 | 
						|
#include <wait_bit.h>
 | 
						|
 | 
						|
/* Register definitions derived from Linux source */
 | 
						|
#define SYS_REV_CTRL			0x00
 | 
						|
 | 
						|
#define SYS_PORT_CTRL			0x04
 | 
						|
#define PORT_MODE_EXT_GPHY		3
 | 
						|
 | 
						|
#define GENET_SYS_OFF			0x0000
 | 
						|
#define SYS_RBUF_FLUSH_CTRL		(GENET_SYS_OFF  + 0x08)
 | 
						|
#define SYS_TBUF_FLUSH_CTRL		(GENET_SYS_OFF  + 0x0c)
 | 
						|
 | 
						|
#define GENET_EXT_OFF			0x0080
 | 
						|
#define EXT_RGMII_OOB_CTRL		(GENET_EXT_OFF + 0x0c)
 | 
						|
#define RGMII_LINK			BIT(4)
 | 
						|
#define OOB_DISABLE			BIT(5)
 | 
						|
#define RGMII_MODE_EN			BIT(6)
 | 
						|
#define ID_MODE_DIS			BIT(16)
 | 
						|
 | 
						|
#define GENET_RBUF_OFF			0x0300
 | 
						|
#define RBUF_TBUF_SIZE_CTRL		(GENET_RBUF_OFF + 0xb4)
 | 
						|
#define RBUF_CTRL			(GENET_RBUF_OFF + 0x00)
 | 
						|
#define RBUF_ALIGN_2B			BIT(1)
 | 
						|
 | 
						|
#define GENET_UMAC_OFF			0x0800
 | 
						|
#define UMAC_MIB_CTRL			(GENET_UMAC_OFF + 0x580)
 | 
						|
#define UMAC_MAX_FRAME_LEN		(GENET_UMAC_OFF + 0x014)
 | 
						|
#define UMAC_MAC0			(GENET_UMAC_OFF + 0x00c)
 | 
						|
#define UMAC_MAC1			(GENET_UMAC_OFF + 0x010)
 | 
						|
#define UMAC_CMD			(GENET_UMAC_OFF + 0x008)
 | 
						|
#define MDIO_CMD			(GENET_UMAC_OFF + 0x614)
 | 
						|
#define UMAC_TX_FLUSH			(GENET_UMAC_OFF + 0x334)
 | 
						|
#define MDIO_START_BUSY			BIT(29)
 | 
						|
#define MDIO_READ_FAIL			BIT(28)
 | 
						|
#define MDIO_RD				(2 << 26)
 | 
						|
#define MDIO_WR				BIT(26)
 | 
						|
#define MDIO_PMD_SHIFT			21
 | 
						|
#define MDIO_PMD_MASK			0x1f
 | 
						|
#define MDIO_REG_SHIFT			16
 | 
						|
#define MDIO_REG_MASK			0x1f
 | 
						|
 | 
						|
#define CMD_TX_EN			BIT(0)
 | 
						|
#define CMD_RX_EN			BIT(1)
 | 
						|
#define UMAC_SPEED_10			0
 | 
						|
#define UMAC_SPEED_100			1
 | 
						|
#define UMAC_SPEED_1000			2
 | 
						|
#define UMAC_SPEED_2500			3
 | 
						|
#define CMD_SPEED_SHIFT			2
 | 
						|
#define CMD_SPEED_MASK			3
 | 
						|
#define CMD_SW_RESET			BIT(13)
 | 
						|
#define CMD_LCL_LOOP_EN			BIT(15)
 | 
						|
#define CMD_TX_EN			BIT(0)
 | 
						|
#define CMD_RX_EN			BIT(1)
 | 
						|
 | 
						|
#define MIB_RESET_RX			BIT(0)
 | 
						|
#define MIB_RESET_RUNT			BIT(1)
 | 
						|
#define MIB_RESET_TX			BIT(2)
 | 
						|
 | 
						|
/* total number of Buffer Descriptors, same for Rx/Tx */
 | 
						|
#define TOTAL_DESCS			256
 | 
						|
#define RX_DESCS			TOTAL_DESCS
 | 
						|
#define TX_DESCS			TOTAL_DESCS
 | 
						|
 | 
						|
#define DEFAULT_Q			0x10
 | 
						|
 | 
						|
/* Body(1500) + EH_SIZE(14) + VLANTAG(4) + BRCMTAG(6) + FCS(4) = 1528.
 | 
						|
 * 1536 is multiple of 256 bytes
 | 
						|
 */
 | 
						|
#define ENET_BRCM_TAG_LEN		6
 | 
						|
#define ENET_PAD			8
 | 
						|
#define ENET_MAX_MTU_SIZE		(ETH_DATA_LEN + ETH_HLEN +	 \
 | 
						|
					 VLAN_HLEN + ENET_BRCM_TAG_LEN + \
 | 
						|
					 ETH_FCS_LEN + ENET_PAD)
 | 
						|
 | 
						|
/* Tx/Rx Dma Descriptor common bits */
 | 
						|
#define DMA_EN				BIT(0)
 | 
						|
#define DMA_RING_BUF_EN_SHIFT		0x01
 | 
						|
#define DMA_RING_BUF_EN_MASK		0xffff
 | 
						|
#define DMA_BUFLENGTH_MASK		0x0fff
 | 
						|
#define DMA_BUFLENGTH_SHIFT		16
 | 
						|
#define DMA_RING_SIZE_SHIFT		16
 | 
						|
#define DMA_OWN				0x8000
 | 
						|
#define DMA_EOP				0x4000
 | 
						|
#define DMA_SOP				0x2000
 | 
						|
#define DMA_WRAP			0x1000
 | 
						|
#define DMA_MAX_BURST_LENGTH		0x8
 | 
						|
/* Tx specific DMA descriptor bits */
 | 
						|
#define DMA_TX_UNDERRUN			0x0200
 | 
						|
#define DMA_TX_APPEND_CRC		0x0040
 | 
						|
#define DMA_TX_OW_CRC			0x0020
 | 
						|
#define DMA_TX_DO_CSUM			0x0010
 | 
						|
#define DMA_TX_QTAG_SHIFT		7
 | 
						|
 | 
						|
/* DMA rings size */
 | 
						|
#define DMA_RING_SIZE			0x40
 | 
						|
#define DMA_RINGS_SIZE			(DMA_RING_SIZE * (DEFAULT_Q + 1))
 | 
						|
 | 
						|
/* DMA descriptor */
 | 
						|
#define DMA_DESC_LENGTH_STATUS		0x00
 | 
						|
#define DMA_DESC_ADDRESS_LO		0x04
 | 
						|
#define DMA_DESC_ADDRESS_HI		0x08
 | 
						|
#define DMA_DESC_SIZE			12
 | 
						|
 | 
						|
#define GENET_RX_OFF			0x2000
 | 
						|
#define GENET_RDMA_REG_OFF					\
 | 
						|
	(GENET_RX_OFF + TOTAL_DESCS * DMA_DESC_SIZE)
 | 
						|
#define GENET_TX_OFF			0x4000
 | 
						|
#define GENET_TDMA_REG_OFF					\
 | 
						|
	(GENET_TX_OFF + TOTAL_DESCS * DMA_DESC_SIZE)
 | 
						|
 | 
						|
#define DMA_FC_THRESH_HI		(RX_DESCS >> 4)
 | 
						|
#define DMA_FC_THRESH_LO		5
 | 
						|
#define DMA_FC_THRESH_VALUE		((DMA_FC_THRESH_LO << 16) |	\
 | 
						|
					  DMA_FC_THRESH_HI)
 | 
						|
 | 
						|
#define DMA_XOFF_THRESHOLD_SHIFT	16
 | 
						|
 | 
						|
#define TDMA_RING_REG_BASE					\
 | 
						|
	(GENET_TDMA_REG_OFF + DEFAULT_Q * DMA_RING_SIZE)
 | 
						|
#define TDMA_READ_PTR			(TDMA_RING_REG_BASE + 0x00)
 | 
						|
#define TDMA_CONS_INDEX			(TDMA_RING_REG_BASE + 0x08)
 | 
						|
#define TDMA_PROD_INDEX			(TDMA_RING_REG_BASE + 0x0c)
 | 
						|
#define DMA_RING_BUF_SIZE		0x10
 | 
						|
#define DMA_START_ADDR			0x14
 | 
						|
#define DMA_END_ADDR			0x1c
 | 
						|
#define DMA_MBUF_DONE_THRESH		0x24
 | 
						|
#define TDMA_FLOW_PERIOD		(TDMA_RING_REG_BASE + 0x28)
 | 
						|
#define TDMA_WRITE_PTR			(TDMA_RING_REG_BASE + 0x2c)
 | 
						|
 | 
						|
#define RDMA_RING_REG_BASE					\
 | 
						|
	(GENET_RDMA_REG_OFF + DEFAULT_Q * DMA_RING_SIZE)
 | 
						|
#define RDMA_WRITE_PTR			(RDMA_RING_REG_BASE + 0x00)
 | 
						|
#define RDMA_PROD_INDEX			(RDMA_RING_REG_BASE + 0x08)
 | 
						|
#define RDMA_CONS_INDEX			(RDMA_RING_REG_BASE + 0x0c)
 | 
						|
#define RDMA_XON_XOFF_THRESH		(RDMA_RING_REG_BASE + 0x28)
 | 
						|
#define RDMA_READ_PTR			(RDMA_RING_REG_BASE + 0x2c)
 | 
						|
 | 
						|
#define TDMA_REG_BASE			(GENET_TDMA_REG_OFF + DMA_RINGS_SIZE)
 | 
						|
#define RDMA_REG_BASE			(GENET_RDMA_REG_OFF + DMA_RINGS_SIZE)
 | 
						|
#define DMA_RING_CFG			0x00
 | 
						|
#define DMA_CTRL			0x04
 | 
						|
#define DMA_SCB_BURST_SIZE		0x0c
 | 
						|
 | 
						|
#define RX_BUF_LENGTH			2048
 | 
						|
#define RX_TOTAL_BUFSIZE		(RX_BUF_LENGTH * RX_DESCS)
 | 
						|
#define RX_BUF_OFFSET			2
 | 
						|
 | 
						|
struct bcmgenet_eth_priv {
 | 
						|
	char rxbuffer[RX_TOTAL_BUFSIZE] __aligned(ARCH_DMA_MINALIGN);
 | 
						|
	void *mac_reg;
 | 
						|
	void *tx_desc_base;
 | 
						|
	void *rx_desc_base;
 | 
						|
	int tx_index;
 | 
						|
	int rx_index;
 | 
						|
	int c_index;
 | 
						|
	int phyaddr;
 | 
						|
	u32 interface;
 | 
						|
	u32 speed;
 | 
						|
	struct phy_device *phydev;
 | 
						|
	struct mii_dev *bus;
 | 
						|
};
 | 
						|
 | 
						|
static void bcmgenet_umac_reset(struct bcmgenet_eth_priv *priv)
 | 
						|
{
 | 
						|
	u32 reg;
 | 
						|
 | 
						|
	reg = readl(priv->mac_reg + SYS_RBUF_FLUSH_CTRL);
 | 
						|
	reg |= BIT(1);
 | 
						|
	writel(reg, (priv->mac_reg + SYS_RBUF_FLUSH_CTRL));
 | 
						|
	udelay(10);
 | 
						|
 | 
						|
	reg &= ~BIT(1);
 | 
						|
	writel(reg, (priv->mac_reg + SYS_RBUF_FLUSH_CTRL));
 | 
						|
	udelay(10);
 | 
						|
 | 
						|
	writel(0, (priv->mac_reg + SYS_RBUF_FLUSH_CTRL));
 | 
						|
	udelay(10);
 | 
						|
 | 
						|
	writel(0, priv->mac_reg + UMAC_CMD);
 | 
						|
 | 
						|
	writel(CMD_SW_RESET | CMD_LCL_LOOP_EN, priv->mac_reg + UMAC_CMD);
 | 
						|
	udelay(2);
 | 
						|
	writel(0, priv->mac_reg + UMAC_CMD);
 | 
						|
 | 
						|
	/* clear tx/rx counter */
 | 
						|
	writel(MIB_RESET_RX | MIB_RESET_TX | MIB_RESET_RUNT,
 | 
						|
	       priv->mac_reg + UMAC_MIB_CTRL);
 | 
						|
	writel(0, priv->mac_reg + UMAC_MIB_CTRL);
 | 
						|
 | 
						|
	writel(ENET_MAX_MTU_SIZE, priv->mac_reg + UMAC_MAX_FRAME_LEN);
 | 
						|
 | 
						|
	/* init rx registers, enable ip header optimization */
 | 
						|
	reg = readl(priv->mac_reg + RBUF_CTRL);
 | 
						|
	reg |= RBUF_ALIGN_2B;
 | 
						|
	writel(reg, (priv->mac_reg + RBUF_CTRL));
 | 
						|
 | 
						|
	writel(1, (priv->mac_reg + RBUF_TBUF_SIZE_CTRL));
 | 
						|
}
 | 
						|
 | 
						|
static int bcmgenet_gmac_write_hwaddr(struct udevice *dev)
 | 
						|
{
 | 
						|
	struct bcmgenet_eth_priv *priv = dev_get_priv(dev);
 | 
						|
	struct eth_pdata *pdata = dev_get_plat(dev);
 | 
						|
	uchar *addr = pdata->enetaddr;
 | 
						|
	u32 reg;
 | 
						|
 | 
						|
	reg = addr[0] << 24 | addr[1] << 16 | addr[2] << 8 | addr[3];
 | 
						|
	writel_relaxed(reg, priv->mac_reg + UMAC_MAC0);
 | 
						|
 | 
						|
	reg = addr[4] << 8 | addr[5];
 | 
						|
	writel_relaxed(reg, priv->mac_reg + UMAC_MAC1);
 | 
						|
 | 
						|
	return 0;
 | 
						|
}
 | 
						|
 | 
						|
static void bcmgenet_disable_dma(struct bcmgenet_eth_priv *priv)
 | 
						|
{
 | 
						|
	clrbits_32(priv->mac_reg + TDMA_REG_BASE + DMA_CTRL, DMA_EN);
 | 
						|
	clrbits_32(priv->mac_reg + RDMA_REG_BASE + DMA_CTRL, DMA_EN);
 | 
						|
 | 
						|
	writel(1, priv->mac_reg + UMAC_TX_FLUSH);
 | 
						|
	udelay(10);
 | 
						|
	writel(0, priv->mac_reg + UMAC_TX_FLUSH);
 | 
						|
}
 | 
						|
 | 
						|
static void bcmgenet_enable_dma(struct bcmgenet_eth_priv *priv)
 | 
						|
{
 | 
						|
	u32 dma_ctrl = (1 << (DEFAULT_Q + DMA_RING_BUF_EN_SHIFT)) | DMA_EN;
 | 
						|
 | 
						|
	writel(dma_ctrl, priv->mac_reg + TDMA_REG_BASE + DMA_CTRL);
 | 
						|
 | 
						|
	setbits_32(priv->mac_reg + RDMA_REG_BASE + DMA_CTRL, dma_ctrl);
 | 
						|
}
 | 
						|
 | 
						|
static int bcmgenet_gmac_eth_send(struct udevice *dev, void *packet, int length)
 | 
						|
{
 | 
						|
	struct bcmgenet_eth_priv *priv = dev_get_priv(dev);
 | 
						|
	void *desc_base = priv->tx_desc_base + priv->tx_index * DMA_DESC_SIZE;
 | 
						|
	u32 len_stat = length << DMA_BUFLENGTH_SHIFT;
 | 
						|
	ulong packet_aligned = rounddown((ulong)packet, ARCH_DMA_MINALIGN);
 | 
						|
	u32 prod_index, cons;
 | 
						|
	u32 tries = 100;
 | 
						|
 | 
						|
	prod_index = readl(priv->mac_reg + TDMA_PROD_INDEX);
 | 
						|
 | 
						|
	/* There is actually no reason for the rounding here, but the ARMv7
 | 
						|
	 * implementation of flush_dcache_range() checks for aligned
 | 
						|
	 * boundaries of the flushed range.
 | 
						|
	 * Adjust them here to pass that check and avoid misleading messages.
 | 
						|
	 */
 | 
						|
	flush_dcache_range(packet_aligned,
 | 
						|
			   packet_aligned + roundup(length, ARCH_DMA_MINALIGN));
 | 
						|
 | 
						|
	len_stat |= 0x3F << DMA_TX_QTAG_SHIFT;
 | 
						|
	len_stat |= DMA_TX_APPEND_CRC | DMA_SOP | DMA_EOP;
 | 
						|
 | 
						|
	/* Set-up packet for transmission */
 | 
						|
	writel(lower_32_bits((ulong)packet), (desc_base + DMA_DESC_ADDRESS_LO));
 | 
						|
	writel(upper_32_bits((ulong)packet), (desc_base + DMA_DESC_ADDRESS_HI));
 | 
						|
	writel(len_stat, (desc_base + DMA_DESC_LENGTH_STATUS));
 | 
						|
 | 
						|
	/* Increment index and start transmission */
 | 
						|
	if (++priv->tx_index >= TX_DESCS)
 | 
						|
		priv->tx_index = 0;
 | 
						|
 | 
						|
	prod_index++;
 | 
						|
 | 
						|
	/* Start Transmisson */
 | 
						|
	writel(prod_index, priv->mac_reg + TDMA_PROD_INDEX);
 | 
						|
 | 
						|
	do {
 | 
						|
		cons = readl(priv->mac_reg + TDMA_CONS_INDEX);
 | 
						|
	} while ((cons & 0xffff) < prod_index && --tries);
 | 
						|
	if (!tries)
 | 
						|
		return -ETIMEDOUT;
 | 
						|
 | 
						|
	return 0;
 | 
						|
}
 | 
						|
 | 
						|
/* Check whether all cache lines affected by an invalidate are within
 | 
						|
 * the buffer, to make sure we don't accidentally lose unrelated dirty
 | 
						|
 * data stored nearby.
 | 
						|
 * Alignment of the buffer start address will be checked in the implementation
 | 
						|
 * of invalidate_dcache_range().
 | 
						|
 */
 | 
						|
static void invalidate_dcache_check(unsigned long addr, size_t size,
 | 
						|
				    size_t buffer_size)
 | 
						|
{
 | 
						|
	size_t inval_size = roundup(size, ARCH_DMA_MINALIGN);
 | 
						|
 | 
						|
	if (unlikely(inval_size > buffer_size))
 | 
						|
		printf("WARNING: Cache invalidate area exceeds buffer size\n");
 | 
						|
 | 
						|
	invalidate_dcache_range(addr, addr + inval_size);
 | 
						|
}
 | 
						|
 | 
						|
static int bcmgenet_gmac_eth_recv(struct udevice *dev,
 | 
						|
				  int flags, uchar **packetp)
 | 
						|
{
 | 
						|
	struct bcmgenet_eth_priv *priv = dev_get_priv(dev);
 | 
						|
	void *desc_base = priv->rx_desc_base + priv->rx_index * DMA_DESC_SIZE;
 | 
						|
	u32 prod_index = readl(priv->mac_reg + RDMA_PROD_INDEX);
 | 
						|
	u32 length, addr;
 | 
						|
 | 
						|
	if (prod_index == priv->c_index)
 | 
						|
		return -EAGAIN;
 | 
						|
 | 
						|
	length = readl(desc_base + DMA_DESC_LENGTH_STATUS);
 | 
						|
	length = (length >> DMA_BUFLENGTH_SHIFT) & DMA_BUFLENGTH_MASK;
 | 
						|
	addr = readl(desc_base + DMA_DESC_ADDRESS_LO);
 | 
						|
 | 
						|
	invalidate_dcache_check(addr, length, RX_BUF_LENGTH);
 | 
						|
 | 
						|
	/* To cater for the IP header alignment the hardware does.
 | 
						|
	 * This would actually not be needed if we don't program
 | 
						|
	 * RBUF_ALIGN_2B
 | 
						|
	 */
 | 
						|
	*packetp = (uchar *)(ulong)addr + RX_BUF_OFFSET;
 | 
						|
 | 
						|
	return length - RX_BUF_OFFSET;
 | 
						|
}
 | 
						|
 | 
						|
static int bcmgenet_gmac_free_pkt(struct udevice *dev, uchar *packet,
 | 
						|
				  int length)
 | 
						|
{
 | 
						|
	struct bcmgenet_eth_priv *priv = dev_get_priv(dev);
 | 
						|
 | 
						|
	/* Tell the MAC we have consumed that last receive buffer. */
 | 
						|
	priv->c_index = (priv->c_index + 1) & 0xFFFF;
 | 
						|
	writel(priv->c_index, priv->mac_reg + RDMA_CONS_INDEX);
 | 
						|
 | 
						|
	/* Forward our descriptor pointer, wrapping around if needed. */
 | 
						|
	if (++priv->rx_index >= RX_DESCS)
 | 
						|
		priv->rx_index = 0;
 | 
						|
 | 
						|
	return 0;
 | 
						|
}
 | 
						|
 | 
						|
static void rx_descs_init(struct bcmgenet_eth_priv *priv)
 | 
						|
{
 | 
						|
	char *rxbuffs = &priv->rxbuffer[0];
 | 
						|
	u32 len_stat, i;
 | 
						|
	void *desc_base = priv->rx_desc_base;
 | 
						|
 | 
						|
	len_stat = (RX_BUF_LENGTH << DMA_BUFLENGTH_SHIFT) | DMA_OWN;
 | 
						|
 | 
						|
	for (i = 0; i < RX_DESCS; i++) {
 | 
						|
		writel(lower_32_bits((uintptr_t)&rxbuffs[i * RX_BUF_LENGTH]),
 | 
						|
		       desc_base + i * DMA_DESC_SIZE + DMA_DESC_ADDRESS_LO);
 | 
						|
		writel(upper_32_bits((uintptr_t)&rxbuffs[i * RX_BUF_LENGTH]),
 | 
						|
		       desc_base + i * DMA_DESC_SIZE + DMA_DESC_ADDRESS_HI);
 | 
						|
		writel(len_stat,
 | 
						|
		       desc_base + i * DMA_DESC_SIZE + DMA_DESC_LENGTH_STATUS);
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
static void rx_ring_init(struct bcmgenet_eth_priv *priv)
 | 
						|
{
 | 
						|
	writel(DMA_MAX_BURST_LENGTH,
 | 
						|
	       priv->mac_reg + RDMA_REG_BASE + DMA_SCB_BURST_SIZE);
 | 
						|
 | 
						|
	writel(0x0, priv->mac_reg + RDMA_RING_REG_BASE + DMA_START_ADDR);
 | 
						|
	writel(0x0, priv->mac_reg + RDMA_READ_PTR);
 | 
						|
	writel(0x0, priv->mac_reg + RDMA_WRITE_PTR);
 | 
						|
	writel(RX_DESCS * DMA_DESC_SIZE / 4 - 1,
 | 
						|
	       priv->mac_reg + RDMA_RING_REG_BASE + DMA_END_ADDR);
 | 
						|
 | 
						|
	/* cannot init RDMA_PROD_INDEX to 0, so align RDMA_CONS_INDEX on it instead */
 | 
						|
	priv->c_index = readl(priv->mac_reg + RDMA_PROD_INDEX);
 | 
						|
	writel(priv->c_index, priv->mac_reg + RDMA_CONS_INDEX);
 | 
						|
	priv->rx_index = priv->c_index;
 | 
						|
	priv->rx_index &= 0xFF;
 | 
						|
	writel((RX_DESCS << DMA_RING_SIZE_SHIFT) | RX_BUF_LENGTH,
 | 
						|
	       priv->mac_reg + RDMA_RING_REG_BASE + DMA_RING_BUF_SIZE);
 | 
						|
	writel(DMA_FC_THRESH_VALUE, priv->mac_reg + RDMA_XON_XOFF_THRESH);
 | 
						|
	writel(1 << DEFAULT_Q, priv->mac_reg + RDMA_REG_BASE + DMA_RING_CFG);
 | 
						|
}
 | 
						|
 | 
						|
static void tx_ring_init(struct bcmgenet_eth_priv *priv)
 | 
						|
{
 | 
						|
	writel(DMA_MAX_BURST_LENGTH,
 | 
						|
	       priv->mac_reg + TDMA_REG_BASE + DMA_SCB_BURST_SIZE);
 | 
						|
 | 
						|
	writel(0x0, priv->mac_reg + TDMA_RING_REG_BASE + DMA_START_ADDR);
 | 
						|
	writel(0x0, priv->mac_reg + TDMA_READ_PTR);
 | 
						|
	writel(0x0, priv->mac_reg + TDMA_WRITE_PTR);
 | 
						|
	writel(TX_DESCS * DMA_DESC_SIZE / 4 - 1,
 | 
						|
	       priv->mac_reg + TDMA_RING_REG_BASE + DMA_END_ADDR);
 | 
						|
	/* cannot init TDMA_CONS_INDEX to 0, so align TDMA_PROD_INDEX on it instead */
 | 
						|
	priv->tx_index = readl(priv->mac_reg + TDMA_CONS_INDEX);
 | 
						|
	writel(priv->tx_index, priv->mac_reg + TDMA_PROD_INDEX);
 | 
						|
	priv->tx_index &= 0xFF;
 | 
						|
	writel(0x1, priv->mac_reg + TDMA_RING_REG_BASE + DMA_MBUF_DONE_THRESH);
 | 
						|
	writel(0x0, priv->mac_reg + TDMA_FLOW_PERIOD);
 | 
						|
	writel((TX_DESCS << DMA_RING_SIZE_SHIFT) | RX_BUF_LENGTH,
 | 
						|
	       priv->mac_reg + TDMA_RING_REG_BASE + DMA_RING_BUF_SIZE);
 | 
						|
 | 
						|
	writel(1 << DEFAULT_Q, priv->mac_reg + TDMA_REG_BASE + DMA_RING_CFG);
 | 
						|
}
 | 
						|
 | 
						|
static int bcmgenet_adjust_link(struct bcmgenet_eth_priv *priv)
 | 
						|
{
 | 
						|
	struct phy_device *phy_dev = priv->phydev;
 | 
						|
	u32 speed;
 | 
						|
 | 
						|
	switch (phy_dev->speed) {
 | 
						|
	case SPEED_1000:
 | 
						|
		speed = UMAC_SPEED_1000;
 | 
						|
		break;
 | 
						|
	case SPEED_100:
 | 
						|
		speed = UMAC_SPEED_100;
 | 
						|
		break;
 | 
						|
	case SPEED_10:
 | 
						|
		speed = UMAC_SPEED_10;
 | 
						|
		break;
 | 
						|
	default:
 | 
						|
		printf("bcmgenet: Unsupported PHY speed: %d\n", phy_dev->speed);
 | 
						|
		return -EINVAL;
 | 
						|
	}
 | 
						|
 | 
						|
	clrsetbits_32(priv->mac_reg + EXT_RGMII_OOB_CTRL, OOB_DISABLE,
 | 
						|
			RGMII_LINK | RGMII_MODE_EN);
 | 
						|
 | 
						|
	if (phy_dev->interface == PHY_INTERFACE_MODE_RGMII ||
 | 
						|
	    phy_dev->interface == PHY_INTERFACE_MODE_RGMII_RXID)
 | 
						|
		setbits_32(priv->mac_reg + EXT_RGMII_OOB_CTRL, ID_MODE_DIS);
 | 
						|
 | 
						|
	writel(speed << CMD_SPEED_SHIFT, (priv->mac_reg + UMAC_CMD));
 | 
						|
 | 
						|
	return 0;
 | 
						|
}
 | 
						|
 | 
						|
static int bcmgenet_gmac_eth_start(struct udevice *dev)
 | 
						|
{
 | 
						|
	struct bcmgenet_eth_priv *priv = dev_get_priv(dev);
 | 
						|
	int ret;
 | 
						|
 | 
						|
	priv->tx_desc_base = priv->mac_reg + GENET_TX_OFF;
 | 
						|
	priv->rx_desc_base = priv->mac_reg + GENET_RX_OFF;
 | 
						|
 | 
						|
	bcmgenet_umac_reset(priv);
 | 
						|
 | 
						|
	bcmgenet_gmac_write_hwaddr(dev);
 | 
						|
 | 
						|
	/* Disable RX/TX DMA and flush TX queues */
 | 
						|
	bcmgenet_disable_dma(priv);
 | 
						|
 | 
						|
	rx_ring_init(priv);
 | 
						|
	rx_descs_init(priv);
 | 
						|
 | 
						|
	tx_ring_init(priv);
 | 
						|
 | 
						|
	/* Enable RX/TX DMA */
 | 
						|
	bcmgenet_enable_dma(priv);
 | 
						|
 | 
						|
	/* read PHY properties over the wire from generic PHY set-up */
 | 
						|
	ret = phy_startup(priv->phydev);
 | 
						|
	if (ret) {
 | 
						|
		printf("bcmgenet: PHY startup failed: %d\n", ret);
 | 
						|
		return ret;
 | 
						|
	}
 | 
						|
 | 
						|
	/* Update MAC registers based on PHY property */
 | 
						|
	ret = bcmgenet_adjust_link(priv);
 | 
						|
	if (ret) {
 | 
						|
		printf("bcmgenet: adjust PHY link failed: %d\n", ret);
 | 
						|
		return ret;
 | 
						|
	}
 | 
						|
 | 
						|
	/* Enable Rx/Tx */
 | 
						|
	setbits_32(priv->mac_reg + UMAC_CMD, CMD_TX_EN | CMD_RX_EN);
 | 
						|
 | 
						|
	return 0;
 | 
						|
}
 | 
						|
 | 
						|
static int bcmgenet_phy_init(struct bcmgenet_eth_priv *priv, void *dev)
 | 
						|
{
 | 
						|
	struct phy_device *phydev;
 | 
						|
	int ret;
 | 
						|
 | 
						|
	phydev = phy_connect(priv->bus, priv->phyaddr, dev, priv->interface);
 | 
						|
	if (!phydev)
 | 
						|
		return -ENODEV;
 | 
						|
 | 
						|
	phydev->supported &= PHY_GBIT_FEATURES;
 | 
						|
	if (priv->speed) {
 | 
						|
		ret = phy_set_supported(priv->phydev, priv->speed);
 | 
						|
		if (ret)
 | 
						|
			return ret;
 | 
						|
	}
 | 
						|
	phydev->advertising = phydev->supported;
 | 
						|
 | 
						|
	priv->phydev = phydev;
 | 
						|
	phy_config(priv->phydev);
 | 
						|
 | 
						|
	return 0;
 | 
						|
}
 | 
						|
 | 
						|
static void bcmgenet_mdio_start(struct bcmgenet_eth_priv *priv)
 | 
						|
{
 | 
						|
	setbits_32(priv->mac_reg + MDIO_CMD, MDIO_START_BUSY);
 | 
						|
}
 | 
						|
 | 
						|
static int bcmgenet_mdio_write(struct mii_dev *bus, int addr, int devad,
 | 
						|
			       int reg, u16 value)
 | 
						|
{
 | 
						|
	struct udevice *dev = bus->priv;
 | 
						|
	struct bcmgenet_eth_priv *priv = dev_get_priv(dev);
 | 
						|
	u32 val;
 | 
						|
 | 
						|
	/* Prepare the read operation */
 | 
						|
	val = MDIO_WR | (addr << MDIO_PMD_SHIFT) |
 | 
						|
		(reg << MDIO_REG_SHIFT) | (0xffff & value);
 | 
						|
	writel_relaxed(val,  priv->mac_reg + MDIO_CMD);
 | 
						|
 | 
						|
	/* Start MDIO transaction */
 | 
						|
	bcmgenet_mdio_start(priv);
 | 
						|
 | 
						|
	return wait_for_bit_32(priv->mac_reg + MDIO_CMD,
 | 
						|
			       MDIO_START_BUSY, false, 20, true);
 | 
						|
}
 | 
						|
 | 
						|
static int bcmgenet_mdio_read(struct mii_dev *bus, int addr, int devad, int reg)
 | 
						|
{
 | 
						|
	struct udevice *dev = bus->priv;
 | 
						|
	struct bcmgenet_eth_priv *priv = dev_get_priv(dev);
 | 
						|
	u32 val;
 | 
						|
	int ret;
 | 
						|
 | 
						|
	/* Prepare the read operation */
 | 
						|
	val = MDIO_RD | (addr << MDIO_PMD_SHIFT) | (reg << MDIO_REG_SHIFT);
 | 
						|
	writel_relaxed(val, priv->mac_reg + MDIO_CMD);
 | 
						|
 | 
						|
	/* Start MDIO transaction */
 | 
						|
	bcmgenet_mdio_start(priv);
 | 
						|
 | 
						|
	ret = wait_for_bit_32(priv->mac_reg + MDIO_CMD,
 | 
						|
			      MDIO_START_BUSY, false, 20, true);
 | 
						|
	if (ret)
 | 
						|
		return ret;
 | 
						|
 | 
						|
	val = readl_relaxed(priv->mac_reg + MDIO_CMD);
 | 
						|
 | 
						|
	return val & 0xffff;
 | 
						|
}
 | 
						|
 | 
						|
static int bcmgenet_mdio_init(const char *name, struct udevice *priv)
 | 
						|
{
 | 
						|
	struct mii_dev *bus = mdio_alloc();
 | 
						|
 | 
						|
	if (!bus) {
 | 
						|
		debug("Failed to allocate MDIO bus\n");
 | 
						|
		return -ENOMEM;
 | 
						|
	}
 | 
						|
 | 
						|
	bus->read = bcmgenet_mdio_read;
 | 
						|
	bus->write = bcmgenet_mdio_write;
 | 
						|
	snprintf(bus->name, sizeof(bus->name), name);
 | 
						|
	bus->priv = (void *)priv;
 | 
						|
 | 
						|
	return mdio_register(bus);
 | 
						|
}
 | 
						|
 | 
						|
/* We only support RGMII (as used on the RPi4). */
 | 
						|
static int bcmgenet_interface_set(struct bcmgenet_eth_priv *priv)
 | 
						|
{
 | 
						|
	phy_interface_t phy_mode = priv->interface;
 | 
						|
 | 
						|
	switch (phy_mode) {
 | 
						|
	case PHY_INTERFACE_MODE_RGMII:
 | 
						|
	case PHY_INTERFACE_MODE_RGMII_RXID:
 | 
						|
		writel(PORT_MODE_EXT_GPHY, priv->mac_reg + SYS_PORT_CTRL);
 | 
						|
		break;
 | 
						|
	default:
 | 
						|
		printf("unknown phy mode: %d\n", priv->interface);
 | 
						|
		return -EINVAL;
 | 
						|
	}
 | 
						|
 | 
						|
	return 0;
 | 
						|
}
 | 
						|
 | 
						|
static int bcmgenet_eth_probe(struct udevice *dev)
 | 
						|
{
 | 
						|
	struct eth_pdata *pdata = dev_get_plat(dev);
 | 
						|
	struct bcmgenet_eth_priv *priv = dev_get_priv(dev);
 | 
						|
	ofnode mdio_node;
 | 
						|
	const char *name;
 | 
						|
	u32 reg;
 | 
						|
	int ret;
 | 
						|
	u8 major;
 | 
						|
 | 
						|
	priv->mac_reg = map_physmem(pdata->iobase, SZ_64K, MAP_NOCACHE);
 | 
						|
	priv->interface = pdata->phy_interface;
 | 
						|
	priv->speed = pdata->max_speed;
 | 
						|
 | 
						|
	/* Read GENET HW version */
 | 
						|
	reg = readl_relaxed(priv->mac_reg + SYS_REV_CTRL);
 | 
						|
	major = (reg >> 24) & 0x0f;
 | 
						|
	if (major != 6) {
 | 
						|
		if (major == 5)
 | 
						|
			major = 4;
 | 
						|
		else if (major == 0)
 | 
						|
			major = 1;
 | 
						|
 | 
						|
		printf("Unsupported GENETv%d.%d\n", major, (reg >> 16) & 0x0f);
 | 
						|
		return -ENODEV;
 | 
						|
	}
 | 
						|
 | 
						|
	ret = bcmgenet_interface_set(priv);
 | 
						|
	if (ret)
 | 
						|
		return ret;
 | 
						|
 | 
						|
	writel(0, priv->mac_reg + SYS_RBUF_FLUSH_CTRL);
 | 
						|
	udelay(10);
 | 
						|
	/* disable MAC while updating its registers */
 | 
						|
	writel(0, priv->mac_reg + UMAC_CMD);
 | 
						|
	/* issue soft reset with (rg)mii loopback to ensure a stable rxclk */
 | 
						|
	writel(CMD_SW_RESET | CMD_LCL_LOOP_EN, priv->mac_reg + UMAC_CMD);
 | 
						|
 | 
						|
	mdio_node = dev_read_first_subnode(dev);
 | 
						|
	name = ofnode_get_name(mdio_node);
 | 
						|
 | 
						|
	ret = bcmgenet_mdio_init(name, dev);
 | 
						|
	if (ret)
 | 
						|
		return ret;
 | 
						|
 | 
						|
	priv->bus = miiphy_get_dev_by_name(name);
 | 
						|
 | 
						|
	return bcmgenet_phy_init(priv, dev);
 | 
						|
}
 | 
						|
 | 
						|
static void bcmgenet_gmac_eth_stop(struct udevice *dev)
 | 
						|
{
 | 
						|
	struct bcmgenet_eth_priv *priv = dev_get_priv(dev);
 | 
						|
 | 
						|
	clrbits_32(priv->mac_reg + UMAC_CMD, CMD_TX_EN | CMD_RX_EN);
 | 
						|
 | 
						|
	bcmgenet_disable_dma(priv);
 | 
						|
}
 | 
						|
 | 
						|
static const struct eth_ops bcmgenet_gmac_eth_ops = {
 | 
						|
	.start                  = bcmgenet_gmac_eth_start,
 | 
						|
	.write_hwaddr           = bcmgenet_gmac_write_hwaddr,
 | 
						|
	.send                   = bcmgenet_gmac_eth_send,
 | 
						|
	.recv                   = bcmgenet_gmac_eth_recv,
 | 
						|
	.free_pkt               = bcmgenet_gmac_free_pkt,
 | 
						|
	.stop                   = bcmgenet_gmac_eth_stop,
 | 
						|
};
 | 
						|
 | 
						|
static int bcmgenet_eth_of_to_plat(struct udevice *dev)
 | 
						|
{
 | 
						|
	struct eth_pdata *pdata = dev_get_plat(dev);
 | 
						|
	struct bcmgenet_eth_priv *priv = dev_get_priv(dev);
 | 
						|
	struct ofnode_phandle_args phy_node;
 | 
						|
	int ret;
 | 
						|
 | 
						|
	pdata->iobase = dev_read_addr(dev);
 | 
						|
 | 
						|
	/* Get phy mode from DT */
 | 
						|
	pdata->phy_interface = dev_read_phy_mode(dev);
 | 
						|
	if (pdata->phy_interface == PHY_INTERFACE_MODE_NA)
 | 
						|
		return -EINVAL;
 | 
						|
 | 
						|
	ret = dev_read_phandle_with_args(dev, "phy-handle", NULL, 0, 0,
 | 
						|
					 &phy_node);
 | 
						|
	if (!ret) {
 | 
						|
		ofnode_read_s32(phy_node.node, "reg", &priv->phyaddr);
 | 
						|
		ofnode_read_s32(phy_node.node, "max-speed", &pdata->max_speed);
 | 
						|
	}
 | 
						|
 | 
						|
	return 0;
 | 
						|
}
 | 
						|
 | 
						|
/* The BCM2711 implementation has a limited burst length compared to a generic
 | 
						|
 * GENETv5 version, but we go with that shorter value (8) in both cases, for
 | 
						|
 * the sake of simplicity.
 | 
						|
 */
 | 
						|
static const struct udevice_id bcmgenet_eth_ids[] = {
 | 
						|
	{.compatible = "brcm,genet-v5"},
 | 
						|
	{.compatible = "brcm,bcm2711-genet-v5"},
 | 
						|
	{}
 | 
						|
};
 | 
						|
 | 
						|
U_BOOT_DRIVER(eth_bcmgenet) = {
 | 
						|
	.name   = "eth_bcmgenet",
 | 
						|
	.id     = UCLASS_ETH,
 | 
						|
	.of_match = bcmgenet_eth_ids,
 | 
						|
	.of_to_plat = bcmgenet_eth_of_to_plat,
 | 
						|
	.probe  = bcmgenet_eth_probe,
 | 
						|
	.ops    = &bcmgenet_gmac_eth_ops,
 | 
						|
	.priv_auto	= sizeof(struct bcmgenet_eth_priv),
 | 
						|
	.plat_auto	= sizeof(struct eth_pdata),
 | 
						|
	.flags = DM_FLAG_ALLOC_PRIV_DMA,
 | 
						|
};
 |