Hi Andrew,

On 25/03/19 7:35 PM, Andrew F. Davis wrote:
> On 3/25/19 3:08 AM, Kishon Vijay Abraham I wrote:
>> Add a new SERDES driver for TI's AM654x SoC which configures
>> the SERDES only for PCIe. Support fo USB3 will be added later.
>>
>> SERDES in am654x has three input clocks (left input, externel reference
>> clock and right input) and two output clocks (left output and right
>> output) in addition to a PLL mux clock which the SERDES uses for Clock
>> Multiplier Unit (CMU refclock).
>>
>> The PLL mux clock can select from one of the three input clocks.
>> The right output can select between left input and external reference
>> clock while the left output can select between the right input and
>> external reference clock.
>>
>> The driver has support to select PLL mux and left/right output mux as
>> specified in device tree.
>>
>> [[email protected]: Fix boot lockup caused by accessing a structure member
>> (hw->init) allocated in stack of probe() and accessed in get_parent]
>> [[email protected]: Fix "Failed to find the parent" warnings]
>> Signed-off-by: Roger Quadros <[email protected]>
>> Signed-off-by: Kishon Vijay Abraham I <[email protected]>
>> ---
>>  drivers/phy/ti/Kconfig            |  12 +
>>  drivers/phy/ti/Makefile           |   1 +
>>  drivers/phy/ti/phy-am654-serdes.c | 539 ++++++++++++++++++++++++++++++
>>  3 files changed, 552 insertions(+)
>>  create mode 100644 drivers/phy/ti/phy-am654-serdes.c
>>
>> diff --git a/drivers/phy/ti/Kconfig b/drivers/phy/ti/Kconfig
>> index 7cdc35f8c862..6931c87235b9 100644
>> --- a/drivers/phy/ti/Kconfig
>> +++ b/drivers/phy/ti/Kconfig
>> @@ -20,6 +20,18 @@ config PHY_DM816X_USB
>>      help
>>        Enable this for dm816x USB to work.
>>  
>> +config PHY_AM654_SERDES
>> +    tristate "TI AM654 SERDES support"
>> +    depends on OF && ARCH_K3 || COMPILE_TEST
>> +    depends on COMMON_CLK
>> +    select GENERIC_PHY
>> +    select MULTIPLEXER
>> +    select REGMAP_MMIO
>> +    select MUX_MMIO
>> +    help
>> +      This option enables support for TI AM654 SerDes PHY used for
>> +      PCIe.
>> +
>>  config OMAP_CONTROL_PHY
>>      tristate "OMAP CONTROL PHY Driver"
>>      depends on ARCH_OMAP2PLUS || COMPILE_TEST
>> diff --git a/drivers/phy/ti/Makefile b/drivers/phy/ti/Makefile
>> index 368de8e1548d..601bbd88f35e 100644
>> --- a/drivers/phy/ti/Makefile
>> +++ b/drivers/phy/ti/Makefile
>> @@ -6,5 +6,6 @@ obj-$(CONFIG_OMAP_USB2)                      += 
>> phy-omap-usb2.o
>>  obj-$(CONFIG_TI_PIPE3)                      += phy-ti-pipe3.o
>>  obj-$(CONFIG_PHY_TUSB1210)          += phy-tusb1210.o
>>  obj-$(CONFIG_TWL4030_USB)           += phy-twl4030-usb.o
>> +obj-$(CONFIG_PHY_AM654_SERDES)              += phy-am654-serdes.o
>>  obj-$(CONFIG_PHY_TI_GMII_SEL)               += phy-gmii-sel.o
>>  obj-$(CONFIG_PHY_TI_KEYSTONE_SERDES)        += phy-keystone-serdes.o
>> diff --git a/drivers/phy/ti/phy-am654-serdes.c 
>> b/drivers/phy/ti/phy-am654-serdes.c
>> new file mode 100644
>> index 000000000000..dfbd2d48503d
>> --- /dev/null
>> +++ b/drivers/phy/ti/phy-am654-serdes.c
>> @@ -0,0 +1,539 @@
>> +// SPDX-License-Identifier: GPL-2.0
>> +/**
>> + * PCIe SERDES driver for AM654x SoC
>> + *
>> + * Copyright (C) 2018 Texas Instruments
> 
> Copyright (C) 2018 Texas Instruments Incorporated - http://www.ti.com/
> 
>> + * Author: Kishon Vijay Abraham I <[email protected]>
>> + */
>> +
>> +#include <dt-bindings/phy/phy.h>
>> +#include <linux/clk.h>
>> +#include <linux/clk-provider.h>
>> +#include <linux/delay.h>
>> +#include <linux/io.h>
>> +#include <linux/module.h>
>> +#include <linux/mux/consumer.h>
>> +#include <linux/of_address.h>
>> +#include <linux/of_device.h>
>> +#include <linux/phy/phy.h>
>> +#include <linux/platform_device.h>
>> +#include <linux/pm_runtime.h>
>> +#include <linux/regmap.h>
>> +#include <linux/slab.h>
>> +#include <linux/mfd/syscon.h>
> 
> alphabetize last item + do we need all of these?

