2019-06-04 10:11:33 +02:00
// SPDX-License-Identifier: GPL-2.0-only
2009-09-22 21:14:00 +05:30
/*
* CPU frequency scaling for DaVinci
*
* Copyright ( C ) 2009 Texas Instruments Incorporated - http : //www.ti.com/
*
* Based on linux / arch / arm / plat - omap / cpu - omap . c . Original Copyright follows :
*
* Copyright ( C ) 2005 Nokia Corporation
* Written by Tony Lindgren < tony @ atomide . com >
*
* Based on cpu - sa1110 . c , Copyright ( C ) 2001 Russell King
*
* Copyright ( C ) 2007 - 2008 Texas Instruments , Inc .
* Updated to support OMAP3
* Rajendra Nayak < rnayak @ ti . com >
*/
# include <linux/types.h>
# include <linux/cpufreq.h>
# include <linux/init.h>
# include <linux/err.h>
# include <linux/clk.h>
2019-02-14 18:05:05 +01:00
# include <linux/platform_data/davinci-cpufreq.h>
2009-09-22 21:14:00 +05:30
# include <linux/platform_device.h>
2011-07-31 16:17:29 -04:00
# include <linux/export.h>
2009-09-22 21:14:00 +05:30
struct davinci_cpufreq {
struct device * dev ;
struct clk * armclk ;
2010-07-20 16:46:50 +05:30
struct clk * asyncclk ;
unsigned long asyncrate ;
2009-09-22 21:14:00 +05:30
} ;
static struct davinci_cpufreq cpufreq ;
2013-10-25 19:45:48 +05:30
static int davinci_target ( struct cpufreq_policy * policy , unsigned int idx )
2009-09-22 21:14:00 +05:30
{
struct davinci_cpufreq_config * pdata = cpufreq . dev - > platform_data ;
struct clk * armclk = cpufreq . armclk ;
2013-08-14 19:38:24 +05:30
unsigned int old_freq , new_freq ;
int ret = 0 ;
2009-09-22 21:14:00 +05:30
2014-01-09 20:38:43 +05:30
old_freq = policy - > cur ;
2013-08-14 19:38:24 +05:30
new_freq = pdata - > freq_table [ idx ] . frequency ;
2009-09-22 21:14:00 +05:30
/* if moving to higher frequency, up the voltage beforehand */
2013-08-14 19:38:24 +05:30
if ( pdata - > set_voltage & & new_freq > old_freq ) {
2010-07-20 16:46:48 +05:30
ret = pdata - > set_voltage ( idx ) ;
if ( ret )
2013-08-14 19:38:24 +05:30
return ret ;
2010-07-20 16:46:48 +05:30
}
2009-09-22 21:14:00 +05:30
2016-12-07 16:22:18 +01:00
ret = clk_set_rate ( armclk , new_freq * 1000 ) ;
2010-07-20 16:46:48 +05:30
if ( ret )
2013-08-14 19:38:24 +05:30
return ret ;
2009-09-22 21:14:00 +05:30
2010-07-20 16:46:50 +05:30
if ( cpufreq . asyncclk ) {
ret = clk_set_rate ( cpufreq . asyncclk , cpufreq . asyncrate ) ;
if ( ret )
2013-08-14 19:38:24 +05:30
return ret ;
2010-07-20 16:46:50 +05:30
}
2009-09-22 21:14:00 +05:30
/* if moving to lower freq, lower the voltage after lowering freq */
2013-08-14 19:38:24 +05:30
if ( pdata - > set_voltage & & new_freq < old_freq )
2009-09-22 21:14:00 +05:30
pdata - > set_voltage ( idx ) ;
2013-08-14 19:38:24 +05:30
return 0 ;
2009-09-22 21:14:00 +05:30
}
2011-02-28 15:51:33 +05:30
static int davinci_cpu_init ( struct cpufreq_policy * policy )
2009-09-22 21:14:00 +05:30
{
int result = 0 ;
struct davinci_cpufreq_config * pdata = cpufreq . dev - > platform_data ;
struct cpufreq_frequency_table * freq_table = pdata - > freq_table ;
if ( policy - > cpu ! = 0 )
return - EINVAL ;
2009-10-22 15:12:16 +05:30
/* Finish platform specific initialization */
if ( pdata - > init ) {
result = pdata - > init ( ) ;
if ( result )
return result ;
}
2014-01-09 20:38:43 +05:30
policy - > clk = cpufreq . armclk ;
2009-09-22 21:14:00 +05:30
/*
* Time measurement across the target ( ) function yields ~ 1500 - 1800u s
* time taken with no drivers on notification list .
2011-03-30 22:57:33 -03:00
* Setting the latency to 2000 us to accommodate addition of drivers
2009-09-22 21:14:00 +05:30
* to pre / post change notification list .
*/
2019-07-16 09:36:08 +05:30
cpufreq_generic_init ( policy , freq_table , 2000 * 1000 ) ;
return 0 ;
2009-09-22 21:14:00 +05:30
}
static struct cpufreq_driver davinci_driver = {
2013-12-03 11:20:45 +05:30
. flags = CPUFREQ_STICKY | CPUFREQ_NEED_INITIAL_FREQ_CHECK ,
2016-06-03 10:58:52 +05:30
. verify = cpufreq_generic_frequency_table_verify ,
2013-10-25 19:45:48 +05:30
. target_index = davinci_target ,
2014-01-09 20:38:43 +05:30
. get = cpufreq_generic_get ,
2009-09-22 21:14:00 +05:30
. init = davinci_cpu_init ,
. name = " davinci " ,
2013-10-03 20:28:02 +05:30
. attr = cpufreq_generic_attr ,
2009-09-22 21:14:00 +05:30
} ;
static int __init davinci_cpufreq_probe ( struct platform_device * pdev )
{
struct davinci_cpufreq_config * pdata = pdev - > dev . platform_data ;
2010-07-20 16:46:50 +05:30
struct clk * asyncclk ;
2009-09-22 21:14:00 +05:30
if ( ! pdata )
return - EINVAL ;
if ( ! pdata - > freq_table )
return - EINVAL ;
cpufreq . dev = & pdev - > dev ;
cpufreq . armclk = clk_get ( NULL , " arm " ) ;
if ( IS_ERR ( cpufreq . armclk ) ) {
dev_err ( cpufreq . dev , " Unable to get ARM clock \n " ) ;
return PTR_ERR ( cpufreq . armclk ) ;
}
2010-07-20 16:46:50 +05:30
asyncclk = clk_get ( cpufreq . dev , " async " ) ;
if ( ! IS_ERR ( asyncclk ) ) {
cpufreq . asyncclk = asyncclk ;
cpufreq . asyncrate = clk_get_rate ( asyncclk ) ;
}
2009-09-22 21:14:00 +05:30
return cpufreq_register_driver ( & davinci_driver ) ;
}
static int __exit davinci_cpufreq_remove ( struct platform_device * pdev )
{
clk_put ( cpufreq . armclk ) ;
2010-07-20 16:46:50 +05:30
if ( cpufreq . asyncclk )
clk_put ( cpufreq . asyncclk ) ;
2009-09-22 21:14:00 +05:30
return cpufreq_unregister_driver ( & davinci_driver ) ;
}
static struct platform_driver davinci_cpufreq_driver = {
. driver = {
. name = " cpufreq-davinci " ,
} ,
. remove = __exit_p ( davinci_cpufreq_remove ) ,
} ;
2012-04-26 09:45:39 +08:00
int __init davinci_cpufreq_init ( void )
2009-09-22 21:14:00 +05:30
{
return platform_driver_probe ( & davinci_cpufreq_driver ,
davinci_cpufreq_probe ) ;
}