From patchwork Thu Feb 3 18:40:24 2022 Content-Type: text/plain; charset="utf-8" MIME-Version: 1.0 Content-Transfer-Encoding: 8bit X-Patchwork-Submitter: Sergio Costas X-Patchwork-Id: 12734555 Return-Path: X-Spam-Checker-Version: SpamAssassin 3.4.0 (2014-02-07) on aws-us-west-2-korg-lkml-1.web.codeaurora.org Received: from vger.kernel.org (vger.kernel.org [23.128.96.18]) by smtp.lore.kernel.org (Postfix) with ESMTP id EDED4C433EF for ; Thu, 3 Feb 2022 18:40:28 +0000 (UTC) Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S244452AbiBCSk2 (ORCPT ); Thu, 3 Feb 2022 13:40:28 -0500 Received: from lindbergh.monkeyblade.net ([23.128.96.19]:60584 "EHLO lindbergh.monkeyblade.net" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S234230AbiBCSk2 (ORCPT ); Thu, 3 Feb 2022 13:40:28 -0500 Received: from mail-wr1-x436.google.com (mail-wr1-x436.google.com [IPv6:2a00:1450:4864:20::436]) by lindbergh.monkeyblade.net (Postfix) with ESMTPS id 880E7C061714 for ; Thu, 3 Feb 2022 10:40:27 -0800 (PST) Received: by mail-wr1-x436.google.com with SMTP id m14so6721958wrg.12 for ; Thu, 03 Feb 2022 10:40:27 -0800 (PST) DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=gmail.com; s=20210112; h=message-id:date:mime-version:user-agent:content-language:to:from :subject:content-transfer-encoding; bh=4gUjm1o+iIrLKEEKlgpZ+gw9JFX3GLTpX+nuuAoIjI0=; b=kfAM06AIcZ6QpHbYVFPNazNEdHmMWf1IY6ItE0yjh93TF2nsL0aifRflVVGw1AFwNq /Rnngwal1JwA2k5hWKXzx6Hv6nNGrOj/rbVLZ9mfvWCUb8Q6ZjPqAb1XCkayUU229ZG1 HEtPrq9qnYXIGwg/TgYrNZnykpv222gmhjBGZMgzeDyZIm+btP56ZgiBxg7KhrediRHC r7NbUrK32rEzGL2dQCHVABkMBrOUqf1MibMBVG2qII69DpBcqk5OElVPnuqXB8+INhiA j3rV33OcllVCYuRTf6otE2ILdIg61Y0p8jbavheMVVU7QtwI51la1Djz3+B89BJqHJgm idQw== X-Google-DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=1e100.net; s=20210112; h=x-gm-message-state:message-id:date:mime-version:user-agent :content-language:to:from:subject:content-transfer-encoding; bh=4gUjm1o+iIrLKEEKlgpZ+gw9JFX3GLTpX+nuuAoIjI0=; b=xhT9bfo1M/ypYv+458n7atBdhqtfIHCBPPG3kjhE1u1A1X4v8N7V3BKyKptA/9ZRWN RqYGV1jYgeae3KxvbXS2CEJn44bVSSU1WYYqVneY684dYJ808Lm2bbW2rPdv+Bcx7yd2 jZIcDPz1Wq+u5inUHCgrmiNVjyFV3FJ2V43zxS6/U7UvmO8S32sBhh/GP4cKEKIEe2Uh O6RpyGamR+bdKfa0fuwgFWfLtT2dYixjeiejYe+moRDAxrlcvjNO3J4xyj9ZAyHTvex2 PeyAYJROV1D1QYxB2ZwfMu0N0Enb1nL0pjf1SpRVMA1OfPUgSBF1SHGkvIWSH89M5UJR NgIw== X-Gm-Message-State: AOAM531QijmEhXvFGoadxC5t1gISMvp9eAGB4BZ3RQsja0DGReb+62Qs cPRbITj0kqDXJfTFGPArEhUBWq2509zHQg== X-Google-Smtp-Source: ABdhPJyLMm22P1wX/KmbkCKL7FnmN4jcnyCieuJzyaQi7MX2DQ2KS+l0dDBVINn6bsns3hGPdCgRHQ== X-Received: by 2002:adf:f984:: with SMTP id f4mr30843301wrr.510.1643913625981; Thu, 03 Feb 2022 10:40:25 -0800 (PST) Received: from [192.168.0.21] (95.138.117.91.dynamic.reverse-mundo-r.com. [91.117.138.95]) by smtp.gmail.com with ESMTPSA id t16sm8942472wmq.21.2022.02.03.10.40.24 (version=TLS1_3 cipher=TLS_AES_128_GCM_SHA256 bits=128/128); Thu, 03 Feb 2022 10:40:24 -0800 (PST) Message-ID: <6e8bd24e-3fad-b0c3-971b-cfd9152bf349@gmail.com> Date: Thu, 3 Feb 2022 19:40:24 +0100 MIME-Version: 1.0 User-Agent: Mozilla/5.0 (X11; Linux x86_64; rv:91.0) Gecko/20100101 Thunderbird/91.5.1 Content-Language: en-US To: linux-input@vger.kernel.org, jikos@kernel.org From: Sergio Costas Subject: [PATCH] HID:Add support for UGTABLET WP5540 with new quirk Precedence: bulk List-ID: X-Mailing-List: linux-input@vger.kernel.org From 82e513c85f4bdc027e5b907000a3bbaf74314bcb Mon Sep 17 00:00:00 2001 From: Sergio Costas Date: Thu, 3 Feb 2022 14:15:07 +0100 Subject: [PATCH] HID:Add support for UGTABLET WP5540 with new quirk This patch adds a new quirk to HID, allowing to configure the usage rules in reverse order. This is needed to enable the use of the UGTABLET WP5540 USB digitizer tablet. This tablet defines three HID collections: one for a digitizer device and two for mouse devices, the first one with relative coordinates, and the second one with absolute coordinates. The problem is that both mouse devices share the "button 1", "button 2" and "button 3" usages, but the Linux HID driver doesn't support that, so those usages remain unassigned for the second mouse because they are considered "duplicated". Unfortunately, the second mouse is, really, the pencil, which means that the movements are detected and managed, but not the pencil touch or any button press, because those usages are already assigned to the first mouse (I suspect that the chip used in this tablet is designed to work both with pencils and with uncorded mouses, allowing dual devices where both a pencil and a mouse can be used over the same rug). The result is that the user can move the cursor with the pencil, but can't "paint" things because the "touch" action doesn't work. Currently there is already a quirk that, at first glance, could seem as the solution to this problem. That quirk is HID_QUIRK_INCREMENT_USAGE_ON_DUPLICATE. Unfortunately, the result is that the "pencil touch" and "press button" actions are translated into "previous page", "next page"... and so on, which is not a desirable option. Although those events could be reasigned in user space using xmodmap, it requires actions from the user, and also could interfere with other devices, like multibutton mouses. I also tried to remove the duplicate test, but the system hangs as soon as it receives an event from the tablet. This patch aims to fix this by configuring the entries in reverse order, thus giving priority to the pencil over the mouse. This makes sense today, because it is more common to use optical mouses instead of tablet-based ones, and also ensures that devices marketed only with the pencil but using this chip do work out-of-the-box. To do this, the patch adds a new quirk: HID_QUIRK_REVERSE_CONFIGURE_USAGES, which can be enabled for an specific device, and it does assign the actions in reverse order, thus reversing the priority. This quirk can still be combined with HID_QUIRK_INCREMENT_USAGE_ON_DUPLICATE, so the events from the physical mouse could still be received, but giving priority to the digitizer, which makes more sense. With this new quirk enabled for it, the UGTABLET WP5540 digitizer tablet works like a charm. Signed-off-by: Sergio Costas Signed-off-by: Sergio Costas ---  drivers/hid/hid-ids.h    |   1 +  drivers/hid/hid-input.c  | 101 +++++++++++++++++++++++++++------------  drivers/hid/hid-quirks.c |   1 +  include/linux/hid.h      |   1 +  4 files changed, 73 insertions(+), 31 deletions(-) diff --git a/drivers/hid/hid-ids.h b/drivers/hid/hid-ids.h index 85975031389b..78bd3ddda442 100644 --- a/drivers/hid/hid-ids.h +++ b/drivers/hid/hid-ids.h @@ -1370,6 +1370,7 @@  #define USB_VENDOR_ID_UGTIZER            0x2179  #define USB_DEVICE_ID_UGTIZER_TABLET_GP0610    0x0053  #define USB_DEVICE_ID_UGTIZER_TABLET_GT5040    0x0077 +#define USB_DEVICE_ID_UGTIZER_TABLET_WP5540    0x0004  #define USB_VENDOR_ID_VIEWSONIC            0x0543  #define USB_DEVICE_ID_VIEWSONIC_PD1011        0xe621 diff --git a/drivers/hid/hid-input.c b/drivers/hid/hid-input.c index 112901d2d8d2..8cf4f63581cf 100644 --- a/drivers/hid/hid-input.c +++ b/drivers/hid/hid-input.c @@ -1938,6 +1938,56 @@ static inline void hidinput_configure_usages(struct hid_input *hidinput,                           report->field[i]->usage + j);  } +static inline void hidinput_configure_usages_reversed(struct hid_input *hidinput, +                              struct hid_report *report) +{ +    int i, j; + +    for (i = report->maxfield; i > 0 ; i--) +        for (j = report->field[i-1]->maxusage; j > 0 ; j--) +            hidinput_configure_usage(hidinput, report->field[i-1], +                         report->field[i-1]->usage + j - 1); +} + +static bool hidinput_configure_usage_list(struct hid_device *hid, +                      struct hid_report *report, +                      bool reverse) +{ + +    unsigned int application; +    struct hid_input *hidinput = NULL; + +    application = report->application; + +    /* +     * Find the previous hidinput report attached +     * to this report id. +     */ +    if (hid->quirks & HID_QUIRK_MULTI_INPUT) +        hidinput = hidinput_match(report); +    else if (hid->maxapplication > 1 && +            (hid->quirks & HID_QUIRK_INPUT_PER_APP)) +        hidinput = hidinput_match_application(report); + +    if (!hidinput) { +        hidinput = hidinput_allocate(hid, application); +        if (!hidinput) +            return true; +    } + +    if (reverse) +        hidinput_configure_usages_reversed(hidinput, report); +    else +        hidinput_configure_usages(hidinput, report); + +    if (hid->quirks & HID_QUIRK_MULTI_INPUT) +        hidinput->report = report; + +    list_add_tail(&report->hidinput_list, +              &hidinput->reports); +    return false; +} +  /*   * Register the input device; print a message.   * Configure the input layer interface @@ -1949,8 +1999,7 @@ int hidinput_connect(struct hid_device *hid, unsigned int force)      struct hid_driver *drv = hid->driver;      struct hid_report *report;      struct hid_input *next, *hidinput = NULL; -    unsigned int application; -    int i, k; +    int i, k, l;      INIT_LIST_HEAD(&hid->inputs);      INIT_WORK(&hid->led_work, hidinput_led_worker); @@ -1972,41 +2021,31 @@ int hidinput_connect(struct hid_device *hid, unsigned int force)      report_features(hid); -    for (k = HID_INPUT_REPORT; k <= HID_OUTPUT_REPORT; k++) { +    for (l = HID_INPUT_REPORT; l <= HID_OUTPUT_REPORT; l++) { +        if (hid->quirks & HID_QUIRK_REVERSE_CONFIGURE_USAGES) +            k = HID_OUTPUT_REPORT - l + HID_INPUT_REPORT; +        else +            k = l;          if (k == HID_OUTPUT_REPORT &&              hid->quirks & HID_QUIRK_SKIP_OUTPUT_REPORTS)              continue; -        list_for_each_entry(report, &hid->report_enum[k].report_list, list) { - -            if (!report->maxfield) -                continue; - -            application = report->application; - -            /* -             * Find the previous hidinput report attached -             * to this report id. -             */ -            if (hid->quirks & HID_QUIRK_MULTI_INPUT) -                hidinput = hidinput_match(report); -            else if (hid->maxapplication > 1 && -                 (hid->quirks & HID_QUIRK_INPUT_PER_APP)) -                hidinput = hidinput_match_application(report); - -            if (!hidinput) { -                hidinput = hidinput_allocate(hid, application); -                if (!hidinput) +        if (hid->quirks & HID_QUIRK_REVERSE_CONFIGURE_USAGES) { +            list_for_each_entry_reverse(report, + &hid->report_enum[k].report_list, +                            list) { +                if (!report->maxfield) +                    continue; +                if (hidinput_configure_usage_list(hid, report, true)) +                    goto out_unwind; +            } +        } else { +            list_for_each_entry(report, &hid->report_enum[k].report_list, list) { +                if (!report->maxfield) +                    continue; +                if (hidinput_configure_usage_list(hid, report, false))                      goto out_unwind;              } - -            hidinput_configure_usages(hidinput, report); - -            if (hid->quirks & HID_QUIRK_MULTI_INPUT) -                hidinput->report = report; - -            list_add_tail(&report->hidinput_list, -                      &hidinput->reports);          }      } diff --git a/drivers/hid/hid-quirks.c b/drivers/hid/hid-quirks.c index 9af1dc8ae3a2..a59510bc76a1 100644 --- a/drivers/hid/hid-quirks.c +++ b/drivers/hid/hid-quirks.c @@ -187,6 +187,7 @@ static const struct hid_device_id hid_quirks[] = {      { HID_USB_DEVICE(USB_VENDOR_ID_TURBOX, USB_DEVICE_ID_TURBOX_KEYBOARD), HID_QUIRK_NOGET },      { HID_USB_DEVICE(USB_VENDOR_ID_UCLOGIC, USB_DEVICE_ID_UCLOGIC_TABLET_KNA5), HID_QUIRK_MULTI_INPUT },      { HID_USB_DEVICE(USB_VENDOR_ID_UCLOGIC, USB_DEVICE_ID_UCLOGIC_TABLET_TWA60), HID_QUIRK_MULTI_INPUT }, +    { HID_USB_DEVICE(USB_VENDOR_ID_UGTIZER, USB_DEVICE_ID_UGTIZER_TABLET_WP5540), HID_QUIRK_REVERSE_CONFIGURE_USAGES },      { HID_USB_DEVICE(USB_VENDOR_ID_WALTOP, USB_DEVICE_ID_WALTOP_MEDIA_TABLET_10_6_INCH), HID_QUIRK_MULTI_INPUT },      { HID_USB_DEVICE(USB_VENDOR_ID_WALTOP, USB_DEVICE_ID_WALTOP_MEDIA_TABLET_14_1_INCH), HID_QUIRK_MULTI_INPUT },      { HID_USB_DEVICE(USB_VENDOR_ID_WALTOP, USB_DEVICE_ID_WALTOP_SIRIUS_BATTERY_FREE_TABLET), HID_QUIRK_MULTI_INPUT }, diff --git a/include/linux/hid.h b/include/linux/hid.h index 7487b0586fe6..61bf462ca658 100644 --- a/include/linux/hid.h +++ b/include/linux/hid.h @@ -361,6 +361,7 @@ struct hid_item {  #define HID_QUIRK_INPUT_PER_APP            BIT(11)  #define HID_QUIRK_X_INVERT            BIT(12)  #define HID_QUIRK_Y_INVERT            BIT(13) +#define HID_QUIRK_REVERSE_CONFIGURE_USAGES    BIT(14)  #define HID_QUIRK_SKIP_OUTPUT_REPORTS        BIT(16)  #define HID_QUIRK_SKIP_OUTPUT_REPORT_ID        BIT(17)  #define HID_QUIRK_NO_OUTPUT_REPORTS_ON_INTR_EP    BIT(18)