Commit 9e723c53 authored by Linus Torvalds's avatar Linus Torvalds

Merge tag 'platform-drivers-x86-v5.14-3' of...

Merge tag 'platform-drivers-x86-v5.14-3' of git://git.kernel.org/pub/scm/linux/kernel/git/pdx86/platform-drivers-x86

Pull x86 platform driver fixes from Hans de Goede:
 "Small set of pdx86 fixes for 5.14"

* tag 'platform-drivers-x86-v5.14-3' of git://git.kernel.org/pub/scm/linux/kernel/git/pdx86/platform-drivers-x86:
  platform/x86: pcengines-apuv2: Add missing terminating entries to gpio-lookup tables
  platform/x86: Make dual_accel_detect() KIOX010A + KIOX020A detect more robust
  platform/x86: Add and use a dual_accel_detect() helper
parents b3f0ccc5 9d7b132e
...@@ -508,6 +508,7 @@ config THINKPAD_ACPI ...@@ -508,6 +508,7 @@ config THINKPAD_ACPI
depends on RFKILL || RFKILL = n depends on RFKILL || RFKILL = n
depends on ACPI_VIDEO || ACPI_VIDEO = n depends on ACPI_VIDEO || ACPI_VIDEO = n
depends on BACKLIGHT_CLASS_DEVICE depends on BACKLIGHT_CLASS_DEVICE
depends on I2C
select ACPI_PLATFORM_PROFILE select ACPI_PLATFORM_PROFILE
select HWMON select HWMON
select NVRAM select NVRAM
...@@ -691,6 +692,7 @@ config INTEL_HID_EVENT ...@@ -691,6 +692,7 @@ config INTEL_HID_EVENT
tristate "INTEL HID Event" tristate "INTEL HID Event"
depends on ACPI depends on ACPI
depends on INPUT depends on INPUT
depends on I2C
select INPUT_SPARSEKMAP select INPUT_SPARSEKMAP
help help
This driver provides support for the Intel HID Event hotkey interface. This driver provides support for the Intel HID Event hotkey interface.
...@@ -742,6 +744,7 @@ config INTEL_VBTN ...@@ -742,6 +744,7 @@ config INTEL_VBTN
tristate "INTEL VIRTUAL BUTTON" tristate "INTEL VIRTUAL BUTTON"
depends on ACPI depends on ACPI
depends on INPUT depends on INPUT
depends on I2C
select INPUT_SPARSEKMAP select INPUT_SPARSEKMAP
help help
This driver provides support for the Intel Virtual Button interface. This driver provides support for the Intel Virtual Button interface.
......
/* SPDX-License-Identifier: GPL-2.0 */
/*
* Helper code to detect 360 degree hinges (yoga) style 2-in-1 devices using 2 accelerometers
* to allow the OS to determine the angle between the display and the base of the device.
*
* On Windows these are read by a special HingeAngleService process which calls undocumented
* ACPI methods, to let the firmware know if the 2-in-1 is in tablet- or laptop-mode.
* The firmware may use this to disable the kbd and touchpad to avoid spurious input in
* tablet-mode as well as to report SW_TABLET_MODE info to the OS.
*
* Since Linux does not call these undocumented methods, the SW_TABLET_MODE info reported
* by various drivers/platform/x86 drivers is incorrect. These drivers use the detection
* code in this file to disable SW_TABLET_MODE reporting to avoid reporting broken info
* (instead userspace can derive the status itself by directly reading the 2 accels).
*/
#include <linux/acpi.h>
#include <linux/i2c.h>
static int dual_accel_i2c_resource_count(struct acpi_resource *ares, void *data)
{
struct acpi_resource_i2c_serialbus *sb;
int *count = data;
if (i2c_acpi_get_i2c_resource(ares, &sb))
*count = *count + 1;
return 1;
}
static int dual_accel_i2c_client_count(struct acpi_device *adev)
{
int ret, count = 0;
LIST_HEAD(r);
ret = acpi_dev_get_resources(adev, &r, dual_accel_i2c_resource_count, &count);
if (ret < 0)
return ret;
acpi_dev_free_resource_list(&r);
return count;
}
static bool dual_accel_detect_bosc0200(void)
{
struct acpi_device *adev;
int count;
adev = acpi_dev_get_first_match_dev("BOSC0200", NULL, -1);
if (!adev)
return false;
count = dual_accel_i2c_client_count(adev);
acpi_dev_put(adev);
return count == 2;
}
static bool dual_accel_detect(void)
{
/* Systems which use a pair of accels with KIOX010A / KIOX020A ACPI ids */
if (acpi_dev_present("KIOX010A", NULL, -1) &&
acpi_dev_present("KIOX020A", NULL, -1))
return true;
/* Systems which use a single DUAL250E ACPI device to model 2 accels */
if (acpi_dev_present("DUAL250E", NULL, -1))
return true;
/* Systems which use a single BOSC0200 ACPI device to model 2 accels */
if (dual_accel_detect_bosc0200())
return true;
return false;
}
...@@ -14,6 +14,7 @@ ...@@ -14,6 +14,7 @@
#include <linux/module.h> #include <linux/module.h>
#include <linux/platform_device.h> #include <linux/platform_device.h>
#include <linux/suspend.h> #include <linux/suspend.h>
#include "dual_accel_detect.h"
/* When NOT in tablet mode, VGBS returns with the flag 0x40 */ /* When NOT in tablet mode, VGBS returns with the flag 0x40 */
#define TABLET_MODE_FLAG BIT(6) #define TABLET_MODE_FLAG BIT(6)
...@@ -122,6 +123,7 @@ struct intel_hid_priv { ...@@ -122,6 +123,7 @@ struct intel_hid_priv {
struct input_dev *array; struct input_dev *array;
struct input_dev *switches; struct input_dev *switches;
bool wakeup_mode; bool wakeup_mode;
bool dual_accel;
}; };
#define HID_EVENT_FILTER_UUID "eeec56b3-4442-408f-a792-4edd4d758054" #define HID_EVENT_FILTER_UUID "eeec56b3-4442-408f-a792-4edd4d758054"
...@@ -451,22 +453,9 @@ static void notify_handler(acpi_handle handle, u32 event, void *context) ...@@ -451,22 +453,9 @@ static void notify_handler(acpi_handle handle, u32 event, void *context)
* SW_TABLET_MODE report, in these cases we enable support when receiving * SW_TABLET_MODE report, in these cases we enable support when receiving
* the first event instead of during driver setup. * the first event instead of during driver setup.
* *
* Some 360 degree hinges (yoga) style 2-in-1 devices use 2 accelerometers * See dual_accel_detect.h for more info on the dual_accel check.
* to allow the OS to determine the angle between the display and the base
* of the device. On Windows these are read by a special HingeAngleService
* process which calls an ACPI DSM (Device Specific Method) on the
* ACPI KIOX010A device node for the sensor in the display, to let the
* firmware know if the 2-in-1 is in tablet- or laptop-mode so that it can
* disable the kbd and touchpad to avoid spurious input in tablet-mode.
*
* The linux kxcjk1013 driver calls the DSM for this once at probe time
* to ensure that the builtin kbd and touchpad work. On some devices this
* causes a "spurious" 0xcd event on the intel-hid ACPI dev. In this case
* there is not a functional tablet-mode switch, so we should not register
* the tablet-mode switch device.
*/ */
if (!priv->switches && (event == 0xcc || event == 0xcd) && if (!priv->switches && !priv->dual_accel && (event == 0xcc || event == 0xcd)) {
!acpi_dev_present("KIOX010A", NULL, -1)) {
dev_info(&device->dev, "switch event received, enable switches supports\n"); dev_info(&device->dev, "switch event received, enable switches supports\n");
err = intel_hid_switches_setup(device); err = intel_hid_switches_setup(device);
if (err) if (err)
...@@ -607,6 +596,8 @@ static int intel_hid_probe(struct platform_device *device) ...@@ -607,6 +596,8 @@ static int intel_hid_probe(struct platform_device *device)
return -ENOMEM; return -ENOMEM;
dev_set_drvdata(&device->dev, priv); dev_set_drvdata(&device->dev, priv);
priv->dual_accel = dual_accel_detect();
err = intel_hid_input_setup(device); err = intel_hid_input_setup(device);
if (err) { if (err) {
pr_err("Failed to setup Intel HID hotkeys\n"); pr_err("Failed to setup Intel HID hotkeys\n");
......
...@@ -14,6 +14,7 @@ ...@@ -14,6 +14,7 @@
#include <linux/module.h> #include <linux/module.h>
#include <linux/platform_device.h> #include <linux/platform_device.h>
#include <linux/suspend.h> #include <linux/suspend.h>
#include "dual_accel_detect.h"
/* Returned when NOT in tablet mode on some HP Stream x360 11 models */ /* Returned when NOT in tablet mode on some HP Stream x360 11 models */
#define VGBS_TABLET_MODE_FLAG_ALT 0x10 #define VGBS_TABLET_MODE_FLAG_ALT 0x10
...@@ -66,6 +67,7 @@ static const struct key_entry intel_vbtn_switchmap[] = { ...@@ -66,6 +67,7 @@ static const struct key_entry intel_vbtn_switchmap[] = {
struct intel_vbtn_priv { struct intel_vbtn_priv {
struct input_dev *buttons_dev; struct input_dev *buttons_dev;
struct input_dev *switches_dev; struct input_dev *switches_dev;
bool dual_accel;
bool has_buttons; bool has_buttons;
bool has_switches; bool has_switches;
bool wakeup_mode; bool wakeup_mode;
...@@ -160,6 +162,10 @@ static void notify_handler(acpi_handle handle, u32 event, void *context) ...@@ -160,6 +162,10 @@ static void notify_handler(acpi_handle handle, u32 event, void *context)
input_dev = priv->buttons_dev; input_dev = priv->buttons_dev;
} else if ((ke = sparse_keymap_entry_from_scancode(priv->switches_dev, event))) { } else if ((ke = sparse_keymap_entry_from_scancode(priv->switches_dev, event))) {
if (!priv->has_switches) { if (!priv->has_switches) {
/* See dual_accel_detect.h for more info */
if (priv->dual_accel)
return;
dev_info(&device->dev, "Registering Intel Virtual Switches input-dev after receiving a switch event\n"); dev_info(&device->dev, "Registering Intel Virtual Switches input-dev after receiving a switch event\n");
ret = input_register_device(priv->switches_dev); ret = input_register_device(priv->switches_dev);
if (ret) if (ret)
...@@ -248,11 +254,15 @@ static const struct dmi_system_id dmi_switches_allow_list[] = { ...@@ -248,11 +254,15 @@ static const struct dmi_system_id dmi_switches_allow_list[] = {
{} /* Array terminator */ {} /* Array terminator */
}; };
static bool intel_vbtn_has_switches(acpi_handle handle) static bool intel_vbtn_has_switches(acpi_handle handle, bool dual_accel)
{ {
unsigned long long vgbs; unsigned long long vgbs;
acpi_status status; acpi_status status;
/* See dual_accel_detect.h for more info */
if (dual_accel)
return false;
if (!dmi_check_system(dmi_switches_allow_list)) if (!dmi_check_system(dmi_switches_allow_list))
return false; return false;
...@@ -263,13 +273,14 @@ static bool intel_vbtn_has_switches(acpi_handle handle) ...@@ -263,13 +273,14 @@ static bool intel_vbtn_has_switches(acpi_handle handle)
static int intel_vbtn_probe(struct platform_device *device) static int intel_vbtn_probe(struct platform_device *device)
{ {
acpi_handle handle = ACPI_HANDLE(&device->dev); acpi_handle handle = ACPI_HANDLE(&device->dev);
bool has_buttons, has_switches; bool dual_accel, has_buttons, has_switches;
struct intel_vbtn_priv *priv; struct intel_vbtn_priv *priv;
acpi_status status; acpi_status status;
int err; int err;
dual_accel = dual_accel_detect();
has_buttons = acpi_has_method(handle, "VBDL"); has_buttons = acpi_has_method(handle, "VBDL");
has_switches = intel_vbtn_has_switches(handle); has_switches = intel_vbtn_has_switches(handle, dual_accel);
if (!has_buttons && !has_switches) { if (!has_buttons && !has_switches) {
dev_warn(&device->dev, "failed to read Intel Virtual Button driver\n"); dev_warn(&device->dev, "failed to read Intel Virtual Button driver\n");
...@@ -281,6 +292,7 @@ static int intel_vbtn_probe(struct platform_device *device) ...@@ -281,6 +292,7 @@ static int intel_vbtn_probe(struct platform_device *device)
return -ENOMEM; return -ENOMEM;
dev_set_drvdata(&device->dev, priv); dev_set_drvdata(&device->dev, priv);
priv->dual_accel = dual_accel;
priv->has_buttons = has_buttons; priv->has_buttons = has_buttons;
priv->has_switches = has_switches; priv->has_switches = has_switches;
......
...@@ -94,6 +94,7 @@ static struct gpiod_lookup_table gpios_led_table = { ...@@ -94,6 +94,7 @@ static struct gpiod_lookup_table gpios_led_table = {
NULL, 1, GPIO_ACTIVE_LOW), NULL, 1, GPIO_ACTIVE_LOW),
GPIO_LOOKUP_IDX(AMD_FCH_GPIO_DRIVER_NAME, APU2_GPIO_LINE_LED3, GPIO_LOOKUP_IDX(AMD_FCH_GPIO_DRIVER_NAME, APU2_GPIO_LINE_LED3,
NULL, 2, GPIO_ACTIVE_LOW), NULL, 2, GPIO_ACTIVE_LOW),
{} /* Terminating entry */
} }
}; };
...@@ -123,6 +124,7 @@ static struct gpiod_lookup_table gpios_key_table = { ...@@ -123,6 +124,7 @@ static struct gpiod_lookup_table gpios_key_table = {
.table = { .table = {
GPIO_LOOKUP_IDX(AMD_FCH_GPIO_DRIVER_NAME, APU2_GPIO_LINE_MODESW, GPIO_LOOKUP_IDX(AMD_FCH_GPIO_DRIVER_NAME, APU2_GPIO_LINE_MODESW,
NULL, 0, GPIO_ACTIVE_LOW), NULL, 0, GPIO_ACTIVE_LOW),
{} /* Terminating entry */
} }
}; };
......
...@@ -73,6 +73,7 @@ ...@@ -73,6 +73,7 @@
#include <linux/uaccess.h> #include <linux/uaccess.h>
#include <acpi/battery.h> #include <acpi/battery.h>
#include <acpi/video.h> #include <acpi/video.h>
#include "dual_accel_detect.h"
/* ThinkPad CMOS commands */ /* ThinkPad CMOS commands */
#define TP_CMOS_VOLUME_DOWN 0 #define TP_CMOS_VOLUME_DOWN 0
...@@ -3232,7 +3233,7 @@ static int hotkey_init_tablet_mode(void) ...@@ -3232,7 +3233,7 @@ static int hotkey_init_tablet_mode(void)
* the laptop/tent/tablet mode to the EC. The bmc150 iio driver * the laptop/tent/tablet mode to the EC. The bmc150 iio driver
* does not support this, so skip the hotkey on these models. * does not support this, so skip the hotkey on these models.
*/ */
if (has_tablet_mode && !acpi_dev_present("BOSC0200", "1", -1)) if (has_tablet_mode && !dual_accel_detect())
tp_features.hotkey_tablet = TP_HOTKEY_TABLET_USES_GMMS; tp_features.hotkey_tablet = TP_HOTKEY_TABLET_USES_GMMS;
type = "GMMS"; type = "GMMS";
} else if (acpi_evalf(hkey_handle, &res, "MHKG", "qd")) { } else if (acpi_evalf(hkey_handle, &res, "MHKG", "qd")) {
......
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