blob: 174d44422035783c9fa916e9fdf655640b147933 [file] [log] [blame]
Anji Jonnala5063e042013-03-09 09:49:11 +05301/* Copyright (c) 2011-2013, The Linux Foundation. All rights reserved.
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -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 */
13
14#include <linux/module.h>
15#include <linux/kernel.h>
16#include <linux/delay.h>
17#include <linux/init.h>
18#include <linux/io.h>
19#include <linux/slab.h>
Praveen Chidambaramaa9d52b2012-04-02 11:09:47 -060020#include <linux/of.h>
21#include <linux/of_address.h>
22#include <linux/platform_device.h>
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -070023#include <mach/msm_iomap.h>
Praveen Chidambaram76679d42011-12-16 14:19:02 -070024#include <mach/socinfo.h>
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -070025#include "spm.h"
26#include "spm_driver.h"
27
28struct msm_spm_power_modes {
29 uint32_t mode;
30 bool notify_rpm;
31 uint32_t start_addr;
32
33};
34
35struct msm_spm_device {
Mahesh Sivasubramanianbc420242013-03-26 17:18:04 -060036 bool initialized;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -070037 struct msm_spm_driver_data reg_data;
38 struct msm_spm_power_modes *modes;
39 uint32_t num_modes;
Anji Jonnala5063e042013-03-09 09:49:11 +053040 uint32_t cpu_vdd;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -070041};
42
Praveen Chidambarambcd483b2012-09-16 14:54:35 -060043struct msm_spm_vdd_info {
44 uint32_t cpu;
45 uint32_t vlevel;
46 int err;
47};
48
Praveen Chidambaramaa9d52b2012-04-02 11:09:47 -060049static struct msm_spm_device msm_spm_l2_device;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -070050static DEFINE_PER_CPU_SHARED_ALIGNED(struct msm_spm_device, msm_cpu_spm_device);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -070051
Praveen Chidambarambcd483b2012-09-16 14:54:35 -060052
Praveen Chidambarambcd483b2012-09-16 14:54:35 -060053static void msm_spm_smp_set_vdd(void *data)
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -070054{
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -070055 struct msm_spm_device *dev;
Praveen Chidambarambcd483b2012-09-16 14:54:35 -060056 struct msm_spm_vdd_info *info = (struct msm_spm_vdd_info *)data;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -070057
Praveen Chidambarambcd483b2012-09-16 14:54:35 -060058 dev = &per_cpu(msm_cpu_spm_device, info->cpu);
Mahesh Sivasubramanianbc420242013-03-26 17:18:04 -060059 if (!dev->initialized)
60 return;
Anji Jonnala5063e042013-03-09 09:49:11 +053061 dev->cpu_vdd = info->vlevel;
Praveen Chidambarambcd483b2012-09-16 14:54:35 -060062 info->err = msm_spm_drv_set_vdd(&dev->reg_data, info->vlevel);
63}
64
Praveen Chidambaram7347bfe2012-11-01 15:21:10 -060065/**
66 * msm_spm_set_vdd(): Set core voltage
67 * @cpu: core id
68 * @vlevel: Encoded PMIC data.
69 */
Praveen Chidambarambcd483b2012-09-16 14:54:35 -060070int msm_spm_set_vdd(unsigned int cpu, unsigned int vlevel)
71{
72 struct msm_spm_vdd_info info;
73 int ret;
74
75 info.cpu = cpu;
76 info.vlevel = vlevel;
Mahesh Sivasubramanian072b2922013-04-16 15:36:54 -060077 info.err = -ENODEV;
Praveen Chidambarambcd483b2012-09-16 14:54:35 -060078
Praveen Chidambaram8624ce72012-10-05 17:11:33 -060079 if (cpu_online(cpu)) {
80 /**
81 * We do not want to set the voltage of another core from
82 * this core, as its possible that we may race the vdd change
83 * with the SPM state machine of that core, which could also
84 * be changing the voltage of that core during power collapse.
85 * Hence, set the function to be executed on that core and block
86 * until the vdd change is complete.
87 */
88 ret = smp_call_function_single(cpu, msm_spm_smp_set_vdd,
89 &info, true);
90 if (!ret)
91 ret = info.err;
92 } else {
93 /**
94 * Since the core is not online, it is safe to set the vdd
95 * directly.
96 */
97 msm_spm_smp_set_vdd(&info);
Praveen Chidambarambcd483b2012-09-16 14:54:35 -060098 ret = info.err;
Praveen Chidambaram8624ce72012-10-05 17:11:33 -060099 }
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700100
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700101 return ret;
102}
Praveen Chidambaramaa9d52b2012-04-02 11:09:47 -0600103EXPORT_SYMBOL(msm_spm_set_vdd);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700104
Praveen Chidambaram7347bfe2012-11-01 15:21:10 -0600105/**
106 * msm_spm_get_vdd(): Get core voltage
107 * @cpu: core id
108 * @return: Returns encoded PMIC data.
109 */
Praveen Chidambaram4133ba12012-09-29 22:27:03 -0600110unsigned int msm_spm_get_vdd(unsigned int cpu)
111{
112 struct msm_spm_device *dev;
113
114 dev = &per_cpu(msm_cpu_spm_device, cpu);
Anji Jonnala5063e042013-03-09 09:49:11 +0530115 return dev->cpu_vdd;
Praveen Chidambaram4133ba12012-09-29 22:27:03 -0600116}
117EXPORT_SYMBOL(msm_spm_get_vdd);
118
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700119static int msm_spm_dev_set_low_power_mode(struct msm_spm_device *dev,
120 unsigned int mode, bool notify_rpm)
121{
122 uint32_t i;
123 uint32_t start_addr = 0;
124 int ret = -EINVAL;
125
Mahesh Sivasubramanianbc420242013-03-26 17:18:04 -0600126 if (!dev->initialized)
127 return -ENXIO;
128
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700129 if (mode == MSM_SPM_MODE_DISABLED) {
130 ret = msm_spm_drv_set_spm_enable(&dev->reg_data, false);
131 } else if (!msm_spm_drv_set_spm_enable(&dev->reg_data, true)) {
132 for (i = 0; i < dev->num_modes; i++) {
133 if ((dev->modes[i].mode == mode) &&
134 (dev->modes[i].notify_rpm == notify_rpm)) {
135 start_addr = dev->modes[i].start_addr;
136 break;
137 }
138 }
139 ret = msm_spm_drv_set_low_power_mode(&dev->reg_data,
140 start_addr);
141 }
142 return ret;
143}
144
Stephen Boyddb354112012-05-09 14:24:58 -0700145static int __devinit msm_spm_dev_init(struct msm_spm_device *dev,
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700146 struct msm_spm_platform_data *data)
147{
148 int i, ret = -ENOMEM;
149 uint32_t offset = 0;
150
151 dev->num_modes = data->num_modes;
152 dev->modes = kmalloc(
153 sizeof(struct msm_spm_power_modes) * dev->num_modes,
154 GFP_KERNEL);
155
156 if (!dev->modes)
157 goto spm_failed_malloc;
158
Praveen Chidambaramaa9d52b2012-04-02 11:09:47 -0600159 dev->reg_data.ver_reg = data->ver_reg;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700160 ret = msm_spm_drv_init(&dev->reg_data, data);
161
162 if (ret)
163 goto spm_failed_init;
164
165 for (i = 0; i < dev->num_modes; i++) {
166
Praveen Chidambaramaa9d52b2012-04-02 11:09:47 -0600167 /* Default offset is 0 and gets updated as we write more
168 * sequences into SPM
169 */
170 dev->modes[i].start_addr = offset;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700171 ret = msm_spm_drv_write_seq_data(&dev->reg_data,
Praveen Chidambaramaa9d52b2012-04-02 11:09:47 -0600172 data->modes[i].cmd, &offset);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700173 if (ret < 0)
174 goto spm_failed_init;
175
176 dev->modes[i].mode = data->modes[i].mode;
177 dev->modes[i].notify_rpm = data->modes[i].notify_rpm;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700178 }
179 msm_spm_drv_flush_seq_entry(&dev->reg_data);
Mahesh Sivasubramanianbc420242013-03-26 17:18:04 -0600180 dev->initialized = true;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700181 return 0;
182
183spm_failed_init:
184 kfree(dev->modes);
185spm_failed_malloc:
186 return ret;
187}
188
Praveen Chidambaram7347bfe2012-11-01 15:21:10 -0600189/**
190 * msm_spm_turn_on_cpu_rail(): Power on cpu rail before turning on core
191 * @cpu: core id
192 */
Praveen Chidambaramc0750ca2012-01-08 10:03:28 -0700193int msm_spm_turn_on_cpu_rail(unsigned int cpu)
194{
195 uint32_t val = 0;
196 uint32_t timeout = 0;
197 void *reg = NULL;
Stepan Moskovchenko2b0b06e2012-02-03 15:03:52 -0800198 void *saw_bases[] = {
199 0,
200 MSM_SAW1_BASE,
201 MSM_SAW2_BASE,
202 MSM_SAW3_BASE
203 };
Praveen Chidambaramc0750ca2012-01-08 10:03:28 -0700204
Stepan Moskovchenko2b0b06e2012-02-03 15:03:52 -0800205 if (cpu == 0 || cpu >= num_possible_cpus())
Praveen Chidambaramc0750ca2012-01-08 10:03:28 -0700206 return -EINVAL;
207
Stepan Moskovchenko2b0b06e2012-02-03 15:03:52 -0800208 reg = saw_bases[cpu];
Praveen Chidambaramc0750ca2012-01-08 10:03:28 -0700209
Stepan Moskovchenko5b9e7762012-09-21 20:32:17 -0700210 if (soc_class_is_msm8960() || soc_class_is_msm8930() ||
211 soc_class_is_apq8064()) {
Stepan Moskovchenko2b0b06e2012-02-03 15:03:52 -0800212 val = 0xA4;
213 reg += 0x14;
214 timeout = 512;
Praveen Chidambaramc0750ca2012-01-08 10:03:28 -0700215 } else {
216 return -ENOSYS;
217 }
218
219 writel_relaxed(val, reg);
220 mb();
221 udelay(timeout);
222
223 return 0;
224}
225EXPORT_SYMBOL(msm_spm_turn_on_cpu_rail);
226
Praveen Chidambaramaa9d52b2012-04-02 11:09:47 -0600227void msm_spm_reinit(void)
228{
229 unsigned int cpu;
230 for_each_possible_cpu(cpu)
231 msm_spm_drv_reinit(&per_cpu(msm_cpu_spm_device.reg_data, cpu));
232}
233EXPORT_SYMBOL(msm_spm_reinit);
234
Praveen Chidambaram7347bfe2012-11-01 15:21:10 -0600235/**
236 * msm_spm_set_low_power_mode() - Configure SPM start address for low power mode
237 * @mode: SPM LPM mode to enter
238 * @notify_rpm: Notify RPM in this mode
239 */
Praveen Chidambaramaa9d52b2012-04-02 11:09:47 -0600240int msm_spm_set_low_power_mode(unsigned int mode, bool notify_rpm)
241{
242 struct msm_spm_device *dev = &__get_cpu_var(msm_cpu_spm_device);
243 return msm_spm_dev_set_low_power_mode(dev, mode, notify_rpm);
244}
245EXPORT_SYMBOL(msm_spm_set_low_power_mode);
246
Praveen Chidambaram7347bfe2012-11-01 15:21:10 -0600247/**
248 * msm_spm_init(): Board initalization function
249 * @data: platform specific SPM register configuration data
250 * @nr_devs: Number of SPM devices being initialized
251 */
Praveen Chidambaramaa9d52b2012-04-02 11:09:47 -0600252int __init msm_spm_init(struct msm_spm_platform_data *data, int nr_devs)
253{
254 unsigned int cpu;
255 int ret = 0;
256
257 BUG_ON((nr_devs < num_possible_cpus()) || !data);
258
259 for_each_possible_cpu(cpu) {
260 struct msm_spm_device *dev = &per_cpu(msm_cpu_spm_device, cpu);
261 ret = msm_spm_dev_init(dev, &data[cpu]);
262 if (ret < 0) {
263 pr_warn("%s():failed CPU:%u ret:%d\n", __func__,
264 cpu, ret);
265 break;
266 }
267 }
268
269 return ret;
270}
271
272#ifdef CONFIG_MSM_L2_SPM
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700273
Praveen Chidambaram7347bfe2012-11-01 15:21:10 -0600274/**
275 * msm_spm_l2_set_low_power_mode(): Configure L2 SPM start address
276 * for low power mode
277 * @mode: SPM LPM mode to enter
278 * @notify_rpm: Notify RPM in this mode
279 */
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700280int msm_spm_l2_set_low_power_mode(unsigned int mode, bool notify_rpm)
281{
282 return msm_spm_dev_set_low_power_mode(
283 &msm_spm_l2_device, mode, notify_rpm);
284}
Praveen Chidambaramaa9d52b2012-04-02 11:09:47 -0600285EXPORT_SYMBOL(msm_spm_l2_set_low_power_mode);
Maheshkumar Sivasubramanian4ac23762011-11-02 10:03:06 -0600286
287void msm_spm_l2_reinit(void)
288{
Mahesh Sivasubramanianbc420242013-03-26 17:18:04 -0600289 if (!msm_spm_l2_device.initialized)
290 return;
Maheshkumar Sivasubramanian4ac23762011-11-02 10:03:06 -0600291 msm_spm_drv_reinit(&msm_spm_l2_device.reg_data);
292}
Praveen Chidambaramaa9d52b2012-04-02 11:09:47 -0600293EXPORT_SYMBOL(msm_spm_l2_reinit);
294
Praveen Chidambaram7347bfe2012-11-01 15:21:10 -0600295/**
296 * msm_spm_apcs_set_vdd(): Set Apps processor core sub-system voltage
297 * @vlevel: Encoded PMIC data.
298 */
Praveen Chidambaramaa9d52b2012-04-02 11:09:47 -0600299int msm_spm_apcs_set_vdd(unsigned int vlevel)
300{
Mahesh Sivasubramanianbc420242013-03-26 17:18:04 -0600301 if (!msm_spm_l2_device.initialized)
302 return -ENXIO;
Praveen Chidambaramaa9d52b2012-04-02 11:09:47 -0600303 return msm_spm_drv_set_vdd(&msm_spm_l2_device.reg_data, vlevel);
304}
305EXPORT_SYMBOL(msm_spm_apcs_set_vdd);
306
Praveen Chidambaram7347bfe2012-11-01 15:21:10 -0600307/**
308 * msm_spm_apcs_set_phase(): Set number of SMPS phases.
309 * phase_cnt: Number of phases to be set active
310 */
Praveen Chidambaramaa9d52b2012-04-02 11:09:47 -0600311int msm_spm_apcs_set_phase(unsigned int phase_cnt)
312{
Mahesh Sivasubramanianbc420242013-03-26 17:18:04 -0600313 if (!msm_spm_l2_device.initialized)
314 return -ENXIO;
Praveen Chidambaram1dbe4952012-10-03 17:06:02 -0600315 return msm_spm_drv_set_pmic_data(&msm_spm_l2_device.reg_data,
316 MSM_SPM_PMIC_PHASE_PORT, phase_cnt);
Praveen Chidambaramaa9d52b2012-04-02 11:09:47 -0600317}
318EXPORT_SYMBOL(msm_spm_apcs_set_phase);
319
Praveen Chidambaram7347bfe2012-11-01 15:21:10 -0600320/** msm_spm_enable_fts_lpm() : Enable FTS to switch to low power
321 * when the cores are in low power modes
322 * @mode: The mode configuration for FTS
323 */
Praveen Chidambaram1dbe4952012-10-03 17:06:02 -0600324int msm_spm_enable_fts_lpm(uint32_t mode)
325{
Mahesh Sivasubramanianbc420242013-03-26 17:18:04 -0600326 if (!msm_spm_l2_device.initialized)
327 return -ENXIO;
Praveen Chidambaram1dbe4952012-10-03 17:06:02 -0600328 return msm_spm_drv_set_pmic_data(&msm_spm_l2_device.reg_data,
329 MSM_SPM_PMIC_PFM_PORT, mode);
330}
331EXPORT_SYMBOL(msm_spm_enable_fts_lpm);
332
Praveen Chidambaram7347bfe2012-11-01 15:21:10 -0600333/**
334 * msm_spm_l2_init(): Board initialization function
335 * @data: SPM target specific register configuration
336 */
Praveen Chidambaramaa9d52b2012-04-02 11:09:47 -0600337int __init msm_spm_l2_init(struct msm_spm_platform_data *data)
338{
339 return msm_spm_dev_init(&msm_spm_l2_device, data);
340}
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700341#endif
Praveen Chidambaramaa9d52b2012-04-02 11:09:47 -0600342
Sathish Ambley86487e52012-06-11 13:46:11 -0700343static int __devinit msm_spm_dev_probe(struct platform_device *pdev)
Praveen Chidambaramaa9d52b2012-04-02 11:09:47 -0600344{
345 int ret = 0;
346 int cpu = 0;
347 int i = 0;
348 struct device_node *node = pdev->dev.of_node;
349 struct msm_spm_platform_data spm_data;
350 char *key = NULL;
351 uint32_t val = 0;
352 struct msm_spm_seq_entry modes[MSM_SPM_MODE_NR];
353 size_t len = 0;
354 struct msm_spm_device *dev = NULL;
355 struct resource *res = NULL;
356 uint32_t mode_count = 0;
357
358 struct spm_of {
359 char *key;
360 uint32_t id;
361 };
362
363 struct spm_of spm_of_data[] = {
364 {"qcom,saw2-cfg", MSM_SPM_REG_SAW2_CFG},
365 {"qcom,saw2-avs-ctl", MSM_SPM_REG_SAW2_AVS_CTL},
366 {"qcom,saw2-avs-hysteresis", MSM_SPM_REG_SAW2_AVS_HYSTERESIS},
Praveen Chidambaramaa9d52b2012-04-02 11:09:47 -0600367 {"qcom,saw2-avs-limit", MSM_SPM_REG_SAW2_AVS_LIMIT},
Praveen Chidambaram2772d832012-08-22 11:50:34 -0600368 {"qcom,saw2-avs-dly", MSM_SPM_REG_SAW2_AVS_DLY},
Praveen Chidambaramaa9d52b2012-04-02 11:09:47 -0600369 {"qcom,saw2-spm-dly", MSM_SPM_REG_SAW2_SPM_DLY},
Praveen Chidambaram2772d832012-08-22 11:50:34 -0600370 {"qcom,saw2-spm-ctl", MSM_SPM_REG_SAW2_SPM_CTL},
Praveen Chidambaramaa9d52b2012-04-02 11:09:47 -0600371 {"qcom,saw2-pmic-data0", MSM_SPM_REG_SAW2_PMIC_DATA_0},
372 {"qcom,saw2-pmic-data1", MSM_SPM_REG_SAW2_PMIC_DATA_1},
373 {"qcom,saw2-pmic-data2", MSM_SPM_REG_SAW2_PMIC_DATA_2},
374 {"qcom,saw2-pmic-data3", MSM_SPM_REG_SAW2_PMIC_DATA_3},
375 {"qcom,saw2-pmic-data4", MSM_SPM_REG_SAW2_PMIC_DATA_4},
376 {"qcom,saw2-pmic-data5", MSM_SPM_REG_SAW2_PMIC_DATA_5},
377 {"qcom,saw2-pmic-data6", MSM_SPM_REG_SAW2_PMIC_DATA_6},
378 {"qcom,saw2-pmic-data7", MSM_SPM_REG_SAW2_PMIC_DATA_7},
379 };
380
381 struct mode_of {
382 char *key;
383 uint32_t id;
384 uint32_t notify_rpm;
385 };
386
Mahesh Sivasubramanian11373322012-06-14 11:17:20 -0600387 struct mode_of of_cpu_modes[] = {
388 {"qcom,saw2-spm-cmd-wfi", MSM_SPM_MODE_CLOCK_GATING, 0},
389 {"qcom,saw2-spm-cmd-ret", MSM_SPM_MODE_POWER_RETENTION, 0},
390 {"qcom,saw2-spm-cmd-spc", MSM_SPM_MODE_POWER_COLLAPSE, 0},
391 {"qcom,saw2-spm-cmd-pc", MSM_SPM_MODE_POWER_COLLAPSE, 1},
Praveen Chidambaramaa9d52b2012-04-02 11:09:47 -0600392 };
393
Mahesh Sivasubramanian11373322012-06-14 11:17:20 -0600394 struct mode_of of_l2_modes[] = {
Mahesh Sivasubramanian1826ae42012-11-07 14:37:13 -0700395 {"qcom,saw2-spm-cmd-ret", MSM_SPM_L2_MODE_RETENTION, 0},
Mahesh Sivasubramanian11373322012-06-14 11:17:20 -0600396 {"qcom,saw2-spm-cmd-gdhs", MSM_SPM_L2_MODE_GDHS, 1},
397 {"qcom,saw2-spm-cmd-pc", MSM_SPM_L2_MODE_POWER_COLLAPSE, 1},
398 };
399
400 struct mode_of *mode_of_data;
401 int num_modes;
402
Praveen Chidambaramaa9d52b2012-04-02 11:09:47 -0600403 memset(&spm_data, 0, sizeof(struct msm_spm_platform_data));
404 memset(&modes, 0,
405 (MSM_SPM_MODE_NR - 2) * sizeof(struct msm_spm_seq_entry));
406
407 res = platform_get_resource(pdev, IORESOURCE_MEM, 0);
408 if (!res)
409 goto fail;
410
411 spm_data.reg_base_addr = devm_ioremap(&pdev->dev, res->start,
412 resource_size(res));
413 if (!spm_data.reg_base_addr)
414 return -ENOMEM;
415
416 key = "qcom,core-id";
417 ret = of_property_read_u32(node, key, &val);
418 if (ret)
419 goto fail;
420 cpu = val;
421
422 key = "qcom,saw2-ver-reg";
423 ret = of_property_read_u32(node, key, &val);
424 if (ret)
425 goto fail;
426 spm_data.ver_reg = val;
427
428 key = "qcom,vctl-timeout-us";
429 ret = of_property_read_u32(node, key, &val);
430 if (!ret)
431 spm_data.vctl_timeout_us = val;
432
Mahesh Sivasubramanian11373322012-06-14 11:17:20 -0600433 /*
434 * Device with id 0..NR_CPUS are SPM for apps cores
435 * Device with id 0xFFFF is for L2 SPM.
436 */
437 if (cpu >= 0 && cpu < num_possible_cpus()) {
438 mode_of_data = of_cpu_modes;
439 num_modes = ARRAY_SIZE(of_cpu_modes);
440 dev = &per_cpu(msm_cpu_spm_device, cpu);
441
442 } else {
443 mode_of_data = of_l2_modes;
444 num_modes = ARRAY_SIZE(of_l2_modes);
445 dev = &msm_spm_l2_device;
446 }
447
Praveen Chidambaram1dbe4952012-10-03 17:06:02 -0600448 spm_data.vctl_port = -1;
449 spm_data.phase_port = -1;
450 spm_data.pfm_port = -1;
451
452 /* optional */
453 if (dev == &msm_spm_l2_device) {
454 key = "qcom,vctl-port";
455 ret = of_property_read_u32(node, key, &val);
456 if (!ret)
457 spm_data.vctl_port = val;
458
459 key = "qcom,phase-port";
460 ret = of_property_read_u32(node, key, &val);
461 if (!ret)
462 spm_data.phase_port = val;
463
464 key = "qcom,pfm-port";
465 ret = of_property_read_u32(node, key, &val);
466 if (!ret)
467 spm_data.pfm_port = val;
468 }
469
470 for (i = 0; i < ARRAY_SIZE(spm_of_data); i++) {
471 ret = of_property_read_u32(node, spm_of_data[i].key, &val);
472 if (ret)
473 continue;
474 spm_data.reg_init_values[spm_of_data[i].id] = val;
475 }
476
Mahesh Sivasubramanian11373322012-06-14 11:17:20 -0600477 for (i = 0; i < num_modes; i++) {
Praveen Chidambaramaa9d52b2012-04-02 11:09:47 -0600478 key = mode_of_data[i].key;
479 modes[mode_count].cmd =
480 (uint8_t *)of_get_property(node, key, &len);
481 if (!modes[mode_count].cmd)
482 continue;
483 modes[mode_count].mode = mode_of_data[i].id;
484 modes[mode_count].notify_rpm = mode_of_data[i].notify_rpm;
485 mode_count++;
486 }
487
488 spm_data.modes = modes;
489 spm_data.num_modes = mode_count;
490
Praveen Chidambaramaa9d52b2012-04-02 11:09:47 -0600491 ret = msm_spm_dev_init(dev, &spm_data);
Mahesh Sivasubramanian11373322012-06-14 11:17:20 -0600492
Praveen Chidambaramaa9d52b2012-04-02 11:09:47 -0600493 if (ret < 0)
494 pr_warn("%s():failed core-id:%u ret:%d\n", __func__, cpu, ret);
495
496 return ret;
497
498fail:
499 pr_err("%s: Failed reading node=%s, key=%s\n",
500 __func__, node->full_name, key);
501 return -EFAULT;
502}
503
Sathish Ambley86487e52012-06-11 13:46:11 -0700504static struct of_device_id msm_spm_match_table[] = {
Praveen Chidambaramaa9d52b2012-04-02 11:09:47 -0600505 {.compatible = "qcom,spm-v2"},
506 {},
507};
508
Sathish Ambley86487e52012-06-11 13:46:11 -0700509static struct platform_driver msm_spm_device_driver = {
Praveen Chidambaramaa9d52b2012-04-02 11:09:47 -0600510 .probe = msm_spm_dev_probe,
511 .driver = {
512 .name = "spm-v2",
513 .owner = THIS_MODULE,
514 .of_match_table = msm_spm_match_table,
515 },
516};
517
Praveen Chidambaram7347bfe2012-11-01 15:21:10 -0600518/**
519 * msm_spm_device_init(): Device tree initialization function
520 */
Praveen Chidambaramaa9d52b2012-04-02 11:09:47 -0600521int __init msm_spm_device_init(void)
522{
523 return platform_driver_register(&msm_spm_device_driver);
524}