2005-04-16 15:20:36 -07:00
/*
* 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
2017-04-12 22:07:36 +02:00
struct cpufreq_target {
struct cpufreq_policy * policy ;
unsigned int freq ;
} ;
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
}
2017-04-12 22:07:36 +02:00
static long __sh_cpufreq_target ( void * arg )
2005-04-16 15:20:36 -07:00
{
2017-04-12 22:07:36 +02:00
struct cpufreq_target * target = arg ;
struct cpufreq_policy * policy = target - > policy ;
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
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
2017-04-12 22:07:36 +02:00
if ( smp_processor_id ( ) ! = cpu )
return - ENODEV ;
2005-04-16 15:20:36 -07:00
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 */
2017-04-12 22:07:36 +02:00
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 ;
2017-04-12 22:07:36 +02: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 . old = sh_cpufreq_get ( cpu ) ;
freqs . new = ( freq + 500 ) / 1000 ;
freqs . flags = 0 ;
2017-04-12 22:07:36 +02:00
cpufreq_freq_transition_begin ( target - > policy , & freqs ) ;
2007-07-20 13:38:19 +09:00
clk_set_rate ( cpuclk , freq ) ;
2017-04-12 22:07:36 +02:00
cpufreq_freq_transition_end ( target - > policy , & freqs , 0 ) ;
2005-04-16 15:20:36 -07:00
2012-01-27 19:44:49 +09:00
dev_dbg ( dev , " set frequency %lu Hz \n " , freq ) ;
2005-04-16 15:20:36 -07:00
return 0 ;
}
2017-04-12 22:07:36 +02:00
/*
* Here we notify other drivers of the proposed change and the final change .
*/
static int sh_cpufreq_target ( struct cpufreq_policy * policy ,
unsigned int target_freq ,
unsigned int relation )
{
struct cpufreq_target data = { . policy = policy , . freq = target_freq } ;
return work_on_cpu ( policy - > cpu , __sh_cpufreq_target , & data ) ;
}
2012-01-27 20:18:24 +09:00
static int sh_cpufreq_verify ( struct cpufreq_policy * policy )
{
struct clk * cpuclk = & per_cpu ( sh_cpuclk , policy - > cpu ) ;
struct cpufreq_frequency_table * freq_table ;
freq_table = cpuclk - > nr_freqs ? cpuclk - > freq_table : NULL ;
if ( freq_table )
return cpufreq_frequency_table_verify ( policy , freq_table ) ;
2013-10-02 14:13:19 +05:30
cpufreq_verify_within_cpu_limits ( policy ) ;
2012-01-27 20:18:24 +09:00
policy - > min = ( clk_round_rate ( cpuclk , 1 ) + 500 ) / 1000 ;
policy - > max = ( clk_round_rate ( cpuclk , ~ 0UL ) + 500 ) / 1000 ;
2013-10-02 14:13:19 +05:30
cpufreq_verify_within_cpu_limits ( policy ) ;
2012-01-27 20:18:24 +09:00
return 0 ;
}
2005-04-16 15:20:36 -07:00
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 20:18:24 +09:00
struct cpufreq_frequency_table * freq_table ;
2012-01-27 19:44:49 +09:00
struct device * dev ;
2012-01-27 17:49:16 +09:00
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
2012-01-27 20:18:24 +09:00
freq_table = cpuclk - > nr_freqs ? cpuclk - > freq_table : NULL ;
if ( freq_table ) {
2012-01-27 20:43:14 +09:00
int result ;
2012-01-27 20:18:24 +09:00
2013-09-16 18:56:35 +05:30
result = cpufreq_table_validate_and_show ( policy , freq_table ) ;
if ( result )
return result ;
2012-01-27 20:18:24 +09:00
} else {
2012-01-27 20:43:14 +09:00
dev_notice ( dev , " no frequency table found, falling back "
" to rate rounding. \n " ) ;
2013-04-01 12:57:48 +00:00
policy - > min = policy - > cpuinfo . min_freq =
2012-01-27 20:43:14 +09:00
( clk_round_rate ( cpuclk , 1 ) + 500 ) / 1000 ;
2013-04-01 12:57:48 +00:00
policy - > max = policy - > cpuinfo . max_freq =
2012-01-27 20:43:14 +09:00
( clk_round_rate ( cpuclk , ~ 0UL ) + 500 ) / 1000 ;
2007-07-20 13:38:19 +09:00
}
2005-04-16 15:20:36 -07:00
2012-01-27 20:18:24 +09:00
policy - > cpuinfo . transition_latency = CPUFREQ_ETERNAL ;
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
2012-01-27 20:18:24 +09:00
static int sh_cpufreq_cpu_exit ( struct cpufreq_policy * policy )
2007-07-20 13:38:19 +09:00
{
2012-01-27 20:18:24 +09:00
unsigned int cpu = policy - > cpu ;
struct clk * cpuclk = & per_cpu ( sh_cpuclk , cpu ) ;
2007-07-20 13:38:19 +09:00
clk_put ( cpuclk ) ;
2012-01-27 20:18:24 +09:00
2005-04-16 15:20:36 -07:00
return 0 ;
}
static struct cpufreq_driver sh_cpufreq_driver = {
2007-07-20 13:38:19 +09:00
. name = " sh " ,
. get = sh_cpufreq_get ,
2012-01-27 20:18:24 +09:00
. target = sh_cpufreq_target ,
. verify = sh_cpufreq_verify ,
. init = sh_cpufreq_cpu_init ,
. exit = sh_cpufreq_cpu_exit ,
2013-10-03 20:28:25 +05:30
. attr = cpufreq_generic_attr ,
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 " ) ;