This driver is ready for mainstream. So, move it out of staging.

Signed-off-by: Mauro Carvalho Chehab <mchehab+hua...@kernel.org>
---
 .../mfd/hisilicon,hi6421-spmi-pmic.yaml       | 135 ++++++++
 MAINTAINERS                                   |   7 +
 drivers/mfd/Kconfig                           |  15 +
 drivers/mfd/Makefile                          |   1 +
 drivers/mfd/hi6421-spmi-pmic.c                | 296 ++++++++++++++++++
 drivers/staging/hikey9xx/Kconfig              |  17 -
 drivers/staging/hikey9xx/Makefile             |   1 -
 drivers/staging/hikey9xx/hi6421-spmi-pmic.c   | 296 ------------------
 .../hikey9xx/hisilicon,hi6421-spmi-pmic.yaml  | 135 --------
 9 files changed, 454 insertions(+), 449 deletions(-)
 create mode 100644 
Documentation/devicetree/bindings/mfd/hisilicon,hi6421-spmi-pmic.yaml
 create mode 100644 drivers/mfd/hi6421-spmi-pmic.c
 delete mode 100644 drivers/staging/hikey9xx/hi6421-spmi-pmic.c
 delete mode 100644 drivers/staging/hikey9xx/hisilicon,hi6421-spmi-pmic.yaml

diff --git 
a/Documentation/devicetree/bindings/mfd/hisilicon,hi6421-spmi-pmic.yaml 
b/Documentation/devicetree/bindings/mfd/hisilicon,hi6421-spmi-pmic.yaml
new file mode 100644
index 000000000000..3b23ad56b31a
--- /dev/null
+++ b/Documentation/devicetree/bindings/mfd/hisilicon,hi6421-spmi-pmic.yaml
@@ -0,0 +1,135 @@
+# SPDX-License-Identifier: GPL-2.0
+%YAML 1.2
+---
+$id: http://devicetree.org/schemas/mfd/hisilicon,hi6421-spmi-pmic.yaml#
+$schema: http://devicetree.org/meta-schemas/core.yaml#
+
+title: HiSilicon 6421v600 SPMI PMIC
+
+maintainers:
+  - Mauro Carvalho Chehab <mchehab+hua...@kernel.org>
+
+description: |
+  HiSilicon 6421v600 should be connected inside a MIPI System Power Management
+  (SPMI) bus. It provides interrupts and power supply.
+
+  The GPIO and interrupt settings are represented as part of the top-level PMIC
+  node.
+
+  The SPMI controller part is provided by
+  drivers/staging/hikey9xx/hisilicon,hisi-spmi-controller.yaml.
+
+properties:
+  $nodename:
+    pattern: "pmic@[0-9a-f]"
+
+  compatible:
+    const: hisilicon,hi6421v600-spmi
+
+  reg:
+    maxItems: 1
+
+  '#interrupt-cells':
+    const: 2
+
+  interrupt-controller:
+    description:
+      Identify that the PMIC is capable of behaving as an interrupt controller.
+
+  gpios:
+    maxItems: 1
+
+  regulators:
+    type: object
+
+    properties:
+      '#address-cells':
+        const: 1
+
+      '#size-cells':
+        const: 0
+
+    patternProperties:
+      '^ldo[0-9]+@[0-9a-f]$':
+        type: object
+
+        $ref: "/schemas/regulator/regulator.yaml#"
+
+required:
+  - compatible
+  - reg
+  - regulators
+
+additionalProperties: false
+
+examples:
+  - |
+    /* pmic properties */
+
+    pmic: pmic@0 {
+      compatible = "hisilicon,hi6421-spmi";
+      reg = <0 0>;
+
+      #interrupt-cells = <2>;
+      interrupt-controller;
+      gpios = <&gpio28 0 0>;
+
+      regulators {
+        #address-cells = <1>;
+        #size-cells = <0>;
+
+        ldo3: LDO3 {
+          regulator-name = "ldo3";
+          regulator-min-microvolt = <1500000>;
+          regulator-max-microvolt = <2000000>;
+          regulator-boot-on;
+        };
+
+        ldo4: LDO4 {
+          regulator-name = "ldo4";
+          regulator-min-microvolt = <1725000>;
+          regulator-max-microvolt = <1900000>;
+          regulator-boot-on;
+        };
+
+        ldo9: LDO9 {
+          regulator-name = "ldo9";
+          regulator-min-microvolt = <1750000>;
+          regulator-max-microvolt = <3300000>;
+          regulator-boot-on;
+        };
+
+        ldo15: LDO15 {
+          regulator-name = "ldo15";
+          regulator-min-microvolt = <1800000>;
+          regulator-max-microvolt = <3000000>;
+          regulator-always-on;
+        };
+
+        ldo16: LDO16 {
+          regulator-name = "ldo16";
+          regulator-min-microvolt = <1800000>;
+          regulator-max-microvolt = <3000000>;
+          regulator-boot-on;
+        };
+
+        ldo17: LDO17 {
+          regulator-name = "ldo17";
+          regulator-min-microvolt = <2500000>;
+          regulator-max-microvolt = <3300000>;
+        };
+
+        ldo33: LDO33 {
+          regulator-name = "ldo33";
+          regulator-min-microvolt = <2500000>;
+          regulator-max-microvolt = <3300000>;
+          regulator-boot-on;
+        };
+
+        ldo34: LDO34 {
+          regulator-name = "ldo34";
+          regulator-min-microvolt = <2600000>;
+          regulator-max-microvolt = <3300000>;
+        };
+      };
+    };
diff --git a/MAINTAINERS b/MAINTAINERS
index 406c2340f221..241f11b7d48a 100644
--- a/MAINTAINERS
+++ b/MAINTAINERS
@@ -8085,6 +8085,13 @@ S:       Maintained
 F:     
