2010-04-23 07:30:13 +04:00
/*
* Copyright ( C ) 2010 Google , Inc .
*
* Author :
* Colin Cross < ccross @ google . com >
* Based on arch / arm / plat - omap / cpu - omap . c , ( C ) 2005 Nokia Corporation
*
* 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/kernel.h>
# include <linux/module.h>
# include <linux/types.h>
# include <linux/sched.h>
# include <linux/cpufreq.h>
# include <linux/delay.h>
# include <linux/init.h>
# include <linux/err.h>
# include <linux/clk.h>
# include <linux/io.h>
2010-08-06 04:40:39 +04:00
# include <linux/suspend.h>
2010-04-23 07:30:13 +04:00
static struct cpufreq_frequency_table freq_table [ ] = {
2013-05-14 17:38:50 +04:00
{ . frequency = 216000 } ,
{ . frequency = 312000 } ,
{ . frequency = 456000 } ,
{ . frequency = 608000 } ,
{ . frequency = 760000 } ,
{ . frequency = 816000 } ,
{ . frequency = 912000 } ,
{ . frequency = 1000000 } ,
{ . frequency = CPUFREQ_TABLE_END } ,
2010-04-23 07:30:13 +04:00
} ;
# define NUM_CPUS 2
static struct clk * cpu_clk ;
2012-09-11 03:05:01 +04:00
static struct clk * pll_x_clk ;
static struct clk * pll_p_clk ;
2010-11-23 05:54:36 +03:00
static struct clk * emc_clk ;
2010-04-23 07:30:13 +04:00
static unsigned long target_cpu_speed [ NUM_CPUS ] ;
2010-08-06 04:40:39 +04:00
static DEFINE_MUTEX ( tegra_cpu_lock ) ;
static bool is_suspended ;
2010-04-23 07:30:13 +04:00
2011-10-10 08:57:04 +04:00
static unsigned int tegra_getspeed ( unsigned int cpu )
2010-04-23 07:30:13 +04:00
{
unsigned long rate ;
if ( cpu > = NUM_CPUS )
return 0 ;
rate = clk_get_rate ( cpu_clk ) / 1000 ;
return rate ;
}
2012-09-11 03:05:01 +04:00
static int tegra_cpu_clk_set_rate ( unsigned long rate )
{
int ret ;
/*
* Take an extra reference to the main pll so it doesn ' t turn
* off when we move the cpu off of it
*/
clk_prepare_enable ( pll_x_clk ) ;
ret = clk_set_parent ( cpu_clk , pll_p_clk ) ;
if ( ret ) {
pr_err ( " Failed to switch cpu to clock pll_p \n " ) ;
goto out ;
}
if ( rate = = clk_get_rate ( pll_p_clk ) )
goto out ;
ret = clk_set_rate ( pll_x_clk , rate ) ;
if ( ret ) {
pr_err ( " Failed to change pll_x to %lu \n " , rate ) ;
goto out ;
}
ret = clk_set_parent ( cpu_clk , pll_x_clk ) ;
if ( ret ) {
pr_err ( " Failed to switch cpu to clock pll_x \n " ) ;
goto out ;
}
out :
clk_disable_unprepare ( pll_x_clk ) ;
return ret ;
}
2013-03-24 10:26:43 +04:00
static int tegra_update_cpu_speed ( struct cpufreq_policy * policy ,
unsigned long rate )
2010-04-23 07:30:13 +04:00
{
int ret = 0 ;
2013-08-14 18:08:24 +04:00
if ( tegra_getspeed ( 0 ) = = rate )
2010-04-23 07:30:13 +04:00
return ret ;
2010-11-23 05:54:36 +03:00
/*
* Vote on memory bus frequency based on cpu frequency
* This sets the minimum frequency , display or avp may request higher
*/
if ( rate > = 816000 )
clk_set_rate ( emc_clk , 600000000 ) ; /* cpu 816 MHz, emc max */
else if ( rate > = 456000 )
clk_set_rate ( emc_clk , 300000000 ) ; /* cpu 456 MHz, emc 150Mhz */
else
clk_set_rate ( emc_clk , 100000000 ) ; /* emc 50Mhz */
2013-08-14 18:08:24 +04:00
ret = tegra_cpu_clk_set_rate ( rate * 1000 ) ;
if ( ret )
pr_err ( " cpu-tegra: Failed to set cpu frequency to %lu kHz \n " ,
rate ) ;
2010-04-23 07:30:13 +04:00
2013-06-19 09:48:20 +04:00
return ret ;
2010-04-23 07:30:13 +04:00
}
2010-08-06 04:40:39 +04:00
static unsigned long tegra_cpu_highest_speed ( void )
{
unsigned long rate = 0 ;
int i ;
for_each_online_cpu ( i )
rate = max ( rate , target_cpu_speed [ i ] ) ;
return rate ;
}
2013-10-25 18:15:48 +04:00
static int tegra_target ( struct cpufreq_policy * policy , unsigned int index )
2010-04-23 07:30:13 +04:00
{
unsigned int freq ;
2010-08-06 04:40:39 +04:00
int ret = 0 ;
mutex_lock ( & tegra_cpu_lock ) ;
if ( is_suspended ) {
ret = - EBUSY ;
goto out ;
}
2010-04-23 07:30:13 +04:00
2013-10-25 18:15:48 +04:00
freq = freq_table [ index ] . frequency ;
2010-04-23 07:30:13 +04:00
target_cpu_speed [ policy - > cpu ] = freq ;
2013-03-24 10:26:43 +04:00
ret = tegra_update_cpu_speed ( policy , tegra_cpu_highest_speed ( ) ) ;
2010-08-06 04:40:39 +04:00
out :
mutex_unlock ( & tegra_cpu_lock ) ;
return ret ;
2010-04-23 07:30:13 +04:00
}
2010-08-06 04:40:39 +04:00
static int tegra_pm_notify ( struct notifier_block * nb , unsigned long event ,
void * dummy )
{
mutex_lock ( & tegra_cpu_lock ) ;
if ( event = = PM_SUSPEND_PREPARE ) {
2013-03-24 10:26:43 +04:00
struct cpufreq_policy * policy = cpufreq_cpu_get ( 0 ) ;
2010-08-06 04:40:39 +04:00
is_suspended = true ;
pr_info ( " Tegra cpufreq suspend: setting frequency to %d kHz \n " ,
freq_table [ 0 ] . frequency ) ;
2013-03-24 10:26:43 +04:00
tegra_update_cpu_speed ( policy , freq_table [ 0 ] . frequency ) ;
cpufreq_cpu_put ( policy ) ;
2010-08-06 04:40:39 +04:00
} else if ( event = = PM_POST_SUSPEND ) {
is_suspended = false ;
}
mutex_unlock ( & tegra_cpu_lock ) ;
return NOTIFY_OK ;
}
static struct notifier_block tegra_cpu_pm_notifier = {
. notifier_call = tegra_pm_notify ,
} ;
2010-04-23 07:30:13 +04:00
static int tegra_cpu_init ( struct cpufreq_policy * policy )
{
2013-10-03 19:12:11 +04:00
int ret ;
2010-04-23 07:30:13 +04:00
if ( policy - > cpu > = NUM_CPUS )
return - EINVAL ;
2012-06-05 08:29:35 +04:00
clk_prepare_enable ( emc_clk ) ;
clk_prepare_enable ( cpu_clk ) ;
2010-10-21 04:47:59 +04:00
2013-10-03 18:59:05 +04:00
target_cpu_speed [ policy - > cpu ] = tegra_getspeed ( policy - > cpu ) ;
2010-04-23 07:30:13 +04:00
/* FIXME: what's the actual transition time? */
2013-10-03 19:12:11 +04:00
ret = cpufreq_generic_init ( policy , freq_table , 300 * 1000 ) ;
if ( ret ) {
clk_disable_unprepare ( cpu_clk ) ;
clk_disable_unprepare ( emc_clk ) ;
return ret ;
}
2010-04-23 07:30:13 +04:00
2010-08-06 04:40:39 +04:00
if ( policy - > cpu = = 0 )
register_pm_notifier ( & tegra_cpu_pm_notifier ) ;
2010-04-23 07:30:13 +04:00
return 0 ;
}
static int tegra_cpu_exit ( struct cpufreq_policy * policy )
{
2013-09-16 17:26:40 +04:00
cpufreq_frequency_table_put_attr ( policy - > cpu ) ;
2013-10-03 19:12:11 +04:00
clk_disable_unprepare ( cpu_clk ) ;
2012-06-05 08:29:35 +04:00
clk_disable_unprepare ( emc_clk ) ;
2010-04-23 07:30:13 +04:00
return 0 ;
}
static struct cpufreq_driver tegra_cpufreq_driver = {
2013-10-03 18:58:29 +04:00
. verify = cpufreq_generic_frequency_table_verify ,
2013-10-25 18:15:48 +04:00
. target_index = tegra_target ,
2010-04-23 07:30:13 +04:00
. get = tegra_getspeed ,
. init = tegra_cpu_init ,
. exit = tegra_cpu_exit ,
. name = " tegra " ,
2013-10-03 18:58:29 +04:00
. attr = cpufreq_generic_attr ,
2010-04-23 07:30:13 +04:00
} ;
static int __init tegra_cpufreq_init ( void )
{
2013-08-23 05:43:58 +04:00
cpu_clk = clk_get_sys ( NULL , " cclk " ) ;
2012-12-21 04:09:55 +04:00
if ( IS_ERR ( cpu_clk ) )
return PTR_ERR ( cpu_clk ) ;
pll_x_clk = clk_get_sys ( NULL , " pll_x " ) ;
if ( IS_ERR ( pll_x_clk ) )
return PTR_ERR ( pll_x_clk ) ;
2013-08-23 05:43:58 +04:00
pll_p_clk = clk_get_sys ( NULL , " pll_p " ) ;
2012-12-21 04:09:55 +04:00
if ( IS_ERR ( pll_p_clk ) )
return PTR_ERR ( pll_p_clk ) ;
emc_clk = clk_get_sys ( " cpu " , " emc " ) ;
if ( IS_ERR ( emc_clk ) ) {
clk_put ( cpu_clk ) ;
return PTR_ERR ( emc_clk ) ;
}
2010-04-23 07:30:13 +04:00
return cpufreq_register_driver ( & tegra_cpufreq_driver ) ;
}
static void __exit tegra_cpufreq_exit ( void )
{
cpufreq_unregister_driver ( & tegra_cpufreq_driver ) ;
2012-12-21 04:09:55 +04:00
clk_put ( emc_clk ) ;
clk_put ( cpu_clk ) ;
2010-04-23 07:30:13 +04:00
}
MODULE_AUTHOR ( " Colin Cross <ccross@android.com> " ) ;
MODULE_DESCRIPTION ( " cpufreq driver for Nvidia Tegra2 " ) ;
MODULE_LICENSE ( " GPL " ) ;
module_init ( tegra_cpufreq_init ) ;
module_exit ( tegra_cpufreq_exit ) ;