blob: 180d277f7e8e567e8eb01c75f33d7ece335f75c4 [file] [log] [blame]
Priyanka Mathur3abfd442013-01-11 12:58:51 -08001/* Copyright (c) 2012-2013, The Linux Foundation. All rights reserved.
Praveen Chidambaram85b7b282012-04-16 13:45:15 -06002 *
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/init.h>
17#include <linux/slab.h>
18#include <linux/platform_device.h>
Girish Mahadevanc45b4c72013-04-24 14:07:11 -060019#include <linux/mutex.h>
20#include <linux/cpu.h>
Praveen Chidambaram85b7b282012-04-16 13:45:15 -060021#include <linux/of.h>
22#include <mach/mpm.h>
Mahesh Sivasubramanian6d06e3a2012-05-16 13:41:07 -060023#include "pm.h"
Mahesh Sivasubramanian11dad772012-07-13 14:00:01 -060024#include "rpm-notifier.h"
Girish Mahadevanc45b4c72013-04-24 14:07:11 -060025#include "spm.h"
26#include "idle.h"
Mahesh Sivasubramanianfae923f2012-10-08 16:22:51 -060027
28enum {
29 MSM_LPM_LVL_DBG_SUSPEND_LIMITS = BIT(0),
30 MSM_LPM_LVL_DBG_IDLE_LIMITS = BIT(1),
31};
32
Girish Mahadevanc45b4c72013-04-24 14:07:11 -060033enum {
34 MSM_SCM_L2_ON = 0,
35 MSM_SCM_L2_OFF = 1,
36 MSM_SCM_L2_GDHS = 3,
37};
38
39struct msm_rpmrs_level {
40 enum msm_pm_sleep_mode sleep_mode;
41 uint32_t l2_cache;
42 bool available;
43 uint32_t latency_us;
44 uint32_t steady_state_power;
45 uint32_t energy_overhead;
46 uint32_t time_overhead_us;
47};
48
49struct lpm_lookup_table {
50 uint32_t modes;
51 const char *mode_name;
52};
53
54static void msm_lpm_level_update(void);
55
56static int msm_lpm_cpu_callback(struct notifier_block *cpu_nb,
57 unsigned long action, void *hcpu);
58
59static struct notifier_block __refdata msm_lpm_cpu_nblk = {
60 .notifier_call = msm_lpm_cpu_callback,
61};
62
63static uint32_t allowed_l2_mode;
64static uint32_t sysfs_dbg_l2_mode = MSM_SPM_L2_MODE_POWER_COLLAPSE;
65static uint32_t default_l2_mode;
66
67static bool no_l2_saw;
68
69static ssize_t msm_lpm_levels_attr_show(
70 struct kobject *kobj, struct kobj_attribute *attr, char *buf);
71static ssize_t msm_lpm_levels_attr_store(struct kobject *kobj,
72 struct kobj_attribute *attr, const char *buf, size_t count);
Archana Sathyakumare6a35102013-01-31 16:18:49 -070073
Girish Mahadevane6b1dd42013-05-22 11:59:08 -060074#define ADJUST_LATENCY(x) \
75 ((x == MSM_PM_SLEEP_MODE_POWER_COLLAPSE_STANDALONE) ?\
76 (num_online_cpus()) / 2 : 0)
77
Mahesh Sivasubramanianfae923f2012-10-08 16:22:51 -060078static int msm_lpm_lvl_dbg_msk;
79
80module_param_named(
81 debug_mask, msm_lpm_lvl_dbg_msk, int, S_IRUGO | S_IWUSR | S_IWGRP
82);
83
Praveen Chidambaram85b7b282012-04-16 13:45:15 -060084static struct msm_rpmrs_level *msm_lpm_levels;
85static int msm_lpm_level_count;
86
Girish Mahadevanc45b4c72013-04-24 14:07:11 -060087static struct kobj_attribute lpm_l2_kattr = __ATTR(l2, S_IRUGO|S_IWUSR,\
88 msm_lpm_levels_attr_show, msm_lpm_levels_attr_store);
89
90static struct attribute *lpm_levels_attr[] = {
91 &lpm_l2_kattr.attr,
92 NULL,
93};
94
95static struct attribute_group lpm_levels_attr_grp = {
96 .attrs = lpm_levels_attr,
97};
98
99/* SYSFS */
100static ssize_t msm_lpm_levels_attr_show(
101 struct kobject *kobj, struct kobj_attribute *attr, char *buf)
102{
103 struct kernel_param kp;
104 int rc;
105
106 kp.arg = &sysfs_dbg_l2_mode;
107
108 rc = param_get_uint(buf, &kp);
109
110 if (rc > 0) {
111 strlcat(buf, "\n", PAGE_SIZE);
112 rc++;
113 }
114
115 return rc;
116}
117
118static ssize_t msm_lpm_levels_attr_store(struct kobject *kobj,
119 struct kobj_attribute *attr, const char *buf, size_t count)
120{
121 struct kernel_param kp;
122 unsigned int temp;
123 int rc;
124
125 kp.arg = &temp;
126 rc = param_set_uint(buf, &kp);
127 if (rc)
128 return rc;
129
130 sysfs_dbg_l2_mode = temp;
131 msm_lpm_level_update();
132
133 return count;
134}
Priyanka Mathur3abfd442013-01-11 12:58:51 -0800135
Archana Sathyakumare6a35102013-01-31 16:18:49 -0700136static int msm_pm_get_sleep_mode_value(struct device_node *node,
137 const char *key, uint32_t *sleep_mode_val)
138{
139 int i;
140 struct lpm_lookup_table {
141 uint32_t modes;
142 const char *mode_name;
143 };
144 struct lpm_lookup_table pm_sm_lookup[] = {
145 {MSM_PM_SLEEP_MODE_WAIT_FOR_INTERRUPT,
146 "wfi"},
147 {MSM_PM_SLEEP_MODE_RAMP_DOWN_AND_WAIT_FOR_INTERRUPT,
148 "ramp_down_and_wfi"},
149 {MSM_PM_SLEEP_MODE_POWER_COLLAPSE_STANDALONE,
150 "standalone_pc"},
151 {MSM_PM_SLEEP_MODE_POWER_COLLAPSE,
152 "pc"},
153 {MSM_PM_SLEEP_MODE_RETENTION,
154 "retention"},
155 {MSM_PM_SLEEP_MODE_POWER_COLLAPSE_SUSPEND,
156 "pc_suspend"},
157 {MSM_PM_SLEEP_MODE_POWER_COLLAPSE_NO_XO_SHUTDOWN,
158 "pc_no_xo_shutdown"}
159 };
160 int ret;
161 const char *mode_name;
162
163 ret = of_property_read_string(node, key, &mode_name);
164 if (!ret) {
165 ret = -EINVAL;
166 for (i = 0; i < ARRAY_SIZE(pm_sm_lookup); i++) {
Girish Mahadevanc45b4c72013-04-24 14:07:11 -0600167 if (!strcmp(mode_name, pm_sm_lookup[i].mode_name)) {
Archana Sathyakumare6a35102013-01-31 16:18:49 -0700168 *sleep_mode_val = pm_sm_lookup[i].modes;
169 ret = 0;
170 break;
171 }
172 }
173 }
174 return ret;
175}
176
Girish Mahadevanc45b4c72013-04-24 14:07:11 -0600177static int msm_lpm_set_l2_mode(int sleep_mode)
178{
179 int lpm = sleep_mode;
180 int rc = 0;
181
182 if (no_l2_saw)
183 goto bail_set_l2_mode;
184
185 msm_pm_set_l2_flush_flag(MSM_SCM_L2_ON);
186
187 switch (sleep_mode) {
188 case MSM_SPM_L2_MODE_POWER_COLLAPSE:
189 msm_pm_set_l2_flush_flag(MSM_SCM_L2_OFF);
190 break;
191 case MSM_SPM_L2_MODE_GDHS:
192 msm_pm_set_l2_flush_flag(MSM_SCM_L2_GDHS);
193 break;
194 case MSM_SPM_L2_MODE_RETENTION:
195 case MSM_SPM_L2_MODE_DISABLED:
196 break;
197 default:
198 lpm = MSM_SPM_L2_MODE_DISABLED;
199 break;
200 }
201
202 rc = msm_spm_l2_set_low_power_mode(lpm, true);
203
204 if (rc) {
205 if (rc == -ENXIO)
206 WARN_ON_ONCE(1);
207 else
208 pr_err("%s: Failed to set L2 low power mode %d, ERR %d",
209 __func__, lpm, rc);
210 }
211
212bail_set_l2_mode:
213 return rc;
214}
215
Girish Mahadevan40abbe12012-04-25 14:58:13 -0600216static void msm_lpm_level_update(void)
217{
Girish Mahadevanc45b4c72013-04-24 14:07:11 -0600218 int lpm_level;
Girish Mahadevan40abbe12012-04-25 14:58:13 -0600219 struct msm_rpmrs_level *level = NULL;
Girish Mahadevanc45b4c72013-04-24 14:07:11 -0600220 uint32_t max_l2_mode;
221 static DEFINE_MUTEX(lpm_lock);
222
223 mutex_lock(&lpm_lock);
224
225 max_l2_mode = min(allowed_l2_mode, sysfs_dbg_l2_mode);
Girish Mahadevan40abbe12012-04-25 14:58:13 -0600226
227 for (lpm_level = 0; lpm_level < msm_lpm_level_count; lpm_level++) {
228 level = &msm_lpm_levels[lpm_level];
Girish Mahadevanc45b4c72013-04-24 14:07:11 -0600229 level->available = !(level->l2_cache > max_l2_mode);
Girish Mahadevan40abbe12012-04-25 14:58:13 -0600230 }
Girish Mahadevanc45b4c72013-04-24 14:07:11 -0600231 mutex_unlock(&lpm_lock);
Girish Mahadevan40abbe12012-04-25 14:58:13 -0600232}
233
234int msm_lpm_enter_sleep(uint32_t sclk_count, void *limits,
Praveen Chidambaram85b7b282012-04-16 13:45:15 -0600235 bool from_idle, bool notify_rpm)
236{
Girish Mahadevan40abbe12012-04-25 14:58:13 -0600237 int ret = 0;
Mahesh Sivasubramanian9b0f4122013-02-15 14:20:52 -0700238 int debug_mask;
Girish Mahadevanc45b4c72013-04-24 14:07:11 -0600239 uint32_t l2 = *(uint32_t *)limits;
Girish Mahadevan40abbe12012-04-25 14:58:13 -0600240
Mahesh Sivasubramanian9b0f4122013-02-15 14:20:52 -0700241 if (from_idle)
242 debug_mask = msm_lpm_lvl_dbg_msk &
Priyanka Mathur86cfc762013-01-10 17:03:04 -0800243 MSM_LPM_LVL_DBG_IDLE_LIMITS;
Mahesh Sivasubramanian9b0f4122013-02-15 14:20:52 -0700244 else
245 debug_mask = msm_lpm_lvl_dbg_msk &
Priyanka Mathur86cfc762013-01-10 17:03:04 -0800246 MSM_LPM_LVL_DBG_SUSPEND_LIMITS;
Mahesh Sivasubramanian9b0f4122013-02-15 14:20:52 -0700247
248 if (debug_mask)
Girish Mahadevanc45b4c72013-04-24 14:07:11 -0600249 pr_info("%s(): l2:%d", __func__, l2);
Mahesh Sivasubramanian9b0f4122013-02-15 14:20:52 -0700250
Girish Mahadevanc45b4c72013-04-24 14:07:11 -0600251 ret = msm_lpm_set_l2_mode(l2);
252
Mahesh Sivasubramanian0558d4b2012-10-12 18:05:28 -0600253 if (ret) {
Girish Mahadevanc45b4c72013-04-24 14:07:11 -0600254 if (ret == -ENXIO)
255 ret = 0;
256 else {
257 pr_warn("%s(): Failed to set L2 SPM Mode %d",
258 __func__, l2);
259 goto bail;
260 }
Mahesh Sivasubramanian0558d4b2012-10-12 18:05:28 -0600261 }
Girish Mahadevanc45b4c72013-04-24 14:07:11 -0600262
Priyanka Mathur86cfc762013-01-10 17:03:04 -0800263 if (notify_rpm) {
264 ret = msm_rpm_enter_sleep(debug_mask);
265 if (ret) {
266 pr_warn("%s(): RPM failed to enter sleep err:%d\n",
267 __func__, ret);
268 goto bail;
269 }
Girish Mahadevanc45b4c72013-04-24 14:07:11 -0600270
271 msm_mpm_enter_sleep(sclk_count, from_idle);
Priyanka Mathur86cfc762013-01-10 17:03:04 -0800272 }
Mahesh Sivasubramanian11dad772012-07-13 14:00:01 -0600273bail:
Girish Mahadevan40abbe12012-04-25 14:58:13 -0600274 return ret;
Praveen Chidambaram85b7b282012-04-16 13:45:15 -0600275}
276
Mahesh Sivasubramanian6d06e3a2012-05-16 13:41:07 -0600277static void msm_lpm_exit_sleep(void *limits, bool from_idle,
Praveen Chidambaram85b7b282012-04-16 13:45:15 -0600278 bool notify_rpm, bool collapsed)
279{
Mahesh Sivasubramanian0558d4b2012-10-12 18:05:28 -0600280
Girish Mahadevanc45b4c72013-04-24 14:07:11 -0600281 msm_lpm_set_l2_mode(default_l2_mode);
282
283 if (notify_rpm) {
284 msm_mpm_exit_sleep(from_idle);
Priyanka Mathur86cfc762013-01-10 17:03:04 -0800285 msm_rpm_exit_sleep();
Girish Mahadevanc45b4c72013-04-24 14:07:11 -0600286 }
Praveen Chidambaram85b7b282012-04-16 13:45:15 -0600287}
288
Girish Mahadevan40abbe12012-04-25 14:58:13 -0600289void msm_lpm_show_resources(void)
Praveen Chidambaram85b7b282012-04-16 13:45:15 -0600290{
291 /* TODO */
292 return;
293}
294
Stephen Boyd3f4bac22012-05-30 10:03:13 -0700295s32 msm_cpuidle_get_deep_idle_latency(void)
296{
297 int i;
298 struct msm_rpmrs_level *level = msm_lpm_levels, *best = level;
299
300 if (!level)
301 return 0;
302
303 for (i = 0; i < msm_lpm_level_count; i++, level++) {
304 if (!level->available)
305 continue;
306 if (level->sleep_mode != MSM_PM_SLEEP_MODE_POWER_COLLAPSE)
307 continue;
308 /* Pick the first power collapse mode by default */
309 if (best->sleep_mode != MSM_PM_SLEEP_MODE_POWER_COLLAPSE)
310 best = level;
311 /* Find the lowest latency for power collapse */
312 if (level->latency_us < best->latency_us)
313 best = level;
314 }
315 return best->latency_us - 1;
316}
Girish Mahadevanc45b4c72013-04-24 14:07:11 -0600317
318static int msm_lpm_cpu_callback(struct notifier_block *cpu_nb,
319 unsigned long action, void *hcpu)
Mahesh Sivasubramanianf1ddf042013-01-08 14:03:32 -0700320{
Girish Mahadevanc45b4c72013-04-24 14:07:11 -0600321 switch (action) {
322 case CPU_UP_PREPARE:
323 case CPU_UP_PREPARE_FROZEN:
324 allowed_l2_mode = default_l2_mode;
325 msm_lpm_level_update();
326 break;
327 case CPU_DEAD_FROZEN:
328 case CPU_DEAD:
329 case CPU_UP_CANCELED:
330 case CPU_UP_CANCELED_FROZEN:
331 if (num_online_cpus() == 1)
332 allowed_l2_mode = MSM_SPM_L2_MODE_POWER_COLLAPSE;
333 msm_lpm_level_update();
334 break;
335 }
336 return NOTIFY_OK;
Mahesh Sivasubramanianf1ddf042013-01-08 14:03:32 -0700337}
Stephen Boyd3f4bac22012-05-30 10:03:13 -0700338
Mahesh Sivasubramanian6d06e3a2012-05-16 13:41:07 -0600339static void *msm_lpm_lowest_limits(bool from_idle,
Girish Mahadevandc318fd2012-08-17 16:48:05 -0600340 enum msm_pm_sleep_mode sleep_mode,
341 struct msm_pm_time_params *time_param, uint32_t *power)
Praveen Chidambaram85b7b282012-04-16 13:45:15 -0600342{
343 unsigned int cpu = smp_processor_id();
344 struct msm_rpmrs_level *best_level = NULL;
Girish Mahadevanc45b4c72013-04-24 14:07:11 -0600345 uint32_t best_level_pwr = 0;
Mahesh Sivasubramanian6d06e3a2012-05-16 13:41:07 -0600346 uint32_t pwr;
Praveen Chidambaram85b7b282012-04-16 13:45:15 -0600347 int i;
Girish Mahadevana351c0b2013-02-22 11:06:00 -0700348 bool modify_event_timer;
349 uint32_t next_wakeup_us = time_param->sleep_us;
Girish Mahadevane6b1dd42013-05-22 11:59:08 -0600350 uint32_t lvl_latency_us = 0;
351 uint32_t lvl_overhead_us = 0;
352 uint32_t lvl_overhead_energy = 0;
Mahesh Sivasubramanianf1ddf042013-01-08 14:03:32 -0700353
Praveen Chidambaram85b7b282012-04-16 13:45:15 -0600354 if (!msm_lpm_levels)
355 return NULL;
356
Praveen Chidambaram85b7b282012-04-16 13:45:15 -0600357 for (i = 0; i < msm_lpm_level_count; i++) {
358 struct msm_rpmrs_level *level = &msm_lpm_levels[i];
Praveen Chidambaram85b7b282012-04-16 13:45:15 -0600359
Girish Mahadevana351c0b2013-02-22 11:06:00 -0700360 modify_event_timer = false;
361
Praveen Chidambaram85b7b282012-04-16 13:45:15 -0600362 if (!level->available)
363 continue;
364
365 if (sleep_mode != level->sleep_mode)
366 continue;
367
Girish Mahadevane6b1dd42013-05-22 11:59:08 -0600368 lvl_latency_us =
369 level->latency_us + (level->latency_us *
370 ADJUST_LATENCY(sleep_mode));
371
372 lvl_overhead_us =
373 level->time_overhead_us + (level->time_overhead_us *
374 ADJUST_LATENCY(sleep_mode));
375
376 lvl_overhead_energy =
377 level->energy_overhead + level->energy_overhead *
378 ADJUST_LATENCY(sleep_mode);
379
380 if (time_param->latency_us < lvl_latency_us)
Praveen Chidambaram85b7b282012-04-16 13:45:15 -0600381 continue;
382
Girish Mahadevana351c0b2013-02-22 11:06:00 -0700383 if (time_param->next_event_us &&
Girish Mahadevane6b1dd42013-05-22 11:59:08 -0600384 time_param->next_event_us < lvl_latency_us)
Girish Mahadevana351c0b2013-02-22 11:06:00 -0700385 continue;
386
387 if (time_param->next_event_us) {
388 if ((time_param->next_event_us < time_param->sleep_us)
Girish Mahadevane6b1dd42013-05-22 11:59:08 -0600389 || ((time_param->next_event_us - lvl_latency_us) <
Girish Mahadevana351c0b2013-02-22 11:06:00 -0700390 time_param->sleep_us)) {
391 modify_event_timer = true;
392 next_wakeup_us = time_param->next_event_us -
Girish Mahadevane6b1dd42013-05-22 11:59:08 -0600393 lvl_latency_us;
Girish Mahadevana351c0b2013-02-22 11:06:00 -0700394 }
395 }
396
Girish Mahadevane6b1dd42013-05-22 11:59:08 -0600397 if (next_wakeup_us <= lvl_overhead_us)
Girish Mahadevana351c0b2013-02-22 11:06:00 -0700398 continue;
399
Mahesh Sivasubramanian9063a292012-11-09 09:15:30 -0700400 if ((MSM_PM_SLEEP_MODE_POWER_COLLAPSE_STANDALONE == sleep_mode)
401 || (MSM_PM_SLEEP_MODE_POWER_COLLAPSE == sleep_mode))
402 if (!cpu && msm_rpm_waiting_for_ack())
403 break;
404
Girish Mahadevana351c0b2013-02-22 11:06:00 -0700405 if (next_wakeup_us <= 1) {
Girish Mahadevane6b1dd42013-05-22 11:59:08 -0600406 pwr = lvl_overhead_energy;
407 } else if (next_wakeup_us <= lvl_overhead_us) {
408 pwr = lvl_overhead_energy / next_wakeup_us;
Girish Mahadevana351c0b2013-02-22 11:06:00 -0700409 } else if ((next_wakeup_us >> 10)
Girish Mahadevane6b1dd42013-05-22 11:59:08 -0600410 > lvl_overhead_us) {
Mahesh Sivasubramanian6d06e3a2012-05-16 13:41:07 -0600411 pwr = level->steady_state_power;
Praveen Chidambaram85b7b282012-04-16 13:45:15 -0600412 } else {
Mahesh Sivasubramanian6d06e3a2012-05-16 13:41:07 -0600413 pwr = level->steady_state_power;
Girish Mahadevane6b1dd42013-05-22 11:59:08 -0600414 pwr -= (lvl_overhead_us *
Girish Mahadevandc318fd2012-08-17 16:48:05 -0600415 level->steady_state_power) /
Girish Mahadevana351c0b2013-02-22 11:06:00 -0700416 next_wakeup_us;
Girish Mahadevane6b1dd42013-05-22 11:59:08 -0600417 pwr += lvl_overhead_energy / next_wakeup_us;
Praveen Chidambaram85b7b282012-04-16 13:45:15 -0600418 }
419
Girish Mahadevanc45b4c72013-04-24 14:07:11 -0600420 if (!best_level || (best_level_pwr >= pwr)) {
Praveen Chidambaram85b7b282012-04-16 13:45:15 -0600421 best_level = level;
Girish Mahadevanc45b4c72013-04-24 14:07:11 -0600422 best_level_pwr = pwr;
Mahesh Sivasubramanian6d06e3a2012-05-16 13:41:07 -0600423 if (power)
424 *power = pwr;
Girish Mahadevana351c0b2013-02-22 11:06:00 -0700425 if (modify_event_timer &&
426 (sleep_mode !=
427 MSM_PM_SLEEP_MODE_WAIT_FOR_INTERRUPT))
428 time_param->modified_time_us =
429 time_param->next_event_us -
Girish Mahadevane6b1dd42013-05-22 11:59:08 -0600430 lvl_latency_us;
Girish Mahadevana351c0b2013-02-22 11:06:00 -0700431 else
432 time_param->modified_time_us = 0;
Praveen Chidambaram85b7b282012-04-16 13:45:15 -0600433 }
434 }
435
Girish Mahadevanc45b4c72013-04-24 14:07:11 -0600436 return best_level ? &best_level->l2_cache : NULL;
Praveen Chidambaram85b7b282012-04-16 13:45:15 -0600437}
Priyanka Mathur3abfd442013-01-11 12:58:51 -0800438
Mahesh Sivasubramanian6d06e3a2012-05-16 13:41:07 -0600439static struct msm_pm_sleep_ops msm_lpm_ops = {
440 .lowest_limits = msm_lpm_lowest_limits,
441 .enter_sleep = msm_lpm_enter_sleep,
442 .exit_sleep = msm_lpm_exit_sleep,
443};
Praveen Chidambaram85b7b282012-04-16 13:45:15 -0600444
Girish Mahadevanc45b4c72013-04-24 14:07:11 -0600445static int msm_lpm_get_l2_cache_value(struct device_node *node,
446 char *key, uint32_t *l2_val)
447{
448 int i;
449 struct lpm_lookup_table l2_mode_lookup[] = {
450 {MSM_SPM_L2_MODE_POWER_COLLAPSE, "l2_cache_pc"},
451 {MSM_SPM_L2_MODE_GDHS, "l2_cache_gdhs"},
452 {MSM_SPM_L2_MODE_RETENTION, "l2_cache_retention"},
453 {MSM_SPM_L2_MODE_DISABLED, "l2_cache_active"}
454 };
455 const char *l2_str;
456 int ret;
457
458 ret = of_property_read_string(node, key, &l2_str);
459 if (!ret) {
460 ret = -EINVAL;
461 for (i = 0; i < ARRAY_SIZE(l2_mode_lookup); i++) {
462 if (!strcmp(l2_str, l2_mode_lookup[i].mode_name)) {
463 *l2_val = l2_mode_lookup[i].modes;
464 ret = 0;
465 break;
466 }
467 }
468 }
469 return ret;
470}
471
472static int __devinit msm_lpm_levels_sysfs_add(void)
473{
474 struct kobject *module_kobj = NULL;
475 struct kobject *low_power_kobj = NULL;
476 int rc = 0;
477
478 module_kobj = kset_find_obj(module_kset, KBUILD_MODNAME);
479 if (!module_kobj) {
480 pr_err("%s: cannot find kobject for module %s\n",
481 __func__, KBUILD_MODNAME);
482 rc = -ENOENT;
483 goto resource_sysfs_add_exit;
484 }
485
486 low_power_kobj = kobject_create_and_add(
487 "enable_low_power", module_kobj);
488 if (!low_power_kobj) {
489 pr_err("%s: cannot create kobject\n", __func__);
490 rc = -ENOMEM;
491 goto resource_sysfs_add_exit;
492 }
493
494 rc = sysfs_create_group(low_power_kobj, &lpm_levels_attr_grp);
495resource_sysfs_add_exit:
496 if (rc) {
497 if (low_power_kobj) {
498 sysfs_remove_group(low_power_kobj,
499 &lpm_levels_attr_grp);
500 kobject_del(low_power_kobj);
501 }
502 }
503
504 return rc;
505}
506
Praveen Chidambaram85b7b282012-04-16 13:45:15 -0600507static int __devinit msm_lpm_levels_probe(struct platform_device *pdev)
508{
509 struct msm_rpmrs_level *levels = NULL;
510 struct msm_rpmrs_level *level = NULL;
511 struct device_node *node = NULL;
512 char *key = NULL;
513 uint32_t val = 0;
514 int ret = 0;
515 uint32_t num_levels = 0;
516 int idx = 0;
517
518 for_each_child_of_node(pdev->dev.of_node, node)
519 num_levels++;
520
521 levels = kzalloc(num_levels * sizeof(struct msm_rpmrs_level),
522 GFP_KERNEL);
523 if (!levels)
524 return -ENOMEM;
525
526 for_each_child_of_node(pdev->dev.of_node, node) {
527 level = &levels[idx++];
528 level->available = false;
529
530 key = "qcom,mode";
Archana Sathyakumare6a35102013-01-31 16:18:49 -0700531 ret = msm_pm_get_sleep_mode_value(node, key, &val);
Praveen Chidambaram85b7b282012-04-16 13:45:15 -0600532 if (ret)
533 goto fail;
534 level->sleep_mode = val;
535
Praveen Chidambaram85b7b282012-04-16 13:45:15 -0600536 key = "qcom,l2";
Archana Sathyakumare6a35102013-01-31 16:18:49 -0700537 ret = msm_lpm_get_l2_cache_value(node, key, &val);
Praveen Chidambaram85b7b282012-04-16 13:45:15 -0600538 if (ret)
539 goto fail;
Girish Mahadevanc45b4c72013-04-24 14:07:11 -0600540 level->l2_cache = val;
Mahesh Sivasubramanianb71ce092013-01-08 13:44:23 -0700541
Praveen Chidambaram85b7b282012-04-16 13:45:15 -0600542 key = "qcom,latency-us";
543 ret = of_property_read_u32(node, key, &val);
544 if (ret)
545 goto fail;
546 level->latency_us = val;
547
548 key = "qcom,ss-power";
549 ret = of_property_read_u32(node, key, &val);
550 if (ret)
551 goto fail;
552 level->steady_state_power = val;
553
554 key = "qcom,energy-overhead";
555 ret = of_property_read_u32(node, key, &val);
556 if (ret)
557 goto fail;
558 level->energy_overhead = val;
559
560 key = "qcom,time-overhead";
561 ret = of_property_read_u32(node, key, &val);
562 if (ret)
563 goto fail;
564 level->time_overhead_us = val;
565
566 level->available = true;
567 }
568
Girish Mahadevanc45b4c72013-04-24 14:07:11 -0600569 node = pdev->dev.of_node;
570 key = "qcom,no-l2-saw";
571 no_l2_saw = of_property_read_bool(node, key);
572
Praveen Chidambaram85b7b282012-04-16 13:45:15 -0600573 msm_lpm_levels = levels;
574 msm_lpm_level_count = idx;
575
Girish Mahadevanc45b4c72013-04-24 14:07:11 -0600576 if (num_online_cpus() == 1)
577 allowed_l2_mode = MSM_SPM_L2_MODE_POWER_COLLAPSE;
Priyanka Mathur3abfd442013-01-11 12:58:51 -0800578
Girish Mahadevanc45b4c72013-04-24 14:07:11 -0600579 /* Do the following two steps only if L2 SAW is present */
580 if (!no_l2_saw) {
581 key = "qcom,default-l2-state";
582 if (msm_lpm_get_l2_cache_value(node, key, &default_l2_mode))
583 goto fail;
Priyanka Mathur3abfd442013-01-11 12:58:51 -0800584
Girish Mahadevanc45b4c72013-04-24 14:07:11 -0600585 if (msm_lpm_levels_sysfs_add())
586 goto fail;
587 register_hotcpu_notifier(&msm_lpm_cpu_nblk);
588 msm_pm_set_l2_flush_flag(0);
589 } else {
590 msm_pm_set_l2_flush_flag(1);
591 default_l2_mode = MSM_SPM_L2_MODE_POWER_COLLAPSE;
592 }
593
594 msm_lpm_level_update();
Mahesh Sivasubramanian6d06e3a2012-05-16 13:41:07 -0600595 msm_pm_set_sleep_ops(&msm_lpm_ops);
Praveen Chidambaram85b7b282012-04-16 13:45:15 -0600596 return 0;
597fail:
598 pr_err("%s: Error in name %s key %s\n", __func__, node->full_name, key);
599 kfree(levels);
600 return -EFAULT;
601}
602
603static struct of_device_id msm_lpm_levels_match_table[] = {
604 {.compatible = "qcom,lpm-levels"},
605 {},
606};
607
608static struct platform_driver msm_lpm_levels_driver = {
609 .probe = msm_lpm_levels_probe,
610 .driver = {
611 .name = "lpm-levels",
612 .owner = THIS_MODULE,
613 .of_match_table = msm_lpm_levels_match_table,
614 },
615};
616
617static int __init msm_lpm_levels_module_init(void)
618{
619 return platform_driver_register(&msm_lpm_levels_driver);
620}
621late_initcall(msm_lpm_levels_module_init);