Looks like no. I'll fix this up.
> 
>> +
>> +#define CMU_R07C            0x7c
>> +#define CMU_MASTER_CDN_O    BIT(24)
>> +
>> +#define COMLANE_R138                0xb38
>> +#define CONFIG_VERSION_REG_MASK     GENMASK(23, 16)
>> +#define CONFIG_VERSION_REG_SHIFT 16
>> +#define VERSION                     0x70
>> +
>> +#define COMLANE_R190                0xb90
>> +#define L1_MASTER_CDN_O             BIT(9)
>> +
>> +#define COMLANE_R194                0xb94
>> +#define CMU_OK_I_0          BIT(19)
>> +
>> +#define SERDES_CTRL         0x1fd0
>> +#define POR_EN                      BIT(29)
>> +
>> +#define WIZ_LANEXCTL_STS    0x1fe0
>> +#define TX0_ENABLE_OVL              BIT(31)
>> +#define TX0_ENABLE_MASK             GENMASK(30, 29)
>> +#define TX0_ENABLE_SHIFT    29
>> +#define TX0_DISABLE_STATE   0x0
>> +#define TX0_SLEEP_STATE             0x1
>> +#define TX0_SNOOZE_STATE    0x2
>> +#define TX0_ENABLE_STATE    0x3
>> +#define RX0_ENABLE_OVL              BIT(15)
>> +#define RX0_ENABLE_MASK             GENMASK(14, 13)
>> +#define RX0_ENABLE_SHIFT    13
>> +#define RX0_DISABLE_STATE   0x0
>> +#define RX0_SLEEP_STATE             0x1
>> +#define RX0_SNOOZE_STATE    0x2
>> +#define RX0_ENABLE_STATE    0x3
>> +
>> +#define WIZ_PLL_CTRL                0x1ff4
>> +#define PLL_ENABLE_OVL              BIT(31)
>> +#define PLL_ENABLE_MASK             GENMASK(30, 29)
>> +#define PLL_ENABLE_SHIFT    29
>> +#define PLL_DISABLE_STATE   0x0
>> +#define PLL_SLEEP_STATE             0x1
>> +#define PLL_SNOOZE_STATE    0x2
>> +#define PLL_ENABLE_STATE    0x3
>> +#define PLL_OK                      BIT(28)
>> +
>> +#define PLL_LOCK_TIME               100000  /* in microseconds */
>> +#define SLEEP_TIME          100     /* in microseconds */
>> +
>> +#define LANE_USB3           0x0
>> +#define LANE_PCIE0_LANE0    0x1
>> +
>> +#define LANE_PCIE1_LANE0    0x0
>> +#define LANE_PCIE0_LANE1    0x1
>> +
>> +#define SERDES_NUM_CLOCKS   3
>> +
>> +struct serdes_am654_clk_mux {
>> +    struct clk_hw   hw;
>> +    struct regmap   *regmap;
>> +    unsigned int    reg;
>> +    int             *table;
>> +    u32             mask;
>> +    u8              shift;
>> +    struct clk_init_data clk_data;
>> +};
>> +
>> +#define to_serdes_am654_clk_mux(_hw)        \
>> +            container_of(_hw, struct serdes_am654_clk_mux, hw)
>> +
>> +static struct regmap_config serdes_am654_regmap_config = {
>> +    .reg_bits = 32,
>> +    .val_bits = 32,
>> +    .reg_stride = 4,
>> +    .fast_io = true,
>> +};
>> +
>> +struct serdes_am654 {
>> +    struct regmap           *regmap;
>> +    struct device           *dev;
>> +    struct mux_control      *control;
>> +    bool                    busy;
>> +    u32                     type;
>> +    struct device_node      *of_node;
>> +    struct clk_onecell_data clk_data;
>> +    struct clk              *clks[SERDES_NUM_CLOCKS];
>> +};
>> +
>> +static int serdes_am654_enable_pll(struct serdes_am654 *phy)
>> +{
>> +    u32 mask = PLL_ENABLE_OVL | PLL_ENABLE_MASK;
>> +    u32 val = PLL_ENABLE_OVL | (PLL_ENABLE_STATE << PLL_ENABLE_SHIFT);
>> +
>> +    regmap_update_bits(phy->regmap, WIZ_PLL_CTRL, mask, val);
> 
> 
> This driver looks like a good candidate for regmap_field

sounds reasonable. Other than in serdes_am654_clk_mux_set_parent below, we
could use regmap_field everywhere else. That will be fixed in the next patch.
> 
>> +
>> +    return regmap_read_poll_timeout(phy->regmap, WIZ_PLL_CTRL, val,
>> +                                    val & PLL_OK, 1000, PLL_LOCK_TIME);
>> +}
>> +
>> +static void serdes_am654_disable_pll(struct serdes_am654 *phy)
>> +{
>> +    u32 mask = PLL_ENABLE_OVL | PLL_ENABLE_MASK;
>> +
>> +    regmap_update_bits(phy->regmap, WIZ_PLL_CTRL, mask, PLL_ENABLE_OVL);
>> +}
>> +
>> +static int serdes_am654_enable_txrx(struct serdes_am654 *phy)
>> +{
>> +    u32 mask;
>> +    u32 val;
>> +
>> +    /* Enable TX */
>> +    mask = TX0_ENABLE_OVL | TX0_ENABLE_MASK;
>> +    val = TX0_ENABLE_OVL | (TX0_ENABLE_STATE << TX0_ENABLE_SHIFT);
>> +    regmap_update_bits(phy->regmap, WIZ_LANEXCTL_STS, mask, val);
>> +
>> +    /* Enable RX */
>> +    mask = RX0_ENABLE_OVL | RX0_ENABLE_MASK;
>> +    val = RX0_ENABLE_OVL | (RX0_ENABLE_STATE << RX0_ENABLE_SHIFT);
>> +    regmap_update_bits(phy->regmap, WIZ_LANEXCTL_STS, mask, val);
>> +
>> +    return 0;
>> +}
>> +
>> +static int serdes_am654_disable_txrx(struct serdes_am654 *phy)
>> +{
>> +    u32 mask;
>> +
>> +    /* Disable TX */
>> +    mask = TX0_ENABLE_OVL | TX0_ENABLE_MASK;
>> +    regmap_update_bits(phy->regmap, WIZ_LANEXCTL_STS, mask, TX0_ENABLE_OVL);
>> +
>> +    /* Disable RX */
>> +    mask = RX0_ENABLE_OVL | RX0_ENABLE_MASK;
>> +    regmap_update_bits(phy->regmap, WIZ_LANEXCTL_STS, mask, RX0_ENABLE_OVL);
>> +
>> +    return 0;
>> +}
>> +
>> +static int serdes_am654_power_on(struct phy *x)
>> +{
>> +    struct serdes_am654 *phy = phy_get_drvdata(x);
>> +    struct device *dev = phy->dev;
>> +    int ret;
>> +    u32 val;
>> +
>> +    ret = serdes_am654_enable_pll(phy);
>> +    if (ret) {
>> +            dev_err(dev, "Failed to enable PLL\n");
>> +            return ret;
>> +    }
>> +
>> +    ret = serdes_am654_enable_txrx(phy);
>> +    if (ret) {
>> +            dev_err(dev, "Failed to enable TX RX\n");
>> +            return ret;
>> +    }
>> +
>> +    return regmap_read_poll_timeout(phy->regmap, COMLANE_R194, val,
>> +                                    val & CMU_OK_I_0, SLEEP_TIME,
>> +                                    PLL_LOCK_TIME);
>> +}
>> +
>> +static int serdes_am654_power_off(struct phy *x)
>> +{
>> +    struct serdes_am654 *phy = phy_get_drvdata(x);
>> +
>> +    serdes_am654_disable_txrx(phy);
>> +    serdes_am654_disable_pll(phy);
>> +
>> +    return 0;
>> +}
>> +
>> +static int serdes_am654_init(struct phy *x)
>> +{
>> +    struct serdes_am654 *phy = phy_get_drvdata(x);
>> +    u32 mask;
>> +    u32 val;
>> +
>> +    mask = CONFIG_VERSION_REG_MASK;
>> +    val = VERSION << CONFIG_VERSION_REG_SHIFT;
>> +    regmap_update_bits(phy->regmap, COMLANE_R138, mask, val);
>> +
>> +    val = CMU_MASTER_CDN_O;
>> +    regmap_update_bits(phy->regmap, CMU_R07C, val, val);
>> +
>> +    val = L1_MASTER_CDN_O;
>> +    regmap_update_bits(phy->regmap, COMLANE_R190, val, val);
>> +
>> +    return 0;
>> +}
>> +
>> +static int serdes_am654_reset(struct phy *x)
>> +{
>> +    struct serdes_am654 *phy = phy_get_drvdata(x);
>> +    u32 val;
>> +
>> +    val = POR_EN;
>> +    regmap_update_bits(phy->regmap, SERDES_CTRL, val, val);
>> +    mdelay(1);
>> +    regmap_update_bits(phy->regmap, SERDES_CTRL, val, 0);
>> +
>> +    return 0;
>> +}
>> +
>> +static void serdes_am654_release(struct phy *x)
>> +{
>> +    struct serdes_am654 *phy = phy_get_drvdata(x);
>> +
>> +    phy->type = PHY_NONE;
>> +    phy->busy = false;
>> +    mux_control_deselect(phy->control);
>> +}
>> +
>> +struct phy *serdes_am654_xlate(struct device *dev, struct of_phandle_args
>> +                             *args)
>> +{
>> +    struct serdes_am654 *am654_phy;
>> +    struct phy *phy;
>> +    int ret;
>> +
>> +    phy = of_phy_simple_xlate(dev, args);
>> +    if (IS_ERR(phy))
>> +            return phy;
>> +
>> +    am654_phy = phy_get_drvdata(phy);
>> +    if (am654_phy->busy)
>> +            return ERR_PTR(-EBUSY);
>> +
>> +    ret = mux_control_select(am654_phy->control, args->args[1]);
>> +    if (ret) {
>> +            dev_err(dev, "Failed to select SERDES Lane Function\n");
>> +            return ERR_PTR(ret);
>> +    }
>> +
>> +    am654_phy->busy = true;
>> +    am654_phy->type = args->args[0];
>> +
>> +    return phy;
>> +}
>> +
>> +static const struct phy_ops ops = {
>> +    .reset          = serdes_am654_reset,
>> +    .init           = serdes_am654_init,
>> +    .power_on       = serdes_am654_power_on,
>> +    .power_off      = serdes_am654_power_off,
>> +    .release        = serdes_am654_release,
>> +    .owner          = THIS_MODULE,
>> +};
>> +
>> +static u8 serdes_am654_clk_mux_get_parent(struct clk_hw *hw)
>> +{
>> +    struct serdes_am654_clk_mux *mux = to_serdes_am654_clk_mux(hw);
>> +    unsigned int num_parents = clk_hw_get_num_parents(hw);
>> +    struct regmap *regmap = mux->regmap;
>> +    unsigned int reg = mux->reg;
>> +    unsigned int val;
>> +    int i;
>> +
>> +    regmap_read(regmap, reg, &val);
>> +    val >>= mux->shift;
>> +    val &= mux->mask;
>> +
>> +    for (i = 0; i < num_parents; i++)
>> +            if (mux->table[i] == val)
>> +                    return i;
>> +
>> +    /*
>> +     * No parent? This should never happen!
>> +     * Verify if we set a valid parent in serdes_am654_clk_register()
>> +     */
>> +    WARN(1, "Failed to find the parent of %s clock\n", hw->init->name);
>> +
>> +    /* Make the parent lookup to fail */
>> +    return num_parents;
>> +}
>> +
>> +static int serdes_am654_clk_mux_set_parent(struct clk_hw *hw, u8 index)
>> +{
>> +    struct serdes_am654_clk_mux *mux = to_serdes_am654_clk_mux(hw);
>> +    struct regmap *regmap = mux->regmap;
>> +    unsigned int reg = mux->reg;
>> +    int val;
>> +    int ret;
>> +
>> +    val = mux->table[index];
>> +
>> +    if (val == -1)
>> +            return -EINVAL;
>> +
>> +    val <<= mux->shift;
>> +    ret = regmap_update_bits(regmap, reg, mux->mask << mux->shift, val);
>> +
>> +    return ret;
>> +}
>> +
>> +static const struct clk_ops serdes_am654_clk_mux_ops = {
>> +    .set_parent = serdes_am654_clk_mux_set_parent,
>> +    .get_parent = serdes_am654_clk_mux_get_parent,
>> +};
>> +
>> +static int mux_table[SERDES_NUM_CLOCKS][3] = {
>> +    /*
>> +     * The entries represent values for selecting between
>> +     * {left input, external reference clock, right input}
>> +     * Only one of Left Output or Right Output should be used since
>> +     * both left and right output clock uses the same bits and modifying
>> +     * one clock will impact the other.
>> +     */
>> +    { BIT(2),               0, BIT(0) }, /* Mux of CMU refclk */
>> +    {     -1,          BIT(3), BIT(1) }, /* Mux of Left Output */
>> +    { BIT(1), BIT(3) | BIT(1),     -1 }, /* Mux of Right Output */
>> +};
>> +
>> +static int mux_mask[SERDES_NUM_CLOCKS] = { 0x5, 0xa, 0xa };
>> +
>> +static int serdes_am654_clk_register(struct serdes_am654 *am654_phy,
>> +                                 const char *clock_name, int clock_num)
>> +{
>> +    struct device_node *node = am654_phy->of_node;
>> +    struct device *dev = am654_phy->dev;
>> +    struct serdes_am654_clk_mux *mux;
>> +    struct device_node *regmap_node;
>> +    const char **parent_names;
>> +    struct clk_init_data *init;
>> +    unsigned int num_parents;
>> +    struct regmap *regmap;
>> +    const __be32 *addr;
>> +    unsigned int reg;
>> +    struct clk *clk;
>> +
>> +    mux = devm_kzalloc(dev, sizeof(*mux), GFP_KERNEL);
>> +    if (!mux)
>> +            return -ENOMEM;
>> +
>> +    init = &mux->clk_data;
>> +
>> +    regmap_node = of_parse_phandle(node, "ti,serdes-clk", 0);
>> +    of_node_put(regmap_node);
>> +    if (!regmap_node) {
>> +            dev_err(dev, "Fail to get serdes-clk node\n");
>> +            return -ENODEV;
>> +    }
>> +
>> +    regmap = syscon_node_to_regmap(regmap_node->parent);
>> +    if (IS_ERR(regmap)) {
>> +            dev_err(dev, "Fail to get Syscon regmap\n");
>> +            return PTR_ERR(regmap);
>> +    }
>> +
>> +    num_parents = of_clk_get_parent_count(node);
>> +    if (num_parents < 2) {
>> +            dev_err(dev, "SERDES clock must have parents\n");
>> +            return -EINVAL;
>> +    }
>> +
>> +    parent_names = devm_kzalloc(dev, (sizeof(char *) * num_parents),
>> +                                GFP_KERNEL);
>> +    if (!parent_names)
>> +            return -ENOMEM;
>> +
>> +    of_clk_parent_fill(node, parent_names, num_parents);
>> +
>> +    addr = of_get_address(regmap_node, 0, NULL, NULL);
>> +    if (!addr)
>> +            return -EINVAL;
>> +
>> +    reg = be32_to_cpu(*addr);
>> +
>> +    init->ops = &serdes_am654_clk_mux_ops;
>> +    init->flags = CLK_SET_RATE_NO_REPARENT;
>> +    init->parent_names = parent_names;
>> +    init->num_parents = num_parents;
>> +    init->name = clock_name;
>> +
>> +    mux->table = mux_table[clock_num];
>> +    mux->regmap = regmap;
>> +    mux->reg = reg;
>> +    mux->shift = 4;
>> +    mux->mask = mux_mask[clock_num];
>> +    mux->hw.init = init;
>> +
>> +    /*
>> +     * setup a sane default so get_parent() call evaluates
>> +     * to a valid parent. Index 1 is the safest choice as
>> +     * the default as it is valid value for all of serdes's
>> +     * output clocks.
>> +     */
>> +    serdes_am654_clk_mux_set_parent(&mux->hw, 1);
>> +    clk = devm_clk_register(dev, &mux->hw);
>> +    if (IS_ERR(clk))
>> +            return PTR_ERR(clk);
>> +
>> +    am654_phy->clks[clock_num] = clk;
>> +
>> +    return 0;
>> +}
>> +
>> +static const struct of_device_id serdes_am654_id_table[] = {
>> +    {
>> +            .compatible = "ti,phy-am654-serdes",
>> +    },
>> +    {}
>> +};
>> +MODULE_DEVICE_TABLE(of, serdes_am654_id_table);
>> +
>> +static int serdes_am654_probe(struct platform_device *pdev)
>> +{
>> +    struct phy_provider *phy_provider;
>> +    struct device *dev = &pdev->dev;
>> +    struct device_node *node = dev->of_node;
>> +    struct clk_onecell_data *clk_data;
>> +    struct serdes_am654 *am654_phy;
>> +    struct mux_control *control;
>> +    const char *clock_name;
>> +    struct regmap *regmap;
>> +    struct resource *res;
>> +    void __iomem *base;
>> +    struct phy *phy;
>> +    int ret;
>> +    int i;
>> +
>> +    am654_phy = devm_kzalloc(dev, sizeof(*am654_phy), GFP_KERNEL);
>> +    if (!am654_phy)
>> +            return -ENOMEM;
>> +
>> +    res = platform_get_resource_byname(pdev, IORESOURCE_MEM, "serdes");
>> +    base = devm_ioremap_resource(dev, res);
>> +    if (IS_ERR(base))
>> +            return PTR_ERR(base);
>> +
>> +    regmap = devm_regmap_init_mmio(dev, base, &serdes_am654_regmap_config);
>> +    if (IS_ERR(regmap)) {
>> +            dev_err(dev, "Failed to initialize regmap\n");
>> +            return PTR_ERR(regmap);
>> +    }
>> +
>> +    control = devm_mux_control_get(dev, NULL);
>> +    if (IS_ERR(control))
>> +            return PTR_ERR(control);
>> +
>> +    am654_phy->dev = dev;
>> +    am654_phy->of_node = node;
>> +    am654_phy->regmap = regmap;
>> +    am654_phy->control = control;
>> +    am654_phy->type = PHY_NONE;
>> +
>> +    platform_set_drvdata(pdev, am654_phy);
>> +
>> +    for (i = 0; i < SERDES_NUM_CLOCKS; i++) {
>> +            ret = of_property_read_string_index(node, "clock-output-names",
>> +                                                i, &clock_name);
>> +            if (ret) {
>> +                    dev_err(dev, "Failed to get clock name\n");
>> +                    return ret;
>> +            }
>> +
>> +            ret = serdes_am654_clk_register(am654_phy, clock_name, i);
>> +            if (ret) {
>> +                    dev_err(dev, "Failed to initialize clock %s\n",
>> +                            clock_name);
>> +                    return ret;
>> +            }
>> +    }
>> +
>> +    clk_data = &am654_phy->clk_data;
>> +    clk_data->clks = am654_phy->clks;
>> +    clk_data->clk_num = SERDES_NUM_CLOCKS;
>> +    ret = of_clk_add_provider(node, of_clk_src_onecell_get, clk_data);
>> +    if (ret)
>> +            return ret;
>> +
>> +    pm_runtime_enable(dev);
>> +
>> +    phy = devm_phy_create(dev, NULL, &ops);
>> +    if (IS_ERR(phy))
>> +            return PTR_ERR(phy);
>> +
>> +    phy_set_drvdata(phy, am654_phy);
>> +    phy_provider = devm_of_phy_provider_register(dev, serdes_am654_xlate);
>> +    if (IS_ERR(phy_provider)) {
>> +            ret = PTR_ERR(phy_provider);
>> +            goto clk_err;
>> +    }
>> +
>> +    return 0;
>> +
>> +clk_err:
>> +    of_clk_del_provider(node);
>> +
>> +    return ret;
>> +}
>> +
>> +static int serdes_am654_remove(struct platform_device *pdev)
>> +{
>> +    struct serdes_am654 *am654_phy = platform_get_drvdata(pdev);
>> +    struct device_node *node = am654_phy->of_node;
>> +
>> +    pm_runtime_disable(&pdev->dev);
>> +    of_clk_del_provider(node);
>> +
>> +    return 0;
>> +}
>> +
>> +static struct platform_driver serdes_am654_driver = {
>> +    .probe          = serdes_am654_probe,
>> +    .remove         = serdes_am654_remove,
>> +    .driver         = {
>> +            .name   = "phy-am654",
>> +            .of_match_table = serdes_am654_id_table,
>> +    },
>> +};
>> +module_platform_driver(serdes_am654_driver);
>> +
>> +MODULE_ALIAS("platform:phy-am654");
> 
> MODULE_ALIAS not needed here, should get one built from the
> MODULE_DEVICE_TABLE above and only the OF one will get searched anyway.

I'll fix this one too.

Thanks
Kishon

Reply via email to