mirror of
https://github.com/openwrt/openwrt.git
synced 2025-01-07 06:18:54 +00:00
161 lines
4.6 KiB
Diff
161 lines
4.6 KiB
Diff
|
From patchwork Fri Dec 8 09:42:28 2017
|
||
|
Content-Type: text/plain; charset="utf-8"
|
||
|
MIME-Version: 1.0
|
||
|
Content-Transfer-Encoding: 7bit
|
||
|
Subject: [v4,10/12] clk: qcom: Add safe switch hook for krait mux clocks
|
||
|
From: Sricharan R <sricharan@codeaurora.org>
|
||
|
X-Patchwork-Id: 10102057
|
||
|
Message-Id: <1512726150-7204-11-git-send-email-sricharan@codeaurora.org>
|
||
|
To: mturquette@baylibre.com, sboyd@codeaurora.org,
|
||
|
devicetree@vger.kernel.org, linux-pm@vger.kernel.org,
|
||
|
linux-arm-msm@vger.kernel.org, linux-kernel@vger.kernel.org,
|
||
|
viresh.kumar@linaro.org, linux-arm-kernel@lists.infradead.org
|
||
|
Cc: sricharan@codeaurora.org
|
||
|
Date: Fri, 8 Dec 2017 15:12:28 +0530
|
||
|
|
||
|
When the Hfplls are reprogrammed during the rate change,
|
||
|
the primary muxes which are sourced from the same hfpll
|
||
|
for higher frequencies, needs to be switched to the 'safe
|
||
|
secondary mux' as the parent for that small window. This
|
||
|
is done by registering a clk notifier for the muxes and
|
||
|
switching to the safe parent in the PRE_RATE_CHANGE notifier
|
||
|
and back to the original parent in the POST_RATE_CHANGE notifier.
|
||
|
|
||
|
Signed-off-by: Sricharan R <sricharan@codeaurora.org>
|
||
|
---
|
||
|
drivers/clk/qcom/clk-krait.c | 2 ++
|
||
|
drivers/clk/qcom/clk-krait.h | 3 +++
|
||
|
drivers/clk/qcom/krait-cc.c | 56 ++++++++++++++++++++++++++++++++++++++++++++
|
||
|
3 files changed, 61 insertions(+)
|
||
|
|
||
|
--- a/drivers/clk/qcom/clk-krait.c
|
||
|
+++ b/drivers/clk/qcom/clk-krait.c
|
||
|
@@ -60,6 +60,8 @@ static int krait_mux_set_parent(struct c
|
||
|
if (__clk_is_enabled(hw->clk))
|
||
|
__krait_mux_set_sel(mux, sel);
|
||
|
|
||
|
+ mux->reparent = true;
|
||
|
+
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
--- a/drivers/clk/qcom/clk-krait.h
|
||
|
+++ b/drivers/clk/qcom/clk-krait.h
|
||
|
@@ -23,6 +23,9 @@ struct krait_mux_clk {
|
||
|
u32 shift;
|
||
|
u32 en_mask;
|
||
|
bool lpl;
|
||
|
+ u8 safe_sel;
|
||
|
+ u8 old_index;
|
||
|
+ bool reparent;
|
||
|
|
||
|
struct clk_hw hw;
|
||
|
struct notifier_block clk_nb;
|
||
|
--- a/drivers/clk/qcom/krait-cc.c
|
||
|
+++ b/drivers/clk/qcom/krait-cc.c
|
||
|
@@ -35,6 +35,49 @@ static unsigned int pri_mux_map[] = {
|
||
|
0,
|
||
|
};
|
||
|
|
||
|
+/*
|
||
|
+ * Notifier function for switching the muxes to safe parent
|
||
|
+ * while the hfpll is getting reprogrammed.
|
||
|
+ */
|
||
|
+static int krait_notifier_cb(struct notifier_block *nb,
|
||
|
+ unsigned long event,
|
||
|
+ void *data)
|
||
|
+{
|
||
|
+ int ret = 0;
|
||
|
+ struct krait_mux_clk *mux = container_of(nb, struct krait_mux_clk,
|
||
|
+ clk_nb);
|
||
|
+ /* Switch to safe parent */
|
||
|
+ if (event == PRE_RATE_CHANGE) {
|
||
|
+ mux->old_index = krait_mux_clk_ops.get_parent(&mux->hw);
|
||
|
+ ret = krait_mux_clk_ops.set_parent(&mux->hw, mux->safe_sel);
|
||
|
+ mux->reparent = false;
|
||
|
+ /*
|
||
|
+ * By the time POST_RATE_CHANGE notifier is called,
|
||
|
+ * clk framework itself would have changed the parent for the new rate.
|
||
|
+ * Only otherwise, put back to the old parent.
|
||
|
+ */
|
||
|
+ } else if (event == POST_RATE_CHANGE) {
|
||
|
+ if (!mux->reparent)
|
||
|
+ ret = krait_mux_clk_ops.set_parent(&mux->hw,
|
||
|
+ mux->old_index);
|
||
|
+ }
|
||
|
+
|
||
|
+ return notifier_from_errno(ret);
|
||
|
+}
|
||
|
+
|
||
|
+static int krait_notifier_register(struct device *dev, struct clk *clk,
|
||
|
+ struct krait_mux_clk *mux)
|
||
|
+{
|
||
|
+ int ret = 0;
|
||
|
+
|
||
|
+ mux->clk_nb.notifier_call = krait_notifier_cb;
|
||
|
+ ret = clk_notifier_register(clk, &mux->clk_nb);
|
||
|
+ if (ret)
|
||
|
+ dev_err(dev, "failed to register clock notifier: %d\n", ret);
|
||
|
+
|
||
|
+ return ret;
|
||
|
+}
|
||
|
+
|
||
|
static int
|
||
|
krait_add_div(struct device *dev, int id, const char *s, unsigned int offset)
|
||
|
{
|
||
|
@@ -79,6 +122,7 @@ static int
|
||
|
krait_add_sec_mux(struct device *dev, int id, const char *s,
|
||
|
unsigned int offset, bool unique_aux)
|
||
|
{
|
||
|
+ int ret;
|
||
|
struct krait_mux_clk *mux;
|
||
|
static const char *sec_mux_list[] = {
|
||
|
"acpu_aux",
|
||
|
@@ -102,6 +146,7 @@ krait_add_sec_mux(struct device *dev, in
|
||
|
mux->shift = 2;
|
||
|
mux->parent_map = sec_mux_map;
|
||
|
mux->hw.init = &init;
|
||
|
+ mux->safe_sel = 0;
|
||
|
|
||
|
init.name = kasprintf(GFP_KERNEL, "krait%s_sec_mux", s);
|
||
|
if (!init.name)
|
||
|
@@ -117,6 +162,11 @@ krait_add_sec_mux(struct device *dev, in
|
||
|
|
||
|
clk = devm_clk_register(dev, &mux->hw);
|
||
|
|
||
|
+ ret = krait_notifier_register(dev, clk, mux);
|
||
|
+ if (ret)
|
||
|
+ goto unique_aux;
|
||
|
+
|
||
|
+unique_aux:
|
||
|
if (unique_aux)
|
||
|
kfree(sec_mux_list[0]);
|
||
|
err_aux:
|
||
|
@@ -128,6 +178,7 @@ static struct clk *
|
||
|
krait_add_pri_mux(struct device *dev, int id, const char *s,
|
||
|
unsigned int offset)
|
||
|
{
|
||
|
+ int ret;
|
||
|
struct krait_mux_clk *mux;
|
||
|
const char *p_names[3];
|
||
|
struct clk_init_data init = {
|
||
|
@@ -148,6 +199,7 @@ krait_add_pri_mux(struct device *dev, in
|
||
|
mux->lpl = id >= 0;
|
||
|
mux->parent_map = pri_mux_map;
|
||
|
mux->hw.init = &init;
|
||
|
+ mux->safe_sel = 2;
|
||
|
|
||
|
init.name = kasprintf(GFP_KERNEL, "krait%s_pri_mux", s);
|
||
|
if (!init.name)
|
||
|
@@ -173,6 +225,10 @@ krait_add_pri_mux(struct device *dev, in
|
||
|
|
||
|
clk = devm_clk_register(dev, &mux->hw);
|
||
|
|
||
|
+ ret = krait_notifier_register(dev, clk, mux);
|
||
|
+ if (ret)
|
||
|
+ goto err_p3;
|
||
|
+err_p3:
|
||
|
kfree(p_names[2]);
|
||
|
err_p2:
|
||
|
kfree(p_names[1]);
|