summaryrefslogtreecommitdiffstats
path: root/drivers/platform/x86/dual_accel_detect.h
blob: 1a069159da91afcdff7c4a8de81607c9248b7c4b (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
/* 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))
		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;
}