Skip to content

Commit 680ee41

Browse files
jwrdegoedeBenjamin Tissoires
authored andcommitted
HID: logitech-hidpp: Fix connect event race
There is a connect event race in hidpp_probe() in these 2 lines: connected = hidpp_root_get_protocol_version(hidpp) == 0; atomic_set(&hidpp->connected, connected); Specifically the following can happen: 1. This line from hidpp_probe() is executed: connected = hidpp_root_get_protocol_version(hidpp) == 0; and sets connected to false; 2. A connect-event packet is received and does: atomic_set(&hidpp->connected, true); 3. The next line from hidpp_probe() is executed: atomic_set(&hidpp->connected, connected); and sets the atomic_t back to 0 again. 4. hidpp_connect_event() runs and sees the connected device as disconnected because of this. To fix this make hidpp_connect_event() query the connection status of the device itself instead of having it rely on possibly stale data cached in struct hidpp_device. This series has been tested on the following devices: Logitech Bluetooth Laser Travel Mouse (bluetooth, HID++ 1.0) Logitech M720 Triathlon (bluetooth, HID++ 4.5) Logitech M720 Triathlon (unifying, HID++ 4.5) Logitech K400 Pro (unifying, HID++ 4.1) Logitech K270 (eQUAD nano Lite, HID++ 2.0) Logitech M185 (eQUAD nano Lite, HID++ 4.5) Logitech LX501 keyboard (27 Mhz, HID++ builtin scroll-wheel, HID++ 1.0) Logitech M-RAZ105 mouse (27 Mhz, HID++ extra mouse buttons, HID++ 1.0) And by bentiss: Logitech Touchpad T650 (unifying) Logitech Touchpad T651 (bluetooth) Logitech MX Master 3B (BLE) Logitech G403 (plain USB / Gaming receiver) Signed-off-by: Hans de Goede <[email protected]> Link: https://lore.kernel.org/r/[email protected] Signed-off-by: Benjamin Tissoires <[email protected]>
1 parent bb17b2c commit 680ee41

File tree

1 file changed

+5
-20
lines changed

1 file changed

+5
-20
lines changed

drivers/hid/hid-logitech-hidpp.c

Lines changed: 5 additions & 20 deletions
Original file line numberDiff line numberDiff line change
@@ -194,7 +194,6 @@ struct hidpp_device {
194194

195195
struct work_struct work;
196196
struct kfifo delayed_work_fifo;
197-
atomic_t connected;
198197
struct input_dev *delayed_input;
199198

200199
unsigned long quirks;
@@ -3893,8 +3892,6 @@ static int hidpp_raw_hidpp_event(struct hidpp_device *hidpp, u8 *data,
38933892
}
38943893

38953894
if (unlikely(hidpp_report_is_connect_event(hidpp, report))) {
3896-
atomic_set(&hidpp->connected,
3897-
!(report->rap.params[0] & (1 << 6)));
38983895
if (schedule_work(&hidpp->work) == 0)
38993896
dbg_hid("%s: connect event already queued\n", __func__);
39003897
return 1;
@@ -4189,12 +4186,14 @@ static struct input_dev *hidpp_allocate_input(struct hid_device *hdev)
41894186
static void hidpp_connect_event(struct hidpp_device *hidpp)
41904187
{
41914188
struct hid_device *hdev = hidpp->hid_dev;
4192-
int ret = 0;
4193-
bool connected = atomic_read(&hidpp->connected);
41944189
struct input_dev *input;
41954190
char *name, *devm_name;
4191+
int ret;
41964192

4197-
if (!connected) {
4193+
/* Get device version to check if it is connected */
4194+
ret = hidpp_root_get_protocol_version(hidpp);
4195+
if (ret) {
4196+
hid_info(hidpp->hid_dev, "Disconnected\n");
41984197
if (hidpp->battery.ps) {
41994198
hidpp->battery.online = false;
42004199
hidpp->battery.status = POWER_SUPPLY_STATUS_UNKNOWN;
@@ -4236,16 +4235,6 @@ static void hidpp_connect_event(struct hidpp_device *hidpp)
42364235
return;
42374236
}
42384237

4239-
/* the device is already connected, we can ask for its name and
4240-
* protocol */
4241-
if (!hidpp->protocol_major) {
4242-
ret = hidpp_root_get_protocol_version(hidpp);
4243-
if (ret) {
4244-
hid_err(hdev, "Can not get the protocol version.\n");
4245-
return;
4246-
}
4247-
}
4248-
42494238
if (hidpp->protocol_major >= 2) {
42504239
u8 feature_index;
42514240

@@ -4395,7 +4384,6 @@ static int hidpp_probe(struct hid_device *hdev, const struct hid_device_id *id)
43954384
{
43964385
struct hidpp_device *hidpp;
43974386
int ret;
4398-
bool connected;
43994387
unsigned int connect_mask = HID_CONNECT_DEFAULT;
44004388

44014389
/* report_fixup needs drvdata to be set before we call hid_parse */
@@ -4485,9 +4473,6 @@ static int hidpp_probe(struct hid_device *hdev, const struct hid_device_id *id)
44854473
else
44864474
hidpp_non_unifying_init(hidpp);
44874475

4488-
connected = hidpp_root_get_protocol_version(hidpp) == 0;
4489-
atomic_set(&hidpp->connected, connected);
4490-
44914476
schedule_work(&hidpp->work);
44924477
flush_work(&hidpp->work);
44934478

0 commit comments

Comments
 (0)