2019-05-27 08:55:01 +02:00
// SPDX-License-Identifier: GPL-2.0-or-later
2013-10-11 11:57:04 +02:00
/*
* Copyright ( C ) 2013 Boris BREZILLON < b . brezillon @ overkiz . com >
*/
# include <linux/clk-provider.h>
# include <linux/clkdev.h>
# include <linux/clk/at91_pmc.h>
# include <linux/of.h>
2014-09-07 08:14:29 +02:00
# include <linux/mfd/syscon.h>
# include <linux/regmap.h>
2013-10-11 11:57:04 +02:00
# include "pmc.h"
# define PROG_ID_MAX 7
# define PROG_STATUS_MASK(id) (1 << ((id) + 8))
2019-03-18 11:50:45 +01:00
# define PROG_PRES(layout, pckr) ((pckr >> layout->pres_shift) & layout->pres_mask)
2013-10-11 11:57:04 +02:00
# define PROG_MAX_RM9200_CSS 3
struct clk_programmable {
struct clk_hw hw ;
2014-09-07 08:14:29 +02:00
struct regmap * regmap ;
2020-07-22 10:38:22 +03:00
u32 * mux_table ;
2013-10-11 11:57:04 +02:00
u8 id ;
const struct clk_programmable_layout * layout ;
2021-10-11 14:27:05 +03:00
struct at91_clk_pms pms ;
2013-10-11 11:57:04 +02:00
} ;
# define to_clk_programmable(hw) container_of(hw, struct clk_programmable, hw)
static unsigned long clk_programmable_recalc_rate ( struct clk_hw * hw ,
unsigned long parent_rate )
{
struct clk_programmable * prog = to_clk_programmable ( hw ) ;
2019-03-18 11:50:45 +01:00
const struct clk_programmable_layout * layout = prog - > layout ;
2014-09-07 08:14:29 +02:00
unsigned int pckr ;
2019-03-18 11:50:45 +01:00
unsigned long rate ;
2014-09-07 08:14:29 +02:00
regmap_read ( prog - > regmap , AT91_PMC_PCKR ( prog - > id ) , & pckr ) ;
2013-10-11 11:57:04 +02:00
2019-03-18 11:50:45 +01:00
if ( layout - > is_pres_direct )
rate = parent_rate / ( PROG_PRES ( layout , pckr ) + 1 ) ;
else
rate = parent_rate > > PROG_PRES ( layout , pckr ) ;
return rate ;
2013-10-11 11:57:04 +02:00
}
2015-07-07 20:48:08 +02:00
static int clk_programmable_determine_rate ( struct clk_hw * hw ,
struct clk_rate_request * req )
2013-10-11 11:57:04 +02:00
{
2019-03-18 11:50:45 +01:00
struct clk_programmable * prog = to_clk_programmable ( hw ) ;
const struct clk_programmable_layout * layout = prog - > layout ;
2015-07-30 17:20:57 -07:00
struct clk_hw * parent ;
2014-03-11 10:00:32 +01:00
long best_rate = - EINVAL ;
unsigned long parent_rate ;
2019-03-18 11:50:45 +01:00
unsigned long tmp_rate = 0 ;
2014-03-11 10:00:32 +01:00
int shift ;
int i ;
2015-06-25 16:53:23 -07:00
for ( i = 0 ; i < clk_hw_get_num_parents ( hw ) ; i + + ) {
2015-07-30 17:20:57 -07:00
parent = clk_hw_get_parent_by_index ( hw , i ) ;
2014-03-11 10:00:32 +01:00
if ( ! parent )
continue ;
2015-07-30 17:20:57 -07:00
parent_rate = clk_hw_get_rate ( parent ) ;
2019-03-18 11:50:45 +01:00
if ( layout - > is_pres_direct ) {
for ( shift = 0 ; shift < = layout - > pres_mask ; shift + + ) {
tmp_rate = parent_rate / ( shift + 1 ) ;
if ( tmp_rate < = req - > rate )
break ;
}
} else {
for ( shift = 0 ; shift < layout - > pres_mask ; shift + + ) {
tmp_rate = parent_rate > > shift ;
if ( tmp_rate < = req - > rate )
break ;
}
2014-03-11 10:00:32 +01:00
}
2013-10-11 11:57:04 +02:00
2015-07-07 20:48:08 +02:00
if ( tmp_rate > req - > rate )
2014-03-11 10:00:32 +01:00
continue ;
2013-10-11 11:57:04 +02:00
2015-07-07 20:48:08 +02:00
if ( best_rate < 0 | |
( req - > rate - tmp_rate ) < ( req - > rate - best_rate ) ) {
2014-03-11 10:00:32 +01:00
best_rate = tmp_rate ;
2015-07-07 20:48:08 +02:00
req - > best_parent_rate = parent_rate ;
2015-07-30 17:20:57 -07:00
req - > best_parent_hw = parent ;
2013-10-11 11:57:04 +02:00
}
2014-03-11 10:00:32 +01:00
if ( ! best_rate )
2013-10-11 11:57:04 +02:00
break ;
}
2015-07-07 20:48:08 +02:00
if ( best_rate < 0 )
return best_rate ;
req - > rate = best_rate ;
return 0 ;
2013-10-11 11:57:04 +02:00
}
static int clk_programmable_set_parent ( struct clk_hw * hw , u8 index )
{
struct clk_programmable * prog = to_clk_programmable ( hw ) ;
const struct clk_programmable_layout * layout = prog - > layout ;
2014-09-07 08:14:29 +02:00
unsigned int mask = layout - > css_mask ;
2016-07-18 09:49:12 +02:00
unsigned int pckr = index ;
2014-03-11 10:00:34 +01:00
if ( layout - > have_slck_mck )
2014-09-07 08:14:29 +02:00
mask | = AT91_PMC_CSSMCK_MCK ;
2014-03-11 10:00:34 +01:00
2020-07-22 10:38:22 +03:00
if ( prog - > mux_table )
pckr = clk_mux_index_to_val ( prog - > mux_table , 0 , index ) ;
2013-10-11 11:57:04 +02:00
if ( index > layout - > css_mask ) {
2014-09-07 08:14:29 +02:00
if ( index > PROG_MAX_RM9200_CSS & & ! layout - > have_slck_mck )
2013-10-11 11:57:04 +02:00
return - EINVAL ;
2014-09-07 08:14:29 +02:00
pckr | = AT91_PMC_CSSMCK_MCK ;
2013-10-11 11:57:04 +02:00
}
2014-09-07 08:14:29 +02:00
regmap_update_bits ( prog - > regmap , AT91_PMC_PCKR ( prog - > id ) , mask , pckr ) ;
2013-10-11 11:57:04 +02:00
return 0 ;
}
static u8 clk_programmable_get_parent ( struct clk_hw * hw )
{
struct clk_programmable * prog = to_clk_programmable ( hw ) ;
const struct clk_programmable_layout * layout = prog - > layout ;
2014-09-07 08:14:29 +02:00
unsigned int pckr ;
u8 ret ;
regmap_read ( prog - > regmap , AT91_PMC_PCKR ( prog - > id ) , & pckr ) ;
ret = pckr & layout - > css_mask ;
2013-10-11 11:57:04 +02:00
2014-09-07 08:14:29 +02:00
if ( layout - > have_slck_mck & & ( pckr & AT91_PMC_CSSMCK_MCK ) & & ! ret )
2014-03-11 10:00:34 +01:00
ret = PROG_MAX_RM9200_CSS + 1 ;
2013-10-11 11:57:04 +02:00
2020-07-22 10:38:22 +03:00
if ( prog - > mux_table )
ret = clk_mux_val_to_index ( & prog - > hw , prog - > mux_table , 0 , ret ) ;
2013-10-11 11:57:04 +02:00
return ret ;
}
static int clk_programmable_set_rate ( struct clk_hw * hw , unsigned long rate ,
unsigned long parent_rate )
{
struct clk_programmable * prog = to_clk_programmable ( hw ) ;
2014-03-11 10:00:34 +01:00
const struct clk_programmable_layout * layout = prog - > layout ;
2014-03-11 10:00:35 +01:00
unsigned long div = parent_rate / rate ;
2013-10-11 11:57:04 +02:00
int shift = 0 ;
2014-09-07 08:14:29 +02:00
2014-03-11 10:00:35 +01:00
if ( ! div )
return - EINVAL ;
2013-10-11 11:57:04 +02:00
2019-03-18 11:50:45 +01:00
if ( layout - > is_pres_direct ) {
shift = div - 1 ;
2013-10-11 11:57:04 +02:00
2019-03-18 11:50:45 +01:00
if ( shift > layout - > pres_mask )
return - EINVAL ;
} else {
shift = fls ( div ) - 1 ;
2013-10-11 11:57:04 +02:00
2019-03-18 11:50:45 +01:00
if ( div ! = ( 1 < < shift ) )
return - EINVAL ;
if ( shift > = layout - > pres_mask )
return - EINVAL ;
}
2013-10-11 11:57:04 +02:00
2014-09-07 08:14:29 +02:00
regmap_update_bits ( prog - > regmap , AT91_PMC_PCKR ( prog - > id ) ,
2019-03-18 11:50:45 +01:00
layout - > pres_mask < < layout - > pres_shift ,
2014-09-07 08:14:29 +02:00
shift < < layout - > pres_shift ) ;
2014-03-11 10:00:34 +01:00
2013-10-11 11:57:04 +02:00
return 0 ;
}
2021-10-11 14:27:05 +03:00
static int clk_programmable_save_context ( struct clk_hw * hw )
{
struct clk_programmable * prog = to_clk_programmable ( hw ) ;
struct clk_hw * parent_hw = clk_hw_get_parent ( hw ) ;
prog - > pms . parent = clk_programmable_get_parent ( hw ) ;
prog - > pms . parent_rate = clk_hw_get_rate ( parent_hw ) ;
prog - > pms . rate = clk_programmable_recalc_rate ( hw , prog - > pms . parent_rate ) ;
return 0 ;
}
static void clk_programmable_restore_context ( struct clk_hw * hw )
{
struct clk_programmable * prog = to_clk_programmable ( hw ) ;
int ret ;
ret = clk_programmable_set_parent ( hw , prog - > pms . parent ) ;
if ( ret )
return ;
clk_programmable_set_rate ( hw , prog - > pms . rate , prog - > pms . parent_rate ) ;
}
2013-10-11 11:57:04 +02:00
static const struct clk_ops programmable_ops = {
. recalc_rate = clk_programmable_recalc_rate ,
2014-03-11 10:00:32 +01:00
. determine_rate = clk_programmable_determine_rate ,
2013-10-11 11:57:04 +02:00
. get_parent = clk_programmable_get_parent ,
. set_parent = clk_programmable_set_parent ,
. set_rate = clk_programmable_set_rate ,
2021-10-11 14:27:05 +03:00
. save_context = clk_programmable_save_context ,
. restore_context = clk_programmable_restore_context ,
2013-10-11 11:57:04 +02:00
} ;
2018-10-16 16:21:44 +02:00
struct clk_hw * __init
2014-09-07 08:14:29 +02:00
at91_clk_register_programmable ( struct regmap * regmap ,
2013-10-11 11:57:04 +02:00
const char * name , const char * * parent_names ,
u8 num_parents , u8 id ,
2020-07-22 10:38:22 +03:00
const struct clk_programmable_layout * layout ,
u32 * mux_table )
2013-10-11 11:57:04 +02:00
{
struct clk_programmable * prog ;
2016-06-01 14:31:22 -07:00
struct clk_hw * hw ;
2013-10-11 11:57:04 +02:00
struct clk_init_data init ;
2016-06-01 14:31:22 -07:00
int ret ;
2013-10-11 11:57:04 +02:00
if ( id > PROG_ID_MAX )
return ERR_PTR ( - EINVAL ) ;
prog = kzalloc ( sizeof ( * prog ) , GFP_KERNEL ) ;
if ( ! prog )
return ERR_PTR ( - ENOMEM ) ;
init . name = name ;
init . ops = & programmable_ops ;
init . parent_names = parent_names ;
init . num_parents = num_parents ;
init . flags = CLK_SET_RATE_GATE | CLK_SET_PARENT_GATE ;
prog - > id = id ;
prog - > layout = layout ;
prog - > hw . init = & init ;
2014-09-07 08:14:29 +02:00
prog - > regmap = regmap ;
2020-07-22 10:38:22 +03:00
prog - > mux_table = mux_table ;
2013-10-11 11:57:04 +02:00
2016-06-01 14:31:22 -07:00
hw = & prog - > hw ;
ret = clk_hw_register ( NULL , & prog - > hw ) ;
if ( ret ) {
2013-10-11 11:57:04 +02:00
kfree ( prog ) ;
2016-09-25 13:53:58 +02:00
hw = ERR_PTR ( ret ) ;
2016-06-01 14:31:22 -07:00
}
2013-10-11 11:57:04 +02:00
2016-06-01 14:31:22 -07:00
return hw ;
2013-10-11 11:57:04 +02:00
}
2018-10-16 16:21:44 +02:00
const struct clk_programmable_layout at91rm9200_programmable_layout = {
2019-03-18 11:50:45 +01:00
. pres_mask = 0x7 ,
2013-10-11 11:57:04 +02:00
. pres_shift = 2 ,
. css_mask = 0x3 ,
. have_slck_mck = 0 ,
2019-03-18 11:50:45 +01:00
. is_pres_direct = 0 ,
2013-10-11 11:57:04 +02:00
} ;
2018-10-16 16:21:44 +02:00
const struct clk_programmable_layout at91sam9g45_programmable_layout = {
2019-03-18 11:50:45 +01:00
. pres_mask = 0x7 ,
2013-10-11 11:57:04 +02:00
. pres_shift = 2 ,
. css_mask = 0x3 ,
. have_slck_mck = 1 ,
2019-03-18 11:50:45 +01:00
. is_pres_direct = 0 ,
2013-10-11 11:57:04 +02:00
} ;
2018-10-16 16:21:44 +02:00
const struct clk_programmable_layout at91sam9x5_programmable_layout = {
2019-03-18 11:50:45 +01:00
. pres_mask = 0x7 ,
2013-10-11 11:57:04 +02:00
. pres_shift = 4 ,
. css_mask = 0x7 ,
. have_slck_mck = 0 ,
2019-03-18 11:50:45 +01:00
. is_pres_direct = 0 ,
2013-10-11 11:57:04 +02:00
} ;