]> git.ipfire.org Git - thirdparty/kernel/linux.git/commitdiff
HID: hid-lenovo-go: cancel cfg_setup work in hid_go_cfg_remove()
authorManish Khadka <maskmemanish@gmail.com>
Fri, 15 May 2026 17:45:11 +0000 (23:30 +0545)
committerJiri Kosina <jkosina@suse.com>
Wed, 3 Jun 2026 12:18:18 +0000 (14:18 +0200)
hid_go_cfg_probe() initialises drvdata.go_cfg_setup and schedules it
to run 2 ms later:

    INIT_DELAYED_WORK(&drvdata.go_cfg_setup, &cfg_setup);
    schedule_delayed_work(&drvdata.go_cfg_setup, msecs_to_jiffies(2));

cfg_setup() dereferences drvdata.hdev to issue MCU command requests.
hid_go_cfg_remove() tears down sysfs and stops the HID device, but
never drains the delayed work.  If the device is unbound within the
2 ms scheduling delay (a probe failure rolling back via remove, or a
fast rmmod after probe), the work fires after hid_destroy_device()
has dropped its reference and released the underlying hdev struct,
leaving cfg_setup() with a stale drvdata.hdev pointer.

Mirror the sibling driver hid-lenovo-go-s.c, whose hid_gos_cfg_remove()
already calls cancel_delayed_work_sync() on its analogous work, and
drain go_cfg_setup at the top of hid_go_cfg_remove().  The cancel
must come before guard(mutex)(&drvdata.cfg_mutex) because cfg_setup()
acquires that mutex; reversing the order would deadlock.

Fixes: d69ccfcbc955 ("HID: hid-lenovo-go: Add Lenovo Legion Go Series HID Driver")
Cc: stable@vger.kernel.org
Signed-off-by: Manish Khadka <maskmemanish@gmail.com>
Signed-off-by: Jiri Kosina <jkosina@suse.com>
drivers/hid/hid-lenovo-go.c

index e0c9d5ec9451bf92a2fec83993702d4d76d0eff2..318b1152ff8ba827967fa9bfe71a1cdc55829c7b 100644 (file)
@@ -2405,6 +2405,15 @@ static int hid_go_cfg_probe(struct hid_device *hdev,
 
 static void hid_go_cfg_remove(struct hid_device *hdev)
 {
+       /*
+        * cfg_setup is scheduled from hid_go_cfg_probe() with a 2 ms delay
+        * and dereferences drvdata.hdev.  Drain it here before tearing
+        * down so the workqueue cannot run after hid_destroy_device()'s
+        * put_device() has released the underlying hdev and dereference
+        * a stale drvdata.hdev pointer.
+        */
+       cancel_delayed_work_sync(&drvdata.go_cfg_setup);
+
        guard(mutex)(&drvdata.cfg_mutex);
        sysfs_remove_groups(&hdev->dev.kobj, top_level_attr_groups);
        hid_hw_close(hdev);