diff mbox series

Fix HOG profile incorrectly stripping off read report bytes.

Message ID a542b260-7eb1-8917-9379-e945003d7817@fourwalledcubicle.com
State New
Headers show
Series Fix HOG profile incorrectly stripping off read report bytes. | expand

Commit Message

Dean Camera Nov. 20, 2020, 1:42 a.m. UTC
If the HID subsystem requests a HID report to be read from the
device, we currently incorrectly strip off the first byte of the
response, if the device has report IDs set in the HID report
descriptor.

This is incorrect; unlike USB HID, the report ID is *not* included
in the HOG profile's HID reports, and instead exists out of band
in a descriptor on the report's bluetooth characteristic in the
device.

In this patch, we remove the erroneous stripping of the first
byte of the report, and (if report IDs are enabled) prepend the
report ID to the front of the result. This makes the HID report
returned indentical in format to that of a USB HID report, so
that the upper HID drivers can consume HOG device reports in the
same way as USB.
---
  profiles/input/hog-lib.c | 18 +++++++++++-------
  1 file changed, 11 insertions(+), 7 deletions(-)

  	rsp.u.get_report_reply.err = status;
@@ -846,7 +850,7 @@ static void get_report(struct uhid_event *ev, void 
*user_data)

  	hog->getrep_att = gatt_read_char(hog->attrib,
  						report->value_handle,
-						get_report_cb, hog);
+						get_report_cb, report);
  	if (!hog->getrep_att) {
  		err = ENOMEM;
  		goto fail;

Comments

Luiz Augusto von Dentz Nov. 20, 2020, 5:49 p.m. UTC | #1
Hi Dean,

On Thu, Nov 19, 2020 at 5:47 PM Dean Camera <dean@fourwalledcubicle.com> wrote:
>
> If the HID subsystem requests a HID report to be read from the
> device, we currently incorrectly strip off the first byte of the
> response, if the device has report IDs set in the HID report
> descriptor.
>
> This is incorrect; unlike USB HID, the report ID is *not* included
> in the HOG profile's HID reports, and instead exists out of band
> in a descriptor on the report's bluetooth characteristic in the
> device.
>
> In this patch, we remove the erroneous stripping of the first
> byte of the report, and (if report IDs are enabled) prepend the
> report ID to the front of the result. This makes the HID report
> returned indentical in format to that of a USB HID report, so
> that the upper HID drivers can consume HOG device reports in the
> same way as USB.
> ---
>   profiles/input/hog-lib.c | 18 +++++++++++-------
>   1 file changed, 11 insertions(+), 7 deletions(-)
>
> diff --git a/profiles/input/hog-lib.c b/profiles/input/hog-lib.c
> index 78018aad3..49d459e21 100644
> --- a/profiles/input/hog-lib.c
> +++ b/profiles/input/hog-lib.c
> @@ -779,7 +779,8 @@ fail:
>   static void get_report_cb(guint8 status, const guint8 *pdu, guint16 len,
>                                                         gpointer user_data)
>   {
> -       struct bt_hog *hog = user_data;
> +       struct report *report = user_data;
> +       struct bt_hog *hog = report->hog;
>         struct uhid_event rsp;
>         int err;
>
> @@ -808,13 +809,16 @@ static void get_report_cb(guint8 status, const
> guint8 *pdu, guint16 len,
>
>         --len;
>         ++pdu;
> +
>         if (hog->has_report_id && len > 0) {
> -               --len;
> -               ++pdu;
> +               rsp.u.get_report_reply.size = len + 1;
> +               rsp.u.get_report_reply.data[0] = report->id;
> +               memcpy(&rsp.u.get_report_reply.data[1], pdu, len);
> +       }
> +       else {
> +               rsp.u.get_report_reply.size = len;
> +               memcpy(rsp.u.get_report_reply.data, pdu, len);
>         }
> -
> -       rsp.u.get_report_reply.size = len;
> -       memcpy(rsp.u.get_report_reply.data, pdu, len);
>
>   exit:
>         rsp.u.get_report_reply.err = status;
> @@ -846,7 +850,7 @@ static void get_report(struct uhid_event *ev, void
> *user_data)
>
>         hog->getrep_att = gatt_read_char(hog->attrib,
>                                                 report->value_handle,
> -                                               get_report_cb, hog);
> +                                               get_report_cb, report);
>         if (!hog->getrep_att) {
>                 err = ENOMEM;
>                 goto fail;
> --
> 2.29.2.windows.2
>

Applied, thanks.
diff mbox series

Patch

diff --git a/profiles/input/hog-lib.c b/profiles/input/hog-lib.c
index 78018aad3..49d459e21 100644
--- a/profiles/input/hog-lib.c
+++ b/profiles/input/hog-lib.c
@@ -779,7 +779,8 @@  fail:
  static void get_report_cb(guint8 status, const guint8 *pdu, guint16 len,
  							gpointer user_data)
  {
-	struct bt_hog *hog = user_data;
+	struct report *report = user_data;
+	struct bt_hog *hog = report->hog;
  	struct uhid_event rsp;
  	int err;

@@ -808,13 +809,16 @@  static void get_report_cb(guint8 status, const 
guint8 *pdu, guint16 len,

  	--len;
  	++pdu;
+
  	if (hog->has_report_id && len > 0) {
-		--len;
-		++pdu;
+		rsp.u.get_report_reply.size = len + 1;
+		rsp.u.get_report_reply.data[0] = report->id;
+		memcpy(&rsp.u.get_report_reply.data[1], pdu, len);
+	}
+	else {
+		rsp.u.get_report_reply.size = len;
+		memcpy(rsp.u.get_report_reply.data, pdu, len);
  	}
-
-	rsp.u.get_report_reply.size = len;
-	memcpy(rsp.u.get_report_reply.data, pdu, len);

  exit: