blob: b707df1d9f858de74d2d37f5b9bf99635cab65d9 [file] [log] [blame]
/* Copyright (c) 2012-2013, The Linux Foundation. All rights reserved.
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License version 2 and
* only version 2 as published by the Free Software Foundation.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
*/
#include <linux/module.h>
#include <linux/kernel.h>
#include <linux/init.h>
#include <linux/slab.h>
#include <linux/platform_device.h>
#include <linux/mutex.h>
#include <linux/cpu.h>
#include <linux/of.h>
#include <linux/hrtimer.h>
#include <linux/ktime.h>
#include <linux/tick.h>
#include <linux/suspend.h>
#include <linux/pm_qos.h>
#include <linux/of_platform.h>
#include <mach/mpm.h>
#include <mach/cpuidle.h>
#include <mach/event_timer.h>
#include "pm.h"
#include "rpm-notifier.h"
#include "spm.h"
#include "idle.h"
#define SCLK_HZ (32768)
enum {
MSM_LPM_LVL_DBG_SUSPEND_LIMITS = BIT(0),
MSM_LPM_LVL_DBG_IDLE_LIMITS = BIT(1),
};
struct power_params {
uint32_t latency_us;
uint32_t ss_power;
uint32_t energy_overhead;
uint32_t time_overhead_us;
uint32_t target_residency_us;
};
struct lpm_cpu_level {
const char *name;
enum msm_pm_sleep_mode mode;
struct power_params pwr;
bool use_bc_timer;
bool sync;
};
struct lpm_system_level {
const char *name;
uint32_t l2_mode;
struct power_params pwr;
enum msm_pm_sleep_mode min_cpu_mode;
int num_cpu_votes;
bool notify_rpm;
bool available;
bool sync;
};
struct lpm_system_state {
struct lpm_cpu_level *cpu_level;
int num_cpu_levels;
struct lpm_system_level *system_level;
int num_system_levels;
enum msm_pm_sleep_mode sync_cpu_mode;
int last_entered_cluster_index;
bool allow_synched_levels;
bool no_l2_saw;
struct spinlock sync_lock;
int num_cores_in_sync;
};
static struct lpm_system_state sys_state;
static bool suspend_in_progress;
struct lpm_lookup_table {
uint32_t modes;
const char *mode_name;
};
static void lpm_system_level_update(void);
static void setup_broadcast_timer(void *arg);
static int lpm_cpu_callback(struct notifier_block *cpu_nb,
unsigned long action, void *hcpu);
static struct notifier_block __refdata lpm_cpu_nblk = {
.notifier_call = lpm_cpu_callback,
};
static uint32_t allowed_l2_mode;
static uint32_t sysfs_dbg_l2_mode = MSM_SPM_L2_MODE_POWER_COLLAPSE;
static uint32_t default_l2_mode;
static ssize_t lpm_levels_attr_show(
struct kobject *kobj, struct kobj_attribute *attr, char *buf);
static ssize_t lpm_levels_attr_store(struct kobject *kobj,
struct kobj_attribute *attr, const char *buf, size_t count);
static int lpm_lvl_dbg_msk;
module_param_named(
debug_mask, lpm_lvl_dbg_msk, int, S_IRUGO | S_IWUSR | S_IWGRP
);
static bool menu_select;
module_param_named(
menu_select, menu_select, bool, S_IRUGO | S_IWUSR | S_IWGRP
);
static int msm_pm_sleep_time_override;
module_param_named(sleep_time_override,
msm_pm_sleep_time_override, int, S_IRUGO | S_IWUSR | S_IWGRP);
static int num_powered_cores;
static struct hrtimer lpm_hrtimer;
static struct kobj_attribute lpm_l2_kattr = __ATTR(l2, S_IRUGO|S_IWUSR,\
lpm_levels_attr_show, lpm_levels_attr_store);
static struct attribute *lpm_levels_attr[] = {
&lpm_l2_kattr.attr,
NULL,
};
static struct attribute_group lpm_levels_attr_grp = {
.attrs = lpm_levels_attr,
};
/* SYSFS */
static ssize_t lpm_levels_attr_show(
struct kobject *kobj, struct kobj_attribute *attr, char *buf)
{
struct kernel_param kp;
int rc;
kp.arg = &sysfs_dbg_l2_mode;
rc = param_get_uint(buf, &kp);
if (rc > 0) {
strlcat(buf, "\n", PAGE_SIZE);
rc++;
}
return rc;
}
static ssize_t lpm_levels_attr_store(struct kobject *kobj,
struct kobj_attribute *attr, const char *buf, size_t count)
{
struct kernel_param kp;
unsigned int temp;
int rc;
kp.arg = &temp;
rc = param_set_uint(buf, &kp);
if (rc)
return rc;
sysfs_dbg_l2_mode = temp;
lpm_system_level_update();
return count;
}
static int msm_pm_get_sleep_mode_value(const char *mode_name)
{
struct lpm_lookup_table pm_sm_lookup[] = {
{MSM_PM_SLEEP_MODE_WAIT_FOR_INTERRUPT,
"wfi"},
{MSM_PM_SLEEP_MODE_POWER_COLLAPSE_STANDALONE,
"standalone_pc"},
{MSM_PM_SLEEP_MODE_POWER_COLLAPSE,
"pc"},
{MSM_PM_SLEEP_MODE_RETENTION,
"retention"},
};
int i;
int ret = -EINVAL;
for (i = 0; i < ARRAY_SIZE(pm_sm_lookup); i++) {
if (!strcmp(mode_name, pm_sm_lookup[i].mode_name)) {
ret = pm_sm_lookup[i].modes;
break;
}
}
return ret;
}
static int lpm_set_l2_mode(struct lpm_system_state *system_state,
int sleep_mode)
{
int lpm = sleep_mode;
int rc = 0;
if (system_state->no_l2_saw)
goto bail_set_l2_mode;
msm_pm_set_l2_flush_flag(MSM_SCM_L2_ON);
switch (sleep_mode) {
case MSM_SPM_L2_MODE_POWER_COLLAPSE:
pr_info("Configuring for L2 power collapse\n");
msm_pm_set_l2_flush_flag(MSM_SCM_L2_OFF);
break;
case MSM_SPM_L2_MODE_GDHS:
msm_pm_set_l2_flush_flag(MSM_SCM_L2_GDHS);
break;
case MSM_SPM_L2_MODE_PC_NO_RPM:
msm_pm_set_l2_flush_flag(MSM_SCM_L2_OFF);
break;
case MSM_SPM_L2_MODE_RETENTION:
case MSM_SPM_L2_MODE_DISABLED:
break;
default:
lpm = MSM_SPM_L2_MODE_DISABLED;
break;
}
rc = msm_spm_l2_set_low_power_mode(lpm, true);
if (rc) {
if (rc == -ENXIO)
WARN_ON_ONCE(1);
else
pr_err("%s: Failed to set L2 low power mode %d, ERR %d",
__func__, lpm, rc);
}
bail_set_l2_mode:
return rc;
}
static void lpm_system_level_update(void)
{
int i;
struct lpm_system_level *l = NULL;
uint32_t max_l2_mode;
static DEFINE_MUTEX(lpm_lock);
mutex_lock(&lpm_lock);
if (num_powered_cores == 1)
allowed_l2_mode = MSM_SPM_L2_MODE_POWER_COLLAPSE;
else if (sys_state.allow_synched_levels)
allowed_l2_mode = MSM_SPM_L2_MODE_POWER_COLLAPSE;
else
allowed_l2_mode = default_l2_mode;
max_l2_mode = min(allowed_l2_mode, sysfs_dbg_l2_mode);
for (i = 0; i < sys_state.num_system_levels; i++) {
l = &sys_state.system_level[i];
l->available = !(l->l2_mode > max_l2_mode);
}
mutex_unlock(&lpm_lock);
}
static int lpm_system_mode_select(
struct lpm_system_state *system_state,
uint32_t sleep_us, bool from_idle)
{
int best_level = -1;
int i;
uint32_t best_level_pwr = ~0UL;
uint32_t pwr;
uint32_t latency_us = pm_qos_request(PM_QOS_CPU_DMA_LATENCY);
if (!system_state->system_level)
return -EINVAL;
for (i = 0; i < system_state->num_system_levels; i++) {
struct lpm_system_level *system_level =
&system_state->system_level[i];
struct power_params *pwr_param = &system_level->pwr;
if (!system_level->available)
continue;
if (system_level->sync &&
system_level->num_cpu_votes != num_powered_cores)
continue;
if (latency_us < pwr_param->latency_us)
continue;
if (sleep_us < pwr_param->time_overhead_us)
continue;
/*
* After the suspend prepare notifications its possible
* for the CPU to enter a system sleep mode. But MPM would have
* already requested a XO clock based on the wakeup irqs. To
* prevent suspend votes from being overriden by idle irqs, MPM
* doesn't send an updated MPM vote after suspend_prepare
* callback.
* To ensure that XO sleep vote isn't used if and when the
* device enters idle PC after suspend prepare callback,
* disallow any low power modes that notifies RPM after suspend
* prepare function is called
*/
if (suspend_in_progress && system_level->notify_rpm &&
from_idle)
continue;
if ((sleep_us >> 10) > pwr_param->time_overhead_us) {
pwr = pwr_param->ss_power;
} else {
pwr = pwr_param->ss_power;
pwr -= (pwr_param->time_overhead_us
* pwr_param->ss_power) / sleep_us;
pwr += pwr_param->energy_overhead / sleep_us;
}
if (best_level_pwr >= pwr) {
best_level = i;
best_level_pwr = pwr;
}
}
return best_level;
}
static void lpm_system_prepare(struct lpm_system_state *system_state,
int index, bool from_idle)
{
struct lpm_system_level *lvl;
struct clock_event_device *bc = tick_get_broadcast_device()->evtdev;
uint32_t sclk;
int64_t us = (~0ULL);
int dbg_mask;
int ret;
const struct cpumask *nextcpu;
spin_lock(&system_state->sync_lock);
if (num_powered_cores != system_state->num_cores_in_sync) {
spin_unlock(&system_state->sync_lock);
return;
}
if (from_idle) {
dbg_mask = lpm_lvl_dbg_msk & MSM_LPM_LVL_DBG_IDLE_LIMITS;
us = ktime_to_us(ktime_sub(bc->next_event, ktime_get()));
nextcpu = bc->cpumask;
} else {
dbg_mask = lpm_lvl_dbg_msk & MSM_LPM_LVL_DBG_SUSPEND_LIMITS;
nextcpu = cpumask_of(smp_processor_id());
}
lvl = &system_state->system_level[index];
ret = lpm_set_l2_mode(system_state, lvl->l2_mode);
if (ret && ret != -ENXIO) {
pr_warn("%s(): Cannot set L2 Mode %d, ret:%d\n",
__func__, lvl->l2_mode, ret);
goto bail_system_sleep;
}
if (lvl->notify_rpm) {
ret = msm_rpm_enter_sleep(dbg_mask, nextcpu);
if (ret) {
pr_err("rpm_enter_sleep() failed with rc = %d\n", ret);
goto bail_system_sleep;
}
if (!from_idle)
us = USEC_PER_SEC * msm_pm_sleep_time_override;
do_div(us, USEC_PER_SEC/SCLK_HZ);
sclk = (uint32_t)us;
msm_mpm_enter_sleep(sclk, from_idle, nextcpu);
}
system_state->last_entered_cluster_index = index;
spin_unlock(&system_state->sync_lock);
return;
bail_system_sleep:
if (default_l2_mode != system_state->system_level[index].l2_mode)
lpm_set_l2_mode(system_state, default_l2_mode);
spin_unlock(&system_state->sync_lock);
}
static void lpm_system_unprepare(struct lpm_system_state *system_state,
int cpu_index, bool from_idle)
{
int index, i;
struct lpm_cpu_level *cpu_level = &system_state->cpu_level[cpu_index];
bool first_core_up;
if (cpu_level->mode < system_state->sync_cpu_mode)
return;
spin_lock(&system_state->sync_lock);
first_core_up = (system_state->num_cores_in_sync == num_powered_cores);
system_state->num_cores_in_sync--;
if (!system_state->system_level)
goto unlock_and_return;
index = system_state->last_entered_cluster_index;
for (i = 0; i < system_state->num_system_levels; i++) {
struct lpm_system_level *system_lvl
= &system_state->system_level[i];
if (cpu_level->mode >= system_lvl->min_cpu_mode)
system_lvl->num_cpu_votes--;
}
if (!first_core_up)
goto unlock_and_return;
if (default_l2_mode != system_state->system_level[index].l2_mode)
lpm_set_l2_mode(system_state, default_l2_mode);
if (system_state->system_level[index].notify_rpm) {
msm_rpm_exit_sleep();
msm_mpm_exit_sleep(from_idle);
}
unlock_and_return:
spin_unlock(&system_state->sync_lock);
}
s32 msm_cpuidle_get_deep_idle_latency(void)
{
int i;
struct lpm_cpu_level *level = sys_state.cpu_level;
if (!level)
return 0;
for (i = 0; i < sys_state.num_cpu_levels; i++, level++) {
if (level->mode == MSM_PM_SLEEP_MODE_POWER_COLLAPSE)
break;
}
if (i == sys_state.num_cpu_levels)
return 0;
else
return level->pwr.latency_us;
}
static int lpm_cpu_callback(struct notifier_block *cpu_nb,
unsigned long action, void *hcpu)
{
switch (action & ~CPU_TASKS_FROZEN) {
case CPU_UP_PREPARE:
++num_powered_cores;
lpm_system_level_update();
break;
case CPU_DEAD:
case CPU_UP_CANCELED:
num_powered_cores = num_online_cpus();
lpm_system_level_update();
break;
case CPU_ONLINE:
smp_call_function_single((unsigned long)hcpu,
setup_broadcast_timer, (void *)true, 1);
break;
default:
break;
}
return NOTIFY_OK;
}
static enum hrtimer_restart lpm_hrtimer_cb(struct hrtimer *h)
{
return HRTIMER_NORESTART;
}
static void msm_pm_set_timer(uint32_t modified_time_us)
{
u64 modified_time_ns = modified_time_us * NSEC_PER_USEC;
ktime_t modified_ktime = ns_to_ktime(modified_time_ns);
lpm_hrtimer.function = lpm_hrtimer_cb;
hrtimer_start(&lpm_hrtimer, modified_ktime, HRTIMER_MODE_REL_PINNED);
}
static noinline int lpm_cpu_power_select(struct cpuidle_device *dev, int *index)
{
int best_level = -1;
uint32_t best_level_pwr = ~0UL;
uint32_t latency_us = pm_qos_request(PM_QOS_CPU_DMA_LATENCY);
uint32_t sleep_us =
(uint32_t)(ktime_to_us(tick_nohz_get_sleep_length()));
uint32_t modified_time_us = 0;
uint32_t next_event_us = 0;
uint32_t power;
int i;
if (!sys_state.cpu_level)
return -EINVAL;
if (!dev->cpu)
next_event_us = (uint32_t)(ktime_to_us(get_next_event_time()));
for (i = 0; i < sys_state.num_cpu_levels; i++) {
struct lpm_cpu_level *level = &sys_state.cpu_level[i];
struct power_params *pwr = &level->pwr;
uint32_t next_wakeup_us = sleep_us;
enum msm_pm_sleep_mode mode = level->mode;
bool allow;
if (level->sync && num_online_cpus() > 1
&& !sys_state.allow_synched_levels)
continue;
allow = msm_cpu_pm_check_mode(dev->cpu, mode, true);
if (!allow)
continue;
if (latency_us < pwr->latency_us)
continue;
if (next_event_us)
if (next_event_us < pwr->latency_us)
continue;
if (((next_event_us - pwr->latency_us) < sleep_us)
|| (next_event_us < sleep_us)) {
next_wakeup_us = next_event_us
- pwr->latency_us;
}
if (next_wakeup_us <= pwr->time_overhead_us)
continue;
if ((MSM_PM_SLEEP_MODE_POWER_COLLAPSE_STANDALONE == mode)
|| (MSM_PM_SLEEP_MODE_POWER_COLLAPSE == mode))
if (!dev->cpu && msm_rpm_waiting_for_ack())
break;
if ((next_wakeup_us >> 10) > pwr->latency_us) {
power = pwr->ss_power;
} else {
power = pwr->ss_power;
power -= (pwr->latency_us * pwr->ss_power)
/ next_wakeup_us;
power += pwr->energy_overhead / next_wakeup_us;
}
if (best_level_pwr >= power) {
best_level = i;
best_level_pwr = power;
if (next_event_us < sleep_us &&
(mode != MSM_PM_SLEEP_MODE_WAIT_FOR_INTERRUPT))
modified_time_us = next_event_us
- pwr->latency_us;
else
modified_time_us = 0;
}
}
if (modified_time_us && !dev->cpu)
msm_pm_set_timer(modified_time_us);
return best_level;
}
static int lpm_get_l2_cache_value(const char *l2_str)
{
int i;
struct lpm_lookup_table l2_mode_lookup[] = {
{MSM_SPM_L2_MODE_POWER_COLLAPSE, "l2_cache_pc"},
{MSM_SPM_L2_MODE_PC_NO_RPM, "l2_cache_pc_no_rpm"},
{MSM_SPM_L2_MODE_GDHS, "l2_cache_gdhs"},
{MSM_SPM_L2_MODE_RETENTION, "l2_cache_retention"},
{MSM_SPM_L2_MODE_DISABLED, "l2_cache_active"}
};
for (i = 0; i < ARRAY_SIZE(l2_mode_lookup); i++)
if (!strcmp(l2_str, l2_mode_lookup[i].mode_name))
return l2_mode_lookup[i].modes;
return -EINVAL;
}
static int lpm_levels_sysfs_add(void)
{
struct kobject *module_kobj = NULL;
struct kobject *low_power_kobj = NULL;
int rc = 0;
module_kobj = kset_find_obj(module_kset, KBUILD_MODNAME);
if (!module_kobj) {
pr_err("%s: cannot find kobject for module %s\n",
__func__, KBUILD_MODNAME);
rc = -ENOENT;
goto resource_sysfs_add_exit;
}
low_power_kobj = kobject_create_and_add(
"enable_low_power", module_kobj);
if (!low_power_kobj) {
pr_err("%s: cannot create kobject\n", __func__);
rc = -ENOMEM;
goto resource_sysfs_add_exit;
}
rc = sysfs_create_group(low_power_kobj, &lpm_levels_attr_grp);
resource_sysfs_add_exit:
if (rc) {
if (low_power_kobj) {
sysfs_remove_group(low_power_kobj,
&lpm_levels_attr_grp);
kobject_del(low_power_kobj);
}
}
return rc;
}
static int lpm_cpu_menu_select(struct cpuidle_device *dev, int *index)
{
int j;
for (; *index >= 0; (*index)--) {
int mode = 0;
bool allow = false;
allow = msm_cpu_pm_check_mode(dev->cpu, mode, true);
if (!allow)
continue;
for (j = sys_state.num_cpu_levels; j >= 0; j--) {
struct lpm_cpu_level *l = &sys_state.cpu_level[j];
if (mode == l->mode)
return j;
}
}
return -EPERM;
}
static inline void lpm_cpu_prepare(struct lpm_system_state *system_state,
int cpu_index, bool from_idle)
{
struct lpm_cpu_level *cpu_level = &system_state->cpu_level[cpu_index];
unsigned int cpu = smp_processor_id();
/* Use broadcast timer for aggregating sleep mode within a cluster.
* A broadcast timer could be used because of harware restriction or
* to ensure that we BC timer is used incase a cpu mode could trigger
* a cluster level sleep
*/
if (from_idle && (cpu_level->use_bc_timer ||
(cpu_level->mode >= system_state->sync_cpu_mode)))
clockevents_notify(CLOCK_EVT_NOTIFY_BROADCAST_ENTER, &cpu);
}
static inline void lpm_cpu_unprepare(struct lpm_system_state *system_state,
int cpu_index, bool from_idle)
{
struct lpm_cpu_level *cpu_level = &system_state->cpu_level[cpu_index];
unsigned int cpu = smp_processor_id();
if (from_idle && (cpu_level->use_bc_timer ||
(cpu_level->mode >= system_state->sync_cpu_mode)))
clockevents_notify(CLOCK_EVT_NOTIFY_BROADCAST_EXIT, &cpu);
}
static int lpm_system_select(struct lpm_system_state *system_state,
int cpu_index, bool from_idle)
{
uint64_t us = (~0ULL);
struct clock_event_device *ed;
struct lpm_cpu_level *cpu_level = &system_state->cpu_level[cpu_index];
int i;
bool last_core_down;
if (cpu_level->mode < system_state->sync_cpu_mode)
return -EINVAL;
spin_lock(&system_state->sync_lock);
last_core_down =
(++system_state->num_cores_in_sync == num_powered_cores);
if (!system_state->system_level) {
spin_unlock(&system_state->sync_lock);
return -EINVAL;
}
for (i = 0; i < system_state->num_system_levels; i++) {
struct lpm_system_level *system_lvl =
&system_state->system_level[i];
if (cpu_level->mode >= system_lvl->min_cpu_mode)
system_lvl->num_cpu_votes++;
}
spin_unlock(&system_state->sync_lock);
if (!last_core_down)
return -EBUSY;
ed = tick_get_broadcast_device()->evtdev;
if (!ed)
return -EINVAL;
if (from_idle)
us = ktime_to_us(ktime_sub(ed->next_event, ktime_get()));
else
us = (~0ULL);
return lpm_system_mode_select(system_state, (uint32_t)(us), from_idle);
}
static void lpm_enter_low_power(struct lpm_system_state *system_state,
int cpu_index, bool from_idle)
{
int idx;
struct lpm_cpu_level *cpu_level = &system_state->cpu_level[cpu_index];
cpu_level = &system_state->cpu_level[cpu_index];
lpm_cpu_prepare(system_state, cpu_index, from_idle);
idx = lpm_system_select(system_state, cpu_index, from_idle);
if (idx >= 0)
lpm_system_prepare(system_state, idx, from_idle);
msm_cpu_pm_enter_sleep(cpu_level->mode, from_idle);
lpm_system_unprepare(system_state, cpu_index, from_idle);
lpm_cpu_unprepare(system_state, cpu_index, from_idle);
}
static int lpm_cpuidle_enter(struct cpuidle_device *dev,
struct cpuidle_driver *drv, int index)
{
int64_t time = ktime_to_ns(ktime_get());
int idx;
idx = menu_select ? lpm_cpu_menu_select(dev, &index) :
lpm_cpu_power_select(dev, &index);
if (idx < 0) {
local_irq_enable();
return -EPERM;
}
lpm_enter_low_power(&sys_state, idx, true);
time = ktime_to_ns(ktime_get()) - time;
do_div(time, 1000);
dev->last_residency = (int)time;
local_irq_enable();
return index;
}
static int lpm_suspend_enter(suspend_state_t state)
{
int i;
for (i = sys_state.num_cpu_levels - 1; i >= 0; i--) {
bool allow = msm_cpu_pm_check_mode(smp_processor_id(),
sys_state.cpu_level[i].mode, false);
if (allow)
break;
}
if (i < 0)
return -EINVAL;
lpm_enter_low_power(&sys_state, i, false);
return 0;
}
static int lpm_suspend_prepare(void)
{
suspend_in_progress = true;
msm_mpm_suspend_prepare();
return 0;
}
static void lpm_suspend_wake(void)
{
msm_mpm_suspend_wake();
suspend_in_progress = false;
}
static struct platform_device lpm_dev = {
.name = "msm_pm",
.id = -1,
};
static const struct platform_suspend_ops lpm_suspend_ops = {
.enter = lpm_suspend_enter,
.valid = suspend_valid_only_mem,
.prepare_late = lpm_suspend_prepare,
.wake = lpm_suspend_wake,
};
static void setup_broadcast_timer(void *arg)
{
unsigned long reason = (unsigned long)arg;
int cpu = smp_processor_id();
reason = reason ?
CLOCK_EVT_NOTIFY_BROADCAST_ON : CLOCK_EVT_NOTIFY_BROADCAST_OFF;
clockevents_notify(reason, &cpu);
}
static struct cpuidle_driver msm_cpuidle_driver = {
.name = "msm_idle",
.owner = THIS_MODULE,
};
static void lpm_cpuidle_init(void)
{
int i = 0;
int state_count = 0;
if (!sys_state.cpu_level)
return;
BUG_ON(sys_state.num_cpu_levels > CPUIDLE_STATE_MAX);
for (i = 0; i < sys_state.num_cpu_levels; i++) {
struct cpuidle_state *st = &msm_cpuidle_driver.states[i];
struct lpm_cpu_level *cpu_level = &sys_state.cpu_level[i];
snprintf(st->name, CPUIDLE_NAME_LEN, "C%u\n", i);
snprintf(st->desc, CPUIDLE_DESC_LEN, cpu_level->name);
st->flags = 0;
st->exit_latency = cpu_level->pwr.latency_us;
st->power_usage = cpu_level->pwr.ss_power;
st->target_residency = 0;
st->enter = lpm_cpuidle_enter;
state_count++;
}
msm_cpuidle_driver.state_count = state_count;
msm_cpuidle_driver.safe_state_index = 0;
if (cpuidle_register(&msm_cpuidle_driver, NULL))
pr_err("%s(): Failed to register CPUIDLE device\n", __func__);
}
static int lpm_parse_power_params(struct device_node *node,
struct power_params *pwr)
{
char *key;
int ret;
key = "qcom,latency-us";
ret = of_property_read_u32(node, key, &pwr->latency_us);
if (ret)
goto fail;
key = "qcom,ss-power";
ret = of_property_read_u32(node, key, &pwr->ss_power);
if (ret)
goto fail;
key = "qcom,energy-overhead";
ret = of_property_read_u32(node, key, &pwr->energy_overhead);
if (ret)
goto fail;
key = "qcom,time-overhead";
ret = of_property_read_u32(node, key, &pwr->time_overhead_us);
if (ret)
goto fail;
fail:
if (ret)
pr_err("%s(): Error reading %s\n", __func__, key);
return ret;
}
static int lpm_cpu_probe(struct platform_device *pdev)
{
struct lpm_cpu_level *level = NULL, *l;
struct device_node *node = NULL;
int num_levels = 0;
char *key;
int ret;
for_each_child_of_node(pdev->dev.of_node, node)
num_levels++;
level = kzalloc(num_levels * sizeof(struct lpm_cpu_level),
GFP_KERNEL);
if (!level)
return -ENOMEM;
l = &level[0];
for_each_child_of_node(pdev->dev.of_node, node) {
key = "qcom,mode";
ret = of_property_read_string(node, key, &l->name);
if (ret) {
pr_err("%s(): Cannot read cpu mode%s\n", __func__, key);
goto fail;
}
l->mode = msm_pm_get_sleep_mode_value(l->name);
if (l->mode < 0) {
pr_err("%s():Cannot parse cpu mode:%s\n", __func__,
l->name);
goto fail;
}
if (l->mode == MSM_PM_SLEEP_MODE_POWER_COLLAPSE)
l->sync = true;
key = "qcom,use-broadcast-timer";
l->use_bc_timer = of_property_read_bool(node, key);
ret = lpm_parse_power_params(node, &l->pwr);
if (ret) {
pr_err("%s(): cannot Parse power params\n", __func__);
goto fail;
}
l++;
}
sys_state.cpu_level = level;
sys_state.num_cpu_levels = num_levels;
return ret;
fail:
kfree(level);
return ret;
}
static int lpm_system_probe(struct platform_device *pdev)
{
struct lpm_system_level *level = NULL, *l;
int num_levels = 0;
struct device_node *node;
char *key;
int ret;
for_each_child_of_node(pdev->dev.of_node, node)
num_levels++;
level = kzalloc(num_levels * sizeof(struct lpm_system_level),
GFP_KERNEL);
if (!level)
return -ENOMEM;
l = &level[0];
for_each_child_of_node(pdev->dev.of_node, node) {
key = "qcom,l2";
ret = of_property_read_string(node, key, &l->name);
if (ret) {
pr_err("%s(): Failed to read L2 mode\n", __func__);
goto fail;
}
l->l2_mode = lpm_get_l2_cache_value(l->name);
if (l->l2_mode < 0) {
pr_err("%s(): Failed to read l2 cache mode\n",
__func__);
goto fail;
}
if (l->l2_mode == MSM_SPM_L2_MODE_GDHS ||
l->l2_mode == MSM_SPM_L2_MODE_POWER_COLLAPSE)
l->notify_rpm = true;
if (l->l2_mode >= MSM_SPM_L2_MODE_GDHS)
l->sync = true;
ret = lpm_parse_power_params(node, &l->pwr);
if (ret) {
pr_err("%s(): Failed to parse power params\n",
__func__);
goto fail;
}
key = "qcom,sync-cpus";
l->sync = of_property_read_bool(node, key);
if (l->sync) {
const char *name;
key = "qcom,min-cpu-mode";
ret = of_property_read_string(node, key, &name);
if (ret) {
pr_err("%s(): Required key %snot found\n",
__func__, name);
goto fail;
}
l->min_cpu_mode = msm_pm_get_sleep_mode_value(name);
if (l->min_cpu_mode < 0) {
pr_err("%s(): Cannot parse cpu mode:%s\n",
__func__, name);
goto fail;
}
if (l->min_cpu_mode < sys_state.sync_cpu_mode)
sys_state.sync_cpu_mode = l->min_cpu_mode;
}
l++;
}
sys_state.system_level = level;
sys_state.num_system_levels = num_levels;
return ret;
fail:
kfree(level);
return ret;
}
static int lpm_probe(struct platform_device *pdev)
{
struct device_node *node = NULL;
char *key = NULL;
int ret;
node = pdev->dev.of_node;
key = "qcom,allow-synced-levels";
sys_state.allow_synched_levels = of_property_read_bool(node, key);
key = "qcom,no-l2-saw";
sys_state.no_l2_saw = of_property_read_bool(node, key);
sys_state.sync_cpu_mode = MSM_PM_SLEEP_MODE_NR;
spin_lock_init(&sys_state.sync_lock);
sys_state.num_cores_in_sync = 0;
ret = of_platform_populate(pdev->dev.of_node, NULL, NULL, &pdev->dev);
if (ret)
goto fail;
/* Do the following two steps only if L2 SAW is present */
num_powered_cores = num_online_cpus();
if (!sys_state.no_l2_saw) {
int ret;
const char *l2;
key = "qcom,default-l2-state";
ret = of_property_read_string(node, key, &l2);
if (ret) {
pr_err("%s(): Failed to read default L2 mode\n",
__func__);
goto fail;
}
default_l2_mode = lpm_get_l2_cache_value(l2);
if (default_l2_mode < 0) {
pr_err("%s(): Unable to parse default L2 mode\n",
__func__);
goto fail;
}
if (lpm_levels_sysfs_add())
goto fail;
msm_pm_set_l2_flush_flag(MSM_SCM_L2_ON);
} else {
msm_pm_set_l2_flush_flag(MSM_SCM_L2_OFF);
default_l2_mode = MSM_SPM_L2_MODE_POWER_COLLAPSE;
}
get_cpu();
on_each_cpu(setup_broadcast_timer, (void *)true, 1);
put_cpu();
register_hotcpu_notifier(&lpm_cpu_nblk);
lpm_system_level_update();
platform_device_register(&lpm_dev);
suspend_set_ops(&lpm_suspend_ops);
hrtimer_init(&lpm_hrtimer, CLOCK_MONOTONIC, HRTIMER_MODE_REL);
lpm_cpuidle_init();
return 0;
fail:
pr_err("%s: Error in name %s key %s\n", __func__, node->full_name, key);
return -EFAULT;
}
static struct of_device_id cpu_modes_mtch_tbl[] = {
{.compatible = "qcom,cpu-modes"},
{},
};
static struct platform_driver cpu_modes_driver = {
.probe = lpm_cpu_probe,
.driver = {
.name = "cpu-modes",
.owner = THIS_MODULE,
.of_match_table = cpu_modes_mtch_tbl,
},
};
static struct of_device_id system_modes_mtch_tbl[] = {
{.compatible = "qcom,system-modes"},
{},
};
static struct platform_driver system_modes_driver = {
.probe = lpm_system_probe,
.driver = {
.name = "system-modes",
.owner = THIS_MODULE,
.of_match_table = system_modes_mtch_tbl,
},
};
static struct of_device_id lpm_levels_match_table[] = {
{.compatible = "qcom,lpm-levels"},
{},
};
static struct platform_driver lpm_levels_driver = {
.probe = lpm_probe,
.driver = {
.name = "lpm-levels",
.owner = THIS_MODULE,
.of_match_table = lpm_levels_match_table,
},
};
static int __init lpm_levels_module_init(void)
{
int rc;
rc = platform_driver_register(&cpu_modes_driver);
if (rc) {
pr_err("Error registering %s\n", cpu_modes_driver.driver.name);
goto fail;
}
rc = platform_driver_register(&system_modes_driver);
if (rc) {
platform_driver_unregister(&cpu_modes_driver);
pr_err("Error registering %s\n",
system_modes_driver.driver.name);
goto fail;
}
rc = platform_driver_register(&lpm_levels_driver);
if (rc) {
platform_driver_unregister(&cpu_modes_driver);
platform_driver_unregister(&system_modes_driver);
pr_err("Error registering %s\n",
lpm_levels_driver.driver.name);
}
fail:
return rc;
}
late_initcall(lpm_levels_module_init);