mirror of
				https://github.com/smaeul/u-boot.git
				synced 2025-11-03 21:48:15 +00:00 
			
		
		
		
	As part of bringing the master branch back in to next, we need to allow for all of these changes to exist here. Reported-by: Jonas Karlman <jonas@kwiboo.se> Signed-off-by: Tom Rini <trini@konsulko.com>
		
			
				
	
	
		
			364 lines
		
	
	
		
			8.3 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			364 lines
		
	
	
		
			8.3 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
// SPDX-License-Identifier: GPL-2.0+
 | 
						|
/*
 | 
						|
 * Atheros PHY drivers
 | 
						|
 *
 | 
						|
 * Copyright 2011, 2013 Freescale Semiconductor, Inc.
 | 
						|
 * author Andy Fleming
 | 
						|
 * Copyright (c) 2019 Michael Walle <michael@walle.cc>
 | 
						|
 */
 | 
						|
#include <phy.h>
 | 
						|
#include <dm/device_compat.h>
 | 
						|
#include <linux/bitfield.h>
 | 
						|
#include <linux/bitops.h>
 | 
						|
#include <dt-bindings/net/qca-ar803x.h>
 | 
						|
 | 
						|
#define AR803x_PHY_DEBUG_ADDR_REG	0x1d
 | 
						|
#define AR803x_PHY_DEBUG_DATA_REG	0x1e
 | 
						|
 | 
						|
/* Debug registers */
 | 
						|
#define AR803x_DEBUG_REG_0		0x0
 | 
						|
#define AR803x_RGMII_RX_CLK_DLY		BIT(15)
 | 
						|
 | 
						|
#define AR803x_DEBUG_REG_5		0x5
 | 
						|
#define AR803x_RGMII_TX_CLK_DLY		BIT(8)
 | 
						|
 | 
						|
#define AR803x_DEBUG_REG_1F		0x1f
 | 
						|
#define AR803x_PLL_ON			BIT(2)
 | 
						|
#define AR803x_RGMII_1V8		BIT(3)
 | 
						|
 | 
						|
/* CLK_25M register is at MMD 7, address 0x8016 */
 | 
						|
#define AR803x_CLK_25M_SEL_REG		0x8016
 | 
						|
 | 
						|
#define AR803x_CLK_25M_MASK		GENMASK(4, 2)
 | 
						|
#define AR803x_CLK_25M_25MHZ_XTAL	0
 | 
						|
#define AR803x_CLK_25M_25MHZ_DSP	1
 | 
						|
#define AR803x_CLK_25M_50MHZ_PLL	2
 | 
						|
#define AR803x_CLK_25M_50MHZ_DSP	3
 | 
						|
#define AR803x_CLK_25M_62_5MHZ_PLL	4
 | 
						|
#define AR803x_CLK_25M_62_5MHZ_DSP	5
 | 
						|
#define AR803x_CLK_25M_125MHZ_PLL	6
 | 
						|
#define AR803x_CLK_25M_125MHZ_DSP	7
 | 
						|
#define AR8035_CLK_25M_MASK		GENMASK(4, 3)
 | 
						|
 | 
						|
#define AR803x_CLK_25M_DR_MASK		GENMASK(8, 7)
 | 
						|
#define AR803x_CLK_25M_DR_FULL		0
 | 
						|
#define AR803x_CLK_25M_DR_HALF		1
 | 
						|
#define AR803x_CLK_25M_DR_QUARTER	2
 | 
						|
 | 
						|
#define AR8021_PHY_ID 0x004dd040
 | 
						|
#define AR8031_PHY_ID 0x004dd074
 | 
						|
#define AR8035_PHY_ID 0x004dd072
 | 
						|
 | 
						|
