root/drivers/mfd/ocelot-core.c
// 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");