Commit 007e50eb authored by Hans de Goede's avatar Hans de Goede Committed by Dmitry Torokhov

Input: hideep - optionally reset controller work mode to native HiDeep protocol

The HiDeep IST940E touchscreen controller used on the Lenovo Yoga Book X90F
convertible comes up in HID mode by default.

This works well on the X91F Windows model where the touchscreen is
correctly described in ACPI and ACPI takes care of controlling
the reset GPIO and regulators.

But the X90F ships with Android and the ACPI tables on this model don't
describe the touchscreen. Instead this is hardcoded in the vendor kernel.

The vendor kernel uses the touchscreen in native HiDeep 20 (2.0?) protocol
mode and switches the controller to this mode by writing 0 to reg 0x081e.

Adjusting the i2c-hid code to deal with the reset-gpio and regulators on
this non devicetree (but rather broken ACPI) convertible is somewhat tricky
and the native protocol reports ABS_MT_PRESSURE and ABS_MT_TOUCH_MAJOR
which are not reported in HID mode, so it is preferable to use the native
mode.

Add support to the hideep driver to reset the work-mode to the native
HiDeep protocol to allow using it on the Lenovo Yoga Book X90F.
This is guarded behind a new "hideep,force-native-protocol" boolean
property, to avoid changing behavior on other devices.

For the record: I did test using the i2c-hid driver with some quick hacks
and it does work. The I2C-HID descriptor is available from address 0x0020,
just like on the X91F Windows model.

So far the new "hideep,force-native-protocol" property is only used on
x86/ACPI (non devicetree) devs. IOW it is not used in actual devicetree
files. The devicetree-bindings maintainers have requested properties like
these to not be added to the devicetree-bindings, so the new property is
deliberately not added to the existing devicetree-bindings.
Signed-off-by: default avatarHans de Goede <hdegoede@redhat.com>
Link: https://lore.kernel.org/r/20230311114726.182789-3-hdegoede@redhat.comSigned-off-by: default avatarDmitry Torokhov <dmitry.torokhov@gmail.com>
parent 10b0a455
...@@ -35,6 +35,7 @@ ...@@ -35,6 +35,7 @@
#define HIDEEP_EVENT_ADDR 0x240 #define HIDEEP_EVENT_ADDR 0x240
/* command list */ /* command list */
#define HIDEEP_WORK_MODE 0x081e
#define HIDEEP_RESET_CMD 0x9800 #define HIDEEP_RESET_CMD 0x9800
/* event bit */ /* event bit */
...@@ -964,6 +965,21 @@ static const struct attribute_group hideep_ts_attr_group = { ...@@ -964,6 +965,21 @@ static const struct attribute_group hideep_ts_attr_group = {
.attrs = hideep_ts_sysfs_entries, .attrs = hideep_ts_sysfs_entries,
}; };
static void hideep_set_work_mode(struct hideep_ts *ts)
{
/*
* Reset touch report format to the native HiDeep 20 protocol if requested.
* This is necessary to make touchscreens which come up in I2C-HID mode
* work with this driver.
*
* Note this is a kernel internal device-property set by x86 platform code,
* this MUST not be used in devicetree files without first adding it to
* the DT bindings.
*/
if (device_property_read_bool(&ts->client->dev, "hideep,force-native-protocol"))
regmap_write(ts->reg, HIDEEP_WORK_MODE, 0x00);
}
static int hideep_suspend(struct device *dev) static int hideep_suspend(struct device *dev)
{ {
struct i2c_client *client = to_i2c_client(dev); struct i2c_client *client = to_i2c_client(dev);
...@@ -987,6 +1003,8 @@ static int hideep_resume(struct device *dev) ...@@ -987,6 +1003,8 @@ static int hideep_resume(struct device *dev)
return error; return error;
} }
hideep_set_work_mode(ts);
enable_irq(client->irq); enable_irq(client->irq);
return 0; return 0;
...@@ -1063,6 +1081,8 @@ static int hideep_probe(struct i2c_client *client) ...@@ -1063,6 +1081,8 @@ static int hideep_probe(struct i2c_client *client)
return error; return error;
} }
hideep_set_work_mode(ts);
error = hideep_init_input(ts); error = hideep_init_input(ts);
if (error) if (error)
return error; return error;
......
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