Commit 5c24e4e9 authored by Linus Torvalds's avatar Linus Torvalds

Merge tag 'hid-for-linus-2024020101' of git://git.kernel.org/pub/scm/linux/kernel/git/hid/hid

Pull HID fixes from Benjamin Tissoires:

 - cleanups in the error path in hid-steam (Dan Carpenter)

 - fixes for Wacom tablets selftests that sneaked in while the CI was
   taking a break during the year end holidays (Benjamin Tissoires)

 - null pointer check in nvidia-shield (Kunwu Chan)

 - memory leak fix in hidraw (Su Hui)

 - another null pointer fix in i2c-hid-of (Johan Hovold)

 - another memory leak fix in HID-BPF this time, as well as a double
   fdget() fix reported by Dan Carpenter (Benjamin Tissoires)

 - fix for Cirque touchpad when they go on suspend (Kai-Heng Feng)

 - new device ID in hid-logitech-hidpp: "Logitech G Pro X SuperLight 2"
   (Jiri Kosina)

* tag 'hid-for-linus-2024020101' of git://git.kernel.org/pub/scm/linux/kernel/git/hid/hid:
  HID: bpf: use __bpf_kfunc instead of noinline
  HID: bpf: actually free hdev memory after attaching a HID-BPF program
  HID: bpf: remove double fdget()
  HID: i2c-hid-of: fix NULL-deref on failed power up
  HID: hidraw: fix a problem of memory leak in hidraw_release()
  HID: i2c-hid: Skip SET_POWER SLEEP for Cirque touchpad on system suspend
  HID: nvidia-shield: Add missing null pointer checks to LED initialization
  HID: logitech-hidpp: add support for Logitech G Pro X Superlight 2
  selftests/hid: wacom: fix confidence tests
  HID: hid-steam: Fix cleanup in probe()
  HID: hid-steam: remove pointless error message
