blob: 7c1b8d6a90831ad798da2205574c267887f229d2 [file] [log] [blame]
Abhijeet Dharmapurikarfe1306b2012-11-15 18:09:31 -08001/* Copyright (c) 2012-2013, The Linux Foundation. All rights reserved.
Abhijeet Dharmapurikar00269e52012-05-14 17:59:10 -07002 *
3 * This program is free software; you can redistribute it and/or modify
4 * it under the terms of the GNU General Public License version 2 and
5 * only version 2 as published by the Free Software Foundation.
6 *
7 * This program is distributed in the hope that it will be useful,
8 * but WITHOUT ANY WARRANTY; without even the implied warranty of
9 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
10 * GNU General Public License for more details.
11 */
12
Abhijeet Dharmapurikarfe1306b2012-11-15 18:09:31 -080013#define pr_fmt(fmt) "PDN %s: " fmt, __func__
Abhijeet Dharmapurikar00269e52012-05-14 17:59:10 -070014
15#include <linux/err.h>
16#include <linux/kernel.h>
17#include <linux/module.h>
18#include <linux/init.h>
19#include <linux/delay.h>
20#include <linux/slab.h>
21#include <linux/string.h>
22#include <linux/io.h>
23#include <linux/of.h>
24#include <linux/of_device.h>
25#include <linux/platform_device.h>
26#include <linux/regulator/driver.h>
27#include <linux/regulator/machine.h>
28#include <linux/regulator/of_regulator.h>
29#include <linux/regulator/krait-regulator.h>
Abhijeet Dharmapurikar37c47a82013-01-08 18:44:21 -080030#include <linux/debugfs.h>
Abhijeet Dharmapurikarc8605a62013-03-11 21:24:25 -070031#include <linux/syscore_ops.h>
Abhijeet Dharmapurikar3ff7cba2012-10-10 17:48:29 -070032#include <mach/msm_iomap.h>
Abhijeet Dharmapurikar00269e52012-05-14 17:59:10 -070033
34#include "spm.h"
Abhijeet Dharmapurikar295c4dc2013-01-08 12:59:01 -080035#include "pm.h"
Abhijeet Dharmapurikar00269e52012-05-14 17:59:10 -070036
Abhijeet Dharmapurikar152a7082012-05-14 20:25:59 -070037/*
38 * supply
39 * from
40 * pmic
41 * gang
Abhijeet Dharmapurikar749ecd42012-11-08 17:29:04 -080042 * |
43 * |________________________________
44 * | | |
45 * ___|___ | |
46 * | | | |
47 * | | / /
48 * | LDO | / /LDO BYP [6]
49 * | | / BHS[6] /(bypass is a weak BHS
50 * |_______| | | needs to be on when in
51 * | | | BHS mode)
52 * |________________|_______________|
Abhijeet Dharmapurikar152a7082012-05-14 20:25:59 -070053 * |
54 * ________|________
55 * | |
56 * | KRAIT |
57 * |_________________|
58 */
59
Abhijeet Dharmapurikar00269e52012-05-14 17:59:10 -070060#define PMIC_VOLTAGE_MIN 350000
61#define PMIC_VOLTAGE_MAX 1355000
62#define LV_RANGE_STEP 5000
Abhijeet Dharmapurikar00269e52012-05-14 17:59:10 -070063
Patrick Cain92f4fa12013-04-22 16:23:19 -070064#define CORE_VOLTAGE_BOOTUP 900000
Abhijeet Dharmapurikar00269e52012-05-14 17:59:10 -070065
Abhijeet Dharmapurikar152a7082012-05-14 20:25:59 -070066#define KRAIT_LDO_VOLTAGE_MIN 465000
Abhijeet Dharmapurikare8571ef2012-08-14 20:44:58 -070067#define KRAIT_LDO_VOLTAGE_OFFSET 465000
Abhijeet Dharmapurikar152a7082012-05-14 20:25:59 -070068#define KRAIT_LDO_STEP 5000
69
70#define BHS_SETTLING_DELAY_US 1
71#define LDO_SETTLING_DELAY_US 1
Abhijeet Dharmapurikare8f83852013-02-06 18:57:17 -080072#define MDD_SETTLING_DELAY_US 5
Abhijeet Dharmapurikar152a7082012-05-14 20:25:59 -070073
74#define _KRAIT_MASK(BITS, POS) (((u32)(1 << (BITS)) - 1) << POS)
75#define KRAIT_MASK(LEFT_BIT_POS, RIGHT_BIT_POS) \
76 _KRAIT_MASK(LEFT_BIT_POS - RIGHT_BIT_POS + 1, RIGHT_BIT_POS)
77
78#define APC_SECURE 0x00000000
79#define CPU_PWR_CTL 0x00000004
80#define APC_PWR_STATUS 0x00000008
81#define APC_TEST_BUS_SEL 0x0000000C
82#define CPU_TRGTD_DBG_RST 0x00000010
83#define APC_PWR_GATE_CTL 0x00000014
84#define APC_LDO_VREF_SET 0x00000018
Abhijeet Dharmapurikar3ff7cba2012-10-10 17:48:29 -070085#define APC_PWR_GATE_MODE 0x0000001C
86#define APC_PWR_GATE_DLY 0x00000020
87
88#define PWR_GATE_CONFIG 0x00000044
89#define VERSION 0x00000FD0
Abhijeet Dharmapurikar152a7082012-05-14 20:25:59 -070090
Abhijeet Dharmapurikar749ecd42012-11-08 17:29:04 -080091/* MDD register group */
92#define MDD_CONFIG_CTL 0x00000000
93#define MDD_MODE 0x00000010
94
David Collinsb80db492013-04-26 15:05:41 -070095#define PHASE_SCALING_REF 4
96
97/* bit definitions for phase scaling eFuses */
98#define PHASE_SCALING_EFUSE_VERSION_POS 26
99#define PHASE_SCALING_EFUSE_VERSION_MASK KRAIT_MASK(27, 26)
100#define PHASE_SCALING_EFUSE_VERSION_SET 1
101
102#define PHASE_SCALING_EFUSE_VALUE_POS 16
103#define PHASE_SCALING_EFUSE_VALUE_MASK KRAIT_MASK(18, 16)
104
Abhijeet Dharmapurikar152a7082012-05-14 20:25:59 -0700105/* bit definitions for APC_PWR_GATE_CTL */
106#define BHS_CNT_BIT_POS 24
107#define BHS_CNT_MASK KRAIT_MASK(31, 24)
108#define BHS_CNT_DEFAULT 64
109
110#define CLK_SRC_SEL_BIT_POS 15
111#define CLK_SRC_SEL_MASK KRAIT_MASK(15, 15)
112#define CLK_SRC_DEFAULT 0
113
114#define LDO_PWR_DWN_BIT_POS 16
115#define LDO_PWR_DWN_MASK KRAIT_MASK(21, 16)
116
117#define LDO_BYP_BIT_POS 8
118#define LDO_BYP_MASK KRAIT_MASK(13, 8)
119
120#define BHS_SEG_EN_BIT_POS 1
121#define BHS_SEG_EN_MASK KRAIT_MASK(6, 1)
122#define BHS_SEG_EN_DEFAULT 0x3F
123
124#define BHS_EN_BIT_POS 0
125#define BHS_EN_MASK KRAIT_MASK(0, 0)
126
127/* bit definitions for APC_LDO_VREF_SET register */
128#define VREF_RET_POS 8
129#define VREF_RET_MASK KRAIT_MASK(14, 8)
130
131#define VREF_LDO_BIT_POS 0
132#define VREF_LDO_MASK KRAIT_MASK(6, 0)
133
Abhijeet Dharmapurikar37c47a82013-01-08 18:44:21 -0800134#define LDO_HDROOM_MIN 50000
135#define LDO_HDROOM_MAX 250000
136
137#define LDO_UV_MIN 465000
138#define LDO_UV_MAX 750000
139
140#define LDO_TH_MIN 600000
141#define LDO_TH_MAX 900000
142
143#define LDO_DELTA_MIN 10000
144#define LDO_DELTA_MAX 100000
145
Mahesh Sivasubramanian3dca48e2013-04-12 14:29:57 -0600146#define MSM_L2_SAW_PHYS 0xf9012000
Abhijeet Dharmapurikar00269e52012-05-14 17:59:10 -0700147/**
148 * struct pmic_gang_vreg -
149 * @name: the string used to represent the gang
150 * @pmic_vmax_uV: the current pmic gang voltage
151 * @pmic_phase_count: the number of phases turned on in the gang
152 * @krait_power_vregs: a list of krait consumers this gang supplies to
153 * @krait_power_vregs_lock: lock to prevent simultaneous access to the list
154 * and its nodes. This needs to be taken by each
155 * regulator's callback functions to prevent
156 * simultaneous updates to the pmic's phase
157 * voltage.
Patrick Cain92f4fa12013-04-22 16:23:19 -0700158 * @apcs_gcc_base: virtual address of the APCS GCC registers
159 * @manage_phases: begin phase control
160 * @pfm_threshold: the sum of coefficients below which PFM can be
161 * enabled
David Collinsb80db492013-04-26 15:05:41 -0700162 * @efuse_phase_scaling_factor: Phase scaling factor read out of an eFuse. When
163 * calculating the appropriate phase count to use,
164 * coeff2 is multiplied by this factor and then
165 * divided by PHASE_SCALING_REF.
Abhijeet Dharmapurikar00269e52012-05-14 17:59:10 -0700166 */
167struct pmic_gang_vreg {
168 const char *name;
169 int pmic_vmax_uV;
170 int pmic_phase_count;
171 struct list_head krait_power_vregs;
172 struct mutex krait_power_vregs_lock;
Abhijeet Dharmapurikar6a42ef82012-10-18 19:37:36 -0700173 bool pfm_mode;
Abhijeet Dharmapurikar295c4dc2013-01-08 12:59:01 -0800174 int pmic_min_uV_for_retention;
175 bool retention_enabled;
Abhijeet Dharmapurikar46a7ad32013-02-23 12:50:19 -0800176 bool use_phase_switching;
Stepan Moskovchenko7663b782013-03-08 20:13:06 -0800177 void __iomem *apcs_gcc_base;
Patrick Cain92f4fa12013-04-22 16:23:19 -0700178 bool manage_phases;
179 int pfm_threshold;
David Collinsb80db492013-04-26 15:05:41 -0700180 int efuse_phase_scaling_factor;
Abhijeet Dharmapurikar00269e52012-05-14 17:59:10 -0700181};
182
183static struct pmic_gang_vreg *the_gang;
184
185enum krait_supply_mode {
186 HS_MODE = REGULATOR_MODE_NORMAL,
187 LDO_MODE = REGULATOR_MODE_IDLE,
188};
189
Patrick Cain955d3522013-04-22 16:25:24 -0700190#define WAIT_FOR_LOAD 0x2
191#define WAIT_FOR_VOLTAGE 0x1
192
Abhijeet Dharmapurikar00269e52012-05-14 17:59:10 -0700193struct krait_power_vreg {
194 struct list_head link;
195 struct regulator_desc desc;
196 struct regulator_dev *rdev;
197 const char *name;
198 struct pmic_gang_vreg *pvreg;
199 int uV;
Patrick Cain92f4fa12013-04-22 16:23:19 -0700200 int load;
Abhijeet Dharmapurikar00269e52012-05-14 17:59:10 -0700201 enum krait_supply_mode mode;
202 void __iomem *reg_base;
Abhijeet Dharmapurikar749ecd42012-11-08 17:29:04 -0800203 void __iomem *mdd_base;
Abhijeet Dharmapurikar34f0d642012-10-25 12:35:23 -0700204 int ldo_default_uV;
205 int retention_uV;
206 int headroom_uV;
207 int ldo_threshold_uV;
Abhijeet Dharmapurikar2269db02012-12-05 15:49:20 -0800208 int ldo_delta_uV;
Abhijeet Dharmapurikare8f83852013-02-06 18:57:17 -0800209 int cpu_num;
Patrick Cain92f4fa12013-04-22 16:23:19 -0700210 int coeff1;
211 int coeff2;
Michael Bohanb5d1cdb2012-11-08 18:09:52 -0800212 bool online;
Patrick Cain955d3522013-04-22 16:25:24 -0700213 int online_at_probe;
Abhijeet Dharmapurikar00269e52012-05-14 17:59:10 -0700214};
215
Abhijeet Dharmapurikare8f83852013-02-06 18:57:17 -0800216DEFINE_PER_CPU(struct krait_power_vreg *, krait_vregs);
217
Abhijeet Dharmapurikar3ff7cba2012-10-10 17:48:29 -0700218static u32 version;
219
David Collinsb80db492013-04-26 15:05:41 -0700220static int use_efuse_phase_scaling_factor;
221module_param_named(
222 use_phase_scaling_efuse, use_efuse_phase_scaling_factor, int,
223 S_IRUSR | S_IWUSR
224);
225
Abhijeet Dharmapurikar37c47a82013-01-08 18:44:21 -0800226static int is_between(int left, int right, int value)
227{
228 if (left >= right && left >= value && value >= right)
229 return 1;
230 if (left <= right && left <= value && value <= right)
231 return 1;
232 return 0;
233}
234
Abhijeet Dharmapurikar152a7082012-05-14 20:25:59 -0700235static void krait_masked_write(struct krait_power_vreg *kvreg,
236 int reg, uint32_t mask, uint32_t val)
237{
238 uint32_t reg_val;
239
240 reg_val = readl_relaxed(kvreg->reg_base + reg);
241 reg_val &= ~mask;
242 reg_val |= (val & mask);
243 writel_relaxed(reg_val, kvreg->reg_base + reg);
244
245 /*
246 * Barrier to ensure that the reads and writes from
247 * other regulator regions (they are 1k apart) execute in
248 * order to the above write.
249 */
250 mb();
251}
252
Abhijeet Dharmapurikar37c47a82013-01-08 18:44:21 -0800253static int get_krait_retention_ldo_uv(struct krait_power_vreg *kvreg)
254{
255 uint32_t reg_val;
256 int uV;
257
258 reg_val = readl_relaxed(kvreg->reg_base + APC_LDO_VREF_SET);
259 reg_val &= VREF_RET_MASK;
260 reg_val >>= VREF_RET_POS;
261
262 if (reg_val == 0)
263 uV = 0;
264 else
265 uV = KRAIT_LDO_VOLTAGE_OFFSET + reg_val * KRAIT_LDO_STEP;
266
267 return uV;
268}
269
Abhijeet Dharmapurikar152a7082012-05-14 20:25:59 -0700270static int get_krait_ldo_uv(struct krait_power_vreg *kvreg)
271{
272 uint32_t reg_val;
273 int uV;
274
275 reg_val = readl_relaxed(kvreg->reg_base + APC_LDO_VREF_SET);
276 reg_val &= VREF_LDO_MASK;
277 reg_val >>= VREF_LDO_BIT_POS;
278
279 if (reg_val == 0)
280 uV = 0;
281 else
282 uV = KRAIT_LDO_VOLTAGE_OFFSET + reg_val * KRAIT_LDO_STEP;
283
284 return uV;
285}
286
Abhijeet Dharmapurikar34f0d642012-10-25 12:35:23 -0700287static int set_krait_retention_uv(struct krait_power_vreg *kvreg, int uV)
Abhijeet Dharmapurikar152a7082012-05-14 20:25:59 -0700288{
289 uint32_t reg_val;
290
Abhijeet Dharmapurikar34f0d642012-10-25 12:35:23 -0700291 reg_val = DIV_ROUND_UP(uV - KRAIT_LDO_VOLTAGE_OFFSET, KRAIT_LDO_STEP);
292 krait_masked_write(kvreg, APC_LDO_VREF_SET, VREF_RET_MASK,
293 reg_val << VREF_RET_POS);
294
295 return 0;
296}
297
298static int set_krait_ldo_uv(struct krait_power_vreg *kvreg, int uV)
299{
300 uint32_t reg_val;
301
302 reg_val = DIV_ROUND_UP(uV - KRAIT_LDO_VOLTAGE_OFFSET, KRAIT_LDO_STEP);
Abhijeet Dharmapurikar152a7082012-05-14 20:25:59 -0700303 krait_masked_write(kvreg, APC_LDO_VREF_SET, VREF_LDO_MASK,
304 reg_val << VREF_LDO_BIT_POS);
305
306 return 0;
307}
308
Abhijeet Dharmapurikare8f83852013-02-06 18:57:17 -0800309static int __krait_power_mdd_enable(struct krait_power_vreg *kvreg, bool on)
310{
311 if (on) {
312 writel_relaxed(0x00000002, kvreg->mdd_base + MDD_MODE);
313 /* complete the above write before the delay */
314 mb();
315 udelay(MDD_SETTLING_DELAY_US);
316 } else {
317 writel_relaxed(0x00000000, kvreg->mdd_base + MDD_MODE);
318 /*
319 * complete the above write before other accesses
320 * to krait regulator
321 */
322 mb();
323 }
324 return 0;
325}
326
Patrick Cain92f4fa12013-04-22 16:23:19 -0700327#define COEFF2_UV_THRESHOLD 850000
David Collinsb80db492013-04-26 15:05:41 -0700328static int get_coeff2(int krait_uV, int phase_scaling_factor)
Patrick Cain92f4fa12013-04-22 16:23:19 -0700329{
330 int coeff2 = 0;
331 int krait_mV = krait_uV / 1000;
332
333 if (krait_uV <= COEFF2_UV_THRESHOLD)
334 coeff2 = (612229 * krait_mV) / 1000 - 211258;
335 else
336 coeff2 = (892564 * krait_mV) / 1000 - 449543;
337
David Collinsb80db492013-04-26 15:05:41 -0700338 coeff2 = coeff2 * phase_scaling_factor / PHASE_SCALING_REF;
339
Patrick Cain92f4fa12013-04-22 16:23:19 -0700340 return coeff2;
341}
342
343static int get_coeff1(int actual_uV, int requested_uV, int load)
344{
345 int ratio = actual_uV * 1000 / requested_uV;
346 int coeff1 = 330 * load + (load * 673 * ratio / 1000);
347
348 return coeff1;
349}
350
351static int get_coeff_total(struct krait_power_vreg *from)
352{
353 int coeff_total = 0;
354 struct krait_power_vreg *kvreg;
355 struct pmic_gang_vreg *pvreg = from->pvreg;
David Collinsb80db492013-04-26 15:05:41 -0700356 int phase_scaling_factor = PHASE_SCALING_REF;
357
358 if (use_efuse_phase_scaling_factor)
359 phase_scaling_factor = pvreg->efuse_phase_scaling_factor;
Patrick Cain92f4fa12013-04-22 16:23:19 -0700360
361 list_for_each_entry(kvreg, &pvreg->krait_power_vregs, link) {
362 if (!kvreg->online)
363 continue;
364
365 if (kvreg->mode == LDO_MODE) {
366 kvreg->coeff1 =
367 get_coeff1(kvreg->uV - kvreg->ldo_delta_uV,
368 kvreg->uV, kvreg->load);
369 kvreg->coeff2 =
David Collinsb80db492013-04-26 15:05:41 -0700370 get_coeff2(kvreg->uV - kvreg->ldo_delta_uV,
371 phase_scaling_factor);
Patrick Cain92f4fa12013-04-22 16:23:19 -0700372 } else {
373 kvreg->coeff1 =
374 get_coeff1(pvreg->pmic_vmax_uV,
375 kvreg->uV, kvreg->load);
David Collinsb80db492013-04-26 15:05:41 -0700376 kvreg->coeff2 = get_coeff2(pvreg->pmic_vmax_uV,
377 phase_scaling_factor);
Patrick Cain92f4fa12013-04-22 16:23:19 -0700378 }
379 coeff_total += kvreg->coeff1 + kvreg->coeff2;
380 }
381
382 return coeff_total;
383}
384
385static int set_pmic_gang_phases(struct pmic_gang_vreg *pvreg, int phase_count)
386{
387 pr_debug("programming phase_count = %d\n", phase_count);
388 if (pvreg->use_phase_switching)
389 /*
390 * note the PMIC sets the phase count to one more than
391 * the value in the register - hence subtract 1 from it
392 */
393 return msm_spm_apcs_set_phase(phase_count - 1);
394 else
395 return 0;
396}
397
398static int num_online(struct pmic_gang_vreg *pvreg)
399{
400 int online_total = 0;
401 struct krait_power_vreg *kvreg;
402
403 list_for_each_entry(kvreg, &pvreg->krait_power_vregs, link) {
404 if (kvreg->online)
405 online_total++;
406 }
407 return online_total;
408}
409
410static int get_total_load(struct krait_power_vreg *from)
411{
412 int load_total = 0;
413 struct krait_power_vreg *kvreg;
414 struct pmic_gang_vreg *pvreg = from->pvreg;
415
416 list_for_each_entry(kvreg, &pvreg->krait_power_vregs, link) {
417 if (!kvreg->online)
418 continue;
419 load_total += kvreg->load;
420 }
421
422 return load_total;
423}
424
Patrick Cain955d3522013-04-22 16:25:24 -0700425static bool enable_phase_management(struct pmic_gang_vreg *pvreg)
426{
427 struct krait_power_vreg *kvreg;
428
429 list_for_each_entry(kvreg, &pvreg->krait_power_vregs, link) {
430 pr_debug("%s online_at_probe:0x%x\n", kvreg->name,
431 kvreg->online_at_probe);
432 if (kvreg->online_at_probe)
433 return false;
434 }
435 return true;
436}
437
Patrick Cain92f4fa12013-04-22 16:23:19 -0700438#define PMIC_FTS_MODE_PFM 0x00
439#define PMIC_FTS_MODE_PWM 0x80
440#define ONE_PHASE_COEFF 1000000
441#define TWO_PHASE_COEFF 2000000
442
Willie Ruanaf5b68b2013-04-24 12:08:00 -0700443#define PWM_SETTLING_TIME_US 50
444#define PHASE_SETTLING_TIME_US 50
Patrick Cain92f4fa12013-04-22 16:23:19 -0700445static unsigned int pmic_gang_set_phases(struct krait_power_vreg *from,
446 int coeff_total)
447{
448 struct pmic_gang_vreg *pvreg = from->pvreg;
449 int phase_count;
450 int rc = 0;
451 int n_online = num_online(pvreg);
452 int load_total;
453
454 load_total = get_total_load(from);
455
Patrick Cain955d3522013-04-22 16:25:24 -0700456 if (pvreg->manage_phases == false) {
457 if (enable_phase_management(pvreg))
458 pvreg->manage_phases = true;
459 else
460 return 0;
461 }
Patrick Cain92f4fa12013-04-22 16:23:19 -0700462
463 /* First check if the coeff is low for PFM mode */
464 if (load_total <= pvreg->pfm_threshold && n_online == 1) {
465 if (!pvreg->pfm_mode) {
466 rc = msm_spm_enable_fts_lpm(PMIC_FTS_MODE_PFM);
467 if (rc) {
468 pr_err("%s PFM en failed load_t %d rc = %d\n",
469 from->name, load_total, rc);
470 return rc;
471 } else {
472 pvreg->pfm_mode = true;
473 }
474 }
475 return rc;
476 }
477
478 /* coeff is high switch to PWM mode before changing phases */
479 if (pvreg->pfm_mode) {
480 rc = msm_spm_enable_fts_lpm(PMIC_FTS_MODE_PWM);
481 if (rc) {
482 pr_err("%s PFM exit failed load %d rc = %d\n",
483 from->name, coeff_total, rc);
484 return rc;
485 } else {
486 pvreg->pfm_mode = false;
Willie Ruanaf5b68b2013-04-24 12:08:00 -0700487 udelay(PWM_SETTLING_TIME_US);
Patrick Cain92f4fa12013-04-22 16:23:19 -0700488 }
489 }
490
491 /* calculate phases */
492 if (coeff_total < ONE_PHASE_COEFF)
493 phase_count = 1;
494 else if (coeff_total < TWO_PHASE_COEFF)
495 phase_count = 2;
496 else
497 phase_count = 4;
498
499 /* don't increase the phase count higher than number of online cpus */
500 if (phase_count > n_online)
501 phase_count = n_online;
502
503 if (phase_count != pvreg->pmic_phase_count) {
504 rc = set_pmic_gang_phases(pvreg, phase_count);
505 if (rc < 0) {
506 pr_err("%s failed set phase %d rc = %d\n",
507 from->name, phase_count, rc);
508 return rc;
509 }
510
511 /* complete the writes before the delay */
512 mb();
513
514 /*
515 * delay until the phases are settled when
516 * the count is raised
517 */
518 if (phase_count > pvreg->pmic_phase_count)
519 udelay(PHASE_SETTLING_TIME_US);
520
521 pvreg->pmic_phase_count = phase_count;
522 }
523
524 return rc;
525}
526
527static unsigned int _get_optimum_mode(struct regulator_dev *rdev,
528 int input_uV, int output_uV, int load)
529{
530 struct krait_power_vreg *kvreg = rdev_get_drvdata(rdev);
531 int coeff_total;
532 int rc;
533
Patrick Cain955d3522013-04-22 16:25:24 -0700534 kvreg->online_at_probe &= ~WAIT_FOR_LOAD;
Patrick Cain92f4fa12013-04-22 16:23:19 -0700535 coeff_total = get_coeff_total(kvreg);
536
537 rc = pmic_gang_set_phases(kvreg, coeff_total);
538 if (rc < 0) {
539 dev_err(&rdev->dev, "%s failed set mode %d rc = %d\n",
540 kvreg->name, coeff_total, rc);
541 }
542
543 return kvreg->mode;
544}
545
546static unsigned int krait_power_get_optimum_mode(struct regulator_dev *rdev,
547 int input_uV, int output_uV, int load_uA)
548{
549 struct krait_power_vreg *kvreg = rdev_get_drvdata(rdev);
550 struct pmic_gang_vreg *pvreg = kvreg->pvreg;
551 int rc;
552
553 mutex_lock(&pvreg->krait_power_vregs_lock);
554 kvreg->load = load_uA;
555 if (!kvreg->online) {
556 mutex_unlock(&pvreg->krait_power_vregs_lock);
557 return kvreg->mode;
558 }
559
560 rc = _get_optimum_mode(rdev, input_uV, output_uV, load_uA);
561 mutex_unlock(&pvreg->krait_power_vregs_lock);
562
563 return rc;
564}
565
566static int krait_power_set_mode(struct regulator_dev *rdev, unsigned int mode)
567{
568 return 0;
569}
570
571static unsigned int krait_power_get_mode(struct regulator_dev *rdev)
572{
573 struct krait_power_vreg *kvreg = rdev_get_drvdata(rdev);
574
575 return kvreg->mode;
576}
577
Abhijeet Dharmapurikar152a7082012-05-14 20:25:59 -0700578static int switch_to_using_hs(struct krait_power_vreg *kvreg)
579{
580 if (kvreg->mode == HS_MODE)
581 return 0;
Abhijeet Dharmapurikar152a7082012-05-14 20:25:59 -0700582 /* enable bhs */
Matt Wagantall5a58f452013-04-26 12:59:49 -0700583 krait_masked_write(kvreg, APC_PWR_GATE_CTL, BHS_EN_MASK, BHS_EN_MASK);
584 /* complete the above write before the delay */
585 mb();
586 /* wait for the bhs to settle */
587 udelay(BHS_SETTLING_DELAY_US);
588
589 /* Turn on BHS segments */
Abhijeet Dharmapurikar749ecd42012-11-08 17:29:04 -0800590 krait_masked_write(kvreg, APC_PWR_GATE_CTL,
Matt Wagantall5a58f452013-04-26 12:59:49 -0700591 BHS_SEG_EN_MASK, BHS_SEG_EN_DEFAULT << BHS_SEG_EN_BIT_POS);
Abhijeet Dharmapurikar749ecd42012-11-08 17:29:04 -0800592
593 /* complete the above write before the delay */
594 mb();
Abhijeet Dharmapurikar152a7082012-05-14 20:25:59 -0700595
596 /*
597 * wait for the bhs to settle - note that
598 * after the voltage has settled both BHS and LDO are supplying power
599 * to the krait. This avoids glitches during switching
600 */
601 udelay(BHS_SETTLING_DELAY_US);
602
Abhijeet Dharmapurikar749ecd42012-11-08 17:29:04 -0800603 /*
604 * enable ldo bypass - the krait is powered still by LDO since
605 * LDO is enabled
606 */
607 krait_masked_write(kvreg, APC_PWR_GATE_CTL, LDO_BYP_MASK, LDO_BYP_MASK);
608
Abhijeet Dharmapurikar152a7082012-05-14 20:25:59 -0700609 /* disable ldo - only the BHS provides voltage to the cpu after this */
610 krait_masked_write(kvreg, APC_PWR_GATE_CTL,
611 LDO_PWR_DWN_MASK, LDO_PWR_DWN_MASK);
612
613 kvreg->mode = HS_MODE;
Abhijeet Dharmapurikarfe1306b2012-11-15 18:09:31 -0800614 pr_debug("%s using BHS\n", kvreg->name);
Abhijeet Dharmapurikar152a7082012-05-14 20:25:59 -0700615 return 0;
616}
617
618static int switch_to_using_ldo(struct krait_power_vreg *kvreg)
619{
Abhijeet Dharmapurikar2269db02012-12-05 15:49:20 -0800620 if (kvreg->mode == LDO_MODE
621 && get_krait_ldo_uv(kvreg) == kvreg->uV - kvreg->ldo_delta_uV)
Abhijeet Dharmapurikar152a7082012-05-14 20:25:59 -0700622 return 0;
623
624 /*
625 * if the krait is in ldo mode and a voltage change is requested on the
626 * ldo switch to using hs before changing ldo voltage
627 */
628 if (kvreg->mode == LDO_MODE)
629 switch_to_using_hs(kvreg);
630
Abhijeet Dharmapurikar2269db02012-12-05 15:49:20 -0800631 set_krait_ldo_uv(kvreg, kvreg->uV - kvreg->ldo_delta_uV);
Abhijeet Dharmapurikar152a7082012-05-14 20:25:59 -0700632
633 /*
634 * enable ldo - note that both LDO and BHS are are supplying voltage to
635 * the cpu after this. This avoids glitches during switching from BHS
636 * to LDO.
637 */
638 krait_masked_write(kvreg, APC_PWR_GATE_CTL, LDO_PWR_DWN_MASK, 0);
639
Abhijeet Dharmapurikar749ecd42012-11-08 17:29:04 -0800640 /* complete the writes before the delay */
641 mb();
642
Abhijeet Dharmapurikar152a7082012-05-14 20:25:59 -0700643 /* wait for the ldo to settle */
644 udelay(LDO_SETTLING_DELAY_US);
645
646 /*
647 * disable BHS and disable LDO bypass seperate from enabling
648 * the LDO above.
649 */
650 krait_masked_write(kvreg, APC_PWR_GATE_CTL,
651 BHS_EN_MASK | LDO_BYP_MASK, 0);
Abhijeet Dharmapurikar749ecd42012-11-08 17:29:04 -0800652 krait_masked_write(kvreg, APC_PWR_GATE_CTL, BHS_SEG_EN_MASK, 0);
Abhijeet Dharmapurikar152a7082012-05-14 20:25:59 -0700653
654 kvreg->mode = LDO_MODE;
Abhijeet Dharmapurikarfe1306b2012-11-15 18:09:31 -0800655 pr_debug("%s using LDO\n", kvreg->name);
Abhijeet Dharmapurikar152a7082012-05-14 20:25:59 -0700656 return 0;
657}
658
Abhijeet Dharmapurikarfe1306b2012-11-15 18:09:31 -0800659static int set_pmic_gang_voltage(struct pmic_gang_vreg *pvreg, int uV)
Abhijeet Dharmapurikar00269e52012-05-14 17:59:10 -0700660{
661 int setpoint;
Abhijeet Dharmapurikarfe1306b2012-11-15 18:09:31 -0800662 int rc;
663
664 if (pvreg->pmic_vmax_uV == uV)
665 return 0;
666
667 pr_debug("%d\n", uV);
Abhijeet Dharmapurikar00269e52012-05-14 17:59:10 -0700668
669 if (uV < PMIC_VOLTAGE_MIN) {
670 pr_err("requested %d < %d, restricting it to %d\n",
671 uV, PMIC_VOLTAGE_MIN, PMIC_VOLTAGE_MIN);
672 uV = PMIC_VOLTAGE_MIN;
673 }
674 if (uV > PMIC_VOLTAGE_MAX) {
675 pr_err("requested %d > %d, restricting it to %d\n",
676 uV, PMIC_VOLTAGE_MAX, PMIC_VOLTAGE_MAX);
677 uV = PMIC_VOLTAGE_MAX;
678 }
679
Abhijeet Dharmapurikar295c4dc2013-01-08 12:59:01 -0800680 if (uV < pvreg->pmic_min_uV_for_retention) {
681 if (pvreg->retention_enabled) {
682 pr_debug("Disabling Retention pmic = %duV, pmic_min_uV_for_retention = %duV",
683 uV, pvreg->pmic_min_uV_for_retention);
684 msm_pm_enable_retention(false);
685 pvreg->retention_enabled = false;
686 }
687 } else {
688 if (!pvreg->retention_enabled) {
689 pr_debug("Enabling Retention pmic = %duV, pmic_min_uV_for_retention = %duV",
690 uV, pvreg->pmic_min_uV_for_retention);
691 msm_pm_enable_retention(true);
692 pvreg->retention_enabled = true;
693 }
694 }
695
Abhijeet Dharmapurikard4f38da2012-08-08 21:42:06 -0700696 setpoint = DIV_ROUND_UP(uV, LV_RANGE_STEP);
Michael Bohanb5d1cdb2012-11-08 18:09:52 -0800697
Abhijeet Dharmapurikarfe1306b2012-11-15 18:09:31 -0800698 rc = msm_spm_apcs_set_vdd(setpoint);
699 if (rc < 0)
700 pr_err("could not set %duV setpt = 0x%x rc = %d\n",
701 uV, setpoint, rc);
702 else
703 pvreg->pmic_vmax_uV = uV;
704
705 return rc;
Abhijeet Dharmapurikar00269e52012-05-14 17:59:10 -0700706}
707
Abhijeet Dharmapurikarfe1306b2012-11-15 18:09:31 -0800708static int configure_ldo_or_hs_all(struct krait_power_vreg *from, int vmax)
Abhijeet Dharmapurikar00269e52012-05-14 17:59:10 -0700709{
Abhijeet Dharmapurikar00269e52012-05-14 17:59:10 -0700710 struct pmic_gang_vreg *pvreg = from->pvreg;
Abhijeet Dharmapurikar152a7082012-05-14 20:25:59 -0700711 struct krait_power_vreg *kvreg;
712 int rc = 0;
713
714 list_for_each_entry(kvreg, &pvreg->krait_power_vregs, link) {
Michael Bohanb5d1cdb2012-11-08 18:09:52 -0800715 if (!kvreg->online)
716 continue;
Abhijeet Dharmapurikar2269db02012-12-05 15:49:20 -0800717 if (kvreg->uV <= kvreg->ldo_threshold_uV
718 && kvreg->uV - kvreg->ldo_delta_uV + kvreg->headroom_uV
719 <= vmax) {
720 rc = switch_to_using_ldo(kvreg);
Abhijeet Dharmapurikar152a7082012-05-14 20:25:59 -0700721 if (rc < 0) {
Abhijeet Dharmapurikar2269db02012-12-05 15:49:20 -0800722 pr_err("could not switch %s to ldo rc = %d\n",
Abhijeet Dharmapurikar152a7082012-05-14 20:25:59 -0700723 kvreg->name, rc);
724 return rc;
725 }
726 } else {
Abhijeet Dharmapurikar2269db02012-12-05 15:49:20 -0800727 rc = switch_to_using_hs(kvreg);
Abhijeet Dharmapurikar152a7082012-05-14 20:25:59 -0700728 if (rc < 0) {
Abhijeet Dharmapurikar2269db02012-12-05 15:49:20 -0800729 pr_err("could not switch %s to hs rc = %d\n",
Abhijeet Dharmapurikar152a7082012-05-14 20:25:59 -0700730 kvreg->name, rc);
731 return rc;
732 }
733 }
734 }
735
736 return rc;
737}
738
739#define SLEW_RATE 2994
Abhijeet Dharmapurikarfe1306b2012-11-15 18:09:31 -0800740static int krait_voltage_increase(struct krait_power_vreg *from,
Abhijeet Dharmapurikar152a7082012-05-14 20:25:59 -0700741 int vmax)
742{
743 struct pmic_gang_vreg *pvreg = from->pvreg;
744 int rc = 0;
Abhijeet Dharmapurikar00269e52012-05-14 17:59:10 -0700745 int settling_us;
746
Abhijeet Dharmapurikar152a7082012-05-14 20:25:59 -0700747 /*
Abhijeet Dharmapurikarfe1306b2012-11-15 18:09:31 -0800748 * since krait voltage is increasing set the gang voltage
Abhijeet Dharmapurikar152a7082012-05-14 20:25:59 -0700749 * prior to changing ldo/hs states of the requesting krait
750 */
Abhijeet Dharmapurikarfe1306b2012-11-15 18:09:31 -0800751 rc = set_pmic_gang_voltage(pvreg, vmax);
Abhijeet Dharmapurikar152a7082012-05-14 20:25:59 -0700752 if (rc < 0) {
753 dev_err(&from->rdev->dev, "%s failed set voltage %d rc = %d\n",
754 pvreg->name, vmax, rc);
Abhijeet Dharmapurikarfe1306b2012-11-15 18:09:31 -0800755 return rc;
Abhijeet Dharmapurikar152a7082012-05-14 20:25:59 -0700756 }
757
Abhijeet Dharmapurikar749ecd42012-11-08 17:29:04 -0800758
759 /* complete the above writes before the delay */
760 mb();
761
Abhijeet Dharmapurikar152a7082012-05-14 20:25:59 -0700762 /* delay until the voltage is settled when it is raised */
763 settling_us = DIV_ROUND_UP(vmax - pvreg->pmic_vmax_uV, SLEW_RATE);
764 udelay(settling_us);
765
Abhijeet Dharmapurikarfe1306b2012-11-15 18:09:31 -0800766 rc = configure_ldo_or_hs_all(from, vmax);
Abhijeet Dharmapurikar152a7082012-05-14 20:25:59 -0700767 if (rc < 0) {
768 dev_err(&from->rdev->dev, "%s failed ldo/hs conf %d rc = %d\n",
769 pvreg->name, vmax, rc);
770 }
771
772 return rc;
773}
774
Abhijeet Dharmapurikarfe1306b2012-11-15 18:09:31 -0800775static int krait_voltage_decrease(struct krait_power_vreg *from,
Abhijeet Dharmapurikar152a7082012-05-14 20:25:59 -0700776 int vmax)
777{
778 struct pmic_gang_vreg *pvreg = from->pvreg;
779 int rc = 0;
780
781 /*
Abhijeet Dharmapurikarfe1306b2012-11-15 18:09:31 -0800782 * since krait voltage is decreasing ldos might get out of their
Abhijeet Dharmapurikar152a7082012-05-14 20:25:59 -0700783 * operating range. Hence configure such kraits to be in hs mode prior
784 * to setting the pmic gang voltage
785 */
Abhijeet Dharmapurikarfe1306b2012-11-15 18:09:31 -0800786 rc = configure_ldo_or_hs_all(from, vmax);
Abhijeet Dharmapurikar152a7082012-05-14 20:25:59 -0700787 if (rc < 0) {
788 dev_err(&from->rdev->dev, "%s failed ldo/hs conf %d rc = %d\n",
789 pvreg->name, vmax, rc);
790 return rc;
791 }
Abhijeet Dharmapurikar00269e52012-05-14 17:59:10 -0700792
Abhijeet Dharmapurikarfe1306b2012-11-15 18:09:31 -0800793 rc = set_pmic_gang_voltage(pvreg, vmax);
Abhijeet Dharmapurikar00269e52012-05-14 17:59:10 -0700794 if (rc < 0) {
795 dev_err(&from->rdev->dev, "%s failed set voltage %d rc = %d\n",
796 pvreg->name, vmax, rc);
Abhijeet Dharmapurikar00269e52012-05-14 17:59:10 -0700797 }
798
Abhijeet Dharmapurikar00269e52012-05-14 17:59:10 -0700799 return rc;
800}
801
Abhijeet Dharmapurikar00269e52012-05-14 17:59:10 -0700802static int krait_power_get_voltage(struct regulator_dev *rdev)
803{
804 struct krait_power_vreg *kvreg = rdev_get_drvdata(rdev);
805
806 return kvreg->uV;
807}
808
Abhijeet Dharmapurikarfe1306b2012-11-15 18:09:31 -0800809static int get_vmax(struct pmic_gang_vreg *pvreg)
Abhijeet Dharmapurikar00269e52012-05-14 17:59:10 -0700810{
811 int vmax = 0;
812 int v;
813 struct krait_power_vreg *kvreg;
Abhijeet Dharmapurikar00269e52012-05-14 17:59:10 -0700814
815 list_for_each_entry(kvreg, &pvreg->krait_power_vregs, link) {
Michael Bohanb5d1cdb2012-11-08 18:09:52 -0800816 if (!kvreg->online)
817 continue;
818
Abhijeet Dharmapurikar00269e52012-05-14 17:59:10 -0700819 v = kvreg->uV;
820
Abhijeet Dharmapurikar00269e52012-05-14 17:59:10 -0700821 if (vmax < v)
822 vmax = v;
823 }
Michael Bohanb5d1cdb2012-11-08 18:09:52 -0800824
Abhijeet Dharmapurikar00269e52012-05-14 17:59:10 -0700825 return vmax;
826}
827
Abhijeet Dharmapurikar152a7082012-05-14 20:25:59 -0700828#define ROUND_UP_VOLTAGE(v, res) (DIV_ROUND_UP(v, res) * res)
Michael Bohanb5d1cdb2012-11-08 18:09:52 -0800829static int _set_voltage(struct regulator_dev *rdev,
Abhijeet Dharmapurikarfe1306b2012-11-15 18:09:31 -0800830 int orig_krait_uV, int requested_uV)
Abhijeet Dharmapurikar00269e52012-05-14 17:59:10 -0700831{
832 struct krait_power_vreg *kvreg = rdev_get_drvdata(rdev);
833 struct pmic_gang_vreg *pvreg = kvreg->pvreg;
834 int rc;
835 int vmax;
Patrick Cain92f4fa12013-04-22 16:23:19 -0700836 int coeff_total;
Abhijeet Dharmapurikar00269e52012-05-14 17:59:10 -0700837
Abhijeet Dharmapurikarfe1306b2012-11-15 18:09:31 -0800838 pr_debug("%s: %d to %d\n", kvreg->name, orig_krait_uV, requested_uV);
839 /*
840 * Assign the voltage before updating the gang voltage as we iterate
841 * over all the core voltages and choose HS or LDO for each of them
842 */
843 kvreg->uV = requested_uV;
844
845 vmax = get_vmax(pvreg);
Michael Bohanb5d1cdb2012-11-08 18:09:52 -0800846
847 /* round up the pmic voltage as per its resolution */
848 vmax = ROUND_UP_VOLTAGE(vmax, LV_RANGE_STEP);
849
Abhijeet Dharmapurikarfe1306b2012-11-15 18:09:31 -0800850 if (requested_uV > orig_krait_uV)
851 rc = krait_voltage_increase(kvreg, vmax);
852 else
853 rc = krait_voltage_decrease(kvreg, vmax);
854
Michael Bohanb5d1cdb2012-11-08 18:09:52 -0800855 if (rc < 0) {
Abhijeet Dharmapurikarfe1306b2012-11-15 18:09:31 -0800856 dev_err(&rdev->dev, "%s failed to set %duV from %duV rc = %d\n",
857 kvreg->name, requested_uV, orig_krait_uV, rc);
Michael Bohanb5d1cdb2012-11-08 18:09:52 -0800858 }
859
Patrick Cain955d3522013-04-22 16:25:24 -0700860 kvreg->online_at_probe &= ~WAIT_FOR_VOLTAGE;
Patrick Cain92f4fa12013-04-22 16:23:19 -0700861 coeff_total = get_coeff_total(kvreg);
862 /* adjust the phases since coeff2 would have changed */
863 rc = pmic_gang_set_phases(kvreg, coeff_total);
864
Michael Bohanb5d1cdb2012-11-08 18:09:52 -0800865 return rc;
866}
867
868static int krait_power_set_voltage(struct regulator_dev *rdev,
869 int min_uV, int max_uV, unsigned *selector)
870{
871 struct krait_power_vreg *kvreg = rdev_get_drvdata(rdev);
872 struct pmic_gang_vreg *pvreg = kvreg->pvreg;
873 int rc;
874
Abhijeet Dharmapurikar152a7082012-05-14 20:25:59 -0700875 /*
876 * if the voltage requested is below LDO_THRESHOLD this cpu could
877 * switch to LDO mode. Hence round the voltage as per the LDO
878 * resolution
879 */
Abhijeet Dharmapurikar34f0d642012-10-25 12:35:23 -0700880 if (min_uV < kvreg->ldo_threshold_uV) {
Abhijeet Dharmapurikar152a7082012-05-14 20:25:59 -0700881 if (min_uV < KRAIT_LDO_VOLTAGE_MIN)
882 min_uV = KRAIT_LDO_VOLTAGE_MIN;
883 min_uV = ROUND_UP_VOLTAGE(min_uV, KRAIT_LDO_STEP);
884 }
885
Abhijeet Dharmapurikar00269e52012-05-14 17:59:10 -0700886 mutex_lock(&pvreg->krait_power_vregs_lock);
Michael Bohanb5d1cdb2012-11-08 18:09:52 -0800887 if (!kvreg->online) {
Abhijeet Dharmapurikarfe1306b2012-11-15 18:09:31 -0800888 kvreg->uV = min_uV;
Michael Bohanb5d1cdb2012-11-08 18:09:52 -0800889 mutex_unlock(&pvreg->krait_power_vregs_lock);
890 return 0;
Abhijeet Dharmapurikar00269e52012-05-14 17:59:10 -0700891 }
Abhijeet Dharmapurikar152a7082012-05-14 20:25:59 -0700892
Abhijeet Dharmapurikarfe1306b2012-11-15 18:09:31 -0800893 rc = _set_voltage(rdev, kvreg->uV, min_uV);
Abhijeet Dharmapurikar00269e52012-05-14 17:59:10 -0700894 mutex_unlock(&pvreg->krait_power_vregs_lock);
Michael Bohanb5d1cdb2012-11-08 18:09:52 -0800895
Abhijeet Dharmapurikar00269e52012-05-14 17:59:10 -0700896 return rc;
897}
898
Michael Bohanb5d1cdb2012-11-08 18:09:52 -0800899static int krait_power_is_enabled(struct regulator_dev *rdev)
900{
901 struct krait_power_vreg *kvreg = rdev_get_drvdata(rdev);
902
903 return kvreg->online;
904}
905
906static int krait_power_enable(struct regulator_dev *rdev)
907{
908 struct krait_power_vreg *kvreg = rdev_get_drvdata(rdev);
909 struct pmic_gang_vreg *pvreg = kvreg->pvreg;
910 int rc;
911
912 mutex_lock(&pvreg->krait_power_vregs_lock);
Abhijeet Dharmapurikarc8605a62013-03-11 21:24:25 -0700913 __krait_power_mdd_enable(kvreg, true);
Michael Bohanb5d1cdb2012-11-08 18:09:52 -0800914 kvreg->online = true;
Patrick Cain92f4fa12013-04-22 16:23:19 -0700915 rc = _get_optimum_mode(rdev, kvreg->uV, kvreg->uV, kvreg->load);
Michael Bohanb5d1cdb2012-11-08 18:09:52 -0800916 if (rc < 0)
917 goto en_err;
Abhijeet Dharmapurikarfe1306b2012-11-15 18:09:31 -0800918 /*
919 * since the core is being enabled, behave as if it is increasing
920 * the core voltage
921 */
922 rc = _set_voltage(rdev, 0, kvreg->uV);
Michael Bohanb5d1cdb2012-11-08 18:09:52 -0800923en_err:
924 mutex_unlock(&pvreg->krait_power_vregs_lock);
925 return rc;
926}
927
928static int krait_power_disable(struct regulator_dev *rdev)
929{
930 struct krait_power_vreg *kvreg = rdev_get_drvdata(rdev);
931 struct pmic_gang_vreg *pvreg = kvreg->pvreg;
932 int rc;
933
934 mutex_lock(&pvreg->krait_power_vregs_lock);
935 kvreg->online = false;
936
Patrick Cain92f4fa12013-04-22 16:23:19 -0700937 rc = _get_optimum_mode(rdev, kvreg->uV, kvreg->uV, kvreg->load);
Michael Bohanb5d1cdb2012-11-08 18:09:52 -0800938 if (rc < 0)
939 goto dis_err;
940
Abhijeet Dharmapurikarfe1306b2012-11-15 18:09:31 -0800941 rc = _set_voltage(rdev, kvreg->uV, kvreg->uV);
Abhijeet Dharmapurikarc8605a62013-03-11 21:24:25 -0700942 __krait_power_mdd_enable(kvreg, false);
Michael Bohanb5d1cdb2012-11-08 18:09:52 -0800943dis_err:
944 mutex_unlock(&pvreg->krait_power_vregs_lock);
945 return rc;
946}
947
Abhijeet Dharmapurikar00269e52012-05-14 17:59:10 -0700948static struct regulator_ops krait_power_ops = {
949 .get_voltage = krait_power_get_voltage,
950 .set_voltage = krait_power_set_voltage,
951 .get_optimum_mode = krait_power_get_optimum_mode,
952 .set_mode = krait_power_set_mode,
953 .get_mode = krait_power_get_mode,
Michael Bohanb5d1cdb2012-11-08 18:09:52 -0800954 .enable = krait_power_enable,
955 .disable = krait_power_disable,
956 .is_enabled = krait_power_is_enabled,
Abhijeet Dharmapurikar00269e52012-05-14 17:59:10 -0700957};
958
Abhijeet Dharmapurikar37c47a82013-01-08 18:44:21 -0800959static struct dentry *dent;
960static int get_retention_dbg_uV(void *data, u64 *val)
961{
962 struct pmic_gang_vreg *pvreg = data;
963 struct krait_power_vreg *kvreg;
964
965 mutex_lock(&pvreg->krait_power_vregs_lock);
966 if (!list_empty(&pvreg->krait_power_vregs)) {
967 /* return the retention voltage on just the first cpu */
968 kvreg = list_entry((&pvreg->krait_power_vregs)->next,
969 typeof(*kvreg), link);
970 *val = get_krait_retention_ldo_uv(kvreg);
971 }
972 mutex_unlock(&pvreg->krait_power_vregs_lock);
973 return 0;
974}
975
976static int set_retention_dbg_uV(void *data, u64 val)
977{
978 struct pmic_gang_vreg *pvreg = data;
979 struct krait_power_vreg *kvreg;
980 int retention_uV = val;
981
982 if (!is_between(LDO_UV_MIN, LDO_UV_MAX, retention_uV))
983 return -EINVAL;
984
985 mutex_lock(&pvreg->krait_power_vregs_lock);
986 list_for_each_entry(kvreg, &pvreg->krait_power_vregs, link) {
987 kvreg->retention_uV = retention_uV;
988 set_krait_retention_uv(kvreg, retention_uV);
989 }
990 mutex_unlock(&pvreg->krait_power_vregs_lock);
991 return 0;
992}
993DEFINE_SIMPLE_ATTRIBUTE(retention_fops,
994 get_retention_dbg_uV, set_retention_dbg_uV, "%llu\n");
995
Patrick Cain955d3522013-04-22 16:25:24 -0700996#define CPU_PWR_CTL_ONLINE_MASK 0x80
Abhijeet Dharmapurikar152a7082012-05-14 20:25:59 -0700997static void kvreg_hw_init(struct krait_power_vreg *kvreg)
998{
Patrick Cain955d3522013-04-22 16:25:24 -0700999 int online;
Abhijeet Dharmapurikar152a7082012-05-14 20:25:59 -07001000 /*
1001 * bhs_cnt value sets the ramp-up time from power collapse,
1002 * initialize the ramp up time
1003 */
Abhijeet Dharmapurikar34f0d642012-10-25 12:35:23 -07001004 set_krait_retention_uv(kvreg, kvreg->retention_uV);
1005 set_krait_ldo_uv(kvreg, kvreg->ldo_default_uV);
Abhijeet Dharmapurikar749ecd42012-11-08 17:29:04 -08001006
1007 /* setup the bandgap that configures the reference to the LDO */
1008 writel_relaxed(0x00000190, kvreg->mdd_base + MDD_CONFIG_CTL);
Abhijeet Dharmapurikarc8605a62013-03-11 21:24:25 -07001009 /* Enable MDD */
1010 writel_relaxed(0x00000002, kvreg->mdd_base + MDD_MODE);
Abhijeet Dharmapurikar749ecd42012-11-08 17:29:04 -08001011 mb();
Patrick Cain955d3522013-04-22 16:25:24 -07001012 online = CPU_PWR_CTL_ONLINE_MASK
1013 & readl_relaxed(kvreg->reg_base + CPU_PWR_CTL);
1014 kvreg->online_at_probe
1015 = online ? (WAIT_FOR_LOAD | WAIT_FOR_VOLTAGE) : 0x0;
Abhijeet Dharmapurikar152a7082012-05-14 20:25:59 -07001016}
1017
Stepan Moskovchenko7663b782013-03-08 20:13:06 -08001018static void glb_init(void __iomem *apcs_gcc_base)
Abhijeet Dharmapurikar3ff7cba2012-10-10 17:48:29 -07001019{
1020 /* configure bi-modal switch */
Stepan Moskovchenko7663b782013-03-08 20:13:06 -08001021 writel_relaxed(0x0008736E, apcs_gcc_base + PWR_GATE_CONFIG);
Abhijeet Dharmapurikar3ff7cba2012-10-10 17:48:29 -07001022 /* read kpss version */
Stepan Moskovchenko7663b782013-03-08 20:13:06 -08001023 version = readl_relaxed(apcs_gcc_base + VERSION);
Abhijeet Dharmapurikar3ff7cba2012-10-10 17:48:29 -07001024 pr_debug("version= 0x%x\n", version);
1025}
1026
Abhijeet Dharmapurikar00269e52012-05-14 17:59:10 -07001027static int __devinit krait_power_probe(struct platform_device *pdev)
1028{
1029 struct krait_power_vreg *kvreg;
Abhijeet Dharmapurikar749ecd42012-11-08 17:29:04 -08001030 struct resource *res, *res_mdd;
Abhijeet Dharmapurikar00269e52012-05-14 17:59:10 -07001031 struct regulator_init_data *init_data = pdev->dev.platform_data;
1032 int rc = 0;
Abhijeet Dharmapurikar34f0d642012-10-25 12:35:23 -07001033 int headroom_uV, retention_uV, ldo_default_uV, ldo_threshold_uV;
Abhijeet Dharmapurikar2269db02012-12-05 15:49:20 -08001034 int ldo_delta_uV;
Abhijeet Dharmapurikare8f83852013-02-06 18:57:17 -08001035 int cpu_num;
Abhijeet Dharmapurikar00269e52012-05-14 17:59:10 -07001036
Abhijeet Dharmapurikar00269e52012-05-14 17:59:10 -07001037 if (pdev->dev.of_node) {
1038 /* Get init_data from device tree. */
1039 init_data = of_get_regulator_init_data(&pdev->dev,
1040 pdev->dev.of_node);
1041 init_data->constraints.valid_ops_mask
1042 |= REGULATOR_CHANGE_VOLTAGE | REGULATOR_CHANGE_DRMS
1043 | REGULATOR_CHANGE_MODE;
1044 init_data->constraints.valid_modes_mask
1045 |= REGULATOR_MODE_NORMAL | REGULATOR_MODE_IDLE
1046 | REGULATOR_MODE_FAST;
1047 init_data->constraints.input_uV = init_data->constraints.max_uV;
Abhijeet Dharmapurikar34f0d642012-10-25 12:35:23 -07001048 rc = of_property_read_u32(pdev->dev.of_node,
1049 "qcom,headroom-voltage",
1050 &headroom_uV);
1051 if (rc < 0) {
1052 pr_err("headroom-voltage missing rc=%d\n", rc);
1053 return rc;
1054 }
1055 if (!is_between(LDO_HDROOM_MIN, LDO_HDROOM_MAX, headroom_uV)) {
1056 pr_err("bad headroom-voltage = %d specified\n",
1057 headroom_uV);
1058 return -EINVAL;
1059 }
1060
1061 rc = of_property_read_u32(pdev->dev.of_node,
1062 "qcom,retention-voltage",
1063 &retention_uV);
1064 if (rc < 0) {
1065 pr_err("retention-voltage missing rc=%d\n", rc);
1066 return rc;
1067 }
1068 if (!is_between(LDO_UV_MIN, LDO_UV_MAX, retention_uV)) {
1069 pr_err("bad retention-voltage = %d specified\n",
1070 retention_uV);
1071 return -EINVAL;
1072 }
1073
1074 rc = of_property_read_u32(pdev->dev.of_node,
1075 "qcom,ldo-default-voltage",
1076 &ldo_default_uV);
1077 if (rc < 0) {
1078 pr_err("ldo-default-voltage missing rc=%d\n", rc);
1079 return rc;
1080 }
1081 if (!is_between(LDO_UV_MIN, LDO_UV_MAX, ldo_default_uV)) {
1082 pr_err("bad ldo-default-voltage = %d specified\n",
1083 ldo_default_uV);
1084 return -EINVAL;
1085 }
1086
1087 rc = of_property_read_u32(pdev->dev.of_node,
1088 "qcom,ldo-threshold-voltage",
1089 &ldo_threshold_uV);
1090 if (rc < 0) {
1091 pr_err("ldo-threshold-voltage missing rc=%d\n", rc);
1092 return rc;
1093 }
1094 if (!is_between(LDO_TH_MIN, LDO_TH_MAX, ldo_threshold_uV)) {
1095 pr_err("bad ldo-threshold-voltage = %d specified\n",
1096 ldo_threshold_uV);
1097 return -EINVAL;
1098 }
Abhijeet Dharmapurikar2269db02012-12-05 15:49:20 -08001099
1100 rc = of_property_read_u32(pdev->dev.of_node,
1101 "qcom,ldo-delta-voltage",
1102 &ldo_delta_uV);
1103 if (rc < 0) {
1104 pr_err("ldo-delta-voltage missing rc=%d\n", rc);
1105 return rc;
1106 }
1107 if (!is_between(LDO_DELTA_MIN, LDO_DELTA_MAX, ldo_delta_uV)) {
1108 pr_err("bad ldo-delta-voltage = %d specified\n",
1109 ldo_delta_uV);
1110 return -EINVAL;
1111 }
Abhijeet Dharmapurikare8f83852013-02-06 18:57:17 -08001112 rc = of_property_read_u32(pdev->dev.of_node,
1113 "qcom,cpu-num",
1114 &cpu_num);
1115 if (cpu_num > num_possible_cpus()) {
1116 pr_err("bad cpu-num= %d specified\n", cpu_num);
1117 return -EINVAL;
1118 }
Abhijeet Dharmapurikar00269e52012-05-14 17:59:10 -07001119 }
1120
1121 if (!init_data) {
1122 dev_err(&pdev->dev, "init data required.\n");
1123 return -EINVAL;
1124 }
1125
1126 if (!init_data->constraints.name) {
1127 dev_err(&pdev->dev,
1128 "regulator name must be specified in constraints.\n");
1129 return -EINVAL;
1130 }
1131
Abhijeet Dharmapurikar749ecd42012-11-08 17:29:04 -08001132 res = platform_get_resource_byname(pdev, IORESOURCE_MEM, "acs");
Abhijeet Dharmapurikar00269e52012-05-14 17:59:10 -07001133 if (!res) {
1134 dev_err(&pdev->dev, "missing physical register addresses\n");
1135 return -EINVAL;
1136 }
1137
Abhijeet Dharmapurikar749ecd42012-11-08 17:29:04 -08001138 res_mdd = platform_get_resource_byname(pdev, IORESOURCE_MEM, "mdd");
1139 if (!res_mdd) {
1140 dev_err(&pdev->dev, "missing mdd register addresses\n");
1141 return -EINVAL;
1142 }
1143
Abhijeet Dharmapurikar00269e52012-05-14 17:59:10 -07001144 kvreg = devm_kzalloc(&pdev->dev,
1145 sizeof(struct krait_power_vreg), GFP_KERNEL);
1146 if (!kvreg) {
1147 dev_err(&pdev->dev, "kzalloc failed.\n");
1148 return -ENOMEM;
1149 }
1150
1151 kvreg->reg_base = devm_ioremap(&pdev->dev,
1152 res->start, resource_size(res));
1153
Abhijeet Dharmapurikar749ecd42012-11-08 17:29:04 -08001154 kvreg->mdd_base = devm_ioremap(&pdev->dev,
1155 res_mdd->start, resource_size(res));
1156
Abhijeet Dharmapurikar34f0d642012-10-25 12:35:23 -07001157 kvreg->pvreg = the_gang;
1158 kvreg->name = init_data->constraints.name;
1159 kvreg->desc.name = kvreg->name;
1160 kvreg->desc.ops = &krait_power_ops;
1161 kvreg->desc.type = REGULATOR_VOLTAGE;
1162 kvreg->desc.owner = THIS_MODULE;
Patrick Cain92f4fa12013-04-22 16:23:19 -07001163 kvreg->uV = CORE_VOLTAGE_BOOTUP;
Abhijeet Dharmapurikar34f0d642012-10-25 12:35:23 -07001164 kvreg->mode = HS_MODE;
1165 kvreg->desc.ops = &krait_power_ops;
1166 kvreg->headroom_uV = headroom_uV;
1167 kvreg->retention_uV = retention_uV;
1168 kvreg->ldo_default_uV = ldo_default_uV;
1169 kvreg->ldo_threshold_uV = ldo_threshold_uV;
Abhijeet Dharmapurikar2269db02012-12-05 15:49:20 -08001170 kvreg->ldo_delta_uV = ldo_delta_uV;
Abhijeet Dharmapurikare8f83852013-02-06 18:57:17 -08001171 kvreg->cpu_num = cpu_num;
Abhijeet Dharmapurikar00269e52012-05-14 17:59:10 -07001172
1173 platform_set_drvdata(pdev, kvreg);
1174
1175 mutex_lock(&the_gang->krait_power_vregs_lock);
Abhijeet Dharmapurikar295c4dc2013-01-08 12:59:01 -08001176 the_gang->pmic_min_uV_for_retention
1177 = min(the_gang->pmic_min_uV_for_retention,
1178 kvreg->retention_uV + kvreg->headroom_uV);
Abhijeet Dharmapurikar00269e52012-05-14 17:59:10 -07001179 list_add_tail(&kvreg->link, &the_gang->krait_power_vregs);
1180 mutex_unlock(&the_gang->krait_power_vregs_lock);
1181
1182 kvreg->rdev = regulator_register(&kvreg->desc, &pdev->dev, init_data,
1183 kvreg, pdev->dev.of_node);
1184 if (IS_ERR(kvreg->rdev)) {
1185 rc = PTR_ERR(kvreg->rdev);
1186 pr_err("regulator_register failed, rc=%d.\n", rc);
1187 goto out;
1188 }
1189
Abhijeet Dharmapurikar152a7082012-05-14 20:25:59 -07001190 kvreg_hw_init(kvreg);
Abhijeet Dharmapurikare8f83852013-02-06 18:57:17 -08001191 per_cpu(krait_vregs, cpu_num) = kvreg;
Abhijeet Dharmapurikar00269e52012-05-14 17:59:10 -07001192 dev_dbg(&pdev->dev, "id=%d, name=%s\n", pdev->id, kvreg->name);
1193
1194 return 0;
1195out:
1196 mutex_lock(&the_gang->krait_power_vregs_lock);
1197 list_del(&kvreg->link);
1198 mutex_unlock(&the_gang->krait_power_vregs_lock);
1199
1200 platform_set_drvdata(pdev, NULL);
1201 return rc;
1202}
1203
1204static int __devexit krait_power_remove(struct platform_device *pdev)
1205{
1206 struct krait_power_vreg *kvreg = platform_get_drvdata(pdev);
1207 struct pmic_gang_vreg *pvreg = kvreg->pvreg;
1208
1209 mutex_lock(&pvreg->krait_power_vregs_lock);
1210 list_del(&kvreg->link);
1211 mutex_unlock(&pvreg->krait_power_vregs_lock);
1212
1213 regulator_unregister(kvreg->rdev);
1214 platform_set_drvdata(pdev, NULL);
1215 return 0;
1216}
1217
1218static struct of_device_id krait_power_match_table[] = {
1219 { .compatible = "qcom,krait-regulator", },
1220 {}
1221};
1222
1223static struct platform_driver krait_power_driver = {
1224 .probe = krait_power_probe,
1225 .remove = __devexit_p(krait_power_remove),
1226 .driver = {
1227 .name = KRAIT_REGULATOR_DRIVER_NAME,
1228 .of_match_table = krait_power_match_table,
1229 .owner = THIS_MODULE,
1230 },
1231};
1232
Abhijeet Dharmapurikar46a7ad32013-02-23 12:50:19 -08001233static struct of_device_id krait_pdn_match_table[] = {
1234 { .compatible = "qcom,krait-pdn", },
1235 {}
1236};
1237
Abhijeet Dharmapurikarc8605a62013-03-11 21:24:25 -07001238static int boot_cpu_mdd_off(void)
1239{
1240 struct krait_power_vreg *kvreg = per_cpu(krait_vregs, 0);
1241
1242 __krait_power_mdd_enable(kvreg, false);
1243 return 0;
1244}
1245
1246static void boot_cpu_mdd_on(void)
1247{
1248 struct krait_power_vreg *kvreg = per_cpu(krait_vregs, 0);
1249
1250 __krait_power_mdd_enable(kvreg, true);
1251}
1252
1253static struct syscore_ops boot_cpu_mdd_ops = {
1254 .suspend = boot_cpu_mdd_off,
1255 .resume = boot_cpu_mdd_on,
1256};
1257
David Collinsb80db492013-04-26 15:05:41 -07001258static int __devinit krait_pdn_phase_scaling_init(struct pmic_gang_vreg *pvreg,
1259 struct platform_device *pdev)
1260{
1261 struct resource *res;
1262 void __iomem *efuse;
1263 u32 efuse_data, efuse_version;
1264 bool scaling_factor_valid, use_efuse;
1265
1266 use_efuse = of_property_read_bool(pdev->dev.of_node,
1267 "qcom,use-phase-scaling-factor");
1268 /*
1269 * Allow usage of the eFuse phase scaling factor if it is enabled in
1270 * either device tree or by module parameter.
1271 */
1272 use_efuse_phase_scaling_factor = use_efuse_phase_scaling_factor
1273 || use_efuse;
1274
1275 res = platform_get_resource_byname(pdev, IORESOURCE_MEM,
1276 "phase-scaling-efuse");
1277 if (!res || !res->start) {
1278 pr_err("phase scaling eFuse address is missing\n");
1279 return -EINVAL;
1280 }
1281
1282 efuse = ioremap(res->start, 8);
1283 if (!efuse) {
1284 pr_err("could not map phase scaling eFuse address\n");
1285 return -EINVAL;
1286 }
1287
1288 efuse_data = readl_relaxed(efuse);
1289 efuse_version = readl_relaxed(efuse + 4);
1290
1291 iounmap(efuse);
1292
1293 scaling_factor_valid
1294 = ((efuse_version & PHASE_SCALING_EFUSE_VERSION_MASK) >>
1295 PHASE_SCALING_EFUSE_VERSION_POS)
1296 == PHASE_SCALING_EFUSE_VERSION_SET;
1297
1298 if (scaling_factor_valid)
1299 pvreg->efuse_phase_scaling_factor
1300 = ((efuse_data & PHASE_SCALING_EFUSE_VALUE_MASK)
1301 >> PHASE_SCALING_EFUSE_VALUE_POS) + 1;
1302 else
1303 pvreg->efuse_phase_scaling_factor = PHASE_SCALING_REF;
1304
1305 pr_info("eFuse phase scaling factor = %d/%d%s\n",
1306 pvreg->efuse_phase_scaling_factor, PHASE_SCALING_REF,
1307 scaling_factor_valid ? "" : " (eFuse not blown)");
1308 pr_info("initial phase scaling factor = %d/%d%s\n",
1309 use_efuse_phase_scaling_factor
1310 ? pvreg->efuse_phase_scaling_factor : PHASE_SCALING_REF,
1311 PHASE_SCALING_REF,
1312 use_efuse_phase_scaling_factor ? "" : " (ignoring eFuse)");
1313
1314 return 0;
1315}
1316
Abhijeet Dharmapurikar46a7ad32013-02-23 12:50:19 -08001317static int __devinit krait_pdn_probe(struct platform_device *pdev)
1318{
1319 int rc;
1320 bool use_phase_switching = false;
Patrick Cain92f4fa12013-04-22 16:23:19 -07001321 int pfm_threshold;
Abhijeet Dharmapurikar46a7ad32013-02-23 12:50:19 -08001322 struct device *dev = &pdev->dev;
1323 struct device_node *node = dev->of_node;
1324 struct pmic_gang_vreg *pvreg;
Stepan Moskovchenko7663b782013-03-08 20:13:06 -08001325 struct resource *res;
Abhijeet Dharmapurikar46a7ad32013-02-23 12:50:19 -08001326
1327 if (!dev->of_node) {
1328 dev_err(dev, "device tree information missing\n");
1329 return -ENODEV;
1330 }
1331
1332 use_phase_switching = of_property_read_bool(node,
1333 "qcom,use-phase-switching");
Patrick Cain92f4fa12013-04-22 16:23:19 -07001334
1335 rc = of_property_read_u32(node, "qcom,pfm-threshold", &pfm_threshold);
1336 if (rc < 0) {
1337 dev_err(dev, "pfm-threshold missing rc=%d, pfm disabled\n", rc);
1338 return -EINVAL;
1339 }
1340
Abhijeet Dharmapurikar46a7ad32013-02-23 12:50:19 -08001341 pvreg = devm_kzalloc(&pdev->dev,
1342 sizeof(struct pmic_gang_vreg), GFP_KERNEL);
1343 if (!pvreg) {
1344 pr_err("kzalloc failed.\n");
1345 return 0;
1346 }
1347
Stepan Moskovchenko7663b782013-03-08 20:13:06 -08001348 res = platform_get_resource_byname(pdev, IORESOURCE_MEM, "apcs_gcc");
1349 if (!res) {
1350 dev_err(&pdev->dev, "missing apcs gcc base addresses\n");
1351 return -EINVAL;
1352 }
1353
1354 pvreg->apcs_gcc_base = devm_ioremap(&pdev->dev, res->start,
1355 resource_size(res));
1356
1357 if (pvreg->apcs_gcc_base == NULL)
1358 return -ENOMEM;
1359
David Collinsb80db492013-04-26 15:05:41 -07001360 rc = krait_pdn_phase_scaling_init(pvreg, pdev);
1361 if (rc)
1362 return rc;
1363
Abhijeet Dharmapurikar46a7ad32013-02-23 12:50:19 -08001364 pvreg->name = "pmic_gang";
1365 pvreg->pmic_vmax_uV = PMIC_VOLTAGE_MIN;
1366 pvreg->pmic_phase_count = -EINVAL;
1367 pvreg->retention_enabled = true;
1368 pvreg->pmic_min_uV_for_retention = INT_MAX;
1369 pvreg->use_phase_switching = use_phase_switching;
Patrick Cain92f4fa12013-04-22 16:23:19 -07001370 pvreg->pfm_threshold = pfm_threshold;
Abhijeet Dharmapurikar46a7ad32013-02-23 12:50:19 -08001371
1372 mutex_init(&pvreg->krait_power_vregs_lock);
1373 INIT_LIST_HEAD(&pvreg->krait_power_vregs);
1374 the_gang = pvreg;
1375
1376 pr_debug("name=%s inited\n", pvreg->name);
1377
1378 /* global initializtion */
Stepan Moskovchenko7663b782013-03-08 20:13:06 -08001379 glb_init(pvreg->apcs_gcc_base);
Abhijeet Dharmapurikar46a7ad32013-02-23 12:50:19 -08001380
1381 rc = of_platform_populate(node, NULL, NULL, dev);
1382 if (rc) {
1383 dev_err(dev, "failed to add child nodes, rc=%d\n", rc);
1384 return rc;
1385 }
1386
1387 dent = debugfs_create_dir(KRAIT_REGULATOR_DRIVER_NAME, NULL);
1388 debugfs_create_file("retention_uV",
1389 0644, dent, the_gang, &retention_fops);
Abhijeet Dharmapurikarc8605a62013-03-11 21:24:25 -07001390 register_syscore_ops(&boot_cpu_mdd_ops);
Abhijeet Dharmapurikar46a7ad32013-02-23 12:50:19 -08001391 return 0;
1392}
1393
1394static int __devexit krait_pdn_remove(struct platform_device *pdev)
1395{
1396 the_gang = NULL;
1397 debugfs_remove_recursive(dent);
1398 return 0;
1399}
1400
1401static struct platform_driver krait_pdn_driver = {
1402 .probe = krait_pdn_probe,
1403 .remove = __devexit_p(krait_pdn_remove),
1404 .driver = {
1405 .name = KRAIT_PDN_DRIVER_NAME,
1406 .of_match_table = krait_pdn_match_table,
1407 .owner = THIS_MODULE,
1408 },
1409};
1410
Abhijeet Dharmapurikar00269e52012-05-14 17:59:10 -07001411int __init krait_power_init(void)
1412{
Abhijeet Dharmapurikar46a7ad32013-02-23 12:50:19 -08001413 int rc = platform_driver_register(&krait_power_driver);
1414 if (rc) {
1415 pr_err("failed to add %s driver rc = %d\n",
1416 KRAIT_REGULATOR_DRIVER_NAME, rc);
1417 return rc;
1418 }
1419 return platform_driver_register(&krait_pdn_driver);
Abhijeet Dharmapurikar00269e52012-05-14 17:59:10 -07001420}
1421
1422static void __exit krait_power_exit(void)
1423{
1424 platform_driver_unregister(&krait_power_driver);
Abhijeet Dharmapurikar46a7ad32013-02-23 12:50:19 -08001425 platform_driver_unregister(&krait_pdn_driver);
Abhijeet Dharmapurikar00269e52012-05-14 17:59:10 -07001426}
Abhijeet Dharmapurikar00269e52012-05-14 17:59:10 -07001427module_exit(krait_power_exit);
1428
Abhijeet Dharmapurikare8571ef2012-08-14 20:44:58 -07001429void secondary_cpu_hs_init(void *base_ptr)
1430{
Matt Wagantall5a58f452013-04-26 12:59:49 -07001431 uint32_t reg_val;
Mahesh Sivasubramanian3dca48e2013-04-12 14:29:57 -06001432 void *l2_saw_base;
Matt Wagantall5a58f452013-04-26 12:59:49 -07001433
Abhijeet Dharmapurikare8571ef2012-08-14 20:44:58 -07001434 /* Turn on the BHS, turn off LDO Bypass and power down LDO */
Matt Wagantall5a58f452013-04-26 12:59:49 -07001435 reg_val = BHS_CNT_DEFAULT << BHS_CNT_BIT_POS
Abhijeet Dharmapurikar749ecd42012-11-08 17:29:04 -08001436 | LDO_PWR_DWN_MASK
1437 | CLK_SRC_DEFAULT << CLK_SRC_SEL_BIT_POS
Matt Wagantall5a58f452013-04-26 12:59:49 -07001438 | BHS_EN_MASK;
1439 writel_relaxed(reg_val, base_ptr + APC_PWR_GATE_CTL);
Abhijeet Dharmapurikar749ecd42012-11-08 17:29:04 -08001440
1441 /* complete the above write before the delay */
Abhijeet Dharmapurikare8571ef2012-08-14 20:44:58 -07001442 mb();
Matt Wagantall5a58f452013-04-26 12:59:49 -07001443 /* wait for the bhs to settle */
1444 udelay(BHS_SETTLING_DELAY_US);
Abhijeet Dharmapurikar749ecd42012-11-08 17:29:04 -08001445
Matt Wagantall5a58f452013-04-26 12:59:49 -07001446 /* Turn on BHS segments */
1447 reg_val |= BHS_SEG_EN_DEFAULT << BHS_SEG_EN_BIT_POS;
1448 writel_relaxed(reg_val, base_ptr + APC_PWR_GATE_CTL);
1449
1450 /* complete the above write before the delay */
1451 mb();
1452 /* wait for the bhs to settle */
Abhijeet Dharmapurikar749ecd42012-11-08 17:29:04 -08001453 udelay(BHS_SETTLING_DELAY_US);
Abhijeet Dharmapurikare8571ef2012-08-14 20:44:58 -07001454
1455 /* Finally turn on the bypass so that BHS supplies power */
Matt Wagantall5a58f452013-04-26 12:59:49 -07001456 reg_val |= LDO_BYP_MASK;
1457 writel_relaxed(reg_val, base_ptr + APC_PWR_GATE_CTL);
Mahesh Sivasubramanian3dca48e2013-04-12 14:29:57 -06001458
1459 if (the_gang && the_gang->manage_phases)
1460 return;
1461
1462 /*
1463 * If the driver has not yet started to manage phases then enable
1464 * max phases.
1465 */
1466 l2_saw_base = ioremap_nocache(MSM_L2_SAW_PHYS, SZ_4K);
1467 if (!l2_saw_base) {
1468 __WARN();
1469 return;
1470 }
1471 writel_relaxed(0x10003, l2_saw_base + 0x1c);
1472 mb();
1473 udelay(PHASE_SETTLING_TIME_US);
1474
1475 iounmap(l2_saw_base);
Abhijeet Dharmapurikare8571ef2012-08-14 20:44:58 -07001476}
1477
Abhijeet Dharmapurikar00269e52012-05-14 17:59:10 -07001478MODULE_LICENSE("GPL v2");
1479MODULE_DESCRIPTION("KRAIT POWER regulator driver");
Abhijeet Dharmapurikar00269e52012-05-14 17:59:10 -07001480MODULE_ALIAS("platform:"KRAIT_REGULATOR_DRIVER_NAME);