[PATCH 2/2] clk: Add support for rate table based dividers

Turquette, Mike mturquette at ti.com
Tue May 29 17:15:43 EDT 2012


On Thu, May 17, 2012 at 3:22 AM, Rajendra Nayak <rnayak at ti.com> wrote:
> Some divider clks do not have any obvious relationship
> between the divider and the value programmed in the
> register. For instance, say a value of 1 could signify divide
> by 6 and a value of 2 could signify divide by 4 etc.
> Also there are dividers where not all values possible
> based on the bitfield width are valid. For instance
> a 3 bit wide bitfield can be used to program a value
> from 0 to 7. However its possible that only 0 to 4
> are valid values.
>
> All these cases need the platform code to pass a simple
> table of divider/value tuple, so the framework knows
> the exact value to be written based on the divider
> calculation and can also do better error checking.
>
> This patch adds support for such rate table based
> dividers.
>
> Signed-off-by: Rajendra Nayak <rnayak at ti.com>

Thanks Rajendra.  I'll take this into my clk-next.

Regards,
Mike

> ---
>  drivers/clk/clk-divider.c    |   67 ++++++++++++++++++++++++++++++++++++++++--
>  include/linux/clk-private.h  |    3 +-
>  include/linux/clk-provider.h |   10 +++++-
>  3 files changed, 75 insertions(+), 5 deletions(-)
>
> diff --git a/drivers/clk/clk-divider.c b/drivers/clk/clk-divider.c
> index e548c43..e4911ee 100644
> --- a/drivers/clk/clk-divider.c
> +++ b/drivers/clk/clk-divider.c
> @@ -32,30 +32,69 @@
>  #define div_mask(d)    ((1 << (d->width)) - 1)
>  #define is_power_of_two(i)     !(i & ~i)
>
> +static unsigned int _get_table_maxdiv(const struct clk_div_table *table)
> +{
> +       unsigned int maxdiv;
> +       const struct clk_div_table *clkt;
> +
> +       for (clkt = table; clkt->div; clkt++)
> +               if (clkt->div > maxdiv)
> +                       maxdiv = clkt->div;
> +       return maxdiv;
> +}
> +
>  static unsigned int _get_maxdiv(struct clk_divider *divider)
>  {
>        if (divider->flags & CLK_DIVIDER_ONE_BASED)
>                return div_mask(divider);
>        if (divider->flags & CLK_DIVIDER_POWER_OF_TWO)
>                return 1 << div_mask(divider);
> +       if (divider->table)
> +               return _get_table_maxdiv(divider->table);
>        return div_mask(divider) + 1;
>  }
>
> +static unsigned int _get_table_div(const struct clk_div_table *table,
> +                                                       unsigned int val)
> +{
> +       const struct clk_div_table *clkt;
> +
> +       for (clkt = table; clkt->div; clkt++)
> +               if (clkt->val == val)
> +                       return clkt->div;
> +       return 0;
> +}
> +
>  static unsigned int _get_div(struct clk_divider *divider, unsigned int val)
>  {
>        if (divider->flags & CLK_DIVIDER_ONE_BASED)
>                return val;
>        if (divider->flags & CLK_DIVIDER_POWER_OF_TWO)
>                return 1 << val;
> +       if (divider->table)
> +               return _get_table_div(divider->table, val);
>        return val + 1;
>  }
>
> +static unsigned int _get_table_val(const struct clk_div_table *table,
> +                                                       unsigned int div)
> +{
> +       const struct clk_div_table *clkt;
> +
> +       for (clkt = table; clkt->div; clkt++)
> +               if (clkt->div == div)
> +                       return clkt->val;
> +       return 0;
> +}
> +
>  static unsigned int _get_val(struct clk_divider *divider, u8 div)
>  {
>        if (divider->flags & CLK_DIVIDER_ONE_BASED)
>                return div;
>        if (divider->flags & CLK_DIVIDER_POWER_OF_TWO)
>                return __ffs(div);
> +       if (divider->table)
> +               return  _get_table_val(divider->table, div);
>        return div - 1;
>  }
>
> @@ -84,6 +123,26 @@ static unsigned long clk_divider_recalc_rate(struct clk_hw *hw,
>  */
>  #define MULT_ROUND_UP(r, m) ((r) * (m) + (m) - 1)
>
> +static bool _is_valid_table_div(const struct clk_div_table *table,
> +                                                        unsigned int div)
> +{
> +       const struct clk_div_table *clkt;
> +
> +       for (clkt = table; clkt->div; clkt++)
> +               if (clkt->div == div)
> +                       return true;
> +       return false;
> +}
> +
> +static bool _is_valid_div(struct clk_divider *divider, unsigned int div)
> +{
> +       if (divider->flags & CLK_DIVIDER_POWER_OF_TWO)
> +               return is_power_of_two(div);
> +       if (divider->table)
> +               return _is_valid_table_div(divider->table, div);
> +       return true;
> +}
> +
>  static int clk_divider_bestdiv(struct clk_hw *hw, unsigned long rate,
>                unsigned long *best_parent_rate)
>  {
> @@ -111,8 +170,7 @@ static int clk_divider_bestdiv(struct clk_hw *hw, unsigned long rate,
>        maxdiv = min(ULONG_MAX / rate, maxdiv);
>
>        for (i = 1; i <= maxdiv; i++) {
> -               if ((divider->flags & CLK_DIVIDER_POWER_OF_TWO)
> -                       && (!is_power_of_two(i)))
> +               if (!_is_valid_div(divider, i))
>                        continue;
>                parent_rate = __clk_round_rate(__clk_get_parent(hw->clk),
>                                MULT_ROUND_UP(rate, i));
> @@ -186,12 +244,14 @@ EXPORT_SYMBOL_GPL(clk_divider_ops);
>  * @shift: number of bits to shift the bitfield
>  * @width: width of the bitfield
>  * @clk_divider_flags: divider-specific flags for this clock
> + * @table: array of divider/value pairs ending with a div set to 0
>  * @lock: shared register lock for this clock
>  */
>  struct clk *clk_register_divider(struct device *dev, const char *name,
>                const char *parent_name, unsigned long flags,
>                void __iomem *reg, u8 shift, u8 width,
> -               u8 clk_divider_flags, spinlock_t *lock)
> +               u8 clk_divider_flags, const struct clk_div_table *table,
> +               spinlock_t *lock)
>  {
>        struct clk_divider *div;
>        struct clk *clk;
> @@ -217,6 +277,7 @@ struct clk *clk_register_divider(struct device *dev, const char *name,
>        div->flags = clk_divider_flags;
>        div->lock = lock;
>        div->hw.init = &init;
> +       div->table = table;
>
>        /* register the clock */
>        clk = clk_register(dev, &div->hw);
> diff --git a/include/linux/clk-private.h b/include/linux/clk-private.h
> index eb3f84b..2479239 100644
> --- a/include/linux/clk-private.h
> +++ b/include/linux/clk-private.h
> @@ -105,7 +105,7 @@ struct clk {
>
>  #define DEFINE_CLK_DIVIDER(_name, _parent_name, _parent_ptr,   \
>                                _flags, _reg, _shift, _width,   \
> -                               _divider_flags, _lock)          \
> +                               _divider_flags, _table, _lock)  \
>        static struct clk _name;                                \
>        static const char *_name##_parent_names[] = {           \
>                _parent_name,                                   \
> @@ -121,6 +121,7 @@ struct clk {
>                .shift = _shift,                                \
>                .width = _width,                                \
>                .flags = _divider_flags,                        \
> +               .table = _table,                                \
>                .lock = _lock,                                  \
>        };                                                      \
>        DEFINE_CLK(_name, clk_divider_ops, _flags,              \
> diff --git a/include/linux/clk-provider.h b/include/linux/clk-provider.h
> index 4a0b483..22bc067 100644
> --- a/include/linux/clk-provider.h
> +++ b/include/linux/clk-provider.h
> @@ -203,6 +203,11 @@ struct clk *clk_register_gate(struct device *dev, const char *name,
>                void __iomem *reg, u8 bit_idx,
>                u8 clk_gate_flags, spinlock_t *lock);
>
> +struct clk_div_table {
> +       unsigned int    val;
> +       unsigned int    div;
> +};
> +
>  /**
>  * struct clk_divider - adjustable divider clock
>  *
> @@ -210,6 +215,7 @@ struct clk *clk_register_gate(struct device *dev, const char *name,
>  * @reg:       register containing the divider
>  * @shift:     shift to the divider bit field
>  * @width:     width of the divider bit field
> + * @table:     array of value/divider pairs, last entry should have div = 0
>  * @lock:      register lock
>  *
>  * Clock with an adjustable divider affecting its output frequency.  Implements
> @@ -229,6 +235,7 @@ struct clk_divider {
>        u8              shift;
>        u8              width;
>        u8              flags;
> +       const struct clk_div_table      *table;
>        spinlock_t      *lock;
>  };
>
> @@ -239,7 +246,8 @@ extern const struct clk_ops clk_divider_ops;
>  struct clk *clk_register_divider(struct device *dev, const char *name,
>                const char *parent_name, unsigned long flags,
>                void __iomem *reg, u8 shift, u8 width,
> -               u8 clk_divider_flags, spinlock_t *lock);
> +               u8 clk_divider_flags, const struct clk_div_table *table,
> +               spinlock_t *lock);
>
>  /**
>  * struct clk_mux - multiplexer clock
> --
> 1.7.1
>



More information about the linux-arm-kernel mailing list