soc: renesas: Add PWC support for RZ/V2M
authorFabrizio Castro <fabrizio.castro.jz@renesas.com>
Fri, 6 Jan 2023 12:58:16 +0000 (12:58 +0000)
committerGeert Uytterhoeven <geert+renesas@glider.be>
Mon, 23 Jan 2023 08:28:28 +0000 (09:28 +0100)
The Renesas RZ/V2M External Power Sequence Controller (PWC)
IP is capable of:
* external power supply on/off sequence generation
* on/off signal generation for the LPDDR4 core power supply (LPVDD)
* key input signals processing
* general-purpose output pins

Add the corresponding device driver.

Signed-off-by: Fabrizio Castro <fabrizio.castro.jz@renesas.com>
Reviewed-by: Geert Uytterhoeven <geert+renesas@glider.be>
Link: https://lore.kernel.org/r/20230106125816.10600-3-fabrizio.castro.jz@renesas.com
Signed-off-by: Geert Uytterhoeven <geert+renesas@glider.be>
drivers/soc/renesas/Kconfig
drivers/soc/renesas/Makefile
drivers/soc/renesas/pwc-rzv2m.c [new file with mode: 0644]

index 660498252ec5b4eb7f74117d3261ccb4b24668af..4e8b51ba226614ef1a1e91b47c8a4d249ceff9ac 100644 (file)
@@ -330,6 +330,7 @@ config ARCH_R9A09G011
        bool "ARM64 Platform support for RZ/V2M"
        select PM
        select PM_GENERIC_DOMAINS
+       select PWC_RZV2M
        help
          This enables support for the Renesas RZ/V2M SoC.
 
@@ -345,6 +346,9 @@ config ARCH_R9A07G043
 
 endif # RISCV
 
+config PWC_RZV2M
+       bool "Renesas RZ/V2M PWC support" if COMPILE_TEST
+
 config RST_RCAR
        bool "Reset Controller support for R-Car" if COMPILE_TEST
 
index 535868c9c7e4b67623c93dacbef42f80a2573382..6e4e77b0afffacde3882ef1076b9da10b67e7230 100644 (file)
@@ -32,6 +32,7 @@ obj-$(CONFIG_ARCH_R9A06G032)  += r9a06g032-smp.o
 endif
 
 # Family
+obj-$(CONFIG_PWC_RZV2M)                += pwc-rzv2m.o
 obj-$(CONFIG_RST_RCAR)         += rcar-rst.o
 obj-$(CONFIG_SYSC_RCAR)                += rcar-sysc.o
 obj-$(CONFIG_SYSC_RCAR_GEN4)   += rcar-gen4-sysc.o
