2 * Supports for the button array on SoC tablets originally running
5 * (C) Copyright 2014 Intel Corporation
7 * This program is free software; you can redistribute it and/or
8 * modify it under the terms of the GNU General Public License
9 * as published by the Free Software Foundation; version 2
13 #include <linux/module.h>
14 #include <linux/input.h>
15 #include <linux/init.h>
16 #include <linux/kernel.h>
17 #include <linux/acpi.h>
18 #include <linux/gpio/consumer.h>
19 #include <linux/gpio_keys.h>
20 #include <linux/gpio.h>
21 #include <linux/platform_device.h>
23 struct soc_button_info
{
26 unsigned int event_type
;
27 unsigned int event_code
;
33 * Some of the buttons like volume up/down are auto repeat, while others
34 * are not. To support both, we register two platform devices, and put
35 * buttons into them based on whether the key should be auto repeat.
37 #define BUTTON_TYPES 2
39 struct soc_button_data
{
40 struct platform_device
*children
[BUTTON_TYPES
];
44 * Get the Nth GPIO number from the ACPI object.
46 static int soc_button_lookup_gpio(struct device
*dev
, int acpi_index
)
48 struct gpio_desc
*desc
;
51 desc
= gpiod_get_index(dev
, NULL
, acpi_index
, GPIOD_ASIS
);
55 gpio
= desc_to_gpio(desc
);
62 static struct platform_device
*
63 soc_button_device_create(struct platform_device
*pdev
,
64 const struct soc_button_info
*button_info
,
67 const struct soc_button_info
*info
;
68 struct platform_device
*pd
;
69 struct gpio_keys_button
*gpio_keys
;
70 struct gpio_keys_platform_data
*gpio_keys_pdata
;
75 for (info
= button_info
; info
->name
; info
++)
76 if (info
->autorepeat
== autorepeat
)
79 gpio_keys_pdata
= devm_kzalloc(&pdev
->dev
,
80 sizeof(*gpio_keys_pdata
) +
81 sizeof(*gpio_keys
) * n_buttons
,
84 return ERR_PTR(-ENOMEM
);
86 gpio_keys
= (void *)(gpio_keys_pdata
+ 1);
89 for (info
= button_info
; info
->name
; info
++) {
90 if (info
->autorepeat
!= autorepeat
)
93 gpio
= soc_button_lookup_gpio(&pdev
->dev
, info
->acpi_index
);
94 if (!gpio_is_valid(gpio
))
97 gpio_keys
[n_buttons
].type
= info
->event_type
;
98 gpio_keys
[n_buttons
].code
= info
->event_code
;
99 gpio_keys
[n_buttons
].gpio
= gpio
;
100 gpio_keys
[n_buttons
].active_low
= 1;
101 gpio_keys
[n_buttons
].desc
= info
->name
;
102 gpio_keys
[n_buttons
].wakeup
= info
->wakeup
;
103 /* These devices often use cheap buttons, use 50 ms debounce */
104 gpio_keys
[n_buttons
].debounce_interval
= 50;
108 if (n_buttons
== 0) {
113 gpio_keys_pdata
->buttons
= gpio_keys
;
114 gpio_keys_pdata
->nbuttons
= n_buttons
;
115 gpio_keys_pdata
->rep
= autorepeat
;
117 pd
= platform_device_alloc("gpio-keys", PLATFORM_DEVID_AUTO
);
123 error
= platform_device_add_data(pd
, gpio_keys_pdata
,
124 sizeof(*gpio_keys_pdata
));
128 error
= platform_device_add(pd
);
135 platform_device_put(pd
);
137 devm_kfree(&pdev
->dev
, gpio_keys_pdata
);
138 return ERR_PTR(error
);
141 static int soc_button_get_acpi_object_int(const union acpi_object
*obj
)
143 if (obj
->type
!= ACPI_TYPE_INTEGER
)
146 return obj
->integer
.value
;
149 /* Parse a single ACPI0011 _DSD button descriptor */
150 static int soc_button_parse_btn_desc(struct device
*dev
,
151 const union acpi_object
*desc
,
153 struct soc_button_info
*info
)
157 if (desc
->type
!= ACPI_TYPE_PACKAGE
||
158 desc
->package
.count
!= 5 ||
159 /* First byte should be 1 (control) */
160 soc_button_get_acpi_object_int(&desc
->package
.elements
[0]) != 1 ||
161 /* Third byte should be collection uid */
162 soc_button_get_acpi_object_int(&desc
->package
.elements
[2]) !=
164 dev_err(dev
, "Invalid ACPI Button Descriptor\n");
168 info
->event_type
= EV_KEY
;
170 soc_button_get_acpi_object_int(&desc
->package
.elements
[1]);
171 upage
= soc_button_get_acpi_object_int(&desc
->package
.elements
[3]);
172 usage
= soc_button_get_acpi_object_int(&desc
->package
.elements
[4]);
175 * The UUID: fa6bd625-9ce8-470d-a2c7-b3ca36c4282e descriptors use HID
176 * usage page and usage codes, but otherwise the device is not HID
177 * compliant: it uses one irq per button instead of generating HID
178 * input reports and some buttons should generate wakeups where as
179 * others should not, so we cannot use the HID subsystem.
181 * Luckily all devices only use a few usage page + usage combinations,
182 * so we can simply check for the known combinations here.
184 if (upage
== 0x01 && usage
== 0x81) {
185 info
->name
= "power";
186 info
->event_code
= KEY_POWER
;
188 } else if (upage
== 0x07 && usage
== 0xe3) {
190 info
->event_code
= KEY_LEFTMETA
;
192 } else if (upage
== 0x0c && usage
== 0xe9) {
193 info
->name
= "volume_up";
194 info
->event_code
= KEY_VOLUMEUP
;
195 info
->autorepeat
= true;
196 } else if (upage
== 0x0c && usage
== 0xea) {
197 info
->name
= "volume_down";
198 info
->event_code
= KEY_VOLUMEDOWN
;
199 info
->autorepeat
= true;
201 dev_warn(dev
, "Unknown button index %d upage %02x usage %02x, ignoring\n",
202 info
->acpi_index
, upage
, usage
);
203 info
->name
= "unknown";
204 info
->event_code
= KEY_RESERVED
;
210 /* ACPI0011 _DSD btns descriptors UUID: fa6bd625-9ce8-470d-a2c7-b3ca36c4282e */
211 static const u8 btns_desc_uuid
[16] = {
212 0x25, 0xd6, 0x6b, 0xfa, 0xe8, 0x9c, 0x0d, 0x47,
213 0xa2, 0xc7, 0xb3, 0xca, 0x36, 0xc4, 0x28, 0x2e
216 /* Parse ACPI0011 _DSD button descriptors */
217 static struct soc_button_info
*soc_button_get_button_info(struct device
*dev
)
219 struct acpi_buffer buf
= { ACPI_ALLOCATE_BUFFER
};
220 const union acpi_object
*desc
, *el0
, *uuid
, *btns_desc
= NULL
;
221 struct soc_button_info
*button_info
;
223 int i
, btn
, collection_uid
= -1;
225 status
= acpi_evaluate_object_typed(ACPI_HANDLE(dev
), "_DSD", NULL
,
226 &buf
, ACPI_TYPE_PACKAGE
);
227 if (ACPI_FAILURE(status
)) {
228 dev_err(dev
, "ACPI _DSD object not found\n");
229 return ERR_PTR(-ENODEV
);
232 /* Look for the Button Descriptors UUID */
234 for (i
= 0; (i
+ 1) < desc
->package
.count
; i
+= 2) {
235 uuid
= &desc
->package
.elements
[i
];
237 if (uuid
->type
!= ACPI_TYPE_BUFFER
||
238 uuid
->buffer
.length
!= 16 ||
239 desc
->package
.elements
[i
+ 1].type
!= ACPI_TYPE_PACKAGE
) {
243 if (memcmp(uuid
->buffer
.pointer
, btns_desc_uuid
, 16) == 0) {
244 btns_desc
= &desc
->package
.elements
[i
+ 1];
250 dev_err(dev
, "ACPI Button Descriptors not found\n");
251 button_info
= ERR_PTR(-ENODEV
);
255 /* The first package describes the collection */
256 el0
= &btns_desc
->package
.elements
[0];
257 if (el0
->type
== ACPI_TYPE_PACKAGE
&&
258 el0
->package
.count
== 5 &&
259 /* First byte should be 0 (collection) */
260 soc_button_get_acpi_object_int(&el0
->package
.elements
[0]) == 0 &&
261 /* Third byte should be 0 (top level collection) */
262 soc_button_get_acpi_object_int(&el0
->package
.elements
[2]) == 0) {
263 collection_uid
= soc_button_get_acpi_object_int(
264 &el0
->package
.elements
[1]);
266 if (collection_uid
== -1) {
267 dev_err(dev
, "Invalid Button Collection Descriptor\n");
268 button_info
= ERR_PTR(-ENODEV
);
272 /* There are package.count - 1 buttons + 1 terminating empty entry */
273 button_info
= devm_kcalloc(dev
, btns_desc
->package
.count
,
274 sizeof(*button_info
), GFP_KERNEL
);
276 button_info
= ERR_PTR(-ENOMEM
);
280 /* Parse the button descriptors */
281 for (i
= 1, btn
= 0; i
< btns_desc
->package
.count
; i
++, btn
++) {
282 if (soc_button_parse_btn_desc(dev
,
283 &btns_desc
->package
.elements
[i
],
285 &button_info
[btn
])) {
286 button_info
= ERR_PTR(-ENODEV
);
296 static int soc_button_remove(struct platform_device
*pdev
)
298 struct soc_button_data
*priv
= platform_get_drvdata(pdev
);
302 for (i
= 0; i
< BUTTON_TYPES
; i
++)
303 if (priv
->children
[i
])
304 platform_device_unregister(priv
->children
[i
]);
309 static int soc_button_probe(struct platform_device
*pdev
)
311 struct device
*dev
= &pdev
->dev
;
312 const struct acpi_device_id
*id
;
313 struct soc_button_info
*button_info
;
314 struct soc_button_data
*priv
;
315 struct platform_device
*pd
;
319 id
= acpi_match_device(dev
->driver
->acpi_match_table
, dev
);
323 if (!id
->driver_data
) {
324 button_info
= soc_button_get_button_info(dev
);
325 if (IS_ERR(button_info
))
326 return PTR_ERR(button_info
);
328 button_info
= (struct soc_button_info
*)id
->driver_data
;
331 error
= gpiod_count(dev
, NULL
);
333 dev_dbg(dev
, "no GPIO attached, ignoring...\n");
337 priv
= devm_kzalloc(dev
, sizeof(*priv
), GFP_KERNEL
);
341 platform_set_drvdata(pdev
, priv
);
343 for (i
= 0; i
< BUTTON_TYPES
; i
++) {
344 pd
= soc_button_device_create(pdev
, button_info
, i
== 0);
347 if (error
!= -ENODEV
) {
348 soc_button_remove(pdev
);
354 priv
->children
[i
] = pd
;
357 if (!priv
->children
[0] && !priv
->children
[1])
360 if (!id
->driver_data
)
361 devm_kfree(dev
, button_info
);
367 * Definition of buttons on the tablet. The ACPI index of each button
368 * is defined in section 2.8.7.2 of "Windows ACPI Design Guide for SoC
371 static struct soc_button_info soc_button_PNP0C40
[] = {
372 { "power", 0, EV_KEY
, KEY_POWER
, false, true },
373 { "home", 1, EV_KEY
, KEY_LEFTMETA
, false, true },
374 { "volume_up", 2, EV_KEY
, KEY_VOLUMEUP
, true, false },
375 { "volume_down", 3, EV_KEY
, KEY_VOLUMEDOWN
, true, false },
376 { "rotation_lock", 4, EV_SW
, SW_ROTATE_LOCK
, false, false },
380 static const struct acpi_device_id soc_button_acpi_match
[] = {
381 { "PNP0C40", (unsigned long)soc_button_PNP0C40
},
386 MODULE_DEVICE_TABLE(acpi
, soc_button_acpi_match
);
388 static struct platform_driver soc_button_driver
= {
389 .probe
= soc_button_probe
,
390 .remove
= soc_button_remove
,
392 .name
= KBUILD_MODNAME
,
393 .acpi_match_table
= ACPI_PTR(soc_button_acpi_match
),
396 module_platform_driver(soc_button_driver
);
398 MODULE_LICENSE("GPL");