1 // SPDX-License-Identifier: GPL-2.0-or-later
3 * lenovo-ymc.c - Lenovo Yoga Mode Control driver
5 * Copyright © 2022 Gergo Koteles <soyer@irl.hu>
8 #define pr_fmt(fmt) KBUILD_MODNAME ": " fmt
10 #include <linux/acpi.h>
11 #include <linux/dmi.h>
12 #include <linux/input.h>
13 #include <linux/input/sparse-keymap.h>
14 #include <linux/wmi.h>
15 #include "ideapad-laptop.h"
17 #define LENOVO_YMC_EVENT_GUID "06129D99-6083-4164-81AD-F092F9D773A6"
18 #define LENOVO_YMC_QUERY_GUID "09B0EE6E-C3FD-4243-8DA1-7911FF80BB8C"
20 #define LENOVO_YMC_QUERY_INSTANCE 0
21 #define LENOVO_YMC_QUERY_METHOD 0x01
24 module_param(force
, bool, 0444);
25 MODULE_PARM_DESC(force
, "Force loading on boards without a convertible DMI chassis-type");
27 static const struct dmi_system_id allowed_chasis_types_dmi_table
[] = {
30 DMI_EXACT_MATCH(DMI_CHASSIS_TYPE
, "31" /* Convertible */),
35 DMI_EXACT_MATCH(DMI_CHASSIS_TYPE
, "32" /* Detachable */),
41 struct lenovo_ymc_private
{
42 struct input_dev
*input_dev
;
45 static const struct key_entry lenovo_ymc_keymap
[] = {
46 /* Ignore the uninitialized state */
49 { KE_SW
, 0x01, { .sw
= { SW_TABLET_MODE
, 0 } } },
51 { KE_SW
, 0x02, { .sw
= { SW_TABLET_MODE
, 1 } } },
53 { KE_SW
, 0x03, { .sw
= { SW_TABLET_MODE
, 1 } } },
55 { KE_SW
, 0x04, { .sw
= { SW_TABLET_MODE
, 1 } } },
59 static void lenovo_ymc_notify(struct wmi_device
*wdev
, union acpi_object
*data
)
61 struct lenovo_ymc_private
*priv
= dev_get_drvdata(&wdev
->dev
);
63 struct acpi_buffer input
= { sizeof(input_val
), &input_val
};
64 struct acpi_buffer output
= { ACPI_ALLOCATE_BUFFER
, NULL
};
65 union acpi_object
*obj
;
69 status
= wmi_evaluate_method(LENOVO_YMC_QUERY_GUID
,
70 LENOVO_YMC_QUERY_INSTANCE
,
71 LENOVO_YMC_QUERY_METHOD
,
74 if (ACPI_FAILURE(status
)) {
76 "Failed to evaluate query method: %s\n",
77 acpi_format_exception(status
));
83 if (obj
->type
!= ACPI_TYPE_INTEGER
) {
85 "WMI event data is not an integer\n");
88 code
= obj
->integer
.value
;
90 if (!sparse_keymap_report_event(priv
->input_dev
, code
, 1, true))
91 dev_warn(&wdev
->dev
, "Unknown key %d pressed\n", code
);
95 ideapad_laptop_call_notifier(IDEAPAD_LAPTOP_YMC_EVENT
, &code
);
98 static int lenovo_ymc_probe(struct wmi_device
*wdev
, const void *ctx
)
100 struct lenovo_ymc_private
*priv
;
101 struct input_dev
*input_dev
;
104 if (!dmi_check_system(allowed_chasis_types_dmi_table
)) {
106 dev_info(&wdev
->dev
, "Force loading Lenovo YMC support\n");
111 priv
= devm_kzalloc(&wdev
->dev
, sizeof(*priv
), GFP_KERNEL
);
115 input_dev
= devm_input_allocate_device(&wdev
->dev
);
119 input_dev
->name
= "Lenovo Yoga Tablet Mode Control switch";
120 input_dev
->phys
= LENOVO_YMC_EVENT_GUID
"/input0";
121 input_dev
->id
.bustype
= BUS_HOST
;
122 input_dev
->dev
.parent
= &wdev
->dev
;
123 err
= sparse_keymap_setup(input_dev
, lenovo_ymc_keymap
, NULL
);
126 "Could not set up input device keymap: %d\n", err
);
130 err
= input_register_device(input_dev
);
133 "Could not register input device: %d\n", err
);
137 priv
->input_dev
= input_dev
;
138 dev_set_drvdata(&wdev
->dev
, priv
);
140 /* Report the state for the first time on probe */
141 lenovo_ymc_notify(wdev
, NULL
);
145 static const struct wmi_device_id lenovo_ymc_wmi_id_table
[] = {
146 { .guid_string
= LENOVO_YMC_EVENT_GUID
},
149 MODULE_DEVICE_TABLE(wmi
, lenovo_ymc_wmi_id_table
);
151 static struct wmi_driver lenovo_ymc_driver
= {
153 .name
= "lenovo-ymc",
155 .id_table
= lenovo_ymc_wmi_id_table
,
156 .probe
= lenovo_ymc_probe
,
157 .notify
= lenovo_ymc_notify
,
160 module_wmi_driver(lenovo_ymc_driver
);
162 MODULE_AUTHOR("Gergo Koteles <soyer@irl.hu>");
163 MODULE_DESCRIPTION("Lenovo Yoga Mode Control driver");
164 MODULE_LICENSE("GPL");
165 MODULE_IMPORT_NS(IDEAPAD_LAPTOP
);