2014-01-15 10:47:25 -08:00
/*
* Copyright ( c ) 2013 , The Linux Foundation . All rights reserved .
*
* 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/bitops.h>
# include <linux/err.h>
# include <linux/bug.h>
# include <linux/export.h>
# include <linux/clk-provider.h>
# include <linux/delay.h>
# include <linux/regmap.h>
2014-05-16 16:07:11 -07:00
# include <linux/math64.h>
2014-01-15 10:47:25 -08:00
# include <asm/div64.h>
# include "clk-rcg.h"
# define CMD_REG 0x0
# define CMD_UPDATE BIT(0)
# define CMD_ROOT_EN BIT(1)
# define CMD_DIRTY_CFG BIT(4)
# define CMD_DIRTY_N BIT(5)
# define CMD_DIRTY_M BIT(6)
# define CMD_DIRTY_D BIT(7)
# define CMD_ROOT_OFF BIT(31)
# define CFG_REG 0x4
# define CFG_SRC_DIV_SHIFT 0
# define CFG_SRC_SEL_SHIFT 8
# define CFG_SRC_SEL_MASK (0x7 << CFG_SRC_SEL_SHIFT)
# define CFG_MODE_SHIFT 12
# define CFG_MODE_MASK (0x3 << CFG_MODE_SHIFT)
# define CFG_MODE_DUAL_EDGE (0x2 << CFG_MODE_SHIFT)
# define M_REG 0x8
# define N_REG 0xc
# define D_REG 0x10
static int clk_rcg2_is_enabled ( struct clk_hw * hw )
{
struct clk_rcg2 * rcg = to_clk_rcg2 ( hw ) ;
u32 cmd ;
int ret ;
ret = regmap_read ( rcg - > clkr . regmap , rcg - > cmd_rcgr + CMD_REG , & cmd ) ;
if ( ret )
return ret ;
2014-05-16 16:07:08 -07:00
return ( cmd & CMD_ROOT_OFF ) = = 0 ;
2014-01-15 10:47:25 -08:00
}
static u8 clk_rcg2_get_parent ( struct clk_hw * hw )
{
struct clk_rcg2 * rcg = to_clk_rcg2 ( hw ) ;
int num_parents = __clk_get_num_parents ( hw - > clk ) ;
u32 cfg ;
int i , ret ;
ret = regmap_read ( rcg - > clkr . regmap , rcg - > cmd_rcgr + CFG_REG , & cfg ) ;
if ( ret )
return ret ;
cfg & = CFG_SRC_SEL_MASK ;
cfg > > = CFG_SRC_SEL_SHIFT ;
for ( i = 0 ; i < num_parents ; i + + )
if ( cfg = = rcg - > parent_map [ i ] )
return i ;
return - EINVAL ;
}
static int update_config ( struct clk_rcg2 * rcg )
{
int count , ret ;
u32 cmd ;
struct clk_hw * hw = & rcg - > clkr . hw ;
const char * name = __clk_get_name ( hw - > clk ) ;
ret = regmap_update_bits ( rcg - > clkr . regmap , rcg - > cmd_rcgr + CMD_REG ,
CMD_UPDATE , CMD_UPDATE ) ;
if ( ret )
return ret ;
/* Wait for update to take effect */
for ( count = 500 ; count > 0 ; count - - ) {
ret = regmap_read ( rcg - > clkr . regmap , rcg - > cmd_rcgr + CMD_REG , & cmd ) ;
if ( ret )
return ret ;
if ( ! ( cmd & CMD_UPDATE ) )
return 0 ;
udelay ( 1 ) ;
}
WARN ( 1 , " %s: rcg didn't update its configuration. " , name ) ;
return 0 ;
}
static int clk_rcg2_set_parent ( struct clk_hw * hw , u8 index )
{
struct clk_rcg2 * rcg = to_clk_rcg2 ( hw ) ;
int ret ;
ret = regmap_update_bits ( rcg - > clkr . regmap , rcg - > cmd_rcgr + CFG_REG ,
CFG_SRC_SEL_MASK ,
rcg - > parent_map [ index ] < < CFG_SRC_SEL_SHIFT ) ;
if ( ret )
return ret ;
return update_config ( rcg ) ;
}
/*
* Calculate m / n : d rate
*
* parent_rate m
* rate = - - - - - - - - - - - x - - -
* hid_div n
*/
static unsigned long
calc_rate ( unsigned long rate , u32 m , u32 n , u32 mode , u32 hid_div )
{
if ( hid_div ) {
rate * = 2 ;
rate / = hid_div + 1 ;
}
if ( mode ) {
u64 tmp = rate ;
tmp * = m ;
do_div ( tmp , n ) ;
rate = tmp ;
}
return rate ;
}
static unsigned long
clk_rcg2_recalc_rate ( struct clk_hw * hw , unsigned long parent_rate )
{
struct clk_rcg2 * rcg = to_clk_rcg2 ( hw ) ;
u32 cfg , hid_div , m = 0 , n = 0 , mode = 0 , mask ;
regmap_read ( rcg - > clkr . regmap , rcg - > cmd_rcgr + CFG_REG , & cfg ) ;
if ( rcg - > mnd_width ) {
mask = BIT ( rcg - > mnd_width ) - 1 ;
regmap_read ( rcg - > clkr . regmap , rcg - > cmd_rcgr + M_REG , & m ) ;
m & = mask ;
regmap_read ( rcg - > clkr . regmap , rcg - > cmd_rcgr + N_REG , & n ) ;
n = ~ n ;
n & = mask ;
n + = m ;
mode = cfg & CFG_MODE_MASK ;
mode > > = CFG_MODE_SHIFT ;
}
mask = BIT ( rcg - > hid_width ) - 1 ;
hid_div = cfg > > CFG_SRC_DIV_SHIFT ;
hid_div & = mask ;
return calc_rate ( parent_rate , m , n , mode , hid_div ) ;
}
static const
struct freq_tbl * find_freq ( const struct freq_tbl * f , unsigned long rate )
{
if ( ! f )
return NULL ;
for ( ; f - > freq ; f + + )
if ( rate < = f - > freq )
return f ;
2014-05-16 16:07:10 -07:00
/* Default to our fastest rate */
return f - 1 ;
2014-01-15 10:47:25 -08:00
}
static long _freq_tbl_determine_rate ( struct clk_hw * hw ,
const struct freq_tbl * f , unsigned long rate ,
unsigned long * p_rate , struct clk * * p )
{
unsigned long clk_flags ;
f = find_freq ( f , rate ) ;
if ( ! f )
return - EINVAL ;
clk_flags = __clk_get_flags ( hw - > clk ) ;
* p = clk_get_parent_by_index ( hw - > clk , f - > src ) ;
if ( clk_flags & CLK_SET_RATE_PARENT ) {
if ( f - > pre_div ) {
rate / = 2 ;
rate * = f - > pre_div + 1 ;
}
if ( f - > n ) {
u64 tmp = rate ;
tmp = tmp * f - > n ;
do_div ( tmp , f - > m ) ;
rate = tmp ;
}
} else {
rate = __clk_get_rate ( * p ) ;
}
* p_rate = rate ;
return f - > freq ;
}
static long clk_rcg2_determine_rate ( struct clk_hw * hw , unsigned long rate ,
unsigned long * p_rate , struct clk * * p )
{
struct clk_rcg2 * rcg = to_clk_rcg2 ( hw ) ;
return _freq_tbl_determine_rate ( hw , rcg - > freq_tbl , rate , p_rate , p ) ;
}
2014-05-16 16:07:11 -07:00
static int clk_rcg2_configure ( struct clk_rcg2 * rcg , const struct freq_tbl * f )
2014-01-15 10:47:25 -08:00
{
u32 cfg , mask ;
int ret ;
if ( rcg - > mnd_width & & f - > n ) {
mask = BIT ( rcg - > mnd_width ) - 1 ;
2014-05-16 16:07:11 -07:00
ret = regmap_update_bits ( rcg - > clkr . regmap ,
rcg - > cmd_rcgr + M_REG , mask , f - > m ) ;
2014-01-15 10:47:25 -08:00
if ( ret )
return ret ;
2014-05-16 16:07:11 -07:00
ret = regmap_update_bits ( rcg - > clkr . regmap ,
rcg - > cmd_rcgr + N_REG , mask , ~ ( f - > n - f - > m ) ) ;
2014-01-15 10:47:25 -08:00
if ( ret )
return ret ;
2014-05-16 16:07:11 -07:00
ret = regmap_update_bits ( rcg - > clkr . regmap ,
rcg - > cmd_rcgr + D_REG , mask , ~ f - > n ) ;
2014-01-15 10:47:25 -08:00
if ( ret )
return ret ;
}
mask = BIT ( rcg - > hid_width ) - 1 ;
mask | = CFG_SRC_SEL_MASK | CFG_MODE_MASK ;
cfg = f - > pre_div < < CFG_SRC_DIV_SHIFT ;
cfg | = rcg - > parent_map [ f - > src ] < < CFG_SRC_SEL_SHIFT ;
if ( rcg - > mnd_width & & f - > n )
cfg | = CFG_MODE_DUAL_EDGE ;
2014-05-16 16:07:11 -07:00
ret = regmap_update_bits ( rcg - > clkr . regmap ,
rcg - > cmd_rcgr + CFG_REG , mask , cfg ) ;
2014-01-15 10:47:25 -08:00
if ( ret )
return ret ;
return update_config ( rcg ) ;
}
2014-05-16 16:07:11 -07:00
static int __clk_rcg2_set_rate ( struct clk_hw * hw , unsigned long rate )
{
struct clk_rcg2 * rcg = to_clk_rcg2 ( hw ) ;
const struct freq_tbl * f ;
f = find_freq ( rcg - > freq_tbl , rate ) ;
if ( ! f )
return - EINVAL ;
return clk_rcg2_configure ( rcg , f ) ;
}
2014-01-15 10:47:25 -08:00
static int clk_rcg2_set_rate ( struct clk_hw * hw , unsigned long rate ,
unsigned long parent_rate )
{
return __clk_rcg2_set_rate ( hw , rate ) ;
}
static int clk_rcg2_set_rate_and_parent ( struct clk_hw * hw ,
unsigned long rate , unsigned long parent_rate , u8 index )
{
return __clk_rcg2_set_rate ( hw , rate ) ;
}
const struct clk_ops clk_rcg2_ops = {
. is_enabled = clk_rcg2_is_enabled ,
. get_parent = clk_rcg2_get_parent ,
. set_parent = clk_rcg2_set_parent ,
. recalc_rate = clk_rcg2_recalc_rate ,
. determine_rate = clk_rcg2_determine_rate ,
. set_rate = clk_rcg2_set_rate ,
. set_rate_and_parent = clk_rcg2_set_rate_and_parent ,
} ;
EXPORT_SYMBOL_GPL ( clk_rcg2_ops ) ;
2014-05-16 16:07:11 -07:00
struct frac_entry {
int num ;
int den ;
} ;
static const struct frac_entry frac_table_675m [ ] = { /* link rate of 270M */
{ 52 , 295 } , /* 119 M */
{ 11 , 57 } , /* 130.25 M */
{ 63 , 307 } , /* 138.50 M */
{ 11 , 50 } , /* 148.50 M */
{ 47 , 206 } , /* 154 M */
{ 31 , 100 } , /* 205.25 M */
{ 107 , 269 } , /* 268.50 M */
{ } ,
} ;
static struct frac_entry frac_table_810m [ ] = { /* Link rate of 162M */
{ 31 , 211 } , /* 119 M */
{ 32 , 199 } , /* 130.25 M */
{ 63 , 307 } , /* 138.50 M */
{ 11 , 60 } , /* 148.50 M */
{ 50 , 263 } , /* 154 M */
{ 31 , 120 } , /* 205.25 M */
{ 119 , 359 } , /* 268.50 M */
{ } ,
} ;
static int clk_edp_pixel_set_rate ( struct clk_hw * hw , unsigned long rate ,
unsigned long parent_rate )
{
struct clk_rcg2 * rcg = to_clk_rcg2 ( hw ) ;
struct freq_tbl f = * rcg - > freq_tbl ;
const struct frac_entry * frac ;
int delta = 100000 ;
s64 src_rate = parent_rate ;
s64 request ;
u32 mask = BIT ( rcg - > hid_width ) - 1 ;
u32 hid_div ;
if ( src_rate = = 810000000 )
frac = frac_table_810m ;
else
frac = frac_table_675m ;
for ( ; frac - > num ; frac + + ) {
request = rate ;
request * = frac - > den ;
request = div_s64 ( request , frac - > num ) ;
if ( ( src_rate < ( request - delta ) ) | |
( src_rate > ( request + delta ) ) )
continue ;
regmap_read ( rcg - > clkr . regmap , rcg - > cmd_rcgr + CFG_REG ,
& hid_div ) ;
f . pre_div = hid_div ;
f . pre_div > > = CFG_SRC_DIV_SHIFT ;
f . pre_div & = mask ;
f . m = frac - > num ;
f . n = frac - > den ;
return clk_rcg2_configure ( rcg , & f ) ;
}
return - EINVAL ;
}
static int clk_edp_pixel_set_rate_and_parent ( struct clk_hw * hw ,
unsigned long rate , unsigned long parent_rate , u8 index )
{
/* Parent index is set statically in frequency table */
return clk_edp_pixel_set_rate ( hw , rate , parent_rate ) ;
}
static long clk_edp_pixel_determine_rate ( struct clk_hw * hw , unsigned long rate ,
unsigned long * p_rate , struct clk * * p )
{
struct clk_rcg2 * rcg = to_clk_rcg2 ( hw ) ;
const struct freq_tbl * f = rcg - > freq_tbl ;
const struct frac_entry * frac ;
int delta = 100000 ;
s64 src_rate = * p_rate ;
s64 request ;
u32 mask = BIT ( rcg - > hid_width ) - 1 ;
u32 hid_div ;
/* Force the correct parent */
* p = clk_get_parent_by_index ( hw - > clk , f - > src ) ;
if ( src_rate = = 810000000 )
frac = frac_table_810m ;
else
frac = frac_table_675m ;
for ( ; frac - > num ; frac + + ) {
request = rate ;
request * = frac - > den ;
request = div_s64 ( request , frac - > num ) ;
if ( ( src_rate < ( request - delta ) ) | |
( src_rate > ( request + delta ) ) )
continue ;
regmap_read ( rcg - > clkr . regmap , rcg - > cmd_rcgr + CFG_REG ,
& hid_div ) ;
hid_div > > = CFG_SRC_DIV_SHIFT ;
hid_div & = mask ;
return calc_rate ( src_rate , frac - > num , frac - > den , ! ! frac - > den ,
hid_div ) ;
}
return - EINVAL ;
}
const struct clk_ops clk_edp_pixel_ops = {
. is_enabled = clk_rcg2_is_enabled ,
. get_parent = clk_rcg2_get_parent ,
. set_parent = clk_rcg2_set_parent ,
. recalc_rate = clk_rcg2_recalc_rate ,
. set_rate = clk_edp_pixel_set_rate ,
. set_rate_and_parent = clk_edp_pixel_set_rate_and_parent ,
. determine_rate = clk_edp_pixel_determine_rate ,
} ;
EXPORT_SYMBOL_GPL ( clk_edp_pixel_ops ) ;
static long clk_byte_determine_rate ( struct clk_hw * hw , unsigned long rate ,
unsigned long * p_rate , struct clk * * p )
{
struct clk_rcg2 * rcg = to_clk_rcg2 ( hw ) ;
const struct freq_tbl * f = rcg - > freq_tbl ;
unsigned long parent_rate , div ;
u32 mask = BIT ( rcg - > hid_width ) - 1 ;
if ( rate = = 0 )
return - EINVAL ;
* p = clk_get_parent_by_index ( hw - > clk , f - > src ) ;
* p_rate = parent_rate = __clk_round_rate ( * p , rate ) ;
div = DIV_ROUND_UP ( ( 2 * parent_rate ) , rate ) - 1 ;
div = min_t ( u32 , div , mask ) ;
return calc_rate ( parent_rate , 0 , 0 , 0 , div ) ;
}
static int clk_byte_set_rate ( struct clk_hw * hw , unsigned long rate ,
unsigned long parent_rate )
{
struct clk_rcg2 * rcg = to_clk_rcg2 ( hw ) ;
struct freq_tbl f = * rcg - > freq_tbl ;
unsigned long div ;
u32 mask = BIT ( rcg - > hid_width ) - 1 ;
div = DIV_ROUND_UP ( ( 2 * parent_rate ) , rate ) - 1 ;
div = min_t ( u32 , div , mask ) ;
f . pre_div = div ;
return clk_rcg2_configure ( rcg , & f ) ;
}
static int clk_byte_set_rate_and_parent ( struct clk_hw * hw ,
unsigned long rate , unsigned long parent_rate , u8 index )
{
/* Parent index is set statically in frequency table */
return clk_byte_set_rate ( hw , rate , parent_rate ) ;
}
const struct clk_ops clk_byte_ops = {
. is_enabled = clk_rcg2_is_enabled ,
. get_parent = clk_rcg2_get_parent ,
. set_parent = clk_rcg2_set_parent ,
. recalc_rate = clk_rcg2_recalc_rate ,
. set_rate = clk_byte_set_rate ,
. set_rate_and_parent = clk_byte_set_rate_and_parent ,
. determine_rate = clk_byte_determine_rate ,
} ;
EXPORT_SYMBOL_GPL ( clk_byte_ops ) ;
static const struct frac_entry frac_table_pixel [ ] = {
{ 3 , 8 } ,
{ 2 , 9 } ,
{ 4 , 9 } ,
{ 1 , 1 } ,
{ }
} ;
static long clk_pixel_determine_rate ( struct clk_hw * hw , unsigned long rate ,
unsigned long * p_rate , struct clk * * p )
{
struct clk_rcg2 * rcg = to_clk_rcg2 ( hw ) ;
unsigned long request , src_rate ;
int delta = 100000 ;
const struct freq_tbl * f = rcg - > freq_tbl ;
const struct frac_entry * frac = frac_table_pixel ;
struct clk * parent = * p = clk_get_parent_by_index ( hw - > clk , f - > src ) ;
for ( ; frac - > num ; frac + + ) {
request = ( rate * frac - > den ) / frac - > num ;
src_rate = __clk_round_rate ( parent , request ) ;
if ( ( src_rate < ( request - delta ) ) | |
( src_rate > ( request + delta ) ) )
continue ;
* p_rate = src_rate ;
return ( src_rate * frac - > num ) / frac - > den ;
}
return - EINVAL ;
}
static int clk_pixel_set_rate ( struct clk_hw * hw , unsigned long rate ,
unsigned long parent_rate )
{
struct clk_rcg2 * rcg = to_clk_rcg2 ( hw ) ;
struct freq_tbl f = * rcg - > freq_tbl ;
const struct frac_entry * frac = frac_table_pixel ;
unsigned long request , src_rate ;
int delta = 100000 ;
u32 mask = BIT ( rcg - > hid_width ) - 1 ;
u32 hid_div ;
struct clk * parent = clk_get_parent_by_index ( hw - > clk , f . src ) ;
for ( ; frac - > num ; frac + + ) {
request = ( rate * frac - > den ) / frac - > num ;
src_rate = __clk_round_rate ( parent , request ) ;
if ( ( src_rate < ( request - delta ) ) | |
( src_rate > ( request + delta ) ) )
continue ;
regmap_read ( rcg - > clkr . regmap , rcg - > cmd_rcgr + CFG_REG ,
& hid_div ) ;
f . pre_div = hid_div ;
f . pre_div > > = CFG_SRC_DIV_SHIFT ;
f . pre_div & = mask ;
f . m = frac - > num ;
f . n = frac - > den ;
return clk_rcg2_configure ( rcg , & f ) ;
}
return - EINVAL ;
}
static int clk_pixel_set_rate_and_parent ( struct clk_hw * hw , unsigned long rate ,
unsigned long parent_rate , u8 index )
{
/* Parent index is set statically in frequency table */
return clk_pixel_set_rate ( hw , rate , parent_rate ) ;
}
const struct clk_ops clk_pixel_ops = {
. is_enabled = clk_rcg2_is_enabled ,
. get_parent = clk_rcg2_get_parent ,
. set_parent = clk_rcg2_set_parent ,
. recalc_rate = clk_rcg2_recalc_rate ,
. set_rate = clk_pixel_set_rate ,
. set_rate_and_parent = clk_pixel_set_rate_and_parent ,
. determine_rate = clk_pixel_determine_rate ,
} ;
EXPORT_SYMBOL_GPL ( clk_pixel_ops ) ;