aboutsummaryrefslogtreecommitdiffstats
path: root/drivers/clk/qcom
diff options
context:
space:
mode:
authorRajendra Nayak <rnayak@codeaurora.org>2015-12-01 11:12:12 -0500
committerStephen Boyd <sboyd@codeaurora.org>2016-02-11 19:24:03 -0500
commit77b1067a19b4986b009f3279cc6b8ad1d29ff51c (patch)
tree0cbe30f554030b78b0bf9779a5d554aeb4f6e43a /drivers/clk/qcom
parentc2c7f0a47493ae23f9a76fabdbdd4f25e1de0925 (diff)
clk: qcom: gdsc: Add support for gdscs with gds hw controller
Some gdsc power domains can have a gds_hw_controller block inside to help ensure all slave devices within the power domain are idle before the gdsc is actually switched off. This is mainly useful in power domains which host a MMU, in which case its necessary to make sure there are no outstanding MMU operations or pending bus transactions before the power domain is turned off. In gdscs with gds_hw_controller block, its necessary to check the gds_hw_ctrl status bits instead of the ones in gdscr, to determine the state of the powerdomain. While at it, also move away from using jiffies and use ktime APIs instead for busy looping on status bits. Signed-off-by: Rajendra Nayak <rnayak@codeaurora.org> Signed-off-by: Stephen Boyd <sboyd@codeaurora.org>
Diffstat (limited to 'drivers/clk/qcom')
-rw-r--r--drivers/clk/qcom/gdsc.c44
-rw-r--r--drivers/clk/qcom/gdsc.h2
2 files changed, 29 insertions, 17 deletions
diff --git a/drivers/clk/qcom/gdsc.c b/drivers/clk/qcom/gdsc.c
index bfab75bd272c..9f530b73bd3e 100644
--- a/drivers/clk/qcom/gdsc.c
+++ b/drivers/clk/qcom/gdsc.c
@@ -16,6 +16,7 @@
16#include <linux/err.h> 16#include <linux/err.h>
17#include <linux/jiffies.h> 17#include <linux/jiffies.h>
18#include <linux/kernel.h> 18#include <linux/kernel.h>
19#include <linux/ktime.h>
19#include <linux/pm_domain.h> 20#include <linux/pm_domain.h>
20#include <linux/regmap.h> 21#include <linux/regmap.h>
21#include <linux/reset-controller.h> 22#include <linux/reset-controller.h>
@@ -42,12 +43,12 @@
42 43
43#define domain_to_gdsc(domain) container_of(domain, struct gdsc, pd) 44#define domain_to_gdsc(domain) container_of(domain, struct gdsc, pd)
44 45
45static int gdsc_is_enabled(struct gdsc *sc) 46static int gdsc_is_enabled(struct gdsc *sc, unsigned int reg)
46{ 47{
47 u32 val; 48 u32 val;
48 int ret; 49 int ret;
49 50
50 ret = regmap_read(sc->regmap, sc->gdscr, &val); 51 ret = regmap_read(sc->regmap, reg, &val);
51 if (ret) 52 if (ret)
52 return ret; 53 return ret;
53 54
@@ -58,28 +59,35 @@ static int gdsc_toggle_logic(struct gdsc *sc, bool en)
58{ 59{
59 int ret; 60 int ret;
60 u32 val = en ? 0 : SW_COLLAPSE_MASK; 61 u32 val = en ? 0 : SW_COLLAPSE_MASK;
61 u32 check = en ? PWR_ON_MASK : 0; 62 ktime_t start;
62 unsigned long timeout; 63 unsigned int status_reg = sc->gdscr;
63 64
64 ret = regmap_update_bits(sc->regmap, sc->gdscr, SW_COLLAPSE_MASK, val); 65 ret = regmap_update_bits(sc->regmap, sc->gdscr, SW_COLLAPSE_MASK, val);
65 if (ret) 66 if (ret)
66 return ret; 67 return ret;
67 68
68 timeout = jiffies + usecs_to_jiffies(TIMEOUT_US); 69 if (sc->gds_hw_ctrl) {
69 do { 70 status_reg = sc->gds_hw_ctrl;
70 ret = regmap_read(sc->regmap, sc->gdscr, &val); 71 /*
71 if (ret) 72 * The gds hw controller asserts/de-asserts the status bit soon
72 return ret; 73 * after it receives a power on/off request from a master.
74 * The controller then takes around 8 xo cycles to start its
75 * internal state machine and update the status bit. During
76 * this time, the status bit does not reflect the true status
77 * of the core.
78 * Add a delay of 1 us between writing to the SW_COLLAPSE bit
79 * and polling the status bit.
80 */
81 udelay(1);
82 }
73 83
74 if ((val & PWR_ON_MASK) == check) 84 start = ktime_get();
85 do {
86 if (gdsc_is_enabled(sc, status_reg) == en)
75 return 0; 87 return 0;
76 } while (time_before(jiffies, timeout)); 88 } while (ktime_us_delta(ktime_get(), start) < TIMEOUT_US);
77
78 ret = regmap_read(sc->regmap, sc->gdscr, &val);
79 if (ret)
80 return ret;
81 89
82 if ((val & PWR_ON_MASK) == check) 90 if (gdsc_is_enabled(sc, status_reg) == en)
83 return 0; 91 return 0;
84 92
85 return -ETIMEDOUT; 93 return -ETIMEDOUT;
@@ -165,6 +173,7 @@ static int gdsc_init(struct gdsc *sc)
165{ 173{
166 u32 mask, val; 174 u32 mask, val;
167 int on, ret; 175 int on, ret;
176 unsigned int reg;
168 177
169 /* 178 /*
170 * Disable HW trigger: collapse/restore occur based on registers writes. 179 * Disable HW trigger: collapse/restore occur based on registers writes.
@@ -185,7 +194,8 @@ static int gdsc_init(struct gdsc *sc)
185 return ret; 194 return ret;
186 } 195 }
187 196
188 on = gdsc_is_enabled(sc); 197 reg = sc->gds_hw_ctrl ? sc->gds_hw_ctrl : sc->gdscr;
198 on = gdsc_is_enabled(sc, reg);
189 if (on < 0) 199 if (on < 0)
190 return on; 200 return on;
191 201
diff --git a/drivers/clk/qcom/gdsc.h b/drivers/clk/qcom/gdsc.h
index 4e9dfc11ef57..66a43beb9f57 100644
--- a/drivers/clk/qcom/gdsc.h
+++ b/drivers/clk/qcom/gdsc.h
@@ -32,6 +32,7 @@ struct reset_controller_dev;
32 * @pd: generic power domain 32 * @pd: generic power domain
33 * @regmap: regmap for MMIO accesses 33 * @regmap: regmap for MMIO accesses
34 * @gdscr: gsdc control register 34 * @gdscr: gsdc control register
35 * @gds_hw_ctrl: gds_hw_ctrl register
35 * @cxcs: offsets of branch registers to toggle mem/periph bits in 36 * @cxcs: offsets of branch registers to toggle mem/periph bits in
36 * @cxc_count: number of @cxcs 37 * @cxc_count: number of @cxcs
37 * @pwrsts: Possible powerdomain power states 38 * @pwrsts: Possible powerdomain power states
@@ -44,6 +45,7 @@ struct gdsc {
44 struct generic_pm_domain *parent; 45 struct generic_pm_domain *parent;
45 struct regmap *regmap; 46 struct regmap *regmap;
46 unsigned int gdscr; 47 unsigned int gdscr;
48 unsigned int gds_hw_ctrl;
47 unsigned int *cxcs; 49 unsigned int *cxcs;
48 unsigned int cxc_count; 50 unsigned int cxc_count;
49 const u8 pwrsts; 51 const u8 pwrsts;