Fix for QCA953X IRQ enable/disable

QCA953x chipset has same issue as QCA955X/QCA956X, fixed by refered commit below.
Unfortunately 953x is not supported in upstream. Made similar change in 953x.
    refered commit a3e485a8788da7e3c50ed22b98e6094dcf48e54b
    Author: nbd <nbd@3c298f89-4303-0410-b956-a3cf2f4a3e73>
    Date:   Thu Jan 15 12:19:20 2015 +0000

    ar71xx: fix disable_irq() on chained irq handlers

    Signed-off-by: Felix Fietkau <nbd@openwrt.org>

    git-svn-id: svn://svn.openwrt.org/openwrt/trunk@43974 3c298f89-4303-0410-b956-a3cf2f4a3e73

Change-Id: I3433697e1130a7d5ffcb0519e9ed9537b476fae8
diff --git a/arch/mips/ath79/irq.c b/arch/mips/ath79/irq.c
index cea67db..66301b2 100644
--- a/arch/mips/ath79/irq.c
+++ b/arch/mips/ath79/irq.c
@@ -27,6 +27,9 @@
 static void (*ath79_ip2_handler)(void);
 static void (*ath79_ip3_handler)(void);
 
+static struct irq_chip ip2_chip;
+static struct irq_chip ip3_chip;
+
 static void ath79_misc_irq_handler(unsigned int irq, struct irq_desc *desc)
 {
 	void __iomem *base = ath79_reset_base;
@@ -157,35 +160,83 @@
 
 static void qca953x_ip2_irq_dispatch(unsigned int irq, struct irq_desc *desc)
 {
+        u32 status;
+
+        disable_irq_nosync(irq);
+
+        status = ath79_reset_rr(QCA953X_RESET_REG_PCIE_WMAC_INT_STATUS);
+        status &= QCA953X_PCIE_WMAC_INT_PCIE_ALL | QCA953X_PCIE_WMAC_INT_WMAC_ALL;
+
+        if (status == 0) {
+                spurious_interrupt();
+                goto enable;
+        }
+
+        if (status & QCA953X_PCIE_WMAC_INT_PCIE_ALL) {
+                /* TODO: flush DDR? */
+                generic_handle_irq(ATH79_IP2_IRQ(0));
+        }
+
+        if (status & QCA953X_PCIE_WMAC_INT_WMAC_ALL) {
+                /* TODO: flsuh DDR? */
+                generic_handle_irq(ATH79_IP2_IRQ(1));
+        }
+
+enable:
+        enable_irq(irq);
+}
+
+static void qca953x_ip3_irq_dispatch(unsigned int irq, struct irq_desc *desc)
+{
 	u32 status;
 
 	disable_irq_nosync(irq);
 
 	status = ath79_reset_rr(QCA953X_RESET_REG_PCIE_WMAC_INT_STATUS);
+	status &= QCA953X_PCIE_WMAC_INT_PCIE_RCX_ALL|
+			QCA953X_PCIE_WMAC_USB1|
+			QCA953X_PCIE_WMAC_USB2;
 
-	if (status & QCA953X_PCIE_WMAC_INT_PCIE_ALL) {
-		ath79_ddr_wb_flush(QCA953X_DDR_REG_FLUSH_PCIE);
-		generic_handle_irq(ATH79_IP2_IRQ(0));
-	} else if (status & QCA953X_PCIE_WMAC_INT_WMAC_ALL) {
-		ath79_ddr_wb_flush(QCA953X_DDR_REG_FLUSH_WMAC);
-		generic_handle_irq(ATH79_IP2_IRQ(1));
-	} else {
+	if (status == 0) {
 		spurious_interrupt();
+		goto enable;
 	}
 
+	if (status & QCA953X_PCIE_WMAC_USB1) {
+		/* TODO: flush DDR? */
+		generic_handle_irq(ATH79_IP3_IRQ(0));
+	}
+
+	if (status & QCA953X_PCIE_WMAC_USB2) {
+		/* TODO: flsuh DDR? */
+		generic_handle_irq(ATH79_IP3_IRQ(1));
+	}
+
+	if (status & QCA953X_PCIE_WMAC_INT_PCIE_RCX_ALL) {
+		/* TODO: flush DDR? */
+		generic_handle_irq(ATH79_IP3_IRQ(2));
+	}
+
+enable:
 	enable_irq(irq);
 }
 
+
 static void qca953x_irq_init(void)
 {
 	int i;
 
 	for (i = ATH79_IP2_IRQ_BASE;
-	     i < ATH79_IP2_IRQ_BASE + ATH79_IP2_IRQ_COUNT; i++)
-		irq_set_chip_and_handler(i, &dummy_irq_chip,
-					 handle_level_irq);
+		i < ATH79_IP2_IRQ_BASE + ATH79_IP2_IRQ_COUNT; i++)
+		irq_set_chip_and_handler(i, &ip2_chip, handle_level_irq);
 
 	irq_set_chained_handler(ATH79_CPU_IRQ_IP2, qca953x_ip2_irq_dispatch);
+
+	for (i = ATH79_IP3_IRQ_BASE;
+		i < ATH79_IP3_IRQ_BASE + ATH79_IP3_IRQ_COUNT; i++)
+		irq_set_chip_and_handler(i, &ip3_chip, handle_level_irq);
+
+	irq_set_chained_handler(ATH79_CPU_IRQ_IP3, qca953x_ip3_irq_dispatch);
 }
 
 static void qca955x_ip2_irq_dispatch(unsigned int irq, struct irq_desc *desc)
@@ -469,8 +520,35 @@
 	do_IRQ(ATH79_CPU_IRQ_USB);
 }
 