Documentation/devicetree/bindings/spmi/hisilicon,hisi-spmi-controller.yaml
 F:     drivers/spmi/hisi-spmi-controller.c
 
+HISILICON SPMI PMIC DRIVER FOR HIKEY 6421v600
+M:     Mauro Carvalho Chehab <mchehab+hua...@kernel.org>
+L:     linux-ker...@vger.kernel.org
+S:     Maintained
+F:     Documentation/devicetree/bindings/mfd/hisilicon,hi6421-spmi-pmic.yaml
+F:     drivers/mfd/hi6421-spmi-pmic.c
+
 HISILICON STAGING DRIVERS FOR HIKEY 960/970
 M:     Mauro Carvalho Chehab <mchehab+hua...@kernel.org>
 L:     de...@driverdev.osuosl.org
diff --git a/drivers/mfd/Kconfig b/drivers/mfd/Kconfig
index bdfce7b15621..d80ccbf50235 100644
--- a/drivers/mfd/Kconfig
+++ b/drivers/mfd/Kconfig
@@ -509,6 +509,21 @@ config MFD_HI6421_PMIC
          menus in order to enable them.
          We communicate with the Hi6421 via memory-mapped I/O.
 
+config MFD_HI6421_SPMI
+       tristate "HiSilicon Hi6421v600 SPMI PMU/Codec IC"
+       depends on OF
+       depends on SPMI
+       select MFD_CORE
+       help
+         Add support for HiSilicon Hi6421v600 SPMI PMIC. Hi6421 includes
+         multi-functions, such as regulators, RTC, codec, Coulomb counter,
+         etc.
+
+         This driver includes core APIs _only_. You have to select
+         individual components like voltage regulators under corresponding
+         menus in order to enable them.
+         We communicate with the Hi6421v600 via a SPMI bus.
+
 config MFD_HI655X_PMIC
        tristate "HiSilicon Hi655X series PMU/Codec IC"
        depends on ARCH_HISI || COMPILE_TEST
diff --git a/drivers/mfd/Makefile b/drivers/mfd/Makefile
index 14fdb188af02..b64e89ade44b 100644
--- a/drivers/mfd/Makefile
+++ b/drivers/mfd/Makefile
@@ -234,6 +234,7 @@ obj-$(CONFIG_MFD_IPAQ_MICRO)        += ipaq-micro.o
 obj-$(CONFIG_MFD_IQS62X)       += iqs62x.o
 obj-$(CONFIG_MFD_MENF21BMC)    += menf21bmc.o
 obj-$(CONFIG_MFD_HI6421_PMIC)  += hi6421-pmic-core.o
+obj-$(CONFIG_MFD_HI6421_SPMI)  += hi6421-spmi-pmic.o
 obj-$(CONFIG_MFD_HI655X_PMIC)   += hi655x-pmic.o
 obj-$(CONFIG_MFD_DLN2)         += dln2.o
 obj-$(CONFIG_MFD_RT5033)       += rt5033.o
