blob: 57712ff1a1deea9a6714c5dd3a84235bda2975fe [file] [log] [blame]
Corentin Charyb229ece2011-02-26 10:20:40 +01001/*
2 * Asus Notebooks WMI hotkey driver
3 *
4 * Copyright(C) 2010 Corentin Chary <corentin.chary@gmail.com>
5 *
6 * This program is free software; you can redistribute it and/or modify
7 * it under the terms of the GNU General Public License as published by
8 * the Free Software Foundation; either version 2 of the License, or
9 * (at your option) any later version.
10 *
11 * This program is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14 * GNU General Public License for more details.
15 *
16 * You should have received a copy of the GNU General Public License
17 * along with this program; if not, write to the Free Software
18 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
19 */
20
21#define pr_fmt(fmt) KBUILD_MODNAME ": " fmt
22
23#include <linux/kernel.h>
24#include <linux/module.h>
25#include <linux/init.h>
26#include <linux/input.h>
27#include <linux/input/sparse-keymap.h>
Corentin Charyfb05b9f2012-03-20 09:53:11 +010028#include <linux/fb.h>
AceLan Kaod2044c52012-07-04 11:19:08 +080029#include <linux/dmi.h>
Corentin Charyb229ece2011-02-26 10:20:40 +010030
31#include "asus-wmi.h"
32
33#define ASUS_NB_WMI_FILE "asus-nb-wmi"
34
35MODULE_AUTHOR("Corentin Chary <corentincj@iksaif.net>");
36MODULE_DESCRIPTION("Asus Notebooks WMI Hotkey Driver");
37MODULE_LICENSE("GPL");
38
39#define ASUS_NB_WMI_EVENT_GUID "0B3CBB35-E3C2-45ED-91C2-4C5A6D195D1C"
40
41MODULE_ALIAS("wmi:"ASUS_NB_WMI_EVENT_GUID);
42
Corentin Charyfddbfed2011-07-01 11:34:39 +020043/*
44 * WAPF defines the behavior of the Fn+Fx wlan key
45 * The significance of values is yet to be found, but
46 * most of the time:
47 * Bit | Bluetooth | WLAN
48 * 0 | Hardware | Hardware
49 * 1 | Hardware | Software
50 * 4 | Software | Software
51 */
AceLan Kaod2044c52012-07-04 11:19:08 +080052static int wapf = -1;
Corentin Charyfddbfed2011-07-01 11:34:39 +020053module_param(wapf, uint, 0444);
54MODULE_PARM_DESC(wapf, "WAPF value");
55
AceLan Kaod2044c52012-07-04 11:19:08 +080056static struct quirk_entry *quirks;
57
Corentin Chary6a2bccc2012-03-20 09:53:10 +010058static struct quirk_entry quirk_asus_unknown = {
AceLan Kaod2044c52012-07-04 11:19:08 +080059 .wapf = 0,
60};
61
62static struct quirk_entry quirk_asus_x401u = {
63 .wapf = 4,
64};
65
66static int dmi_matched(const struct dmi_system_id *dmi)
67{
68 quirks = dmi->driver_data;
69 return 1;
70}
71
72static struct dmi_system_id asus_quirks[] = {
73 {
74 .callback = dmi_matched,
75 .ident = "ASUSTeK COMPUTER INC. X401U",
76 .matches = {
77 DMI_MATCH(DMI_SYS_VENDOR, "ASUSTeK COMPUTER INC."),
78 DMI_MATCH(DMI_PRODUCT_NAME, "X401U"),
79 },
80 .driver_data = &quirk_asus_x401u,
81 },
82 {
83 .callback = dmi_matched,
84 .ident = "ASUSTeK COMPUTER INC. X401A1",
85 .matches = {
86 DMI_MATCH(DMI_SYS_VENDOR, "ASUSTeK COMPUTER INC."),
87 DMI_MATCH(DMI_PRODUCT_NAME, "X401A1"),
88 },
89 .driver_data = &quirk_asus_x401u,
90 },
91 {
92 .callback = dmi_matched,
93 .ident = "ASUSTeK COMPUTER INC. X501U",
94 .matches = {
95 DMI_MATCH(DMI_SYS_VENDOR, "ASUSTeK COMPUTER INC."),
96 DMI_MATCH(DMI_PRODUCT_NAME, "X501U"),
97 },
98 .driver_data = &quirk_asus_x401u,
99 },
100 {
101 .callback = dmi_matched,
102 .ident = "ASUSTeK COMPUTER INC. X501A1",
103 .matches = {
104 DMI_MATCH(DMI_SYS_VENDOR, "ASUSTeK COMPUTER INC."),
105 DMI_MATCH(DMI_PRODUCT_NAME, "X501A1"),
106 },
107 .driver_data = &quirk_asus_x401u,
108 },
109 {
110 .callback = dmi_matched,
111 .ident = "ASUSTeK COMPUTER INC. X55A",
112 .matches = {
113 DMI_MATCH(DMI_SYS_VENDOR, "ASUSTeK COMPUTER INC."),
114 DMI_MATCH(DMI_PRODUCT_NAME, "X55A"),
115 },
116 .driver_data = &quirk_asus_x401u,
117 },
118 {
119 .callback = dmi_matched,
120 .ident = "ASUSTeK COMPUTER INC. X55C",
121 .matches = {
122 DMI_MATCH(DMI_SYS_VENDOR, "ASUSTeK COMPUTER INC."),
123 DMI_MATCH(DMI_PRODUCT_NAME, "X55C"),
124 },
125 .driver_data = &quirk_asus_x401u,
126 },
127 {
128 .callback = dmi_matched,
129 .ident = "ASUSTeK COMPUTER INC. X55U",
130 .matches = {
131 DMI_MATCH(DMI_SYS_VENDOR, "ASUSTeK COMPUTER INC."),
132 DMI_MATCH(DMI_PRODUCT_NAME, "X55U"),
133 },
134 .driver_data = &quirk_asus_x401u,
135 },
136 {
137 .callback = dmi_matched,
138 .ident = "ASUSTeK COMPUTER INC. X55VD",
139 .matches = {
140 DMI_MATCH(DMI_SYS_VENDOR, "ASUSTeK COMPUTER INC."),
141 DMI_MATCH(DMI_PRODUCT_NAME, "X55VD"),
142 },
143 .driver_data = &quirk_asus_x401u,
144 },
145 {},
Corentin Chary6a2bccc2012-03-20 09:53:10 +0100146};
147
Corentin Charyfddbfed2011-07-01 11:34:39 +0200148static void asus_nb_wmi_quirks(struct asus_wmi_driver *driver)
149{
AceLan Kaod2044c52012-07-04 11:19:08 +0800150 quirks = &quirk_asus_unknown;
151 dmi_check_system(asus_quirks);
152
153 driver->quirks = quirks;
Corentin Charyfb05b9f2012-03-20 09:53:11 +0100154 driver->panel_power = FB_BLANK_UNBLANK;
AceLan Kaod2044c52012-07-04 11:19:08 +0800155
156 /* overwrite the wapf setting if the wapf paramater is specified */
157 if (wapf != -1)
158 quirks->wapf = wapf;
159 else
160 wapf = quirks->wapf;
Corentin Charyfddbfed2011-07-01 11:34:39 +0200161}
162
Corentin Charyb229ece2011-02-26 10:20:40 +0100163static const struct key_entry asus_nb_wmi_keymap[] = {
164 { KE_KEY, 0x30, { KEY_VOLUMEUP } },
165 { KE_KEY, 0x31, { KEY_VOLUMEDOWN } },
166 { KE_KEY, 0x32, { KEY_MUTE } },
167 { KE_KEY, 0x33, { KEY_DISPLAYTOGGLE } }, /* LCD on */
168 { KE_KEY, 0x34, { KEY_DISPLAY_OFF } }, /* LCD off */
169 { KE_KEY, 0x40, { KEY_PREVIOUSSONG } },
170 { KE_KEY, 0x41, { KEY_NEXTSONG } },
171 { KE_KEY, 0x43, { KEY_STOPCD } },
172 { KE_KEY, 0x45, { KEY_PLAYPAUSE } },
173 { KE_KEY, 0x4c, { KEY_MEDIA } },
174 { KE_KEY, 0x50, { KEY_EMAIL } },
175 { KE_KEY, 0x51, { KEY_WWW } },
176 { KE_KEY, 0x55, { KEY_CALC } },
Corentin Chary9b05ea22012-03-20 09:53:02 +0100177 { KE_IGNORE, 0x57, }, /* Battery mode */
178 { KE_IGNORE, 0x58, }, /* AC mode */
Corentin Charyb229ece2011-02-26 10:20:40 +0100179 { KE_KEY, 0x5C, { KEY_F15 } }, /* Power Gear key */
Corentin Charyfddbfed2011-07-01 11:34:39 +0200180 { KE_KEY, 0x5D, { KEY_WLAN } }, /* Wireless console Toggle */
181 { KE_KEY, 0x5E, { KEY_WLAN } }, /* Wireless console Enable */
182 { KE_KEY, 0x5F, { KEY_WLAN } }, /* Wireless console Disable */
Corentin Charyb229ece2011-02-26 10:20:40 +0100183 { KE_KEY, 0x60, { KEY_SWITCHVIDEOMODE } },
184 { KE_KEY, 0x61, { KEY_SWITCHVIDEOMODE } },
185 { KE_KEY, 0x62, { KEY_SWITCHVIDEOMODE } },
186 { KE_KEY, 0x63, { KEY_SWITCHVIDEOMODE } },
187 { KE_KEY, 0x6B, { KEY_TOUCHPAD_TOGGLE } },
Corentin Charyb229ece2011-02-26 10:20:40 +0100188 { KE_KEY, 0x7D, { KEY_BLUETOOTH } },
Corentin Charyfddbfed2011-07-01 11:34:39 +0200189 { KE_KEY, 0x7E, { KEY_BLUETOOTH } },
Corentin Charyb229ece2011-02-26 10:20:40 +0100190 { KE_KEY, 0x82, { KEY_CAMERA } },
191 { KE_KEY, 0x88, { KEY_RFKILL } },
192 { KE_KEY, 0x8A, { KEY_PROG1 } },
193 { KE_KEY, 0x95, { KEY_MEDIA } },
194 { KE_KEY, 0x99, { KEY_PHONE } },
195 { KE_KEY, 0xb5, { KEY_CALC } },
196 { KE_KEY, 0xc4, { KEY_KBDILLUMUP } },
197 { KE_KEY, 0xc5, { KEY_KBDILLUMDOWN } },
198 { KE_END, 0},
199};
200
201static struct asus_wmi_driver asus_nb_wmi_driver = {
202 .name = ASUS_NB_WMI_FILE,
203 .owner = THIS_MODULE,
204 .event_guid = ASUS_NB_WMI_EVENT_GUID,
205 .keymap = asus_nb_wmi_keymap,
206 .input_name = "Asus WMI hotkeys",
207 .input_phys = ASUS_NB_WMI_FILE "/input0",
AceLan Kaoc87992d2012-03-20 09:53:08 +0100208 .detect_quirks = asus_nb_wmi_quirks,
Corentin Charyb229ece2011-02-26 10:20:40 +0100209};
210
211
212static int __init asus_nb_wmi_init(void)
213{
214 return asus_wmi_register_driver(&asus_nb_wmi_driver);
215}
216
217static void __exit asus_nb_wmi_exit(void)
218{
219 asus_wmi_unregister_driver(&asus_nb_wmi_driver);
220}
221
222module_init(asus_nb_wmi_init);
223module_exit(asus_nb_wmi_exit);