summaryrefslogtreecommitdiff
path: root/arch/arm/mach-tegra/tegra3_clocks.c
diff options
context:
space:
mode:
authorAlex Frid <afrid@nvidia.com>2011-07-23 23:59:20 -0700
committerVarun Colbert <vcolbert@nvidia.com>2011-08-04 10:51:00 -0700
commite989fd407a1f51c94d10841f0195da3272b6d0a8 (patch)
tree8543d3e742167e594311f3e0d381a4aba8b505cc /arch/arm/mach-tegra/tegra3_clocks.c
parent1bf381d9553fb5108eff327893e3da3538094db2 (diff)
ARM: tegra: clock: Update Tegra3 SCLK divider selection
Integrated the following limitation on 7.1 dividers setting when used as a source for system clock: no fractional settings are allowed with the exception of 1 : 2.5 ratio at core voltage 1.2V and above. Bug 840399 Change-Id: I3a5f65c8d8112e2ffe98165f25f64fd385e2a5d4 Reviewed-on: http://git-master/r/44181 Reviewed-by: Varun Colbert <vcolbert@nvidia.com> Tested-by: Varun Colbert <vcolbert@nvidia.com>
Diffstat (limited to 'arch/arm/mach-tegra/tegra3_clocks.c')
-rw-r--r--arch/arm/mach-tegra/tegra3_clocks.c98
1 files changed, 52 insertions, 46 deletions
diff --git a/arch/arm/mach-tegra/tegra3_clocks.c b/arch/arm/mach-tegra/tegra3_clocks.c
index 366dec405ed1..a4d059dccb83 100644
--- a/arch/arm/mach-tegra/tegra3_clocks.c
+++ b/arch/arm/mach-tegra/tegra3_clocks.c
@@ -1017,64 +1017,70 @@ static void tegra3_sbus_cmplx_init(struct clk *c)
BUG_ON(!(c->u.system.sclk_high->flags & DIV_U71));
}
-static long tegra3_sbus_div_round_rate(
- struct clk *d, unsigned long max_rate, unsigned long rate)
-{
- int divider;
- unsigned long d_input = clk_get_rate(d->parent);
-
- /* This special sbus round function is implemented because:
-
- (a) system bus clock duty cycle requirements can not be met by
- 1 : 1.5 divider, but we do need fractional division for better
- granularity - hence, special case with only 1.5 fraction skipped
-
- (b) since sbus is a shared bus, and its frequency is set to the
- highest enabled shared_bus_user clock, the target rate should be
- rounded up divider ladder (if max limit allows it) - for pll_div
- and peripheral_div common is rounding down - special case again.
-
- Note that final rate is trimmed, not rounded up to avoid multiple
- round spiral up. Lost 1Hz is added in tegra3_sbus_cmplx_set_rate
- just before actually setting divider rate */
-
- if (rate > d_input / 2)
- return (d_input > max_rate) ? (d_input / 2) : d_input;
-
- /* round divider down => round rate up */
- divider = clk_div71_get_divider(
- d_input, rate, d->flags, ROUND_DIVIDER_DOWN);
- if (divider < 0)
- return divider;
-
- rate = 2 * d_input / (divider + 2);
- if (rate > max_rate)
- rate = 2 * d_input / (divider + 3);
-
- return rate;
-}
+/* This special sbus round function is implemented because:
+ *
+ * (a) fractional dividers can not be used to derive system bus clock with one
+ * exception: 1 : 2.5 divider is allowed at 1.2V and above (and we do need this
+ * divider to reach top sbus frequencies from high frequency source).
+ *
+ * (b) since sbus is a shared bus, and its frequency is set to the highest
+ * enabled shared_bus_user clock, the target rate should be rounded up divider
+ * ladder (if max limit allows it) - for pll_div and peripheral_div common is
+ * rounding down - special case again.
+ *
+ * Note that final rate is trimmed (not rounded up) to avoid spiraling up in
+ * recursive calls. Lost 1Hz is added in tegra3_sbus_cmplx_set_rate before
+ * actually setting divider rate.
+ */
+static unsigned long sclk_high_2_5_rate;
+static bool sclk_high_2_5_valid;
static long tegra3_sbus_cmplx_round_rate(struct clk *c, unsigned long rate)
{
+ int i, divider;
+ unsigned long source_rate, round_rate;
struct clk *new_parent;
- struct clk *new_parent_after_round;
rate = max(rate, c->min_rate);
+ if (!sclk_high_2_5_rate) {
+ source_rate = clk_get_rate(c->u.system.sclk_high->parent);
+ sclk_high_2_5_rate = 2 * source_rate / 5;
+ i = tegra_dvfs_predict_millivolts(c, sclk_high_2_5_rate);
+ if (!IS_ERR_VALUE(i) && (i >= 1200) &&
+ (sclk_high_2_5_rate <= c->max_rate))
+ sclk_high_2_5_valid = true;
+ }
+
new_parent = (rate <= c->u.system.threshold) ?
c->u.system.sclk_low : c->u.system.sclk_high;
+ source_rate = clk_get_rate(new_parent->parent);
- rate = tegra3_sbus_div_round_rate(new_parent, c->max_rate, rate);
+ divider = clk_div71_get_divider(source_rate, rate,
+ new_parent->flags | DIV_U71_INT, ROUND_DIVIDER_DOWN);
+ if (divider < 0)
+ return divider;
- if(!IS_ERR_VALUE(rate)) {
- new_parent_after_round = (rate <= c->u.system.threshold) ?
- c->u.system.sclk_low : c->u.system.sclk_high;
- /* if parent is oscillating across threshold -
- use threshold as a target */
- if (new_parent != new_parent_after_round)
- return c->u.system.threshold;
+ round_rate = source_rate * 2 / (divider + 2);
+ if (round_rate > c->max_rate) {
+ divider += 2;
+ round_rate = source_rate * 2 / (divider + 2);
}
- return rate;
+
+ if (new_parent == c->u.system.sclk_high) {
+ /* Check if 1 : 2.5 ratio provides better approximation */
+ if (sclk_high_2_5_valid) {
+ if (((sclk_high_2_5_rate < round_rate) &&
+ (sclk_high_2_5_rate >= rate)) ||
+ ((round_rate < sclk_high_2_5_rate) &&
+ (round_rate < rate)))
+ round_rate = sclk_high_2_5_rate;
+ }
+
+ if (round_rate <= c->u.system.threshold)
+ round_rate = c->u.system.threshold;
+ }
+ return round_rate;
}
static int tegra3_sbus_cmplx_set_rate(struct clk *c, unsigned long rate)