2019-05-27 08:55:06 +02:00
// SPDX-License-Identifier: GPL-2.0-or-later
2014-07-03 01:59:39 +02:00
/*
* Copyright ( c ) 2014 MundoReader S . L .
* Author : Heiko Stuebner < heiko @ sntech . de >
*/
# include <linux/slab.h>
# include <linux/io.h>
# include <linux/reset-controller.h>
# include <linux/spinlock.h>
# include "clk.h"
struct rockchip_softrst {
struct reset_controller_dev rcdev ;
2022-10-18 17:14:06 +02:00
const int * lut ;
2014-07-03 01:59:39 +02:00
void __iomem * reg_base ;
int num_regs ;
int num_per_reg ;
u8 flags ;
spinlock_t lock ;
} ;
static int rockchip_softrst_assert ( struct reset_controller_dev * rcdev ,
unsigned long id )
{
struct rockchip_softrst * softrst = container_of ( rcdev ,
struct rockchip_softrst ,
rcdev ) ;
2022-10-18 17:14:06 +02:00
int bank , offset ;
if ( softrst - > lut )
id = softrst - > lut [ id ] ;
bank = id / softrst - > num_per_reg ;
offset = id % softrst - > num_per_reg ;
2014-07-03 01:59:39 +02:00
if ( softrst - > flags & ROCKCHIP_SOFTRST_HIWORD_MASK ) {
writel ( BIT ( offset ) | ( BIT ( offset ) < < 16 ) ,
softrst - > reg_base + ( bank * 4 ) ) ;
} else {
unsigned long flags ;
u32 reg ;
spin_lock_irqsave ( & softrst - > lock , flags ) ;
reg = readl ( softrst - > reg_base + ( bank * 4 ) ) ;
writel ( reg | BIT ( offset ) , softrst - > reg_base + ( bank * 4 ) ) ;
spin_unlock_irqrestore ( & softrst - > lock , flags ) ;
}
return 0 ;
}
static int rockchip_softrst_deassert ( struct reset_controller_dev * rcdev ,
unsigned long id )
{
struct rockchip_softrst * softrst = container_of ( rcdev ,
struct rockchip_softrst ,
rcdev ) ;
2022-10-18 17:14:06 +02:00
int bank , offset ;
if ( softrst - > lut )
id = softrst - > lut [ id ] ;
bank = id / softrst - > num_per_reg ;
offset = id % softrst - > num_per_reg ;
2014-07-03 01:59:39 +02:00
if ( softrst - > flags & ROCKCHIP_SOFTRST_HIWORD_MASK ) {
writel ( ( BIT ( offset ) < < 16 ) , softrst - > reg_base + ( bank * 4 ) ) ;
} else {
unsigned long flags ;
u32 reg ;
spin_lock_irqsave ( & softrst - > lock , flags ) ;
reg = readl ( softrst - > reg_base + ( bank * 4 ) ) ;
writel ( reg & ~ BIT ( offset ) , softrst - > reg_base + ( bank * 4 ) ) ;
spin_unlock_irqrestore ( & softrst - > lock , flags ) ;
}
return 0 ;
}
2016-02-25 10:45:08 +01:00
static const struct reset_control_ops rockchip_softrst_ops = {
2014-07-03 01:59:39 +02:00
. assert = rockchip_softrst_assert ,
. deassert = rockchip_softrst_deassert ,
} ;
2022-10-18 17:14:06 +02:00
void rockchip_register_softrst_lut ( struct device_node * np ,
const int * lookup_table ,
unsigned int num_regs ,
void __iomem * base , u8 flags )
2014-07-03 01:59:39 +02:00
{
struct rockchip_softrst * softrst ;
int ret ;
softrst = kzalloc ( sizeof ( * softrst ) , GFP_KERNEL ) ;
if ( ! softrst )
return ;
spin_lock_init ( & softrst - > lock ) ;
softrst - > reg_base = base ;
2022-10-18 17:14:06 +02:00
softrst - > lut = lookup_table ;
2014-07-03 01:59:39 +02:00
softrst - > flags = flags ;
softrst - > num_regs = num_regs ;
softrst - > num_per_reg = ( flags & ROCKCHIP_SOFTRST_HIWORD_MASK ) ? 16
: 32 ;
softrst - > rcdev . owner = THIS_MODULE ;
2022-10-18 17:14:06 +02:00
if ( lookup_table )
softrst - > rcdev . nr_resets = num_regs ;
else
softrst - > rcdev . nr_resets = num_regs * softrst - > num_per_reg ;
2014-07-03 01:59:39 +02:00
softrst - > rcdev . ops = & rockchip_softrst_ops ;
softrst - > rcdev . of_node = np ;
ret = reset_controller_register ( & softrst - > rcdev ) ;
if ( ret ) {
pr_err ( " %s: could not register reset controller, %d \n " ,
__func__ , ret ) ;
kfree ( softrst ) ;
}
} ;
2022-10-18 17:14:06 +02:00
EXPORT_SYMBOL_GPL ( rockchip_register_softrst_lut ) ;