struct ar803x_priv {
 | 
						|
	int flags;
 | 
						|
#define AR803x_FLAG_KEEP_PLL_ENABLED	BIT(0) /* don't turn off internal PLL */
 | 
						|
#define AR803x_FLAG_RGMII_1V8		BIT(1) /* use 1.8V RGMII I/O voltage */
 | 
						|
	u16 clk_25m_reg;
 | 
						|
	u16 clk_25m_mask;
 | 
						|
};
 | 
						|
 | 
						|
static int ar803x_debug_reg_read(struct phy_device *phydev, u16 reg)
 | 
						|
{
 | 
						|
	int ret;
 | 
						|
 | 
						|
	ret = phy_write(phydev, MDIO_DEVAD_NONE, AR803x_PHY_DEBUG_ADDR_REG,
 | 
						|
			reg);
 | 
						|
	if (ret < 0)
 | 
						|
		return ret;
 | 
						|
 | 
						|
	return phy_read(phydev, MDIO_DEVAD_NONE, AR803x_PHY_DEBUG_DATA_REG);
 | 
						|
}
 | 
						|
 | 
						|
static int ar803x_debug_reg_mask(struct phy_device *phydev, u16 reg,
 | 
						|
				 u16 clear, u16 set)
 | 
						|
{
 | 
						|
	int val;
 | 
						|
 | 
						|
	val = ar803x_debug_reg_read(phydev, reg);
 | 
						|
	if (val < 0)
 | 
						|
		return val;
 | 
						|
 | 
						|
	val &= 0xffff;
 | 
						|
	val &= ~clear;
 | 
						|
	val |= set;
 | 
						|
 | 
						|
	return phy_write(phydev, MDIO_DEVAD_NONE, AR803x_PHY_DEBUG_DATA_REG,
 | 
						|
			 val);
 | 
						|
}
 | 
						|
 | 
						|
static int ar803x_enable_rx_delay(struct phy_device *phydev, bool on)
 | 
						|
{
 | 
						|
	u16 clear = 0, set = 0;
 | 
						|
 | 
						|
	if (on)
 | 
						|
		set = AR803x_RGMII_RX_CLK_DLY;
 | 
						|
	else
 | 
						|
		clear = AR803x_RGMII_RX_CLK_DLY;
 | 
						|
 | 
						|
	return ar803x_debug_reg_mask(phydev, AR803x_DEBUG_REG_0, clear, set);
 | 
						|
}
 | 
						|
 | 
						|
static int ar803x_enable_tx_delay(struct phy_device *phydev, bool on)
 | 
						|
{
 | 
						|
	u16 clear = 0, set = 0;
 | 
						|
 | 
						|
	if (on)
 | 
						|
		set = AR803x_RGMII_TX_CLK_DLY;
 | 
						|
	else
 | 
						|
		clear = AR803x_RGMII_TX_CLK_DLY;
 | 
						|
 | 
						|
	return ar803x_debug_reg_mask(phydev, AR803x_DEBUG_REG_5, clear, set);
 | 
						|
}
 | 
						|
 | 
						|
static int ar8021_config(struct phy_device *phydev)
 | 
						|
{
 | 
						|
	phy_write(phydev, MDIO_DEVAD_NONE, MII_BMCR,
 | 
						|
		  BMCR_ANENABLE | BMCR_ANRESTART);
 | 
						|
 | 
						|
	ar803x_enable_tx_delay(phydev, true);
 | 
						|
 | 
						|
	phydev->supported = phydev->drv->features;
 | 
						|
	return 0;
 | 
						|
}
 | 
						|
 | 
						|
