#include <linux/types.h>
 
 #define PDC_MAX_GPIO_IRQS      256
+#define PDC_DRV_OFFSET         0x10000
 
 /* Valid only on HW version < 3.2 */
 #define IRQ_ENABLE_BANK                0x10
+#define IRQ_ENABLE_BANK_MAX    (IRQ_ENABLE_BANK + BITS_TO_BYTES(PDC_MAX_GPIO_IRQS))
 #define IRQ_i_CFG              0x110
 
 /* Valid only on HW version >= 3.2 */
 
 static DEFINE_RAW_SPINLOCK(pdc_lock);
 static void __iomem *pdc_base;
+static void __iomem *pdc_prev_base;
 static struct pdc_pin_region *pdc_region;
 static int pdc_region_cnt;
 static unsigned int pdc_version;
+static bool pdc_x1e_quirk;
+
+static void pdc_base_reg_write(void __iomem *base, int reg, u32 i, u32 val)
+{
+       writel_relaxed(val, base + reg + i * sizeof(u32));
+}
 
 static void pdc_reg_write(int reg, u32 i, u32 val)
 {
-       writel_relaxed(val, pdc_base + reg + i * sizeof(u32));
+       pdc_base_reg_write(pdc_base, reg, i, val);
 }
 
 static u32 pdc_reg_read(int reg, u32 i)
        return readl_relaxed(pdc_base + reg + i * sizeof(u32));
 }
 
+static void pdc_x1e_irq_enable_write(u32 bank, u32 enable)
+{
+       void __iomem *base;
+
+       /* Remap the write access to work around a hardware bug on X1E */
+       switch (bank) {
+       case 0 ... 1:
+               /* Use previous DRV (client) region and shift to bank 3-4 */
+               base = pdc_prev_base;
+               bank += 3;
+               break;
+       case 2 ... 4:
+               /* Use our own region and shift to bank 0-2 */
+               base = pdc_base;
+               bank -= 2;
+               break;
+       case 5:
+               /* No fixup required for bank 5 */
+               base = pdc_base;
+               break;
+       default:
+               WARN_ON(1);
+               return;
+       }
+
+       pdc_base_reg_write(base, IRQ_ENABLE_BANK, bank, enable);
+}
+
 static void __pdc_enable_intr(int pin_out, bool on)
 {
        unsigned long enable;
 
                enable = pdc_reg_read(IRQ_ENABLE_BANK, index);
                __assign_bit(mask, &enable, on);
-               pdc_reg_write(IRQ_ENABLE_BANK, index, enable);
+
+               if (pdc_x1e_quirk)
+                       pdc_x1e_irq_enable_write(index, enable);
+               else
+                       pdc_reg_write(IRQ_ENABLE_BANK, index, enable);
        } else {
                enable = pdc_reg_read(IRQ_i_CFG, pin_out);
                __assign_bit(IRQ_i_CFG_IRQ_ENABLE, &enable, on);
        if (res_size > resource_size(&res))
                pr_warn("%pOF: invalid reg size, please fix DT\n", node);
 
+       /*
+        * PDC has multiple DRV regions, each one provides the same set of
+        * registers for a particular client in the system. Due to a hardware
+        * bug on X1E, some writes to the IRQ_ENABLE_BANK register must be
+        * issued inside the previous region. This region belongs to
+        * a different client and is not described in the device tree. Map the
+        * region with the expected offset to preserve support for old DTs.
+        */
+       if (of_device_is_compatible(node, "qcom,x1e80100-pdc")) {
+               pdc_prev_base = ioremap(res.start - PDC_DRV_OFFSET, IRQ_ENABLE_BANK_MAX);
+               if (!pdc_prev_base) {
+                       pr_err("%pOF: unable to map previous PDC DRV region\n", node);
+                       return -ENXIO;
+               }
+
+               pdc_x1e_quirk = true;
+       }
+
        pdc_base = ioremap(res.start, res_size);
        if (!pdc_base) {
                pr_err("%pOF: unable to map PDC registers\n", node);
-               return -ENXIO;
+               ret = -ENXIO;
+               goto fail;
        }
 
        pdc_version = pdc_reg_read(PDC_VERSION_REG, 0);
 fail:
        kfree(pdc_region);
        iounmap(pdc_base);
+       iounmap(pdc_prev_base);
        return ret;
 }