2009-11-16 20:32:59 +03:00
/*
* Cpufreq driver for the loongson - 2 processors
*
* The 2 E revision of loongson processor not support this feature .
*
* Copyright ( C ) 2006 - 2008 Lemote Inc . & Insititute of Computing Technology
* Author : Yanhua , yanh @ lemote . com
*
* 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 .
*/
# include <linux/cpufreq.h>
# include <linux/module.h>
# include <linux/err.h>
# include <linux/sched.h> /* set_cpus_allowed() */
# include <linux/delay.h>
# include <linux/platform_device.h>
# include <asm/clock.h>
2013-05-21 18:59:19 +04:00
# include <asm/idle.h>
2009-11-16 20:32:59 +03:00
2012-08-01 19:15:32 +04:00
# include <asm/mach-loongson/loongson.h>
2009-11-16 20:32:59 +03:00
static uint nowait ;
static void ( * saved_cpu_wait ) ( void ) ;
static int loongson2_cpu_freq_notifier ( struct notifier_block * nb ,
unsigned long val , void * data ) ;
static struct notifier_block loongson2_cpufreq_notifier_block = {
. notifier_call = loongson2_cpu_freq_notifier
} ;
static int loongson2_cpu_freq_notifier ( struct notifier_block * nb ,
unsigned long val , void * data )
{
if ( val = = CPUFREQ_POSTCHANGE )
current_cpu_data . udelay_val = loops_per_jiffy ;
return 0 ;
}
/*
* Here we notify other drivers of the proposed change and the final change .
*/
static int loongson2_cpufreq_target ( struct cpufreq_policy * policy ,
2013-10-25 18:15:48 +04:00
unsigned int index )
2009-11-16 20:32:59 +03:00
{
unsigned int cpu = policy - > cpu ;
cpumask_t cpus_allowed ;
unsigned int freq ;
cpus_allowed = current - > cpus_allowed ;
2010-03-27 01:03:07 +03:00
set_cpus_allowed_ptr ( current , cpumask_of ( cpu ) ) ;
2009-11-16 20:32:59 +03:00
freq =
( ( cpu_clock_freq / 1000 ) *
2013-10-25 18:15:48 +04:00
loongson2_clockmod_table [ index ] . driver_data ) / 8 ;
2009-11-16 20:32:59 +03:00
2010-03-27 01:03:07 +03:00
set_cpus_allowed_ptr ( current , & cpus_allowed ) ;
2009-11-16 20:32:59 +03:00
/* setting the cpu frequency */
2014-04-03 23:24:01 +04:00
clk_set_rate ( policy - > clk , freq * 1000 ) ;
2009-11-16 20:32:59 +03:00
return 0 ;
}
static int loongson2_cpufreq_cpu_init ( struct cpufreq_policy * policy )
{
2014-04-02 08:44:24 +04:00
struct clk * cpuclk ;
2009-11-16 20:32:59 +03:00
int i ;
2013-01-03 13:34:20 +04:00
unsigned long rate ;
int ret ;
2009-11-16 20:32:59 +03:00
cpuclk = clk_get ( NULL , " cpu_clk " ) ;
if ( IS_ERR ( cpuclk ) ) {
printk ( KERN_ERR " cpufreq: couldn't get CPU clk \n " ) ;
return PTR_ERR ( cpuclk ) ;
}
2013-01-03 13:34:20 +04:00
rate = cpu_clock_freq / 1000 ;
if ( ! rate ) {
clk_put ( cpuclk ) ;
2009-11-16 20:32:59 +03:00
return - EINVAL ;
2013-01-03 13:34:20 +04:00
}
2009-11-16 20:32:59 +03:00
/* clock table init */
for ( i = 2 ;
( loongson2_clockmod_table [ i ] . frequency ! = CPUFREQ_TABLE_END ) ;
i + + )
2013-01-03 13:34:20 +04:00
loongson2_clockmod_table [ i ] . frequency = ( rate * i ) / 8 ;
2009-11-16 20:32:59 +03:00
2014-04-03 23:24:01 +04:00
ret = clk_set_rate ( cpuclk , rate * 1000 ) ;
2013-08-05 22:27:12 +04:00
if ( ret ) {
clk_put ( cpuclk ) ;
return ret ;
}
2014-01-09 19:08:43 +04:00
policy - > clk = cpuclk ;
2013-10-03 18:59:16 +04:00
return cpufreq_generic_init ( policy , & loongson2_clockmod_table [ 0 ] , 0 ) ;
2009-11-16 20:32:59 +03:00
}
static int loongson2_cpufreq_exit ( struct cpufreq_policy * policy )
{
2014-01-09 19:08:43 +04:00
clk_put ( policy - > clk ) ;
2009-11-16 20:32:59 +03:00
return 0 ;
}
static struct cpufreq_driver loongson2_cpufreq_driver = {
. name = " loongson2 " ,
. init = loongson2_cpufreq_cpu_init ,
2013-10-03 18:58:11 +04:00
. verify = cpufreq_generic_frequency_table_verify ,
2013-10-25 18:15:48 +04:00
. target_index = loongson2_cpufreq_target ,
2014-01-09 19:08:43 +04:00
. get = cpufreq_generic_get ,
2009-11-16 20:32:59 +03:00
. exit = loongson2_cpufreq_exit ,
2013-10-03 18:58:11 +04:00
. attr = cpufreq_generic_attr ,
2009-11-16 20:32:59 +03:00
} ;
static struct platform_device_id platform_device_ids [ ] = {
{
. name = " loongson2_cpufreq " ,
} ,
{ }
} ;
MODULE_DEVICE_TABLE ( platform , platform_device_ids ) ;
static struct platform_driver platform_driver = {
. driver = {
. name = " loongson2_cpufreq " ,
} ,
. id_table = platform_device_ids ,
} ;
2012-08-01 19:15:32 +04:00
/*
* This is the simple version of Loongson - 2 wait , Maybe we need do this in
* interrupt disabled context .
*/
static DEFINE_SPINLOCK ( loongson2_wait_lock ) ;
static void loongson2_cpu_wait ( void )
{
unsigned long flags ;
u32 cpu_freq ;
spin_lock_irqsave ( & loongson2_wait_lock , flags ) ;
2014-06-26 07:41:27 +04:00
cpu_freq = LOONGSON_CHIPCFG ( 0 ) ;
LOONGSON_CHIPCFG ( 0 ) & = ~ 0x7 ; /* Put CPU into wait mode */
LOONGSON_CHIPCFG ( 0 ) = cpu_freq ; /* Restore CPU state */
2012-08-01 19:15:32 +04:00
spin_unlock_irqrestore ( & loongson2_wait_lock , flags ) ;
2013-05-21 16:05:27 +04:00
local_irq_enable ( ) ;
2012-08-01 19:15:32 +04:00
}
2009-11-16 20:32:59 +03:00
static int __init cpufreq_init ( void )
{
int ret ;
/* Register platform stuff */
ret = platform_driver_register ( & platform_driver ) ;
if ( ret )
return ret ;
pr_info ( " cpufreq: Loongson-2F CPU frequency driver. \n " ) ;
cpufreq_register_notifier ( & loongson2_cpufreq_notifier_block ,
CPUFREQ_TRANSITION_NOTIFIER ) ;
ret = cpufreq_register_driver ( & loongson2_cpufreq_driver ) ;
if ( ! ret & & ! nowait ) {
saved_cpu_wait = cpu_wait ;
cpu_wait = loongson2_cpu_wait ;
}
return ret ;
}
static void __exit cpufreq_exit ( void )
{
if ( ! nowait & & saved_cpu_wait )
cpu_wait = saved_cpu_wait ;
cpufreq_unregister_driver ( & loongson2_cpufreq_driver ) ;
cpufreq_unregister_notifier ( & loongson2_cpufreq_notifier_block ,
CPUFREQ_TRANSITION_NOTIFIER ) ;
platform_driver_unregister ( & platform_driver ) ;
}
module_init ( cpufreq_init ) ;
module_exit ( cpufreq_exit ) ;
module_param ( nowait , uint , 0644 ) ;
MODULE_PARM_DESC ( nowait , " Disable Loongson-2F specific wait " ) ;
MODULE_AUTHOR ( " Yanhua <yanh@lemote.com> " ) ;
MODULE_DESCRIPTION ( " cpufreq driver for Loongson2F " ) ;
MODULE_LICENSE ( " GPL " ) ;