Commit 7953794f authored by Ping Cheng's avatar Ping Cheng Committed by Jiri Kosina
Browse files

HID: input: map HID_GD_Z to ABS_DISTANCE for stylus/pen



HID_GD_Z is mapped to ABS_Z for stylus and pen in hid-input.c. But HID_GD_Z
should be used to report ABS_DISTANCE for stylus and pen as described at:
Documentation/input/event-codes.rst#n226

* ABS_DISTANCE:

  - Used to describe the distance of a tool from an interaction surface. This
    event should only be emitted while the tool is hovering, meaning in close
    proximity of the device and while the value of the BTN_TOUCH code is 0. If
    the input device may be used freely in three dimensions, consider ABS_Z
    instead.
  - BTN_TOOL_<name> should be set to 1 when the tool comes into detectable
    proximity and set to 0 when the tool leaves detectable proximity.
    BTN_TOOL_<name> signals the type of tool that is currently detected by the
    hardware and is otherwise independent of ABS_DISTANCE and/or BTN_TOUCH.

This patch makes the correct mapping. The ABS_DISTANCE is currently not mapped
by any HID usage in hid-generic driver.

Signed-off-by: default avatarPing Cheng <ping.cheng@wacom.com>
Cc: stable@kernel.org
Signed-off-by: default avatarJiri Kosina <jkosina@suse.com>
parent 0678f563
Loading
Loading
Loading
Loading
+17 −1
Original line number Diff line number Diff line
@@ -878,7 +878,7 @@ static void hidinput_configure_usage(struct hid_input *hidinput, struct hid_fiel

		switch (usage->hid) {
		/* These usage IDs map directly to the usage codes. */
		case HID_GD_X: case HID_GD_Y: case HID_GD_Z:
		case HID_GD_X: case HID_GD_Y:
		case HID_GD_RX: case HID_GD_RY: case HID_GD_RZ:
			if (field->flags & HID_MAIN_ITEM_RELATIVE)
				map_rel(usage->hid & 0xf);
@@ -886,6 +886,22 @@ static void hidinput_configure_usage(struct hid_input *hidinput, struct hid_fiel
				map_abs_clear(usage->hid & 0xf);
			break;

		case HID_GD_Z:
			/* HID_GD_Z is mapped to ABS_DISTANCE for stylus/pen */
			if (field->flags & HID_MAIN_ITEM_RELATIVE) {
				map_rel(usage->hid & 0xf);
			} else {
				if (field->application == HID_DG_PEN ||
				    field->physical == HID_DG_PEN ||
				    field->logical == HID_DG_STYLUS ||
				    field->physical == HID_DG_STYLUS ||
				    field->application == HID_DG_DIGITIZER)
					map_abs_clear(ABS_DISTANCE);
				else
					map_abs_clear(usage->hid & 0xf);
			}
			break;

		case HID_GD_WHEEL:
			if (field->flags & HID_MAIN_ITEM_RELATIVE) {
				set_bit(REL_WHEEL, input->relbit);