2014-07-03 01:58:39 +02:00
/*
* Copyright ( c ) 2014 MundoReader S . L .
* Author : Heiko Stuebner < heiko @ sntech . de >
*
2016-03-09 10:37:04 +08:00
* Copyright ( c ) 2016 Rockchip Electronics Co . Ltd .
* Author : Xing Zheng < zhengxing @ rock - chips . com >
*
2014-07-03 01:58:39 +02:00
* based on
*
* samsung / clk . c
* Copyright ( c ) 2013 Samsung Electronics Co . , Ltd .
* Copyright ( c ) 2013 Linaro Ltd .
* Author : Thomas Abraham < thomas . ab @ samsung . com >
*
* This program is free software ; you can redistribute it and / or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation ; either version 2 of the License , or
* ( at your option ) any later version .
*
* This program is distributed in the hope that it will be useful ,
* but WITHOUT ANY WARRANTY ; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE . See the
* GNU General Public License for more details .
*/
# include <linux/slab.h>
# include <linux/clk.h>
# include <linux/clk-provider.h>
2014-07-03 01:59:10 +02:00
# include <linux/mfd/syscon.h>
# include <linux/regmap.h>
2014-08-19 17:45:38 -07:00
# include <linux/reboot.h>
2014-07-03 01:58:39 +02:00
# include "clk.h"
/**
* Register a clock branch .
* Most clock branches have a form like
*
* src1 - - | - - \
* | M | - - [ GATE ] - [ DIV ] -
* src2 - - | - - /
*
* sometimes without one of those components .
*/
2014-08-27 00:54:56 +02:00
static struct clk * rockchip_clk_register_branch ( const char * name ,
2016-04-19 21:29:27 +02:00
const char * const * parent_names , u8 num_parents ,
void __iomem * base ,
2014-07-03 01:58:39 +02:00
int muxdiv_offset , u8 mux_shift , u8 mux_width , u8 mux_flags ,
u8 div_shift , u8 div_width , u8 div_flags ,
struct clk_div_table * div_table , int gate_offset ,
u8 gate_shift , u8 gate_flags , unsigned long flags ,
spinlock_t * lock )
{
struct clk * clk ;
struct clk_mux * mux = NULL ;
struct clk_gate * gate = NULL ;
struct clk_divider * div = NULL ;
const struct clk_ops * mux_ops = NULL , * div_ops = NULL ,
* gate_ops = NULL ;
if ( num_parents > 1 ) {
mux = kzalloc ( sizeof ( * mux ) , GFP_KERNEL ) ;
if ( ! mux )
return ERR_PTR ( - ENOMEM ) ;
mux - > reg = base + muxdiv_offset ;
mux - > shift = mux_shift ;
mux - > mask = BIT ( mux_width ) - 1 ;
mux - > flags = mux_flags ;
mux - > lock = lock ;
mux_ops = ( mux_flags & CLK_MUX_READ_ONLY ) ? & clk_mux_ro_ops
: & clk_mux_ops ;
}
if ( gate_offset > = 0 ) {
gate = kzalloc ( sizeof ( * gate ) , GFP_KERNEL ) ;
if ( ! gate )
2016-02-02 11:37:50 +08:00
goto err_gate ;
2014-07-03 01:58:39 +02:00
gate - > flags = gate_flags ;
gate - > reg = base + gate_offset ;
gate - > bit_idx = gate_shift ;
gate - > lock = lock ;
gate_ops = & clk_gate_ops ;
}
if ( div_width > 0 ) {
div = kzalloc ( sizeof ( * div ) , GFP_KERNEL ) ;
if ( ! div )
2016-02-02 11:37:50 +08:00
goto err_div ;
2014-07-03 01:58:39 +02:00
div - > flags = div_flags ;
div - > reg = base + muxdiv_offset ;
div - > shift = div_shift ;
div - > width = div_width ;
div - > lock = lock ;
div - > table = div_table ;
2016-01-21 21:53:09 +01:00
div_ops = ( div_flags & CLK_DIVIDER_READ_ONLY )
? & clk_divider_ro_ops
: & clk_divider_ops ;
2014-07-03 01:58:39 +02:00
}
clk = clk_register_composite ( NULL , name , parent_names , num_parents ,
mux ? & mux - > hw : NULL , mux_ops ,
div ? & div - > hw : NULL , div_ops ,
gate ? & gate - > hw : NULL , gate_ops ,
flags ) ;
return clk ;
2016-02-02 11:37:50 +08:00
err_div :
kfree ( gate ) ;
err_gate :
kfree ( mux ) ;
return ERR_PTR ( - ENOMEM ) ;
2014-07-03 01:58:39 +02:00
}
2015-12-22 22:27:59 +01:00
struct rockchip_clk_frac {
struct notifier_block clk_nb ;
struct clk_fractional_divider div ;
struct clk_gate gate ;
struct clk_mux mux ;
const struct clk_ops * mux_ops ;
int mux_frac_idx ;
bool rate_change_remuxed ;
int rate_change_idx ;
} ;
# define to_rockchip_clk_frac_nb(nb) \
container_of ( nb , struct rockchip_clk_frac , clk_nb )
static int rockchip_clk_frac_notifier_cb ( struct notifier_block * nb ,
unsigned long event , void * data )
{
struct clk_notifier_data * ndata = data ;
struct rockchip_clk_frac * frac = to_rockchip_clk_frac_nb ( nb ) ;
struct clk_mux * frac_mux = & frac - > mux ;
int ret = 0 ;
pr_debug ( " %s: event %lu, old_rate %lu, new_rate: %lu \n " ,
__func__ , event , ndata - > old_rate , ndata - > new_rate ) ;
if ( event = = PRE_RATE_CHANGE ) {
2016-04-19 21:29:27 +02:00
frac - > rate_change_idx =
frac - > mux_ops - > get_parent ( & frac_mux - > hw ) ;
2015-12-22 22:27:59 +01:00
if ( frac - > rate_change_idx ! = frac - > mux_frac_idx ) {
2016-04-19 21:29:27 +02:00
frac - > mux_ops - > set_parent ( & frac_mux - > hw ,
frac - > mux_frac_idx ) ;
2015-12-22 22:27:59 +01:00
frac - > rate_change_remuxed = 1 ;
}
} else if ( event = = POST_RATE_CHANGE ) {
/*
* The POST_RATE_CHANGE notifier runs directly after the
* divider clock is set in clk_change_rate , so we ' ll have
* remuxed back to the original parent before clk_change_rate
* reaches the mux itself .
*/
if ( frac - > rate_change_remuxed ) {
2016-04-19 21:29:27 +02:00
frac - > mux_ops - > set_parent ( & frac_mux - > hw ,
frac - > rate_change_idx ) ;
2015-12-22 22:27:59 +01:00
frac - > rate_change_remuxed = 0 ;
}
}
return notifier_from_errno ( ret ) ;
}
2016-03-09 10:37:04 +08:00
static struct clk * rockchip_clk_register_frac_branch (
struct rockchip_clk_provider * ctx , const char * name ,
2015-05-28 10:45:51 +02:00
const char * const * parent_names , u8 num_parents ,
void __iomem * base , int muxdiv_offset , u8 div_flags ,
2014-08-27 00:54:21 +02:00
int gate_offset , u8 gate_shift , u8 gate_flags ,
2015-12-22 22:27:59 +01:00
unsigned long flags , struct rockchip_clk_branch * child ,
spinlock_t * lock )
2014-08-27 00:54:21 +02:00
{
2015-12-22 22:27:59 +01:00
struct rockchip_clk_frac * frac ;
2014-08-27 00:54:21 +02:00
struct clk * clk ;
struct clk_gate * gate = NULL ;
struct clk_fractional_divider * div = NULL ;
const struct clk_ops * div_ops = NULL , * gate_ops = NULL ;
2015-12-22 22:27:59 +01:00
if ( muxdiv_offset < 0 )
return ERR_PTR ( - EINVAL ) ;
if ( child & & child - > branch_type ! = branch_mux ) {
pr_err ( " %s: fractional child clock for %s can only be a mux \n " ,
__func__ , name ) ;
return ERR_PTR ( - EINVAL ) ;
}
2014-08-27 00:54:21 +02:00
2015-12-22 22:27:59 +01:00
frac = kzalloc ( sizeof ( * frac ) , GFP_KERNEL ) ;
if ( ! frac )
return ERR_PTR ( - ENOMEM ) ;
if ( gate_offset > = 0 ) {
gate = & frac - > gate ;
2014-08-27 00:54:21 +02:00
gate - > flags = gate_flags ;
gate - > reg = base + gate_offset ;
gate - > bit_idx = gate_shift ;
gate - > lock = lock ;
gate_ops = & clk_gate_ops ;
}
2015-12-22 22:27:59 +01:00
div = & frac - > div ;
2014-08-27 00:54:21 +02:00
div - > flags = div_flags ;
div - > reg = base + muxdiv_offset ;
div - > mshift = 16 ;
2015-09-22 18:54:10 +03:00
div - > mwidth = 16 ;
div - > mmask = GENMASK ( div - > mwidth - 1 , 0 ) < < div - > mshift ;
2014-08-27 00:54:21 +02:00
div - > nshift = 0 ;
2015-09-22 18:54:10 +03:00
div - > nwidth = 16 ;
div - > nmask = GENMASK ( div - > nwidth - 1 , 0 ) < < div - > nshift ;
2014-08-27 00:54:21 +02:00
div - > lock = lock ;
div_ops = & clk_fractional_divider_ops ;
clk = clk_register_composite ( NULL , name , parent_names , num_parents ,
NULL , NULL ,
& div - > hw , div_ops ,
gate ? & gate - > hw : NULL , gate_ops ,
2015-12-22 22:27:59 +01:00
flags | CLK_SET_RATE_UNGATE ) ;
if ( IS_ERR ( clk ) ) {
kfree ( frac ) ;
return clk ;
}
if ( child ) {
struct clk_mux * frac_mux = & frac - > mux ;
struct clk_init_data init ;
struct clk * mux_clk ;
int i , ret ;
frac - > mux_frac_idx = - 1 ;
for ( i = 0 ; i < child - > num_parents ; i + + ) {
if ( ! strcmp ( name , child - > parent_names [ i ] ) ) {
pr_debug ( " %s: found fractional parent in mux at pos %d \n " ,
__func__ , i ) ;
frac - > mux_frac_idx = i ;
break ;
}
}
frac - > mux_ops = & clk_mux_ops ;
frac - > clk_nb . notifier_call = rockchip_clk_frac_notifier_cb ;
frac_mux - > reg = base + child - > muxdiv_offset ;
frac_mux - > shift = child - > mux_shift ;
frac_mux - > mask = BIT ( child - > mux_width ) - 1 ;
frac_mux - > flags = child - > mux_flags ;
frac_mux - > lock = lock ;
frac_mux - > hw . init = & init ;
init . name = child - > name ;
init . flags = child - > flags | CLK_SET_RATE_PARENT ;
init . ops = frac - > mux_ops ;
init . parent_names = child - > parent_names ;
init . num_parents = child - > num_parents ;
mux_clk = clk_register ( NULL , & frac_mux - > hw ) ;
if ( IS_ERR ( mux_clk ) )
return clk ;
2016-03-09 10:37:04 +08:00
rockchip_clk_add_lookup ( ctx , mux_clk , child - > id ) ;
2015-12-22 22:27:59 +01:00
/* notifier on the fraction divider to catch rate changes */
if ( frac - > mux_frac_idx > = 0 ) {
ret = clk_notifier_register ( clk , & frac - > clk_nb ) ;
if ( ret )
pr_err ( " %s: failed to register clock notifier for %s \n " ,
__func__ , name ) ;
} else {
pr_warn ( " %s: could not find %s as parent of %s, rate changes may not work \n " ,
__func__ , name , child - > name ) ;
}
}
2014-08-27 00:54:21 +02:00
return clk ;
}
2015-06-20 13:08:57 +02:00
static struct clk * rockchip_clk_register_factor_branch ( const char * name ,
const char * const * parent_names , u8 num_parents ,
void __iomem * base , unsigned int mult , unsigned int div ,
int gate_offset , u8 gate_shift , u8 gate_flags ,
unsigned long flags , spinlock_t * lock )
{
struct clk * clk ;
struct clk_gate * gate = NULL ;
struct clk_fixed_factor * fix = NULL ;
/* without gate, register a simple factor clock */
if ( gate_offset = = 0 ) {
return clk_register_fixed_factor ( NULL , name ,
parent_names [ 0 ] , flags , mult ,
div ) ;
}
gate = kzalloc ( sizeof ( * gate ) , GFP_KERNEL ) ;
if ( ! gate )
return ERR_PTR ( - ENOMEM ) ;
gate - > flags = gate_flags ;
gate - > reg = base + gate_offset ;
gate - > bit_idx = gate_shift ;
gate - > lock = lock ;
fix = kzalloc ( sizeof ( * fix ) , GFP_KERNEL ) ;
if ( ! fix ) {
kfree ( gate ) ;
return ERR_PTR ( - ENOMEM ) ;
}
fix - > mult = mult ;
fix - > div = div ;
clk = clk_register_composite ( NULL , name , parent_names , num_parents ,
NULL , NULL ,
& fix - > hw , & clk_fixed_factor_ops ,
& gate - > hw , & clk_gate_ops , flags ) ;
if ( IS_ERR ( clk ) ) {
kfree ( fix ) ;
kfree ( gate ) ;
}
return clk ;
}
2016-03-09 10:37:04 +08:00
struct rockchip_clk_provider * __init rockchip_clk_init ( struct device_node * np ,
void __iomem * base , unsigned long nr_clks )
2014-07-03 01:58:39 +02:00
{
2016-03-09 10:37:04 +08:00
struct rockchip_clk_provider * ctx ;
struct clk * * clk_table ;
int i ;
ctx = kzalloc ( sizeof ( struct rockchip_clk_provider ) , GFP_KERNEL ) ;
2016-04-19 21:29:27 +02:00
if ( ! ctx )
2016-03-09 10:37:04 +08:00
return ERR_PTR ( - ENOMEM ) ;
2014-07-03 01:58:39 +02:00
clk_table = kcalloc ( nr_clks , sizeof ( struct clk * ) , GFP_KERNEL ) ;
2016-04-19 21:29:27 +02:00
if ( ! clk_table )
2016-03-09 10:37:04 +08:00
goto err_free ;
for ( i = 0 ; i < nr_clks ; + + i )
clk_table [ i ] = ERR_PTR ( - ENOENT ) ;
2014-07-03 01:58:39 +02:00
2016-03-09 10:37:04 +08:00
ctx - > reg_base = base ;
ctx - > clk_data . clks = clk_table ;
ctx - > clk_data . clk_num = nr_clks ;
ctx - > cru_node = np ;
ctx - > grf = ERR_PTR ( - EPROBE_DEFER ) ;
spin_lock_init ( & ctx - > lock ) ;
return ctx ;
err_free :
kfree ( ctx ) ;
return ERR_PTR ( - ENOMEM ) ;
}
void __init rockchip_clk_of_add_provider ( struct device_node * np ,
struct rockchip_clk_provider * ctx )
{
2016-03-13 00:25:53 +08:00
if ( of_clk_add_provider ( np , of_clk_src_onecell_get ,
& ctx - > clk_data ) )
pr_err ( " %s: could not register clk provider \n " , __func__ ) ;
2014-07-03 01:58:39 +02:00
}
2016-03-09 10:37:04 +08:00
struct regmap * rockchip_clk_get_grf ( struct rockchip_clk_provider * ctx )
2014-07-03 01:59:10 +02:00
{
2016-03-09 10:37:04 +08:00
if ( IS_ERR ( ctx - > grf ) )
2016-04-19 21:29:27 +02:00
ctx - > grf = syscon_regmap_lookup_by_phandle ( ctx - > cru_node ,
" rockchip,grf " ) ;
2016-03-09 10:37:04 +08:00
return ctx - > grf ;
2014-07-03 01:59:10 +02:00
}
2016-03-09 10:37:04 +08:00
void rockchip_clk_add_lookup ( struct rockchip_clk_provider * ctx ,
struct clk * clk , unsigned int id )
2014-07-03 01:58:39 +02:00
{
2016-03-09 10:37:04 +08:00
if ( ctx - > clk_data . clks & & id )
ctx - > clk_data . clks [ id ] = clk ;
2014-07-03 01:58:39 +02:00
}
2016-03-09 10:37:04 +08:00
void __init rockchip_clk_register_plls ( struct rockchip_clk_provider * ctx ,
struct rockchip_pll_clock * list ,
2014-07-03 01:59:10 +02:00
unsigned int nr_pll , int grf_lock_offset )
{
struct clk * clk ;
int idx ;
for ( idx = 0 ; idx < nr_pll ; idx + + , list + + ) {
2016-03-09 10:37:04 +08:00
clk = rockchip_clk_register_pll ( ctx , list - > type , list - > name ,
2014-07-03 01:59:10 +02:00
list - > parent_names , list - > num_parents ,
2016-03-09 10:37:04 +08:00
list - > con_offset , grf_lock_offset ,
2014-07-03 01:59:10 +02:00
list - > lock_shift , list - > mode_offset ,
2014-11-20 20:38:50 +01:00
list - > mode_shift , list - > rate_table ,
2016-03-09 10:37:04 +08:00
list - > pll_flags ) ;
2014-07-03 01:59:10 +02:00
if ( IS_ERR ( clk ) ) {
pr_err ( " %s: failed to register clock %s \n " , __func__ ,
list - > name ) ;
continue ;
}
2016-03-09 10:37:04 +08:00
rockchip_clk_add_lookup ( ctx , clk , list - > id ) ;
2014-07-03 01:59:10 +02:00
}
}
2014-07-03 01:58:39 +02:00
void __init rockchip_clk_register_branches (
2016-03-09 10:37:04 +08:00
struct rockchip_clk_provider * ctx ,
2014-07-03 01:58:39 +02:00
struct rockchip_clk_branch * list ,
unsigned int nr_clk )
{
struct clk * clk = NULL ;
unsigned int idx ;
unsigned long flags ;
for ( idx = 0 ; idx < nr_clk ; idx + + , list + + ) {
flags = list - > flags ;
/* catch simple muxes */
switch ( list - > branch_type ) {
case branch_mux :
clk = clk_register_mux ( NULL , list - > name ,
list - > parent_names , list - > num_parents ,
2016-03-09 10:37:04 +08:00
flags , ctx - > reg_base + list - > muxdiv_offset ,
2014-07-03 01:58:39 +02:00
list - > mux_shift , list - > mux_width ,
2016-03-09 10:37:04 +08:00
list - > mux_flags , & ctx - > lock ) ;
2014-07-03 01:58:39 +02:00
break ;
case branch_divider :
if ( list - > div_table )
clk = clk_register_divider_table ( NULL ,
list - > name , list - > parent_names [ 0 ] ,
2016-04-19 21:29:27 +02:00
flags ,
ctx - > reg_base + list - > muxdiv_offset ,
2014-07-03 01:58:39 +02:00
list - > div_shift , list - > div_width ,
list - > div_flags , list - > div_table ,
2016-03-09 10:37:04 +08:00
& ctx - > lock ) ;
2014-07-03 01:58:39 +02:00
else
clk = clk_register_divider ( NULL , list - > name ,
list - > parent_names [ 0 ] , flags ,
2016-03-09 10:37:04 +08:00
ctx - > reg_base + list - > muxdiv_offset ,
2014-07-03 01:58:39 +02:00
list - > div_shift , list - > div_width ,
2016-03-09 10:37:04 +08:00
list - > div_flags , & ctx - > lock ) ;
2014-07-03 01:58:39 +02:00
break ;
case branch_fraction_divider :
2016-03-09 10:37:04 +08:00
clk = rockchip_clk_register_frac_branch ( ctx , list - > name ,
2014-08-27 00:54:21 +02:00
list - > parent_names , list - > num_parents ,
2016-04-19 21:29:27 +02:00
ctx - > reg_base , list - > muxdiv_offset ,
list - > div_flags ,
2014-08-27 00:54:21 +02:00
list - > gate_offset , list - > gate_shift ,
2015-12-22 22:27:59 +01:00
list - > gate_flags , flags , list - > child ,
2016-03-09 10:37:04 +08:00
& ctx - > lock ) ;
2014-07-03 01:58:39 +02:00
break ;
case branch_gate :
flags | = CLK_SET_RATE_PARENT ;
clk = clk_register_gate ( NULL , list - > name ,
list - > parent_names [ 0 ] , flags ,
2016-03-09 10:37:04 +08:00
ctx - > reg_base + list - > gate_offset ,
list - > gate_shift , list - > gate_flags , & ctx - > lock ) ;
2014-07-03 01:58:39 +02:00
break ;
case branch_composite :
clk = rockchip_clk_register_branch ( list - > name ,
list - > parent_names , list - > num_parents ,
2016-04-19 21:29:27 +02:00
ctx - > reg_base , list - > muxdiv_offset ,
list - > mux_shift ,
2014-07-03 01:58:39 +02:00
list - > mux_width , list - > mux_flags ,
list - > div_shift , list - > div_width ,
list - > div_flags , list - > div_table ,
list - > gate_offset , list - > gate_shift ,
2016-03-09 10:37:04 +08:00
list - > gate_flags , flags , & ctx - > lock ) ;
2014-07-03 01:58:39 +02:00
break ;
2014-11-26 17:30:27 -08:00
case branch_mmc :
clk = rockchip_clk_register_mmc (
list - > name ,
list - > parent_names , list - > num_parents ,
2016-03-09 10:37:04 +08:00
ctx - > reg_base + list - > muxdiv_offset ,
2014-11-26 17:30:27 -08:00
list - > div_shift
) ;
break ;
2015-07-05 11:00:14 +02:00
case branch_inverter :
clk = rockchip_clk_register_inverter (
list - > name , list - > parent_names ,
list - > num_parents ,
2016-03-09 10:37:04 +08:00
ctx - > reg_base + list - > muxdiv_offset ,
list - > div_shift , list - > div_flags , & ctx - > lock ) ;
2015-07-05 11:00:14 +02:00
break ;
2015-06-20 13:08:57 +02:00
case branch_factor :
clk = rockchip_clk_register_factor_branch (
list - > name , list - > parent_names ,
2016-03-09 10:37:04 +08:00
list - > num_parents , ctx - > reg_base ,
2015-06-20 13:08:57 +02:00
list - > div_shift , list - > div_width ,
list - > gate_offset , list - > gate_shift ,
2016-03-09 10:37:04 +08:00
list - > gate_flags , flags , & ctx - > lock ) ;
2015-06-20 13:08:57 +02:00
break ;
2014-07-03 01:58:39 +02:00
}
/* none of the cases above matched */
if ( ! clk ) {
pr_err ( " %s: unknown clock type %d \n " ,
__func__ , list - > branch_type ) ;
continue ;
}
if ( IS_ERR ( clk ) ) {
pr_err ( " %s: failed to register clock %s: %ld \n " ,
__func__ , list - > name , PTR_ERR ( clk ) ) ;
continue ;
}
2016-03-09 10:37:04 +08:00
rockchip_clk_add_lookup ( ctx , clk , list - > id ) ;
2014-07-03 01:58:39 +02:00
}
}
2014-08-14 23:00:26 +02:00
2016-03-09 10:37:04 +08:00
void __init rockchip_clk_register_armclk ( struct rockchip_clk_provider * ctx ,
unsigned int lookup_id ,
2015-05-28 10:45:51 +02:00
const char * name , const char * const * parent_names ,
2014-09-04 22:10:43 +02:00
u8 num_parents ,
const struct rockchip_cpuclk_reg_data * reg_data ,
const struct rockchip_cpuclk_rate_table * rates ,
int nrates )
{
struct clk * clk ;
clk = rockchip_clk_register_cpuclk ( name , parent_names , num_parents ,
2016-04-19 21:29:27 +02:00
reg_data , rates , nrates ,
ctx - > reg_base , & ctx - > lock ) ;
2014-09-04 22:10:43 +02:00
if ( IS_ERR ( clk ) ) {
pr_err ( " %s: failed to register clock %s: %ld \n " ,
__func__ , name , PTR_ERR ( clk ) ) ;
return ;
}
2016-03-09 10:37:04 +08:00
rockchip_clk_add_lookup ( ctx , clk , lookup_id ) ;
2014-09-04 22:10:43 +02:00
}
2015-02-18 10:59:45 +01:00
void __init rockchip_clk_protect_critical ( const char * const clocks [ ] ,
int nclocks )
2014-08-14 23:00:26 +02:00
{
int i ;
/* Protect the clocks that needs to stay on */
for ( i = 0 ; i < nclocks ; i + + ) {
struct clk * clk = __clk_lookup ( clocks [ i ] ) ;
if ( clk )
clk_prepare_enable ( clk ) ;
}
}
2014-08-19 17:45:38 -07:00
2016-03-09 10:37:04 +08:00
static void __iomem * rst_base ;
2014-08-19 17:45:38 -07:00
static unsigned int reg_restart ;
2015-12-18 17:51:55 +01:00
static void ( * cb_restart ) ( void ) ;
2014-08-19 17:45:38 -07:00
static int rockchip_restart_notify ( struct notifier_block * this ,
unsigned long mode , void * cmd )
{
2015-12-18 17:51:55 +01:00
if ( cb_restart )
cb_restart ( ) ;
2016-03-09 10:37:04 +08:00
writel ( 0xfdb9 , rst_base + reg_restart ) ;
2014-08-19 17:45:38 -07:00
return NOTIFY_DONE ;
}
static struct notifier_block rockchip_restart_handler = {
. notifier_call = rockchip_restart_notify ,
. priority = 128 ,
} ;
2016-04-19 21:29:27 +02:00
void __init
rockchip_register_restart_notifier ( struct rockchip_clk_provider * ctx ,
unsigned int reg ,
void ( * cb ) ( void ) )
2014-08-19 17:45:38 -07:00
{
int ret ;
2016-03-09 10:37:04 +08:00
rst_base = ctx - > reg_base ;
2014-08-19 17:45:38 -07:00
reg_restart = reg ;
2015-12-18 17:51:55 +01:00
cb_restart = cb ;
2014-08-19 17:45:38 -07:00
ret = register_restart_handler ( & rockchip_restart_handler ) ;
if ( ret )
pr_err ( " %s: cannot register restart handler, %d \n " ,
__func__ , ret ) ;
}