2018-12-11 09:57:48 -08:00
// SPDX-License-Identifier: GPL-2.0
2012-03-15 23:11:20 -07:00
/*
* Copyright ( C ) 2011 Sascha Hauer , Pengutronix < s . hauer @ pengutronix . de >
* Copyright ( C ) 2011 Richard Zhao , Linaro < richard . zhao @ linaro . org >
* Copyright ( C ) 2011 - 2012 Mike Turquette , Linaro Ltd < mturquette @ linaro . org >
*
* Simple multiplexer clock implementation
*/
# include <linux/clk-provider.h>
2021-03-31 13:57:12 +03:00
# include <linux/device.h>
2012-03-15 23:11:20 -07:00
# include <linux/module.h>
# include <linux/slab.h>
# include <linux/io.h>
# include <linux/err.h>
/*
* DOC : basic adjustable multiplexer clock that cannot gate
*
* Traits of this clock :
* prepare - clk_prepare only ensures that parents are prepared
* enable - clk_enable only ensures that parents are enabled
* rate - rate is only affected by parent switching . No clk_set_rate support
* parent - parent is adjustable through clk_set_parent
*/
2019-04-18 13:12:08 +02:00
static inline u32 clk_mux_readl ( struct clk_mux * mux )
{
if ( mux - > flags & CLK_MUX_BIG_ENDIAN )
return ioread32be ( mux - > reg ) ;
2019-04-18 13:12:11 +02:00
return readl ( mux - > reg ) ;
2019-04-18 13:12:08 +02:00
}
static inline void clk_mux_writel ( struct clk_mux * mux , u32 val )
{
if ( mux - > flags & CLK_MUX_BIG_ENDIAN )
iowrite32be ( val , mux - > reg ) ;
else
2019-04-18 13:12:11 +02:00
writel ( val , mux - > reg ) ;
2019-04-18 13:12:08 +02:00
}
2022-02-05 11:36:09 +01:00
int clk_mux_val_to_index ( struct clk_hw * hw , const u32 * table , unsigned int flags ,
2018-02-14 14:43:34 +01:00
unsigned int val )
2012-03-15 23:11:20 -07:00
{
2015-06-25 16:53:23 -07:00
int num_parents = clk_hw_get_num_parents ( hw ) ;
2012-03-15 23:11:20 -07:00
2018-02-14 14:43:34 +01:00
if ( table ) {
2013-03-22 14:07:53 +02:00
int i ;
for ( i = 0 ; i < num_parents ; i + + )
2018-02-14 14:43:34 +01:00
if ( table [ i ] = = val )
2013-03-22 14:07:53 +02:00
return i ;
return - EINVAL ;
}
2012-03-15 23:11:20 -07:00
2018-02-14 14:43:34 +01:00
if ( val & & ( flags & CLK_MUX_INDEX_BIT ) )
2012-03-15 23:11:20 -07:00
val = ffs ( val ) - 1 ;
2018-02-14 14:43:34 +01:00
if ( val & & ( flags & CLK_MUX_INDEX_ONE ) )
2012-03-15 23:11:20 -07:00
val - - ;
2013-03-22 14:07:53 +02:00
if ( val > = num_parents )
2012-03-15 23:11:20 -07:00
return - EINVAL ;
return val ;
}
2018-02-14 14:43:34 +01:00
EXPORT_SYMBOL_GPL ( clk_mux_val_to_index ) ;
2012-03-15 23:11:20 -07:00
2022-02-05 11:36:09 +01:00
unsigned int clk_mux_index_to_val ( const u32 * table , unsigned int flags , u8 index )
2012-03-15 23:11:20 -07:00
{
2018-02-14 14:43:34 +01:00
unsigned int val = index ;
2012-03-15 23:11:20 -07:00
2018-02-14 14:43:34 +01:00
if ( table ) {
val = table [ index ] ;
2015-11-05 17:59:39 +09:00
} else {
2018-02-14 14:43:34 +01:00
if ( flags & CLK_MUX_INDEX_BIT )
val = 1 < < index ;
2013-03-22 14:07:53 +02:00
2018-02-14 14:43:34 +01:00
if ( flags & CLK_MUX_INDEX_ONE )
val + + ;
2013-03-22 14:07:53 +02:00
}
2012-03-15 23:11:20 -07:00
2018-02-14 14:43:34 +01:00
return val ;
}
EXPORT_SYMBOL_GPL ( clk_mux_index_to_val ) ;
static u8 clk_mux_get_parent ( struct clk_hw * hw )
{
struct clk_mux * mux = to_clk_mux ( hw ) ;
u32 val ;
2019-04-18 13:12:08 +02:00
val = clk_mux_readl ( mux ) > > mux - > shift ;
2018-02-14 14:43:34 +01:00
val & = mux - > mask ;
return clk_mux_val_to_index ( hw , mux - > table , mux - > flags , val ) ;
}
static int clk_mux_set_parent ( struct clk_hw * hw , u8 index )
{
struct clk_mux * mux = to_clk_mux ( hw ) ;
u32 val = clk_mux_index_to_val ( mux - > table , mux - > flags , index ) ;
unsigned long flags = 0 ;
u32 reg ;
2012-03-15 23:11:20 -07:00
if ( mux - > lock )
spin_lock_irqsave ( mux - > lock , flags ) ;
2015-07-24 12:21:12 -07:00
else
__acquire ( mux - > lock ) ;
2012-03-15 23:11:20 -07:00
2013-06-08 22:47:17 +08:00
if ( mux - > flags & CLK_MUX_HIWORD_MASK ) {
2018-02-14 14:43:34 +01:00
reg = mux - > mask < < ( mux - > shift + 16 ) ;
2013-06-08 22:47:17 +08:00
} else {
2019-04-18 13:12:08 +02:00
reg = clk_mux_readl ( mux ) ;
2018-02-14 14:43:34 +01:00
reg & = ~ ( mux - > mask < < mux - > shift ) ;
2013-06-08 22:47:17 +08:00
}
2018-02-14 14:43:34 +01:00
val = val < < mux - > shift ;
reg | = val ;
2019-04-18 13:12:08 +02:00
clk_mux_writel ( mux , reg ) ;
2012-03-15 23:11:20 -07:00
if ( mux - > lock )
spin_unlock_irqrestore ( mux - > lock , flags ) ;
2015-07-24 12:21:12 -07:00
else
__release ( mux - > lock ) ;
2012-03-15 23:11:20 -07:00
return 0 ;
}
2018-04-09 15:59:20 +02:00
static int clk_mux_determine_rate ( struct clk_hw * hw ,
struct clk_rate_request * req )
{
struct clk_mux * mux = to_clk_mux ( hw ) ;
return clk_mux_determine_rate_flags ( hw , req , mux - > flags ) ;
}
2012-03-27 15:23:22 +08:00
const struct clk_ops clk_mux_ops = {
2012-03-15 23:11:20 -07:00
. get_parent = clk_mux_get_parent ,
. set_parent = clk_mux_set_parent ,
2018-04-09 15:59:20 +02:00
. determine_rate = clk_mux_determine_rate ,
2012-03-15 23:11:20 -07:00
} ;
EXPORT_SYMBOL_GPL ( clk_mux_ops ) ;
2013-07-23 01:49:18 +02:00
const struct clk_ops clk_mux_ro_ops = {
. get_parent = clk_mux_get_parent ,
} ;
EXPORT_SYMBOL_GPL ( clk_mux_ro_ops ) ;
2019-08-30 08:09:21 -07:00
struct clk_hw * __clk_hw_register_mux ( struct device * dev , struct device_node * np ,
const char * name , u8 num_parents ,
const char * const * parent_names ,
const struct clk_hw * * parent_hws ,
const struct clk_parent_data * parent_data ,
unsigned long flags , void __iomem * reg , u8 shift , u32 mask ,
2022-02-05 11:36:09 +01:00
u8 clk_mux_flags , const u32 * table , spinlock_t * lock )
2012-03-15 23:11:20 -07:00
{
struct clk_mux * mux ;
2016-02-07 00:05:48 -08:00
struct clk_hw * hw ;
2019-11-15 21:58:55 +05:30
struct clk_init_data init = { } ;
2019-08-30 08:09:21 -07:00
int ret = - EINVAL ;
2013-06-08 22:47:17 +08:00
if ( clk_mux_flags & CLK_MUX_HIWORD_MASK ) {
2022-04-24 19:22:27 +01:00
u8 width = fls ( mask ) - ffs ( mask ) + 1 ;
2013-06-08 22:47:17 +08:00
if ( width + shift > 16 ) {
pr_err ( " mux value exceeds LOWORD field \n " ) ;
return ERR_PTR ( - EINVAL ) ;
}
}
2012-03-15 23:11:20 -07:00
2012-03-26 17:51:03 -07:00
/* allocate the mux */
2017-09-26 17:30:06 +02:00
mux = kzalloc ( sizeof ( * mux ) , GFP_KERNEL ) ;
2017-09-26 17:23:04 +02:00
if ( ! mux )
2012-03-15 23:11:20 -07:00
return ERR_PTR ( - ENOMEM ) ;
2012-04-25 22:58:56 -07:00
init . name = name ;
2013-07-23 01:49:18 +02:00
if ( clk_mux_flags & CLK_MUX_READ_ONLY )
init . ops = & clk_mux_ro_ops ;
else
init . ops = & clk_mux_ops ;
2019-04-25 10:57:37 -07:00
init . flags = flags ;
2012-04-25 22:58:56 -07:00
init . parent_names = parent_names ;
2019-08-30 08:09:21 -07:00
init . parent_data = parent_data ;
init . parent_hws = parent_hws ;
2012-04-25 22:58:56 -07:00
init . num_parents = num_parents ;
2012-03-15 23:11:20 -07:00
/* struct clk_mux assignments */
mux - > reg = reg ;
mux - > shift = shift ;
2013-03-22 14:07:53 +02:00
mux - > mask = mask ;
2012-03-15 23:11:20 -07:00
mux - > flags = clk_mux_flags ;
mux - > lock = lock ;
2013-03-22 14:07:53 +02:00
mux - > table = table ;
2012-05-06 18:48:11 -07:00
mux - > hw . init = & init ;
2012-03-15 23:11:20 -07:00
2016-02-07 00:05:48 -08:00
hw = & mux - > hw ;
2019-08-30 08:09:21 -07:00
if ( dev | | ! np )
ret = clk_hw_register ( dev , hw ) ;
else if ( np )
ret = of_clk_hw_register ( np , hw ) ;
2016-02-07 00:05:48 -08:00
if ( ret ) {
2012-03-26 17:51:03 -07:00
kfree ( mux ) ;
2016-02-07 00:05:48 -08:00
hw = ERR_PTR ( ret ) ;
}
2012-03-26 17:51:03 -07:00
2016-02-07 00:05:48 -08:00
return hw ;
}
2019-08-30 08:09:21 -07:00
EXPORT_SYMBOL_GPL ( __clk_hw_register_mux ) ;
2016-02-07 00:05:48 -08:00
2021-03-31 13:57:12 +03:00
static void devm_clk_hw_release_mux ( struct device * dev , void * res )
{
clk_hw_unregister_mux ( * ( struct clk_hw * * ) res ) ;
}
struct clk_hw * __devm_clk_hw_register_mux ( struct device * dev , struct device_node * np ,
const char * name , u8 num_parents ,
const char * const * parent_names ,
const struct clk_hw * * parent_hws ,
const struct clk_parent_data * parent_data ,
unsigned long flags , void __iomem * reg , u8 shift , u32 mask ,
2022-02-05 11:36:09 +01:00
u8 clk_mux_flags , const u32 * table , spinlock_t * lock )
2021-03-31 13:57:12 +03:00
{
struct clk_hw * * ptr , * hw ;
ptr = devres_alloc ( devm_clk_hw_release_mux , sizeof ( * ptr ) , GFP_KERNEL ) ;
if ( ! ptr )
return ERR_PTR ( - ENOMEM ) ;
hw = __clk_hw_register_mux ( dev , np , name , num_parents , parent_names , parent_hws ,
parent_data , flags , reg , shift , mask ,
clk_mux_flags , table , lock ) ;
if ( ! IS_ERR ( hw ) ) {
* ptr = hw ;
devres_add ( dev , ptr ) ;
} else {
devres_free ( ptr ) ;
}
return hw ;
}
EXPORT_SYMBOL_GPL ( __devm_clk_hw_register_mux ) ;
2016-02-07 00:05:48 -08:00
struct clk * clk_register_mux_table ( struct device * dev , const char * name ,
const char * const * parent_names , u8 num_parents ,
2019-08-30 08:09:21 -07:00
unsigned long flags , void __iomem * reg , u8 shift , u32 mask ,
2022-02-05 11:36:09 +01:00
u8 clk_mux_flags , const u32 * table , spinlock_t * lock )
2016-02-07 00:05:48 -08:00
{
struct clk_hw * hw ;
2019-08-30 08:09:21 -07:00
hw = clk_hw_register_mux_table ( dev , name , parent_names ,
num_parents , flags , reg , shift , mask ,
clk_mux_flags , table , lock ) ;
2016-02-07 00:05:48 -08:00
if ( IS_ERR ( hw ) )
return ERR_CAST ( hw ) ;
return hw - > clk ;
2012-03-15 23:11:20 -07:00
}
2013-08-15 19:06:29 -07:00
EXPORT_SYMBOL_GPL ( clk_register_mux_table ) ;
2013-03-22 14:07:53 +02:00
2015-01-05 10:52:40 +01:00
void clk_unregister_mux ( struct clk * clk )
{
struct clk_mux * mux ;
struct clk_hw * hw ;
hw = __clk_get_hw ( clk ) ;
if ( ! hw )
return ;
mux = to_clk_mux ( hw ) ;
clk_unregister ( clk ) ;
kfree ( mux ) ;
}
EXPORT_SYMBOL_GPL ( clk_unregister_mux ) ;
2016-02-07 00:05:48 -08:00
void clk_hw_unregister_mux ( struct clk_hw * hw )
{
struct clk_mux * mux ;
mux = to_clk_mux ( hw ) ;
clk_hw_unregister ( hw ) ;
kfree ( mux ) ;
}
EXPORT_SYMBOL_GPL ( clk_hw_unregister_mux ) ;