2013-02-06 01:52:51 +04:00
/*
* kirkwood_freq . c : cpufreq driver for the Marvell kirkwood
*
* Copyright ( C ) 2013 Andrew Lunn < andrew @ lunn . ch >
*
* This program is free software ; you can redistribute it and / or
* modify it under the terms of the GNU General Public License
* as published by the Free Software Foundation ; either version
* 2 of the License , or ( at your option ) any later version .
*/
# include <linux/kernel.h>
# include <linux/module.h>
# include <linux/clk.h>
# include <linux/clk-provider.h>
# include <linux/cpufreq.h>
2013-06-17 18:09:51 +04:00
# include <linux/of_device.h>
2013-02-06 01:52:51 +04:00
# include <linux/platform_device.h>
# include <linux/io.h>
# include <asm/proc-fns.h>
# define CPU_SW_INT_BLK BIT(28)
static struct priv
{
struct clk * cpu_clk ;
struct clk * ddr_clk ;
struct clk * powersave_clk ;
struct device * dev ;
void __iomem * base ;
} priv ;
# define STATE_CPU_FREQ 0x01
# define STATE_DDR_FREQ 0x02
/*
* Kirkwood can swap the clock to the CPU between two clocks :
*
* - cpu clk
* - ddr clk
*
* The frequencies are set at runtime before registering this *
* table .
*/
static struct cpufreq_frequency_table kirkwood_freq_table [ ] = {
{ STATE_CPU_FREQ , 0 } , /* CPU uses cpuclk */
{ STATE_DDR_FREQ , 0 } , /* CPU uses ddrclk */
{ 0 , CPUFREQ_TABLE_END } ,
} ;
static unsigned int kirkwood_cpufreq_get_cpu_frequency ( unsigned int cpu )
{
if ( __clk_is_enabled ( priv . powersave_clk ) )
return kirkwood_freq_table [ 1 ] . frequency ;
return kirkwood_freq_table [ 0 ] . frequency ;
}
2013-10-25 18:15:48 +04:00
static int kirkwood_cpufreq_target ( struct cpufreq_policy * policy ,
unsigned int index )
2013-02-06 01:52:51 +04:00
{
2013-03-30 14:55:15 +04:00
unsigned int state = kirkwood_freq_table [ index ] . driver_data ;
2013-02-06 01:52:51 +04:00
unsigned long reg ;
2013-08-14 18:08:24 +04:00
local_irq_disable ( ) ;
2013-02-06 01:52:51 +04:00
2013-08-14 18:08:24 +04:00
/* Disable interrupts to the CPU */
reg = readl_relaxed ( priv . base ) ;
reg | = CPU_SW_INT_BLK ;
writel_relaxed ( reg , priv . base ) ;
2013-02-06 01:52:51 +04:00
2013-08-14 18:08:24 +04:00
switch ( state ) {
case STATE_CPU_FREQ :
clk_disable ( priv . powersave_clk ) ;
break ;
case STATE_DDR_FREQ :
clk_enable ( priv . powersave_clk ) ;
break ;
}
2013-02-06 01:52:51 +04:00
2013-08-14 18:08:24 +04:00
/* Wait-for-Interrupt, while the hardware changes frequency */
cpu_do_idle ( ) ;
2013-02-06 01:52:51 +04:00
2013-08-14 18:08:24 +04:00
/* Enable interrupts to the CPU */
reg = readl_relaxed ( priv . base ) ;
reg & = ~ CPU_SW_INT_BLK ;
writel_relaxed ( reg , priv . base ) ;
2013-02-06 01:52:51 +04:00
2013-08-14 18:08:24 +04:00
local_irq_enable ( ) ;
2013-02-06 01:52:51 +04:00
return 0 ;
}
/* Module init and exit code */
static int kirkwood_cpufreq_cpu_init ( struct cpufreq_policy * policy )
{
2013-10-03 18:59:15 +04:00
return cpufreq_generic_init ( policy , kirkwood_freq_table , 5000 ) ;
2013-02-06 01:52:51 +04:00
}
static struct cpufreq_driver kirkwood_cpufreq_driver = {
. get = kirkwood_cpufreq_get_cpu_frequency ,
2013-10-03 18:58:09 +04:00
. verify = cpufreq_generic_frequency_table_verify ,
2013-10-25 18:15:48 +04:00
. target_index = kirkwood_cpufreq_target ,
2013-02-06 01:52:51 +04:00
. init = kirkwood_cpufreq_cpu_init ,
2013-10-03 18:58:09 +04:00
. exit = cpufreq_generic_exit ,
2013-02-06 01:52:51 +04:00
. name = " kirkwood-cpufreq " ,
2013-10-03 18:58:09 +04:00
. attr = cpufreq_generic_attr ,
2013-02-06 01:52:51 +04:00
} ;
static int kirkwood_cpufreq_probe ( struct platform_device * pdev )
{
struct device_node * np ;
struct resource * res ;
int err ;
priv . dev = & pdev - > dev ;
res = platform_get_resource ( pdev , IORESOURCE_MEM , 0 ) ;
2013-03-22 04:10:22 +04:00
priv . base = devm_ioremap_resource ( & pdev - > dev , res ) ;
if ( IS_ERR ( priv . base ) )
return PTR_ERR ( priv . base ) ;
2013-02-06 01:52:51 +04:00
2013-06-17 18:09:51 +04:00
np = of_cpu_device_node_get ( 0 ) ;
if ( ! np ) {
dev_err ( & pdev - > dev , " failed to get cpu device node \n " ) ;
2013-02-06 01:52:51 +04:00
return - ENODEV ;
2013-06-17 18:09:51 +04:00
}
2013-02-06 01:52:51 +04:00
priv . cpu_clk = of_clk_get_by_name ( np , " cpu_clk " ) ;
if ( IS_ERR ( priv . cpu_clk ) ) {
dev_err ( priv . dev , " Unable to get cpuclk " ) ;
return PTR_ERR ( priv . cpu_clk ) ;
}
clk_prepare_enable ( priv . cpu_clk ) ;
kirkwood_freq_table [ 0 ] . frequency = clk_get_rate ( priv . cpu_clk ) / 1000 ;
priv . ddr_clk = of_clk_get_by_name ( np , " ddrclk " ) ;
if ( IS_ERR ( priv . ddr_clk ) ) {
dev_err ( priv . dev , " Unable to get ddrclk " ) ;
err = PTR_ERR ( priv . ddr_clk ) ;
goto out_cpu ;
}
clk_prepare_enable ( priv . ddr_clk ) ;
kirkwood_freq_table [ 1 ] . frequency = clk_get_rate ( priv . ddr_clk ) / 1000 ;
priv . powersave_clk = of_clk_get_by_name ( np , " powersave " ) ;
if ( IS_ERR ( priv . powersave_clk ) ) {
dev_err ( priv . dev , " Unable to get powersave " ) ;
err = PTR_ERR ( priv . powersave_clk ) ;
goto out_ddr ;
}
clk_prepare ( priv . powersave_clk ) ;
of_node_put ( np ) ;
np = NULL ;
err = cpufreq_register_driver ( & kirkwood_cpufreq_driver ) ;
if ( ! err )
return 0 ;
dev_err ( priv . dev , " Failed to register cpufreq driver " ) ;
clk_disable_unprepare ( priv . powersave_clk ) ;
out_ddr :
clk_disable_unprepare ( priv . ddr_clk ) ;
out_cpu :
clk_disable_unprepare ( priv . cpu_clk ) ;
of_node_put ( np ) ;
return err ;
}
static int kirkwood_cpufreq_remove ( struct platform_device * pdev )
{
cpufreq_unregister_driver ( & kirkwood_cpufreq_driver ) ;
clk_disable_unprepare ( priv . powersave_clk ) ;
clk_disable_unprepare ( priv . ddr_clk ) ;
clk_disable_unprepare ( priv . cpu_clk ) ;
return 0 ;
}
static struct platform_driver kirkwood_cpufreq_platform_driver = {
. probe = kirkwood_cpufreq_probe ,
. remove = kirkwood_cpufreq_remove ,
. driver = {
. name = " kirkwood-cpufreq " ,
. owner = THIS_MODULE ,
} ,
} ;
module_platform_driver ( kirkwood_cpufreq_platform_driver ) ;
MODULE_LICENSE ( " GPL v2 " ) ;
MODULE_AUTHOR ( " Andrew Lunn <andrew@lunn.ch " ) ;
MODULE_DESCRIPTION ( " cpufreq driver for Marvell's kirkwood CPU " ) ;
MODULE_ALIAS ( " platform:kirkwood-cpufreq " ) ;