2018-07-11 13:30:52 +09:00
// SPDX-License-Identifier: GPL-2.0
//
// Regulator controller driver for UniPhier SoC
// Copyright 2018 Socionext Inc.
// Author: Kunihiko Hayashi <hayashi.kunihiko@socionext.com>
# include <linux/clk.h>
# include <linux/io.h>
# include <linux/module.h>
# include <linux/of_device.h>
# include <linux/platform_device.h>
# include <linux/regmap.h>
# include <linux/regulator/driver.h>
# include <linux/regulator/of_regulator.h>
# include <linux/reset.h>
# define MAX_CLKS 2
# define MAX_RSTS 2
struct uniphier_regulator_soc_data {
int nclks ;
const char * const * clock_names ;
int nrsts ;
const char * const * reset_names ;
const struct regulator_desc * desc ;
const struct regmap_config * regconf ;
} ;
struct uniphier_regulator_priv {
struct clk_bulk_data clk [ MAX_CLKS ] ;
struct reset_control * rst [ MAX_RSTS ] ;
const struct uniphier_regulator_soc_data * data ;
} ;
2019-01-30 15:11:10 +08:00
static const struct regulator_ops uniphier_regulator_ops = {
2018-07-11 13:30:52 +09:00
. enable = regulator_enable_regmap ,
. disable = regulator_disable_regmap ,
. is_enabled = regulator_is_enabled_regmap ,
} ;
static int uniphier_regulator_probe ( struct platform_device * pdev )
{
struct device * dev = & pdev - > dev ;
struct uniphier_regulator_priv * priv ;
struct regulator_config config = { } ;
struct regulator_dev * rdev ;
struct regmap * regmap ;
void __iomem * base ;
const char * name ;
int i , ret , nr ;
priv = devm_kzalloc ( dev , sizeof ( * priv ) , GFP_KERNEL ) ;
if ( ! priv )
return - ENOMEM ;
priv - > data = of_device_get_match_data ( dev ) ;
if ( WARN_ON ( ! priv - > data ) )
return - EINVAL ;
2019-10-09 23:02:03 +08:00
base = devm_platform_ioremap_resource ( pdev , 0 ) ;
2018-07-11 13:30:52 +09:00
if ( IS_ERR ( base ) )
return PTR_ERR ( base ) ;
for ( i = 0 ; i < priv - > data - > nclks ; i + + )
priv - > clk [ i ] . id = priv - > data - > clock_names [ i ] ;
ret = devm_clk_bulk_get ( dev , priv - > data - > nclks , priv - > clk ) ;
if ( ret )
return ret ;
for ( i = 0 ; i < priv - > data - > nrsts ; i + + ) {
name = priv - > data - > reset_names [ i ] ;
priv - > rst [ i ] = devm_reset_control_get_shared ( dev , name ) ;
if ( IS_ERR ( priv - > rst [ i ] ) )
return PTR_ERR ( priv - > rst [ i ] ) ;
}
ret = clk_bulk_prepare_enable ( priv - > data - > nclks , priv - > clk ) ;
if ( ret )
return ret ;
for ( nr = 0 ; nr < priv - > data - > nrsts ; nr + + ) {
ret = reset_control_deassert ( priv - > rst [ nr ] ) ;
if ( ret )
goto out_rst_assert ;
}
regmap = devm_regmap_init_mmio ( dev , base , priv - > data - > regconf ) ;
2019-01-30 15:11:09 +08:00
if ( IS_ERR ( regmap ) ) {
ret = PTR_ERR ( regmap ) ;
goto out_rst_assert ;
}
2018-07-11 13:30:52 +09:00
config . dev = dev ;
config . driver_data = priv ;
config . of_node = dev - > of_node ;
config . regmap = regmap ;
config . init_data = of_get_regulator_init_data ( dev , dev - > of_node ,
priv - > data - > desc ) ;
rdev = devm_regulator_register ( dev , priv - > data - > desc , & config ) ;
if ( IS_ERR ( rdev ) ) {
ret = PTR_ERR ( rdev ) ;
goto out_rst_assert ;
}
platform_set_drvdata ( pdev , priv ) ;
return 0 ;
out_rst_assert :
while ( nr - - )
reset_control_assert ( priv - > rst [ nr ] ) ;
clk_bulk_disable_unprepare ( priv - > data - > nclks , priv - > clk ) ;
return ret ;
}
static int uniphier_regulator_remove ( struct platform_device * pdev )
{
struct uniphier_regulator_priv * priv = platform_get_drvdata ( pdev ) ;
int i ;
for ( i = 0 ; i < priv - > data - > nrsts ; i + + )
reset_control_assert ( priv - > rst [ i ] ) ;
clk_bulk_disable_unprepare ( priv - > data - > nclks , priv - > clk ) ;
return 0 ;
}
/* USB3 controller data */
# define USB3VBUS_OFFSET 0x0
# define USB3VBUS_REG BIT(4)
# define USB3VBUS_REG_EN BIT(3)
static const struct regulator_desc uniphier_usb3_regulator_desc = {
. name = " vbus " ,
. of_match = of_match_ptr ( " vbus " ) ,
. ops = & uniphier_regulator_ops ,
. type = REGULATOR_VOLTAGE ,
. owner = THIS_MODULE ,
. enable_reg = USB3VBUS_OFFSET ,
. enable_mask = USB3VBUS_REG_EN | USB3VBUS_REG ,
. enable_val = USB3VBUS_REG_EN | USB3VBUS_REG ,
. disable_val = USB3VBUS_REG_EN ,
} ;
static const struct regmap_config uniphier_usb3_regulator_regconf = {
. reg_bits = 32 ,
. val_bits = 32 ,
. reg_stride = 4 ,
. max_register = 1 ,
} ;
static const char * const uniphier_pro4_clock_reset_names [ ] = {
" gio " , " link " ,
} ;
static const struct uniphier_regulator_soc_data uniphier_pro4_usb3_data = {
. nclks = ARRAY_SIZE ( uniphier_pro4_clock_reset_names ) ,
. clock_names = uniphier_pro4_clock_reset_names ,
. nrsts = ARRAY_SIZE ( uniphier_pro4_clock_reset_names ) ,
. reset_names = uniphier_pro4_clock_reset_names ,
. desc = & uniphier_usb3_regulator_desc ,
. regconf = & uniphier_usb3_regulator_regconf ,
} ;
static const char * const uniphier_pxs2_clock_reset_names [ ] = {
" link " ,
} ;
static const struct uniphier_regulator_soc_data uniphier_pxs2_usb3_data = {
. nclks = ARRAY_SIZE ( uniphier_pxs2_clock_reset_names ) ,
. clock_names = uniphier_pxs2_clock_reset_names ,
. nrsts = ARRAY_SIZE ( uniphier_pxs2_clock_reset_names ) ,
. reset_names = uniphier_pxs2_clock_reset_names ,
. desc = & uniphier_usb3_regulator_desc ,
. regconf = & uniphier_usb3_regulator_regconf ,
} ;
static const struct of_device_id uniphier_regulator_match [ ] = {
/* USB VBUS */
{
. compatible = " socionext,uniphier-pro4-usb3-regulator " ,
. data = & uniphier_pro4_usb3_data ,
} ,
2019-09-10 10:51:44 +09:00
{
. compatible = " socionext,uniphier-pro5-usb3-regulator " ,
. data = & uniphier_pro4_usb3_data ,
} ,
2018-07-11 13:30:52 +09:00
{
. compatible = " socionext,uniphier-pxs2-usb3-regulator " ,
. data = & uniphier_pxs2_usb3_data ,
} ,
{
. compatible = " socionext,uniphier-ld20-usb3-regulator " ,
. data = & uniphier_pxs2_usb3_data ,
} ,
{
. compatible = " socionext,uniphier-pxs3-usb3-regulator " ,
. data = & uniphier_pxs2_usb3_data ,
} ,
{ /* Sentinel */ } ,
} ;
static struct platform_driver uniphier_regulator_driver = {
. probe = uniphier_regulator_probe ,
. remove = uniphier_regulator_remove ,
. driver = {
. name = " uniphier-regulator " ,
. of_match_table = uniphier_regulator_match ,
} ,
} ;
module_platform_driver ( uniphier_regulator_driver ) ;
MODULE_AUTHOR ( " Kunihiko Hayashi <hayashi.kunihiko@socionext.com> " ) ;
MODULE_DESCRIPTION ( " UniPhier Regulator Controller Driver " ) ;
MODULE_LICENSE ( " GPL " ) ;