diff --git a/drivers/mfd/hi6421-spmi-pmic.c b/drivers/mfd/hi6421-spmi-pmic.c
new file mode 100644
index 000000000000..e91ff07b5af8
--- /dev/null
+++ b/drivers/mfd/hi6421-spmi-pmic.c
@@ -0,0 +1,296 @@
+// SPDX-License-Identifier: GPL-2.0
+/*
+ * Device driver for regulators in HISI PMIC IC
+ *
+ * Copyright (c) 2013 Linaro Ltd.
+ * Copyright (c) 2011 Hisilicon.
+ * Copyright (c) 2020-2021 Huawei Technologies Co., Ltd
+ */
+
+#include <linux/bitops.h>
+#include <linux/interrupt.h>
+#include <linux/irq.h>
+#include <linux/mfd/core.h>
+#include <linux/mfd/hi6421-spmi-pmic.h>
+#include <linux/module.h>
+#include <linux/of_gpio.h>
+#include <linux/platform_device.h>
+#include <linux/slab.h>
+#include <linux/spmi.h>
+
+enum hi6421_spmi_pmic_irq_list {
+       OTMP = 0,
+       VBUS_CONNECT,
+       VBUS_DISCONNECT,
+       ALARMON_R,
+       HOLD_6S,
+       HOLD_1S,
+       POWERKEY_UP,
+       POWERKEY_DOWN,
+       OCP_SCP_R,
+       COUL_R,
+       SIM0_HPD_R,
+       SIM0_HPD_F,
+       SIM1_HPD_R,
+       SIM1_HPD_F,
+       PMIC_IRQ_LIST_MAX,
+};
+
+#define HISI_IRQ_ARRAY                 2
+#define HISI_IRQ_NUM                   (HISI_IRQ_ARRAY * 8)
+
+#define HISI_IRQ_KEY_NUM               0
+
+#define HISI_BITS                      8
+#define HISI_IRQ_KEY_VALUE             (BIT(POWERKEY_DOWN) | BIT(POWERKEY_UP))
+#define HISI_MASK                      GENMASK(HISI_BITS - 1, 0)
+
+/*
+ * The IRQs are mapped as:
+ *
+ *     ======================  =============   ============    =====
+ *     IRQ                     MASK REGISTER   IRQ REGISTER    BIT
+ *     ======================  =============   ============    =====
+ *     OTMP                    0x0202          0x212           bit 0
+ *     VBUS_CONNECT            0x0202          0x212           bit 1
+ *     VBUS_DISCONNECT         0x0202          0x212           bit 2
+ *     ALARMON_R               0x0202          0x212           bit 3
+ *     HOLD_6S                 0x0202          0x212           bit 4
+ *     HOLD_1S                 0x0202          0x212           bit 5
+ *     POWERKEY_UP             0x0202          0x212           bit 6
+ *     POWERKEY_DOWN           0x0202          0x212           bit 7
+ *
+ *     OCP_SCP_R               0x0203          0x213           bit 0
+ *     COUL_R                  0x0203          0x213           bit 1
+ *     SIM0_HPD_R              0x0203          0x213           bit 2
+ *     SIM0_HPD_F              0x0203          0x213           bit 3
+ *     SIM1_HPD_R              0x0203          0x213           bit 4
+ *     SIM1_HPD_F              0x0203          0x213           bit 5
+ *     ======================  =============   ============    =====
+ */
+#define SOC_PMIC_IRQ_MASK_0_ADDR       0x0202
+#define SOC_PMIC_IRQ0_ADDR             0x0212
+
+#define IRQ_MASK_REGISTER(irq_data)    (SOC_PMIC_IRQ_MASK_0_ADDR + \           
                                         irqd_to_hwirq(irq_data) >> 3)
