clk: agilex/stratix10: add support for the 2nd bypass
authorDinh Nguyen <dinguyen@kernel.org>
Fri, 11 Jun 2021 02:52:00 +0000 (21:52 -0500)
committerStephen Boyd <sboyd@kernel.org>
Sun, 27 Jun 2021 23:39:59 +0000 (16:39 -0700)
The EMAC clocks on Stratix10/Agilex/N5X have an additional bypass that
was not being accounted for. The bypass selects between
emaca_clk/emacb_clk and boot_clk.

Because the bypass register offset is different between Stratix10 and
Agilex/N5X, it's best to create a new function to calculate the bypass.

Fixes: 80c6b7a0894f ("clk: socfpga: agilex: add clock driver for the Agilex platform")
Cc: stable@vger.kernel.org
Signed-off-by: Dinh Nguyen <dinguyen@kernel.org>
Link: https://lore.kernel.org/r/20210611025201.118799-3-dinguyen@kernel.org
Signed-off-by: Stephen Boyd <sboyd@kernel.org>
drivers/clk/socfpga/clk-agilex.c
drivers/clk/socfpga/clk-gate-s10.c
drivers/clk/socfpga/stratix10-clk.h

index edfa87d..1cb21ea 100644 (file)
@@ -177,6 +177,8 @@ static const struct clk_parent_data emac_mux[] = {
          .name = "emaca_free_clk", },
        { .fw_name = "emacb_free_clk",
          .name = "emacb_free_clk", },