static int ar803x_delay_config(struct phy_device *phydev)
 | 
						|
{
 | 
						|
	int ret;
 | 
						|
 | 
						|
	if (phydev->interface == PHY_INTERFACE_MODE_RGMII_TXID ||
 | 
						|
	    phydev->interface == PHY_INTERFACE_MODE_RGMII_ID)
 | 
						|
		ret = ar803x_enable_tx_delay(phydev, true);
 | 
						|
	else
 | 
						|
		ret = ar803x_enable_tx_delay(phydev, false);
 | 
						|
 | 
						|
	if (phydev->interface == PHY_INTERFACE_MODE_RGMII_RXID ||
 | 
						|
	    phydev->interface == PHY_INTERFACE_MODE_RGMII_ID)
 | 
						|
		ret = ar803x_enable_rx_delay(phydev, true);
 | 
						|
	else
 | 
						|
		ret = ar803x_enable_rx_delay(phydev, false);
 | 
						|
 | 
						|
	return ret;
 | 
						|
}
 | 
						|
 | 
						|
static int ar803x_regs_config(struct phy_device *phydev)
 | 
						|
{
 | 
						|
	struct ar803x_priv *priv = phydev->priv;
 | 
						|
	u16 set = 0, clear = 0;
 | 
						|
	int val;
 | 
						|
	int ret;
 | 
						|
 | 
						|
	/* no configuration available */
 | 
						|
	if (!priv)
 | 
						|
		return 0;
 | 
						|
 | 
						|
	/*
 | 
						|
	 * Only supported on the AR8031, AR8035 has strappings for the PLL mode
 | 
						|
	 * as well as the RGMII voltage.
 | 
						|
	 */
 | 
						|
	if (phydev->drv->uid == AR8031_PHY_ID) {
 | 
						|
		if (priv->flags & AR803x_FLAG_KEEP_PLL_ENABLED)
 | 
						|
			set |= AR803x_PLL_ON;
 | 
						|
		else
 | 
						|
			clear |= AR803x_PLL_ON;
 | 
						|
 | 
						|
		if (priv->flags & AR803x_FLAG_RGMII_1V8)
 | 
						|
			set |= AR803x_RGMII_1V8;
 | 
						|
		else
 | 
						|
			clear |= AR803x_RGMII_1V8;
 | 
						|
 | 
						|
		ret = ar803x_debug_reg_mask(phydev, AR803x_DEBUG_REG_1F, clear,
 | 
						|
					    set);
 | 
						|
		if (ret < 0)
 | 
						|
			return ret;
 | 
						|
	}
 | 
						|
 | 
						|
	/* save the write access if the mask is empty */
 | 
						|
	if (priv->clk_25m_mask) {
 | 
						|
		val = phy_read_mmd(phydev, MDIO_MMD_AN, AR803x_CLK_25M_SEL_REG);
 | 
						|
		if (val < 0)
 | 
						|
			return val;
 | 
						|
		val &= ~priv->clk_25m_mask;
 | 
						|
		val |= priv->clk_25m_reg;
 | 
						|
		ret = phy_write_mmd(phydev, MDIO_MMD_AN,
 | 
						|
				    AR803x_CLK_25M_SEL_REG, val);
 | 
						|
		if (ret < 0)
 | 
						|
			return ret;
 | 
						|
	}
 | 
						|
 | 
						|
	return 0;
 | 
						|
}
 | 
						|
 | 
						|