+#define IRQ_MASK_BIT(irq_data)         BIT(irqd_to_hwirq(irq_data) & 0x07)
+
+static const struct mfd_cell hi6421v600_devs[] = {
+       { .name = "hi6421v600-regulator", },
+};
+
+static irqreturn_t hi6421_spmi_irq_handler(int irq, void *priv)
+{
+       struct hi6421_spmi_pmic *ddata = (struct hi6421_spmi_pmic *)priv;
+       unsigned long pending;
+       unsigned int in;
+       int i, offset;
+
+       for (i = 0; i < HISI_IRQ_ARRAY; i++) {
+               regmap_read(ddata->regmap, SOC_PMIC_IRQ0_ADDR + i, &in);
+               pending = HISI_MASK & in;
+               regmap_write(ddata->regmap, SOC_PMIC_IRQ0_ADDR + i, pending);
+
+               if (i == HISI_IRQ_KEY_NUM &&
+                   (pending & HISI_IRQ_KEY_VALUE) == HISI_IRQ_KEY_VALUE) {
+                       generic_handle_irq(ddata->irqs[POWERKEY_DOWN]);
+                       generic_handle_irq(ddata->irqs[POWERKEY_UP]);
+                       pending &= (~HISI_IRQ_KEY_VALUE);
+               }
+
+               if (!pending)
+                       continue;
+
+               for_each_set_bit(offset, &pending, HISI_BITS)
+                       generic_handle_irq(ddata->irqs[offset + i * HISI_BITS]);
+       }
+
+       return IRQ_HANDLED;
+}
+
+static void hi6421_spmi_irq_mask(struct irq_data *d)
+{
+       struct hi6421_spmi_pmic *ddata = irq_data_get_irq_chip_data(d);
+       unsigned long flags;
+       unsigned int data;
+       u32 offset;
+
+       offset = IRQ_MASK_REGISTER(d);
+
+       spin_lock_irqsave(&ddata->lock, flags);
+
+       regmap_read(ddata->regmap, offset, &data);
+       data |= IRQ_MASK_BIT(d);
+       regmap_write(ddata->regmap, offset, data);
+
+       spin_unlock_irqrestore(&ddata->lock, flags);
+}
+
+static void hi6421_spmi_irq_unmask(struct irq_data *d)
+{
+       struct hi6421_spmi_pmic *ddata = irq_data_get_irq_chip_data(d);
+       u32 data, offset;
+       unsigned long flags;
+
+       offset = (irqd_to_hwirq(d) >> 3);
+       offset += SOC_PMIC_IRQ_MASK_0_ADDR;
+
+       spin_lock_irqsave(&ddata->lock, flags);
+
+       regmap_read(ddata->regmap, offset, &data);
+       data &= ~(1 << (irqd_to_hwirq(d) & 0x07));
+       regmap_write(ddata->regmap, offset, data);
+
+       spin_unlock_irqrestore(&ddata->lock, flags);
+}
+
+static struct irq_chip hi6421_spmi_pmu_irqchip = {
+       .name           = "hisi-irq",
+       .irq_mask       = hi6421_spmi_irq_mask,
+       .irq_unmask     = hi6421_spmi_irq_unmask,
+       .irq_disable    = hi6421_spmi_irq_mask,
+       .irq_enable     = hi6421_spmi_irq_unmask,
+};
+
+static int hi6421_spmi_irq_map(struct irq_domain *d, unsigned int virq,
+                              irq_hw_number_t hw)
+{
+       struct hi6421_spmi_pmic *ddata = d->host_data;
+
+       irq_set_chip_and_handler_name(virq, &hi6421_spmi_pmu_irqchip,
+                                     handle_simple_irq, "hisi");
+       irq_set_chip_data(virq, ddata);
+       irq_set_irq_type(virq, IRQ_TYPE_NONE);
+
+       return 0;
+}
+
+static const struct irq_domain_ops hi6421_spmi_domain_ops = {
+       .map    = hi6421_spmi_irq_map,
+       .xlate  = irq_domain_xlate_twocell,
+};
+
+static void hi6421_spmi_pmic_irq_init(struct hi6421_spmi_pmic *ddata)
+{
+       int i;
+       unsigned int pending;
+
+       for (i = 0; i < HISI_IRQ_ARRAY; i++)
+               regmap_write(ddata->regmap, SOC_PMIC_IRQ_MASK_0_ADDR + i,
+                                       HISI_MASK);
+
+       for (i = 0; i < HISI_IRQ_ARRAY; i++) {
+               regmap_read(ddata->regmap, SOC_PMIC_IRQ0_ADDR + i, &pending);
+               regmap_write(ddata->regmap, SOC_PMIC_IRQ0_ADDR + i,
+                            HISI_MASK);
+       }
+}
+
+static const struct regmap_config regmap_config = {
+       .reg_bits               = 16,
+       .val_bits               = HISI_BITS,
+       .max_register           = 0xffff,
+       .fast_io                = true
+};
+
+static int hi6421_spmi_pmic_probe(struct spmi_device *pdev)
+{
+       struct device *dev = &pdev->dev;
+       struct device_node *np = dev->of_node;
+       struct hi6421_spmi_pmic *ddata;
+       unsigned int virq;
+       int ret, i;
+
+       ddata = devm_kzalloc(dev, sizeof(*ddata), GFP_KERNEL);
+       if (!ddata)
+               return -ENOMEM;
+
+       ddata->regmap = devm_regmap_init_spmi_ext(pdev, &regmap_config);
+       if (IS_ERR(ddata->regmap))
+               return PTR_ERR(ddata->regmap);
+
+       spin_lock_init(&ddata->lock);
+
+       ddata->dev = dev;
+
+       ddata->gpio = of_get_gpio(np, 0);
+       if (ddata->gpio < 0)
+               return ddata->gpio;
+
+       if (!gpio_is_valid(ddata->gpio))
+               return -EINVAL;
+
+       ret = devm_gpio_request_one(dev, ddata->gpio, GPIOF_IN, "pmic");
+       if (ret < 0) {
+               dev_err(dev, "Failed to request gpio%d\n", ddata->gpio);
+               return ret;
+       }
+
+       ddata->irq = gpio_to_irq(ddata->gpio);
+
+       hi6421_spmi_pmic_irq_init(ddata);
+
+       ddata->irqs = devm_kzalloc(dev, HISI_IRQ_NUM * sizeof(int), GFP_KERNEL);
+       if (!ddata->irqs)
+               return -ENOMEM;
+
+       ddata->domain = irq_domain_add_simple(np, HISI_IRQ_NUM, 0,
+                                            &hi6421_spmi_domain_ops, ddata);
+       if (!ddata->domain) {
+               dev_err(dev, "Failed to create IRQ domain\n");
+               return -ENODEV;
+       }
+
+       for (i = 0; i < HISI_IRQ_NUM; i++) {
+               virq = irq_create_mapping(ddata->domain, i);
+               if (!virq) {
+                       dev_err(dev, "Failed to map H/W IRQ\n");
+                       return -ENOSPC;
+               }
+               ddata->irqs[i] = virq;
+       }
+
+       ret = request_threaded_irq(ddata->irq, hi6421_spmi_irq_handler, NULL,
+                                  IRQF_TRIGGER_LOW | IRQF_SHARED | 
IRQF_NO_SUSPEND,
+                                  "pmic", ddata);
+       if (ret < 0) {
+               dev_err(dev, "Failed to start IRQ handling thread: error %d\n",
+                       ret);
+               return ret;
+       }
+
+       dev_set_drvdata(&pdev->dev, ddata);
+
+       ret = devm_mfd_add_devices(&pdev->dev, PLATFORM_DEVID_NONE,
+                                  hi6421v600_devs, ARRAY_SIZE(hi6421v600_devs),
+                                  NULL, 0, NULL);
+       if (ret < 0)
+               dev_err(dev, "Failed to add child devices: %d\n", ret);
+
+       return ret;
+}
+
+static void hi6421_spmi_pmic_remove(struct spmi_device *pdev)
+{
+       struct hi6421_spmi_pmic *ddata = dev_get_drvdata(&pdev->dev);
+
+       free_irq(ddata->irq, ddata);
+}
+
+static const struct of_device_id pmic_spmi_id_table[] = {
+       { .compatible = "hisilicon,hi6421-spmi" },
+       { }
+};
+MODULE_DEVICE_TABLE(of, pmic_spmi_id_table);
+
+static struct spmi_driver hi6421_spmi_pmic_driver = {
+       .driver = {
+               .name   = "hi6421-spmi-pmic",
+               .of_match_table = pmic_spmi_id_table,
+       },
+       .probe  = hi6421_spmi_pmic_probe,
+       .remove = hi6421_spmi_pmic_remove,
+};
+module_spmi_driver(hi6421_spmi_pmic_driver);
+
+MODULE_DESCRIPTION("HiSilicon Hi6421v600 SPMI PMIC driver");
+MODULE_LICENSE("GPL v2");
diff --git a/drivers/staging/hikey9xx/Kconfig b/drivers/staging/hikey9xx/Kconfig
index c2b16ebfd52c..1afb8648a2c4 100644
--- a/drivers/staging/hikey9xx/Kconfig
+++ b/drivers/staging/hikey9xx/Kconfig
@@ -1,22 +1,5 @@
 # SPDX-License-Identifier: GPL-2.0
 
