2008-10-21 14:06:54 +01:00
/* linux/arch/arm/plat-s3c24xx/clock.c
*
2009-11-13 22:54:12 +00:00
* Copyright 2004 - 2005 Simtec Electronics
2008-10-21 14:06:54 +01:00
* Ben Dooks < ben @ simtec . co . uk >
*
* S3C24XX Core clock control support
*
* Based on , and code from linux / arch / arm / mach - versatile / clock . c
* *
* * Copyright ( C ) 2004 ARM Limited .
* * Written by Deep Blue Solutions Limited .
*
*
* 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 .
*
* 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 .
*
* You should have received a copy of the GNU General Public License
* along with this program ; if not , write to the Free Software
* Foundation , Inc . , 59 Temple Place , Suite 330 , Boston , MA 02111 - 1307 USA
*/
# include <linux/init.h>
# include <linux/module.h>
# include <linux/kernel.h>
# include <linux/list.h>
# include <linux/errno.h>
# include <linux/err.h>
# include <linux/platform_device.h>
# include <linux/sysdev.h>
# include <linux/interrupt.h>
# include <linux/ioport.h>
# include <linux/clk.h>
# include <linux/spinlock.h>
# include <linux/io.h>
2011-01-12 13:40:04 +09:00
# if defined(CONFIG_DEBUG_FS)
# include <linux/debugfs.h>
# endif
2008-10-21 14:06:54 +01:00
# include <mach/hardware.h>
# include <asm/irq.h>
# include <plat/cpu-freq.h>
# include <plat/clock.h>
# include <plat/cpu.h>
2010-10-07 17:19:10 +09:00
# include <linux/serial_core.h>
# include <plat/regs-serial.h> /* for s3c24xx_uart_devs */
2008-10-21 14:06:54 +01:00
/* clock information */
static LIST_HEAD ( clocks ) ;
/* We originally used an mutex here, but some contexts (see resume)
* are calling functions such as clk_set_parent ( ) with IRQs disabled
* causing an BUG to be triggered .
*/
DEFINE_SPINLOCK ( clocks_lock ) ;
2011-09-05 20:10:06 +09:00
/* Global watchdog clock used by arch_wtd_reset() callback */
struct clk * s3c2410_wdtclk ;
static int __init s3c_wdt_reset_init ( void )
{
s3c2410_wdtclk = clk_get ( NULL , " watchdog " ) ;
if ( IS_ERR ( s3c2410_wdtclk ) )
printk ( KERN_WARNING " %s: warning: cannot get watchdog clock \n " , __func__ ) ;
return 0 ;
}
arch_initcall ( s3c_wdt_reset_init ) ;
2008-10-21 14:06:54 +01:00
/* enable and disable calls for use with the clk struct */
static int clk_null_enable ( struct clk * clk , int enable )
{
return 0 ;
}
int clk_enable ( struct clk * clk )
{
if ( IS_ERR ( clk ) | | clk = = NULL )
return - EINVAL ;
clk_enable ( clk - > parent ) ;
spin_lock ( & clocks_lock ) ;
if ( ( clk - > usage + + ) = = 0 )
( clk - > enable ) ( clk , 1 ) ;
spin_unlock ( & clocks_lock ) ;
return 0 ;
}
void clk_disable ( struct clk * clk )
{
if ( IS_ERR ( clk ) | | clk = = NULL )
return ;
spin_lock ( & clocks_lock ) ;
if ( ( - - clk - > usage ) = = 0 )
( clk - > enable ) ( clk , 0 ) ;
spin_unlock ( & clocks_lock ) ;
clk_disable ( clk - > parent ) ;
}
unsigned long clk_get_rate ( struct clk * clk )
{
if ( IS_ERR ( clk ) )
return 0 ;
if ( clk - > rate ! = 0 )
return clk - > rate ;
2009-12-01 01:24:37 +00:00
if ( clk - > ops ! = NULL & & clk - > ops - > get_rate ! = NULL )
return ( clk - > ops - > get_rate ) ( clk ) ;
2008-10-21 14:06:54 +01:00
if ( clk - > parent ! = NULL )
return clk_get_rate ( clk - > parent ) ;
return clk - > rate ;
}
long clk_round_rate ( struct clk * clk , unsigned long rate )
{
2009-12-01 01:24:37 +00:00
if ( ! IS_ERR ( clk ) & & clk - > ops & & clk - > ops - > round_rate )
return ( clk - > ops - > round_rate ) ( clk , rate ) ;
2008-10-21 14:06:54 +01:00
return rate ;
}
int clk_set_rate ( struct clk * clk , unsigned long rate )
{
int ret ;
if ( IS_ERR ( clk ) )
return - EINVAL ;
/* We do not default just do a clk->rate = rate as
* the clock may have been made this way by choice .
*/
2009-12-01 01:24:37 +00:00
WARN_ON ( clk - > ops = = NULL ) ;
WARN_ON ( clk - > ops & & clk - > ops - > set_rate = = NULL ) ;
2008-10-21 14:06:54 +01:00
2009-12-01 01:24:37 +00:00
if ( clk - > ops = = NULL | | clk - > ops - > set_rate = = NULL )
2008-10-21 14:06:54 +01:00
return - EINVAL ;
spin_lock ( & clocks_lock ) ;
2009-12-01 01:24:37 +00:00
ret = ( clk - > ops - > set_rate ) ( clk , rate ) ;
2008-10-21 14:06:54 +01:00
spin_unlock ( & clocks_lock ) ;
return ret ;
}
struct clk * clk_get_parent ( struct clk * clk )
{
return clk - > parent ;
}
int clk_set_parent ( struct clk * clk , struct clk * parent )
{
int ret = 0 ;
if ( IS_ERR ( clk ) )
return - EINVAL ;
spin_lock ( & clocks_lock ) ;
2009-12-01 01:24:37 +00:00
if ( clk - > ops & & clk - > ops - > set_parent )
ret = ( clk - > ops - > set_parent ) ( clk , parent ) ;
2008-10-21 14:06:54 +01:00
spin_unlock ( & clocks_lock ) ;
return ret ;
}
EXPORT_SYMBOL ( clk_enable ) ;
EXPORT_SYMBOL ( clk_disable ) ;
EXPORT_SYMBOL ( clk_get_rate ) ;
EXPORT_SYMBOL ( clk_round_rate ) ;
EXPORT_SYMBOL ( clk_set_rate ) ;
EXPORT_SYMBOL ( clk_get_parent ) ;
EXPORT_SYMBOL ( clk_set_parent ) ;
/* base clocks */
2010-01-14 12:50:23 +09:00
int clk_default_setrate ( struct clk * clk , unsigned long rate )
2008-10-21 14:06:54 +01:00
{
clk - > rate = rate ;
return 0 ;
}
2010-01-14 12:50:23 +09:00
struct clk_ops clk_ops_def_setrate = {
2009-12-01 01:24:37 +00:00
. set_rate = clk_default_setrate ,
} ;
2008-10-21 14:06:54 +01:00
struct clk clk_xtal = {
. name = " xtal " ,
. rate = 0 ,
. parent = NULL ,
. ctrlbit = 0 ,
} ;
2008-10-21 14:07:00 +01:00
struct clk clk_ext = {
. name = " ext " ,
} ;
struct clk clk_epll = {
. name = " epll " ,
} ;
2008-10-21 14:06:54 +01:00
struct clk clk_mpll = {
. name = " mpll " ,
2009-12-01 01:24:37 +00:00
. ops = & clk_ops_def_setrate ,
2008-10-21 14:06:54 +01:00
} ;
struct clk clk_upll = {
. name = " upll " ,
. parent = NULL ,
. ctrlbit = 0 ,
} ;
struct clk clk_f = {
. name = " fclk " ,
. rate = 0 ,
. parent = & clk_mpll ,
. ctrlbit = 0 ,
} ;
struct clk clk_h = {
. name = " hclk " ,
. rate = 0 ,
. parent = NULL ,
. ctrlbit = 0 ,
2009-12-01 01:24:37 +00:00
. ops = & clk_ops_def_setrate ,
2008-10-21 14:06:54 +01:00
} ;
struct clk clk_p = {
. name = " pclk " ,
. rate = 0 ,
. parent = NULL ,
. ctrlbit = 0 ,
2009-12-01 01:24:37 +00:00
. ops = & clk_ops_def_setrate ,
2008-10-21 14:06:54 +01:00
} ;
struct clk clk_usb_bus = {
. name = " usb-bus " ,
. rate = 0 ,
. parent = & clk_upll ,
} ;
struct clk s3c24xx_uclk = {
. name = " uclk " ,
} ;
/* initialise the clock system */
2010-01-25 10:44:10 +09:00
/**
* s3c24xx_register_clock ( ) - register a clock
* @ clk : The clock to register
*
* Add the specified clock to the list of clocks known by the system .
*/
2008-10-21 14:06:54 +01:00
int s3c24xx_register_clock ( struct clk * clk )
{
if ( clk - > enable = = NULL )
clk - > enable = clk_null_enable ;
2011-06-14 19:12:26 +09:00
/* fill up the clk_lookup structure and register it*/
clk - > lookup . dev_id = clk - > devname ;
clk - > lookup . con_id = clk - > name ;
clk - > lookup . clk = clk ;
clkdev_add ( & clk - > lookup ) ;
2008-10-21 14:06:54 +01:00
return 0 ;
}
2010-01-25 10:44:10 +09:00
/**
* s3c24xx_register_clocks ( ) - register an array of clock pointers
* @ clks : Pointer to an array of struct clk pointers
* @ nr_clks : The number of clocks in the @ clks array .
*
* Call s3c24xx_register_clock ( ) for all the clock pointers contained
* in the @ clks list . Returns the number of failures .
*/
2008-10-21 14:06:54 +01:00
int s3c24xx_register_clocks ( struct clk * * clks , int nr_clks )
{
int fails = 0 ;
for ( ; nr_clks > 0 ; nr_clks - - , clks + + ) {
2010-01-25 10:46:51 +09:00
if ( s3c24xx_register_clock ( * clks ) < 0 ) {
struct clk * clk = * clks ;
printk ( KERN_ERR " %s: failed to register %p: %s \n " ,
__func__ , clk , clk - > name ) ;
2008-10-21 14:06:54 +01:00
fails + + ;
2010-01-25 10:46:51 +09:00
}
2008-10-21 14:06:54 +01:00
}
return fails ;
}
2010-01-06 01:21:38 +09:00
/**
* s3c_register_clocks ( ) - register an array of clocks
* @ clkp : Pointer to the first clock in the array .
* @ nr_clks : Number of clocks to register .
*
* Call s3c24xx_register_clock ( ) on the @ clkp array given , printing an
* error if it fails to register the clock ( unlikely ) .
*/
2010-01-25 10:39:23 +09:00
void __init s3c_register_clocks ( struct clk * clkp , int nr_clks )
2010-01-06 01:21:38 +09:00
{
int ret ;
for ( ; nr_clks > 0 ; nr_clks - - , clkp + + ) {
ret = s3c24xx_register_clock ( clkp ) ;
if ( ret < 0 ) {
printk ( KERN_ERR " Failed to register clock %s (%d) \n " ,
clkp - > name , ret ) ;
}
}
}
2010-04-28 12:58:13 +09:00
/**
* s3c_disable_clocks ( ) - disable an array of clocks
* @ clkp : Pointer to the first clock in the array .
* @ nr_clks : Number of clocks to register .
*
* for internal use only at initialisation time . disable the clocks in the
* @ clkp array .
*/
void __init s3c_disable_clocks ( struct clk * clkp , int nr_clks )
{
for ( ; nr_clks > 0 ; nr_clks - - , clkp + + )
( clkp - > enable ) ( clkp , 0 ) ;
}
2010-06-11 12:17:00 +02:00
/* initialise all the clocks */
2008-10-21 14:06:54 +01:00
int __init s3c24xx_register_baseclocks ( unsigned long xtal )
{
2009-11-13 22:54:12 +00:00
printk ( KERN_INFO " S3C24XX Clocks, Copyright 2004 Simtec Electronics \n " ) ;
2008-10-21 14:06:54 +01:00
clk_xtal . rate = xtal ;
/* register our clocks */
if ( s3c24xx_register_clock ( & clk_xtal ) < 0 )
printk ( KERN_ERR " failed to register master xtal \n " ) ;
if ( s3c24xx_register_clock ( & clk_mpll ) < 0 )
printk ( KERN_ERR " failed to register mpll clock \n " ) ;
if ( s3c24xx_register_clock ( & clk_upll ) < 0 )
printk ( KERN_ERR " failed to register upll clock \n " ) ;
if ( s3c24xx_register_clock ( & clk_f ) < 0 )
printk ( KERN_ERR " failed to register cpu fclk \n " ) ;
if ( s3c24xx_register_clock ( & clk_h ) < 0 )
printk ( KERN_ERR " failed to register cpu hclk \n " ) ;
if ( s3c24xx_register_clock ( & clk_p ) < 0 )
printk ( KERN_ERR " failed to register cpu pclk \n " ) ;
return 0 ;
}
2011-01-12 13:40:04 +09:00
# if defined(CONFIG_PM_DEBUG) && defined(CONFIG_DEBUG_FS)
/* debugfs support to trace clock tree hierarchy and attributes */
static struct dentry * clk_debugfs_root ;
static int clk_debugfs_register_one ( struct clk * c )
{
int err ;
2011-07-16 12:37:57 -04:00
struct dentry * d ;
2011-01-12 13:40:04 +09:00
struct clk * pa = c - > parent ;
char s [ 255 ] ;
char * p = s ;
2011-06-14 19:12:26 +09:00
p + = sprintf ( p , " %s " , c - > devname ) ;
2011-01-12 13:40:04 +09:00
d = debugfs_create_dir ( s , pa ? pa - > dent : clk_debugfs_root ) ;
if ( ! d )
return - ENOMEM ;
c - > dent = d ;
d = debugfs_create_u8 ( " usecount " , S_IRUGO , c - > dent , ( u8 * ) & c - > usage ) ;
if ( ! d ) {
err = - ENOMEM ;
goto err_out ;
}
d = debugfs_create_u32 ( " rate " , S_IRUGO , c - > dent , ( u32 * ) & c - > rate ) ;
if ( ! d ) {
err = - ENOMEM ;
goto err_out ;
}
return 0 ;
err_out :
2011-07-16 12:37:57 -04:00
debugfs_remove_recursive ( c - > dent ) ;
2011-01-12 13:40:04 +09:00
return err ;
}
static int clk_debugfs_register ( struct clk * c )
{
int err ;
struct clk * pa = c - > parent ;
if ( pa & & ! pa - > dent ) {
err = clk_debugfs_register ( pa ) ;
if ( err )
return err ;
}
if ( ! c - > dent ) {
err = clk_debugfs_register_one ( c ) ;
if ( err )
return err ;
}
return 0 ;
}
static int __init clk_debugfs_init ( void )
{
struct clk * c ;
struct dentry * d ;
int err ;
d = debugfs_create_dir ( " clock " , NULL ) ;
if ( ! d )
return - ENOMEM ;
clk_debugfs_root = d ;
list_for_each_entry ( c , & clocks , list ) {
err = clk_debugfs_register ( c ) ;
if ( err )
goto err_out ;
}
return 0 ;
err_out :
debugfs_remove_recursive ( clk_debugfs_root ) ;
return err ;
}
late_initcall ( clk_debugfs_init ) ;
# endif /* defined(CONFIG_PM_DEBUG) && defined(CONFIG_DEBUG_FS) */