2018-09-12 15:40:17 +02:00
// SPDX-License-Identifier: GPL-2.0+
2016-07-19 15:42:20 +02:00
/*
* Marvell Armada 37 xx SoC Time Base Generator clocks
*
* Copyright ( C ) 2016 Marvell
*
* Gregory CLEMENT < gregory . clement @ free - electrons . com >
*/
# include <linux/clk-provider.h>
# include <linux/clk.h>
2019-04-18 15:20:22 -07:00
# include <linux/io.h>
2016-07-19 15:42:20 +02:00
# include <linux/of.h>
# include <linux/of_address.h>
# include <linux/platform_device.h>
# include <linux/slab.h>
# define NUM_TBG 4
# define TBG_CTRL0 0x4
# define TBG_CTRL1 0x8
# define TBG_CTRL7 0x20
# define TBG_CTRL8 0x30
# define TBG_DIV_MASK 0x1FF
# define TBG_A_REFDIV 0
# define TBG_B_REFDIV 16
# define TBG_A_FBDIV 2
# define TBG_B_FBDIV 18
# define TBG_A_VCODIV_SE 0
# define TBG_B_VCODIV_SE 16
# define TBG_A_VCODIV_DIFF 1
# define TBG_B_VCODIV_DIFF 17
struct tbg_def {
char * name ;
u32 refdiv_offset ;
u32 fbdiv_offset ;
u32 vcodiv_reg ;
u32 vcodiv_offset ;
} ;
static const struct tbg_def tbg [ NUM_TBG ] = {
{ " TBG-A-P " , TBG_A_REFDIV , TBG_A_FBDIV , TBG_CTRL8 , TBG_A_VCODIV_DIFF } ,
{ " TBG-B-P " , TBG_B_REFDIV , TBG_B_FBDIV , TBG_CTRL8 , TBG_B_VCODIV_DIFF } ,
{ " TBG-A-S " , TBG_A_REFDIV , TBG_A_FBDIV , TBG_CTRL1 , TBG_A_VCODIV_SE } ,
{ " TBG-B-S " , TBG_B_REFDIV , TBG_B_FBDIV , TBG_CTRL1 , TBG_B_VCODIV_SE } ,
} ;
static unsigned int tbg_get_mult ( void __iomem * reg , const struct tbg_def * ptbg )
{
u32 val ;
val = readl ( reg + TBG_CTRL0 ) ;
return ( ( val > > ptbg - > fbdiv_offset ) & TBG_DIV_MASK ) < < 2 ;
}
static unsigned int tbg_get_div ( void __iomem * reg , const struct tbg_def * ptbg )
{
u32 val ;
unsigned int div ;
val = readl ( reg + TBG_CTRL7 ) ;
div = ( val > > ptbg - > refdiv_offset ) & TBG_DIV_MASK ;
if ( div = = 0 )
div = 1 ;
val = readl ( reg + ptbg - > vcodiv_reg ) ;
div * = 1 < < ( ( val > > ptbg - > vcodiv_offset ) & TBG_DIV_MASK ) ;
return div ;
}
static int armada_3700_tbg_clock_probe ( struct platform_device * pdev )
{
struct device_node * np = pdev - > dev . of_node ;
struct clk_hw_onecell_data * hw_tbg_data ;
struct device * dev = & pdev - > dev ;
const char * parent_name ;
struct resource * res ;
struct clk * parent ;
void __iomem * reg ;
2022-09-06 07:23:22 +00:00
int i ;
2016-07-19 15:42:20 +02:00
treewide: Use struct_size() for devm_kmalloc() and friends
Replaces open-coded struct size calculations with struct_size() for
devm_*, f2fs_*, and sock_* allocations. Automatically generated (and
manually adjusted) from the following Coccinelle script:
// Direct reference to struct field.
@@
identifier alloc =~ "devm_kmalloc|devm_kzalloc|sock_kmalloc|f2fs_kmalloc|f2fs_kzalloc";
expression HANDLE;
expression GFP;
identifier VAR, ELEMENT;
expression COUNT;
@@
- alloc(HANDLE, sizeof(*VAR) + COUNT * sizeof(*VAR->ELEMENT), GFP)
+ alloc(HANDLE, struct_size(VAR, ELEMENT, COUNT), GFP)
// mr = kzalloc(sizeof(*mr) + m * sizeof(mr->map[0]), GFP_KERNEL);
@@
identifier alloc =~ "devm_kmalloc|devm_kzalloc|sock_kmalloc|f2fs_kmalloc|f2fs_kzalloc";
expression HANDLE;
expression GFP;
identifier VAR, ELEMENT;
expression COUNT;
@@
- alloc(HANDLE, sizeof(*VAR) + COUNT * sizeof(VAR->ELEMENT[0]), GFP)
+ alloc(HANDLE, struct_size(VAR, ELEMENT, COUNT), GFP)
// Same pattern, but can't trivially locate the trailing element name,
// or variable name.
@@
identifier alloc =~ "devm_kmalloc|devm_kzalloc|sock_kmalloc|f2fs_kmalloc|f2fs_kzalloc";
expression HANDLE;
expression GFP;
expression SOMETHING, COUNT, ELEMENT;
@@
- alloc(HANDLE, sizeof(SOMETHING) + COUNT * sizeof(ELEMENT), GFP)
+ alloc(HANDLE, CHECKME_struct_size(&SOMETHING, ELEMENT, COUNT), GFP)
Signed-off-by: Kees Cook <keescook@chromium.org>
2018-05-08 16:08:53 -07:00
hw_tbg_data = devm_kzalloc ( & pdev - > dev ,
struct_size ( hw_tbg_data , hws , NUM_TBG ) ,
2016-07-19 15:42:20 +02:00
GFP_KERNEL ) ;
if ( ! hw_tbg_data )
return - ENOMEM ;
hw_tbg_data - > num = NUM_TBG ;
platform_set_drvdata ( pdev , hw_tbg_data ) ;
2018-10-10 20:18:38 +02:00
parent = clk_get ( dev , NULL ) ;
2016-07-19 15:42:20 +02:00
if ( IS_ERR ( parent ) ) {
dev_err ( dev , " Could get the clock parent \n " ) ;
return - EINVAL ;
}
parent_name = __clk_get_name ( parent ) ;
2018-10-10 20:18:38 +02:00
clk_put ( parent ) ;
2016-07-19 15:42:20 +02:00
res = platform_get_resource ( pdev , IORESOURCE_MEM , 0 ) ;
reg = devm_ioremap_resource ( dev , res ) ;
if ( IS_ERR ( reg ) )
return PTR_ERR ( reg ) ;
for ( i = 0 ; i < NUM_TBG ; i + + ) {
const char * name ;
unsigned int mult , div ;
name = tbg [ i ] . name ;
mult = tbg_get_mult ( reg , & tbg [ i ] ) ;
div = tbg_get_div ( reg , & tbg [ i ] ) ;
hw_tbg_data - > hws [ i ] = clk_hw_register_fixed_factor ( NULL , name ,
parent_name , 0 , mult , div ) ;
if ( IS_ERR ( hw_tbg_data - > hws [ i ] ) )
dev_err ( dev , " Can't register TBG clock %s \n " , name ) ;
}
2022-09-06 07:23:22 +00:00
return of_clk_add_hw_provider ( np , of_clk_hw_onecell_get , hw_tbg_data ) ;
2016-07-19 15:42:20 +02:00
}
2023-03-12 17:15:03 +01:00
static void armada_3700_tbg_clock_remove ( struct platform_device * pdev )
2016-07-19 15:42:20 +02:00
{
int i ;
struct clk_hw_onecell_data * hw_tbg_data = platform_get_drvdata ( pdev ) ;
of_clk_del_provider ( pdev - > dev . of_node ) ;
for ( i = 0 ; i < hw_tbg_data - > num ; i + + )
clk_hw_unregister_fixed_factor ( hw_tbg_data - > hws [ i ] ) ;
}
static const struct of_device_id armada_3700_tbg_clock_of_match [ ] = {
{ . compatible = " marvell,armada-3700-tbg-clock " , } ,
{ }
} ;
static struct platform_driver armada_3700_tbg_clock_driver = {
. probe = armada_3700_tbg_clock_probe ,
2023-03-12 17:15:03 +01:00
. remove_new = armada_3700_tbg_clock_remove ,
2016-07-19 15:42:20 +02:00
. driver = {
. name = " marvell-armada-3700-tbg-clock " ,
. of_match_table = armada_3700_tbg_clock_of_match ,
} ,
} ;
builtin_platform_driver ( armada_3700_tbg_clock_driver ) ;