]> git.ipfire.org Git - thirdparty/kernel/stable.git/commitdiff
PCI: xilinx-nwl: Clean up clock on probe failure/removal
authorSean Anderson <sean.anderson@linux.dev>
Fri, 31 May 2024 16:13:35 +0000 (12:13 -0400)
committerGreg Kroah-Hartman <gregkh@linuxfoundation.org>
Fri, 4 Oct 2024 14:29:30 +0000 (16:29 +0200)
[ Upstream commit cfd67903977b13f63340a4eb5a1cc890994f2c62 ]

Make sure we turn off the clock on probe failure and device removal.

Fixes: de0a01f52966 ("PCI: xilinx-nwl: Enable the clock through CCF")
Link: https://lore.kernel.org/r/20240531161337.864994-6-sean.anderson@linux.dev
Signed-off-by: Sean Anderson <sean.anderson@linux.dev>
Signed-off-by: Bjorn Helgaas <bhelgaas@google.com>
Signed-off-by: Sasha Levin <sashal@kernel.org>
drivers/pci/controller/pcie-xilinx-nwl.c

index 7cf05431d54bc32c88ed28c287ca388d063f52da..60afbb51511c3fc059247674c788433b5961d15e 100644 (file)
@@ -790,6 +790,7 @@ static int nwl_pcie_probe(struct platform_device *pdev)
                return -ENODEV;
 
        pcie = pci_host_bridge_priv(bridge);
+       platform_set_drvdata(pdev, pcie);
 
        pcie->dev = dev;
        pcie->ecam_value = NWL_ECAM_VALUE_DEFAULT;
@@ -813,13 +814,13 @@ static int nwl_pcie_probe(struct platform_device *pdev)
        err = nwl_pcie_bridge_init(pcie);
        if (err) {
                dev_err(dev, "HW Initialization failed\n");
-               return err;
+               goto err_clk;
        }
 
        err = nwl_pcie_init_irq_domain(pcie);
        if (err) {
                dev_err(dev, "Failed creating IRQ Domain\n");
-               return err;
+               goto err_clk;
        }
 
        bridge->sysdata = pcie;
@@ -829,11 +830,24 @@ static int nwl_pcie_probe(struct platform_device *pdev)
                err = nwl_pcie_enable_msi(pcie);
                if (err < 0) {
                        dev_err(dev, "failed to enable MSI support: %d\n", err);
-                       return err;
+                       goto err_clk;
                }
        }
 
-       return pci_host_probe(bridge);
+       err = pci_host_probe(bridge);
+       if (!err)
+               return 0;
+
+err_clk:
+       clk_disable_unprepare(pcie->clk);
+       return err;
+}
+
+static void nwl_pcie_remove(struct platform_device *pdev)
+{
+       struct nwl_pcie *pcie = platform_get_drvdata(pdev);
+
+       clk_disable_unprepare(pcie->clk);
 }
 
 static struct platform_driver nwl_pcie_driver = {
@@ -843,5 +857,6 @@ static struct platform_driver nwl_pcie_driver = {
                .of_match_table = nwl_pcie_of_match,
        },
        .probe = nwl_pcie_probe,
+       .remove_new = nwl_pcie_remove,
 };
 builtin_platform_driver(nwl_pcie_driver);