]> git.dujemihanovic.xyz Git - u-boot.git/commitdiff
clk: exynos: Fix incorrect clock lookup for non-top CMUs
authorSam Protsenko <semen.protsenko@linaro.org>
Fri, 8 Mar 2024 02:18:57 +0000 (20:18 -0600)
committerMinkyu Kang <mk7.kang@samsung.com>
Tue, 26 Mar 2024 09:56:55 +0000 (18:56 +0900)
Samsung clock drivers usually define the clock indices that are unique
per one CMU, but are not unique across all CMUs. That is, clock indices
start from 1 for each CMU, as provided in CMU bindings header. The way
the clock lookup via clk_get_by_index() works at the moment is by using
clk_of_xlate_default(), which returns globally non-unique clock ids for
for clocks registered with Samsung CCF API, which leads to incorrect
clocks being obtained. One way to fix that would be to make all clock
ids defined in the bindings header unique, but it'd make it incompatible
with Linux kernel bindings header. A better way to solve this issue is
to calculate the global clock id and use it when registering a clock
with clk_dm() and when obtaining it, in a custom .of_xlate function.

This patch adds an API for such mapping calculation, introducing the
necessary modifications to CMU registering functions in Samsung CCF.
Exynos850 clock driver (the only driver that uses Samsung CCF at the
moment) is modified accordingly, as it uses the changed API. So the
clock lookup with clk-exynos850.c driver is also fixed here.

The global clock id is calculated from CMU id and local clock id in
SAMSUNG_TO_CLK_ID() macro like this:

    clk_id_global = cmu_id * 256 + clk_id_local

leaving a range of up to 256 clocks for each CMU. Then this mapping
macro is used in clk_dm() to register clocks using their global ids, and
in .of_xlate() to lookup the clock by its local id correctly. Because
.of_xlate() operation has a separate function for each CMU, it "knows"
the correct way of finding the correct clk_id_global by provided
clk_id_local.

Fixes: ff3e8b8c6c22 ("clk: exynos: Add Samsung clock framework")
Fixes: a36cc5e3ef4d ("clk: exynos: Add Exynos850 clock driver")
Signed-off-by: Sam Protsenko <semen.protsenko@linaro.org>
Signed-off-by: Minkyu Kang <mk7.kang@samsung.com>
drivers/clk/exynos/clk-exynos850.c
drivers/clk/exynos/clk-pll.c
drivers/clk/exynos/clk-pll.h
drivers/clk/exynos/clk.c
drivers/clk/exynos/clk.h

index de4170cdc2f317c8f0e06ab7026ab093eaf76000..f11c1ff29bdde9cea4ee734afc5abefb5ca47a5a 100644 (file)
 #include <dt-bindings/clock/exynos850.h>
 #include "clk.h"
 
+enum exynos850_cmu_id {
+       CMU_TOP,
+       CMU_PERI,
+};
+
 /* ---- CMU_TOP ------------------------------------------------------------- */
 
 /* Register Offset definitions for CMU_TOP (0x120e0000) */
