Some devices like SoCs from Mediatek need to use the clock muxes
through a regmap interface.
This patch adds regmap support for the simple multiplexer
clock code.

Signed-off-by: Matthias Brugger <matthias....@gmail.com>
---
 drivers/clk/clk-mux.c        | 134 ++++++++++++++++++++++++++++++++++++++-----
 include/linux/clk-provider.h |  44 +++++++++++++-
 2 files changed, 162 insertions(+), 16 deletions(-)

diff --git a/drivers/clk/clk-mux.c b/drivers/clk/clk-mux.c
index 1fa2a8d..6f75116 100644
--- a/drivers/clk/clk-mux.c
+++ b/drivers/clk/clk-mux.c
@@ -29,6 +29,43 @@
 
 #define to_clk_mux(_hw) container_of(_hw, struct clk_mux, hw)
 
+static void clk_mux_writel(struct clk_mux *mux, u32 val)
+{
+       if (mux->flags & CLK_MUX_USE_REGMAP)
+               regmap_write(mux->regmap, mux->offset, val);
+       else
+               clk_writel(val, mux->reg);
+}
+
+static u32 clk_mux_readl(struct clk_mux *mux)
+{
+       u32 val;
+
+       if (mux->flags & CLK_MUX_USE_REGMAP)
+               regmap_read(mux->regmap, mux->offset, &val);
+       else
+               val = clk_readl(mux->reg);
+
+       return val;
+}
+
+static int clk_mux_update_bits(struct clk_mux *mux, u32 mask, u32 val)
+{
+       unsigned int ret;
+
+       if (mux->flags & CLK_MUX_USE_REGMAP) {
+               ret = regmap_update_bits(mux->regmap, mux->offset, mask, val);
+       } else {
+               ret = clk_readl(mux->reg);
+               ret &= ~mask;
+               ret |= val;
+               clk_writel(ret, mux->reg);
+               ret = 0;
+       }
+
+       return ret;
+}
+
 static u8 clk_mux_get_parent(struct clk_hw *hw)
 {
        struct clk_mux *mux = to_clk_mux(hw);
@@ -42,7 +79,10 @@ static u8 clk_mux_get_parent(struct clk_hw *hw)
         * OTOH, pmd_trace_clk_mux_ck uses a separate bit for each clock, so
         * val = 0x4 really means "bit 2, index starts at bit 0"
         */
-       val = clk_readl(mux->reg) >> mux->shift;
+
+       val = clk_mux_readl(mux);
+
+       val >>= mux->shift;
        val &= mux->mask;
 
        if (mux->table) {
@@ -71,6 +111,7 @@ static int clk_mux_set_parent(struct clk_hw *hw, u8 index)
        struct clk_mux *mux = to_clk_mux(hw);
        u32 val;
        unsigned long flags = 0;
+       int ret = 0;
 
        if (mux->table)
                index = mux->table[index];
@@ -88,17 +129,19 @@ static int clk_mux_set_parent(struct clk_hw *hw, u8 index)
 
        if (mux->flags & CLK_MUX_HIWORD_MASK) {
                val = mux->mask << (mux->shift + 16);
+               val |= index << mux->shift;
+               clk_mux_writel(mux, val);
        } else {
-               val = clk_readl(mux->reg);
-               val &= ~(mux->mask << mux->shift);
+               u32 mask = mux->mask << mux->shift;
+
+               val = index << mux->shift;
+               ret = clk_mux_update_bits(mux, mask, val);
        }
-       val |= index << mux->shift;
-       clk_writel(val, mux->reg);
 
        if (mux->lock)
                spin_unlock_irqrestore(mux->lock, flags);
 
-       return 0;
+       return ret;
 }
 
 const struct clk_ops clk_mux_ops = {
@@ -113,9 +156,10 @@ const struct clk_ops clk_mux_ro_ops = {
 };
 EXPORT_SYMBOL_GPL(clk_mux_ro_ops);
 
-struct clk *clk_register_mux_table(struct device *dev, const char *name,
+struct clk *__clk_register_mux_table(struct device *dev, const char *name,
                const char * const *parent_names, u8 num_parents, unsigned long 
flags,
-               void __iomem *reg, u8 shift, u32 mask,
+               void __iomem *reg, struct regmap *regmap,
+               u32 offset, u8 shift, u32 mask,
                u8 clk_mux_flags, u32 *table, spinlock_t *lock)
 {
        struct clk_mux *mux;
@@ -148,7 +192,12 @@ struct clk *clk_register_mux_table(struct device *dev, 
const char *name,
        init.num_parents = num_parents;
 
        /* struct clk_mux assignments */
-       mux->reg = reg;
+       if (clk_mux_flags & CLK_MUX_USE_REGMAP)
+               mux->regmap = regmap;
+       else
+               mux->reg = reg;
+
+       mux->offset = offset;
        mux->shift = shift;
        mux->mask = mask;
        mux->flags = clk_mux_flags;
@@ -163,18 +212,40 @@ struct clk *clk_register_mux_table(struct device *dev, 
const char *name,
 
        return clk;
 }
+
+struct clk *clk_register_mux_table(struct device *dev, const char *name,
+               const char * const *parent_names, u8 num_parents,
+               unsigned long flags, void __iomem *reg, u8 shift, u32 mask,
+               u8 clk_mux_flags, u32 *table, spinlock_t *lock)
+{
+       return __clk_register_mux_table(dev, name, parent_names, num_parents,
+                                               flags, reg, NULL, 0,
+                                               shift, mask, clk_mux_flags,
+                                               table, lock);
+}
 EXPORT_SYMBOL_GPL(clk_register_mux_table);
 
-struct clk *clk_register_mux(struct device *dev, const char *name,
-               const char * const *parent_names, u8 num_parents, unsigned long 
flags,
-               void __iomem *reg, u8 shift, u8 width,
+struct clk *__clk_register_mux(struct device *dev, const char *name,
+               const char * const *parent_names, u8 num_parents,
+               unsigned long flags, void __iomem *reg, struct regmap *regmap,
+               u32 offset, u8 shift, u8 width,
                u8 clk_mux_flags, spinlock_t *lock)
 {
        u32 mask = BIT(width) - 1;
 
-       return clk_register_mux_table(dev, name, parent_names, num_parents,
-                                     flags, reg, shift, mask, clk_mux_flags,
-                                     NULL, lock);
+       return __clk_register_mux_table(dev, name, parent_names, num_parents,
+                                       flags, reg, regmap, offset, shift, mask,
+                                       clk_mux_flags, NULL, lock);
+}
+
+struct clk *clk_register_mux(struct device *dev, const char *name,
+               const char * const *parent_names, u8 num_parents,
+               unsigned long flags, void __iomem *reg, u8 shift, u8 width,
+               u8 clk_mux_flags, spinlock_t *lock)
+{
+       return __clk_register_mux(dev, name, parent_names, num_parents, flags,
+                                       reg, NULL, 0, shift, width,
+                                       clk_mux_flags, lock);
 }
 EXPORT_SYMBOL_GPL(clk_register_mux);
 
@@ -193,3 +264,36 @@ void clk_unregister_mux(struct clk *clk)
        kfree(mux);
 }
 EXPORT_SYMBOL_GPL(clk_unregister_mux);
+
+#ifdef CONFIG_REGMAP
+
+struct clk *clk_regm_register_mux(struct device *dev, const char *name,
+               const char * const *parent_names, u8 num_parents,
+               unsigned long flags, struct regmap *regmap,
+               u32 offset, u8 shift, u8 width,
+               u8 clk_mux_flags, spinlock_t *lock)
+{
+       clk_mux_flags |= CLK_MUX_USE_REGMAP;
+
+       return __clk_register_mux(dev, name, parent_names, num_parents, flags,
+                                       NULL, regmap, offset, shift, width,
+                                       clk_mux_flags, lock);
+}
+EXPORT_SYMBOL_GPL(clk_regm_register_mux);
+
+struct clk *clk_regm_register_mux_table(struct device *dev, const char *name,
+               const char * const *parent_names, u8 num_parents,
+               unsigned long flags, struct regmap *regmap,
+               u32 offset, u8 shift, u32 mask,
+               u8 clk_mux_flags, u32 *table, spinlock_t *lock)
+{
+       clk_mux_flags |= CLK_MUX_USE_REGMAP;
+
+       return __clk_register_mux_table(dev, name, parent_names, num_parents,
+                                       flags, NULL, regmap, offset,
+                                       shift, mask, clk_mux_flags,
+                                       table, lock);
+}
+EXPORT_SYMBOL_GPL(clk_regm_register_mux_table);
+
+#endif
diff --git a/include/linux/clk-provider.h b/include/linux/clk-provider.h
index ec609e5..c7487ad 100644
--- a/include/linux/clk-provider.h
+++ b/include/linux/clk-provider.h
@@ -14,6 +14,7 @@
 #include <linux/clk.h>
 #include <linux/io.h>
 #include <linux/of.h>
+#include <linux/regmap.h>
 
 #ifdef CONFIG_COMMON_CLK
 
@@ -408,8 +409,12 @@ void clk_unregister_divider(struct clk *clk);
  */
 struct clk_mux {
        struct clk_hw   hw;
-       void __iomem    *reg;
+       union {
+               void __iomem    *reg;
+               struct regmap   *regmap;
+       };
        u32             *table;
+       u32             offset;
        u32             mask;
        u8              shift;
        u8              flags;
@@ -421,6 +426,7 @@ struct clk_mux {
 #define CLK_MUX_HIWORD_MASK            BIT(2)
 #define CLK_MUX_READ_ONLY              BIT(3) /* mux can't be changed */
 #define CLK_MUX_ROUND_CLOSEST          BIT(4)
+#define CLK_MUX_USE_REGMAP             BIT(5)
 
 extern const struct clk_ops clk_mux_ops;
 extern const struct clk_ops clk_mux_ro_ops;
@@ -437,6 +443,42 @@ struct clk *clk_register_mux_table(struct device *dev, 
const char *name,
 
 void clk_unregister_mux(struct clk *clk);
 
+#ifdef CONFIG_REGMAP
+
+struct clk *clk_regm_register_mux_table(struct device *dev, const char *name,
+               const char * const *parent_names, u8 num_parents,
+               unsigned long flags, struct regmap *regmap,
+               u32 offset, u8 shift, u32 mask,
+               u8 clk_mux_flags, u32 *table, spinlock_t *lock);
+
+struct clk *clk_regm_register_mux(struct device *dev, const char *name,
+               const char * const *parent_names, u8 num_parents,
+               unsigned long flags, struct regmap *regmap,
+               u32 offset, u8 shift, u8 width,
+               u8 clk_mux_flags, spinlock_t *lock);
+
+#else
+
+struct clk *clk_regm_register_mux_table(struct device *dev, const char *name,
+               const char * const *parent_names, u8 num_parents,
+               unsigned long flags, struct regmap *reg,
+               u32 offset, u8 shift, u32 mask,
+               u8 clk_mux_flags, u32 *table, spinlock_t *lock)
+{
+       return NULL;
+}
+
+struct clk *clk_regm_register_mux(struct device *dev, const char *name,
+               const char * const *parent_names, u8 num_parents,
+               unsigned long flags, struct regmap *reg,
+               u32 offset, u8 shift, u8 width,
+               u8 clk_mux_flags, spinlock_t *lock)
+{
+       return NULL;
+}
+
+#endif
+
 void of_fixed_factor_clk_setup(struct device_node *node);
 
 /**
-- 
1.9.1

--
To unsubscribe from this list: send the line "unsubscribe linux-kernel" in
the body of a message to majord...@vger.kernel.org
More majordomo info at  http://vger.kernel.org/majordomo-info.html
Please read the FAQ at  http://www.tux.org/lkml/

Reply via email to