]> git.ipfire.org Git - thirdparty/linux.git/commitdiff
PCI: altera: Fix resource leaks on probe failure
authorMahesh Vaidya <mahesh.vaidya@altera.com>
Thu, 30 Apr 2026 20:43:30 +0000 (13:43 -0700)
committerManivannan Sadhasivam <mani@kernel.org>
Fri, 15 May 2026 17:34:29 +0000 (23:04 +0530)
The chained IRQ handler is set during probe, but is only removed during the
driver remove(). If pci_host_probe() fails, the handler and INTx IRQ
domain remain set even though the devm-managed host bridge storage
containing struct altera_pcie will be released, leaving the handler with
a stale data pointer.

Interrupts are also enabled before pci_host_probe() is called. If probe
fails after that point, the controller interrupt source should be disabled
before the chained handler and INTx domain are removed.

So set the chained handler only after the INTx domain has been created.
Disable controller interrupts during IRQ teardown, and tear the IRQ setup
down if pci_host_probe() fails.

Fixes: c63aed7334c2 ("PCI: altera: Use pci_host_probe() to register host")
Signed-off-by: Mahesh Vaidya <mahesh.vaidya@altera.com>
[mani: commit log]
Signed-off-by: Manivannan Sadhasivam <mani@kernel.org>
Reviewed-by: Subhransu S. Prusty <subhransu.sekhar.prusty@altera.com>
Cc: stable@vger.kernel.org
Link: https://patch.msgid.link/20260430204330.3121003-3-mahesh.vaidya@altera.com
drivers/pci/controller/pcie-altera.c

index 3d3519b8d88f0bf931bf90595a3518098fb1dc78..76f3823d961395967fc705a267ba75234d39c15c 100644 (file)
@@ -864,8 +864,23 @@ static int altera_pcie_init_irq_domain(struct altera_pcie *pcie)
        return 0;
 }
 
+static void altera_pcie_disable_irq(struct altera_pcie *pcie)
+{
+       if (pcie->pcie_data->version == ALTERA_PCIE_V1 ||
+           pcie->pcie_data->version == ALTERA_PCIE_V2) {
+               /* Disable all P2A interrupts */
+               cra_writel(pcie, 0, P2A_INT_ENABLE);
+       } else if (pcie->pcie_data->version == ALTERA_PCIE_V3) {
+               /* Disable port-level interrupts (CFG_AER, etc.) */
+               writel(0, pcie->hip_base +
+                         pcie->pcie_data->port_conf_offset +
+                         pcie->pcie_data->port_irq_enable_offset);
+       }
+}
+
 static void altera_pcie_irq_teardown(struct altera_pcie *pcie)
 {
+       altera_pcie_disable_irq(pcie);
        irq_set_chained_handler_and_data(pcie->irq, NULL, NULL);
        irq_domain_remove(pcie->irq_domain);
 }
@@ -890,7 +905,6 @@ static int altera_pcie_parse_dt(struct altera_pcie *pcie)
        if (pcie->irq < 0)
                return pcie->irq;
 
-       irq_set_chained_handler_and_data(pcie->irq, pcie->pcie_data->ops->rp_isr, pcie);
        return 0;
 }
 
@@ -1019,6 +1033,14 @@ static int altera_pcie_probe(struct platform_device *pdev)
                return ret;
        }
 
+       /*
+        * The chained handler uses pcie->irq_domain, so set it only after the
+        * INTx domain has been created.
+        */
+       irq_set_chained_handler_and_data(pcie->irq,
+                                        pcie->pcie_data->ops->rp_isr,
+                                        pcie);
+
        if (pcie->pcie_data->version == ALTERA_PCIE_V1 ||
            pcie->pcie_data->version == ALTERA_PCIE_V2) {
                /* clear all interrupts */
@@ -1036,7 +1058,16 @@ static int altera_pcie_probe(struct platform_device *pdev)
        bridge->busnr = pcie->root_bus_nr;
        bridge->ops = &altera_pcie_ops;
 
-       return pci_host_probe(bridge);
+       ret = pci_host_probe(bridge);
+       if (ret)
+               goto err_teardown_irq;
+
+       return 0;
+
+err_teardown_irq:
+       altera_pcie_irq_teardown(pcie);
+
+       return ret;
 }
 
 static void altera_pcie_remove(struct platform_device *pdev)