From: Neil Armstrong <narmstrong@baylibre.com>
To: Jerome Brunet <jbrunet@baylibre.com>,
Carlo Caione <carlo@caione.org>,
Kevin Hilman <khilman@baylibre.com>
Cc: Michael Turquette <mturquette@baylibre.com>,
Stephen Boyd <sboyd@kernel.org>,
linux-amlogic@lists.infradead.org, linux-clk@vger.kernel.org,
devicetree@vger.kernel.org, linux-kernel@vger.kernel.org
Subject: Re: [PATCH 4/7] clk: meson: add axg audio sclk divider driver
Date: Thu, 26 Apr 2018 10:47:38 +0200 [thread overview]
Message-ID: <d0435ffb-1101-6f1b-b2de-1dc74f6a9543@baylibre.com> (raw)
In-Reply-To: <20180425163304.10852-5-jbrunet@baylibre.com>
On 25/04/2018 18:33, Jerome Brunet wrote:
> Add a driver to control the clock divider found in the sample clock
> generator of the axg audio clock controller.
>
> The sclk divider accumulates specific features which make the generic
> divider unsuitable to control it:
> - zero based divider (div = val + 1), but zero value gates the clock,
> so minimum divider value is 2.
> - lrclk variant may adjust the duty cycle depending the divider value
> and the 'hi' value.
>
> Signed-off-by: Jerome Brunet <jbrunet@baylibre.com>
> ---
> drivers/clk/meson/Makefile | 2 +-
> drivers/clk/meson/clkc-audio.h | 8 ++
> drivers/clk/meson/sclk-div.c | 243 +++++++++++++++++++++++++++++++++++++++++
> 3 files changed, 252 insertions(+), 1 deletion(-)
> create mode 100644 drivers/clk/meson/sclk-div.c
>
> diff --git a/drivers/clk/meson/Makefile b/drivers/clk/meson/Makefile
> index 64bb917fe1f0..f51b4754c31b 100644
> --- a/drivers/clk/meson/Makefile
> +++ b/drivers/clk/meson/Makefile
> @@ -4,7 +4,7 @@
>
> obj-$(CONFIG_COMMON_CLK_AMLOGIC) += clk-pll.o clk-mpll.o clk-audio-divider.o
> obj-$(CONFIG_COMMON_CLK_AMLOGIC) += clk-phase.o
> -obj-$(CONFIG_COMMON_CLK_AMLOGIC_AUDIO) += clk-triphase.o
> +obj-$(CONFIG_COMMON_CLK_AMLOGIC_AUDIO) += clk-triphase.o sclk-div.o
> obj-$(CONFIG_COMMON_CLK_MESON8B) += meson8b.o
> obj-$(CONFIG_COMMON_CLK_GXBB) += gxbb.o gxbb-aoclk.o gxbb-aoclk-32k.o
> obj-$(CONFIG_COMMON_CLK_AXG) += axg.o
> diff --git a/drivers/clk/meson/clkc-audio.h b/drivers/clk/meson/clkc-audio.h
> index 286ff1201258..0a7c157ebf81 100644
> --- a/drivers/clk/meson/clkc-audio.h
> +++ b/drivers/clk/meson/clkc-audio.h
> @@ -15,6 +15,14 @@ struct meson_clk_triphase_data {
> struct parm ph2;
> };
>
> +struct meson_sclk_div_data {
> + struct parm div;
> + struct parm hi;
> + unsigned int cached_div;
> + struct clk_duty cached_duty;
> +};
> +
> extern const struct clk_ops meson_clk_triphase_ops;
> +extern const struct clk_ops meson_sclk_div_ops;
>
> #endif /* __MESON_CLKC_AUDIO_H */
> diff --git a/drivers/clk/meson/sclk-div.c b/drivers/clk/meson/sclk-div.c
> new file mode 100644
> index 000000000000..8c0bc914a6d7
> --- /dev/null
> +++ b/drivers/clk/meson/sclk-div.c
> @@ -0,0 +1,243 @@
> +// SPDX-License-Identifier: (GPL-2.0+ OR MIT)
> +/*
> + * Copyright (c) 2018 BayLibre, SAS.
> + * Author: Jerome Brunet <jbrunet@baylibre.com>
> + *
> + * Sample clock generator divider:
> + * This HW divider gates with value 0 but is otherwise a zero based divider:
> + *
> + * val >= 1
> + * divider = val + 1
> + *
> + * The duty cycle may also be set for the LR clock variant. The duty cycle
> + * ratio is:
> + *
> + * hi = [0 - val]
> + * duty_cycle = (1 + hi) / (1 + val)
> + */
> +
> +#include "clkc-audio.h"
> +
> +static inline struct meson_sclk_div_data *
> +meson_sclk_div_data(struct clk_regmap *clk)
> +{
> + return (struct meson_sclk_div_data *)clk->data;
> +}
> +
> +static int sclk_div_maxval(struct meson_sclk_div_data *sclk)
> +{
> + return (1 << sclk->div.width) - 1;
> +}
> +
> +static int sclk_div_maxdiv(struct meson_sclk_div_data *sclk)
> +{
> + return sclk_div_maxval(sclk) + 1;
> +}
> +
> +static int sclk_div_getdiv(struct clk_hw *hw, unsigned long rate,
> + unsigned long prate, int maxdiv)
> +{
> + int div = DIV_ROUND_CLOSEST_ULL((u64)prate, rate);
> +
> + return clamp(div, 2, maxdiv);
> +}
> +
> +static int sclk_div_bestdiv(struct clk_hw *hw, unsigned long rate,
> + unsigned long *prate,
> + struct meson_sclk_div_data *sclk)
> +{
> + struct clk_hw *parent = clk_hw_get_parent(hw);
> + int bestdiv = 0, i;
> + unsigned long maxdiv, now, parent_now;
> + unsigned long best = 0, best_parent = 0;
> +
> + if (!rate)
> + rate = 1;
> +
> + maxdiv = sclk_div_maxdiv(sclk);
> +
> + if (!(clk_hw_get_flags(hw) & CLK_SET_RATE_PARENT))
> + return sclk_div_getdiv(hw, rate, *prate, maxdiv);
> +
> + /*
> + * The maximum divider we can use without overflowing
> + * unsigned long in rate * i below
> + */
> + maxdiv = min(ULONG_MAX / rate, maxdiv);
> +
> + for (i = 2; i <= maxdiv; i++) {
> + /*
> + * It's the most ideal case if the requested rate can be
> + * divided from parent clock without needing to change
> + * parent rate, so return the divider immediately.
> + */
> + if (rate * i == *prate)
> + return i;
> +
> + parent_now = clk_hw_round_rate(parent, rate * i);
> + now = DIV_ROUND_UP_ULL((u64)parent_now, i);
> +
> + if (abs(rate - now) < abs(rate - best)) {
> + bestdiv = i;
> + best = now;
> + best_parent = parent_now;
> + }
> + }
> +
> + if (!bestdiv)
> + bestdiv = sclk_div_maxdiv(sclk);
> + else
> + *prate = best_parent;
> +
> + return bestdiv;
> +}
> +
> +static long sclk_div_round_rate(struct clk_hw *hw, unsigned long rate,
> + unsigned long *prate)
> +{
> + struct clk_regmap *clk = to_clk_regmap(hw);
> + struct meson_sclk_div_data *sclk = meson_sclk_div_data(clk);
> + int div;
> +
> + div = sclk_div_bestdiv(hw, rate, prate, sclk);
> +
> + return DIV_ROUND_UP_ULL((u64)*prate, div);
> +}
> +
> +static void sclk_apply_ratio(struct clk_regmap *clk,
> + struct meson_sclk_div_data *sclk)
> +{
> + unsigned int hi = DIV_ROUND_CLOSEST(sclk->cached_div *
> + sclk->cached_duty.num,
> + sclk->cached_duty.den);
> +
> + if (hi)
> + hi -= 1;
> +
> + meson_parm_write(clk->map, &sclk->hi, hi);
> +}
> +
> +static int sclk_div_set_duty_cycle(struct clk_hw *hw,
> + struct clk_duty *duty)
> +{
> + struct clk_regmap *clk = to_clk_regmap(hw);
> + struct meson_sclk_div_data *sclk = meson_sclk_div_data(clk);
> +
> + if (MESON_PARM_APPLICABLE(&sclk->hi)) {
> + memcpy(&sclk->cached_duty, duty, sizeof(*duty));
> + sclk_apply_ratio(clk, sclk);
> + }
> +
> + return 0;
> +}
> +
> +static int sclk_div_get_duty_cycle(struct clk_hw *hw,
> + struct clk_duty *duty)
> +{
> + struct clk_regmap *clk = to_clk_regmap(hw);
> + struct meson_sclk_div_data *sclk = meson_sclk_div_data(clk);
> + int hi;
> +
> + if (!MESON_PARM_APPLICABLE(&sclk->hi)) {
> + duty->num = 1;
> + duty->den = 2;
> + return 0;
> + }
> +
> + hi = meson_parm_read(clk->map, &sclk->hi);
> + duty->num = hi + 1;
> + duty->den = sclk->cached_div;
> + return 0;
> +}
> +
> +static void sclk_apply_divider(struct clk_regmap *clk,
> + struct meson_sclk_div_data *sclk)
> +{
> + if (MESON_PARM_APPLICABLE(&sclk->hi))
> + sclk_apply_ratio(clk, sclk);
> +
> + meson_parm_write(clk->map, &sclk->div, sclk->cached_div - 1);
> +}
> +
> +static int sclk_div_set_rate(struct clk_hw *hw, unsigned long rate,
> + unsigned long prate)
> +{
> + struct clk_regmap *clk = to_clk_regmap(hw);
> + struct meson_sclk_div_data *sclk = meson_sclk_div_data(clk);
> + unsigned long maxdiv = sclk_div_maxdiv(sclk);
> +
> + sclk->cached_div = sclk_div_getdiv(hw, rate, prate, maxdiv);
> +
> + if (clk_hw_is_enabled(hw))
> + sclk_apply_divider(clk, sclk);
> +
> + return 0;
> +}
> +
> +static unsigned long sclk_div_recalc_rate(struct clk_hw *hw,
> + unsigned long prate)
> +{
> + struct clk_regmap *clk = to_clk_regmap(hw);
> + struct meson_sclk_div_data *sclk = meson_sclk_div_data(clk);
> +
> + return DIV_ROUND_UP_ULL((u64)prate, sclk->cached_div);
> +}
> +
> +static int sclk_div_enable(struct clk_hw *hw)
> +{
> + struct clk_regmap *clk = to_clk_regmap(hw);
> + struct meson_sclk_div_data *sclk = meson_sclk_div_data(clk);
> +
> + sclk_apply_divider(clk, sclk);
> +
> + return 0;
> +}
> +
> +static void sclk_div_disable(struct clk_hw *hw)
> +{
> + struct clk_regmap *clk = to_clk_regmap(hw);
> + struct meson_sclk_div_data *sclk = meson_sclk_div_data(clk);
> +
> + meson_parm_write(clk->map, &sclk->div, 0);
> +}
> +
> +static int sclk_div_is_enabled(struct clk_hw *hw)
> +{
> + struct clk_regmap *clk = to_clk_regmap(hw);
> + struct meson_sclk_div_data *sclk = meson_sclk_div_data(clk);
> +
> + if (meson_parm_read(clk->map, &sclk->div))
> + return 1;
> +
> + return 0;
> +}
> +
> +static void sclk_div_init(struct clk_hw *hw)
> +{
> + struct clk_regmap *clk = to_clk_regmap(hw);
> + struct meson_sclk_div_data *sclk = meson_sclk_div_data(clk);
> + unsigned int val;
> +
> + val = meson_parm_read(clk->map, &sclk->div);
> +
> + /* if the divider is initially disabled, assume max */
> + if (!val)
> + sclk->cached_div = sclk_div_maxdiv(sclk);
> + else
> + sclk->cached_div = val + 1;
> +
> + sclk_div_get_duty_cycle(hw, &sclk->cached_duty);
> +}
> +
> +const struct clk_ops meson_sclk_div_ops = {
> + .recalc_rate = sclk_div_recalc_rate,
> + .round_rate = sclk_div_round_rate,
> + .set_rate = sclk_div_set_rate,
> + .enable = sclk_div_enable,
> + .disable = sclk_div_disable,
> + .is_enabled = sclk_div_is_enabled,
> + .get_duty_cycle = sclk_div_get_duty_cycle,
> + .set_duty_cycle = sclk_div_set_duty_cycle,
> + .init = sclk_div_init,
> +};
> +EXPORT_SYMBOL_GPL(meson_sclk_div_ops);
>
Acked-by: Neil Armstrong <narmstrong@baylibre.com>
next prev parent reply other threads:[~2018-04-26 8:47 UTC|newest]
Thread overview: 20+ messages / expand[flat|nested] mbox.gz Atom feed top
2018-04-25 16:32 [PATCH 0/7] clk: meson: axg: add audio clock controller support Jerome Brunet
2018-04-25 16:32 ` [PATCH 1/7] clk: meson: clean-up meson clock configuration Jerome Brunet
2018-04-26 8:46 ` Neil Armstrong
2018-04-25 16:32 ` [PATCH 2/7] clk: meson: add clk-phase clock driver Jerome Brunet
2018-04-26 8:46 ` Neil Armstrong
2018-04-25 16:33 ` [PATCH 3/7] clk: meson: add triple phase " Jerome Brunet
2018-04-26 8:47 ` Neil Armstrong
2018-04-26 8:50 ` Neil Armstrong
2018-04-25 16:33 ` [PATCH 4/7] clk: meson: add axg audio sclk divider driver Jerome Brunet
2018-04-26 8:47 ` Neil Armstrong [this message]
2018-04-25 16:33 ` [PATCH 5/7] clk: meson: axg: export audio clock controller id bindings Jerome Brunet
2018-04-26 8:48 ` Neil Armstrong
2018-05-01 14:31 ` Rob Herring
2018-04-25 16:33 ` [PATCH 6/7] clk: meson: axg: document bindings for the audio clock controller Jerome Brunet
2018-05-01 14:37 ` Rob Herring
2018-05-14 14:16 ` Jerome Brunet
2018-04-25 16:33 ` [PATCH 7/7] clk: meson: axg: add the audio clock controller driver Jerome Brunet
2018-04-26 8:49 ` Neil Armstrong
2018-05-15 23:41 ` Stephen Boyd
2018-04-27 1:13 ` kbuild test robot
Reply instructions:
You may reply publicly to this message via plain-text email
using any one of the following methods:
* Save the following mbox file, import it into your mail client,
and reply-to-all from there: mbox
Avoid top-posting and favor interleaved quoting:
https://en.wikipedia.org/wiki/Posting_style#Interleaved_style
* Reply using the --to, --cc, and --in-reply-to
switches of git-send-email(1):
git send-email \
--in-reply-to=d0435ffb-1101-6f1b-b2de-1dc74f6a9543@baylibre.com \
--to=narmstrong@baylibre.com \
--cc=carlo@caione.org \
--cc=devicetree@vger.kernel.org \
--cc=jbrunet@baylibre.com \
--cc=khilman@baylibre.com \
--cc=linux-amlogic@lists.infradead.org \
--cc=linux-clk@vger.kernel.org \
--cc=linux-kernel@vger.kernel.org \
--cc=mturquette@baylibre.com \
--cc=sboyd@kernel.org \
/path/to/YOUR_REPLY
https://kernel.org/pub/software/scm/git/docs/git-send-email.html
* If your mail client supports setting the In-Reply-To header
via mailto: links, try the mailto: link
Be sure your reply has a Subject: header at the top and a blank line
before the message body.
This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox