2019-05-27 08:55:08 +02:00
// SPDX-License-Identifier: GPL-2.0-or-later
2013-07-04 17:35:46 +08:00
/*
* Copyright ( C ) 2013 Freescale Semiconductor , Inc .
*/
# include <linux/clk-provider.h>
# include <linux/err.h>
# include <linux/io.h>
# include <linux/slab.h>
# include "clk.h"
/**
* struct clk_fixup_mux - imx integer fixup multiplexer clock
* @ mux : the parent class
* @ ops : pointer to clk_ops of parent class
* @ fixup : a hook to fixup the write value
*
* The imx fixup multiplexer clock is a subclass of basic clk_mux
* with an addtional fixup hook .
*/
struct clk_fixup_mux {
struct clk_mux mux ;
const struct clk_ops * ops ;
void ( * fixup ) ( u32 * val ) ;
} ;
static inline struct clk_fixup_mux * to_clk_fixup_mux ( struct clk_hw * hw )
{
struct clk_mux * mux = to_clk_mux ( hw ) ;
return container_of ( mux , struct clk_fixup_mux , mux ) ;
}
static u8 clk_fixup_mux_get_parent ( struct clk_hw * hw )
{
struct clk_fixup_mux * fixup_mux = to_clk_fixup_mux ( hw ) ;
return fixup_mux - > ops - > get_parent ( & fixup_mux - > mux . hw ) ;
}
static int clk_fixup_mux_set_parent ( struct clk_hw * hw , u8 index )
{
struct clk_fixup_mux * fixup_mux = to_clk_fixup_mux ( hw ) ;
struct clk_mux * mux = to_clk_mux ( hw ) ;
unsigned long flags = 0 ;
u32 val ;
spin_lock_irqsave ( mux - > lock , flags ) ;
val = readl ( mux - > reg ) ;
val & = ~ ( mux - > mask < < mux - > shift ) ;
val | = index < < mux - > shift ;
fixup_mux - > fixup ( & val ) ;
writel ( val , mux - > reg ) ;
spin_unlock_irqrestore ( mux - > lock , flags ) ;
return 0 ;
}
static const struct clk_ops clk_fixup_mux_ops = {
. get_parent = clk_fixup_mux_get_parent ,
. set_parent = clk_fixup_mux_set_parent ,
} ;
2019-05-29 12:26:44 +00:00
struct clk_hw * imx_clk_hw_fixup_mux ( const char * name , void __iomem * reg ,
2018-11-14 13:02:00 +00:00
u8 shift , u8 width , const char * const * parents ,
2013-07-04 17:35:46 +08:00
int num_parents , void ( * fixup ) ( u32 * val ) )
{
struct clk_fixup_mux * fixup_mux ;
2019-05-29 12:26:44 +00:00
struct clk_hw * hw ;
2013-07-04 17:35:46 +08:00
struct clk_init_data init ;
2019-05-29 12:26:44 +00:00
int ret ;
2013-07-04 17:35:46 +08:00
if ( ! fixup )
return ERR_PTR ( - EINVAL ) ;
fixup_mux = kzalloc ( sizeof ( * fixup_mux ) , GFP_KERNEL ) ;
if ( ! fixup_mux )
return ERR_PTR ( - ENOMEM ) ;
init . name = name ;
init . ops = & clk_fixup_mux_ops ;
init . parent_names = parents ;
init . num_parents = num_parents ;
2013-09-04 20:49:04 +08:00
init . flags = 0 ;
2013-07-04 17:35:46 +08:00
fixup_mux - > mux . reg = reg ;
fixup_mux - > mux . shift = shift ;
fixup_mux - > mux . mask = BIT ( width ) - 1 ;
fixup_mux - > mux . lock = & imx_ccm_lock ;
fixup_mux - > mux . hw . init = & init ;
fixup_mux - > ops = & clk_mux_ops ;
fixup_mux - > fixup = fixup ;
2019-05-29 12:26:44 +00:00
hw = & fixup_mux - > mux . hw ;
ret = clk_hw_register ( NULL , hw ) ;
if ( ret ) {
2013-07-04 17:35:46 +08:00
kfree ( fixup_mux ) ;
2019-05-29 12:26:44 +00:00
return ERR_PTR ( ret ) ;
}
2013-07-04 17:35:46 +08:00
2019-05-29 12:26:44 +00:00
return hw ;
2013-07-04 17:35:46 +08:00
}