From: Zhangfei Gao
...
> diff --git a/drivers/clk/hisilicon/clk-hix5hd2.c 
> b/drivers/clk/hisilicon/clk-hix5hd2.c
> index e5fcfb4..1b1347f 100644
> --- a/drivers/clk/hisilicon/clk-hix5hd2.c
> +++ b/drivers/clk/hisilicon/clk-hix5hd2.c
> @@ -9,6 +9,8 @@
> 
>  #include <linux/of_address.h>
>  #include <dt-bindings/clock/hix5hd2-clock.h>
> +#include <linux/slab.h>
> +#include <linux/delay.h>
>  #include "clk.h"
> 
>  static struct hisi_fixed_rate_clock hix5hd2_fixed_rate_clks[] __initdata = {
> @@ -79,8 +81,186 @@ static struct hisi_gate_clock hix5hd2_gate_clks[] 
> __initdata = {
>               CLK_SET_RATE_PARENT, 0xa0, 1, 0, },
>       { HIX5HD2_MMC_CIU_RST, "rst_mmc_ciu", "clk_mmc_ciu",
>               CLK_SET_RATE_PARENT, 0xa0, 4, CLK_GATE_SET_TO_DISABLE, },
> +     /*gsf*/
> +     { HIX5HD2_FWD_BUS_CLK, "clk_fwd_bus", NULL, 0, 0xcc, 0, 0, },
> +     { HIX5HD2_FWD_SYS_CLK, "clk_fwd_sys", "clk_fwd_bus", 0, 0xcc, 5, 0, },
> +     { HIX5HD2_MAC0_PHY_CLK, "clk_fephy", "clk_fwd_sys",
> +              CLK_SET_RATE_PARENT, 0x120, 0, 0, },
>  };
> 
> +enum {TYPE_COMPLEX, TYPE_ETHER};

Shouldn't this be a named enum to make it more obvious
where the values should be used?

