1 // SPDX-License-Identifier: GPL-2.0-only
3 * Supports for the button array on SoC tablets originally running
6 * (C) Copyright 2014 Intel Corporation
9 #include <linux/module.h>
10 #include <linux/input.h>
11 #include <linux/init.h>
12 #include <linux/kernel.h>
13 #include <linux/acpi.h>
14 #include <linux/gpio/consumer.h>
15 #include <linux/gpio_keys.h>
16 #include <linux/gpio.h>
17 #include <linux/platform_device.h>
19 struct soc_button_info
{
22 unsigned int event_type
;
23 unsigned int event_code
;
28 struct soc_device_data
{
29 const struct soc_button_info
*button_info
;
30 int (*check
)(struct device
*dev
);
34 * Some of the buttons like volume up/down are auto repeat, while others
35 * are not. To support both, we register two platform devices, and put
36 * buttons into them based on whether the key should be auto repeat.
38 #define BUTTON_TYPES 2
40 struct soc_button_data
{
41 struct platform_device
*children
[BUTTON_TYPES
];
45 * Get the Nth GPIO number from the ACPI object.
47 static int soc_button_lookup_gpio(struct device
*dev
, int acpi_index
)
49 struct gpio_desc
*desc
;
52 desc
= gpiod_get_index(dev
, NULL
, acpi_index
, GPIOD_ASIS
);
56 gpio
= desc_to_gpio(desc
);
63 static struct platform_device
*
64 soc_button_device_create(struct platform_device
*pdev
,
65 const struct soc_button_info
*button_info
,
68 const struct soc_button_info
*info
;
69 struct platform_device
*pd
;
70 struct gpio_keys_button
*gpio_keys
;
71 struct gpio_keys_platform_data
*gpio_keys_pdata
;
76 for (info
= button_info
; info
->name
; info
++)
77 if (info
->autorepeat
== autorepeat
)
80 gpio_keys_pdata
= devm_kzalloc(&pdev
->dev
,
81 sizeof(*gpio_keys_pdata
) +
82 sizeof(*gpio_keys
) * n_buttons
,
85 return ERR_PTR(-ENOMEM
);
87 gpio_keys
= (void *)(gpio_keys_pdata
+ 1);
90 for (info
= button_info
; info
->name
; info
++) {
91 if (info
->autorepeat
!= autorepeat
)
94 gpio
= soc_button_lookup_gpio(&pdev
->dev
, info
->acpi_index
);
95 if (!gpio_is_valid(gpio
)) {
97 * Skip GPIO if not present. Note we deliberately
98 * ignore -EPROBE_DEFER errors here. On some devices
99 * Intel is using so called virtual GPIOs which are not
100 * GPIOs at all but some way for AML code to check some
101 * random status bits without need a custom opregion.
102 * In some cases the resources table we parse points to
103 * such a virtual GPIO, since these are not real GPIOs
104 * we do not have a driver for these so they will never
105 * show up, therefore we ignore -EPROBE_DEFER.
110 gpio_keys
[n_buttons
].type
= info
->event_type
;
111 gpio_keys
[n_buttons
].code
= info
->event_code
;
112 gpio_keys
[n_buttons
].gpio
= gpio
;
113 gpio_keys
[n_buttons
].active_low
= 1;
114 gpio_keys
[n_buttons
].desc
= info
->name
;
115 gpio_keys
[n_buttons
].wakeup
= info
->wakeup
;
116 /* These devices often use cheap buttons, use 50 ms debounce */
117 gpio_keys
[n_buttons
].debounce_interval
= 50;
121 if (n_buttons
== 0) {
126 gpio_keys_pdata
->buttons
= gpio_keys
;
127 gpio_keys_pdata
->nbuttons
= n_buttons
;
128 gpio_keys_pdata
->rep
= autorepeat
;
130 pd
= platform_device_register_resndata(&pdev
->dev
, "gpio-keys",
131 PLATFORM_DEVID_AUTO
, NULL
, 0,
133 sizeof(*gpio_keys_pdata
));
134 error
= PTR_ERR_OR_ZERO(pd
);
137 "failed registering gpio-keys: %d\n", error
);
144 devm_kfree(&pdev
->dev
, gpio_keys_pdata
);
145 return ERR_PTR(error
);
148 static int soc_button_get_acpi_object_int(const union acpi_object
*obj
)
150 if (obj
->type
!= ACPI_TYPE_INTEGER
)
153 return obj
->integer
.value
;
156 /* Parse a single ACPI0011 _DSD button descriptor */
157 static int soc_button_parse_btn_desc(struct device
*dev
,
158 const union acpi_object
*desc
,
160 struct soc_button_info
*info
)
164 if (desc
->type
!= ACPI_TYPE_PACKAGE
||
165 desc
->package
.count
!= 5 ||
166 /* First byte should be 1 (control) */
167 soc_button_get_acpi_object_int(&desc
->package
.elements
[0]) != 1 ||
168 /* Third byte should be collection uid */
169 soc_button_get_acpi_object_int(&desc
->package
.elements
[2]) !=
171 dev_err(dev
, "Invalid ACPI Button Descriptor\n");
175 info
->event_type
= EV_KEY
;
177 soc_button_get_acpi_object_int(&desc
->package
.elements
[1]);
178 upage
= soc_button_get_acpi_object_int(&desc
->package
.elements
[3]);
179 usage
= soc_button_get_acpi_object_int(&desc
->package
.elements
[4]);
182 * The UUID: fa6bd625-9ce8-470d-a2c7-b3ca36c4282e descriptors use HID
183 * usage page and usage codes, but otherwise the device is not HID
184 * compliant: it uses one irq per button instead of generating HID
185 * input reports and some buttons should generate wakeups where as
186 * others should not, so we cannot use the HID subsystem.
188 * Luckily all devices only use a few usage page + usage combinations,
189 * so we can simply check for the known combinations here.
191 if (upage
== 0x01 && usage
== 0x81) {
192 info
->name
= "power";
193 info
->event_code
= KEY_POWER
;
195 } else if (upage
== 0x01 && usage
== 0xca) {
196 info
->name
= "rotation lock switch";
197 info
->event_type
= EV_SW
;
198 info
->event_code
= SW_ROTATE_LOCK
;
199 } else if (upage
== 0x07 && usage
== 0xe3) {
201 info
->event_code
= KEY_LEFTMETA
;
203 } else if (upage
== 0x0c && usage
== 0xe9) {
204 info
->name
= "volume_up";
205 info
->event_code
= KEY_VOLUMEUP
;
206 info
->autorepeat
= true;
207 } else if (upage
== 0x0c && usage
== 0xea) {
208 info
->name
= "volume_down";
209 info
->event_code
= KEY_VOLUMEDOWN
;
210 info
->autorepeat
= true;
212 dev_warn(dev
, "Unknown button index %d upage %02x usage %02x, ignoring\n",
213 info
->acpi_index
, upage
, usage
);
214 info
->name
= "unknown";
215 info
->event_code
= KEY_RESERVED
;
221 /* ACPI0011 _DSD btns descriptors UUID: fa6bd625-9ce8-470d-a2c7-b3ca36c4282e */
222 static const u8 btns_desc_uuid
[16] = {
223 0x25, 0xd6, 0x6b, 0xfa, 0xe8, 0x9c, 0x0d, 0x47,
224 0xa2, 0xc7, 0xb3, 0xca, 0x36, 0xc4, 0x28, 0x2e
227 /* Parse ACPI0011 _DSD button descriptors */
228 static struct soc_button_info
*soc_button_get_button_info(struct device
*dev
)
230 struct acpi_buffer buf
= { ACPI_ALLOCATE_BUFFER
};
231 const union acpi_object
*desc
, *el0
, *uuid
, *btns_desc
= NULL
;
232 struct soc_button_info
*button_info
;
234 int i
, btn
, collection_uid
= -1;
236 status
= acpi_evaluate_object_typed(ACPI_HANDLE(dev
), "_DSD", NULL
,
237 &buf
, ACPI_TYPE_PACKAGE
);
238 if (ACPI_FAILURE(status
)) {
239 dev_err(dev
, "ACPI _DSD object not found\n");
240 return ERR_PTR(-ENODEV
);
243 /* Look for the Button Descriptors UUID */
245 for (i
= 0; (i
+ 1) < desc
->package
.count
; i
+= 2) {
246 uuid
= &desc
->package
.elements
[i
];
248 if (uuid
->type
!= ACPI_TYPE_BUFFER
||
249 uuid
->buffer
.length
!= 16 ||
250 desc
->package
.elements
[i
+ 1].type
!= ACPI_TYPE_PACKAGE
) {
254 if (memcmp(uuid
->buffer
.pointer
, btns_desc_uuid
, 16) == 0) {
255 btns_desc
= &desc
->package
.elements
[i
+ 1];
261 dev_err(dev
, "ACPI Button Descriptors not found\n");
262 button_info
= ERR_PTR(-ENODEV
);
266 /* The first package describes the collection */
267 el0
= &btns_desc
->package
.elements
[0];
268 if (el0
->type
== ACPI_TYPE_PACKAGE
&&
269 el0
->package
.count
== 5 &&
270 /* First byte should be 0 (collection) */
271 soc_button_get_acpi_object_int(&el0
->package
.elements
[0]) == 0 &&
272 /* Third byte should be 0 (top level collection) */
273 soc_button_get_acpi_object_int(&el0
->package
.elements
[2]) == 0) {
274 collection_uid
= soc_button_get_acpi_object_int(
275 &el0
->package
.elements
[1]);
277 if (collection_uid
== -1) {
278 dev_err(dev
, "Invalid Button Collection Descriptor\n");
279 button_info
= ERR_PTR(-ENODEV
);
283 /* There are package.count - 1 buttons + 1 terminating empty entry */
284 button_info
= devm_kcalloc(dev
, btns_desc
->package
.count
,
285 sizeof(*button_info
), GFP_KERNEL
);
287 button_info
= ERR_PTR(-ENOMEM
);
291 /* Parse the button descriptors */
292 for (i
= 1, btn
= 0; i
< btns_desc
->package
.count
; i
++, btn
++) {
293 if (soc_button_parse_btn_desc(dev
,
294 &btns_desc
->package
.elements
[i
],
296 &button_info
[btn
])) {
297 button_info
= ERR_PTR(-ENODEV
);
307 static int soc_button_remove(struct platform_device
*pdev
)
309 struct soc_button_data
*priv
= platform_get_drvdata(pdev
);
313 for (i
= 0; i
< BUTTON_TYPES
; i
++)
314 if (priv
->children
[i
])
315 platform_device_unregister(priv
->children
[i
]);
320 static int soc_button_probe(struct platform_device
*pdev
)
322 struct device
*dev
= &pdev
->dev
;
323 const struct soc_device_data
*device_data
;
324 const struct soc_button_info
*button_info
;
325 struct soc_button_data
*priv
;
326 struct platform_device
*pd
;
330 device_data
= acpi_device_get_match_data(dev
);
331 if (device_data
&& device_data
->check
) {
332 error
= device_data
->check(dev
);
337 if (device_data
&& device_data
->button_info
) {
338 button_info
= device_data
->button_info
;
340 button_info
= soc_button_get_button_info(dev
);
341 if (IS_ERR(button_info
))
342 return PTR_ERR(button_info
);
345 error
= gpiod_count(dev
, NULL
);
347 dev_dbg(dev
, "no GPIO attached, ignoring...\n");
351 priv
= devm_kzalloc(dev
, sizeof(*priv
), GFP_KERNEL
);
355 platform_set_drvdata(pdev
, priv
);
357 for (i
= 0; i
< BUTTON_TYPES
; i
++) {
358 pd
= soc_button_device_create(pdev
, button_info
, i
== 0);
361 if (error
!= -ENODEV
) {
362 soc_button_remove(pdev
);
368 priv
->children
[i
] = pd
;
371 if (!priv
->children
[0] && !priv
->children
[1])
374 if (!device_data
|| !device_data
->button_info
)
375 devm_kfree(dev
, button_info
);
381 * Definition of buttons on the tablet. The ACPI index of each button
382 * is defined in section 2.8.7.2 of "Windows ACPI Design Guide for SoC
385 static const struct soc_button_info soc_button_PNP0C40
[] = {
386 { "power", 0, EV_KEY
, KEY_POWER
, false, true },
387 { "home", 1, EV_KEY
, KEY_LEFTMETA
, false, true },
388 { "volume_up", 2, EV_KEY
, KEY_VOLUMEUP
, true, false },
389 { "volume_down", 3, EV_KEY
, KEY_VOLUMEDOWN
, true, false },
390 { "rotation_lock", 4, EV_KEY
, KEY_ROTATE_LOCK_TOGGLE
, false, false },
394 static const struct soc_device_data soc_device_PNP0C40
= {
395 .button_info
= soc_button_PNP0C40
,
399 * Special device check for Surface Book 2 and Surface Pro (2017).
400 * Both, the Surface Pro 4 (surfacepro3_button.c) and the above mentioned
401 * devices use MSHW0040 for power and volume buttons, however the way they
402 * have to be addressed differs. Make sure that we only load this drivers
403 * for the correct devices by checking the OEM Platform Revision provided by
406 #define MSHW0040_DSM_REVISION 0x01
407 #define MSHW0040_DSM_GET_OMPR 0x02 // get OEM Platform Revision
408 static const guid_t MSHW0040_DSM_UUID
=
409 GUID_INIT(0x6fd05c69, 0xcde3, 0x49f4, 0x95, 0xed, 0xab, 0x16, 0x65,
412 static int soc_device_check_MSHW0040(struct device
*dev
)
414 acpi_handle handle
= ACPI_HANDLE(dev
);
415 union acpi_object
*result
;
416 u64 oem_platform_rev
= 0; // valid revisions are nonzero
418 // get OEM platform revision
419 result
= acpi_evaluate_dsm_typed(handle
, &MSHW0040_DSM_UUID
,
420 MSHW0040_DSM_REVISION
,
421 MSHW0040_DSM_GET_OMPR
, NULL
,
425 oem_platform_rev
= result
->integer
.value
;
430 * If the revision is zero here, the _DSM evaluation has failed. This
431 * indicates that we have a Pro 4 or Book 1 and this driver should not
434 if (oem_platform_rev
== 0)
437 dev_dbg(dev
, "OEM Platform Revision %llu\n", oem_platform_rev
);
443 * Button infos for Microsoft Surface Book 2 and Surface Pro (2017).
444 * Obtained from DSDT/testing.
446 static const struct soc_button_info soc_button_MSHW0040
[] = {
447 { "power", 0, EV_KEY
, KEY_POWER
, false, true },
448 { "volume_up", 2, EV_KEY
, KEY_VOLUMEUP
, true, false },
449 { "volume_down", 4, EV_KEY
, KEY_VOLUMEDOWN
, true, false },
453 static const struct soc_device_data soc_device_MSHW0040
= {
454 .button_info
= soc_button_MSHW0040
,
455 .check
= soc_device_check_MSHW0040
,
458 static const struct acpi_device_id soc_button_acpi_match
[] = {
459 { "PNP0C40", (unsigned long)&soc_device_PNP0C40
},
462 /* Microsoft Surface Devices (5th and 6th generation) */
463 { "MSHW0040", (unsigned long)&soc_device_MSHW0040
},
468 MODULE_DEVICE_TABLE(acpi
, soc_button_acpi_match
);
470 static struct platform_driver soc_button_driver
= {
471 .probe
= soc_button_probe
,
472 .remove
= soc_button_remove
,
474 .name
= KBUILD_MODNAME
,
475 .acpi_match_table
= ACPI_PTR(soc_button_acpi_match
),
478 module_platform_driver(soc_button_driver
);
480 MODULE_LICENSE("GPL");