static int ar803x_of_init(struct phy_device *phydev)
 | 
						|
{
 | 
						|
	struct ar803x_priv *priv;
 | 
						|
	ofnode node, vddio_reg_node;
 | 
						|
	u32 strength, freq, min_uV, max_uV;
 | 
						|
	int sel;
 | 
						|
 | 
						|
	node = phy_get_ofnode(phydev);
 | 
						|
	if (!ofnode_valid(node))
 | 
						|
		return 0;
 | 
						|
 | 
						|
	priv = malloc(sizeof(*priv));
 | 
						|
	if (!priv)
 | 
						|
		return -ENOMEM;
 | 
						|
	memset(priv, 0, sizeof(*priv));
 | 
						|
 | 
						|
	phydev->priv = priv;
 | 
						|
 | 
						|
	debug("%s: found PHY node: %s\n", __func__, ofnode_get_name(node));
 | 
						|
 | 
						|
	if (ofnode_read_bool(node, "qca,keep-pll-enabled"))
 | 
						|
		priv->flags |= AR803x_FLAG_KEEP_PLL_ENABLED;
 | 
						|
 | 
						|
	/*
 | 
						|
	 * We can't use the regulator framework because the regulator is
 | 
						|
	 * a subnode of the PHY. So just read the two properties we are
 | 
						|
	 * interested in.
 | 
						|
	 */
 | 
						|
	vddio_reg_node = ofnode_find_subnode(node, "vddio-regulator");
 | 
						|
	if (ofnode_valid(vddio_reg_node)) {
 | 
						|
		min_uV = ofnode_read_u32_default(vddio_reg_node,
 | 
						|
						 "regulator-min-microvolt", 0);
 | 
						|
		max_uV = ofnode_read_u32_default(vddio_reg_node,
 | 
						|
						 "regulator-max-microvolt", 0);
 | 
						|
 | 
						|
		if (min_uV != max_uV) {
 | 
						|
			free(priv);
 | 
						|
			return -EINVAL;
 | 
						|
		}
 | 
						|
 | 
						|
		switch (min_uV) {
 | 
						|
		case 1500000:
 | 
						|
			break;
 | 
						|
		case 1800000:
 | 
						|
			priv->flags |= AR803x_FLAG_RGMII_1V8;
 | 
						|
			break;
 | 
						|
		default:
 | 
						|
			free(priv);
 | 
						|
			return -EINVAL;
 | 
						|
		}
 | 
						|
	}
 | 
						|
 | 
						|
	/*
 | 
						|
	 * Get the CLK_25M frequency from the device tree. Only XTAL and PLL
 | 
						|
	 * sources are supported right now. There is also the possibilty to use
 | 
						|
	 * the DSP as frequency reference, this is used for synchronous
 | 
						|
	 * ethernet.
 | 
						|
	 */
 | 
						|
	if (!ofnode_read_u32(node, "qca,clk-out-frequency", &freq)) {
 | 
						|
		switch (freq) {
 | 
						|
		case 25000000:
 | 
						|
			sel = AR803x_CLK_25M_25MHZ_XTAL;
 | 
						|
			break;
 | 
						|
		case 50000000:
 | 
						|
			sel = AR803x_CLK_25M_50MHZ_PLL;
 | 
						|
			break;
 | 
						|
		case 62500000:
 | 
						|
			sel = AR803x_CLK_25M_62_5MHZ_PLL;
 | 
						|
			break;
 | 
						|
		case 125000000:
 | 
						|
			sel = AR803x_CLK_25M_125MHZ_PLL;
 | 
						|
			break;
 | 
						|
		default:
 | 
						|
			dev_err(phydev->dev,
 | 
						|
				"invalid qca,clk-out-frequency\n");
 | 
						|
			free(priv);
 | 
						|
			return -EINVAL;
 | 
						|
		}
 | 
						|
 | 
						|
		priv->clk_25m_mask |= AR803x_CLK_25M_MASK;
 | 
						|
		priv->clk_25m_reg |= FIELD_PREP(AR803x_CLK_25M_MASK, sel);
 | 
						|
		/*
 | 
						|
		 * Fixup for the AR8035 which only has two bits. The two
 | 
						|
		 * remaining bits map to the same frequencies.
 | 
						|
		 */
 | 
						|
 | 
						|
		if (phydev->drv->uid == AR8035_PHY_ID) {
 | 
						|
			priv->clk_25m_reg &= AR8035_CLK_25M_MASK;
 | 
						|
			priv->clk_25m_mask &= AR8035_CLK_25M_MASK;
 | 
						|
		}
 | 
						|
	}
 | 
						|
 | 
						|
	if (phydev->drv->uid == AR8031_PHY_ID &&
 | 
						|
	    !ofnode_read_u32(node, "qca,clk-out-strength", &strength)) {
 | 
						|
		switch (strength) {
 | 
						|
		case AR803X_STRENGTH_FULL:
 | 
						|
			sel = AR803x_CLK_25M_DR_FULL;
 | 
						|
			break;
 | 
						|
		case AR803X_STRENGTH_HALF:
 | 
						|
			sel = AR803x_CLK_25M_DR_HALF;
 | 
						|
			break;
 | 
						|
		case AR803X_STRENGTH_QUARTER:
 | 
						|
			sel = AR803x_CLK_25M_DR_QUARTER;
 | 
						|
			break;
 | 
						|
		default:
 | 
						|
			dev_err(phydev->dev,
 | 
						|
				"invalid qca,clk-out-strength\n");
 | 
						|
			free(priv);
 | 
						|
			return -EINVAL;
 | 
						|
		}
 | 
						|
		priv->clk_25m_mask |= AR803x_CLK_25M_DR_MASK;
 | 
						|
		priv->clk_25m_reg |= FIELD_PREP(AR803x_CLK_25M_DR_MASK, sel);
 | 
						|
	}
 | 
						|
 | 
						|
	debug("%s: flags=%x clk_25m_reg=%04x clk_25m_mask=%04x\n", __func__,
 | 
						|
	      priv->flags, priv->clk_25m_reg, priv->clk_25m_mask);
 | 
						|
 | 
						|
	return 0;
 | 
						|
}
 | 
						|
 | 
						|
