162306a36Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0 262306a36Sopenharmony_ci// 362306a36Sopenharmony_ci// Regulator controller driver for UniPhier SoC 462306a36Sopenharmony_ci// Copyright 2018 Socionext Inc. 562306a36Sopenharmony_ci// Author: Kunihiko Hayashi <hayashi.kunihiko@socionext.com> 662306a36Sopenharmony_ci 762306a36Sopenharmony_ci#include <linux/clk.h> 862306a36Sopenharmony_ci#include <linux/io.h> 962306a36Sopenharmony_ci#include <linux/module.h> 1062306a36Sopenharmony_ci#include <linux/of.h> 1162306a36Sopenharmony_ci#include <linux/platform_device.h> 1262306a36Sopenharmony_ci#include <linux/regmap.h> 1362306a36Sopenharmony_ci#include <linux/regulator/driver.h> 1462306a36Sopenharmony_ci#include <linux/regulator/of_regulator.h> 1562306a36Sopenharmony_ci#include <linux/reset.h> 1662306a36Sopenharmony_ci 1762306a36Sopenharmony_ci#define MAX_CLKS 2 1862306a36Sopenharmony_ci#define MAX_RSTS 2 1962306a36Sopenharmony_ci 2062306a36Sopenharmony_cistruct uniphier_regulator_soc_data { 2162306a36Sopenharmony_ci int nclks; 2262306a36Sopenharmony_ci const char * const *clock_names; 2362306a36Sopenharmony_ci int nrsts; 2462306a36Sopenharmony_ci const char * const *reset_names; 2562306a36Sopenharmony_ci const struct regulator_desc *desc; 2662306a36Sopenharmony_ci const struct regmap_config *regconf; 2762306a36Sopenharmony_ci}; 2862306a36Sopenharmony_ci 2962306a36Sopenharmony_cistruct uniphier_regulator_priv { 3062306a36Sopenharmony_ci struct clk_bulk_data clk[MAX_CLKS]; 3162306a36Sopenharmony_ci struct reset_control *rst[MAX_RSTS]; 3262306a36Sopenharmony_ci const struct uniphier_regulator_soc_data *data; 3362306a36Sopenharmony_ci}; 3462306a36Sopenharmony_ci 3562306a36Sopenharmony_cistatic const struct regulator_ops uniphier_regulator_ops = { 3662306a36Sopenharmony_ci .enable = regulator_enable_regmap, 3762306a36Sopenharmony_ci .disable = regulator_disable_regmap, 3862306a36Sopenharmony_ci .is_enabled = regulator_is_enabled_regmap, 3962306a36Sopenharmony_ci}; 4062306a36Sopenharmony_ci 4162306a36Sopenharmony_cistatic int uniphier_regulator_probe(struct platform_device *pdev) 4262306a36Sopenharmony_ci{ 4362306a36Sopenharmony_ci struct device *dev = &pdev->dev; 4462306a36Sopenharmony_ci struct uniphier_regulator_priv *priv; 4562306a36Sopenharmony_ci struct regulator_config config = { }; 4662306a36Sopenharmony_ci struct regulator_dev *rdev; 4762306a36Sopenharmony_ci struct regmap *regmap; 4862306a36Sopenharmony_ci void __iomem *base; 4962306a36Sopenharmony_ci const char *name; 5062306a36Sopenharmony_ci int i, ret, nr; 5162306a36Sopenharmony_ci 5262306a36Sopenharmony_ci priv = devm_kzalloc(dev, sizeof(*priv), GFP_KERNEL); 5362306a36Sopenharmony_ci if (!priv) 5462306a36Sopenharmony_ci return -ENOMEM; 5562306a36Sopenharmony_ci 5662306a36Sopenharmony_ci priv->data = of_device_get_match_data(dev); 5762306a36Sopenharmony_ci if (WARN_ON(!priv->data)) 5862306a36Sopenharmony_ci return -EINVAL; 5962306a36Sopenharmony_ci 6062306a36Sopenharmony_ci base = devm_platform_ioremap_resource(pdev, 0); 6162306a36Sopenharmony_ci if (IS_ERR(base)) 6262306a36Sopenharmony_ci return PTR_ERR(base); 6362306a36Sopenharmony_ci 6462306a36Sopenharmony_ci for (i = 0; i < priv->data->nclks; i++) 6562306a36Sopenharmony_ci priv->clk[i].id = priv->data->clock_names[i]; 6662306a36Sopenharmony_ci ret = devm_clk_bulk_get(dev, priv->data->nclks, priv->clk); 6762306a36Sopenharmony_ci if (ret) 6862306a36Sopenharmony_ci return ret; 6962306a36Sopenharmony_ci 7062306a36Sopenharmony_ci for (i = 0; i < priv->data->nrsts; i++) { 7162306a36Sopenharmony_ci name = priv->data->reset_names[i]; 7262306a36Sopenharmony_ci priv->rst[i] = devm_reset_control_get_shared(dev, name); 7362306a36Sopenharmony_ci if (IS_ERR(priv->rst[i])) 7462306a36Sopenharmony_ci return PTR_ERR(priv->rst[i]); 7562306a36Sopenharmony_ci } 7662306a36Sopenharmony_ci 7762306a36Sopenharmony_ci ret = clk_bulk_prepare_enable(priv->data->nclks, priv->clk); 7862306a36Sopenharmony_ci if (ret) 7962306a36Sopenharmony_ci return ret; 8062306a36Sopenharmony_ci 8162306a36Sopenharmony_ci for (nr = 0; nr < priv->data->nrsts; nr++) { 8262306a36Sopenharmony_ci ret = reset_control_deassert(priv->rst[nr]); 8362306a36Sopenharmony_ci if (ret) 8462306a36Sopenharmony_ci goto out_rst_assert; 8562306a36Sopenharmony_ci } 8662306a36Sopenharmony_ci 8762306a36Sopenharmony_ci regmap = devm_regmap_init_mmio(dev, base, priv->data->regconf); 8862306a36Sopenharmony_ci if (IS_ERR(regmap)) { 8962306a36Sopenharmony_ci ret = PTR_ERR(regmap); 9062306a36Sopenharmony_ci goto out_rst_assert; 9162306a36Sopenharmony_ci } 9262306a36Sopenharmony_ci 9362306a36Sopenharmony_ci config.dev = dev; 9462306a36Sopenharmony_ci config.driver_data = priv; 9562306a36Sopenharmony_ci config.of_node = dev->of_node; 9662306a36Sopenharmony_ci config.regmap = regmap; 9762306a36Sopenharmony_ci config.init_data = of_get_regulator_init_data(dev, dev->of_node, 9862306a36Sopenharmony_ci priv->data->desc); 9962306a36Sopenharmony_ci rdev = devm_regulator_register(dev, priv->data->desc, &config); 10062306a36Sopenharmony_ci if (IS_ERR(rdev)) { 10162306a36Sopenharmony_ci ret = PTR_ERR(rdev); 10262306a36Sopenharmony_ci goto out_rst_assert; 10362306a36Sopenharmony_ci } 10462306a36Sopenharmony_ci 10562306a36Sopenharmony_ci platform_set_drvdata(pdev, priv); 10662306a36Sopenharmony_ci 10762306a36Sopenharmony_ci return 0; 10862306a36Sopenharmony_ci 10962306a36Sopenharmony_ciout_rst_assert: 11062306a36Sopenharmony_ci while (nr--) 11162306a36Sopenharmony_ci reset_control_assert(priv->rst[nr]); 11262306a36Sopenharmony_ci 11362306a36Sopenharmony_ci clk_bulk_disable_unprepare(priv->data->nclks, priv->clk); 11462306a36Sopenharmony_ci 11562306a36Sopenharmony_ci return ret; 11662306a36Sopenharmony_ci} 11762306a36Sopenharmony_ci 11862306a36Sopenharmony_cistatic int uniphier_regulator_remove(struct platform_device *pdev) 11962306a36Sopenharmony_ci{ 12062306a36Sopenharmony_ci struct uniphier_regulator_priv *priv = platform_get_drvdata(pdev); 12162306a36Sopenharmony_ci int i; 12262306a36Sopenharmony_ci 12362306a36Sopenharmony_ci for (i = 0; i < priv->data->nrsts; i++) 12462306a36Sopenharmony_ci reset_control_assert(priv->rst[i]); 12562306a36Sopenharmony_ci 12662306a36Sopenharmony_ci clk_bulk_disable_unprepare(priv->data->nclks, priv->clk); 12762306a36Sopenharmony_ci 12862306a36Sopenharmony_ci return 0; 12962306a36Sopenharmony_ci} 13062306a36Sopenharmony_ci 13162306a36Sopenharmony_ci/* USB3 controller data */ 13262306a36Sopenharmony_ci#define USB3VBUS_OFFSET 0x0 13362306a36Sopenharmony_ci#define USB3VBUS_REG BIT(4) 13462306a36Sopenharmony_ci#define USB3VBUS_REG_EN BIT(3) 13562306a36Sopenharmony_cistatic const struct regulator_desc uniphier_usb3_regulator_desc = { 13662306a36Sopenharmony_ci .name = "vbus", 13762306a36Sopenharmony_ci .of_match = of_match_ptr("vbus"), 13862306a36Sopenharmony_ci .ops = &uniphier_regulator_ops, 13962306a36Sopenharmony_ci .type = REGULATOR_VOLTAGE, 14062306a36Sopenharmony_ci .owner = THIS_MODULE, 14162306a36Sopenharmony_ci .enable_reg = USB3VBUS_OFFSET, 14262306a36Sopenharmony_ci .enable_mask = USB3VBUS_REG_EN | USB3VBUS_REG, 14362306a36Sopenharmony_ci .enable_val = USB3VBUS_REG_EN | USB3VBUS_REG, 14462306a36Sopenharmony_ci .disable_val = USB3VBUS_REG_EN, 14562306a36Sopenharmony_ci}; 14662306a36Sopenharmony_ci 14762306a36Sopenharmony_cistatic const struct regmap_config uniphier_usb3_regulator_regconf = { 14862306a36Sopenharmony_ci .reg_bits = 32, 14962306a36Sopenharmony_ci .val_bits = 32, 15062306a36Sopenharmony_ci .reg_stride = 4, 15162306a36Sopenharmony_ci .max_register = 1, 15262306a36Sopenharmony_ci}; 15362306a36Sopenharmony_ci 15462306a36Sopenharmony_cistatic const char * const uniphier_pro4_clock_reset_names[] = { 15562306a36Sopenharmony_ci "gio", "link", 15662306a36Sopenharmony_ci}; 15762306a36Sopenharmony_ci 15862306a36Sopenharmony_cistatic const struct uniphier_regulator_soc_data uniphier_pro4_usb3_data = { 15962306a36Sopenharmony_ci .nclks = ARRAY_SIZE(uniphier_pro4_clock_reset_names), 16062306a36Sopenharmony_ci .clock_names = uniphier_pro4_clock_reset_names, 16162306a36Sopenharmony_ci .nrsts = ARRAY_SIZE(uniphier_pro4_clock_reset_names), 16262306a36Sopenharmony_ci .reset_names = uniphier_pro4_clock_reset_names, 16362306a36Sopenharmony_ci .desc = &uniphier_usb3_regulator_desc, 16462306a36Sopenharmony_ci .regconf = &uniphier_usb3_regulator_regconf, 16562306a36Sopenharmony_ci}; 16662306a36Sopenharmony_ci 16762306a36Sopenharmony_cistatic const char * const uniphier_pxs2_clock_reset_names[] = { 16862306a36Sopenharmony_ci "link", 16962306a36Sopenharmony_ci}; 17062306a36Sopenharmony_ci 17162306a36Sopenharmony_cistatic const struct uniphier_regulator_soc_data uniphier_pxs2_usb3_data = { 17262306a36Sopenharmony_ci .nclks = ARRAY_SIZE(uniphier_pxs2_clock_reset_names), 17362306a36Sopenharmony_ci .clock_names = uniphier_pxs2_clock_reset_names, 17462306a36Sopenharmony_ci .nrsts = ARRAY_SIZE(uniphier_pxs2_clock_reset_names), 17562306a36Sopenharmony_ci .reset_names = uniphier_pxs2_clock_reset_names, 17662306a36Sopenharmony_ci .desc = &uniphier_usb3_regulator_desc, 17762306a36Sopenharmony_ci .regconf = &uniphier_usb3_regulator_regconf, 17862306a36Sopenharmony_ci}; 17962306a36Sopenharmony_ci 18062306a36Sopenharmony_cistatic const struct of_device_id uniphier_regulator_match[] = { 18162306a36Sopenharmony_ci /* USB VBUS */ 18262306a36Sopenharmony_ci { 18362306a36Sopenharmony_ci .compatible = "socionext,uniphier-pro4-usb3-regulator", 18462306a36Sopenharmony_ci .data = &uniphier_pro4_usb3_data, 18562306a36Sopenharmony_ci }, 18662306a36Sopenharmony_ci { 18762306a36Sopenharmony_ci .compatible = "socionext,uniphier-pro5-usb3-regulator", 18862306a36Sopenharmony_ci .data = &uniphier_pro4_usb3_data, 18962306a36Sopenharmony_ci }, 19062306a36Sopenharmony_ci { 19162306a36Sopenharmony_ci .compatible = "socionext,uniphier-pxs2-usb3-regulator", 19262306a36Sopenharmony_ci .data = &uniphier_pxs2_usb3_data, 19362306a36Sopenharmony_ci }, 19462306a36Sopenharmony_ci { 19562306a36Sopenharmony_ci .compatible = "socionext,uniphier-ld20-usb3-regulator", 19662306a36Sopenharmony_ci .data = &uniphier_pxs2_usb3_data, 19762306a36Sopenharmony_ci }, 19862306a36Sopenharmony_ci { 19962306a36Sopenharmony_ci .compatible = "socionext,uniphier-pxs3-usb3-regulator", 20062306a36Sopenharmony_ci .data = &uniphier_pxs2_usb3_data, 20162306a36Sopenharmony_ci }, 20262306a36Sopenharmony_ci { 20362306a36Sopenharmony_ci .compatible = "socionext,uniphier-nx1-usb3-regulator", 20462306a36Sopenharmony_ci .data = &uniphier_pxs2_usb3_data, 20562306a36Sopenharmony_ci }, 20662306a36Sopenharmony_ci { /* Sentinel */ }, 20762306a36Sopenharmony_ci}; 20862306a36Sopenharmony_ciMODULE_DEVICE_TABLE(of, uniphier_regulator_match); 20962306a36Sopenharmony_ci 21062306a36Sopenharmony_cistatic struct platform_driver uniphier_regulator_driver = { 21162306a36Sopenharmony_ci .probe = uniphier_regulator_probe, 21262306a36Sopenharmony_ci .remove = uniphier_regulator_remove, 21362306a36Sopenharmony_ci .driver = { 21462306a36Sopenharmony_ci .name = "uniphier-regulator", 21562306a36Sopenharmony_ci .probe_type = PROBE_PREFER_ASYNCHRONOUS, 21662306a36Sopenharmony_ci .of_match_table = uniphier_regulator_match, 21762306a36Sopenharmony_ci }, 21862306a36Sopenharmony_ci}; 21962306a36Sopenharmony_cimodule_platform_driver(uniphier_regulator_driver); 22062306a36Sopenharmony_ci 22162306a36Sopenharmony_ciMODULE_AUTHOR("Kunihiko Hayashi <hayashi.kunihiko@socionext.com>"); 22262306a36Sopenharmony_ciMODULE_DESCRIPTION("UniPhier Regulator Controller Driver"); 22362306a36Sopenharmony_ciMODULE_LICENSE("GPL"); 224