xref: /OK3568_Linux_fs/kernel/drivers/net/ethernet/samsung/sxgbe/sxgbe_mdio.c (revision 4882a59341e53eb6f0b4789bf948001014eff981)
1*4882a593Smuzhiyun // SPDX-License-Identifier: GPL-2.0-only
2*4882a593Smuzhiyun /* 10G controller driver for Samsung SoCs
3*4882a593Smuzhiyun  *
4*4882a593Smuzhiyun  * Copyright (C) 2013 Samsung Electronics Co., Ltd.
5*4882a593Smuzhiyun  *		http://www.samsung.com
6*4882a593Smuzhiyun  *
7*4882a593Smuzhiyun  * Author: Siva Reddy Kallam <siva.kallam@samsung.com>
8*4882a593Smuzhiyun  */
9*4882a593Smuzhiyun 
10*4882a593Smuzhiyun #define pr_fmt(fmt) KBUILD_MODNAME ": " fmt
11*4882a593Smuzhiyun 
12*4882a593Smuzhiyun #include <linux/io.h>
13*4882a593Smuzhiyun #include <linux/mii.h>
14*4882a593Smuzhiyun #include <linux/netdevice.h>
15*4882a593Smuzhiyun #include <linux/platform_device.h>
16*4882a593Smuzhiyun #include <linux/phy.h>
17*4882a593Smuzhiyun #include <linux/slab.h>
18*4882a593Smuzhiyun #include <linux/sxgbe_platform.h>
19*4882a593Smuzhiyun 
20*4882a593Smuzhiyun #include "sxgbe_common.h"
21*4882a593Smuzhiyun #include "sxgbe_reg.h"
22*4882a593Smuzhiyun 
23*4882a593Smuzhiyun #define SXGBE_SMA_WRITE_CMD	0x01 /* write command */
24*4882a593Smuzhiyun #define SXGBE_SMA_PREAD_CMD	0x02 /* post read  increament address */
25*4882a593Smuzhiyun #define SXGBE_SMA_READ_CMD	0x03 /* read command */
26*4882a593Smuzhiyun #define SXGBE_SMA_SKIP_ADDRFRM	0x00040000 /* skip the address frame */
27*4882a593Smuzhiyun #define SXGBE_MII_BUSY		0x00400000 /* mii busy */
28*4882a593Smuzhiyun 
sxgbe_mdio_busy_wait(void __iomem * ioaddr,unsigned int mii_data)29*4882a593Smuzhiyun static int sxgbe_mdio_busy_wait(void __iomem *ioaddr, unsigned int mii_data)
30*4882a593Smuzhiyun {
31*4882a593Smuzhiyun 	unsigned long fin_time = jiffies + 3 * HZ; /* 3 seconds */
32*4882a593Smuzhiyun 
33*4882a593Smuzhiyun 	while (!time_after(jiffies, fin_time)) {
34*4882a593Smuzhiyun 		if (!(readl(ioaddr + mii_data) & SXGBE_MII_BUSY))
35*4882a593Smuzhiyun 			return 0;
36*4882a593Smuzhiyun 		cpu_relax();
37*4882a593Smuzhiyun 	}
38*4882a593Smuzhiyun 
39*4882a593Smuzhiyun 	return -EBUSY;
40*4882a593Smuzhiyun }
41*4882a593Smuzhiyun 
sxgbe_mdio_ctrl_data(struct sxgbe_priv_data * sp,u32 cmd,u16 phydata)42*4882a593Smuzhiyun static void sxgbe_mdio_ctrl_data(struct sxgbe_priv_data *sp, u32 cmd,
43*4882a593Smuzhiyun 				 u16 phydata)
44*4882a593Smuzhiyun {
45*4882a593Smuzhiyun 	u32 reg = phydata;
46*4882a593Smuzhiyun 
47*4882a593Smuzhiyun 	reg |= (cmd << 16) | SXGBE_SMA_SKIP_ADDRFRM |
48*4882a593Smuzhiyun 	       ((sp->clk_csr & 0x7) << 19) | SXGBE_MII_BUSY;
49*4882a593Smuzhiyun 	writel(reg, sp->ioaddr + sp->hw->mii.data);
50*4882a593Smuzhiyun }
51*4882a593Smuzhiyun 
sxgbe_mdio_c45(struct sxgbe_priv_data * sp,u32 cmd,int phyaddr,int phyreg,u16 phydata)52*4882a593Smuzhiyun static void sxgbe_mdio_c45(struct sxgbe_priv_data *sp, u32 cmd, int phyaddr,
53*4882a593Smuzhiyun 			   int phyreg, u16 phydata)
54*4882a593Smuzhiyun {
55*4882a593Smuzhiyun 	u32 reg;
56*4882a593Smuzhiyun 
57*4882a593Smuzhiyun 	/* set mdio address register */
58*4882a593Smuzhiyun 	reg = ((phyreg >> 16) & 0x1f) << 21;
59*4882a593Smuzhiyun 	reg |= (phyaddr << 16) | (phyreg & 0xffff);
60*4882a593Smuzhiyun 	writel(reg, sp->ioaddr + sp->hw->mii.addr);
61*4882a593Smuzhiyun 
62*4882a593Smuzhiyun 	sxgbe_mdio_ctrl_data(sp, cmd, phydata);
63*4882a593Smuzhiyun }
64*4882a593Smuzhiyun 
sxgbe_mdio_c22(struct sxgbe_priv_data * sp,u32 cmd,int phyaddr,int phyreg,u16 phydata)65*4882a593Smuzhiyun static void sxgbe_mdio_c22(struct sxgbe_priv_data *sp, u32 cmd, int phyaddr,
66*4882a593Smuzhiyun 			   int phyreg, u16 phydata)
67*4882a593Smuzhiyun {
68*4882a593Smuzhiyun 	u32 reg;
69*4882a593Smuzhiyun 
70*4882a593Smuzhiyun 	writel(1 << phyaddr, sp->ioaddr + SXGBE_MDIO_CLAUSE22_PORT_REG);
71*4882a593Smuzhiyun 
72*4882a593Smuzhiyun 	/* set mdio address register */
73*4882a593Smuzhiyun 	reg = (phyaddr << 16) | (phyreg & 0x1f);
74*4882a593Smuzhiyun 	writel(reg, sp->ioaddr + sp->hw->mii.addr);
75*4882a593Smuzhiyun 
76*4882a593Smuzhiyun 	sxgbe_mdio_ctrl_data(sp, cmd, phydata);
77*4882a593Smuzhiyun }
78*4882a593Smuzhiyun 
sxgbe_mdio_access(struct sxgbe_priv_data * sp,u32 cmd,int phyaddr,int phyreg,u16 phydata)79*4882a593Smuzhiyun static int sxgbe_mdio_access(struct sxgbe_priv_data *sp, u32 cmd, int phyaddr,
80*4882a593Smuzhiyun 			     int phyreg, u16 phydata)
81*4882a593Smuzhiyun {
82*4882a593Smuzhiyun 	const struct mii_regs *mii = &sp->hw->mii;
83*4882a593Smuzhiyun 	int rc;
84*4882a593Smuzhiyun 
85*4882a593Smuzhiyun 	rc = sxgbe_mdio_busy_wait(sp->ioaddr, mii->data);
86*4882a593Smuzhiyun 	if (rc < 0)
87*4882a593Smuzhiyun 		return rc;
88*4882a593Smuzhiyun 
89*4882a593Smuzhiyun 	if (phyreg & MII_ADDR_C45) {
90*4882a593Smuzhiyun 		sxgbe_mdio_c45(sp, cmd, phyaddr, phyreg, phydata);
91*4882a593Smuzhiyun 	} else {
92*4882a593Smuzhiyun 		 /* Ports 0-3 only support C22. */
93*4882a593Smuzhiyun 		if (phyaddr >= 4)
94*4882a593Smuzhiyun 			return -ENODEV;
95*4882a593Smuzhiyun 
96*4882a593Smuzhiyun 		sxgbe_mdio_c22(sp, cmd, phyaddr, phyreg, phydata);
97*4882a593Smuzhiyun 	}
98*4882a593Smuzhiyun 
99*4882a593Smuzhiyun 	return sxgbe_mdio_busy_wait(sp->ioaddr, mii->data);
100*4882a593Smuzhiyun }
101*4882a593Smuzhiyun 
102*4882a593Smuzhiyun /**
103*4882a593Smuzhiyun  * sxgbe_mdio_read
104*4882a593Smuzhiyun  * @bus: points to the mii_bus structure
105*4882a593Smuzhiyun  * @phyaddr: address of phy port
106*4882a593Smuzhiyun  * @phyreg: address of register with in phy register
107*4882a593Smuzhiyun  * Description: this function used for C45 and C22 MDIO Read
108*4882a593Smuzhiyun  */
sxgbe_mdio_read(struct mii_bus * bus,int phyaddr,int phyreg)109*4882a593Smuzhiyun static int sxgbe_mdio_read(struct mii_bus *bus, int phyaddr, int phyreg)
110*4882a593Smuzhiyun {
111*4882a593Smuzhiyun 	struct net_device *ndev = bus->priv;
112*4882a593Smuzhiyun 	struct sxgbe_priv_data *priv = netdev_priv(ndev);
113*4882a593Smuzhiyun 	int rc;
114*4882a593Smuzhiyun 
115*4882a593Smuzhiyun 	rc = sxgbe_mdio_access(priv, SXGBE_SMA_READ_CMD, phyaddr, phyreg, 0);
116*4882a593Smuzhiyun 	if (rc < 0)
117*4882a593Smuzhiyun 		return rc;
118*4882a593Smuzhiyun 
119*4882a593Smuzhiyun 	return readl(priv->ioaddr + priv->hw->mii.data) & 0xffff;
120*4882a593Smuzhiyun }
121*4882a593Smuzhiyun 
122*4882a593Smuzhiyun /**
123*4882a593Smuzhiyun  * sxgbe_mdio_write
124*4882a593Smuzhiyun  * @bus: points to the mii_bus structure
125*4882a593Smuzhiyun  * @phyaddr: address of phy port
126*4882a593Smuzhiyun  * @phyreg: address of phy registers
127*4882a593Smuzhiyun  * @phydata: data to be written into phy register
128*4882a593Smuzhiyun  * Description: this function is used for C45 and C22 MDIO write
129*4882a593Smuzhiyun  */
sxgbe_mdio_write(struct mii_bus * bus,int phyaddr,int phyreg,u16 phydata)130*4882a593Smuzhiyun static int sxgbe_mdio_write(struct mii_bus *bus, int phyaddr, int phyreg,
131*4882a593Smuzhiyun 			     u16 phydata)
132*4882a593Smuzhiyun {
133*4882a593Smuzhiyun 	struct net_device *ndev = bus->priv;
134*4882a593Smuzhiyun 	struct sxgbe_priv_data *priv = netdev_priv(ndev);
135*4882a593Smuzhiyun 
136*4882a593Smuzhiyun 	return sxgbe_mdio_access(priv, SXGBE_SMA_WRITE_CMD, phyaddr, phyreg,
137*4882a593Smuzhiyun 				 phydata);
138*4882a593Smuzhiyun }
139*4882a593Smuzhiyun 
sxgbe_mdio_register(struct net_device * ndev)140*4882a593Smuzhiyun int sxgbe_mdio_register(struct net_device *ndev)
141*4882a593Smuzhiyun {
142*4882a593Smuzhiyun 	struct mii_bus *mdio_bus;
143*4882a593Smuzhiyun 	struct sxgbe_priv_data *priv = netdev_priv(ndev);
144*4882a593Smuzhiyun 	struct sxgbe_mdio_bus_data *mdio_data = priv->plat->mdio_bus_data;
145*4882a593Smuzhiyun 	int err, phy_addr;
146*4882a593Smuzhiyun 	int *irqlist;
147*4882a593Smuzhiyun 	bool phy_found = false;
148*4882a593Smuzhiyun 	bool act;
149*4882a593Smuzhiyun 
150*4882a593Smuzhiyun 	/* allocate the new mdio bus */
151*4882a593Smuzhiyun 	mdio_bus = mdiobus_alloc();
152*4882a593Smuzhiyun 	if (!mdio_bus) {
153*4882a593Smuzhiyun 		netdev_err(ndev, "%s: mii bus allocation failed\n", __func__);
154*4882a593Smuzhiyun 		return -ENOMEM;
155*4882a593Smuzhiyun 	}
156*4882a593Smuzhiyun 
157*4882a593Smuzhiyun 	if (mdio_data->irqs)
158*4882a593Smuzhiyun 		irqlist = mdio_data->irqs;
159*4882a593Smuzhiyun 	else
160*4882a593Smuzhiyun 		irqlist = priv->mii_irq;
161*4882a593Smuzhiyun 
162*4882a593Smuzhiyun 	/* assign mii bus fields */
163*4882a593Smuzhiyun 	mdio_bus->name = "sxgbe";
164*4882a593Smuzhiyun 	mdio_bus->read = &sxgbe_mdio_read;
165*4882a593Smuzhiyun 	mdio_bus->write = &sxgbe_mdio_write;
166*4882a593Smuzhiyun 	snprintf(mdio_bus->id, MII_BUS_ID_SIZE, "%s-%x",
167*4882a593Smuzhiyun 		 mdio_bus->name, priv->plat->bus_id);
168*4882a593Smuzhiyun 	mdio_bus->priv = ndev;
169*4882a593Smuzhiyun 	mdio_bus->phy_mask = mdio_data->phy_mask;
170*4882a593Smuzhiyun 	mdio_bus->parent = priv->device;
171*4882a593Smuzhiyun 
172*4882a593Smuzhiyun 	/* register with kernel subsystem */
173*4882a593Smuzhiyun 	err = mdiobus_register(mdio_bus);
174*4882a593Smuzhiyun 	if (err != 0) {
175*4882a593Smuzhiyun 		netdev_err(ndev, "mdiobus register failed\n");
176*4882a593Smuzhiyun 		goto mdiobus_err;
177*4882a593Smuzhiyun 	}
178*4882a593Smuzhiyun 
179*4882a593Smuzhiyun 	for (phy_addr = 0; phy_addr < PHY_MAX_ADDR; phy_addr++) {
180*4882a593Smuzhiyun 		struct phy_device *phy = mdiobus_get_phy(mdio_bus, phy_addr);
181*4882a593Smuzhiyun 
182*4882a593Smuzhiyun 		if (phy) {
183*4882a593Smuzhiyun 			char irq_num[4];
184*4882a593Smuzhiyun 			char *irq_str;
185*4882a593Smuzhiyun 			/* If an IRQ was provided to be assigned after
186*4882a593Smuzhiyun 			 * the bus probe, do it here.
187*4882a593Smuzhiyun 			 */
188*4882a593Smuzhiyun 			if ((mdio_data->irqs == NULL) &&
189*4882a593Smuzhiyun 			    (mdio_data->probed_phy_irq > 0)) {
190*4882a593Smuzhiyun 				irqlist[phy_addr] = mdio_data->probed_phy_irq;
191*4882a593Smuzhiyun 				phy->irq = mdio_data->probed_phy_irq;
192*4882a593Smuzhiyun 			}
193*4882a593Smuzhiyun 
194*4882a593Smuzhiyun 			/* If we're  going to bind the MAC to this PHY bus,
195*4882a593Smuzhiyun 			 * and no PHY number was provided to the MAC,
196*4882a593Smuzhiyun 			 * use the one probed here.
197*4882a593Smuzhiyun 			 */
198*4882a593Smuzhiyun 			if (priv->plat->phy_addr == -1)
199*4882a593Smuzhiyun 				priv->plat->phy_addr = phy_addr;
200*4882a593Smuzhiyun 
201*4882a593Smuzhiyun 			act = (priv->plat->phy_addr == phy_addr);
202*4882a593Smuzhiyun 			switch (phy->irq) {
203*4882a593Smuzhiyun 			case PHY_POLL:
204*4882a593Smuzhiyun 				irq_str = "POLL";
205*4882a593Smuzhiyun 				break;
206*4882a593Smuzhiyun 			case PHY_IGNORE_INTERRUPT:
207*4882a593Smuzhiyun 				irq_str = "IGNORE";
208*4882a593Smuzhiyun 				break;
209*4882a593Smuzhiyun 			default:
210*4882a593Smuzhiyun 				sprintf(irq_num, "%d", phy->irq);
211*4882a593Smuzhiyun 				irq_str = irq_num;
212*4882a593Smuzhiyun 				break;
213*4882a593Smuzhiyun 			}
214*4882a593Smuzhiyun 			netdev_info(ndev, "PHY ID %08x at %d IRQ %s (%s)%s\n",
215*4882a593Smuzhiyun 				    phy->phy_id, phy_addr, irq_str,
216*4882a593Smuzhiyun 				    phydev_name(phy), act ? " active" : "");
217*4882a593Smuzhiyun 			phy_found = true;
218*4882a593Smuzhiyun 		}
219*4882a593Smuzhiyun 	}
220*4882a593Smuzhiyun 
221*4882a593Smuzhiyun 	if (!phy_found) {
222*4882a593Smuzhiyun 		netdev_err(ndev, "PHY not found\n");
223*4882a593Smuzhiyun 		goto phyfound_err;
224*4882a593Smuzhiyun 	}
225*4882a593Smuzhiyun 
226*4882a593Smuzhiyun 	priv->mii = mdio_bus;
227*4882a593Smuzhiyun 
228*4882a593Smuzhiyun 	return 0;
229*4882a593Smuzhiyun 
230*4882a593Smuzhiyun phyfound_err:
231*4882a593Smuzhiyun 	err = -ENODEV;
232*4882a593Smuzhiyun 	mdiobus_unregister(mdio_bus);
233*4882a593Smuzhiyun mdiobus_err:
234*4882a593Smuzhiyun 	mdiobus_free(mdio_bus);
235*4882a593Smuzhiyun 	return err;
236*4882a593Smuzhiyun }
237*4882a593Smuzhiyun 
sxgbe_mdio_unregister(struct net_device * ndev)238*4882a593Smuzhiyun int sxgbe_mdio_unregister(struct net_device *ndev)
239*4882a593Smuzhiyun {
240*4882a593Smuzhiyun 	struct sxgbe_priv_data *priv = netdev_priv(ndev);
241*4882a593Smuzhiyun 
242*4882a593Smuzhiyun 	if (!priv->mii)
243*4882a593Smuzhiyun 		return 0;
244*4882a593Smuzhiyun 
245*4882a593Smuzhiyun 	mdiobus_unregister(priv->mii);
246*4882a593Smuzhiyun 	priv->mii->priv = NULL;
247*4882a593Smuzhiyun 	mdiobus_free(priv->mii);
248*4882a593Smuzhiyun 	priv->mii = NULL;
249*4882a593Smuzhiyun 
250*4882a593Smuzhiyun 	return 0;
251*4882a593Smuzhiyun }
252