]> git.ipfire.org Git - thirdparty/kernel/linux.git/commitdiff
thunderbolt: Wait for tb_domain_release() to complete when driver is removed
authorMika Westerberg <mika.westerberg@linux.intel.com>
Wed, 19 Nov 2025 10:53:58 +0000 (12:53 +0200)
committerMika Westerberg <mika.westerberg@linux.intel.com>
Tue, 5 May 2026 11:53:46 +0000 (13:53 +0200)
We should not call nhi_shutdown() before the domain structure and the
control channel rings are completely released. Otherwise we might
release resources like the nhi->msix_ida that are still referenced in
tb_domain_release(). For this reason wait for the tb_domain_release() to
be completed before continuing to nhi_shutdown() and eventually
releasing of the rest of the data structures.

Signed-off-by: Mika Westerberg <mika.westerberg@linux.intel.com>
drivers/thunderbolt/domain.c
drivers/thunderbolt/nhi.c
include/linux/thunderbolt.h

index 317780b999929368fbb2378ed86bf61d388acd66..df4d7dd45adf322d9b13c7b26ea6fcd00ba18ae6 100644 (file)
@@ -319,12 +319,15 @@ const struct bus_type tb_bus_type = {
 static void tb_domain_release(struct device *dev)
 {
        struct tb *tb = container_of(dev, struct tb, dev);
+       struct tb_nhi *nhi = tb->nhi;
 
        tb_ctl_free(tb->ctl);
        destroy_workqueue(tb->wq);
        ida_free(&tb_domain_ida, tb->index);
        mutex_destroy(&tb->lock);
        kfree(tb);
+
+       complete(&nhi->domain_released);
 }
 
 const struct device_type tb_domain_type = {
index 2bb2e79ca3cb35df1565c35ad22c0f52a41454f9..1a20516730677f1b0206e71b7cc2586a35c6ea43 100644 (file)
@@ -1401,6 +1401,8 @@ static int nhi_probe(struct pci_dev *pdev, const struct pci_device_id *id)
 
        dev_dbg(dev, "NHI initialized, starting thunderbolt\n");
 
+       init_completion(&nhi->domain_released);
+
        res = tb_domain_add(tb, host_reset);
        if (res) {
                /*
@@ -1408,6 +1410,7 @@ static int nhi_probe(struct pci_dev *pdev, const struct pci_device_id *id)
                 * activated. Do a proper shutdown.
                 */
                tb_domain_put(tb);
+               wait_for_completion(&nhi->domain_released);
                nhi_shutdown(nhi);
                return res;
        }
@@ -1433,6 +1436,7 @@ static void nhi_remove(struct pci_dev *pdev)
        pm_runtime_forbid(&pdev->dev);
 
        tb_domain_remove(tb);
+       wait_for_completion(&nhi->domain_released);
        nhi_shutdown(nhi);
 }
 
index 0ba112175bb39dac8d6756659ad6b6fe1bb6552b..a5ef7100a6d3fb30c2f86efe1d646cb25f24c1ac 100644 (file)
@@ -493,6 +493,7 @@ static inline struct tb_xdomain *tb_service_parent(struct tb_service *svc)
  *                 MSI-X is used.
  * @hop_count: Number of rings (end point hops) supported by NHI.
  * @quirks: NHI specific quirks if any
+ * @domain_released: Completed when domain has been fully released
  */
 struct tb_nhi {
        spinlock_t lock;
@@ -507,6 +508,7 @@ struct tb_nhi {
        struct work_struct interrupt_work;
        u32 hop_count;
        unsigned long quirks;
+       struct completion domain_released;
 };
 
 /**