]> git.ipfire.org Git - thirdparty/linux.git/commitdiff
Bluetooth: 6lowpan: fix cyclic locking warning on netdev unregister
authorPauli Virtanen <pav@iki.fi>
Sat, 11 Apr 2026 18:15:09 +0000 (21:15 +0300)
committerLuiz Augusto von Dentz <luiz.von.dentz@intel.com>
Thu, 11 Jun 2026 18:24:37 +0000 (14:24 -0400)
6lowpan.c has theoretically conflicting lock orderings, which lockdep
complains about:

    a) rtnl_lock > hdev->workqueue

    from 6lowpan.c:delete_netdev -> rtnl_lock -> device_del
    -> put_device(parent) -> hci_release_dev -> destroy_workqueue

    b) hdev->workqueue > l2cap_conn->lock > chan->lock > rtnl_lock

    from hci_rx_work -> 6lowpan.c:chan_ready_cb
    -> lowpan_register_netdev, ifup -> rtnl_lock

Actual deadlock appears not possible, as hci_rx_work is disabled and
l2cap_conn flushed already on hdev unregister. Hence, do minimal thing
to make lockdep happy by breaking chain a) by holding hdev refcount
until after netdev put in 6lowpan.c.

Fixes the lockdep complaint:
WARNING: possible circular locking dependency detected.
kworker/0:1/11 is trying to acquire lock:
ffff8880023b3940 ((wq_completion)hci0#2){+.+.}-{0:0}, at: touch_wq_lockdep_map+0x8b/0x130
but task is already holding lock:
ffffffff95e4f9c0 (rtnl_mutex){+.+.}-{4:4}, at: lowpan_unregister_netdev+0xd/0x30
Workqueue: events delete_netdev

Signed-off-by: Pauli Virtanen <pav@iki.fi>
Signed-off-by: Luiz Augusto von Dentz <luiz.von.dentz@intel.com>
net/bluetooth/6lowpan.c

index 960a19b3e26da1b98d66dc4bdc40197768f78bd8..cb1e329d66fd4e5161e274a6c3d32c37dbf6fb81 100644 (file)
@@ -760,13 +760,33 @@ static inline struct l2cap_chan *chan_new_conn_cb(struct l2cap_chan *pchan)
        return chan;
 }
 
+static void unregister_dev(struct lowpan_btle_dev *dev)
+{
+       struct hci_dev *hdev = READ_ONCE(dev->hdev);
+
+       /* If netdev holds last reference to hci_dev (its parent device), this
+        * leads to theoretical cyclic locking on lowpan_unregister_netdev:
+        *
+        * rtnl_lock -> put_device(parent) -> hci_release_dev ->
+        * destroy_workqueue -> hci_rx_work -> l2cap_recv_acldata ->
+        * chan_ready_cb -> ifup -> rtnl_lock
+        *
+        * However, hci_rx_work is disabled in hci_unregister_dev, so this
+        * should not occur. Make lockdep happy by postponing hdev release after
+        * netdev put.
+        */
+       hci_dev_hold(hdev);
+       lowpan_unregister_netdev(dev->netdev);
+       hci_dev_put(hdev);
+}
+
 static void delete_netdev(struct work_struct *work)
 {
        struct lowpan_btle_dev *entry = container_of(work,
                                                     struct lowpan_btle_dev,
                                                     delete_netdev);
 
-       lowpan_unregister_netdev(entry->netdev);
+       unregister_dev(entry);
 
        /* The entry pointer is deleted by the netdev destructor. */
 }
@@ -1252,6 +1272,7 @@ static void disconnect_devices(void)
                        break;
 
                new_dev->netdev = entry->netdev;
+               new_dev->hdev = entry->hdev;
                INIT_LIST_HEAD(&new_dev->list);
 
                list_add_rcu(&new_dev->list, &devices);
@@ -1263,7 +1284,7 @@ static void disconnect_devices(void)
                ifdown(entry->netdev);
                BT_DBG("Unregistering netdev %s %p",
                       entry->netdev->name, entry->netdev);
-               lowpan_unregister_netdev(entry->netdev);
+               unregister_dev(entry);
                kfree(entry);
        }
 }