root/drivers/net/mdio/mdio-regmap.c
// 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 <maxime.chevallier@bootlin.com>
 */
#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 <maxime.chevallier@bootlin.com>");
MODULE_LICENSE("GPL");