diff --git a/drivers/soc/renesas/pwc-rzv2m.c b/drivers/soc/renesas/pwc-rzv2m.c
new file mode 100644 (file)
index 0000000..c83bdbd
--- /dev/null
@@ -0,0 +1,141 @@
+// SPDX-License-Identifier: GPL-2.0-only
+/*
+ * Copyright (C) 2023 Renesas Electronics Corporation
+ */
+
+#include <linux/delay.h>
+#include <linux/gpio/driver.h>
+#include <linux/platform_device.h>
+#include <linux/reboot.h>
+
+#define PWC_PWCRST                     0x00
+#define PWC_PWCCKEN                    0x04
+#define PWC_PWCCTL                     0x50
+#define PWC_GPIO                       0x80
+
+#define PWC_PWCRST_RSTSOFTAX           0x1
+#define PWC_PWCCKEN_ENGCKMAIN          0x1
+#define PWC_PWCCTL_PWOFF               0x1
+
+struct rzv2m_pwc_priv {
+       void __iomem *base;
+       struct device *dev;
+       struct gpio_chip gp;
+       DECLARE_BITMAP(ch_en_bits, 2);
+};
+
+static void rzv2m_pwc_gpio_set(struct gpio_chip *chip, unsigned int offset,
+                              int value)
+{
+       struct rzv2m_pwc_priv *priv = gpiochip_get_data(chip);
+       u32 reg;
+
+       /* BIT 16 enables write to BIT 0, and BIT 17 enables write to BIT 1 */
+       reg = BIT(offset + 16);
+       if (value)
+               reg |= BIT(offset);
+
+       writel(reg, priv->base + PWC_GPIO);
+
+       assign_bit(offset, priv->ch_en_bits, value);
+}
+
+static int rzv2m_pwc_gpio_get(struct gpio_chip *chip, unsigned int offset)
+{
+       struct rzv2m_pwc_priv *priv = gpiochip_get_data(chip);
+
+       return test_bit(offset, priv->ch_en_bits);
+}
+
+static int rzv2m_pwc_gpio_direction_output(struct gpio_chip *gc,
+                                          unsigned int nr, int value)
+{
+       if (nr > 1)
+               return -EINVAL;
+
+       rzv2m_pwc_gpio_set(gc, nr, value);
+
+       return 0;
+}
+
+static const struct gpio_chip rzv2m_pwc_gc = {
+       .label = "gpio_rzv2m_pwc",
+       .owner = THIS_MODULE,
+       .get = rzv2m_pwc_gpio_get,
+       .set = rzv2m_pwc_gpio_set,
+       .direction_output = rzv2m_pwc_gpio_direction_output,
+       .can_sleep = false,
+       .ngpio = 2,
+       .base = -1,
+};
+
+static int rzv2m_pwc_poweroff(struct sys_off_data *data)
+{
+       struct rzv2m_pwc_priv *priv = data->cb_data;
+
+       writel(PWC_PWCRST_RSTSOFTAX, priv->base + PWC_PWCRST);
+       writel(PWC_PWCCKEN_ENGCKMAIN, priv->base + PWC_PWCCKEN);
+       writel(PWC_PWCCTL_PWOFF, priv->base + PWC_PWCCTL);
+
+       mdelay(150);
+
+       dev_err(priv->dev, "Failed to power off the system");
+
+       return NOTIFY_DONE;
+}
+
+static int rzv2m_pwc_probe(struct platform_device *pdev)
+{
+       struct rzv2m_pwc_priv *priv;
+       int ret;
+
+       priv = devm_kzalloc(&pdev->dev, sizeof(*priv), GFP_KERNEL);
+       if (!priv)
+               return -ENOMEM;
+
+       priv->base = devm_platform_ioremap_resource(pdev, 0);
+       if (IS_ERR(priv->base))
+               return PTR_ERR(priv->base);
+
+       /*
+        * The register used by this driver cannot be read, therefore set the
+        * outputs to their default values and initialize priv->ch_en_bits
+        * accordingly. BIT 16 enables write to BIT 0, BIT 17 enables write to
+        * BIT 1, and the default value of both BIT 0 and BIT 1 is 0.
+        */
+       writel(BIT(17) | BIT(16), priv->base + PWC_GPIO);
+       bitmap_zero(priv->ch_en_bits, 2);
+
+       priv->gp = rzv2m_pwc_gc;
+       priv->gp.parent = pdev->dev.parent;
+       priv->gp.fwnode = dev_fwnode(&pdev->dev);
+
+       ret = devm_gpiochip_add_data(&pdev->dev, &priv->gp, priv);
+       if (ret)
+               return ret;
+
+       if (device_property_read_bool(&pdev->dev, "renesas,rzv2m-pwc-power"))
+               ret = devm_register_power_off_handler(&pdev->dev,
+                                                     rzv2m_pwc_poweroff, priv);
+
+       return ret;
+}
+
+static const struct of_device_id rzv2m_pwc_of_match[] = {
+       { .compatible = "renesas,rzv2m-pwc" },
+       { /* sentinel */ }
+};
+MODULE_DEVICE_TABLE(of, rzv2m_pwc_of_match);
+
+static struct platform_driver rzv2m_pwc_driver = {
+       .probe = rzv2m_pwc_probe,
+       .driver = {
+               .name = "rzv2m_pwc",
+               .of_match_table = of_match_ptr(rzv2m_pwc_of_match),
+       },
+};
+module_platform_driver(rzv2m_pwc_driver);
+
+MODULE_LICENSE("GPL");
+MODULE_AUTHOR("Fabrizio Castro <castro.fabrizio.jz@renesas.com>");
+MODULE_DESCRIPTION("Renesas RZ/V2M PWC driver");