| /* MOXA ART Ethernet (RTL8201CP) MDIO interface driver | 
 |  * | 
 |  * Copyright (C) 2013 Jonas Jensen <jonas.jensen@gmail.com> | 
 |  * | 
 |  * This file is licensed under the terms of the GNU General Public | 
 |  * License version 2.  This program is licensed "as is" without any | 
 |  * warranty of any kind, whether express or implied. | 
 |  */ | 
 |  | 
 | #include <linux/delay.h> | 
 | #include <linux/kernel.h> | 
 | #include <linux/module.h> | 
 | #include <linux/mutex.h> | 
 | #include <linux/of_address.h> | 
 | #include <linux/of_mdio.h> | 
 | #include <linux/phy.h> | 
 | #include <linux/platform_device.h> | 
 | #include <linux/regulator/consumer.h> | 
 |  | 
 | #define REG_PHY_CTRL            0 | 
 | #define REG_PHY_WRITE_DATA      4 | 
 |  | 
 | /* REG_PHY_CTRL */ | 
 | #define MIIWR                   BIT(27) /* init write sequence (auto cleared)*/ | 
 | #define MIIRD                   BIT(26) | 
 | #define REGAD_MASK              0x3e00000 | 
 | #define PHYAD_MASK              0x1f0000 | 
 | #define MIIRDATA_MASK           0xffff | 
 |  | 
 | /* REG_PHY_WRITE_DATA */ | 
 | #define MIIWDATA_MASK           0xffff | 
 |  | 
 | struct moxart_mdio_data { | 
 | 	void __iomem		*base; | 
 | }; | 
 |  | 
 | static int moxart_mdio_read(struct mii_bus *bus, int mii_id, int regnum) | 
 | { | 
 | 	struct moxart_mdio_data *data = bus->priv; | 
 | 	u32 ctrl = 0; | 
 | 	unsigned int count = 5; | 
 |  | 
 | 	dev_dbg(&bus->dev, "%s\n", __func__); | 
 |  | 
 | 	ctrl |= MIIRD | ((mii_id << 16) & PHYAD_MASK) | | 
 | 		((regnum << 21) & REGAD_MASK); | 
 |  | 
 | 	writel(ctrl, data->base + REG_PHY_CTRL); | 
 |  | 
 | 	do { | 
 | 		ctrl = readl(data->base + REG_PHY_CTRL); | 
 |  | 
 | 		if (!(ctrl & MIIRD)) | 
 | 			return ctrl & MIIRDATA_MASK; | 
 |  | 
 | 		mdelay(10); | 
 | 		count--; | 
 | 	} while (count > 0); | 
 |  | 
 | 	dev_dbg(&bus->dev, "%s timed out\n", __func__); | 
 |  | 
 | 	return -ETIMEDOUT; | 
 | } | 
 |  | 
 | static int moxart_mdio_write(struct mii_bus *bus, int mii_id, | 
 | 			     int regnum, u16 value) | 
 | { | 
 | 	struct moxart_mdio_data *data = bus->priv; | 
 | 	u32 ctrl = 0; | 
 | 	unsigned int count = 5; | 
 |  | 
 | 	dev_dbg(&bus->dev, "%s\n", __func__); | 
 |  | 
 | 	ctrl |= MIIWR | ((mii_id << 16) & PHYAD_MASK) | | 
 | 		((regnum << 21) & REGAD_MASK); | 
 |  | 
 | 	value &= MIIWDATA_MASK; | 
 |  | 
 | 	writel(value, data->base + REG_PHY_WRITE_DATA); | 
 | 	writel(ctrl, data->base + REG_PHY_CTRL); | 
 |  | 
 | 	do { | 
 | 		ctrl = readl(data->base + REG_PHY_CTRL); | 
 |  | 
 | 		if (!(ctrl & MIIWR)) | 
 | 			return 0; | 
 |  | 
 | 		mdelay(10); | 
 | 		count--; | 
 | 	} while (count > 0); | 
 |  | 
 | 	dev_dbg(&bus->dev, "%s timed out\n", __func__); | 
 |  | 
 | 	return -ETIMEDOUT; | 
 | } | 
 |  | 
 | static int moxart_mdio_reset(struct mii_bus *bus) | 
 | { | 
 | 	int data, i; | 
 |  | 
 | 	for (i = 0; i < PHY_MAX_ADDR; i++) { | 
 | 		data = moxart_mdio_read(bus, i, MII_BMCR); | 
 | 		if (data < 0) | 
 | 			continue; | 
 |  | 
 | 		data |= BMCR_RESET; | 
 | 		if (moxart_mdio_write(bus, i, MII_BMCR, data) < 0) | 
 | 			continue; | 
 | 	} | 
 |  | 
 | 	return 0; | 
 | } | 
 |  | 
 | static int moxart_mdio_probe(struct platform_device *pdev) | 
 | { | 
 | 	struct device_node *np = pdev->dev.of_node; | 
 | 	struct mii_bus *bus; | 
 | 	struct moxart_mdio_data *data; | 
 | 	struct resource *res; | 
 | 	int ret, i; | 
 |  | 
 | 	bus = mdiobus_alloc_size(sizeof(*data)); | 
 | 	if (!bus) | 
 | 		return -ENOMEM; | 
 |  | 
 | 	bus->name = "MOXA ART Ethernet MII"; | 
 | 	bus->read = &moxart_mdio_read; | 
 | 	bus->write = &moxart_mdio_write; | 
 | 	bus->reset = &moxart_mdio_reset; | 
 | 	snprintf(bus->id, MII_BUS_ID_SIZE, "%s-%d-mii", pdev->name, pdev->id); | 
 | 	bus->parent = &pdev->dev; | 
 |  | 
 | 	/* Setting PHY_IGNORE_INTERRUPT here even if it has no effect, | 
 | 	 * of_mdiobus_register() sets these PHY_POLL. | 
 | 	 * Ideally, the interrupt from MAC controller could be used to | 
 | 	 * detect link state changes, not polling, i.e. if there was | 
 | 	 * a way phy_driver could set PHY_HAS_INTERRUPT but have that | 
 | 	 * interrupt handled in ethernet drivercode. | 
 | 	 */ | 
 | 	for (i = 0; i < PHY_MAX_ADDR; i++) | 
 | 		bus->irq[i] = PHY_IGNORE_INTERRUPT; | 
 |  | 
 | 	data = bus->priv; | 
 | 	res = platform_get_resource(pdev, IORESOURCE_MEM, 0); | 
 | 	data->base = devm_ioremap_resource(&pdev->dev, res); | 
 | 	if (IS_ERR(data->base)) { | 
 | 		ret = PTR_ERR(data->base); | 
 | 		goto err_out_free_mdiobus; | 
 | 	} | 
 |  | 
 | 	ret = of_mdiobus_register(bus, np); | 
 | 	if (ret < 0) | 
 | 		goto err_out_free_mdiobus; | 
 |  | 
 | 	platform_set_drvdata(pdev, bus); | 
 |  | 
 | 	return 0; | 
 |  | 
 | err_out_free_mdiobus: | 
 | 	mdiobus_free(bus); | 
 | 	return ret; | 
 | } | 
 |  | 
 | static int moxart_mdio_remove(struct platform_device *pdev) | 
 | { | 
 | 	struct mii_bus *bus = platform_get_drvdata(pdev); | 
 |  | 
 | 	mdiobus_unregister(bus); | 
 | 	mdiobus_free(bus); | 
 |  | 
 | 	return 0; | 
 | } | 
 |  | 
 | static const struct of_device_id moxart_mdio_dt_ids[] = { | 
 | 	{ .compatible = "moxa,moxart-mdio" }, | 
 | 	{ } | 
 | }; | 
 | MODULE_DEVICE_TABLE(of, moxart_mdio_dt_ids); | 
 |  | 
 | static struct platform_driver moxart_mdio_driver = { | 
 | 	.probe = moxart_mdio_probe, | 
 | 	.remove = moxart_mdio_remove, | 
 | 	.driver = { | 
 | 		.name = "moxart-mdio", | 
 | 		.of_match_table = moxart_mdio_dt_ids, | 
 | 	}, | 
 | }; | 
 |  | 
 | module_platform_driver(moxart_mdio_driver); | 
 |  | 
 | MODULE_DESCRIPTION("MOXA ART MDIO interface driver"); | 
 | MODULE_AUTHOR("Jonas Jensen <jonas.jensen@gmail.com>"); | 
 | MODULE_LICENSE("GPL"); |