static int ar803x_config(struct phy_device *phydev)
 | 
						|
{
 | 
						|
	int ret;
 | 
						|
 | 
						|
	ret = ar803x_of_init(phydev);
 | 
						|
	if (ret < 0)
 | 
						|
		return ret;
 | 
						|
 | 
						|
	ret = ar803x_delay_config(phydev);
 | 
						|
	if (ret < 0)
 | 
						|
		return ret;
 | 
						|
 | 
						|
	ret = ar803x_regs_config(phydev);
 | 
						|
	if (ret < 0)
 | 
						|
		return ret;
 | 
						|
 | 
						|
	phydev->supported = phydev->drv->features;
 | 
						|
 | 
						|
	genphy_config_aneg(phydev);
 | 
						|
	genphy_restart_aneg(phydev);
 | 
						|
 | 
						|
	return 0;
 | 
						|
}
 | 
						|
 | 
						|
U_BOOT_PHY_DRIVER(AR8021) = {
 | 
						|
	.name = "AR8021",
 | 
						|
	.uid = AR8021_PHY_ID,
 | 
						|
	.mask = 0xfffffff0,
 | 
						|
	.features = PHY_GBIT_FEATURES,
 | 
						|
	.config = ar8021_config,
 | 
						|
	.startup = genphy_startup,
 | 
						|
	.shutdown = genphy_shutdown,
 | 
						|
};
 | 
						|
 | 
						|
U_BOOT_PHY_DRIVER(AR8031) = {
 | 
						|
	.name = "AR8031/AR8033",
 | 
						|
	.uid = AR8031_PHY_ID,
 | 
						|
	.mask = 0xffffffef,
 | 
						|
	.features = PHY_GBIT_FEATURES,
 | 
						|
	.config = ar803x_config,
 | 
						|
	.startup = genphy_startup,
 | 
						|
	.shutdown = genphy_shutdown,
 | 
						|
};
 | 
						|
 | 
						|
U_BOOT_PHY_DRIVER(AR8035) = {
 | 
						|
	.name = "AR8035",
 | 
						|
	.uid = AR8035_PHY_ID,
 | 
						|
	.mask = 0xffffffef,
 | 
						|
	.features = PHY_GBIT_FEATURES,
 | 
						|
	.config = ar803x_config,
 | 
						|
	.startup = genphy_startup,
 | 
						|
	.shutdown = genphy_shutdown,
 | 
						|
};
 |