2019-05-27 08:55:21 +02:00
// SPDX-License-Identifier: GPL-2.0-only
2015-04-23 10:35:39 +02:00
/*
* Copyright ( c ) 2014 MediaTek Inc .
* Author : James Liao < jamesjj . liao @ mediatek . com >
*/
# include <linux/of.h>
# include <linux/of_address.h>
# include <linux/io.h>
# include <linux/slab.h>
# include <linux/delay.h>
# include <linux/clkdev.h>
2021-09-02 06:25:25 +08:00
# include <linux/module.h>
2015-04-23 10:35:39 +02:00
# include "clk-mtk.h"
# include "clk-gate.h"
static int mtk_cg_bit_is_cleared ( struct clk_hw * hw )
{
2016-01-08 23:51:46 +08:00
struct mtk_clk_gate * cg = to_mtk_clk_gate ( hw ) ;
2015-04-23 10:35:39 +02:00
u32 val ;
regmap_read ( cg - > regmap , cg - > sta_ofs , & val ) ;
val & = BIT ( cg - > bit ) ;
return val = = 0 ;
}
static int mtk_cg_bit_is_set ( struct clk_hw * hw )
{
2016-01-08 23:51:46 +08:00
struct mtk_clk_gate * cg = to_mtk_clk_gate ( hw ) ;
2015-04-23 10:35:39 +02:00
u32 val ;
regmap_read ( cg - > regmap , cg - > sta_ofs , & val ) ;
val & = BIT ( cg - > bit ) ;
return val ! = 0 ;
}
static void mtk_cg_set_bit ( struct clk_hw * hw )
{
2016-01-08 23:51:46 +08:00
struct mtk_clk_gate * cg = to_mtk_clk_gate ( hw ) ;
2015-04-23 10:35:39 +02:00
regmap_write ( cg - > regmap , cg - > set_ofs , BIT ( cg - > bit ) ) ;
}
static void mtk_cg_clr_bit ( struct clk_hw * hw )
{
2016-01-08 23:51:46 +08:00
struct mtk_clk_gate * cg = to_mtk_clk_gate ( hw ) ;
2015-04-23 10:35:39 +02:00
regmap_write ( cg - > regmap , cg - > clr_ofs , BIT ( cg - > bit ) ) ;
}
2016-11-04 15:43:05 +08:00
static void mtk_cg_set_bit_no_setclr ( struct clk_hw * hw )
{
struct mtk_clk_gate * cg = to_mtk_clk_gate ( hw ) ;
u32 cgbit = BIT ( cg - > bit ) ;
regmap_update_bits ( cg - > regmap , cg - > sta_ofs , cgbit , cgbit ) ;
}
static void mtk_cg_clr_bit_no_setclr ( struct clk_hw * hw )
{
struct mtk_clk_gate * cg = to_mtk_clk_gate ( hw ) ;
u32 cgbit = BIT ( cg - > bit ) ;
regmap_update_bits ( cg - > regmap , cg - > sta_ofs , cgbit , 0 ) ;
}
2015-04-23 10:35:39 +02:00
static int mtk_cg_enable ( struct clk_hw * hw )
{
mtk_cg_clr_bit ( hw ) ;
return 0 ;
}
static void mtk_cg_disable ( struct clk_hw * hw )
{
mtk_cg_set_bit ( hw ) ;
}
static int mtk_cg_enable_inv ( struct clk_hw * hw )
{
mtk_cg_set_bit ( hw ) ;
return 0 ;
}
static void mtk_cg_disable_inv ( struct clk_hw * hw )
{
mtk_cg_clr_bit ( hw ) ;
}
2016-11-04 15:43:05 +08:00
static int mtk_cg_enable_no_setclr ( struct clk_hw * hw )
{
mtk_cg_clr_bit_no_setclr ( hw ) ;
return 0 ;
}
static void mtk_cg_disable_no_setclr ( struct clk_hw * hw )
{
mtk_cg_set_bit_no_setclr ( hw ) ;
}
static int mtk_cg_enable_inv_no_setclr ( struct clk_hw * hw )
{
mtk_cg_set_bit_no_setclr ( hw ) ;
return 0 ;
}
static void mtk_cg_disable_inv_no_setclr ( struct clk_hw * hw )
{
mtk_cg_clr_bit_no_setclr ( hw ) ;
}
2015-04-23 10:35:39 +02:00
const struct clk_ops mtk_clk_gate_ops_setclr = {
. is_enabled = mtk_cg_bit_is_cleared ,
. enable = mtk_cg_enable ,
. disable = mtk_cg_disable ,
} ;
2021-09-02 06:25:25 +08:00
EXPORT_SYMBOL_GPL ( mtk_clk_gate_ops_setclr ) ;
2015-04-23 10:35:39 +02:00
const struct clk_ops mtk_clk_gate_ops_setclr_inv = {
. is_enabled = mtk_cg_bit_is_set ,
. enable = mtk_cg_enable_inv ,
. disable = mtk_cg_disable_inv ,
} ;
2021-09-02 06:25:25 +08:00
EXPORT_SYMBOL_GPL ( mtk_clk_gate_ops_setclr_inv ) ;
2015-04-23 10:35:39 +02:00
2016-11-04 15:43:05 +08:00
const struct clk_ops mtk_clk_gate_ops_no_setclr = {
. is_enabled = mtk_cg_bit_is_cleared ,
. enable = mtk_cg_enable_no_setclr ,
. disable = mtk_cg_disable_no_setclr ,
} ;
2021-09-02 06:25:25 +08:00
EXPORT_SYMBOL_GPL ( mtk_clk_gate_ops_no_setclr ) ;
2016-11-04 15:43:05 +08:00
const struct clk_ops mtk_clk_gate_ops_no_setclr_inv = {
. is_enabled = mtk_cg_bit_is_set ,
. enable = mtk_cg_enable_inv_no_setclr ,
. disable = mtk_cg_disable_inv_no_setclr ,
} ;
2021-09-02 06:25:25 +08:00
EXPORT_SYMBOL_GPL ( mtk_clk_gate_ops_no_setclr_inv ) ;
2016-11-04 15:43:05 +08:00
2016-08-16 15:30:21 +08:00
struct clk * mtk_clk_register_gate (
2015-04-23 10:35:39 +02:00
const char * name ,
const char * parent_name ,
struct regmap * regmap ,
int set_ofs ,
int clr_ofs ,
int sta_ofs ,
u8 bit ,
2019-02-14 17:32:30 +01:00
const struct clk_ops * ops ,
2019-09-02 17:00:57 +08:00
unsigned long flags ,
struct device * dev )
2015-04-23 10:35:39 +02:00
{
struct mtk_clk_gate * cg ;
struct clk * clk ;
2015-05-18 22:00:26 +08:00
struct clk_init_data init = { } ;
2015-04-23 10:35:39 +02:00
cg = kzalloc ( sizeof ( * cg ) , GFP_KERNEL ) ;
if ( ! cg )
return ERR_PTR ( - ENOMEM ) ;
init . name = name ;
2019-04-12 11:30:27 +08:00
init . flags = flags | CLK_SET_RATE_PARENT ;
2015-04-23 10:35:39 +02:00
init . parent_names = parent_name ? & parent_name : NULL ;
init . num_parents = parent_name ? 1 : 0 ;
init . ops = ops ;
cg - > regmap = regmap ;
cg - > set_ofs = set_ofs ;
cg - > clr_ofs = clr_ofs ;
cg - > sta_ofs = sta_ofs ;
cg - > bit = bit ;
cg - > hw . init = & init ;
2019-09-02 17:00:57 +08:00
clk = clk_register ( dev , & cg - > hw ) ;
2015-04-23 10:35:39 +02:00
if ( IS_ERR ( clk ) )
kfree ( cg ) ;
return clk ;
}
2021-09-02 06:25:25 +08:00
EXPORT_SYMBOL_GPL ( mtk_clk_register_gate ) ;
MODULE_LICENSE ( " GPL " ) ;