+       { .fw_name = "boot_clk",
+         .name = "boot_clk", },
 };
 
 static const struct clk_parent_data noc_mux[] = {
@@ -399,7 +401,7 @@ static int agilex_clk_register_gate(const struct stratix10_gate_clock *clks,
        int i;
 
        for (i = 0; i < nums; i++) {
-               hw_clk = s10_register_gate(&clks[i], base);
+               hw_clk = agilex_register_gate(&clks[i], base);
                if (IS_ERR(hw_clk)) {
                        pr_err("%s: failed to register clock %s\n",
                               __func__, clks[i].name);
index b84f262..3256779 100644 (file)
 #define SOCFPGA_CS_PDBG_CLK    "cs_pdbg_clk"
 #define to_socfpga_gate_clk(p) container_of(p, struct socfpga_gate_clk, hw.hw)
 
+#define SOCFPGA_EMAC0_CLK              "emac0_clk"
+#define SOCFPGA_EMAC1_CLK              "emac1_clk"
+#define SOCFPGA_EMAC2_CLK              "emac2_clk"
+#define AGILEX_BYPASS_OFFSET           0xC
+#define STRATIX10_BYPASS_OFFSET                0x2C
+#define BOOTCLK_BYPASS                 2
+
 static unsigned long socfpga_gate_clk_recalc_rate(struct clk_hw *hwclk,
                                                  unsigned long parent_rate)
 {
@@ -44,14 +51,61 @@ static unsigned long socfpga_dbg_clk_recalc_rate(struct clk_hw *hwclk,
 static u8 socfpga_gate_get_parent(struct clk_hw *hwclk)
 {
        struct socfpga_gate_clk *socfpgaclk = to_socfpga_gate_clk(hwclk);
-       u32 mask;
+       u32 mask, second_bypass;
+       u8 parent = 0;
+       const char *name = clk_hw_get_name(hwclk);
+
+       if (socfpgaclk->bypass_reg) {
+               mask = (0x1 << socfpgaclk->bypass_shift);
+               parent = ((readl(socfpgaclk->bypass_reg) & mask) >>
+                         socfpgaclk->bypass_shift);
+       }
+
+       if (streq(name, SOCFPGA_EMAC0_CLK) ||
+           streq(name, SOCFPGA_EMAC1_CLK) ||
+           streq(name, SOCFPGA_EMAC2_CLK)) {
+               second_bypass = readl(socfpgaclk->bypass_reg -
+                                     STRATIX10_BYPASS_OFFSET);
+               /* EMACA bypass to bootclk @0xB0 offset */
+               if (second_bypass & 0x1)
+                       if (parent == 0) /* only applicable if parent is maca */
+                               parent = BOOTCLK_BYPASS;
+
+               if (second_bypass & 0x2)
+                       if (parent == 1) /* only applicable if parent is macb */
+                               parent = BOOTCLK_BYPASS;
+       }
+       return parent;
+}
+
+static u8 socfpga_agilex_gate_get_parent(struct clk_hw *hwclk)
+{
+       struct socfpga_gate_clk *socfpgaclk = to_socfpga_gate_clk(hwclk);
+       u32 mask, second_bypass;
        u8 parent = 0;
+       const char *name = clk_hw_get_name(hwclk);
 
        if (socfpgaclk->bypass_reg) {
                mask = (0x1 << socfpgaclk->bypass_shift);
                parent = ((readl(socfpgaclk->bypass_reg) & mask) >>
                          socfpgaclk->bypass_shift);
        }
+
+       if (streq(name, SOCFPGA_EMAC0_CLK) ||
+           streq(name, SOCFPGA_EMAC1_CLK) ||
+           streq(name, SOCFPGA_EMAC2_CLK)) {
+               second_bypass = readl(socfpgaclk->bypass_reg -
+                                     AGILEX_BYPASS_OFFSET);
+               /* EMACA bypass to bootclk @0x88 offset */
+               if (second_bypass & 0x1)
+                       if (parent == 0) /* only applicable if parent is maca */
+                               parent = BOOTCLK_BYPASS;
+
+               if (second_bypass & 0x2)
+                       if (parent == 1) /* only applicable if parent is macb */
+                               parent = BOOTCLK_BYPASS;
+       }
+
        return parent;
 }
 
@@ -60,6 +114,11 @@ static struct clk_ops gateclk_ops = {
        .get_parent = socfpga_gate_get_parent,
 };
 
+static const struct clk_ops agilex_gateclk_ops = {
+       .recalc_rate = socfpga_gate_clk_recalc_rate,
+       .get_parent = socfpga_agilex_gate_get_parent,
+};
+
 static const struct clk_ops dbgclk_ops = {
        .recalc_rate = socfpga_dbg_clk_recalc_rate,
        .get_parent = socfpga_gate_get_parent,
@@ -122,3 +181,61 @@ struct clk_hw *s10_register_gate(const struct stratix10_gate_clock *clks, void _
        }
        return hw_clk;
 }
+
+struct clk_hw *agilex_register_gate(const struct stratix10_gate_clock *clks, void __iomem *regbase)
+{
+       struct clk_hw *hw_clk;
+       struct socfpga_gate_clk *socfpga_clk;
+       struct clk_init_data init;
+       const char *parent_name = clks->parent_name;
+       int ret;
+
+       socfpga_clk = kzalloc(sizeof(*socfpga_clk), GFP_KERNEL);
+       if (!socfpga_clk)
+               return NULL;
+
+       socfpga_clk->hw.reg = regbase + clks->gate_reg;
+       socfpga_clk->hw.bit_idx = clks->gate_idx;
+
+       gateclk_ops.enable = clk_gate_ops.enable;
+       gateclk_ops.disable = clk_gate_ops.disable;
+
+       socfpga_clk->fixed_div = clks->fixed_div;
+
+       if (clks->div_reg)
+               socfpga_clk->div_reg = regbase + clks->div_reg;
+       else
+               socfpga_clk->div_reg = NULL;
+
+       socfpga_clk->width = clks->div_width;
+       socfpga_clk->shift = clks->div_offset;
+
+       if (clks->bypass_reg)
+               socfpga_clk->bypass_reg = regbase + clks->bypass_reg;
+       else
+               socfpga_clk->bypass_reg = NULL;
+       socfpga_clk->bypass_shift = clks->bypass_shift;
+
+       if (streq(clks->name, "cs_pdbg_clk"))
+               init.ops = &dbgclk_ops;
+       else
+               init.ops = &agilex_gateclk_ops;
+
+       init.name = clks->name;
+       init.flags = clks->flags;
+
+       init.num_parents = clks->num_parents;
+       init.parent_names = parent_name ? &parent_name : NULL;
+       if (init.parent_names == NULL)
+               init.parent_data = clks->parent_data;
+       socfpga_clk->hw.hw.init = &init;
+
+       hw_clk = &socfpga_clk->hw.hw;
+
+       ret = clk_hw_register(NULL, &socfpga_clk->hw.hw);
+       if (ret) {
+               kfree(socfpga_clk);
+               return ERR_PTR(ret);
+       }
+       return hw_clk;
+}
index 61eaf3a..75234e0 100644 (file)
@@ -85,4 +85,6 @@ struct clk_hw *s10_register_cnt_periph(const struct stratix10_perip_cnt_clock *c
                                    void __iomem *reg);
 struct clk_hw *s10_register_gate(const struct stratix10_gate_clock *clks,
                              void __iomem *reg);
+struct clk_hw *agilex_register_gate(const struct stratix10_gate_clock *clks,
+                             void __iomem *reg);
 #endif /* __STRATIX10_CLK_H */