133 lines
3.5 KiB
C
133 lines
3.5 KiB
C
/* arch/arm/mach-msm/cpufreq.c
|
|
*
|
|
* MSM architecture cpufreq driver
|
|
*
|
|
* Copyright (C) 2007 Google, Inc.
|
|
* Copyright (c) 2007 QUALCOMM Incorporated
|
|
* Author: Mike A. Chan <mikechan@google.com>
|
|
*
|
|
* This software is licensed under the terms of the GNU General Public
|
|
* License version 2, as published by the Free Software Foundation, and
|
|
* may be copied, distributed, and modified under those terms.
|
|
*
|
|
* 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/workqueue.h>
|
|
#include <linux/completion.h>
|
|
#include <linux/cpu.h>
|
|
#include <linux/cpumask.h>
|
|
#include <linux/sched.h>
|
|
#include <linux/suspend.h>
|
|
|
|
#include <linux/cpufreq.h>
|
|
#include <linux/earlysuspend.h>
|
|
#include <linux/init.h>
|
|
#include "acpuclock.h"
|
|
|
|
#ifdef CONFIG_MSM_CPU_FREQ_SCREEN
|
|
static void msm_early_suspend(struct early_suspend *handler) {
|
|
acpuclk_set_rate(CONFIG_MSM_CPU_FREQ_SCREEN_OFF * 1000, 0);
|
|
}
|
|
|
|
static void msm_late_resume(struct early_suspend *handler) {
|
|
acpuclk_set_rate(CONFIG_MSM_CPU_FREQ_SCREEN_ON * 1000, 0);
|
|
}
|
|
|
|
static struct early_suspend msm_power_suspend = {
|
|
.suspend = msm_early_suspend,
|
|
.resume = msm_late_resume,
|
|
};
|
|
|
|
static int __init clock_late_init(void)
|
|
{
|
|
register_early_suspend(&msm_power_suspend);
|
|
return 0;
|
|
}
|
|
|
|
late_initcall(clock_late_init);
|
|
#else
|
|
|
|
static int msm_cpufreq_target(struct cpufreq_policy *policy,
|
|
unsigned int target_freq,
|
|
unsigned int relation)
|
|
{
|
|
int index;
|
|
struct cpufreq_freqs freqs;
|
|
struct cpufreq_frequency_table *table =
|
|
cpufreq_frequency_get_table(smp_processor_id());
|
|
|
|
if (cpufreq_frequency_table_target(policy, table, target_freq, relation,
|
|
&index)) {
|
|
pr_err("cpufreq: invalid target_freq: %d\n", target_freq);
|
|
return -EINVAL;
|
|
}
|
|
|
|
if (policy->cur == table[index].frequency)
|
|
return 0;
|
|
|
|
#ifdef CONFIG_CPU_FREQ_DEBUG
|
|
printk("msm_cpufreq_target %d r %d (%d-%d) selected %d\n", target_freq,
|
|
relation, policy->min, policy->max, table[index].frequency);
|
|
#endif
|
|
freqs.old = policy->cur;
|
|
freqs.new = table[index].frequency;
|
|
freqs.cpu = smp_processor_id();
|
|
cpufreq_notify_transition(&freqs, CPUFREQ_PRECHANGE);
|
|
acpuclk_set_rate(table[index].frequency * 1000, 0);
|
|
cpufreq_notify_transition(&freqs, CPUFREQ_POSTCHANGE);
|
|
return 0;
|
|
}
|
|
|
|
static int msm_cpufreq_verify(struct cpufreq_policy *policy)
|
|
{
|
|
cpufreq_verify_within_limits(policy, policy->cpuinfo.min_freq,
|
|
policy->cpuinfo.max_freq);
|
|
return 0;
|
|
}
|
|
|
|
static int __init msm_cpufreq_init(struct cpufreq_policy *policy)
|
|
{
|
|
struct cpufreq_frequency_table *table =
|
|
cpufreq_frequency_get_table(smp_processor_id());
|
|
|
|
BUG_ON(cpufreq_frequency_table_cpuinfo(policy, table));
|
|
policy->cur = acpuclk_get_rate();
|
|
#ifdef CONFIG_HTCLEO_OVERCLOCK
|
|
policy->max = 998400;
|
|
#endif
|
|
#ifdef CONFIG_HTCLEO_EXOVERCLOCK
|
|
policy->max = 1113600;
|
|
#endif
|
|
policy->cpuinfo.transition_latency =
|
|
acpuclk_get_switch_time() * NSEC_PER_USEC;
|
|
return 0;
|
|
}
|
|
|
|
static struct freq_attr *msm_cpufreq_attr[] = {
|
|
&cpufreq_freq_attr_scaling_available_freqs,
|
|
NULL,
|
|
};
|
|
|
|
static struct cpufreq_driver msm_cpufreq_driver = {
|
|
/* lps calculations are handled here. */
|
|
.flags = CPUFREQ_STICKY | CPUFREQ_CONST_LOOPS,
|
|
.init = msm_cpufreq_init,
|
|
.verify = msm_cpufreq_verify,
|
|
.target = msm_cpufreq_target,
|
|
.name = "msm",
|
|
.attr = msm_cpufreq_attr,
|
|
};
|
|
|
|
static int __init msm_cpufreq_register(void)
|
|
{
|
|
return cpufreq_register_driver(&msm_cpufreq_driver);
|
|
}
|
|
|
|
device_initcall(msm_cpufreq_register);
|
|
#endif
|