-# to be placed at drivers/mfd
-config MFD_HI6421_SPMI
-       tristate "HiSilicon Hi6421v600 SPMI PMU/Codec IC"
-       depends on HAS_IOMEM
-       depends on OF
-       depends on SPMI
-       select MFD_CORE
-       help
-         Add support for HiSilicon Hi6421v600 SPMI PMIC. Hi6421 includes
-         multi-functions, such as regulators, RTC, codec, Coulomb counter,
-         etc.
-
-         This driver includes core APIs _only_. You have to select
-         individual components like voltage regulators under corresponding
-         menus in order to enable them.
-         We communicate with the Hi6421v600 via a SPMI bus.
-
 # to be placed at drivers/regulator
 config REGULATOR_HI6421V600
        tristate "HiSilicon Hi6421v600 PMIC voltage regulator support"
diff --git a/drivers/staging/hikey9xx/Makefile 
b/drivers/staging/hikey9xx/Makefile
index 347880fd378f..4d63184e6086 100644
--- a/drivers/staging/hikey9xx/Makefile
+++ b/drivers/staging/hikey9xx/Makefile
@@ -1,4 +1,3 @@
 # SPDX-License-Identifier: GPL-2.0
 
-obj-$(CONFIG_MFD_HI6421_SPMI)          += hi6421-spmi-pmic.o
 obj-$(CONFIG_REGULATOR_HI6421V600)     += hi6421v600-regulator.o
diff --git a/drivers/staging/hikey9xx/hi6421-spmi-pmic.c 
b/drivers/staging/hikey9xx/hi6421-spmi-pmic.c
deleted file mode 100644
index e91ff07b5af8..000000000000
--- a/drivers/staging/hikey9xx/hi6421-spmi-pmic.c
+++ /dev/null
@@ -1,296 +0,0 @@
-// SPDX-License-Identifier: GPL-2.0
-/*
- * Device driver for regulators in HISI PMIC IC
- *
- * Copyright (c) 2013 Linaro Ltd.
- * Copyright (c) 2011 Hisilicon.
- * Copyright (c) 2020-2021 Huawei Technologies Co., Ltd
- */
-
-#include <linux/bitops.h>
-#include <linux/interrupt.h>
-#include <linux/irq.h>
-#include <linux/mfd/core.h>
-#include <linux/mfd/hi6421-spmi-pmic.h>
-#include <linux/module.h>
-#include <linux/of_gpio.h>
-#include <linux/platform_device.h>
-#include <linux/slab.h>
-#include <linux/spmi.h>
-
-enum hi6421_spmi_pmic_irq_list {
-       OTMP = 0,
-       VBUS_CONNECT,
-       VBUS_DISCONNECT,
-       ALARMON_R,
-       HOLD_6S,
-       HOLD_1S,
-       POWERKEY_UP,
-       POWERKEY_DOWN,
-       OCP_SCP_R,
-       COUL_R,
-       SIM0_HPD_R,
-       SIM0_HPD_F,
-       SIM1_HPD_R,
-       SIM1_HPD_F,
-       PMIC_IRQ_LIST_MAX,
-};
-
-#define HISI_IRQ_ARRAY                 2
-#define HISI_IRQ_NUM                   (HISI_IRQ_ARRAY * 8)
-
-#define HISI_IRQ_KEY_NUM               0
-
-#define HISI_BITS                      8
-#define HISI_IRQ_KEY_VALUE             (BIT(POWERKEY_DOWN) | BIT(POWERKEY_UP))
-#define HISI_MASK                      GENMASK(HISI_BITS - 1, 0)
-
-/*
- * The IRQs are mapped as:
- *
- *     ======================  =============   ============    =====
- *     IRQ                     MASK REGISTER   IRQ REGISTER    BIT
- *     ======================  =============   ============    =====
- *     OTMP                    0x0202          0x212           bit 0
- *     VBUS_CONNECT            0x0202          0x212           bit 1
- *     VBUS_DISCONNECT         0x0202          0x212           bit 2
- *     ALARMON_R               0x0202          0x212           bit 3
- *     HOLD_6S                 0x0202          0x212           bit 4
- *     HOLD_1S                 0x0202          0x212           bit 5
- *     POWERKEY_UP             0x0202          0x212           bit 6
- *     POWERKEY_DOWN           0x0202          0x212           bit 7
- *
- *     OCP_SCP_R               0x0203          0x213           bit 0
- *     COUL_R                  0x0203          0x213           bit 1
- *     SIM0_HPD_R              0x0203          0x213           bit 2
- *     SIM0_HPD_F              0x0203          0x213           bit 3
- *     SIM1_HPD_R              0x0203          0x213           bit 4
- *     SIM1_HPD_F              0x0203          0x213           bit 5
- *     ======================  =============   ============    =====
- */
-#define SOC_PMIC_IRQ_MASK_0_ADDR       0x0202
-#define SOC_PMIC_IRQ0_ADDR             0x0212
-
-#define IRQ_MASK_REGISTER(irq_data)    (SOC_PMIC_IRQ_MASK_0_ADDR + \           
                                         irqd_to_hwirq(irq_data) >> 3)
