blob: 9104f98c9e11c425421b027da8fd077a350bb21c [file] [log] [blame]
Patrick Daly985c14b2012-12-03 17:12:37 -08001/*
2 * Copyright (c) 2013, The Linux Foundation. All rights reserved.
3 *
4 * This program is free software; you can redistribute it and/or modify
5 * it under the terms of the GNU General Public License version 2 and
6 * only version 2 as published by the Free Software Foundation.
7 *
8 * This program is distributed in the hope that it will be useful,
9 * but WITHOUT ANY WARRANTY; without even the implied warranty of
10 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
11 * GNU General Public License for more details.
12 */
13
14#define pr_fmt(fmt) "%s: " fmt, __func__
15
16#include <linux/kernel.h>
17#include <linux/module.h>
18#include <linux/init.h>
19#include <linux/io.h>
20#include <linux/delay.h>
21#include <linux/mutex.h>
22#include <linux/spinlock.h>
23#include <linux/errno.h>
24#include <linux/cpufreq.h>
25#include <linux/clk.h>
26#include <linux/platform_device.h>
27#include <linux/iopoll.h>
28
29#include <mach/board.h>
30#include <mach/msm_iomap.h>
31#include <mach/msm_bus.h>
32#include <mach/msm_bus_board.h>
33#include <mach/rpm-regulator.h>
34#include <mach/clk-provider.h>
35#include <mach/rpm-regulator-smd.h>
36
37#include "acpuclock.h"
38#include "acpuclock-cortex.h"
39
40#define POLL_INTERVAL_US 1
41#define APCS_RCG_UPDATE_TIMEOUT_US 20
42
43static struct acpuclk_drv_data *acpuclk_init_data;
44static uint32_t bus_perf_client;
45
46/* Update the bus bandwidth request. */
47static void set_bus_bw(unsigned int bw)
48{
49 int ret;
50
51 if (bw >= acpuclk_init_data->bus_scale->num_usecases) {
52 pr_err("invalid bandwidth request (%d)\n", bw);
53 return;
54 }
55
56 /* Update bandwidth if request has changed. This may sleep. */
57 ret = msm_bus_scale_client_update_request(bus_perf_client, bw);
58 if (ret)
59 pr_err("bandwidth request failed (%d)\n", ret);
60
61 return;
62}
63
64/* Apply any voltage increases. */
65static int increase_vdd(unsigned int vdd_cpu, unsigned int vdd_mem)
66{
67 int rc = 0;
68
69 /* Increase vdd_mem before vdd_cpu. vdd_mem should be >= vdd_cpu. */
70 rc = regulator_set_voltage(acpuclk_init_data->vdd_mem, vdd_mem,
71 acpuclk_init_data->vdd_max_mem);
72 if (rc) {
73 pr_err("vdd_mem increase failed (%d)\n", rc);
74 return rc;
75 }
76
77 rc = regulator_set_voltage(acpuclk_init_data->vdd_cpu, vdd_cpu,
78 acpuclk_init_data->vdd_max_cpu);
79 if (rc)
80 pr_err("vdd_cpu increase failed (%d)\n", rc);
81
82 return rc;
83}
84
85/* Apply any per-cpu voltage decreases. */
86static void decrease_vdd(unsigned int vdd_cpu, unsigned int vdd_mem)
87{
88 int ret;
89
90 /* Update CPU voltage. */
91 ret = regulator_set_voltage(acpuclk_init_data->vdd_cpu, vdd_cpu,
92 acpuclk_init_data->vdd_max_cpu);
93 if (ret) {
94 pr_err("vdd_cpu decrease failed (%d)\n", ret);
95 return;
96 }
97
98 /* Decrease vdd_mem after vdd_cpu. vdd_mem should be >= vdd_cpu. */
99 ret = regulator_set_voltage(acpuclk_init_data->vdd_mem, vdd_mem,
100 acpuclk_init_data->vdd_max_mem);
101 if (ret)
102 pr_err("vdd_mem decrease failed (%d)\n", ret);
103}
104
105static void select_clk_source_div(struct acpuclk_drv_data *drv_data,
106 struct clkctl_acpu_speed *s)
107{
108 u32 regval, rc, src_div;
109 void __iomem *apcs_rcg_config = drv_data->apcs_rcg_config;
110 void __iomem *apcs_rcg_cmd = drv_data->apcs_rcg_cmd;
111 struct acpuclk_reg_data *r = &drv_data->reg_data;
112
113 src_div = s->src_div ? ((2 * s->src_div) - 1) : s->src_div;
114
115 regval = readl_relaxed(apcs_rcg_config);
116 regval &= ~r->cfg_src_mask;
117 regval |= s->src_sel << r->cfg_src_shift;
118 regval &= ~r->cfg_div_mask;
119 regval |= src_div << r->cfg_div_shift;
120 writel_relaxed(regval, apcs_rcg_config);
121
122 /* Update the configuration */
123 regval = readl_relaxed(apcs_rcg_cmd);
124 regval |= r->update_mask;
125 writel_relaxed(regval, apcs_rcg_cmd);
126
127 /* Wait for the update to take effect */
128 rc = readl_poll_timeout(apcs_rcg_cmd, regval,
129 !(regval & r->poll_mask),
130 POLL_INTERVAL_US,
131 APCS_RCG_UPDATE_TIMEOUT_US);
132 if (rc)
133 pr_warn("acpu rcg didn't update its configuration\n");
134}
135
Patrick Dalye85c6bc2013-01-31 14:14:28 -0800136/*
137 * This function can be called in both atomic and nonatomic context.
138 * Since regulator APIS can sleep, we cannot always use the clk prepare
139 * unprepare API.
140 */
141static int set_speed(struct clkctl_acpu_speed *tgt_s, bool atomic)
Patrick Daly985c14b2012-12-03 17:12:37 -0800142{
143 int rc = 0;
144 unsigned int tgt_freq_hz = tgt_s->khz * 1000;
145 struct clkctl_acpu_speed *strt_s = acpuclk_init_data->current_speed;
146 struct clkctl_acpu_speed *cxo_s = &acpuclk_init_data->freq_tbl[0];
147 struct clk *strt = acpuclk_init_data->src_clocks[strt_s->src].clk;
148 struct clk *tgt = acpuclk_init_data->src_clocks[tgt_s->src].clk;
149
150 if (strt_s->src == ACPUPLL && tgt_s->src == ACPUPLL) {
151 /* Switch to another always on src */
152 select_clk_source_div(acpuclk_init_data, cxo_s);
153
154 /* Re-program acpu pll */
Patrick Dalye85c6bc2013-01-31 14:14:28 -0800155 if (atomic)
156 clk_disable(tgt);
157 else
158 clk_disable_unprepare(tgt);
159
Patrick Daly985c14b2012-12-03 17:12:37 -0800160 rc = clk_set_rate(tgt, tgt_freq_hz);
161 if (rc)
162 pr_err("Failed to set ACPU PLL to %u\n", tgt_freq_hz);
Patrick Dalye85c6bc2013-01-31 14:14:28 -0800163
164 if (atomic)
165 BUG_ON(clk_enable(tgt));
166 else
167 BUG_ON(clk_prepare_enable(tgt));
Patrick Daly985c14b2012-12-03 17:12:37 -0800168
169 /* Switch back to acpu pll */
170 select_clk_source_div(acpuclk_init_data, tgt_s);
171
172 } else if (strt_s->src != ACPUPLL && tgt_s->src == ACPUPLL) {
173 rc = clk_set_rate(tgt, tgt_freq_hz);
174 if (rc) {
175 pr_err("Failed to set ACPU PLL to %u\n", tgt_freq_hz);
176 return rc;
177 }
178
Patrick Dalye85c6bc2013-01-31 14:14:28 -0800179 if (atomic)
180 clk_enable(tgt);
181 else
182 clk_prepare_enable(tgt);
183
Patrick Daly985c14b2012-12-03 17:12:37 -0800184 if (rc) {
185 pr_err("ACPU PLL enable failed\n");
186 return rc;
187 }
188
189 select_clk_source_div(acpuclk_init_data, tgt_s);
190
Patrick Dalye85c6bc2013-01-31 14:14:28 -0800191 if (atomic)
192 clk_disable(strt);
193 else
194 clk_disable_unprepare(strt);
195
Patrick Daly985c14b2012-12-03 17:12:37 -0800196 } else {
Patrick Dalye85c6bc2013-01-31 14:14:28 -0800197 if (atomic)
198 clk_enable(tgt);
199 else
200 clk_prepare_enable(tgt);
201
Patrick Daly985c14b2012-12-03 17:12:37 -0800202 if (rc) {
203 pr_err("%s enable failed\n",
204 acpuclk_init_data->src_clocks[tgt_s->src].name);
205 return rc;
206 }
207
208 select_clk_source_div(acpuclk_init_data, tgt_s);
209
Patrick Dalye85c6bc2013-01-31 14:14:28 -0800210 if (atomic)
211 clk_disable(strt);
212 else
213 clk_disable_unprepare(strt);
214
Patrick Daly985c14b2012-12-03 17:12:37 -0800215 }
216
217 return rc;
218}
219
220static int acpuclk_cortex_set_rate(int cpu, unsigned long rate,
221 enum setrate_reason reason)
222{
223 struct clkctl_acpu_speed *tgt_s, *strt_s;
224 int rc = 0;
225
226 if (reason == SETRATE_CPUFREQ)
227 mutex_lock(&acpuclk_init_data->lock);
228
229 strt_s = acpuclk_init_data->current_speed;
230
231 /* Return early if rate didn't change */
232 if (rate == strt_s->khz)
233 goto out;
234
235 /* Find target frequency */
236 for (tgt_s = acpuclk_init_data->freq_tbl; tgt_s->khz != 0; tgt_s++)
237 if (tgt_s->khz == rate)
238 break;
239 if (tgt_s->khz == 0) {
240 rc = -EINVAL;
241 goto out;
242 }
243
244 /* Increase VDD levels if needed */
245 if ((reason == SETRATE_CPUFREQ || reason == SETRATE_INIT)
246 && (tgt_s->khz > strt_s->khz)) {
247 rc = increase_vdd(tgt_s->vdd_cpu, tgt_s->vdd_mem);
248 if (rc)
249 goto out;
250 }
251
252 pr_debug("Switching from CPU rate %u KHz -> %u KHz\n",
253 strt_s->khz, tgt_s->khz);
254
Patrick Dalye85c6bc2013-01-31 14:14:28 -0800255 /* Switch CPU speed. Flag indicates atomic context */
256 if (reason == SETRATE_CPUFREQ || reason == SETRATE_INIT)
257 rc = set_speed(tgt_s, false);
258 else
259 rc = set_speed(tgt_s, true);
260
Patrick Daly985c14b2012-12-03 17:12:37 -0800261 if (rc)
262 goto out;
263
264 acpuclk_init_data->current_speed = tgt_s;
265 pr_debug("CPU speed change complete\n");
266
267 /* Nothing else to do for SWFI or power-collapse. */
268 if (reason == SETRATE_SWFI || reason == SETRATE_PC)
269 goto out;
270
271 /* Update bus bandwith request */
272 set_bus_bw(tgt_s->bw_level);
273
274 /* Drop VDD levels if we can. */
275 if (tgt_s->khz < strt_s->khz)
276 decrease_vdd(tgt_s->vdd_cpu, tgt_s->vdd_mem);
277
278out:
279 if (reason == SETRATE_CPUFREQ)
280 mutex_unlock(&acpuclk_init_data->lock);
281 return rc;
282}
283
284static unsigned long acpuclk_cortex_get_rate(int cpu)
285{
286 return acpuclk_init_data->current_speed->khz;
287}
288
289#ifdef CONFIG_CPU_FREQ_MSM
290static struct cpufreq_frequency_table freq_table[30];
291
292static void __init cpufreq_table_init(void)
293{
294 int i, freq_cnt = 0;
295
296 /* Construct the freq_table tables from acpuclk_init_data->freq_tbl. */
297 for (i = 0; acpuclk_init_data->freq_tbl[i].khz != 0
298 && freq_cnt < ARRAY_SIZE(freq_table); i++) {
299 if (!acpuclk_init_data->freq_tbl[i].use_for_scaling)
300 continue;
301 freq_table[freq_cnt].index = freq_cnt;
302 freq_table[freq_cnt].frequency =
303 acpuclk_init_data->freq_tbl[i].khz;
304 freq_cnt++;
305 }
306 /* freq_table not big enough to store all usable freqs. */
307 BUG_ON(acpuclk_init_data->freq_tbl[i].khz != 0);
308
309 freq_table[freq_cnt].index = freq_cnt;
310 freq_table[freq_cnt].frequency = CPUFREQ_TABLE_END;
311
312 pr_info("CPU: %d scaling frequencies supported.\n", freq_cnt);
313
314 /* Register table with CPUFreq. */
315 for_each_possible_cpu(i)
316 cpufreq_frequency_table_get_attr(freq_table, i);
317}
318#else
319static void __init cpufreq_table_init(void) {}
320#endif
321
322static struct acpuclk_data acpuclk_cortex_data = {
323 .set_rate = acpuclk_cortex_set_rate,
324 .get_rate = acpuclk_cortex_get_rate,
325 .power_collapse_khz = 19200,
326 .wait_for_irq_khz = 19200,
327};
328
329int __init acpuclk_cortex_init(struct platform_device *pdev,
330 struct acpuclk_drv_data *data)
331{
332 unsigned long max_cpu_khz = 0;
333 int i, rc;
334
335 acpuclk_init_data = data;
336 mutex_init(&acpuclk_init_data->lock);
337
338 bus_perf_client = msm_bus_scale_register_client(
339 acpuclk_init_data->bus_scale);
340 if (!bus_perf_client) {
341 pr_err("Unable to register bus client\n");
342 BUG();
343 }
344
345 for (i = 0; i < NUM_SRC; i++) {
346 if (!acpuclk_init_data->src_clocks[i].name)
347 continue;
348 acpuclk_init_data->src_clocks[i].clk =
349 clk_get(&pdev->dev,
350 acpuclk_init_data->src_clocks[i].name);
351 BUG_ON(IS_ERR(acpuclk_init_data->src_clocks[i].clk));
Patrick Daly985c14b2012-12-03 17:12:37 -0800352 }
353
354 /* Improve boot time by ramping up CPU immediately */
Patrick Daly71839d42013-02-25 13:05:05 -0800355 for (i = 0; acpuclk_init_data->freq_tbl[i].khz != 0; i++)
356 if (acpuclk_init_data->freq_tbl[i].use_for_scaling)
357 max_cpu_khz = acpuclk_init_data->freq_tbl[i].khz;
Patrick Daly985c14b2012-12-03 17:12:37 -0800358
359 /* Initialize regulators */
Patrick Daly973c8b72013-02-28 19:23:03 -0800360 rc = increase_vdd(acpuclk_init_data->vdd_max_cpu,
361 acpuclk_init_data->vdd_max_mem);
Patrick Daly985c14b2012-12-03 17:12:37 -0800362 if (rc)
363 goto err_vdd;
364
365 rc = regulator_enable(acpuclk_init_data->vdd_mem);
366 if (rc) {
367 dev_err(&pdev->dev, "regulator_enable for mem failed\n");
368 goto err_vdd;
369 }
370
371 rc = regulator_enable(acpuclk_init_data->vdd_cpu);
372 if (rc) {
373 dev_err(&pdev->dev, "regulator_enable for cpu failed\n");
374 goto err_vdd_cpu;
375 }
376
Patrick Daly71839d42013-02-25 13:05:05 -0800377 /*
378 * Select a state which is always a valid transition to align SW with
379 * the HW configuration set by the bootloaders.
380 */
381 acpuclk_cortex_set_rate(0, acpuclk_cortex_data.power_collapse_khz,
382 SETRATE_INIT);
Patrick Daly985c14b2012-12-03 17:12:37 -0800383 acpuclk_cortex_set_rate(0, max_cpu_khz, SETRATE_INIT);
384
385 acpuclk_register(&acpuclk_cortex_data);
386 cpufreq_table_init();
387
388 return 0;
389
390err_vdd_cpu:
391 regulator_disable(acpuclk_init_data->vdd_mem);
392err_vdd:
393 regulator_put(acpuclk_init_data->vdd_mem);
394 regulator_put(acpuclk_init_data->vdd_cpu);
395
396 for (i = 0; i < NUM_SRC; i++) {
397 if (!acpuclk_init_data->src_clocks[i].name)
398 continue;
Patrick Daly985c14b2012-12-03 17:12:37 -0800399 clk_put(acpuclk_init_data->src_clocks[i].clk);
400 }
401 return rc;
402}