2005-04-16 15:20:36 -07:00
/*
* arch / sh / kernel / cpufreq . c
*
* cpufreq driver for the SuperH processors .
*
2012-01-27 17:49:16 +09:00
* Copyright ( C ) 2002 - 2012 Paul Mundt
2005-04-16 15:20:36 -07:00
* Copyright ( C ) 2002 M . R . Brown
*
2007-07-20 13:38:19 +09:00
* Clock framework bits from arch / avr32 / mach - at32ap / cpufreq . c
*
* Copyright ( C ) 2004 - 2007 Atmel Corporation
*
* This file is subject to the terms and conditions of the GNU General Public
* License . See the file " COPYING " in the main directory of this archive
* for more details .
2005-04-16 15:20:36 -07:00
*/
2012-01-27 19:44:49 +09:00
# define pr_fmt(fmt) "cpufreq: " fmt
2005-04-16 15:20:36 -07:00
# include <linux/types.h>
# include <linux/cpufreq.h>
# include <linux/kernel.h>
# include <linux/module.h>
# include <linux/init.h>
2007-07-20 13:38:19 +09:00
# include <linux/err.h>
2005-04-16 15:20:36 -07:00
# include <linux/cpumask.h>
2012-01-27 19:44:49 +09:00
# include <linux/cpu.h>
2005-04-16 15:20:36 -07:00
# include <linux/smp.h>
2005-10-30 15:03:48 -08:00
# include <linux/sched.h> /* set_cpus_allowed() */
2007-07-20 13:38:19 +09:00
# include <linux/clk.h>
2012-01-27 17:49:16 +09:00
# include <linux/percpu.h>
# include <linux/sh_clk.h>
2005-04-16 15:20:36 -07:00
2012-01-27 17:49:16 +09:00
static DEFINE_PER_CPU ( struct clk , sh_cpuclk ) ;
2005-04-16 15:20:36 -07:00
2007-07-20 13:38:19 +09:00
static unsigned int sh_cpufreq_get ( unsigned int cpu )
2005-04-16 15:20:36 -07:00
{
2012-01-27 17:49:16 +09:00
return ( clk_get_rate ( & per_cpu ( sh_cpuclk , cpu ) ) + 500 ) / 1000 ;
2005-04-16 15:20:36 -07:00
}
/*
* Here we notify other drivers of the proposed change and the final change .
*/
2007-07-20 13:38:19 +09:00
static int sh_cpufreq_target ( struct cpufreq_policy * policy ,
unsigned int target_freq ,
unsigned int relation )
2005-04-16 15:20:36 -07:00
{
2007-07-20 13:38:19 +09:00
unsigned int cpu = policy - > cpu ;
2012-01-27 17:49:16 +09:00
struct clk * cpuclk = & per_cpu ( sh_cpuclk , cpu ) ;
2005-04-16 15:20:36 -07:00
cpumask_t cpus_allowed ;
struct cpufreq_freqs freqs ;
2012-01-27 19:44:49 +09:00
struct device * dev ;
2007-07-20 13:38:19 +09:00
long freq ;
2005-04-16 15:20:36 -07:00
if ( ! cpu_online ( cpu ) )
return - ENODEV ;
cpus_allowed = current - > cpus_allowed ;
2010-03-26 22:03:49 +00:00
set_cpus_allowed_ptr ( current , cpumask_of ( cpu ) ) ;
2005-04-16 15:20:36 -07:00
BUG_ON ( smp_processor_id ( ) ! = cpu ) ;
2012-01-27 19:44:49 +09:00
dev = get_cpu_device ( cpu ) ;
2007-07-20 13:38:19 +09:00
/* Convert target_freq from kHz to Hz */
freq = clk_round_rate ( cpuclk , target_freq * 1000 ) ;
2005-04-16 15:20:36 -07:00
2007-07-20 13:38:19 +09:00
if ( freq < ( policy - > min * 1000 ) | | freq > ( policy - > max * 1000 ) )
return - EINVAL ;
2012-01-27 19:44:49 +09:00
dev_dbg ( dev , " requested frequency %u Hz \n " , target_freq * 1000 ) ;
2005-04-16 15:20:36 -07:00
2007-07-20 13:38:19 +09:00
freqs . cpu = cpu ;
freqs . old = sh_cpufreq_get ( cpu ) ;
freqs . new = ( freq + 500 ) / 1000 ;
freqs . flags = 0 ;
cpufreq_notify_transition ( & freqs , CPUFREQ_PRECHANGE ) ;
2010-03-26 22:03:49 +00:00
set_cpus_allowed_ptr ( current , & cpus_allowed ) ;
2007-07-20 13:38:19 +09:00
clk_set_rate ( cpuclk , freq ) ;
2005-04-16 15:20:36 -07:00
cpufreq_notify_transition ( & freqs , CPUFREQ_POSTCHANGE ) ;
2012-01-27 19:44:49 +09:00
dev_dbg ( dev , " set frequency %lu Hz \n " , freq ) ;
2007-07-20 13:38:19 +09:00
2005-04-16 15:20:36 -07:00
return 0 ;
}
static int sh_cpufreq_cpu_init ( struct cpufreq_policy * policy )
{
2012-01-27 17:49:16 +09:00
unsigned int cpu = policy - > cpu ;
struct clk * cpuclk = & per_cpu ( sh_cpuclk , cpu ) ;
2012-01-27 19:44:49 +09:00
struct device * dev ;
2012-01-27 17:49:16 +09:00
if ( ! cpu_online ( cpu ) )
2005-04-16 15:20:36 -07:00
return - ENODEV ;
2012-01-27 19:44:49 +09:00
dev = get_cpu_device ( cpu ) ;
cpuclk = clk_get ( dev , " cpu_clk " ) ;
2007-07-20 13:38:19 +09:00
if ( IS_ERR ( cpuclk ) ) {
2012-01-27 19:44:49 +09:00
dev_err ( dev , " couldn't get CPU clk \n " ) ;
2007-07-20 13:38:19 +09:00
return PTR_ERR ( cpuclk ) ;
}
2005-04-16 15:20:36 -07:00
/* cpuinfo and default policy values */
2007-07-20 13:38:19 +09:00
policy - > cpuinfo . min_freq = ( clk_round_rate ( cpuclk , 1 ) + 500 ) / 1000 ;
policy - > cpuinfo . max_freq = ( clk_round_rate ( cpuclk , ~ 0UL ) + 500 ) / 1000 ;
2005-04-16 15:20:36 -07:00
policy - > cpuinfo . transition_latency = CPUFREQ_ETERNAL ;
2012-01-27 17:49:16 +09:00
policy - > cur = sh_cpufreq_get ( cpu ) ;
2007-07-20 13:38:19 +09:00
policy - > min = policy - > cpuinfo . min_freq ;
policy - > max = policy - > cpuinfo . max_freq ;
2005-04-16 15:20:36 -07:00
2007-07-20 13:38:19 +09:00
/*
* Catch the cases where the clock framework hasn ' t been wired up
* properly to support scaling .
*/
if ( unlikely ( policy - > min = = policy - > max ) ) {
2012-01-27 19:44:49 +09:00
dev_err ( dev , " rate rounding not supported on this CPU. \n " ) ;
2007-07-20 13:38:19 +09:00
clk_put ( cpuclk ) ;
2005-04-16 15:20:36 -07:00
return - EINVAL ;
2007-07-20 13:38:19 +09:00
}
2005-04-16 15:20:36 -07:00
2012-01-27 19:44:49 +09:00
dev_info ( dev , " CPU Frequencies - Minimum %u.%03u MHz, "
2007-07-20 13:38:19 +09:00
" Maximum %u.%03u MHz. \n " ,
2012-01-27 19:44:49 +09:00
policy - > min / 1000 , policy - > min % 1000 ,
2007-07-20 13:38:19 +09:00
policy - > max / 1000 , policy - > max % 1000 ) ;
2005-04-16 15:20:36 -07:00
2007-07-20 13:38:19 +09:00
return 0 ;
}
2005-04-16 15:20:36 -07:00
2007-07-20 13:38:19 +09:00
static int sh_cpufreq_verify ( struct cpufreq_policy * policy )
{
cpufreq_verify_within_limits ( policy , policy - > cpuinfo . min_freq ,
policy - > cpuinfo . max_freq ) ;
return 0 ;
}
static int sh_cpufreq_exit ( struct cpufreq_policy * policy )
{
2012-01-27 17:49:16 +09:00
struct clk * cpuclk = & per_cpu ( sh_cpuclk , policy - > cpu ) ;
2007-07-20 13:38:19 +09:00
clk_put ( cpuclk ) ;
2005-04-16 15:20:36 -07:00
return 0 ;
}
static struct cpufreq_driver sh_cpufreq_driver = {
. owner = THIS_MODULE ,
2007-07-20 13:38:19 +09:00
. name = " sh " ,
2005-04-16 15:20:36 -07:00
. init = sh_cpufreq_cpu_init ,
. verify = sh_cpufreq_verify ,
. target = sh_cpufreq_target ,
2007-07-20 13:38:19 +09:00
. get = sh_cpufreq_get ,
. exit = sh_cpufreq_exit ,
2005-04-16 15:20:36 -07:00
} ;
2007-07-20 13:38:19 +09:00
static int __init sh_cpufreq_module_init ( void )
2005-04-16 15:20:36 -07:00
{
2012-01-27 19:44:49 +09:00
pr_notice ( " SuperH CPU frequency driver. \n " ) ;
2007-07-20 13:38:19 +09:00
return cpufreq_register_driver ( & sh_cpufreq_driver ) ;
2005-04-16 15:20:36 -07:00
}
2007-07-20 13:38:19 +09:00
static void __exit sh_cpufreq_module_exit ( void )
2005-04-16 15:20:36 -07:00
{
cpufreq_unregister_driver ( & sh_cpufreq_driver ) ;
}
2007-07-20 13:38:19 +09:00
module_init ( sh_cpufreq_module_init ) ;
module_exit ( sh_cpufreq_module_exit ) ;
2005-04-16 15:20:36 -07:00
MODULE_AUTHOR ( " Paul Mundt <lethal@linux-sh.org> " ) ;
MODULE_DESCRIPTION ( " cpufreq driver for SuperH " ) ;
MODULE_LICENSE ( " GPL " ) ;