]> git.ipfire.org Git - thirdparty/linux.git/commitdiff
thunderbolt: Prevent XDomain delayed work use-after-free on disconnect
authorMichael Bommarito <michael.bommarito@gmail.com>
Wed, 27 May 2026 11:46:04 +0000 (07:46 -0400)
committerMika Westerberg <mika.westerberg@linux.intel.com>
Thu, 28 May 2026 10:04:43 +0000 (12:04 +0200)
tb_xdp_handle_request() runs on system_wq and queues
xd->state_work via queue_delayed_work() in three request handlers:
PROPERTIES_CHANGED_REQUEST, UUID_REQUEST (via start_handshake),
and LINK_STATE_CHANGE_REQUEST.  Similarly, update_xdomain() queues
xd->properties_changed_work when local properties change.

Concurrently, tb_xdomain_remove() calls stop_handshake() which does
cancel_delayed_work_sync() on both delayed works.  Later,
tb_xdomain_unregister() calls device_unregister() which eventually
frees the xdomain.  Since commit 559c1e1e0134 ("thunderbolt: Run
tb_xdp_handle_request() in system workqueue") moved the request
handler off tb->wq, the handler and the remove path are no longer
serialized.  If queue_delayed_work() executes after
cancel_delayed_work_sync() but before the xdomain is freed, the
delayed work fires on a freed object.

Add xd->removing that tb_xdomain_remove() sets under xd->lock
before calling stop_handshake().  Each external queue site holds
the same lock and checks removing before calling
queue_delayed_work().  This provides the mutual exclusion needed:
either the queue site acquires the lock first and queues work that
the subsequent cancel will see, or the remove path acquires the
lock first and the queue site observes removing == true and skips
the queue.

Fixes: 559c1e1e0134 ("thunderbolt: Run tb_xdp_handle_request() in system workqueue")
Cc: stable@vger.kernel.org
Assisted-by: Claude:claude-opus-4-7
Signed-off-by: Michael Bommarito <michael.bommarito@gmail.com>
Signed-off-by: Mika Westerberg <mika.westerberg@linux.intel.com>
drivers/thunderbolt/xdomain.c
include/linux/thunderbolt.h

index 781d88d06b93583645e2d9bae67ec725553585e2..fe6c5ac703f4dac2a6746cc02286821cca0a4bcd 100644 (file)
@@ -803,9 +803,13 @@ static void tb_xdp_handle_request(struct work_struct *work)
                 * the xdomain related to this connection as well in
                 * case there is a change in services it offers.
                 */
-               if (xd && device_is_registered(&xd->dev))
-                       queue_delayed_work(tb->wq, &xd->state_work,
-                                          msecs_to_jiffies(XDOMAIN_SHORT_TIMEOUT));
+               if (xd) {
+                       mutex_lock(&xd->lock);
+                       if (!xd->removing && device_is_registered(&xd->dev))
+                               queue_delayed_work(tb->wq, &xd->state_work,
+                                                  msecs_to_jiffies(XDOMAIN_SHORT_TIMEOUT));
+                       mutex_unlock(&xd->lock);
+               }
                break;
 
        case UUID_REQUEST_OLD:
@@ -818,8 +822,12 @@ static void tb_xdp_handle_request(struct work_struct *work)
                 * received UUID request from the remote host.
                 */
                if (!ret && xd && xd->state == XDOMAIN_STATE_ERROR) {
-                       dev_dbg(&xd->dev, "restarting handshake\n");
-                       start_handshake(xd);
+                       mutex_lock(&xd->lock);
+                       if (!xd->removing) {
+                               dev_dbg(&xd->dev, "restarting handshake\n");
+                               start_handshake(xd);
+                       }
+                       mutex_unlock(&xd->lock);
                }
                break;
 
@@ -885,9 +893,13 @@ static void tb_xdp_handle_request(struct work_struct *work)
 
                        ret = tb_xdp_link_state_change_response(ctl, route,
                                                                sequence, 0);
-                       xd->target_link_width = lsc->tlw;
-                       queue_delayed_work(tb->wq, &xd->state_work,
-                                          msecs_to_jiffies(XDOMAIN_SHORT_TIMEOUT));
+                       mutex_lock(&xd->lock);
+                       if (!xd->removing) {
+                               xd->target_link_width = lsc->tlw;
+                               queue_delayed_work(tb->wq, &xd->state_work,
+                                                  msecs_to_jiffies(XDOMAIN_SHORT_TIMEOUT));
+                       }
+                       mutex_unlock(&xd->lock);
                } else {
                        tb_xdp_error_response(ctl, route, sequence,
                                              ERROR_NOT_READY);
@@ -971,8 +983,12 @@ static int update_xdomain(struct device *dev, void *data)
 
        xd = tb_to_xdomain(dev);
        if (xd) {
-               queue_delayed_work(xd->tb->wq, &xd->properties_changed_work,
-                                  msecs_to_jiffies(50));
+               mutex_lock(&xd->lock);
+               if (!xd->removing)
+                       queue_delayed_work(xd->tb->wq,
+                                          &xd->properties_changed_work,
+                                          msecs_to_jiffies(50));
+               mutex_unlock(&xd->lock);
        }
 
        return 0;
@@ -2200,6 +2216,11 @@ static int unregister_service(struct device *dev, void *data)
 void tb_xdomain_remove(struct tb_xdomain *xd)
 {
        tb_xdomain_debugfs_remove(xd);
+
+       mutex_lock(&xd->lock);
+       xd->removing = true;
+       mutex_unlock(&xd->lock);
+
        stop_handshake(xd);
        tb_xdomain_link_exit(xd);
 
index b5659f8835171193261d4a8fa83a4fb906674053..feb1af175cfdecb12def02eed36aabbc37220945 100644 (file)
@@ -213,6 +213,8 @@ enum tb_link_width {
  * @link_width: Width of the downstream facing link
  * @link_usb4: Downstream link is USB4
  * @is_unplugged: The XDomain is unplugged
+ * @removing: Set by tb_xdomain_remove() under @lock to prevent
+ *           concurrent delayed work queueing
  * @needs_uuid: If the XDomain does not have @remote_uuid it will be
  *             queried first
  * @service_ids: Used to generate IDs for the services
@@ -262,6 +264,7 @@ struct tb_xdomain {
        enum tb_link_width link_width;
        bool link_usb4;
        bool is_unplugged;
+       bool removing;
        bool needs_uuid;
        struct ida service_ids;
        struct ida in_hopids;