2013-03-09 12:02:44 +04:00
/*
* Copyright ( c ) 2013 Samsung Electronics Co . , Ltd .
* Copyright ( c ) 2013 Linaro Ltd .
* Author : Thomas Abraham < thomas . ab @ samsung . com >
*
* This program is free software ; you can redistribute it and / or modify
* it under the terms of the GNU General Public License version 2 as
* published by the Free Software Foundation .
*
* Common Clock Framework support for all Samsung platforms
*/
# ifndef __SAMSUNG_CLK_H
# define __SAMSUNG_CLK_H
# include <linux/clk.h>
# include <linux/clkdev.h>
# include <linux/io.h>
# include <linux/clk-provider.h>
# include <linux/of.h>
# include <linux/of_address.h>
2013-06-11 13:31:07 +04:00
# include "clk-pll.h"
2013-03-09 12:02:44 +04:00
2013-03-18 08:43:56 +04:00
/**
* struct samsung_clock_alias : information about mux clock
* @ id : platform specific id of the clock .
* @ dev_name : name of the device to which this clock belongs .
* @ alias : optional clock alias name to be assigned to this clock .
*/
struct samsung_clock_alias {
unsigned int id ;
const char * dev_name ;
const char * alias ;
} ;
# define ALIAS(_id, dname, a) \
{ \
. id = _id , \
. dev_name = dname , \
. alias = a , \
}
2013-06-11 13:31:16 +04:00
# define MHZ (1000 * 1000)
2013-03-09 12:02:44 +04:00
/**
* struct samsung_fixed_rate_clock : information about fixed - rate clock
* @ id : platform specific id of the clock .
* @ name : name of this fixed - rate clock .
* @ parent_name : optional parent clock name .
* @ flags : optional fixed - rate clock flags .
* @ fixed - rate : fixed clock rate of this clock .
*/
struct samsung_fixed_rate_clock {
unsigned int id ;
char * name ;
const char * parent_name ;
unsigned long flags ;
unsigned long fixed_rate ;
} ;
# define FRATE(_id, cname, pname, f, frate) \
{ \
. id = _id , \
. name = cname , \
. parent_name = pname , \
. flags = f , \
. fixed_rate = frate , \
}
/*
* struct samsung_fixed_factor_clock : information about fixed - factor clock
* @ id : platform specific id of the clock .
* @ name : name of this fixed - factor clock .
* @ parent_name : parent clock name .
* @ mult : fixed multiplication factor .
* @ div : fixed division factor .
* @ flags : optional fixed - factor clock flags .
*/
struct samsung_fixed_factor_clock {
unsigned int id ;
char * name ;
const char * parent_name ;
unsigned long mult ;
unsigned long div ;
unsigned long flags ;
} ;
# define FFACTOR(_id, cname, pname, m, d, f) \
{ \
. id = _id , \
. name = cname , \
. parent_name = pname , \
. mult = m , \
. div = d , \
. flags = f , \
}
/**
* struct samsung_mux_clock : information about mux clock
* @ id : platform specific id of the clock .
* @ dev_name : name of the device to which this clock belongs .
* @ name : name of this mux clock .
* @ parent_names : array of pointer to parent clock names .
* @ num_parents : number of parents listed in @ parent_names .
* @ flags : optional flags for basic clock .
* @ offset : offset of the register for configuring the mux .
* @ shift : starting bit location of the mux control bit - field in @ reg .
* @ width : width of the mux control bit - field in @ reg .
* @ mux_flags : flags for mux - type clock .
* @ alias : optional clock alias name to be assigned to this clock .
*/
struct samsung_mux_clock {
unsigned int id ;
const char * dev_name ;
const char * name ;
const char * * parent_names ;
u8 num_parents ;
unsigned long flags ;
unsigned long offset ;
u8 shift ;
u8 width ;
u8 mux_flags ;
const char * alias ;
} ;
# define __MUX(_id, dname, cname, pnames, o, s, w, f, mf, a) \
{ \
. id = _id , \
. dev_name = dname , \
. name = cname , \
. parent_names = pnames , \
. num_parents = ARRAY_SIZE ( pnames ) , \
2013-07-29 15:25:01 +04:00
. flags = ( f ) | CLK_SET_RATE_NO_REPARENT , \
2013-03-09 12:02:44 +04:00
. offset = o , \
. shift = s , \
. width = w , \
. mux_flags = mf , \
. alias = a , \
}
# define MUX(_id, cname, pnames, o, s, w) \
__MUX ( _id , NULL , cname , pnames , o , s , w , 0 , 0 , NULL )
# define MUX_A(_id, cname, pnames, o, s, w, a) \
__MUX ( _id , NULL , cname , pnames , o , s , w , 0 , 0 , a )
# define MUX_F(_id, cname, pnames, o, s, w, f, mf) \
__MUX ( _id , NULL , cname , pnames , o , s , w , f , mf , NULL )
2013-06-20 14:47:17 +04:00
# define MUX_FA(_id, cname, pnames, o, s, w, f, mf, a) \
__MUX ( _id , NULL , cname , pnames , o , s , w , f , mf , a )
2013-03-09 12:02:44 +04:00
/**
* @ id : platform specific id of the clock .
* struct samsung_div_clock : information about div clock
* @ dev_name : name of the device to which this clock belongs .
* @ name : name of this div clock .
* @ parent_name : name of the parent clock .
* @ flags : optional flags for basic clock .
* @ offset : offset of the register for configuring the div .
* @ shift : starting bit location of the div control bit - field in @ reg .
* @ div_flags : flags for div - type clock .
* @ alias : optional clock alias name to be assigned to this clock .
*/
struct samsung_div_clock {
unsigned int id ;
const char * dev_name ;
const char * name ;
const char * parent_name ;
unsigned long flags ;
unsigned long offset ;
u8 shift ;
u8 width ;
u8 div_flags ;
const char * alias ;
2013-03-18 08:43:52 +04:00
struct clk_div_table * table ;
2013-03-09 12:02:44 +04:00
} ;
2013-03-18 08:43:52 +04:00
# define __DIV(_id, dname, cname, pname, o, s, w, f, df, a, t) \
2013-03-09 12:02:44 +04:00
{ \
. id = _id , \
. dev_name = dname , \
. name = cname , \
. parent_name = pname , \
. flags = f , \
. offset = o , \
. shift = s , \
. width = w , \
. div_flags = df , \
. alias = a , \
2013-03-18 08:43:52 +04:00
. table = t , \
2013-03-09 12:02:44 +04:00
}
# define DIV(_id, cname, pname, o, s, w) \
2013-03-18 08:43:52 +04:00
__DIV ( _id , NULL , cname , pname , o , s , w , 0 , 0 , NULL , NULL )
2013-03-09 12:02:44 +04:00
# define DIV_A(_id, cname, pname, o, s, w, a) \
2013-03-18 08:43:52 +04:00
__DIV ( _id , NULL , cname , pname , o , s , w , 0 , 0 , a , NULL )
2013-03-09 12:02:44 +04:00
# define DIV_F(_id, cname, pname, o, s, w, f, df) \
2013-03-18 08:43:52 +04:00
__DIV ( _id , NULL , cname , pname , o , s , w , f , df , NULL , NULL )
# define DIV_T(_id, cname, pname, o, s, w, t) \
__DIV ( _id , NULL , cname , pname , o , s , w , 0 , 0 , NULL , t )
2013-03-09 12:02:44 +04:00
/**
* struct samsung_gate_clock : information about gate clock
* @ id : platform specific id of the clock .
* @ dev_name : name of the device to which this clock belongs .
* @ name : name of this gate clock .
* @ parent_name : name of the parent clock .
* @ flags : optional flags for basic clock .
* @ offset : offset of the register for configuring the gate .
* @ bit_idx : bit index of the gate control bit - field in @ reg .
* @ gate_flags : flags for gate - type clock .
* @ alias : optional clock alias name to be assigned to this clock .
*/
struct samsung_gate_clock {
unsigned int id ;
const char * dev_name ;
const char * name ;
const char * parent_name ;
unsigned long flags ;
unsigned long offset ;
u8 bit_idx ;
u8 gate_flags ;
const char * alias ;
} ;
# define __GATE(_id, dname, cname, pname, o, b, f, gf, a) \
{ \
. id = _id , \
. dev_name = dname , \
. name = cname , \
. parent_name = pname , \
. flags = f , \
. offset = o , \
. bit_idx = b , \
. gate_flags = gf , \
. alias = a , \
}
# define GATE(_id, cname, pname, o, b, f, gf) \
__GATE ( _id , NULL , cname , pname , o , b , f , gf , NULL )
# define GATE_A(_id, cname, pname, o, b, f, gf, a) \
__GATE ( _id , NULL , cname , pname , o , b , f , gf , a )
# define GATE_D(_id, dname, cname, pname, o, b, f, gf) \
__GATE ( _id , dname , cname , pname , o , b , f , gf , NULL )
# define GATE_DA(_id, dname, cname, pname, o, b, f, gf, a) \
__GATE ( _id , dname , cname , pname , o , b , f , gf , a )
# define PNAME(x) static const char *x[] __initdata
/**
* struct samsung_clk_reg_dump : register dump of clock controller registers .
* @ offset : clock register offset from the controller base address .
* @ value : the value to be register at offset .
*/
struct samsung_clk_reg_dump {
u32 offset ;
u32 value ;
} ;
2013-06-11 13:31:07 +04:00
/**
* struct samsung_pll_clock : information about pll clock
* @ id : platform specific id of the clock .
* @ dev_name : name of the device to which this clock belongs .
* @ name : name of this pll clock .
* @ parent_name : name of the parent clock .
* @ flags : optional flags for basic clock .
* @ con_offset : offset of the register for configuring the PLL .
* @ lock_offset : offset of the register for locking the PLL .
* @ type : Type of PLL to be registered .
* @ alias : optional clock alias name to be assigned to this clock .
*/
struct samsung_pll_clock {
unsigned int id ;
const char * dev_name ;
const char * name ;
const char * parent_name ;
unsigned long flags ;
int con_offset ;
int lock_offset ;
enum samsung_pll_type type ;
2013-06-11 13:31:12 +04:00
const struct samsung_pll_rate_table * rate_table ;
2013-06-11 13:31:07 +04:00
const char * alias ;
} ;
2013-06-11 13:31:12 +04:00
# define __PLL(_typ, _id, _dname, _name, _pname, _flags, _lock, _con, \
_rtable , _alias ) \
2013-06-11 13:31:07 +04:00
{ \
. id = _id , \
. type = _typ , \
. dev_name = _dname , \
. name = _name , \
. parent_name = _pname , \
. flags = CLK_GET_RATE_NOCACHE , \
. con_offset = _con , \
. lock_offset = _lock , \
2013-06-11 13:31:12 +04:00
. rate_table = _rtable , \
2013-06-11 13:31:07 +04:00
. alias = _alias , \
}
2013-06-11 13:31:12 +04:00
# define PLL(_typ, _id, _name, _pname, _lock, _con, _rtable) \
2013-06-11 13:31:07 +04:00
__PLL ( _typ , _id , NULL , _name , _pname , CLK_GET_RATE_NOCACHE , \
2013-06-11 13:31:12 +04:00
_lock , _con , _rtable , _name )
2013-06-11 13:31:07 +04:00
2013-06-11 13:31:12 +04:00
# define PLL_A(_typ, _id, _name, _pname, _lock, _con, _alias, _rtable) \
2013-06-11 13:31:07 +04:00
__PLL ( _typ , _id , NULL , _name , _pname , CLK_GET_RATE_NOCACHE , \
2013-06-11 13:31:12 +04:00
_lock , _con , _rtable , _alias )
2013-06-11 13:31:07 +04:00
2013-03-09 12:02:44 +04:00
extern void __init samsung_clk_init ( struct device_node * np , void __iomem * base ,
2014-02-14 03:16:01 +04:00
unsigned long nr_clks ) ;
2013-03-09 12:02:44 +04:00
extern void __init samsung_clk_of_register_fixed_ext (
struct samsung_fixed_rate_clock * fixed_rate_clk ,
unsigned int nr_fixed_rate_clk ,
struct of_device_id * clk_matches ) ;
extern void samsung_clk_add_lookup ( struct clk * clk , unsigned int id ) ;
2013-03-18 08:43:56 +04:00
extern void samsung_clk_register_alias ( struct samsung_clock_alias * list ,
unsigned int nr_clk ) ;
2013-03-09 12:02:44 +04:00
extern void __init samsung_clk_register_fixed_rate (
struct samsung_fixed_rate_clock * clk_list , unsigned int nr_clk ) ;
extern void __init samsung_clk_register_fixed_factor (
struct samsung_fixed_factor_clock * list , unsigned int nr_clk ) ;
extern void __init samsung_clk_register_mux ( struct samsung_mux_clock * clk_list ,
unsigned int nr_clk ) ;
extern void __init samsung_clk_register_div ( struct samsung_div_clock * clk_list ,
unsigned int nr_clk ) ;
extern void __init samsung_clk_register_gate (
struct samsung_gate_clock * clk_list , unsigned int nr_clk ) ;
2013-06-11 13:31:07 +04:00
extern void __init samsung_clk_register_pll ( struct samsung_pll_clock * pll_list ,
unsigned int nr_clk , void __iomem * base ) ;
2013-03-09 12:02:44 +04:00
extern unsigned long _get_rate ( const char * clk_name ) ;
2014-02-14 03:16:00 +04:00
extern void samsung_clk_save ( void __iomem * base ,
struct samsung_clk_reg_dump * rd ,
unsigned int num_regs ) ;
extern void samsung_clk_restore ( void __iomem * base ,
const struct samsung_clk_reg_dump * rd ,
unsigned int num_regs ) ;
extern struct samsung_clk_reg_dump * samsung_clk_alloc_reg_dump (
2014-02-14 03:16:00 +04:00
const unsigned long * rdump ,
unsigned long nr_rdump ) ;
2014-02-14 03:16:00 +04:00
2013-03-09 12:02:44 +04:00
# endif /* __SAMSUNG_CLK_H */