> +
> +struct hix5hd2_complex_clock {
> +     unsigned int    id;

Reorder the fields to avoid the implicit pad here.

> +     const char      *name;
> +     const char      *parent_name;
> +     u32             ctrl_reg;
> +     u32             ctrl_clk_mask;
> +     u32             ctrl_rst_mask;
> +     u32             phy_reg;
> +     u32             phy_clk_mask;
> +     u32             phy_rst_mask;
> +     u32             type;
> +};
> +
> +struct hix5hd2_clk_complex {
> +     struct clk_hw   hw;
> +     u32             id;
> +     void __iomem    *ctrl_reg;
> +     u32             ctrl_clk_mask;
> +     u32             ctrl_rst_mask;
> +     void __iomem    *phy_reg;
> +     u32             phy_clk_mask;
> +     u32             phy_rst_mask;
> +};
> +
> +static struct hix5hd2_complex_clock hix5hd2_complex_clks[] __initdata = {
> +     {HIX5HD2_MAC0_CLK, "clk_mac0", "clk_fephy",
> +             0xcc, 0xa, 0x500, 0x120, 0, 0x10, TYPE_ETHER},
> +     {HIX5HD2_MAC1_CLK, "clk_mac1", "clk_fwd_sys",
> +             0xcc, 0x14, 0xa00, 0x168, 0x2, 0, TYPE_ETHER},
> +     {HIX5HD2_SATA_CLK, "clk_sata", NULL,
> +             0xa8, 0x1f, 0x300, 0xac, 0x1, 0x0, TYPE_COMPLEX},
> +     {HIX5HD2_USB_CLK, "clk_usb", NULL,
> +             0xb8, 0xff, 0x3f00, 0xbc, 0x7, 0x3f00, TYPE_COMPLEX},
> +};
> +
> +#define to_complex_clk(_hw) container_of(_hw, struct hix5hd2_clk_complex, hw)
> +
> +static int clk_ether_enable(struct clk_hw *hw)
> +{
> +     struct hix5hd2_clk_complex *clk = to_complex_clk(hw);
> +     u32 val;
> +
> +     val = readl(clk->ctrl_reg);
> +     val |= clk->ctrl_clk_mask | clk->ctrl_rst_mask;
> +     writel(val, clk->ctrl_reg);
> +     udelay(50);
> +     val &= ~(clk->ctrl_rst_mask);
> +     writel(val, clk->ctrl_reg);

I'd need to be convinced that the udelay() has the desired effect.
I suspect you are trying to assert reset for a minimum period.
However the first write can be 'posted' by all sorts of hardware
for all sorts of reasons - so the writes can actually be back to back.

        David

> +
> +     val = readl(clk->phy_reg);
> +     val |= clk->phy_clk_mask;
> +     val &= ~(clk->phy_rst_mask);
> +     writel(val, clk->phy_reg);
> +     mdelay(10);
> +
> +     val &= ~(clk->phy_clk_mask);
> +     val |= clk->phy_rst_mask;
> +     writel(val, clk->phy_reg);
> +     mdelay(10);
> +
> +     val |= clk->phy_clk_mask;
> +     val &= ~(clk->phy_rst_mask);
> +     writel(val, clk->phy_reg);
> +     mdelay(30);
> +     return 0;
> +}
> +
> +static void clk_ether_disable(struct clk_hw *hw)
> +{
> +     struct hix5hd2_clk_complex *clk = to_complex_clk(hw);
> +     u32 val;
> +
> +     val = readl(clk->ctrl_reg);
> +     val &= ~(clk->ctrl_clk_mask);
> +     writel(val, clk->ctrl_reg);
> +}
> +
> +static struct clk_ops clk_ether_ops = {
> +     .enable = clk_ether_enable,
> +     .disable = clk_ether_disable,
> +};
> +
> +static int clk_complex_enable(struct clk_hw *hw)
> +{
> +     struct hix5hd2_clk_complex *clk = to_complex_clk(hw);
> +     u32 val;
> +
> +     val = readl(clk->ctrl_reg);
> +     val |= clk->ctrl_clk_mask;
> +     val &= ~(clk->ctrl_rst_mask);
> +     writel(val, clk->ctrl_reg);
> +
> +     val = readl(clk->phy_reg);
> +     val |= clk->phy_clk_mask;
> +     val &= ~(clk->phy_rst_mask);
> +     writel(val, clk->phy_reg);
> +
> +     return 0;
> +}
> +
> +static void clk_complex_disable(struct clk_hw *hw)
> +{
> +     struct hix5hd2_clk_complex *clk = to_complex_clk(hw);
> +     u32 val;
> +
> +     val = readl(clk->ctrl_reg);
> +     val |= clk->ctrl_rst_mask;
> +     val &= ~(clk->ctrl_clk_mask);
> +     writel(val, clk->ctrl_reg);
> +
> +     val = readl(clk->phy_reg);
> +     val |= clk->phy_rst_mask;
> +     val &= ~(clk->phy_clk_mask);
> +     writel(val, clk->phy_reg);
> +
> +     return;
> +}
> +
> +static struct clk_ops clk_complex_ops = {
> +     .enable = clk_complex_enable,
> +     .disable = clk_complex_disable,
> +};
> +
> +void __init hix5hd2_clk_register_complex_clk(struct hix5hd2_complex_clock 
> *clks,
> +                                    int nums, struct hisi_clock_data *data)
> +{
> +     void __iomem *base = data->base;
> +     int i;
> +
> +     for (i = 0; i < nums; i++) {
> +             struct hix5hd2_clk_complex *p_clk;
> +             struct clk *clk;
> +             struct clk_init_data init;
> +
> +             p_clk = kzalloc(sizeof(*p_clk), GFP_KERNEL);
> +             if (!p_clk) {
> +                     pr_err("%s: fail to allocate clk\n", __func__);
> +                     return;
> +             }
> +
> +             init.name = clks[i].name;
> +             if (clks[i].type == TYPE_ETHER)
> +                     init.ops = &clk_ether_ops;
> +             else
> +                     init.ops = &clk_complex_ops;
> +
> +             init.flags = CLK_IS_BASIC;
> +             init.parent_names =
> +                     (clks[i].parent_name ? &clks[i].parent_name : NULL);
> +             init.num_parents = (clks[i].parent_name ? 1 : 0);
> +
> +             p_clk->ctrl_reg = base + clks[i].ctrl_reg;
> +             p_clk->ctrl_clk_mask = clks[i].ctrl_clk_mask;
> +             p_clk->ctrl_rst_mask = clks[i].ctrl_rst_mask;
> +             p_clk->phy_reg = base + clks[i].phy_reg;
> +             p_clk->phy_clk_mask = clks[i].phy_clk_mask;
> +             p_clk->phy_rst_mask = clks[i].phy_rst_mask;
> +             p_clk->hw.init = &init;
> +
> +             clk = clk_register(NULL, &p_clk->hw);
> +             if (IS_ERR(clk)) {
> +                     kfree(p_clk);
> +                     pr_err("%s: failed to register clock %s\n",
> +                            __func__, clks[i].name);
> +                     continue;
> +             }
> +
> +             data->clk_data.clks[clks[i].id] = clk;
> +     }
> +}
> +
>  static void __init hix5hd2_clk_init(struct device_node *np)
>  {
>       struct hisi_clock_data *clk_data;
> @@ -96,6 +276,8 @@ static void __init hix5hd2_clk_init(struct device_node *np)
>                                       clk_data);
>       hisi_clk_register_gate(hix5hd2_gate_clks,
>                       ARRAY_SIZE(hix5hd2_gate_clks), clk_data);
> +     hix5hd2_clk_register_complex_clk(hix5hd2_complex_clks,
> +                     ARRAY_SIZE(hix5hd2_complex_clks), clk_data);
>  }
> 
>  CLK_OF_DECLARE(hix5hd2_clk, "hisilicon,hix5hd2-clock", hix5hd2_clk_init);
> diff --git a/include/dt-bindings/clock/hix5hd2-clock.h 
> b/include/dt-bindings/clock/hix5hd2-clock.h
> index aad579a..e328669 100644
> --- a/include/dt-bindings/clock/hix5hd2-clock.h
> +++ b/include/dt-bindings/clock/hix5hd2-clock.h
> @@ -53,6 +53,15 @@
>  #define HIX5HD2_MMC_CIU_CLK          130
>  #define HIX5HD2_MMC_BIU_CLK          131
>  #define HIX5HD2_MMC_CIU_RST          132
> +#define HIX5HD2_FWD_BUS_CLK          133
> +#define HIX5HD2_FWD_SYS_CLK          134
> +#define HIX5HD2_MAC0_PHY_CLK         135
> +
> +/* complex */
> +#define HIX5HD2_MAC0_CLK             192
> +#define HIX5HD2_MAC1_CLK             193
> +#define HIX5HD2_SATA_CLK             194
> +#define HIX5HD2_USB_CLK                      195
> 
>  #define HIX5HD2_NR_CLKS                      256
>  #endif       /* __DTS_HIX5HD2_CLOCK_H */
> --
> 1.7.9.5



--
To unsubscribe from this list: send the line "unsubscribe devicetree" in
the body of a message to [email protected]
More majordomo info at  http://vger.kernel.org/majordomo-info.html

Reply via email to