diff options
Diffstat (limited to 'drivers/net/mdio/mdio-regmap.c')
| -rw-r--r-- | drivers/net/mdio/mdio-regmap.c | 93 | 
1 files changed, 93 insertions, 0 deletions
diff --git a/drivers/net/mdio/mdio-regmap.c b/drivers/net/mdio/mdio-regmap.c new file mode 100644 index 000000000000..8a742a8d6387 --- /dev/null +++ b/drivers/net/mdio/mdio-regmap.c @@ -0,0 +1,93 @@ +// SPDX-License-Identifier: GPL-2.0-or-later +/* Driver for MMIO-Mapped MDIO devices. Some IPs expose internal PHYs or PCS + * within the MMIO-mapped area + * + * Copyright (C) 2023 Maxime Chevallier <[email protected]> + */ +#include <linux/bitfield.h> +#include <linux/delay.h> +#include <linux/mdio.h> +#include <linux/module.h> +#include <linux/of.h> +#include <linux/of_mdio.h> +#include <linux/phy.h> +#include <linux/platform_device.h> +#include <linux/regmap.h> +#include <linux/mdio/mdio-regmap.h> + +#define DRV_NAME "mdio-regmap" + +struct mdio_regmap_priv { +	struct regmap *regmap; +	u8 valid_addr; +}; + +static int mdio_regmap_read_c22(struct mii_bus *bus, int addr, int regnum) +{ +	struct mdio_regmap_priv *ctx = bus->priv; +	unsigned int val; +	int ret; + +	if (ctx->valid_addr != addr) +		return -ENODEV; + +	ret = regmap_read(ctx->regmap, regnum, &val); +	if (ret < 0) +		return ret; + +	return val; +} + +static int mdio_regmap_write_c22(struct mii_bus *bus, int addr, int regnum, +				 u16 val) +{ +	struct mdio_regmap_priv *ctx = bus->priv; + +	if (ctx->valid_addr != addr) +		return -ENODEV; + +	return regmap_write(ctx->regmap, regnum, val); +} + +struct mii_bus *devm_mdio_regmap_register(struct device *dev, +					  const struct mdio_regmap_config *config) +{ +	struct mdio_regmap_priv *mr; +	struct mii_bus *mii; +	int rc; + +	if (!config->parent) +		return ERR_PTR(-EINVAL); + +	mii = devm_mdiobus_alloc_size(config->parent, sizeof(*mr)); +	if (!mii) +		return ERR_PTR(-ENOMEM); + +	mr = mii->priv; +	mr->regmap = config->regmap; +	mr->valid_addr = config->valid_addr; + +	mii->name = DRV_NAME; +	strscpy(mii->id, config->name, MII_BUS_ID_SIZE); +	mii->parent = config->parent; +	mii->read = mdio_regmap_read_c22; +	mii->write = mdio_regmap_write_c22; + +	if (config->autoscan) +		mii->phy_mask = ~BIT(config->valid_addr); +	else +		mii->phy_mask = ~0; + +	rc = devm_mdiobus_register(dev, mii); +	if (rc) { +		dev_err(config->parent, "Cannot register MDIO bus![%s] (%d)\n", mii->id, rc); +		return ERR_PTR(rc); +	} + +	return mii; +} +EXPORT_SYMBOL_GPL(devm_mdio_regmap_register); + +MODULE_DESCRIPTION("MDIO API over regmap"); +MODULE_AUTHOR("Maxime Chevallier <[email protected]>"); +MODULE_LICENSE("GPL");  |