parents f6cdd897 764ad6b0
...@@ -143,6 +143,9 @@ u8 *call_hid_bpf_rdesc_fixup(struct hid_device *hdev, u8 *rdesc, unsigned int *s ...@@ -143,6 +143,9 @@ u8 *call_hid_bpf_rdesc_fixup(struct hid_device *hdev, u8 *rdesc, unsigned int *s
} }
EXPORT_SYMBOL_GPL(call_hid_bpf_rdesc_fixup); EXPORT_SYMBOL_GPL(call_hid_bpf_rdesc_fixup);
/* Disables missing prototype warnings */
__bpf_kfunc_start_defs();
/** /**
* hid_bpf_get_data - Get the kernel memory pointer associated with the context @ctx * hid_bpf_get_data - Get the kernel memory pointer associated with the context @ctx
* *
...@@ -152,7 +155,7 @@ EXPORT_SYMBOL_GPL(call_hid_bpf_rdesc_fixup); ...@@ -152,7 +155,7 @@ EXPORT_SYMBOL_GPL(call_hid_bpf_rdesc_fixup);
* *
* @returns %NULL on error, an %__u8 memory pointer on success * @returns %NULL on error, an %__u8 memory pointer on success
*/ */
noinline __u8 * __bpf_kfunc __u8 *
hid_bpf_get_data(struct hid_bpf_ctx *ctx, unsigned int offset, const size_t rdwr_buf_size) hid_bpf_get_data(struct hid_bpf_ctx *ctx, unsigned int offset, const size_t rdwr_buf_size)
{ {
struct hid_bpf_ctx_kern *ctx_kern; struct hid_bpf_ctx_kern *ctx_kern;
...@@ -167,6 +170,7 @@ hid_bpf_get_data(struct hid_bpf_ctx *ctx, unsigned int offset, const size_t rdwr ...@@ -167,6 +170,7 @@ hid_bpf_get_data(struct hid_bpf_ctx *ctx, unsigned int offset, const size_t rdwr
return ctx_kern->data + offset; return ctx_kern->data + offset;
} }
__bpf_kfunc_end_defs();
/* /*
* The following set contains all functions we agree BPF programs * The following set contains all functions we agree BPF programs
...@@ -241,6 +245,42 @@ int hid_bpf_reconnect(struct hid_device *hdev) ...@@ -241,6 +245,42 @@ int hid_bpf_reconnect(struct hid_device *hdev)
return 0; return 0;
} }
static int do_hid_bpf_attach_prog(struct hid_device *hdev, int prog_fd, struct bpf_prog *prog,
__u32 flags)
{
int fd, err, prog_type;
prog_type = hid_bpf_get_prog_attach_type(prog);
if (prog_type < 0)
return prog_type;
if (prog_type >= HID_BPF_PROG_TYPE_MAX)
return -EINVAL;
if (prog_type == HID_BPF_PROG_TYPE_DEVICE_EVENT) {
err = hid_bpf_allocate_event_data(hdev);
if (err)
return err;
}
fd = __hid_bpf_attach_prog(hdev, prog_type, prog_fd, prog, flags);
if (fd < 0)
return fd;
if (prog_type == HID_BPF_PROG_TYPE_RDESC_FIXUP) {
err = hid_bpf_reconnect(hdev);
if (err) {
close_fd(fd);
return err;
}
}
return fd;
}
/* Disables missing prototype warnings */
__bpf_kfunc_start_defs();
/** /**
* hid_bpf_attach_prog - Attach the given @prog_fd to the given HID device * hid_bpf_attach_prog - Attach the given @prog_fd to the given HID device
* *
...@@ -253,22 +293,17 @@ int hid_bpf_reconnect(struct hid_device *hdev) ...@@ -253,22 +293,17 @@ int hid_bpf_reconnect(struct hid_device *hdev)
* is pinned to the BPF file system). * is pinned to the BPF file system).
*/ */
/* called from syscall */ /* called from syscall */
noinline int __bpf_kfunc int
hid_bpf_attach_prog(unsigned int hid_id, int prog_fd, __u32 flags) hid_bpf_attach_prog(unsigned int hid_id, int prog_fd, __u32 flags)
{ {
struct hid_device *hdev; struct hid_device *hdev;
struct bpf_prog *prog;
struct device *dev; struct device *dev;
int fd, err, prog_type = hid_bpf_get_prog_attach_type(prog_fd); int err, fd;
if (!hid_bpf_ops) if (!hid_bpf_ops)
return -EINVAL; return -EINVAL;
if (prog_type < 0)
return prog_type;
if (prog_type >= HID_BPF_PROG_TYPE_MAX)
return -EINVAL;
if ((flags & ~HID_BPF_FLAG_MASK)) if ((flags & ~HID_BPF_FLAG_MASK))
return -EINVAL; return -EINVAL;
...@@ -278,25 +313,29 @@ hid_bpf_attach_prog(unsigned int hid_id, int prog_fd, __u32 flags) ...@@ -278,25 +313,29 @@ hid_bpf_attach_prog(unsigned int hid_id, int prog_fd, __u32 flags)
hdev = to_hid_device(dev); hdev = to_hid_device(dev);
if (prog_type == HID_BPF_PROG_TYPE_DEVICE_EVENT) { /*
err = hid_bpf_allocate_event_data(hdev); * take a ref on the prog itself, it will be released
if (err) * on errors or when it'll be detached
return err; */
prog = bpf_prog_get(prog_fd);
if (IS_ERR(prog)) {
err = PTR_ERR(prog);
goto out_dev_put;
} }
fd = __hid_bpf_attach_prog(hdev, prog_type, prog_fd, flags); fd = do_hid_bpf_attach_prog(hdev, prog_fd, prog, flags);
if (fd < 0) if (fd < 0) {
return fd; err = fd;
goto out_prog_put;
if (prog_type == HID_BPF_PROG_TYPE_RDESC_FIXUP) {
err = hid_bpf_reconnect(hdev);
if (err) {
close_fd(fd);
return err;
}
} }
return fd; return fd;
out_prog_put:
bpf_prog_put(prog);
out_dev_put:
put_device(dev);
return err;
} }
/** /**
...@@ -306,7 +345,7 @@ hid_bpf_attach_prog(unsigned int hid_id, int prog_fd, __u32 flags) ...@@ -306,7 +345,7 @@ hid_bpf_attach_prog(unsigned int hid_id, int prog_fd, __u32 flags)
* *
* @returns A pointer to &struct hid_bpf_ctx on success, %NULL on error. * @returns A pointer to &struct hid_bpf_ctx on success, %NULL on error.
*/ */
noinline struct hid_bpf_ctx * __bpf_kfunc struct hid_bpf_ctx *
hid_bpf_allocate_context(unsigned int hid_id) hid_bpf_allocate_context(unsigned int hid_id)
{ {
struct hid_device *hdev; struct hid_device *hdev;
...@@ -323,8 +362,10 @@ hid_bpf_allocate_context(unsigned int hid_id) ...@@ -323,8 +362,10 @@ hid_bpf_allocate_context(unsigned int hid_id)
hdev = to_hid_device(dev); hdev = to_hid_device(dev);
ctx_kern = kzalloc(sizeof(*ctx_kern), GFP_KERNEL); ctx_kern = kzalloc(sizeof(*ctx_kern), GFP_KERNEL);
if (!ctx_kern) if (!ctx_kern) {
put_device(dev);
return NULL; return NULL;
}
ctx_kern->ctx.hid = hdev; ctx_kern->ctx.hid = hdev;
...@@ -337,14 +378,19 @@ hid_bpf_allocate_context(unsigned int hid_id) ...@@ -337,14 +378,19 @@ hid_bpf_allocate_context(unsigned int hid_id)
* @ctx: the HID-BPF context to release * @ctx: the HID-BPF context to release
* *
*/ */
noinline void __bpf_kfunc void
hid_bpf_release_context(struct hid_bpf_ctx *ctx) hid_bpf_release_context(struct hid_bpf_ctx *ctx)
{ {
struct hid_bpf_ctx_kern *ctx_kern; struct hid_bpf_ctx_kern *ctx_kern;
struct hid_device *hid;
ctx_kern = container_of(ctx, struct hid_bpf_ctx_kern, ctx); ctx_kern = container_of(ctx, struct hid_bpf_ctx_kern, ctx);
hid = (struct hid_device *)ctx_kern->ctx.hid; /* ignore const */
kfree(ctx_kern); kfree(ctx_kern);
/* get_device() is called by bus_find_device() */
put_device(&hid->dev);
} }
/** /**
...@@ -358,7 +404,7 @@ hid_bpf_release_context(struct hid_bpf_ctx *ctx) ...@@ -358,7 +404,7 @@ hid_bpf_release_context(struct hid_bpf_ctx *ctx)
* *
* @returns %0 on success, a negative error code otherwise. * @returns %0 on success, a negative error code otherwise.
*/ */
noinline int __bpf_kfunc int
hid_bpf_hw_request(struct hid_bpf_ctx *ctx, __u8 *buf, size_t buf__sz, hid_bpf_hw_request(struct hid_bpf_ctx *ctx, __u8 *buf, size_t buf__sz,
enum hid_report_type rtype, enum hid_class_request reqtype) enum hid_report_type rtype, enum hid_class_request reqtype)
{ {
...@@ -426,6 +472,7 @@ hid_bpf_hw_request(struct hid_bpf_ctx *ctx, __u8 *buf, size_t buf__sz, ...@@ -426,6 +472,7 @@ hid_bpf_hw_request(struct hid_bpf_ctx *ctx, __u8 *buf, size_t buf__sz,
kfree(dma_data); kfree(dma_data);
return ret; return ret;
} }
__bpf_kfunc_end_defs();
/* our HID-BPF entrypoints */ /* our HID-BPF entrypoints */
BTF_SET8_START(hid_bpf_fmodret_ids) BTF_SET8_START(hid_bpf_fmodret_ids)
......
...@@ -12,9 +12,9 @@ struct hid_bpf_ctx_kern { ...@@ -12,9 +12,9 @@ struct hid_bpf_ctx_kern {
int hid_bpf_preload_skel(void); int hid_bpf_preload_skel(void);
void hid_bpf_free_links_and_skel(void); void hid_bpf_free_links_and_skel(void);
int hid_bpf_get_prog_attach_type(int prog_fd); int hid_bpf_get_prog_attach_type(struct bpf_prog *prog);
int __hid_bpf_attach_prog(struct hid_device *hdev, enum hid_bpf_prog_type prog_type, int prog_fd, int __hid_bpf_attach_prog(struct hid_device *hdev, enum hid_bpf_prog_type prog_type, int prog_fd,
__u32 flags); struct bpf_prog *prog, __u32 flags);
void __hid_bpf_destroy_device(struct hid_device *hdev); void __hid_bpf_destroy_device(struct hid_device *hdev);
int hid_bpf_prog_run(struct hid_device *hdev, enum hid_bpf_prog_type type, int hid_bpf_prog_run(struct hid_device *hdev, enum hid_bpf_prog_type type,
struct hid_bpf_ctx_kern *ctx_kern); struct hid_bpf_ctx_kern *ctx_kern);
......
...@@ -196,6 +196,7 @@ static void __hid_bpf_do_release_prog(int map_fd, unsigned int idx) ...@@ -196,6 +196,7 @@ static void __hid_bpf_do_release_prog(int map_fd, unsigned int idx)
static void hid_bpf_release_progs(struct work_struct *work) static void hid_bpf_release_progs(struct work_struct *work)
{ {
int i, j, n, map_fd = -1; int i, j, n, map_fd = -1;
bool hdev_destroyed;
if (!jmp_table.map) if (!jmp_table.map)
return; return;
...@@ -220,6 +221,12 @@ static void hid_bpf_release_progs(struct work_struct *work) ...@@ -220,6 +221,12 @@ static void hid_bpf_release_progs(struct work_struct *work)
if (entry->hdev) { if (entry->hdev) {
hdev = entry->hdev; hdev = entry->hdev;
type = entry->type; type = entry->type;
/*
* hdev is still valid, even if we are called after hid_destroy_device():
* when hid_bpf_attach() gets called, it takes a ref on the dev through
* bus_find_device()
*/
hdev_destroyed = hdev->bpf.destroyed;
hid_bpf_populate_hdev(hdev, type); hid_bpf_populate_hdev(hdev, type);
...@@ -232,12 +239,19 @@ static void hid_bpf_release_progs(struct work_struct *work) ...@@ -232,12 +239,19 @@ static void hid_bpf_release_progs(struct work_struct *work)
if (test_bit(next->idx, jmp_table.enabled)) if (test_bit(next->idx, jmp_table.enabled))
continue; continue;
if (next->hdev == hdev && next->type == type) if (next->hdev == hdev && next->type == type) {
/*
* clear the hdev reference and decrement the device ref
* that was taken during bus_find_device() while calling
* hid_bpf_attach()
*/
next->hdev = NULL; next->hdev = NULL;
put_device(&hdev->dev);
}
} }
/* if type was rdesc fixup, reconnect device */ /* if type was rdesc fixup and the device is not gone, reconnect device */
if (type == HID_BPF_PROG_TYPE_RDESC_FIXUP) if (type == HID_BPF_PROG_TYPE_RDESC_FIXUP && !hdev_destroyed)
hid_bpf_reconnect(hdev); hid_bpf_reconnect(hdev);
} }
} }
...@@ -333,15 +347,10 @@ static int hid_bpf_insert_prog(int prog_fd, struct bpf_prog *prog) ...@@ -333,15 +347,10 @@ static int hid_bpf_insert_prog(int prog_fd, struct bpf_prog *prog)
return err; return err;
} }
int hid_bpf_get_prog_attach_type(int prog_fd) int hid_bpf_get_prog_attach_type(struct bpf_prog *prog)
{ {
struct bpf_prog *prog = NULL;
int i;
int prog_type = HID_BPF_PROG_TYPE_UNDEF; int prog_type = HID_BPF_PROG_TYPE_UNDEF;
int i;
prog = bpf_prog_get(prog_fd);
if (IS_ERR(prog))
return PTR_ERR(prog);
for (i = 0; i < HID_BPF_PROG_TYPE_MAX; i++) { for (i = 0; i < HID_BPF_PROG_TYPE_MAX; i++) {
if (hid_bpf_btf_ids[i] == prog->aux->attach_btf_id) { if (hid_bpf_btf_ids[i] == prog->aux->attach_btf_id) {
...@@ -350,8 +359,6 @@ int hid_bpf_get_prog_attach_type(int prog_fd) ...@@ -350,8 +359,6 @@ int hid_bpf_get_prog_attach_type(int prog_fd)
} }
} }
bpf_prog_put(prog);
return prog_type; return prog_type;
} }
...@@ -388,19 +395,13 @@ static const struct bpf_link_ops hid_bpf_link_lops = { ...@@ -388,19 +395,13 @@ static const struct bpf_link_ops hid_bpf_link_lops = {
/* called from syscall */ /* called from syscall */
noinline int noinline int
__hid_bpf_attach_prog(struct hid_device *hdev, enum hid_bpf_prog_type prog_type, __hid_bpf_attach_prog(struct hid_device *hdev, enum hid_bpf_prog_type prog_type,
int prog_fd, __u32 flags) int prog_fd, struct bpf_prog *prog, __u32 flags)
{ {
struct bpf_link_primer link_primer; struct bpf_link_primer link_primer;
struct hid_bpf_link *link; struct hid_bpf_link *link;
struct bpf_prog *prog = NULL;
struct hid_bpf_prog_entry *prog_entry; struct hid_bpf_prog_entry *prog_entry;
int cnt, err = -EINVAL, prog_table_idx = -1; int cnt, err = -EINVAL, prog_table_idx = -1;
/* take a ref on the prog itself */
prog = bpf_prog_get(prog_fd);
if (IS_ERR(prog))
return PTR_ERR(prog);
mutex_lock(&hid_bpf_attach_lock); mutex_lock(&hid_bpf_attach_lock);
link = kzalloc(sizeof(*link), GFP_USER); link = kzalloc(sizeof(*link), GFP_USER);
...@@ -467,7 +468,6 @@ __hid_bpf_attach_prog(struct hid_device *hdev, enum hid_bpf_prog_type prog_type, ...@@ -467,7 +468,6 @@ __hid_bpf_attach_prog(struct hid_device *hdev, enum hid_bpf_prog_type prog_type,
err_unlock: err_unlock:
mutex_unlock(&hid_bpf_attach_lock); mutex_unlock(&hid_bpf_attach_lock);
bpf_prog_put(prog);
kfree(link); kfree(link);
return err; return err;
......
...@@ -298,6 +298,9 @@ ...@@ -298,6 +298,9 @@
#define USB_VENDOR_ID_CIDC 0x1677 #define USB_VENDOR_ID_CIDC 0x1677
#define I2C_VENDOR_ID_CIRQUE 0x0488
#define I2C_PRODUCT_ID_CIRQUE_1063 0x1063
#define USB_VENDOR_ID_CJTOUCH 0x24b8 #define USB_VENDOR_ID_CJTOUCH 0x24b8
#define USB_DEVICE_ID_CJTOUCH_MULTI_TOUCH_0020 0x0020 #define USB_DEVICE_ID_CJTOUCH_MULTI_TOUCH_0020 0x0020
#define USB_DEVICE_ID_CJTOUCH_MULTI_TOUCH_0040 0x0040 #define USB_DEVICE_ID_CJTOUCH_MULTI_TOUCH_0040 0x0040
......
...@@ -4610,6 +4610,8 @@ static const struct hid_device_id hidpp_devices[] = { ...@@ -4610,6 +4610,8 @@ static const struct hid_device_id hidpp_devices[] = {
HID_USB_DEVICE(USB_VENDOR_ID_LOGITECH, 0xC088) }, HID_USB_DEVICE(USB_VENDOR_ID_LOGITECH, 0xC088) },
{ /* Logitech G Pro X Superlight Gaming Mouse over USB */ { /* Logitech G Pro X Superlight Gaming Mouse over USB */
HID_USB_DEVICE(USB_VENDOR_ID_LOGITECH, 0xC094) }, HID_USB_DEVICE(USB_VENDOR_ID_LOGITECH, 0xC094) },
{ /* Logitech G Pro X Superlight 2 Gaming Mouse over USB */
HID_USB_DEVICE(USB_VENDOR_ID_LOGITECH, 0xC09b) },
{ /* G935 Gaming Headset */ { /* G935 Gaming Headset */
HID_USB_DEVICE(USB_VENDOR_ID_LOGITECH, 0x0a87), HID_USB_DEVICE(USB_VENDOR_ID_LOGITECH, 0x0a87),
......
...@@ -800,6 +800,8 @@ static inline int thunderstrike_led_create(struct thunderstrike *ts) ...@@ -800,6 +800,8 @@ static inline int thunderstrike_led_create(struct thunderstrike *ts)
led->name = devm_kasprintf(&ts->base.hdev->dev, GFP_KERNEL, led->name = devm_kasprintf(&ts->base.hdev->dev, GFP_KERNEL,
"thunderstrike%d:blue:led", ts->id); "thunderstrike%d:blue:led", ts->id);
if (!led->name)
return -ENOMEM;
led->max_brightness = 1; led->max_brightness = 1;
led->flags = LED_CORE_SUSPENDRESUME | LED_RETAIN_AT_SHUTDOWN; led->flags = LED_CORE_SUSPENDRESUME | LED_RETAIN_AT_SHUTDOWN;
led->brightness_get = &thunderstrike_led_get_brightness; led->brightness_get = &thunderstrike_led_get_brightness;
...@@ -831,6 +833,8 @@ static inline int thunderstrike_psy_create(struct shield_device *shield_dev) ...@@ -831,6 +833,8 @@ static inline int thunderstrike_psy_create(struct shield_device *shield_dev)
shield_dev->battery_dev.desc.name = shield_dev->battery_dev.desc.name =
devm_kasprintf(&ts->base.hdev->dev, GFP_KERNEL, devm_kasprintf(&ts->base.hdev->dev, GFP_KERNEL,
"thunderstrike_%d", ts->id); "thunderstrike_%d", ts->id);
if (!shield_dev->battery_dev.desc.name)
return -ENOMEM;
shield_dev->battery_dev.psy = power_supply_register( shield_dev->battery_dev.psy = power_supply_register(
&hdev->dev, &shield_dev->battery_dev.desc, &psy_cfg); &hdev->dev, &shield_dev->battery_dev.desc, &psy_cfg);
......
...@@ -1109,10 +1109,9 @@ static int steam_probe(struct hid_device *hdev, ...@@ -1109,10 +1109,9 @@ static int steam_probe(struct hid_device *hdev,
return hid_hw_start(hdev, HID_CONNECT_DEFAULT); return hid_hw_start(hdev, HID_CONNECT_DEFAULT);
steam = devm_kzalloc(&hdev->dev, sizeof(*steam), GFP_KERNEL); steam = devm_kzalloc(&hdev->dev, sizeof(*steam), GFP_KERNEL);
if (!steam) { if (!steam)
ret = -ENOMEM; return -ENOMEM;
goto steam_alloc_fail;
}
steam->hdev = hdev; steam->hdev = hdev;
hid_set_drvdata(hdev, steam); hid_set_drvdata(hdev, steam);
spin_lock_init(&steam->lock); spin_lock_init(&steam->lock);
...@@ -1129,14 +1128,14 @@ static int steam_probe(struct hid_device *hdev, ...@@ -1129,14 +1128,14 @@ static int steam_probe(struct hid_device *hdev,
*/ */
ret = hid_hw_start(hdev, HID_CONNECT_DEFAULT & ~HID_CONNECT_HIDRAW); ret = hid_hw_start(hdev, HID_CONNECT_DEFAULT & ~HID_CONNECT_HIDRAW);
if (ret) if (ret)
goto hid_hw_start_fail; goto err_cancel_work;
ret = hid_hw_open(hdev); ret = hid_hw_open(hdev);
if (ret) { if (ret) {
hid_err(hdev, hid_err(hdev,
"%s:hid_hw_open\n", "%s:hid_hw_open\n",
__func__); __func__);
goto hid_hw_open_fail; goto err_hw_stop;
} }
if (steam->quirks & STEAM_QUIRK_WIRELESS) { if (steam->quirks & STEAM_QUIRK_WIRELESS) {
...@@ -1152,36 +1151,37 @@ static int steam_probe(struct hid_device *hdev, ...@@ -1152,36 +1151,37 @@ static int steam_probe(struct hid_device *hdev,
hid_err(hdev, hid_err(hdev,
"%s:steam_register failed with error %d\n", "%s:steam_register failed with error %d\n",
__func__, ret); __func__, ret);
goto input_register_fail; goto err_hw_close;
} }
} }
steam->client_hdev = steam_create_client_hid(hdev); steam->client_hdev = steam_create_client_hid(hdev);
if (IS_ERR(steam->client_hdev)) { if (IS_ERR(steam->client_hdev)) {
ret = PTR_ERR(steam->client_hdev); ret = PTR_ERR(steam->client_hdev);
goto client_hdev_fail; goto err_stream_unregister;
} }
steam->client_hdev->driver_data = steam; steam->client_hdev->driver_data = steam;
ret = hid_add_device(steam->client_hdev); ret = hid_add_device(steam->client_hdev);
if (ret) if (ret)
goto client_hdev_add_fail; goto err_destroy;
return 0; return 0;
client_hdev_add_fail: err_destroy:
hid_hw_stop(hdev);
client_hdev_fail:
hid_destroy_device(steam->client_hdev); hid_destroy_device(steam->client_hdev);
input_register_fail: err_stream_unregister:
hid_hw_open_fail: if (steam->connected)
hid_hw_start_fail: steam_unregister(steam);
err_hw_close:
hid_hw_close(hdev);
err_hw_stop:
hid_hw_stop(hdev);
err_cancel_work:
cancel_work_sync(&steam->work_connect); cancel_work_sync(&steam->work_connect);
cancel_delayed_work_sync(&steam->mode_switch); cancel_delayed_work_sync(&steam->mode_switch);
cancel_work_sync(&steam->rumble_work); cancel_work_sync(&steam->rumble_work);
steam_alloc_fail:
hid_err(hdev, "%s: failed with error %d\n",
__func__, ret);
return ret; return ret;
} }
......
...@@ -357,8 +357,11 @@ static int hidraw_release(struct inode * inode, struct file * file) ...@@ -357,8 +357,11 @@ static int hidraw_release(struct inode * inode, struct file * file)
down_write(&minors_rwsem); down_write(&minors_rwsem);
spin_lock_irqsave(&hidraw_table[minor]->list_lock, flags); spin_lock_irqsave(&hidraw_table[minor]->list_lock, flags);
for (int i = list->tail; i < list->head; i++) while (list->tail != list->head) {
kfree(list->buffer[i].value); kfree(list->buffer[list->tail].value);
list->buffer[list->tail].value = NULL;
list->tail = (list->tail + 1) & (HIDRAW_BUFFER_SIZE - 1);
}
list_del(&list->node); list_del(&list->node);
spin_unlock_irqrestore(&hidraw_table[minor]->list_lock, flags); spin_unlock_irqrestore(&hidraw_table[minor]->list_lock, flags);
kfree(list); kfree(list);
......
...@@ -49,6 +49,7 @@ ...@@ -49,6 +49,7 @@
#define I2C_HID_QUIRK_RESET_ON_RESUME BIT(2) #define I2C_HID_QUIRK_RESET_ON_RESUME BIT(2)
#define I2C_HID_QUIRK_BAD_INPUT_SIZE BIT(3) #define I2C_HID_QUIRK_BAD_INPUT_SIZE BIT(3)
#define I2C_HID_QUIRK_NO_WAKEUP_AFTER_RESET BIT(4) #define I2C_HID_QUIRK_NO_WAKEUP_AFTER_RESET BIT(4)
#define I2C_HID_QUIRK_NO_SLEEP_ON_SUSPEND BIT(5)
/* Command opcodes */ /* Command opcodes */
#define I2C_HID_OPCODE_RESET 0x01 #define I2C_HID_OPCODE_RESET 0x01
...@@ -131,6 +132,8 @@ static const struct i2c_hid_quirks { ...@@ -131,6 +132,8 @@ static const struct i2c_hid_quirks {
I2C_HID_QUIRK_RESET_ON_RESUME }, I2C_HID_QUIRK_RESET_ON_RESUME },
{ USB_VENDOR_ID_ITE, I2C_DEVICE_ID_ITE_LENOVO_LEGION_Y720, { USB_VENDOR_ID_ITE, I2C_DEVICE_ID_ITE_LENOVO_LEGION_Y720,
I2C_HID_QUIRK_BAD_INPUT_SIZE }, I2C_HID_QUIRK_BAD_INPUT_SIZE },
{ I2C_VENDOR_ID_CIRQUE, I2C_PRODUCT_ID_CIRQUE_1063,
I2C_HID_QUIRK_NO_SLEEP_ON_SUSPEND },
/* /*
* Sending the wakeup after reset actually break ELAN touchscreen controller * Sending the wakeup after reset actually break ELAN touchscreen controller
*/ */
...@@ -956,6 +959,7 @@ static int i2c_hid_core_suspend(struct i2c_hid *ihid, bool force_poweroff) ...@@ -956,6 +959,7 @@ static int i2c_hid_core_suspend(struct i2c_hid *ihid, bool force_poweroff)
return ret; return ret;
/* Save some power */ /* Save some power */
if (!(ihid->quirks & I2C_HID_QUIRK_NO_SLEEP_ON_SUSPEND))
i2c_hid_set_power(ihid, I2C_HID_PWR_SLEEP); i2c_hid_set_power(ihid, I2C_HID_PWR_SLEEP);
disable_irq(client->irq); disable_irq(client->irq);
......
...@@ -87,6 +87,7 @@ static int i2c_hid_of_probe(struct i2c_client *client) ...@@ -87,6 +87,7 @@ static int i2c_hid_of_probe(struct i2c_client *client)
if (!ihid_of) if (!ihid_of)
return -ENOMEM; return -ENOMEM;
ihid_of->client = client;
ihid_of->ops.power_up = i2c_hid_of_power_up; ihid_of->ops.power_up = i2c_hid_of_power_up;
ihid_of->ops.power_down = i2c_hid_of_power_down; ihid_of->ops.power_down = i2c_hid_of_power_down;
......
...@@ -77,17 +77,6 @@ enum hid_bpf_attach_flags { ...@@ -77,17 +77,6 @@ enum hid_bpf_attach_flags {
int hid_bpf_device_event(struct hid_bpf_ctx *ctx); int hid_bpf_device_event(struct hid_bpf_ctx *ctx);
int hid_bpf_rdesc_fixup(struct hid_bpf_ctx *ctx); int hid_bpf_rdesc_fixup(struct hid_bpf_ctx *ctx);
/* Following functions are kfunc that we export to BPF programs */
/* available everywhere in HID-BPF */
__u8 *hid_bpf_get_data(struct hid_bpf_ctx *ctx, unsigned int offset, const size_t __sz);
/* only available in syscall */
int hid_bpf_attach_prog(unsigned int hid_id, int prog_fd, __u32 flags);
int hid_bpf_hw_request(struct hid_bpf_ctx *ctx, __u8 *buf, size_t buf__sz,
enum hid_report_type rtype, enum hid_class_request reqtype);
struct hid_bpf_ctx *hid_bpf_allocate_context(unsigned int hid_id);
void hid_bpf_release_context(struct hid_bpf_ctx *ctx);
/* /*
* Below is HID internal * Below is HID internal
*/ */
......
...@@ -880,8 +880,8 @@ class TestDTH2452Tablet(test_multitouch.BaseTest.TestMultitouch, TouchTabletTest ...@@ -880,8 +880,8 @@ class TestDTH2452Tablet(test_multitouch.BaseTest.TestMultitouch, TouchTabletTest
does not overlap with other contacts. The value of `t` may be does not overlap with other contacts. The value of `t` may be
incremented over time to move the point along a linear path. incremented over time to move the point along a linear path.
""" """
x = 50 + 10 * contact_id + t x = 50 + 10 * contact_id + t * 11
y = 100 + 100 * contact_id + t y = 100 + 100 * contact_id + t * 11
return test_multitouch.Touch(contact_id, x, y) return test_multitouch.Touch(contact_id, x, y)
def make_contacts(self, n, t=0): def make_contacts(self, n, t=0):
...@@ -902,8 +902,8 @@ class TestDTH2452Tablet(test_multitouch.BaseTest.TestMultitouch, TouchTabletTest ...@@ -902,8 +902,8 @@ class TestDTH2452Tablet(test_multitouch.BaseTest.TestMultitouch, TouchTabletTest
tracking_id = contact_ids.tracking_id tracking_id = contact_ids.tracking_id
slot_num = contact_ids.slot_num slot_num = contact_ids.slot_num
x = 50 + 10 * contact_id + t x = 50 + 10 * contact_id + t * 11
y = 100 + 100 * contact_id + t y = 100 + 100 * contact_id + t * 11
# If the data isn't supposed to be stored in any slots, there is # If the data isn't supposed to be stored in any slots, there is
# nothing we can check for in the evdev stream. # nothing we can check for in the evdev stream.
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment