1 // SPDX-License-Identifier: GPL-2.0+
3 * DMI based code to deal with broken DSDTs on X86 tablets which ship with
4 * Android as (part of) the factory image. The factory kernels shipped on these
5 * devices typically have a bunch of things hardcoded, rather than specified
11 #define pr_fmt(fmt) KBUILD_MODNAME ": " fmt
13 #include <linux/acpi.h>
14 #include <linux/dmi.h>
15 #include <linux/efi.h>
16 #include <linux/gpio_keys.h>
17 #include <linux/gpio/consumer.h>
18 #include <linux/gpio/driver.h>
19 #include <linux/gpio/machine.h>
20 #include <linux/i2c.h>
21 #include <linux/input.h>
22 #include <linux/irq.h>
23 #include <linux/irqdomain.h>
24 #include <linux/module.h>
25 #include <linux/mod_devicetable.h>
26 #include <linux/pinctrl/consumer.h>
27 #include <linux/pinctrl/machine.h>
28 #include <linux/platform_data/lp855x.h>
29 #include <linux/platform_device.h>
30 #include <linux/power/bq24190_charger.h>
31 #include <linux/reboot.h>
32 #include <linux/rmi.h>
33 #include <linux/serdev.h>
34 #include <linux/spi/spi.h>
35 #include <linux/string.h>
36 /* For gpio_get_desc() which is EXPORT_SYMBOL_GPL() */
37 #include "../../gpio/gpiolib.h"
38 #include "../../gpio/gpiolib-acpi.h"
41 * Helper code to get Linux IRQ numbers given a description of the IRQ source
42 * (either IOAPIC index, or GPIO chip name + pin-number).
44 enum x86_acpi_irq_type {
45 X86_ACPI_IRQ_TYPE_NONE,
46 X86_ACPI_IRQ_TYPE_APIC,
47 X86_ACPI_IRQ_TYPE_GPIOINT,
48 X86_ACPI_IRQ_TYPE_PMIC,
51 struct x86_acpi_irq_data {
52 char *chip; /* GPIO chip label (GPIOINT) or PMIC ACPI path (PMIC) */
53 enum x86_acpi_irq_type type;
54 enum irq_domain_bus_token domain;
56 int trigger; /* ACPI_EDGE_SENSITIVE / ACPI_LEVEL_SENSITIVE */
57 int polarity; /* ACPI_ACTIVE_HIGH / ACPI_ACTIVE_LOW / ACPI_ACTIVE_BOTH */
60 static int gpiochip_find_match_label(struct gpio_chip *gc, void *data)
62 return gc->label && !strcmp(gc->label, data);
65 static int x86_android_tablet_get_gpiod(char *label, int pin, struct gpio_desc **desc)
67 struct gpio_desc *gpiod;
68 struct gpio_chip *chip;
70 chip = gpiochip_find(label, gpiochip_find_match_label);
72 pr_err("error cannot find GPIO chip %s\n", label);
76 gpiod = gpiochip_get_desc(chip, pin);
78 pr_err("error %ld getting GPIO %s %d\n", PTR_ERR(gpiod), label, pin);
79 return PTR_ERR(gpiod);
86 static int x86_acpi_irq_helper_get(const struct x86_acpi_irq_data *data)
88 struct irq_fwspec fwspec = { };
89 struct irq_domain *domain;
90 struct acpi_device *adev;
91 struct gpio_desc *gpiod;
92 unsigned int irq_type;
98 case X86_ACPI_IRQ_TYPE_APIC:
100 * The DSDT may already reference the GSI in a device skipped by
101 * acpi_quirk_skip_i2c_client_enumeration(). Unregister the GSI
102 * to avoid EBUSY errors in this case.
104 acpi_unregister_gsi(data->index);
105 irq = acpi_register_gsi(NULL, data->index, data->trigger, data->polarity);
107 pr_err("error %d getting APIC IRQ %d\n", irq, data->index);
110 case X86_ACPI_IRQ_TYPE_GPIOINT:
111 /* Like acpi_dev_gpio_irq_get(), but without parsing ACPI resources */
112 ret = x86_android_tablet_get_gpiod(data->chip, data->index, &gpiod);
116 irq = gpiod_to_irq(gpiod);
118 pr_err("error %d getting IRQ %s %d\n", irq, data->chip, data->index);
122 irq_type = acpi_dev_get_irq_type(data->trigger, data->polarity);
123 if (irq_type != IRQ_TYPE_NONE && irq_type != irq_get_trigger_type(irq))
124 irq_set_irq_type(irq, irq_type);
127 case X86_ACPI_IRQ_TYPE_PMIC:
128 status = acpi_get_handle(NULL, data->chip, &handle);
129 if (ACPI_FAILURE(status)) {
130 pr_err("error could not get %s handle\n", data->chip);
134 adev = acpi_fetch_acpi_dev(handle);
136 pr_err("error could not get %s adev\n", data->chip);
140 fwspec.fwnode = acpi_fwnode_handle(adev);
141 domain = irq_find_matching_fwspec(&fwspec, data->domain);
143 pr_err("error could not find IRQ domain for %s\n", data->chip);
147 return irq_create_mapping(domain, data->index);
153 struct x86_i2c_client_info {
154 struct i2c_board_info board_info;
156 struct x86_acpi_irq_data irq_data;
159 struct x86_serdev_info {
160 const char *ctrl_hid;
161 const char *ctrl_uid;
162 const char *ctrl_devname;
164 * ATM the serdev core only supports of or ACPI matching; and sofar all
165 * Android x86 tablets DSDTs have usable serdev nodes, but sometimes
166 * under the wrong controller. So we just tie the existing serdev ACPI
167 * node to the right controller.
169 const char *serdev_hid;
172 struct x86_dev_info {
173 char *invalid_aei_gpiochip;
174 const char * const *modules;
175 const struct software_node *bat_swnode;
176 struct gpiod_lookup_table * const *gpiod_lookup_tables;
177 const struct x86_i2c_client_info *i2c_client_info;
178 const struct platform_device_info *pdev_info;
179 const struct x86_serdev_info *serdev_info;
180 int i2c_client_count;
187 /* Generic / shared charger / battery settings */
188 static const char * const tusb1211_chg_det_psy[] = { "tusb1211-charger-detect" };
189 static const char * const bq24190_psy[] = { "bq24190-charger" };
190 static const char * const bq25890_psy[] = { "bq25890-charger-0" };
192 static const struct property_entry fg_bq24190_supply_props[] = {
193 PROPERTY_ENTRY_STRING_ARRAY("supplied-from", bq24190_psy),
197 static const struct software_node fg_bq24190_supply_node = {
198 .properties = fg_bq24190_supply_props,
201 static const struct property_entry fg_bq25890_supply_props[] = {
202 PROPERTY_ENTRY_STRING_ARRAY("supplied-from", bq25890_psy),
206 static const struct software_node fg_bq25890_supply_node = {
207 .properties = fg_bq25890_supply_props,
210 /* LiPo HighVoltage (max 4.35V) settings used by most devs with a HV bat. */
211 static const struct property_entry generic_lipo_hv_4v35_battery_props[] = {
212 PROPERTY_ENTRY_STRING("compatible", "simple-battery"),
213 PROPERTY_ENTRY_STRING("device-chemistry", "lithium-ion"),
214 PROPERTY_ENTRY_U32("precharge-current-microamp", 256000),
215 PROPERTY_ENTRY_U32("charge-term-current-microamp", 128000),
216 PROPERTY_ENTRY_U32("constant-charge-current-max-microamp", 1856000),
217 PROPERTY_ENTRY_U32("constant-charge-voltage-max-microvolt", 4352000),
218 PROPERTY_ENTRY_U32("factory-internal-resistance-micro-ohms", 150000),
222 static const struct software_node generic_lipo_hv_4v35_battery_node = {
223 .properties = generic_lipo_hv_4v35_battery_props,
226 /* For enabling the bq24190 5V boost based on id-pin */
227 static struct regulator_consumer_supply intel_int3496_consumer = {
229 .dev_name = "intel-int3496",
232 static const struct regulator_init_data bq24190_vbus_init_data = {
234 .name = "bq24190_vbus",
235 .valid_ops_mask = REGULATOR_CHANGE_STATUS,
237 .consumer_supplies = &intel_int3496_consumer,
238 .num_consumer_supplies = 1,
241 static struct bq24190_platform_data bq24190_pdata = {
242 .regulator_init_data = &bq24190_vbus_init_data,
245 static const char * const bq24190_modules[] __initconst = {
246 "intel_crystal_cove_charger", /* For the bq24190 IRQ */
247 "bq24190_charger", /* For the Vbus regulator for intel-int3496 */
251 /* Generic pdevs array and gpio-lookups for micro USB ID pin handling */
252 static const struct platform_device_info int3496_pdevs[] __initconst = {
254 /* For micro USB ID pin handling */
255 .name = "intel-int3496",
256 .id = PLATFORM_DEVID_NONE,
260 static struct gpiod_lookup_table int3496_gpo2_pin22_gpios = {
261 .dev_id = "intel-int3496",
263 GPIO_LOOKUP("INT33FC:02", 22, "id", GPIO_ACTIVE_HIGH),
270 * This is a standard Windows tablet, but it has an extra "quick launch" button
271 * which is not described in the ACPI tables in anyway.
272 * Use the x86-android-tablets infra to create a gpio-button device for this.
274 static struct gpio_keys_button advantech_mica_071_button = {
276 /* .gpio gets filled in by advantech_mica_071_init() */
281 .debounce_interval = 50,
284 static const struct gpio_keys_platform_data advantech_mica_071_button_pdata __initconst = {
285 .buttons = &advantech_mica_071_button,
290 static const struct platform_device_info advantech_mica_071_pdevs[] __initconst = {
293 .id = PLATFORM_DEVID_AUTO,
294 .data = &advantech_mica_071_button_pdata,
295 .size_data = sizeof(advantech_mica_071_button_pdata),
299 static int __init advantech_mica_071_init(void)
301 struct gpio_desc *gpiod;
304 ret = x86_android_tablet_get_gpiod("INT33FC:00", 2, &gpiod);
307 advantech_mica_071_button.gpio = desc_to_gpio(gpiod);
312 static const struct x86_dev_info advantech_mica_071_info __initconst = {
313 .pdev_info = advantech_mica_071_pdevs,
314 .pdev_count = ARRAY_SIZE(advantech_mica_071_pdevs),
315 .init = advantech_mica_071_init,
318 /* Asus ME176C and TF103C tablets shared data */
319 static struct gpio_keys_button asus_me176c_tf103c_lid = {
321 /* .gpio gets filled in by asus_me176c_tf103c_init() */
326 .debounce_interval = 50,
329 static const struct gpio_keys_platform_data asus_me176c_tf103c_lid_pdata __initconst = {
330 .buttons = &asus_me176c_tf103c_lid,
335 static const struct platform_device_info asus_me176c_tf103c_pdevs[] __initconst = {
338 .id = PLATFORM_DEVID_AUTO,
339 .data = &asus_me176c_tf103c_lid_pdata,
340 .size_data = sizeof(asus_me176c_tf103c_lid_pdata),
343 /* For micro USB ID pin handling */
344 .name = "intel-int3496",
345 .id = PLATFORM_DEVID_NONE,
349 static int __init asus_me176c_tf103c_init(void)
351 struct gpio_desc *gpiod;
354 ret = x86_android_tablet_get_gpiod("INT33FC:02", 12, &gpiod);
357 asus_me176c_tf103c_lid.gpio = desc_to_gpio(gpiod);
363 /* Asus ME176C tablets have an Android factory img with everything hardcoded */
364 static const char * const asus_me176c_accel_mount_matrix[] = {
370 static const struct property_entry asus_me176c_accel_props[] = {
371 PROPERTY_ENTRY_STRING_ARRAY("mount-matrix", asus_me176c_accel_mount_matrix),
375 static const struct software_node asus_me176c_accel_node = {
376 .properties = asus_me176c_accel_props,
379 static const struct property_entry asus_me176c_bq24190_props[] = {
380 PROPERTY_ENTRY_STRING_ARRAY("supplied-from", tusb1211_chg_det_psy),
381 PROPERTY_ENTRY_REF("monitored-battery", &generic_lipo_hv_4v35_battery_node),
382 PROPERTY_ENTRY_U32("ti,system-minimum-microvolt", 3600000),
383 PROPERTY_ENTRY_BOOL("omit-battery-class"),
384 PROPERTY_ENTRY_BOOL("disable-reset"),
388 static const struct software_node asus_me176c_bq24190_node = {
389 .properties = asus_me176c_bq24190_props,
392 static const struct property_entry asus_me176c_ug3105_props[] = {
393 PROPERTY_ENTRY_STRING_ARRAY("supplied-from", bq24190_psy),
394 PROPERTY_ENTRY_REF("monitored-battery", &generic_lipo_hv_4v35_battery_node),
395 PROPERTY_ENTRY_U32("upisemi,rsns-microohm", 10000),
399 static const struct software_node asus_me176c_ug3105_node = {
400 .properties = asus_me176c_ug3105_props,
403 static const struct x86_i2c_client_info asus_me176c_i2c_clients[] __initconst = {
405 /* bq24297 battery charger */
409 .dev_name = "bq24297",
410 .swnode = &asus_me176c_bq24190_node,
411 .platform_data = &bq24190_pdata,
413 .adapter_path = "\\_SB_.I2C1",
415 .type = X86_ACPI_IRQ_TYPE_PMIC,
416 .chip = "\\_SB_.I2C7.PMIC",
417 .domain = DOMAIN_BUS_WAKEUP,
421 /* ug3105 battery monitor */
425 .dev_name = "ug3105",
426 .swnode = &asus_me176c_ug3105_node,
428 .adapter_path = "\\_SB_.I2C1",
430 /* ak09911 compass */
434 .dev_name = "ak09911",
436 .adapter_path = "\\_SB_.I2C5",
438 /* kxtj21009 accel */
442 .dev_name = "kxtj21009",
443 .swnode = &asus_me176c_accel_node,
445 .adapter_path = "\\_SB_.I2C5",
447 .type = X86_ACPI_IRQ_TYPE_APIC,
449 .trigger = ACPI_EDGE_SENSITIVE,
450 .polarity = ACPI_ACTIVE_LOW,
453 /* goodix touchscreen */
455 .type = "GDIX1001:00",
457 .dev_name = "goodix_ts",
459 .adapter_path = "\\_SB_.I2C6",
461 .type = X86_ACPI_IRQ_TYPE_APIC,
463 .trigger = ACPI_EDGE_SENSITIVE,
464 .polarity = ACPI_ACTIVE_LOW,
469 static const struct x86_serdev_info asus_me176c_serdevs[] __initconst = {
471 .ctrl_hid = "80860F0A",
473 .ctrl_devname = "serial0",
474 .serdev_hid = "BCM2E3A",
478 static struct gpiod_lookup_table asus_me176c_goodix_gpios = {
479 .dev_id = "i2c-goodix_ts",
481 GPIO_LOOKUP("INT33FC:00", 60, "reset", GPIO_ACTIVE_HIGH),
482 GPIO_LOOKUP("INT33FC:02", 28, "irq", GPIO_ACTIVE_HIGH),
487 static struct gpiod_lookup_table * const asus_me176c_gpios[] = {
488 &int3496_gpo2_pin22_gpios,
489 &asus_me176c_goodix_gpios,
493 static const struct x86_dev_info asus_me176c_info __initconst = {
494 .i2c_client_info = asus_me176c_i2c_clients,
495 .i2c_client_count = ARRAY_SIZE(asus_me176c_i2c_clients),
496 .pdev_info = asus_me176c_tf103c_pdevs,
497 .pdev_count = ARRAY_SIZE(asus_me176c_tf103c_pdevs),
498 .serdev_info = asus_me176c_serdevs,
499 .serdev_count = ARRAY_SIZE(asus_me176c_serdevs),
500 .gpiod_lookup_tables = asus_me176c_gpios,
501 .bat_swnode = &generic_lipo_hv_4v35_battery_node,
502 .modules = bq24190_modules,
503 .invalid_aei_gpiochip = "INT33FC:02",
504 .init = asus_me176c_tf103c_init,
507 /* Asus TF103C tablets have an Android factory img with everything hardcoded */
508 static const char * const asus_tf103c_accel_mount_matrix[] = {
514 static const struct property_entry asus_tf103c_accel_props[] = {
515 PROPERTY_ENTRY_STRING_ARRAY("mount-matrix", asus_tf103c_accel_mount_matrix),
519 static const struct software_node asus_tf103c_accel_node = {
520 .properties = asus_tf103c_accel_props,
523 static const struct property_entry asus_tf103c_touchscreen_props[] = {
524 PROPERTY_ENTRY_STRING("compatible", "atmel,atmel_mxt_ts"),
528 static const struct software_node asus_tf103c_touchscreen_node = {
529 .properties = asus_tf103c_touchscreen_props,
532 static const struct property_entry asus_tf103c_battery_props[] = {
533 PROPERTY_ENTRY_STRING("compatible", "simple-battery"),
534 PROPERTY_ENTRY_STRING("device-chemistry", "lithium-ion-polymer"),
535 PROPERTY_ENTRY_U32("precharge-current-microamp", 256000),
536 PROPERTY_ENTRY_U32("charge-term-current-microamp", 128000),
537 PROPERTY_ENTRY_U32("constant-charge-current-max-microamp", 2048000),
538 PROPERTY_ENTRY_U32("constant-charge-voltage-max-microvolt", 4208000),
539 PROPERTY_ENTRY_U32("factory-internal-resistance-micro-ohms", 150000),
543 static const struct software_node asus_tf103c_battery_node = {
544 .properties = asus_tf103c_battery_props,
547 static const struct property_entry asus_tf103c_bq24190_props[] = {
548 PROPERTY_ENTRY_STRING_ARRAY("supplied-from", tusb1211_chg_det_psy),
549 PROPERTY_ENTRY_REF("monitored-battery", &asus_tf103c_battery_node),
550 PROPERTY_ENTRY_U32("ti,system-minimum-microvolt", 3600000),
551 PROPERTY_ENTRY_BOOL("omit-battery-class"),
552 PROPERTY_ENTRY_BOOL("disable-reset"),
556 static const struct software_node asus_tf103c_bq24190_node = {
557 .properties = asus_tf103c_bq24190_props,
560 static const struct property_entry asus_tf103c_ug3105_props[] = {
561 PROPERTY_ENTRY_STRING_ARRAY("supplied-from", bq24190_psy),
562 PROPERTY_ENTRY_REF("monitored-battery", &asus_tf103c_battery_node),
563 PROPERTY_ENTRY_U32("upisemi,rsns-microohm", 5000),
567 static const struct software_node asus_tf103c_ug3105_node = {
568 .properties = asus_tf103c_ug3105_props,
571 static const struct x86_i2c_client_info asus_tf103c_i2c_clients[] __initconst = {
573 /* bq24297 battery charger */
577 .dev_name = "bq24297",
578 .swnode = &asus_tf103c_bq24190_node,
579 .platform_data = &bq24190_pdata,
581 .adapter_path = "\\_SB_.I2C1",
583 .type = X86_ACPI_IRQ_TYPE_PMIC,
584 .chip = "\\_SB_.I2C7.PMIC",
585 .domain = DOMAIN_BUS_WAKEUP,
589 /* ug3105 battery monitor */
593 .dev_name = "ug3105",
594 .swnode = &asus_tf103c_ug3105_node,
596 .adapter_path = "\\_SB_.I2C1",
598 /* ak09911 compass */
602 .dev_name = "ak09911",
604 .adapter_path = "\\_SB_.I2C5",
606 /* kxtj21009 accel */
610 .dev_name = "kxtj21009",
611 .swnode = &asus_tf103c_accel_node,
613 .adapter_path = "\\_SB_.I2C5",
615 /* atmel touchscreen */
617 .type = "atmel_mxt_ts",
619 .dev_name = "atmel_mxt_ts",
620 .swnode = &asus_tf103c_touchscreen_node,
622 .adapter_path = "\\_SB_.I2C6",
624 .type = X86_ACPI_IRQ_TYPE_GPIOINT,
625 .chip = "INT33FC:02",
627 .trigger = ACPI_EDGE_SENSITIVE,
628 .polarity = ACPI_ACTIVE_LOW,
633 static struct gpiod_lookup_table * const asus_tf103c_gpios[] = {
634 &int3496_gpo2_pin22_gpios,
638 static const struct x86_dev_info asus_tf103c_info __initconst = {
639 .i2c_client_info = asus_tf103c_i2c_clients,
640 .i2c_client_count = ARRAY_SIZE(asus_tf103c_i2c_clients),
641 .pdev_info = asus_me176c_tf103c_pdevs,
642 .pdev_count = ARRAY_SIZE(asus_me176c_tf103c_pdevs),
643 .gpiod_lookup_tables = asus_tf103c_gpios,
644 .bat_swnode = &asus_tf103c_battery_node,
645 .modules = bq24190_modules,
646 .invalid_aei_gpiochip = "INT33FC:02",
647 .init = asus_me176c_tf103c_init,
651 * When booted with the BIOS set to Android mode the Chuwi Hi8 (CWI509) DSDT
652 * contains a whole bunch of bogus ACPI I2C devices and is missing entries
653 * for the touchscreen and the accelerometer.
655 static const struct property_entry chuwi_hi8_gsl1680_props[] = {
656 PROPERTY_ENTRY_U32("touchscreen-size-x", 1665),
657 PROPERTY_ENTRY_U32("touchscreen-size-y", 1140),
658 PROPERTY_ENTRY_BOOL("touchscreen-swapped-x-y"),
659 PROPERTY_ENTRY_BOOL("silead,home-button"),
660 PROPERTY_ENTRY_STRING("firmware-name", "gsl1680-chuwi-hi8.fw"),
664 static const struct software_node chuwi_hi8_gsl1680_node = {
665 .properties = chuwi_hi8_gsl1680_props,
668 static const char * const chuwi_hi8_mount_matrix[] = {
674 static const struct property_entry chuwi_hi8_bma250e_props[] = {
675 PROPERTY_ENTRY_STRING_ARRAY("mount-matrix", chuwi_hi8_mount_matrix),
679 static const struct software_node chuwi_hi8_bma250e_node = {
680 .properties = chuwi_hi8_bma250e_props,
683 static const struct x86_i2c_client_info chuwi_hi8_i2c_clients[] __initconst = {
685 /* Silead touchscreen */
689 .swnode = &chuwi_hi8_gsl1680_node,
691 .adapter_path = "\\_SB_.I2C4",
693 .type = X86_ACPI_IRQ_TYPE_APIC,
695 .trigger = ACPI_EDGE_SENSITIVE,
696 .polarity = ACPI_ACTIVE_HIGH,
699 /* BMA250E accelerometer */
703 .swnode = &chuwi_hi8_bma250e_node,
705 .adapter_path = "\\_SB_.I2C3",
707 .type = X86_ACPI_IRQ_TYPE_GPIOINT,
708 .chip = "INT33FC:02",
710 .trigger = ACPI_LEVEL_SENSITIVE,
711 .polarity = ACPI_ACTIVE_HIGH,
716 static int __init chuwi_hi8_init(void)
719 * Avoid the acpi_unregister_gsi() call in x86_acpi_irq_helper_get()
720 * breaking the touchscreen + logging various errors when the Windows
723 if (acpi_dev_present("MSSL0001", NULL, 1))
729 static const struct x86_dev_info chuwi_hi8_info __initconst = {
730 .i2c_client_info = chuwi_hi8_i2c_clients,
731 .i2c_client_count = ARRAY_SIZE(chuwi_hi8_i2c_clients),
732 .init = chuwi_hi8_init,
735 #define CZC_EC_EXTRA_PORT 0x68
736 #define CZC_EC_ANDROID_KEYS 0x63
738 static int __init czc_p10t_init(void)
741 * The device boots up in "Windows 7" mode, when the home button sends a
742 * Windows specific key sequence (Left Meta + D) and the second button
743 * sends an unknown one while also toggling the Radio Kill Switch.
744 * This is a surprising behavior when the second button is labeled "Back".
746 * The vendor-supplied Android-x86 build switches the device to a "Android"
747 * mode by writing value 0x63 to the I/O port 0x68. This just seems to just
748 * set bit 6 on address 0x96 in the EC region; switching the bit directly
749 * seems to achieve the same result. It uses a "p10t_switcher" to do the
750 * job. It doesn't seem to be able to do anything else, and no other use
751 * of the port 0x68 is known.
753 * In the Android mode, the home button sends just a single scancode,
754 * which can be handled in Linux userspace more reasonably and the back
755 * button only sends a scancode without toggling the kill switch.
756 * The scancode can then be mapped either to Back or RF Kill functionality
757 * in userspace, depending on how the button is labeled on that particular
760 outb(CZC_EC_ANDROID_KEYS, CZC_EC_EXTRA_PORT);
764 static const struct x86_dev_info czc_p10t __initconst = {
765 .init = czc_p10t_init,
768 /* Lenovo Yoga Book X90F / X91F / X91L need manual instantiation of the fg client */
769 static const struct x86_i2c_client_info lenovo_yogabook_x9x_i2c_clients[] __initconst = {
771 /* BQ27542 fuel-gauge */
775 .dev_name = "bq27542",
776 .swnode = &fg_bq25890_supply_node,
778 .adapter_path = "\\_SB_.PCI0.I2C1",
782 static const struct x86_dev_info lenovo_yogabook_x9x_info __initconst = {
783 .i2c_client_info = lenovo_yogabook_x9x_i2c_clients,
784 .i2c_client_count = ARRAY_SIZE(lenovo_yogabook_x9x_i2c_clients),
787 /* Lenovo Yoga Tablet 2 1050F/L's Android factory img has everything hardcoded */
788 static const struct property_entry lenovo_yoga_tab2_830_1050_bq24190_props[] = {
789 PROPERTY_ENTRY_STRING_ARRAY("supplied-from", tusb1211_chg_det_psy),
790 PROPERTY_ENTRY_REF("monitored-battery", &generic_lipo_hv_4v35_battery_node),
791 PROPERTY_ENTRY_BOOL("omit-battery-class"),
792 PROPERTY_ENTRY_BOOL("disable-reset"),
796 static const struct software_node lenovo_yoga_tab2_830_1050_bq24190_node = {
797 .properties = lenovo_yoga_tab2_830_1050_bq24190_props,
800 /* This gets filled by lenovo_yoga_tab2_830_1050_init() */
801 static struct rmi_device_platform_data lenovo_yoga_tab2_830_1050_rmi_pdata = { };
803 static struct lp855x_platform_data lenovo_yoga_tab2_830_1050_lp8557_pdata = {
804 .device_control = 0x86,
805 .initial_brightness = 128,
808 static const struct x86_i2c_client_info lenovo_yoga_tab2_830_1050_i2c_clients[] __initconst = {
810 /* bq24292i battery charger */
814 .dev_name = "bq24292i",
815 .swnode = &lenovo_yoga_tab2_830_1050_bq24190_node,
816 .platform_data = &bq24190_pdata,
818 .adapter_path = "\\_SB_.I2C1",
820 .type = X86_ACPI_IRQ_TYPE_GPIOINT,
821 .chip = "INT33FC:02",
823 .trigger = ACPI_EDGE_SENSITIVE,
824 .polarity = ACPI_ACTIVE_HIGH,
827 /* BQ27541 fuel-gauge */
831 .dev_name = "bq27541",
832 .swnode = &fg_bq24190_supply_node,
834 .adapter_path = "\\_SB_.I2C1",
836 /* Synaptics RMI touchscreen */
840 .dev_name = "rmi4_i2c",
841 .platform_data = &lenovo_yoga_tab2_830_1050_rmi_pdata,
843 .adapter_path = "\\_SB_.I2C6",
845 .type = X86_ACPI_IRQ_TYPE_APIC,
847 .trigger = ACPI_EDGE_SENSITIVE,
848 .polarity = ACPI_ACTIVE_HIGH,
851 /* LP8557 Backlight controller */
855 .dev_name = "lp8557",
856 .platform_data = &lenovo_yoga_tab2_830_1050_lp8557_pdata,
858 .adapter_path = "\\_SB_.I2C3",
862 static struct gpiod_lookup_table lenovo_yoga_tab2_830_1050_int3496_gpios = {
863 .dev_id = "intel-int3496",
865 GPIO_LOOKUP("INT33FC:02", 1, "mux", GPIO_ACTIVE_LOW),
866 GPIO_LOOKUP("INT33FC:02", 24, "id", GPIO_ACTIVE_HIGH),
871 #define LENOVO_YOGA_TAB2_830_1050_CODEC_NAME "spi-10WM5102:00"
873 static struct gpiod_lookup_table lenovo_yoga_tab2_830_1050_codec_gpios = {
874 .dev_id = LENOVO_YOGA_TAB2_830_1050_CODEC_NAME,
876 GPIO_LOOKUP("gpio_crystalcove", 3, "reset", GPIO_ACTIVE_HIGH),
877 GPIO_LOOKUP("INT33FC:01", 23, "wlf,ldoena", GPIO_ACTIVE_HIGH),
878 GPIO_LOOKUP("arizona", 2, "wlf,spkvdd-ena", GPIO_ACTIVE_HIGH),
879 GPIO_LOOKUP("arizona", 4, "wlf,micd-pol", GPIO_ACTIVE_LOW),
884 static struct gpiod_lookup_table * const lenovo_yoga_tab2_830_1050_gpios[] = {
885 &lenovo_yoga_tab2_830_1050_int3496_gpios,
886 &lenovo_yoga_tab2_830_1050_codec_gpios,
890 static int __init lenovo_yoga_tab2_830_1050_init(void);
891 static void lenovo_yoga_tab2_830_1050_exit(void);
893 static struct x86_dev_info lenovo_yoga_tab2_830_1050_info __initdata = {
894 .i2c_client_info = lenovo_yoga_tab2_830_1050_i2c_clients,
895 /* i2c_client_count gets set by lenovo_yoga_tab2_830_1050_init() */
896 .pdev_info = int3496_pdevs,
897 .pdev_count = ARRAY_SIZE(int3496_pdevs),
898 .gpiod_lookup_tables = lenovo_yoga_tab2_830_1050_gpios,
899 .bat_swnode = &generic_lipo_hv_4v35_battery_node,
900 .modules = bq24190_modules,
901 .invalid_aei_gpiochip = "INT33FC:02",
902 .init = lenovo_yoga_tab2_830_1050_init,
903 .exit = lenovo_yoga_tab2_830_1050_exit,
907 * The Lenovo Yoga Tablet 2 830 and 1050 (8" vs 10") versions use the same
908 * mainboard, but they need some different treatment related to the display:
909 * 1. The 830 uses a portrait LCD panel with a landscape touchscreen, requiring
910 * the touchscreen driver to adjust the touch-coords to match the LCD.
911 * 2. Both use an TI LP8557 LED backlight controller. On the 1050 the LP8557's
912 * PWM input is connected to the PMIC's PWM output and everything works fine
913 * with the defaults programmed into the LP8557 by the BIOS.
914 * But on the 830 the LP8557's PWM input is connected to a PWM output coming
915 * from the LCD panel's controller. The Android code has a hack in the i915
916 * driver to write the non-standard DSI reg 0x9f with the desired backlight
917 * level to set the duty-cycle of the LCD's PWM output.
919 * To avoid having to have a similar hack in the mainline kernel the LP8557
920 * entry in lenovo_yoga_tab2_830_1050_i2c_clients instead just programs the
921 * LP8557 to directly set the level, ignoring the PWM input. This means that
922 * the LP8557 i2c_client should only be instantiated on the 830.
924 static int __init lenovo_yoga_tab2_830_1050_init_display(void)
926 struct gpio_desc *gpiod;
929 /* Use PMIC GPIO 10 bootstrap pin to differentiate 830 vs 1050 */
930 ret = x86_android_tablet_get_gpiod("gpio_crystalcove", 10, &gpiod);
934 ret = gpiod_get_value_cansleep(gpiod);
936 pr_info("detected Lenovo Yoga Tablet 2 1050F/L\n");
937 lenovo_yoga_tab2_830_1050_info.i2c_client_count =
938 ARRAY_SIZE(lenovo_yoga_tab2_830_1050_i2c_clients) - 1;
940 pr_info("detected Lenovo Yoga Tablet 2 830F/L\n");
941 lenovo_yoga_tab2_830_1050_rmi_pdata.sensor_pdata.axis_align.swap_axes = true;
942 lenovo_yoga_tab2_830_1050_rmi_pdata.sensor_pdata.axis_align.flip_y = true;
943 lenovo_yoga_tab2_830_1050_info.i2c_client_count =
944 ARRAY_SIZE(lenovo_yoga_tab2_830_1050_i2c_clients);
950 /* SUS (INT33FC:02) pin 6 needs to be configured as pmu_clk for the audio codec */
951 static const struct pinctrl_map lenovo_yoga_tab2_830_1050_codec_pinctrl_map =
952 PIN_MAP_MUX_GROUP(LENOVO_YOGA_TAB2_830_1050_CODEC_NAME, "codec_32khz_clk",
953 "INT33FC:02", "pmu_clk2_grp", "pmu_clk");
955 static struct pinctrl *lenovo_yoga_tab2_830_1050_codec_pinctrl;
956 static struct sys_off_handler *lenovo_yoga_tab2_830_1050_sys_off_handler;
958 static int __init lenovo_yoga_tab2_830_1050_init_codec(void)
960 struct device *codec_dev;
961 struct pinctrl *pinctrl;
964 codec_dev = bus_find_device_by_name(&spi_bus_type, NULL,
965 LENOVO_YOGA_TAB2_830_1050_CODEC_NAME);
967 pr_err("error cannot find %s device\n", LENOVO_YOGA_TAB2_830_1050_CODEC_NAME);
971 ret = pinctrl_register_mappings(&lenovo_yoga_tab2_830_1050_codec_pinctrl_map, 1);
975 pinctrl = pinctrl_get_select(codec_dev, "codec_32khz_clk");
976 if (IS_ERR(pinctrl)) {
977 ret = dev_err_probe(codec_dev, PTR_ERR(pinctrl), "selecting codec_32khz_clk\n");
978 goto err_unregister_mappings;
981 /* We're done with the codec_dev now */
982 put_device(codec_dev);
984 lenovo_yoga_tab2_830_1050_codec_pinctrl = pinctrl;
987 err_unregister_mappings:
988 pinctrl_unregister_mappings(&lenovo_yoga_tab2_830_1050_codec_pinctrl_map);
990 put_device(codec_dev);
995 * These tablet's DSDT does not set acpi_gbl_reduced_hardware, so acpi_power_off
996 * gets used as pm_power_off handler. This causes "poweroff" on these tablets
997 * to hang hard. Requiring pressing the powerbutton for 30 seconds *twice*
998 * followed by a normal 3 second press to recover. Avoid this by doing an EFI
1001 static int lenovo_yoga_tab2_830_1050_power_off(struct sys_off_data *data)
1003 efi.reset_system(EFI_RESET_SHUTDOWN, EFI_SUCCESS, 0, NULL);
1008 static int __init lenovo_yoga_tab2_830_1050_init(void)
1012 ret = lenovo_yoga_tab2_830_1050_init_display();
1016 ret = lenovo_yoga_tab2_830_1050_init_codec();
1020 /* SYS_OFF_PRIO_FIRMWARE + 1 so that it runs before acpi_power_off */
1021 lenovo_yoga_tab2_830_1050_sys_off_handler =
1022 register_sys_off_handler(SYS_OFF_MODE_POWER_OFF, SYS_OFF_PRIO_FIRMWARE + 1,
1023 lenovo_yoga_tab2_830_1050_power_off, NULL);
1024 if (IS_ERR(lenovo_yoga_tab2_830_1050_sys_off_handler))
1025 return PTR_ERR(lenovo_yoga_tab2_830_1050_sys_off_handler);
1030 static void lenovo_yoga_tab2_830_1050_exit(void)
1032 unregister_sys_off_handler(lenovo_yoga_tab2_830_1050_sys_off_handler);
1034 if (lenovo_yoga_tab2_830_1050_codec_pinctrl) {
1035 pinctrl_put(lenovo_yoga_tab2_830_1050_codec_pinctrl);
1036 pinctrl_unregister_mappings(&lenovo_yoga_tab2_830_1050_codec_pinctrl_map);
1040 /* Lenovo Yoga Tab 3 Pro YT3-X90F */
1043 * There are 2 batteries, with 2 bq27500 fuel-gauges and 2 bq25892 chargers,
1044 * "bq25890-charger-1" is instantiated from: drivers/i2c/busses/i2c-cht-wc.c.
1046 static const char * const lenovo_yt3_bq25892_0_suppliers[] = { "cht_wcove_pwrsrc" };
1047 static const char * const bq25890_1_psy[] = { "bq25890-charger-1" };
1049 static const struct property_entry fg_bq25890_1_supply_props[] = {
1050 PROPERTY_ENTRY_STRING_ARRAY("supplied-from", bq25890_1_psy),
1054 static const struct software_node fg_bq25890_1_supply_node = {
1055 .properties = fg_bq25890_1_supply_props,
1058 /* bq25892 charger settings for the flat lipo battery behind the screen */
1059 static const struct property_entry lenovo_yt3_bq25892_0_props[] = {
1060 PROPERTY_ENTRY_STRING_ARRAY("supplied-from", lenovo_yt3_bq25892_0_suppliers),
1061 PROPERTY_ENTRY_STRING("linux,power-supply-name", "bq25892-second-chrg"),
1062 PROPERTY_ENTRY_U32("linux,iinlim-percentage", 40),
1063 PROPERTY_ENTRY_BOOL("linux,skip-reset"),
1064 /* Values taken from Android Factory Image */
1065 PROPERTY_ENTRY_U32("ti,charge-current", 2048000),
1066 PROPERTY_ENTRY_U32("ti,battery-regulation-voltage", 4352000),
1067 PROPERTY_ENTRY_U32("ti,termination-current", 128000),
1068 PROPERTY_ENTRY_U32("ti,precharge-current", 128000),
1069 PROPERTY_ENTRY_U32("ti,minimum-sys-voltage", 3700000),
1070 PROPERTY_ENTRY_U32("ti,boost-voltage", 4998000),
1071 PROPERTY_ENTRY_U32("ti,boost-max-current", 500000),
1072 PROPERTY_ENTRY_BOOL("ti,use-ilim-pin"),
1076 static const struct software_node lenovo_yt3_bq25892_0_node = {
1077 .properties = lenovo_yt3_bq25892_0_props,
1080 static const struct x86_i2c_client_info lenovo_yt3_i2c_clients[] __initconst = {
1082 /* bq27500 fuel-gauge for the flat lipo battery behind the screen */
1086 .dev_name = "bq27500_0",
1087 .swnode = &fg_bq25890_supply_node,
1089 .adapter_path = "\\_SB_.PCI0.I2C1",
1091 /* bq25892 charger for the flat lipo battery behind the screen */
1095 .dev_name = "bq25892_0",
1096 .swnode = &lenovo_yt3_bq25892_0_node,
1098 .adapter_path = "\\_SB_.PCI0.I2C1",
1100 .type = X86_ACPI_IRQ_TYPE_GPIOINT,
1101 .chip = "INT33FF:01",
1103 .trigger = ACPI_EDGE_SENSITIVE,
1104 .polarity = ACPI_ACTIVE_LOW,
1107 /* bq27500 fuel-gauge for the round li-ion cells in the hinge */
1111 .dev_name = "bq27500_1",
1112 .swnode = &fg_bq25890_1_supply_node,
1114 .adapter_path = "\\_SB_.PCI0.I2C2",
1118 static int __init lenovo_yt3_init(void)
1120 struct gpio_desc *gpiod;
1124 * The "bq25892_0" charger IC has its /CE (Charge-Enable) and OTG pins
1125 * connected to GPIOs, rather then having them hardwired to the correct
1126 * values as is normally done.
1128 * The bq25890_charger driver controls these through I2C, but this only
1129 * works if not overridden by the pins. Set these pins here:
1130 * 1. Set /CE to 0 to allow charging.
1131 * 2. Set OTG to 0 disable V5 boost output since the 5V boost output of
1132 * the main "bq25892_1" charger is used when necessary.
1136 ret = x86_android_tablet_get_gpiod("INT33FF:02", 22, &gpiod);
1141 * The gpio_desc returned by x86_android_tablet_get_gpiod() is a "raw"
1142 * gpio_desc, that is there is no way to pass lookup-flags like
1143 * GPIO_ACTIVE_LOW. Set the GPIO to 0 here to enable charging since
1144 * the /CE pin is active-low, but not marked as such in the gpio_desc.
1146 gpiod_set_value(gpiod, 0);
1149 ret = x86_android_tablet_get_gpiod("INT33FF:03", 19, &gpiod);
1153 gpiod_set_value(gpiod, 0);
1158 static const struct x86_dev_info lenovo_yt3_info __initconst = {
1159 .i2c_client_info = lenovo_yt3_i2c_clients,
1160 .i2c_client_count = ARRAY_SIZE(lenovo_yt3_i2c_clients),
1161 .init = lenovo_yt3_init,
1164 /* Medion Lifetab S10346 tablets have an Android factory img with everything hardcoded */
1165 static const char * const medion_lifetab_s10346_accel_mount_matrix[] = {
1171 static const struct property_entry medion_lifetab_s10346_accel_props[] = {
1172 PROPERTY_ENTRY_STRING_ARRAY("mount-matrix", medion_lifetab_s10346_accel_mount_matrix),
1176 static const struct software_node medion_lifetab_s10346_accel_node = {
1177 .properties = medion_lifetab_s10346_accel_props,
1180 /* Note the LCD panel is mounted upside down, this is correctly indicated in the VBT */
1181 static const struct property_entry medion_lifetab_s10346_touchscreen_props[] = {
1182 PROPERTY_ENTRY_BOOL("touchscreen-inverted-x"),
1183 PROPERTY_ENTRY_BOOL("touchscreen-swapped-x-y"),
1187 static const struct software_node medion_lifetab_s10346_touchscreen_node = {
1188 .properties = medion_lifetab_s10346_touchscreen_props,
1191 static const struct x86_i2c_client_info medion_lifetab_s10346_i2c_clients[] __initconst = {
1193 /* kxtj21009 accel */
1195 .type = "kxtj21009",
1197 .dev_name = "kxtj21009",
1198 .swnode = &medion_lifetab_s10346_accel_node,
1200 .adapter_path = "\\_SB_.I2C3",
1202 .type = X86_ACPI_IRQ_TYPE_GPIOINT,
1203 .chip = "INT33FC:02",
1205 .trigger = ACPI_EDGE_SENSITIVE,
1206 .polarity = ACPI_ACTIVE_HIGH,
1209 /* goodix touchscreen */
1211 .type = "GDIX1001:00",
1213 .dev_name = "goodix_ts",
1214 .swnode = &medion_lifetab_s10346_touchscreen_node,
1216 .adapter_path = "\\_SB_.I2C4",
1218 .type = X86_ACPI_IRQ_TYPE_APIC,
1220 .trigger = ACPI_EDGE_SENSITIVE,
1221 .polarity = ACPI_ACTIVE_LOW,
1226 static struct gpiod_lookup_table medion_lifetab_s10346_goodix_gpios = {
1227 .dev_id = "i2c-goodix_ts",
1229 GPIO_LOOKUP("INT33FC:01", 26, "reset", GPIO_ACTIVE_HIGH),
1230 GPIO_LOOKUP("INT33FC:02", 3, "irq", GPIO_ACTIVE_HIGH),
1235 static struct gpiod_lookup_table * const medion_lifetab_s10346_gpios[] = {
1236 &medion_lifetab_s10346_goodix_gpios,
1240 static const struct x86_dev_info medion_lifetab_s10346_info __initconst = {
1241 .i2c_client_info = medion_lifetab_s10346_i2c_clients,
1242 .i2c_client_count = ARRAY_SIZE(medion_lifetab_s10346_i2c_clients),
1243 .gpiod_lookup_tables = medion_lifetab_s10346_gpios,
1246 /* Nextbook Ares 8 tablets have an Android factory img with everything hardcoded */
1247 static const char * const nextbook_ares8_accel_mount_matrix[] = {
1253 static const struct property_entry nextbook_ares8_accel_props[] = {
1254 PROPERTY_ENTRY_STRING_ARRAY("mount-matrix", nextbook_ares8_accel_mount_matrix),
1258 static const struct software_node nextbook_ares8_accel_node = {
1259 .properties = nextbook_ares8_accel_props,
1262 static const struct property_entry nextbook_ares8_touchscreen_props[] = {
1263 PROPERTY_ENTRY_U32("touchscreen-size-x", 800),
1264 PROPERTY_ENTRY_U32("touchscreen-size-y", 1280),
1268 static const struct software_node nextbook_ares8_touchscreen_node = {
1269 .properties = nextbook_ares8_touchscreen_props,
1272 static const struct x86_i2c_client_info nextbook_ares8_i2c_clients[] __initconst = {
1274 /* Freescale MMA8653FC accel */
1278 .dev_name = "mma8653",
1279 .swnode = &nextbook_ares8_accel_node,
1281 .adapter_path = "\\_SB_.I2C3",
1283 /* FT5416DQ9 touchscreen controller */
1285 .type = "edt-ft5x06",
1287 .dev_name = "ft5416",
1288 .swnode = &nextbook_ares8_touchscreen_node,
1290 .adapter_path = "\\_SB_.I2C4",
1292 .type = X86_ACPI_IRQ_TYPE_GPIOINT,
1293 .chip = "INT33FC:02",
1295 .trigger = ACPI_EDGE_SENSITIVE,
1296 .polarity = ACPI_ACTIVE_LOW,
1301 static struct gpiod_lookup_table nextbook_ares8_int3496_gpios = {
1302 .dev_id = "intel-int3496",
1304 GPIO_LOOKUP("INT33FC:02", 1, "mux", GPIO_ACTIVE_HIGH),
1305 GPIO_LOOKUP("INT33FC:02", 18, "id", GPIO_ACTIVE_HIGH),
1310 static struct gpiod_lookup_table * const nextbook_ares8_gpios[] = {
1311 &nextbook_ares8_int3496_gpios,
1315 static const struct x86_dev_info nextbook_ares8_info __initconst = {
1316 .i2c_client_info = nextbook_ares8_i2c_clients,
1317 .i2c_client_count = ARRAY_SIZE(nextbook_ares8_i2c_clients),
1318 .pdev_info = int3496_pdevs,
1319 .pdev_count = ARRAY_SIZE(int3496_pdevs),
1320 .gpiod_lookup_tables = nextbook_ares8_gpios,
1321 .invalid_aei_gpiochip = "INT33FC:02",
1325 * Whitelabel (sold as various brands) TM800A550L tablets.
1326 * These tablet's DSDT contains a whole bunch of bogus ACPI I2C devices
1327 * (removed through acpi_quirk_skip_i2c_client_enumeration()) and
1328 * the touchscreen fwnode has the wrong GPIOs.
1330 static const char * const whitelabel_tm800a550l_accel_mount_matrix[] = {
1336 static const struct property_entry whitelabel_tm800a550l_accel_props[] = {
1337 PROPERTY_ENTRY_STRING_ARRAY("mount-matrix", whitelabel_tm800a550l_accel_mount_matrix),
1341 static const struct software_node whitelabel_tm800a550l_accel_node = {
1342 .properties = whitelabel_tm800a550l_accel_props,
1345 static const struct property_entry whitelabel_tm800a550l_goodix_props[] = {
1346 PROPERTY_ENTRY_STRING("firmware-name", "gt912-tm800a550l.fw"),
1347 PROPERTY_ENTRY_STRING("goodix,config-name", "gt912-tm800a550l.cfg"),
1348 PROPERTY_ENTRY_U32("goodix,main-clk", 54),
1352 static const struct software_node whitelabel_tm800a550l_goodix_node = {
1353 .properties = whitelabel_tm800a550l_goodix_props,
1356 static const struct x86_i2c_client_info whitelabel_tm800a550l_i2c_clients[] __initconst = {
1358 /* goodix touchscreen */
1360 .type = "GDIX1001:00",
1362 .dev_name = "goodix_ts",
1363 .swnode = &whitelabel_tm800a550l_goodix_node,
1365 .adapter_path = "\\_SB_.I2C2",
1367 .type = X86_ACPI_IRQ_TYPE_APIC,
1369 .trigger = ACPI_EDGE_SENSITIVE,
1370 .polarity = ACPI_ACTIVE_HIGH,
1373 /* kxcj91008 accel */
1375 .type = "kxcj91008",
1377 .dev_name = "kxcj91008",
1378 .swnode = &whitelabel_tm800a550l_accel_node,
1380 .adapter_path = "\\_SB_.I2C3",
1384 static struct gpiod_lookup_table whitelabel_tm800a550l_goodix_gpios = {
1385 .dev_id = "i2c-goodix_ts",
1387 GPIO_LOOKUP("INT33FC:01", 26, "reset", GPIO_ACTIVE_HIGH),
1388 GPIO_LOOKUP("INT33FC:02", 3, "irq", GPIO_ACTIVE_HIGH),
1393 static struct gpiod_lookup_table * const whitelabel_tm800a550l_gpios[] = {
1394 &whitelabel_tm800a550l_goodix_gpios,
1398 static const struct x86_dev_info whitelabel_tm800a550l_info __initconst = {
1399 .i2c_client_info = whitelabel_tm800a550l_i2c_clients,
1400 .i2c_client_count = ARRAY_SIZE(whitelabel_tm800a550l_i2c_clients),
1401 .gpiod_lookup_tables = whitelabel_tm800a550l_gpios,
1405 * If the EFI bootloader is not Xiaomi's own signed Android loader, then the
1406 * Xiaomi Mi Pad 2 X86 tablet sets OSID in the DSDT to 1 (Windows), causing
1407 * a bunch of devices to be hidden.
1409 * This takes care of instantiating the hidden devices manually.
1411 static const struct x86_i2c_client_info xiaomi_mipad2_i2c_clients[] __initconst = {
1413 /* BQ27520 fuel-gauge */
1417 .dev_name = "bq27520",
1418 .swnode = &fg_bq25890_supply_node,
1420 .adapter_path = "\\_SB_.PCI0.I2C1",
1422 /* KTD2026 RGB notification LED controller */
1426 .dev_name = "ktd2026",
1428 .adapter_path = "\\_SB_.PCI0.I2C3",
1432 static const struct x86_dev_info xiaomi_mipad2_info __initconst = {
1433 .i2c_client_info = xiaomi_mipad2_i2c_clients,
1434 .i2c_client_count = ARRAY_SIZE(xiaomi_mipad2_i2c_clients),
1437 static const struct dmi_system_id x86_android_tablet_ids[] __initconst = {
1439 /* Advantech MICA-071 */
1441 DMI_EXACT_MATCH(DMI_SYS_VENDOR, "Advantech"),
1442 DMI_EXACT_MATCH(DMI_PRODUCT_NAME, "MICA-071"),
1444 .driver_data = (void *)&advantech_mica_071_info,
1447 /* Asus MeMO Pad 7 ME176C */
1449 DMI_EXACT_MATCH(DMI_SYS_VENDOR, "ASUSTeK COMPUTER INC."),
1450 DMI_EXACT_MATCH(DMI_PRODUCT_NAME, "ME176C"),
1452 .driver_data = (void *)&asus_me176c_info,
1457 DMI_MATCH(DMI_SYS_VENDOR, "ASUSTeK COMPUTER INC."),
1458 DMI_MATCH(DMI_PRODUCT_NAME, "TF103C"),
1460 .driver_data = (void *)&asus_tf103c_info,
1463 /* Chuwi Hi8 (CWI509) */
1465 DMI_MATCH(DMI_BOARD_VENDOR, "Hampoo"),
1466 DMI_MATCH(DMI_BOARD_NAME, "BYT-PA03C"),
1467 DMI_MATCH(DMI_SYS_VENDOR, "ilife"),
1468 DMI_MATCH(DMI_PRODUCT_NAME, "S806"),
1470 .driver_data = (void *)&chuwi_hi8_info,
1474 .ident = "CZC ODEON TPC-10 (\"P10T\")",
1476 DMI_MATCH(DMI_SYS_VENDOR, "CZC"),
1477 DMI_MATCH(DMI_PRODUCT_NAME, "ODEON*TPC-10"),
1479 .driver_data = (void *)&czc_p10t,
1482 /* CZC P10T variant */
1483 .ident = "ViewSonic ViewPad 10",
1485 DMI_MATCH(DMI_SYS_VENDOR, "ViewSonic"),
1486 DMI_MATCH(DMI_PRODUCT_NAME, "VPAD10"),
1488 .driver_data = (void *)&czc_p10t,
1491 /* Lenovo Yoga Book X90F / X91F / X91L */
1493 /* Non exact match to match all versions */
1494 DMI_MATCH(DMI_PRODUCT_NAME, "Lenovo YB1-X9"),
1496 .driver_data = (void *)&lenovo_yogabook_x9x_info,
1500 * Lenovo Yoga Tablet 2 830F/L or 1050F/L (The 8" and 10"
1501 * Lenovo Yoga Tablet 2 use the same mainboard)
1504 DMI_MATCH(DMI_SYS_VENDOR, "Intel Corp."),
1505 DMI_MATCH(DMI_PRODUCT_NAME, "VALLEYVIEW C0 PLATFORM"),
1506 DMI_MATCH(DMI_BOARD_NAME, "BYT-T FFD8"),
1507 /* Partial match on beginning of BIOS version */
1508 DMI_MATCH(DMI_BIOS_VERSION, "BLADE_21"),
1510 .driver_data = (void *)&lenovo_yoga_tab2_830_1050_info,
1513 /* Lenovo Yoga Tab 3 Pro YT3-X90F */
1515 DMI_MATCH(DMI_SYS_VENDOR, "Intel Corporation"),
1516 DMI_MATCH(DMI_PRODUCT_NAME, "CHERRYVIEW D1 PLATFORM"),
1517 DMI_MATCH(DMI_PRODUCT_VERSION, "Blade3-10A-001"),
1519 .driver_data = (void *)&lenovo_yt3_info,
1522 /* Medion Lifetab S10346 */
1524 DMI_MATCH(DMI_BOARD_VENDOR, "AMI Corporation"),
1525 DMI_MATCH(DMI_BOARD_NAME, "Aptio CRB"),
1526 /* Above strings are much too generic, also match on BIOS date */
1527 DMI_MATCH(DMI_BIOS_DATE, "10/22/2015"),
1529 .driver_data = (void *)&medion_lifetab_s10346_info,
1532 /* Nextbook Ares 8 */
1534 DMI_MATCH(DMI_SYS_VENDOR, "Insyde"),
1535 DMI_MATCH(DMI_PRODUCT_NAME, "M890BAP"),
1537 .driver_data = (void *)&nextbook_ares8_info,
1540 /* Whitelabel (sold as various brands) TM800A550L */
1542 DMI_MATCH(DMI_BOARD_VENDOR, "AMI Corporation"),
1543 DMI_MATCH(DMI_BOARD_NAME, "Aptio CRB"),
1544 /* Above strings are too generic, also match on BIOS version */
1545 DMI_MATCH(DMI_BIOS_VERSION, "ZY-8-BI-PX4S70VTR400-X423B-005-D"),
1547 .driver_data = (void *)&whitelabel_tm800a550l_info,
1550 /* Xiaomi Mi Pad 2 */
1552 DMI_MATCH(DMI_SYS_VENDOR, "Xiaomi Inc"),
1553 DMI_MATCH(DMI_PRODUCT_NAME, "Mipad2"),
1555 .driver_data = (void *)&xiaomi_mipad2_info,
1559 MODULE_DEVICE_TABLE(dmi, x86_android_tablet_ids);
1561 static int i2c_client_count;
1562 static int pdev_count;
1563 static int serdev_count;
1564 static struct i2c_client **i2c_clients;
1565 static struct platform_device **pdevs;
1566 static struct serdev_device **serdevs;
1567 static struct gpiod_lookup_table * const *gpiod_lookup_tables;
1568 static const struct software_node *bat_swnode;
1569 static void (*exit_handler)(void);
1571 static __init int x86_instantiate_i2c_client(const struct x86_dev_info *dev_info,
1574 const struct x86_i2c_client_info *client_info = &dev_info->i2c_client_info[idx];
1575 struct i2c_board_info board_info = client_info->board_info;
1576 struct i2c_adapter *adap;
1580 board_info.irq = x86_acpi_irq_helper_get(&client_info->irq_data);
1581 if (board_info.irq < 0)
1582 return board_info.irq;
1584 status = acpi_get_handle(NULL, client_info->adapter_path, &handle);
1585 if (ACPI_FAILURE(status)) {
1586 pr_err("Error could not get %s handle\n", client_info->adapter_path);
1590 adap = i2c_acpi_find_adapter_by_handle(handle);
1592 pr_err("error could not get %s adapter\n", client_info->adapter_path);
1596 i2c_clients[idx] = i2c_new_client_device(adap, &board_info);
1597 put_device(&adap->dev);
1598 if (IS_ERR(i2c_clients[idx]))
1599 return dev_err_probe(&adap->dev, PTR_ERR(i2c_clients[idx]),
1600 "creating I2C-client %d\n", idx);
1605 static __init int x86_instantiate_serdev(const struct x86_serdev_info *info, int idx)
1607 struct acpi_device *ctrl_adev, *serdev_adev;
1608 struct serdev_device *serdev;
1609 struct device *ctrl_dev;
1612 ctrl_adev = acpi_dev_get_first_match_dev(info->ctrl_hid, info->ctrl_uid, -1);
1614 pr_err("error could not get %s/%s ctrl adev\n",
1615 info->ctrl_hid, info->ctrl_uid);
1619 serdev_adev = acpi_dev_get_first_match_dev(info->serdev_hid, NULL, -1);
1621 pr_err("error could not get %s serdev adev\n", info->serdev_hid);
1625 /* get_first_physical_node() returns a weak ref, no need to put() it */
1626 ctrl_dev = acpi_get_first_physical_node(ctrl_adev);
1628 pr_err("error could not get %s/%s ctrl physical dev\n",
1629 info->ctrl_hid, info->ctrl_uid);
1630 goto put_serdev_adev;
1633 /* ctrl_dev now points to the controller's parent, get the controller */
1634 ctrl_dev = device_find_child_by_name(ctrl_dev, info->ctrl_devname);
1636 pr_err("error could not get %s/%s %s ctrl dev\n",
1637 info->ctrl_hid, info->ctrl_uid, info->ctrl_devname);
1638 goto put_serdev_adev;
1641 serdev = serdev_device_alloc(to_serdev_controller(ctrl_dev));
1644 goto put_serdev_adev;
1647 ACPI_COMPANION_SET(&serdev->dev, serdev_adev);
1648 acpi_device_set_enumerated(serdev_adev);
1650 ret = serdev_device_add(serdev);
1652 dev_err(&serdev->dev, "error %d adding serdev\n", ret);
1653 serdev_device_put(serdev);
1654 goto put_serdev_adev;
1657 serdevs[idx] = serdev;
1660 acpi_dev_put(serdev_adev);
1662 acpi_dev_put(ctrl_adev);
1666 static void x86_android_tablet_cleanup(void)
1670 for (i = 0; i < serdev_count; i++) {
1672 serdev_device_remove(serdevs[i]);
1677 for (i = 0; i < pdev_count; i++)
1678 platform_device_unregister(pdevs[i]);
1682 for (i = 0; i < i2c_client_count; i++)
1683 i2c_unregister_device(i2c_clients[i]);
1690 for (i = 0; gpiod_lookup_tables && gpiod_lookup_tables[i]; i++)
1691 gpiod_remove_lookup_table(gpiod_lookup_tables[i]);
1693 software_node_unregister(bat_swnode);
1696 static __init int x86_android_tablet_init(void)
1698 const struct x86_dev_info *dev_info;
1699 const struct dmi_system_id *id;
1700 struct gpio_chip *chip;
1703 id = dmi_first_match(x86_android_tablet_ids);
1707 dev_info = id->driver_data;
1710 * The broken DSDTs on these devices often also include broken
1711 * _AEI (ACPI Event Interrupt) handlers, disable these.
1713 if (dev_info->invalid_aei_gpiochip) {
1714 chip = gpiochip_find(dev_info->invalid_aei_gpiochip,
1715 gpiochip_find_match_label);
1717 pr_err("error cannot find GPIO chip %s\n", dev_info->invalid_aei_gpiochip);
1720 acpi_gpiochip_free_interrupts(chip);
1724 * Since this runs from module_init() it cannot use -EPROBE_DEFER,
1725 * instead pre-load any modules which are listed as requirements.
1727 for (i = 0; dev_info->modules && dev_info->modules[i]; i++)
1728 request_module(dev_info->modules[i]);
1730 bat_swnode = dev_info->bat_swnode;
1732 ret = software_node_register(bat_swnode);
1737 gpiod_lookup_tables = dev_info->gpiod_lookup_tables;
1738 for (i = 0; gpiod_lookup_tables && gpiod_lookup_tables[i]; i++)
1739 gpiod_add_lookup_table(gpiod_lookup_tables[i]);
1741 if (dev_info->init) {
1742 ret = dev_info->init();
1744 x86_android_tablet_cleanup();
1747 exit_handler = dev_info->exit;
1750 i2c_clients = kcalloc(dev_info->i2c_client_count, sizeof(*i2c_clients), GFP_KERNEL);
1752 x86_android_tablet_cleanup();
1756 i2c_client_count = dev_info->i2c_client_count;
1757 for (i = 0; i < i2c_client_count; i++) {
1758 ret = x86_instantiate_i2c_client(dev_info, i);
1760 x86_android_tablet_cleanup();
1765 pdevs = kcalloc(dev_info->pdev_count, sizeof(*pdevs), GFP_KERNEL);
1767 x86_android_tablet_cleanup();
1771 pdev_count = dev_info->pdev_count;
1772 for (i = 0; i < pdev_count; i++) {
1773 pdevs[i] = platform_device_register_full(&dev_info->pdev_info[i]);
1774 if (IS_ERR(pdevs[i])) {
1775 x86_android_tablet_cleanup();
1776 return PTR_ERR(pdevs[i]);
1780 serdevs = kcalloc(dev_info->serdev_count, sizeof(*serdevs), GFP_KERNEL);
1782 x86_android_tablet_cleanup();
1786 serdev_count = dev_info->serdev_count;
1787 for (i = 0; i < serdev_count; i++) {
1788 ret = x86_instantiate_serdev(&dev_info->serdev_info[i], i);
1790 x86_android_tablet_cleanup();
1798 module_init(x86_android_tablet_init);
1799 module_exit(x86_android_tablet_cleanup);
1802 MODULE_DESCRIPTION("X86 Android tablets DSDT fixups driver");
1803 MODULE_LICENSE("GPL");