+static void ath79_ip2_disable(struct irq_data *data)
+{
+	disable_irq(ATH79_CPU_IRQ(2));
+}
+
+static void ath79_ip2_enable(struct irq_data *data)
+{
+	enable_irq(ATH79_CPU_IRQ(2));
+}
+
+static void ath79_ip3_disable(struct irq_data *data)
+{
+	disable_irq(ATH79_CPU_IRQ(3));
+}
+
+static void ath79_ip3_enable(struct irq_data *data)
+{
+	enable_irq(ATH79_CPU_IRQ(3));
+}
+
 void __init arch_init_irq(void)
 {
+	ip2_chip = dummy_irq_chip;
+	ip3_chip = dummy_irq_chip;
+	ip2_chip.irq_disable = ath79_ip2_disable;
+	ip2_chip.irq_enable = ath79_ip2_enable;
+	ip3_chip.irq_disable = ath79_ip3_disable;
+	ip3_chip.irq_enable = ath79_ip3_enable;
+
 	if (soc_is_ar71xx()) {
 		ath79_ip2_handler = ar71xx_ip2_handler;
 		ath79_ip3_handler = ar71xx_ip3_handler;
diff --git a/arch/mips/include/asm/mach-ath79/ar71xx_regs.h b/arch/mips/include/asm/mach-ath79/ar71xx_regs.h
index b7d5b83..bb20226 100644
--- a/arch/mips/include/asm/mach-ath79/ar71xx_regs.h
+++ b/arch/mips/include/asm/mach-ath79/ar71xx_regs.h
@@ -688,6 +688,14 @@
 #define QCA953X_PCIE_WMAC_INT_PCIE_RC1		BIT(6)
 #define QCA953X_PCIE_WMAC_INT_PCIE_RC2		BIT(7)
 #define QCA953X_PCIE_WMAC_INT_PCIE_RC3		BIT(8)
+#define QCA953X_PCIE_WMAC_INT_PCIE_RCX		BIT(12)
+#define QCA953X_PCIE_WMAC_INT_PCIE_RCX_INT0	BIT(13)
+#define QCA953X_PCIE_WMAC_INT_PCIE_RCX_INT1	BIT(14)
+#define QCA953X_PCIE_WMAC_INT_PCIE_RCX_INT2	BIT(15)
+#define QCA953X_PCIE_WMAC_INT_PCIE_RCX_INT3	BIT(16)
+#define QCA953X_PCIE_WMAC_USB1			BIT(24)
+#define QCA953X_PCIE_WMAC_USB2			BIT(28)
+
 #define QCA953X_PCIE_WMAC_INT_WMAC_ALL \
 	(QCA953X_PCIE_WMAC_INT_WMAC_MISC | QCA953X_PCIE_WMAC_INT_WMAC_TX | \
 	 QCA953X_PCIE_WMAC_INT_WMAC_RXLP | QCA953X_PCIE_WMAC_INT_WMAC_RXHP)
@@ -696,7 +704,14 @@
 	(QCA953X_PCIE_WMAC_INT_PCIE_RC | QCA953X_PCIE_WMAC_INT_PCIE_RC0 | \
 	 QCA953X_PCIE_WMAC_INT_PCIE_RC1 | QCA953X_PCIE_WMAC_INT_PCIE_RC2 | \
 	 QCA953X_PCIE_WMAC_INT_PCIE_RC3)
- 
+
+#define QCA953X_PCIE_WMAC_INT_PCIE_RCX_ALL	\
+	(QCA953X_PCIE_WMAC_INT_PCIE_RCX |	\
+	 QCA953X_PCIE_WMAC_INT_PCIE_RCX_INT0 |  \
+	 QCA953X_PCIE_WMAC_INT_PCIE_RCX_INT1 |  \
+	 QCA953X_PCIE_WMAC_INT_PCIE_RCX_INT2 |  \
+	 QCA953X_PCIE_WMAC_INT_PCIE_RCX_INT3)
+
 #define QCA955X_EXT_INT_WMAC_MISC		BIT(0)
 #define QCA955X_EXT_INT_WMAC_TX			BIT(1)
 #define QCA955X_EXT_INT_WMAC_RXLP		BIT(2)
diff --git a/arch/mips/include/asm/mach-ath79/irq.h b/arch/mips/include/asm/mach-ath79/irq.h
index be87b54..2e7b026 100644
--- a/arch/mips/include/asm/mach-ath79/irq.h
+++ b/arch/mips/include/asm/mach-ath79/irq.h
@@ -13,6 +13,8 @@
 #define MIPS_CPU_IRQ_BASE	0
 #define NR_IRQS			83
 
+#define ATH79_CPU_IRQ(_x)	(MIPS_CPU_IRQ_BASE + (_x))
+
 #define ATH79_MISC_IRQ_BASE	8
 #define ATH79_MISC_IRQ_COUNT	32
 #define ATH79_MISC_IRQ(_x)	(ATH79_MISC_IRQ_BASE + (_x))