-#define IRQ_MASK_BIT(irq_data)         BIT(irqd_to_hwirq(irq_data) & 0x07)
-
-static const struct mfd_cell hi6421v600_devs[] = {
-       { .name = "hi6421v600-regulator", },
-};
-
-static irqreturn_t hi6421_spmi_irq_handler(int irq, void *priv)
-{
-       struct hi6421_spmi_pmic *ddata = (struct hi6421_spmi_pmic *)priv;
-       unsigned long pending;
-       unsigned int in;
-       int i, offset;
-
-       for (i = 0; i < HISI_IRQ_ARRAY; i++) {
-               regmap_read(ddata->regmap, SOC_PMIC_IRQ0_ADDR + i, &in);
-               pending = HISI_MASK & in;
-               regmap_write(ddata->regmap, SOC_PMIC_IRQ0_ADDR + i, pending);
-
-               if (i == HISI_IRQ_KEY_NUM &&
-                   (pending & HISI_IRQ_KEY_VALUE) == HISI_IRQ_KEY_VALUE) {
-                       generic_handle_irq(ddata->irqs[POWERKEY_DOWN]);
-                       generic_handle_irq(ddata->irqs[POWERKEY_UP]);
-                       pending &= (~HISI_IRQ_KEY_VALUE);
-               }
-
-               if (!pending)
-                       continue;
-
-               for_each_set_bit(offset, &pending, HISI_BITS)
-                       generic_handle_irq(ddata->irqs[offset + i * HISI_BITS]);
-       }
-
-       return IRQ_HANDLED;
-}
-
-static void hi6421_spmi_irq_mask(struct irq_data *d)
-{
-       struct hi6421_spmi_pmic *ddata = irq_data_get_irq_chip_data(d);
-       unsigned long flags;
-       unsigned int data;
-       u32 offset;
-
-       offset = IRQ_MASK_REGISTER(d);
-
-       spin_lock_irqsave(&ddata->lock, flags);
-
-       regmap_read(ddata->regmap, offset, &data);
-       data |= IRQ_MASK_BIT(d);
-       regmap_write(ddata->regmap, offset, data);
-
-       spin_unlock_irqrestore(&ddata->lock, flags);
-}
-
-static void hi6421_spmi_irq_unmask(struct irq_data *d)
-{
-       struct hi6421_spmi_pmic *ddata = irq_data_get_irq_chip_data(d);
-       u32 data, offset;
-       unsigned long flags;
-
-       offset = (irqd_to_hwirq(d) >> 3);
-       offset += SOC_PMIC_IRQ_MASK_0_ADDR;
-
-       spin_lock_irqsave(&ddata->lock, flags);
-
-       regmap_read(ddata->regmap, offset, &data);
-       data &= ~(1 << (irqd_to_hwirq(d) & 0x07));
-       regmap_write(ddata->regmap, offset, data);
-
-       spin_unlock_irqrestore(&ddata->lock, flags);
-}
-
-static struct irq_chip hi6421_spmi_pmu_irqchip = {
-       .name           = "hisi-irq",
-       .irq_mask       = hi6421_spmi_irq_mask,
-       .irq_unmask     = hi6421_spmi_irq_unmask,
-       .irq_disable    = hi6421_spmi_irq_mask,
-       .irq_enable     = hi6421_spmi_irq_unmask,
-};
-
-static int hi6421_spmi_irq_map(struct irq_domain *d, unsigned int virq,
-                              irq_hw_number_t hw)
-{
-       struct hi6421_spmi_pmic *ddata = d->host_data;
-
-       irq_set_chip_and_handler_name(virq, &hi6421_spmi_pmu_irqchip,
-                                     handle_simple_irq, "hisi");
-       irq_set_chip_data(virq, ddata);
-       irq_set_irq_type(virq, IRQ_TYPE_NONE);
-
-       return 0;
-}
-
-static const struct irq_domain_ops hi6421_spmi_domain_ops = {
-       .map    = hi6421_spmi_irq_map,
-       .xlate  = irq_domain_xlate_twocell,
-};
-
-static void hi6421_spmi_pmic_irq_init(struct hi6421_spmi_pmic *ddata)
-{
-       int i;
-       unsigned int pending;
-
-       for (i = 0; i < HISI_IRQ_ARRAY; i++)
-               regmap_write(ddata->regmap, SOC_PMIC_IRQ_MASK_0_ADDR + i,
-                                       HISI_MASK);
-
-       for (i = 0; i < HISI_IRQ_ARRAY; i++) {
-               regmap_read(ddata->regmap, SOC_PMIC_IRQ0_ADDR + i, &pending);
-               regmap_write(ddata->regmap, SOC_PMIC_IRQ0_ADDR + i,
-                            HISI_MASK);
-       }
-}
-
-static const struct regmap_config regmap_config = {
-       .reg_bits               = 16,
-       .val_bits               = HISI_BITS,
-       .max_register           = 0xffff,
-       .fast_io                = true
-};
-
-static int hi6421_spmi_pmic_probe(struct spmi_device *pdev)
-{
-       struct device *dev = &pdev->dev;
-       struct device_node *np = dev->of_node;
-       struct hi6421_spmi_pmic *ddata;
-       unsigned int virq;
-       int ret, i;
-
-       ddata = devm_kzalloc(dev, sizeof(*ddata), GFP_KERNEL);
-       if (!ddata)
-               return -ENOMEM;
-
-       ddata->regmap = devm_regmap_init_spmi_ext(pdev, &regmap_config);
-       if (IS_ERR(ddata->regmap))
-               return PTR_ERR(ddata->regmap);
-
-       spin_lock_init(&ddata->lock);
-
-       ddata->dev = dev;
-
-       ddata->gpio = of_get_gpio(np, 0);
-       if (ddata->gpio < 0)
-               return ddata->gpio;
-
-       if (!gpio_is_valid(ddata->gpio))
-               return -EINVAL;
-
-       ret = devm_gpio_request_one(dev, ddata->gpio, GPIOF_IN, "pmic");
-       if (ret < 0) {
-               dev_err(dev, "Failed to request gpio%d\n", ddata->gpio);
-               return ret;
-       }
-
-       ddata->irq = gpio_to_irq(ddata->gpio);
-
-       hi6421_spmi_pmic_irq_init(ddata);
-
-       ddata->irqs = devm_kzalloc(dev, HISI_IRQ_NUM * sizeof(int), GFP_KERNEL);
-       if (!ddata->irqs)
-               return -ENOMEM;
-
-       ddata->domain = irq_domain_add_simple(np, HISI_IRQ_NUM, 0,
-                                            &hi6421_spmi_domain_ops, ddata);
-       if (!ddata->domain) {
-               dev_err(dev, "Failed to create IRQ domain\n");
-               return -ENODEV;
-       }
-
-       for (i = 0; i < HISI_IRQ_NUM; i++) {
-               virq = irq_create_mapping(ddata->domain, i);
-               if (!virq) {
-                       dev_err(dev, "Failed to map H/W IRQ\n");
-                       return -ENOSPC;
-               }
-               ddata->irqs[i] = virq;
-       }
-
-       ret = request_threaded_irq(ddata->irq, hi6421_spmi_irq_handler, NULL,
-                                  IRQF_TRIGGER_LOW | IRQF_SHARED | 
IRQF_NO_SUSPEND,
-                                  "pmic", ddata);
-       if (ret < 0) {
-               dev_err(dev, "Failed to start IRQ handling thread: error %d\n",
-                       ret);
-               return ret;
-       }
-
-       dev_set_drvdata(&pdev->dev, ddata);
-
-       ret = devm_mfd_add_devices(&pdev->dev, PLATFORM_DEVID_NONE,
-                                  hi6421v600_devs, ARRAY_SIZE(hi6421v600_devs),
-                                  NULL, 0, NULL);
-       if (ret < 0)
-               dev_err(dev, "Failed to add child devices: %d\n", ret);
-
-       return ret;
-}
-
-static void hi6421_spmi_pmic_remove(struct spmi_device *pdev)
-{
-       struct hi6421_spmi_pmic *ddata = dev_get_drvdata(&pdev->dev);
-
-       free_irq(ddata->irq, ddata);
-}
-
-static const struct of_device_id pmic_spmi_id_table[] = {
-       { .compatible = "hisilicon,hi6421-spmi" },
-       { }
-};
-MODULE_DEVICE_TABLE(of, pmic_spmi_id_table);
-
-static struct spmi_driver hi6421_spmi_pmic_driver = {
-       .driver = {
-               .name   = "hi6421-spmi-pmic",
-               .of_match_table = pmic_spmi_id_table,
-       },
-       .probe  = hi6421_spmi_pmic_probe,
-       .remove = hi6421_spmi_pmic_remove,
-};
-module_spmi_driver(hi6421_spmi_pmic_driver);
-
-MODULE_DESCRIPTION("HiSilicon Hi6421v600 SPMI PMIC driver");
-MODULE_LICENSE("GPL v2");
diff --git a/drivers/staging/hikey9xx/hisilicon,hi6421-spmi-pmic.yaml 
b/drivers/staging/hikey9xx/hisilicon,hi6421-spmi-pmic.yaml
deleted file mode 100644
index 3b23ad56b31a..000000000000
--- a/drivers/staging/hikey9xx/hisilicon,hi6421-spmi-pmic.yaml
+++ /dev/null
@@ -1,135 +0,0 @@
-# SPDX-License-Identifier: GPL-2.0
-%YAML 1.2
----
-$id: http://devicetree.org/schemas/mfd/hisilicon,hi6421-spmi-pmic.yaml#
-$schema: http://devicetree.org/meta-schemas/core.yaml#
-
-title: HiSilicon 6421v600 SPMI PMIC
-
-maintainers:
-  - Mauro Carvalho Chehab <mchehab+hua...@kernel.org>
-
-description: |
-  HiSilicon 6421v600 should be connected inside a MIPI System Power Management
-  (SPMI) bus. It provides interrupts and power supply.
-
-  The GPIO and interrupt settings are represented as part of the top-level PMIC
-  node.
-
-  The SPMI controller part is provided by
-  drivers/staging/hikey9xx/hisilicon,hisi-spmi-controller.yaml.
-
-properties:
-  $nodename:
-    pattern: "pmic@[0-9a-f]"
-
-  compatible:
-    const: hisilicon,hi6421v600-spmi
-
-  reg:
-    maxItems: 1
-
-  '#interrupt-cells':
-    const: 2
-
-  interrupt-controller:
-    description:
-      Identify that the PMIC is capable of behaving as an interrupt controller.
-
-  gpios:
-    maxItems: 1
-
-  regulators:
-    type: object
-
-    properties:
-      '#address-cells':
-        const: 1
-
-      '#size-cells':
-        const: 0
-
-    patternProperties:
-      '^ldo[0-9]+@[0-9a-f]$':
-        type: object
-
-        $ref: "/schemas/regulator/regulator.yaml#"
-
-required:
-  - compatible
-  - reg
-  - regulators
-
-additionalProperties: false
-
-examples:
-  - |
-    /* pmic properties */
-
-    pmic: pmic@0 {
-      compatible = "hisilicon,hi6421-spmi";
-      reg = <0 0>;
-
-      #interrupt-cells = <2>;
-      interrupt-controller;
-      gpios = <&gpio28 0 0>;
-
-      regulators {
-        #address-cells = <1>;
-        #size-cells = <0>;
-
-        ldo3: LDO3 {
-          regulator-name = "ldo3";
-          regulator-min-microvolt = <1500000>;
-          regulator-max-microvolt = <2000000>;
-          regulator-boot-on;
-        };
-
-        ldo4: LDO4 {
-          regulator-name = "ldo4";
-          regulator-min-microvolt = <1725000>;
-          regulator-max-microvolt = <1900000>;
-          regulator-boot-on;
-        };
-
-        ldo9: LDO9 {
-          regulator-name = "ldo9";
-          regulator-min-microvolt = <1750000>;
-          regulator-max-microvolt = <3300000>;
-          regulator-boot-on;
-        };
-
-        ldo15: LDO15 {
-          regulator-name = "ldo15";
-          regulator-min-microvolt = <1800000>;
-          regulator-max-microvolt = <3000000>;
-          regulator-always-on;
-        };
-
-        ldo16: LDO16 {
-          regulator-name = "ldo16";
-          regulator-min-microvolt = <1800000>;
-          regulator-max-microvolt = <3000000>;
-          regulator-boot-on;
-        };
-
-        ldo17: LDO17 {
-          regulator-name = "ldo17";
-          regulator-min-microvolt = <2500000>;
-          regulator-max-microvolt = <3300000>;
-        };
-
-        ldo33: LDO33 {
-          regulator-name = "ldo33";
-          regulator-min-microvolt = <2500000>;
-          regulator-max-microvolt = <3300000>;
-          regulator-boot-on;
-        };
-
-        ldo34: LDO34 {
-          regulator-name = "ldo34";
-          regulator-min-microvolt = <2600000>;
-          regulator-max-microvolt = <3300000>;
-        };
-      };
-    };
-- 
2.29.2

_______________________________________________
devel mailing list
de...@linuxdriverproject.org
http://driverdev.linuxdriverproject.org/mailman/listinfo/driverdev-devel

Reply via email to