| // SPDX-License-Identifier: (GPL-2.0 OR MIT) | 
 | /* | 
 |  * Core driver for the Ocelot chip family. | 
 |  * | 
 |  * The VSC7511, 7512, 7513, and 7514 can be controlled internally via an | 
 |  * on-chip MIPS processor, or externally via SPI, I2C, PCIe. This core driver is | 
 |  * intended to be the bus-agnostic glue between, for example, the SPI bus and | 
 |  * the child devices. | 
 |  * | 
 |  * Copyright 2021-2022 Innovative Advantage Inc. | 
 |  * | 
 |  * Author: Colin Foster <colin.foster@in-advantage.com> | 
 |  */ | 
 |  | 
 | #include <linux/bits.h> | 
 | #include <linux/device.h> | 
 | #include <linux/export.h> | 
 | #include <linux/iopoll.h> | 
 | #include <linux/ioport.h> | 
 | #include <linux/kernel.h> | 
 | #include <linux/mfd/core.h> | 
 | #include <linux/mfd/ocelot.h> | 
 | #include <linux/module.h> | 
 | #include <linux/regmap.h> | 
 | #include <linux/types.h> | 
 |  | 
 | #include <soc/mscc/ocelot.h> | 
 |  | 
 | #include "ocelot.h" | 
 |  | 
 | #define REG_GCB_SOFT_RST		0x0008 | 
 |  | 
 | #define BIT_SOFT_CHIP_RST		BIT(0) | 
 |  | 
 | #define VSC7512_MIIM0_RES_START		0x7107009c | 
 | #define VSC7512_MIIM1_RES_START		0x710700c0 | 
 | #define VSC7512_MIIM_RES_SIZE		0x00000024 | 
 |  | 
 | #define VSC7512_PHY_RES_START		0x710700f0 | 
 | #define VSC7512_PHY_RES_SIZE		0x00000004 | 
 |  | 
 | #define VSC7512_GPIO_RES_START		0x71070034 | 
 | #define VSC7512_GPIO_RES_SIZE		0x0000006c | 
 |  | 
 | #define VSC7512_SIO_CTRL_RES_START	0x710700f8 | 
 | #define VSC7512_SIO_CTRL_RES_SIZE	0x00000100 | 
 |  | 
 | #define VSC7512_HSIO_RES_START		0x710d0000 | 
 | #define VSC7512_HSIO_RES_SIZE		0x00000128 | 
 |  | 
 | #define VSC7512_ANA_RES_START		0x71880000 | 
 | #define VSC7512_ANA_RES_SIZE		0x00010000 | 
 |  | 
 | #define VSC7512_QS_RES_START		0x71080000 | 
 | #define VSC7512_QS_RES_SIZE		0x00000100 | 
 |  | 
 | #define VSC7512_QSYS_RES_START		0x71800000 | 
 | #define VSC7512_QSYS_RES_SIZE		0x00200000 | 
 |  | 
 | #define VSC7512_REW_RES_START		0x71030000 | 
 | #define VSC7512_REW_RES_SIZE		0x00010000 | 
 |  | 
 | #define VSC7512_SYS_RES_START		0x71010000 | 
 | #define VSC7512_SYS_RES_SIZE		0x00010000 | 
 |  | 
 | #define VSC7512_S0_RES_START		0x71040000 | 
 | #define VSC7512_S1_RES_START		0x71050000 | 
 | #define VSC7512_S2_RES_START		0x71060000 | 
 | #define VCAP_RES_SIZE			0x00000400 | 
 |  | 
 | #define VSC7512_PORT_0_RES_START	0x711e0000 | 
 | #define VSC7512_PORT_1_RES_START	0x711f0000 | 
 | #define VSC7512_PORT_2_RES_START	0x71200000 | 
 | #define VSC7512_PORT_3_RES_START	0x71210000 | 
 | #define VSC7512_PORT_4_RES_START	0x71220000 | 
 | #define VSC7512_PORT_5_RES_START	0x71230000 | 
 | #define VSC7512_PORT_6_RES_START	0x71240000 | 
 | #define VSC7512_PORT_7_RES_START	0x71250000 | 
 | #define VSC7512_PORT_8_RES_START	0x71260000 | 
 | #define VSC7512_PORT_9_RES_START	0x71270000 | 
 | #define VSC7512_PORT_10_RES_START	0x71280000 | 
 | #define VSC7512_PORT_RES_SIZE		0x00010000 | 
 |  | 
 | #define VSC7512_GCB_RST_SLEEP_US	100 | 
 | #define VSC7512_GCB_RST_TIMEOUT_US	100000 | 
 |  | 
 | static int ocelot_gcb_chip_rst_status(struct ocelot_ddata *ddata) | 
 | { | 
 | 	int val, err; | 
 |  | 
 | 	err = regmap_read(ddata->gcb_regmap, REG_GCB_SOFT_RST, &val); | 
 | 	if (err) | 
 | 		return err; | 
 |  | 
 | 	return val; | 
 | } | 
 |  | 
 | int ocelot_chip_reset(struct device *dev) | 
 | { | 
 | 	struct ocelot_ddata *ddata = dev_get_drvdata(dev); | 
 | 	int ret, val; | 
 |  | 
 | 	/* | 
 | 	 * Reset the entire chip here to put it into a completely known state. | 
 | 	 * Other drivers may want to reset their own subsystems. The register | 
 | 	 * self-clears, so one write is all that is needed and wait for it to | 
 | 	 * clear. | 
 | 	 */ | 
 | 	ret = regmap_write(ddata->gcb_regmap, REG_GCB_SOFT_RST, BIT_SOFT_CHIP_RST); | 
 | 	if (ret) | 
 | 		return ret; | 
 |  | 
 | 	return readx_poll_timeout(ocelot_gcb_chip_rst_status, ddata, val, !val, | 
 | 				  VSC7512_GCB_RST_SLEEP_US, VSC7512_GCB_RST_TIMEOUT_US); | 
 | } | 
 | EXPORT_SYMBOL_NS(ocelot_chip_reset, MFD_OCELOT); | 
 |  | 
 | static const struct resource vsc7512_miim0_resources[] = { | 
 | 	DEFINE_RES_REG_NAMED(VSC7512_MIIM0_RES_START, VSC7512_MIIM_RES_SIZE, "gcb_miim0"), | 
 | 	DEFINE_RES_REG_NAMED(VSC7512_PHY_RES_START, VSC7512_PHY_RES_SIZE, "gcb_phy"), | 
 | }; | 
 |  | 
 | static const struct resource vsc7512_miim1_resources[] = { | 
 | 	DEFINE_RES_REG_NAMED(VSC7512_MIIM1_RES_START, VSC7512_MIIM_RES_SIZE, "gcb_miim1"), | 
 | }; | 
 |  | 
 | static const struct resource vsc7512_pinctrl_resources[] = { | 
 | 	DEFINE_RES_REG_NAMED(VSC7512_GPIO_RES_START, VSC7512_GPIO_RES_SIZE, "gcb_gpio"), | 
 | }; | 
 |  | 
 | static const struct resource vsc7512_sgpio_resources[] = { | 
 | 	DEFINE_RES_REG_NAMED(VSC7512_SIO_CTRL_RES_START, VSC7512_SIO_CTRL_RES_SIZE, "gcb_sio"), | 
 | }; | 
 |  | 
 | static const struct resource vsc7512_serdes_resources[] = { | 
 | 	DEFINE_RES_REG_NAMED(VSC7512_HSIO_RES_START, VSC7512_HSIO_RES_SIZE, "hsio"), | 
 | }; | 
 |  | 
 | static const struct resource vsc7512_switch_resources[] = { | 
 | 	DEFINE_RES_REG_NAMED(VSC7512_ANA_RES_START, VSC7512_ANA_RES_SIZE, "ana"), | 
 | 	DEFINE_RES_REG_NAMED(VSC7512_HSIO_RES_START, VSC7512_HSIO_RES_SIZE, "hsio"), | 
 | 	DEFINE_RES_REG_NAMED(VSC7512_QS_RES_START, VSC7512_QS_RES_SIZE, "qs"), | 
 | 	DEFINE_RES_REG_NAMED(VSC7512_QSYS_RES_START, VSC7512_QSYS_RES_SIZE, "qsys"), | 
 | 	DEFINE_RES_REG_NAMED(VSC7512_REW_RES_START, VSC7512_REW_RES_SIZE, "rew"), | 
 | 	DEFINE_RES_REG_NAMED(VSC7512_SYS_RES_START, VSC7512_SYS_RES_SIZE, "sys"), | 
 | 	DEFINE_RES_REG_NAMED(VSC7512_S0_RES_START, VCAP_RES_SIZE, "s0"), | 
 | 	DEFINE_RES_REG_NAMED(VSC7512_S1_RES_START, VCAP_RES_SIZE, "s1"), | 
 | 	DEFINE_RES_REG_NAMED(VSC7512_S2_RES_START, VCAP_RES_SIZE, "s2"), | 
 | 	DEFINE_RES_REG_NAMED(VSC7512_PORT_0_RES_START, VSC7512_PORT_RES_SIZE, "port0"), | 
 | 	DEFINE_RES_REG_NAMED(VSC7512_PORT_1_RES_START, VSC7512_PORT_RES_SIZE, "port1"), | 
 | 	DEFINE_RES_REG_NAMED(VSC7512_PORT_2_RES_START, VSC7512_PORT_RES_SIZE, "port2"), | 
 | 	DEFINE_RES_REG_NAMED(VSC7512_PORT_3_RES_START, VSC7512_PORT_RES_SIZE, "port3"), | 
 | 	DEFINE_RES_REG_NAMED(VSC7512_PORT_4_RES_START, VSC7512_PORT_RES_SIZE, "port4"), | 
 | 	DEFINE_RES_REG_NAMED(VSC7512_PORT_5_RES_START, VSC7512_PORT_RES_SIZE, "port5"), | 
 | 	DEFINE_RES_REG_NAMED(VSC7512_PORT_6_RES_START, VSC7512_PORT_RES_SIZE, "port6"), | 
 | 	DEFINE_RES_REG_NAMED(VSC7512_PORT_7_RES_START, VSC7512_PORT_RES_SIZE, "port7"), | 
 | 	DEFINE_RES_REG_NAMED(VSC7512_PORT_8_RES_START, VSC7512_PORT_RES_SIZE, "port8"), | 
 | 	DEFINE_RES_REG_NAMED(VSC7512_PORT_9_RES_START, VSC7512_PORT_RES_SIZE, "port9"), | 
 | 	DEFINE_RES_REG_NAMED(VSC7512_PORT_10_RES_START, VSC7512_PORT_RES_SIZE, "port10") | 
 | }; | 
 |  | 
 | static const struct mfd_cell vsc7512_devs[] = { | 
 | 	{ | 
 | 		.name = "ocelot-pinctrl", | 
 | 		.of_compatible = "mscc,ocelot-pinctrl", | 
 | 		.num_resources = ARRAY_SIZE(vsc7512_pinctrl_resources), | 
 | 		.resources = vsc7512_pinctrl_resources, | 
 | 	}, { | 
 | 		.name = "ocelot-sgpio", | 
 | 		.of_compatible = "mscc,ocelot-sgpio", | 
 | 		.num_resources = ARRAY_SIZE(vsc7512_sgpio_resources), | 
 | 		.resources = vsc7512_sgpio_resources, | 
 | 	}, { | 
 | 		.name = "ocelot-miim0", | 
 | 		.of_compatible = "mscc,ocelot-miim", | 
 | 		.of_reg = VSC7512_MIIM0_RES_START, | 
 | 		.use_of_reg = true, | 
 | 		.num_resources = ARRAY_SIZE(vsc7512_miim0_resources), | 
 | 		.resources = vsc7512_miim0_resources, | 
 | 	}, { | 
 | 		.name = "ocelot-miim1", | 
 | 		.of_compatible = "mscc,ocelot-miim", | 
 | 		.of_reg = VSC7512_MIIM1_RES_START, | 
 | 		.use_of_reg = true, | 
 | 		.num_resources = ARRAY_SIZE(vsc7512_miim1_resources), | 
 | 		.resources = vsc7512_miim1_resources, | 
 | 	}, { | 
 | 		.name = "ocelot-serdes", | 
 | 		.of_compatible = "mscc,vsc7514-serdes", | 
 | 		.num_resources = ARRAY_SIZE(vsc7512_serdes_resources), | 
 | 		.resources = vsc7512_serdes_resources, | 
 | 	}, { | 
 | 		.name = "ocelot-ext-switch", | 
 | 		.of_compatible = "mscc,vsc7512-switch", | 
 | 		.num_resources = ARRAY_SIZE(vsc7512_switch_resources), | 
 | 		.resources = vsc7512_switch_resources, | 
 | 	}, | 
 | }; | 
 |  | 
 | static void ocelot_core_try_add_regmap(struct device *dev, | 
 | 				       const struct resource *res) | 
 | { | 
 | 	if (dev_get_regmap(dev, res->name)) | 
 | 		return; | 
 |  | 
 | 	ocelot_spi_init_regmap(dev, res); | 
 | } | 
 |  | 
 | static void ocelot_core_try_add_regmaps(struct device *dev, | 
 | 					const struct mfd_cell *cell) | 
 | { | 
 | 	int i; | 
 |  | 
 | 	for (i = 0; i < cell->num_resources; i++) | 
 | 		ocelot_core_try_add_regmap(dev, &cell->resources[i]); | 
 | } | 
 |  | 
 | int ocelot_core_init(struct device *dev) | 
 | { | 
 | 	int i, ndevs; | 
 |  | 
 | 	ndevs = ARRAY_SIZE(vsc7512_devs); | 
 |  | 
 | 	for (i = 0; i < ndevs; i++) | 
 | 		ocelot_core_try_add_regmaps(dev, &vsc7512_devs[i]); | 
 |  | 
 | 	return devm_mfd_add_devices(dev, PLATFORM_DEVID_AUTO, vsc7512_devs, ndevs, NULL, 0, NULL); | 
 | } | 
 | EXPORT_SYMBOL_NS(ocelot_core_init, MFD_OCELOT); | 
 |  | 
 | MODULE_DESCRIPTION("Externally Controlled Ocelot Chip Driver"); | 
 | MODULE_AUTHOR("Colin Foster <colin.foster@in-advantage.com>"); | 
 | MODULE_LICENSE("GPL"); | 
 | MODULE_IMPORT_NS(MFD_OCELOT_SPI); |