@@ -124,7 +129,7 @@ static const struct samsung_clk_group top_cmu_clks[] = {
 
 static int exynos850_cmu_top_probe(struct udevice *dev)
 {
-       return samsung_cmu_register_one(dev, top_cmu_clks,
+       return samsung_cmu_register_one(dev, CMU_TOP, top_cmu_clks,
                                        ARRAY_SIZE(top_cmu_clks));
 }
 
@@ -133,11 +138,13 @@ static const struct udevice_id exynos850_cmu_top_ids[] = {
        { }
 };
 
+SAMSUNG_CLK_OPS(exynos850_cmu_top, CMU_TOP);
+
 U_BOOT_DRIVER(exynos850_cmu_top) = {
        .name           = "exynos850-cmu-top",
        .id             = UCLASS_CLK,
        .of_match       = exynos850_cmu_top_ids,
-       .ops            = &ccf_clk_ops,
+       .ops            = &exynos850_cmu_top_clk_ops,
        .probe          = exynos850_cmu_top_probe,
        .flags          = DM_FLAG_PRE_RELOC,
 };
@@ -175,7 +182,8 @@ static const struct samsung_clk_group peri_cmu_clks[] = {
 
 static int exynos850_cmu_peri_probe(struct udevice *dev)
 {
-       return samsung_register_cmu(dev, peri_cmu_clks, exynos850_cmu_top);
+       return samsung_register_cmu(dev, CMU_PERI, peri_cmu_clks,
+                                   exynos850_cmu_top);
 }
 
 static const struct udevice_id exynos850_cmu_peri_ids[] = {
@@ -183,11 +191,13 @@ static const struct udevice_id exynos850_cmu_peri_ids[] = {
        { }
 };
 
+SAMSUNG_CLK_OPS(exynos850_cmu_peri, CMU_PERI);
+
 U_BOOT_DRIVER(exynos850_cmu_peri) = {
        .name           = "exynos850-cmu-peri",
        .id             = UCLASS_CLK,
        .of_match       = exynos850_cmu_peri_ids,
-       .ops            = &ccf_clk_ops,
+       .ops            = &exynos850_cmu_peri_clk_ops,
        .probe          = exynos850_cmu_peri_probe,
        .flags          = DM_FLAG_PRE_RELOC,
 };
index 4aacbc26b25d2959206e273e1d9101cec2021f52..542d577eaa6f667ba4fb31590eee01b034b22007 100644 (file)
@@ -136,7 +136,7 @@ static struct clk *_samsung_clk_register_pll(void __iomem *base,
        return clk;
 }
 
-void samsung_clk_register_pll(void __iomem *base,
+void samsung_clk_register_pll(void __iomem *base, unsigned int cmu_id,
                              const struct samsung_pll_clock *clk_list,
                              unsigned int nr_clk)
 {
@@ -145,10 +145,12 @@ void samsung_clk_register_pll(void __iomem *base,
        for (cnt = 0; cnt < nr_clk; cnt++) {
                struct clk *clk;
                const struct samsung_pll_clock *pll_clk;
+               unsigned long clk_id;
 
                pll_clk = &clk_list[cnt];
                clk = _samsung_clk_register_pll(base, pll_clk);
-               clk_dm(pll_clk->id, clk);
+               clk_id = SAMSUNG_TO_CLK_ID(cmu_id, pll_clk->id);
+               clk_dm(clk_id, clk);
        }
 }
 
index 00c75068707237c6667223fe486586145d625cc0..bdc94e7624d21a5da95282f686354da0fc771826 100644 (file)
@@ -22,7 +22,7 @@ enum samsung_pll_type {
        pll_0831x,
 };
 
-void samsung_clk_register_pll(void __iomem *base,
+void samsung_clk_register_pll(void __iomem *base, unsigned int cmu_id,
                              const struct samsung_pll_clock *clk_list,
                              unsigned int nr_clk);
 
index 14ccd2cba3745219e3c6766281b52aba24fec4de..943e8bd0189236f4e6035aaf2ac612075a7ee126 100644 (file)
 #include <dm.h>
 #include "clk.h"
 
-static void samsung_clk_register_mux(void __iomem *base,
-                             const struct samsung_mux_clock *clk_list,
-                             unsigned int nr_clk)
+static void samsung_clk_register_mux(void __iomem *base, unsigned int cmu_id,
+                                    const struct samsung_mux_clock *clk_list,
+                                    unsigned int nr_clk)
 {
        unsigned int cnt;
 
        for (cnt = 0; cnt < nr_clk; cnt++) {
                struct clk *clk;
                const struct samsung_mux_clock *m;
+               unsigned long clk_id;
 
                m = &clk_list[cnt];
                clk = clk_register_mux(NULL, m->name, m->parent_names,
                        m->num_parents, m->flags, base + m->offset, m->shift,
                        m->width, m->mux_flags);
-               clk_dm(m->id, clk);
+               clk_id = SAMSUNG_TO_CLK_ID(cmu_id, m->id);
+               clk_dm(clk_id, clk);
        }
 }
 
-static void samsung_clk_register_div(void __iomem *base,
-                             const struct samsung_div_clock *clk_list,
-                             unsigned int nr_clk)
+static void samsung_clk_register_div(void __iomem *base, unsigned int cmu_id,
+                                    const struct samsung_div_clock *clk_list,
+                                    unsigned int nr_clk)
 {
        unsigned int cnt;
 
        for (cnt = 0; cnt < nr_clk; cnt++) {
                struct clk *clk;
                const struct samsung_div_clock *d;
+               unsigned long clk_id;
 
                d = &clk_list[cnt];
                clk = clk_register_divider(NULL, d->name, d->parent_name,
                        d->flags, base + d->offset, d->shift,
                        d->width, d->div_flags);
-               clk_dm(d->id, clk);
+               clk_id = SAMSUNG_TO_CLK_ID(cmu_id, d->id);
+               clk_dm(clk_id, clk);
        }
 }
 
-static void samsung_clk_register_gate(void __iomem *base,
-                              const struct samsung_gate_clock *clk_list,
-                              unsigned int nr_clk)
+static void samsung_clk_register_gate(void __iomem *base, unsigned int cmu_id,
+                                     const struct samsung_gate_clock *clk_list,
+                                     unsigned int nr_clk)
 {
        unsigned int cnt;
 
        for (cnt = 0; cnt < nr_clk; cnt++) {
                struct clk *clk;
                const struct samsung_gate_clock *g;
+               unsigned long clk_id;
 
                g = &clk_list[cnt];
                clk = clk_register_gate(NULL, g->name, g->parent_name,
                        g->flags, base + g->offset, g->bit_idx,
                        g->gate_flags, NULL);
-               clk_dm(g->id, clk);
+               clk_id = SAMSUNG_TO_CLK_ID(cmu_id, g->id);
+               clk_dm(clk_id, clk);
        }
 }
 
-typedef void (*samsung_clk_register_fn)(void __iomem *base,
+typedef void (*samsung_clk_register_fn)(void __iomem *base, unsigned int cmu_id,
                                        const void *clk_list,
                                        unsigned int nr_clk);
 
@@ -78,13 +84,14 @@ static const samsung_clk_register_fn samsung_clk_register_fns[] = {
 /**
  * samsung_cmu_register_clocks() - Register provided clock groups
  * @base: Base address of CMU registers
+ * @cmu_id: CMU index number
  * @clk_groups: list of clock groups
  * @nr_groups: count of clock groups in @clk_groups
  *
  * Having the array of clock groups @clk_groups makes it possible to keep a
  * correct clocks registration order.
  */
-static void samsung_cmu_register_clocks(void __iomem *base,
+static void samsung_cmu_register_clocks(void __iomem *base, unsigned int cmu_id,
                                const struct samsung_clk_group *clk_groups,
                                unsigned int nr_groups)
 {
@@ -93,19 +100,21 @@ static void samsung_cmu_register_clocks(void __iomem *base,
        for (i = 0; i < nr_groups; i++) {
                const struct samsung_clk_group *g = &clk_groups[i];
 
-               samsung_clk_register_fns[g->type](base, g->clk_list, g->nr_clk);
+               samsung_clk_register_fns[g->type](base, cmu_id,
+                                                 g->clk_list, g->nr_clk);
        }
 }
 
 /**
  * samsung_cmu_register_one - Register all CMU clocks
  * @dev: CMU device
+ * @cmu_id: CMU index number
  * @clk_groups: list of CMU clock groups
  * @nr_groups: count of CMU clock groups in @clk_groups
  *
  * Return: 0 on success or negative value on error.
  */
-int samsung_cmu_register_one(struct udevice *dev,
+int samsung_cmu_register_one(struct udevice *dev, unsigned int cmu_id,
                             const struct samsung_clk_group *clk_groups,
                             unsigned int nr_groups)
 {
@@ -115,7 +124,7 @@ int samsung_cmu_register_one(struct udevice *dev,
        if (!base)
                return -EINVAL;
 
-       samsung_cmu_register_clocks(base, clk_groups, nr_groups);
+       samsung_cmu_register_clocks(base, cmu_id, clk_groups, nr_groups);
 
        return 0;
 }
index 14e06b2030fb63dfccb492d73d1863fd7efb3b07..ed0a395f0f636eea40c63dd7bbd2dbcaf8ae2b0b 100644 (file)
 #include <linux/clk-provider.h>
 #include "clk-pll.h"
 
+#define _SAMSUNG_CLK_OPS(_name, _cmu)                                  \
+static int _name##_of_xlate(struct clk *clk,                           \
+                           struct ofnode_phandle_args *args)           \
+{                                                                      \
+       if (args->args_count > 1) {                                     \
+               debug("Invalid args_count: %d\n", args->args_count);    \
+               return -EINVAL;                                         \
+       }                                                               \
+                                                                       \
+       if (args->args_count)                                           \
+               clk->id = SAMSUNG_TO_CLK_ID(_cmu, args->args[0]);       \
+       else                                                            \
+               clk->id = 0;                                            \
+                                                                       \
+       return 0;                                                       \
+}                                                                      \
+                                                                       \
+static const struct clk_ops _name##_clk_ops = {                                \
+       .set_rate = ccf_clk_set_rate,                                   \
+       .get_rate = ccf_clk_get_rate,                                   \
+       .set_parent = ccf_clk_set_parent,                               \
+       .enable = ccf_clk_enable,                                       \
+       .disable = ccf_clk_disable,                                     \
+       .of_xlate = _name##_of_xlate,                                   \
+}
+
+/**
+ * SAMSUNG_CLK_OPS - Define clock operations structure for specified CMU.
+ * @name: name of generated structure
+ * @cmu: CMU index
+ *
+ * Like ccf_clk_ops, but with custom .of_xlate callback.
+ */
+#define SAMSUNG_CLK_OPS(name, cmu) _SAMSUNG_CLK_OPS(name, cmu)
+
+/**
+ * SAMSUNG_TO_CLK_ID - Calculate a global clock index.
+ * @_cmu: CMU index
+ * @_id: local clock index (unique across @_cmu)
+ *
+ * Return: A global clock index unique across all CMUs.
+ * Keeps a range of 256 available clocks for every CMU.
+ */
+#define SAMSUNG_TO_CLK_ID(_cmu, _id)   (((_cmu) << 8) | ((_id) & 0xff))
+
 /**
  * struct samsung_mux_clock - information about mux clock
  * @id: platform specific id of the clock
@@ -179,13 +224,14 @@ struct samsung_clk_group {
        unsigned int nr_clk;
 };
 
-int samsung_cmu_register_one(struct udevice *dev,
+int samsung_cmu_register_one(struct udevice *dev, unsigned int cmu_id,
                             const struct samsung_clk_group *clk_groups,
                             unsigned int nr_groups);
 
 /**
  * samsung_register_cmu - Register CMU clocks ensuring parent CMU is present
  * @dev: CMU device
+ * @cmu_id: CMU index number
  * @clk_groups: list of CMU clock groups
  * @parent_drv: name of parent CMU driver
  *
@@ -194,7 +240,7 @@ int samsung_cmu_register_one(struct udevice *dev,
  *
  * Return: 0 on success or negative value on error.
  */
-#define samsung_register_cmu(dev, clk_groups, parent_drv)              \
+#define samsung_register_cmu(dev, cmu_id, clk_groups, parent_drv)      \
 ({                                                                     \
        struct udevice *__parent;                                       \
        int __ret;                                                      \
@@ -204,8 +250,8 @@ int samsung_cmu_register_one(struct udevice *dev,
        if (__ret || !__parent)                                         \
                __ret = -ENOENT;                                        \
        else                                                            \
-               __ret = samsung_cmu_register_one(dev, clk_groups,       \
-                       ARRAY_SIZE(clk_groups));                        \
+               __ret = samsung_cmu_register_one(dev, cmu_id,           \
+                       clk_groups, ARRAY_SIZE(clk_groups));            \
        __ret;                                                          \
 })