asus-nb-wmi.c 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620
  1. // SPDX-License-Identifier: GPL-2.0-or-later
  2. /*
  3. * Asus Notebooks WMI hotkey driver
  4. *
  5. * Copyright(C) 2010 Corentin Chary <[email protected]>
  6. */
  7. #define pr_fmt(fmt) KBUILD_MODNAME ": " fmt
  8. #include <linux/kernel.h>
  9. #include <linux/module.h>
  10. #include <linux/init.h>
  11. #include <linux/input.h>
  12. #include <linux/input/sparse-keymap.h>
  13. #include <linux/fb.h>
  14. #include <linux/dmi.h>
  15. #include <linux/i8042.h>
  16. #include "asus-wmi.h"
  17. #define ASUS_NB_WMI_FILE "asus-nb-wmi"
  18. MODULE_AUTHOR("Corentin Chary <[email protected]>");
  19. MODULE_DESCRIPTION("Asus Notebooks WMI Hotkey Driver");
  20. MODULE_LICENSE("GPL");
  21. #define ASUS_NB_WMI_EVENT_GUID "0B3CBB35-E3C2-45ED-91C2-4C5A6D195D1C"
  22. MODULE_ALIAS("wmi:"ASUS_NB_WMI_EVENT_GUID);
  23. /*
  24. * WAPF defines the behavior of the Fn+Fx wlan key
  25. * The significance of values is yet to be found, but
  26. * most of the time:
  27. * Bit | Bluetooth | WLAN
  28. * 0 | Hardware | Hardware
  29. * 1 | Hardware | Software
  30. * 4 | Software | Software
  31. */
  32. static int wapf = -1;
  33. module_param(wapf, uint, 0444);
  34. MODULE_PARM_DESC(wapf, "WAPF value");
  35. static int tablet_mode_sw = -1;
  36. module_param(tablet_mode_sw, uint, 0444);
  37. MODULE_PARM_DESC(tablet_mode_sw, "Tablet mode detect: -1:auto 0:disable 1:kbd-dock 2:lid-flip 3:lid-flip-rog");
  38. static struct quirk_entry *quirks;
  39. static bool asus_q500a_i8042_filter(unsigned char data, unsigned char str,
  40. struct serio *port)
  41. {
  42. static bool extended;
  43. bool ret = false;
  44. if (str & I8042_STR_AUXDATA)
  45. return false;
  46. if (unlikely(data == 0xe1)) {
  47. extended = true;
  48. ret = true;
  49. } else if (unlikely(extended)) {
  50. extended = false;
  51. ret = true;
  52. }
  53. return ret;
  54. }
  55. static struct quirk_entry quirk_asus_unknown = {
  56. .wapf = 0,
  57. .wmi_backlight_set_devstate = true,
  58. };
  59. static struct quirk_entry quirk_asus_q500a = {
  60. .i8042_filter = asus_q500a_i8042_filter,
  61. .wmi_backlight_set_devstate = true,
  62. };
  63. /*
  64. * For those machines that need software to control bt/wifi status
  65. * and have duplicate events(ACPI and WMI) for display toggle
  66. */
  67. static struct quirk_entry quirk_asus_x55u = {
  68. .wapf = 4,
  69. .wmi_backlight_set_devstate = true,
  70. .no_display_toggle = true,
  71. };
  72. static struct quirk_entry quirk_asus_wapf4 = {
  73. .wapf = 4,
  74. .wmi_backlight_set_devstate = true,
  75. };
  76. static struct quirk_entry quirk_asus_x200ca = {
  77. .wapf = 2,
  78. .wmi_backlight_set_devstate = true,
  79. };
  80. static struct quirk_entry quirk_asus_x550lb = {
  81. .wmi_backlight_set_devstate = true,
  82. .xusb2pr = 0x01D9,
  83. };
  84. static struct quirk_entry quirk_asus_forceals = {
  85. .wmi_backlight_set_devstate = true,
  86. .wmi_force_als_set = true,
  87. };
  88. static struct quirk_entry quirk_asus_use_kbd_dock_devid = {
  89. .tablet_switch_mode = asus_wmi_kbd_dock_devid,
  90. };
  91. static struct quirk_entry quirk_asus_use_lid_flip_devid = {
  92. .wmi_backlight_set_devstate = true,
  93. .tablet_switch_mode = asus_wmi_lid_flip_devid,
  94. };
  95. static struct quirk_entry quirk_asus_tablet_mode = {
  96. .wmi_backlight_set_devstate = true,
  97. .tablet_switch_mode = asus_wmi_lid_flip_rog_devid,
  98. };
  99. static struct quirk_entry quirk_asus_ignore_fan = {
  100. .wmi_ignore_fan = true,
  101. };
  102. static int dmi_matched(const struct dmi_system_id *dmi)
  103. {
  104. pr_info("Identified laptop model '%s'\n", dmi->ident);
  105. quirks = dmi->driver_data;
  106. return 1;
  107. }
  108. static const struct dmi_system_id asus_quirks[] = {
  109. {
  110. .callback = dmi_matched,
  111. .ident = "ASUSTeK COMPUTER INC. Q500A",
  112. .matches = {
  113. DMI_MATCH(DMI_SYS_VENDOR, "ASUSTeK COMPUTER INC."),
  114. DMI_MATCH(DMI_PRODUCT_NAME, "Q500A"),
  115. },
  116. .driver_data = &quirk_asus_q500a,
  117. },
  118. {
  119. .callback = dmi_matched,
  120. .ident = "ASUSTeK COMPUTER INC. U32U",
  121. .matches = {
  122. DMI_MATCH(DMI_SYS_VENDOR, "ASUSTeK Computer Inc."),
  123. DMI_MATCH(DMI_PRODUCT_NAME, "U32U"),
  124. },
  125. .driver_data = &quirk_asus_wapf4,
  126. },
  127. {
  128. .callback = dmi_matched,
  129. .ident = "ASUSTeK COMPUTER INC. X302UA",
  130. .matches = {
  131. DMI_MATCH(DMI_SYS_VENDOR, "ASUSTeK COMPUTER INC."),
  132. DMI_MATCH(DMI_PRODUCT_NAME, "X302UA"),
  133. },
  134. .driver_data = &quirk_asus_wapf4,
  135. },
  136. {
  137. .callback = dmi_matched,
  138. .ident = "ASUSTeK COMPUTER INC. X401U",
  139. .matches = {
  140. DMI_MATCH(DMI_SYS_VENDOR, "ASUSTeK COMPUTER INC."),
  141. DMI_MATCH(DMI_PRODUCT_NAME, "X401U"),
  142. },
  143. .driver_data = &quirk_asus_x55u,
  144. },
  145. {
  146. .callback = dmi_matched,
  147. .ident = "ASUSTeK COMPUTER INC. X401A",
  148. .matches = {
  149. DMI_MATCH(DMI_SYS_VENDOR, "ASUSTeK COMPUTER INC."),
  150. DMI_MATCH(DMI_PRODUCT_NAME, "X401A"),
  151. },
  152. .driver_data = &quirk_asus_wapf4,
  153. },
  154. {
  155. .callback = dmi_matched,
  156. .ident = "ASUSTeK COMPUTER INC. X401A1",
  157. .matches = {
  158. DMI_MATCH(DMI_SYS_VENDOR, "ASUSTeK COMPUTER INC."),
  159. DMI_MATCH(DMI_PRODUCT_NAME, "X401A1"),
  160. },
  161. .driver_data = &quirk_asus_wapf4,
  162. },
  163. {
  164. .callback = dmi_matched,
  165. .ident = "ASUSTeK COMPUTER INC. X45U",
  166. .matches = {
  167. DMI_MATCH(DMI_SYS_VENDOR, "ASUSTeK COMPUTER INC."),
  168. DMI_MATCH(DMI_PRODUCT_NAME, "X45U"),
  169. },
  170. .driver_data = &quirk_asus_wapf4,
  171. },
  172. {
  173. .callback = dmi_matched,
  174. .ident = "ASUSTeK COMPUTER INC. X456UA",
  175. .matches = {
  176. DMI_MATCH(DMI_SYS_VENDOR, "ASUSTeK COMPUTER INC."),
  177. DMI_MATCH(DMI_PRODUCT_NAME, "X456UA"),
  178. },
  179. .driver_data = &quirk_asus_wapf4,
  180. },
  181. {
  182. .callback = dmi_matched,
  183. .ident = "ASUSTeK COMPUTER INC. X456UF",
  184. .matches = {
  185. DMI_MATCH(DMI_SYS_VENDOR, "ASUSTeK COMPUTER INC."),
  186. DMI_MATCH(DMI_PRODUCT_NAME, "X456UF"),
  187. },
  188. .driver_data = &quirk_asus_wapf4,
  189. },
  190. {
  191. .callback = dmi_matched,
  192. .ident = "ASUSTeK COMPUTER INC. X501U",
  193. .matches = {
  194. DMI_MATCH(DMI_SYS_VENDOR, "ASUSTeK COMPUTER INC."),
  195. DMI_MATCH(DMI_PRODUCT_NAME, "X501U"),
  196. },
  197. .driver_data = &quirk_asus_x55u,
  198. },
  199. {
  200. .callback = dmi_matched,
  201. .ident = "ASUSTeK COMPUTER INC. X501A",
  202. .matches = {
  203. DMI_MATCH(DMI_SYS_VENDOR, "ASUSTeK COMPUTER INC."),
  204. DMI_MATCH(DMI_PRODUCT_NAME, "X501A"),
  205. },
  206. .driver_data = &quirk_asus_wapf4,
  207. },
  208. {
  209. .callback = dmi_matched,
  210. .ident = "ASUSTeK COMPUTER INC. X501A1",
  211. .matches = {
  212. DMI_MATCH(DMI_SYS_VENDOR, "ASUSTeK COMPUTER INC."),
  213. DMI_MATCH(DMI_PRODUCT_NAME, "X501A1"),
  214. },
  215. .driver_data = &quirk_asus_wapf4,
  216. },
  217. {
  218. .callback = dmi_matched,
  219. .ident = "ASUSTeK COMPUTER INC. X550CA",
  220. .matches = {
  221. DMI_MATCH(DMI_SYS_VENDOR, "ASUSTeK COMPUTER INC."),
  222. DMI_MATCH(DMI_PRODUCT_NAME, "X550CA"),
  223. },
  224. .driver_data = &quirk_asus_wapf4,
  225. },
  226. {
  227. .callback = dmi_matched,
  228. .ident = "ASUSTeK COMPUTER INC. X550CC",
  229. .matches = {
  230. DMI_MATCH(DMI_SYS_VENDOR, "ASUSTeK COMPUTER INC."),
  231. DMI_MATCH(DMI_PRODUCT_NAME, "X550CC"),
  232. },
  233. .driver_data = &quirk_asus_wapf4,
  234. },
  235. {
  236. .callback = dmi_matched,
  237. .ident = "ASUSTeK COMPUTER INC. X550CL",
  238. .matches = {
  239. DMI_MATCH(DMI_SYS_VENDOR, "ASUSTeK COMPUTER INC."),
  240. DMI_MATCH(DMI_PRODUCT_NAME, "X550CL"),
  241. },
  242. .driver_data = &quirk_asus_wapf4,
  243. },
  244. {
  245. .callback = dmi_matched,
  246. .ident = "ASUSTeK COMPUTER INC. X550VB",
  247. .matches = {
  248. DMI_MATCH(DMI_SYS_VENDOR, "ASUSTeK COMPUTER INC."),
  249. DMI_MATCH(DMI_PRODUCT_NAME, "X550VB"),
  250. },
  251. .driver_data = &quirk_asus_wapf4,
  252. },
  253. {
  254. .callback = dmi_matched,
  255. .ident = "ASUSTeK COMPUTER INC. X551CA",
  256. .matches = {
  257. DMI_MATCH(DMI_SYS_VENDOR, "ASUSTeK COMPUTER INC."),
  258. DMI_MATCH(DMI_PRODUCT_NAME, "X551CA"),
  259. },
  260. .driver_data = &quirk_asus_wapf4,
  261. },
  262. {
  263. .callback = dmi_matched,
  264. .ident = "ASUSTeK COMPUTER INC. X55A",
  265. .matches = {
  266. DMI_MATCH(DMI_SYS_VENDOR, "ASUSTeK COMPUTER INC."),
  267. DMI_MATCH(DMI_PRODUCT_NAME, "X55A"),
  268. },
  269. .driver_data = &quirk_asus_wapf4,
  270. },
  271. {
  272. .callback = dmi_matched,
  273. .ident = "ASUSTeK COMPUTER INC. X55C",
  274. .matches = {
  275. DMI_MATCH(DMI_SYS_VENDOR, "ASUSTeK COMPUTER INC."),
  276. DMI_MATCH(DMI_PRODUCT_NAME, "X55C"),
  277. },
  278. .driver_data = &quirk_asus_wapf4,
  279. },
  280. {
  281. .callback = dmi_matched,
  282. .ident = "ASUSTeK COMPUTER INC. X55U",
  283. .matches = {
  284. DMI_MATCH(DMI_SYS_VENDOR, "ASUSTeK COMPUTER INC."),
  285. DMI_MATCH(DMI_PRODUCT_NAME, "X55U"),
  286. },
  287. .driver_data = &quirk_asus_x55u,
  288. },
  289. {
  290. .callback = dmi_matched,
  291. .ident = "ASUSTeK COMPUTER INC. X55VD",
  292. .matches = {
  293. DMI_MATCH(DMI_SYS_VENDOR, "ASUSTeK COMPUTER INC."),
  294. DMI_MATCH(DMI_PRODUCT_NAME, "X55VD"),
  295. },
  296. .driver_data = &quirk_asus_wapf4,
  297. },
  298. {
  299. .callback = dmi_matched,
  300. .ident = "ASUSTeK COMPUTER INC. X75A",
  301. .matches = {
  302. DMI_MATCH(DMI_SYS_VENDOR, "ASUSTeK COMPUTER INC."),
  303. DMI_MATCH(DMI_PRODUCT_NAME, "X75A"),
  304. },
  305. .driver_data = &quirk_asus_wapf4,
  306. },
  307. {
  308. .callback = dmi_matched,
  309. .ident = "ASUSTeK COMPUTER INC. X75VBP",
  310. .matches = {
  311. DMI_MATCH(DMI_SYS_VENDOR, "ASUSTeK COMPUTER INC."),
  312. DMI_MATCH(DMI_PRODUCT_NAME, "X75VBP"),
  313. },
  314. .driver_data = &quirk_asus_wapf4,
  315. },
  316. {
  317. .callback = dmi_matched,
  318. .ident = "ASUSTeK COMPUTER INC. X75VD",
  319. .matches = {
  320. DMI_MATCH(DMI_SYS_VENDOR, "ASUSTeK COMPUTER INC."),
  321. DMI_MATCH(DMI_PRODUCT_NAME, "X75VD"),
  322. },
  323. .driver_data = &quirk_asus_wapf4,
  324. },
  325. {
  326. .callback = dmi_matched,
  327. .ident = "ASUSTeK COMPUTER INC. 1015E",
  328. .matches = {
  329. DMI_MATCH(DMI_SYS_VENDOR, "ASUSTeK COMPUTER INC."),
  330. DMI_MATCH(DMI_PRODUCT_NAME, "1015E"),
  331. },
  332. .driver_data = &quirk_asus_wapf4,
  333. },
  334. {
  335. .callback = dmi_matched,
  336. .ident = "ASUSTeK COMPUTER INC. 1015U",
  337. .matches = {
  338. DMI_MATCH(DMI_SYS_VENDOR, "ASUSTeK COMPUTER INC."),
  339. DMI_MATCH(DMI_PRODUCT_NAME, "1015U"),
  340. },
  341. .driver_data = &quirk_asus_wapf4,
  342. },
  343. {
  344. .callback = dmi_matched,
  345. .ident = "ASUSTeK COMPUTER INC. X200CA",
  346. .matches = {
  347. DMI_MATCH(DMI_SYS_VENDOR, "ASUSTeK COMPUTER INC."),
  348. DMI_MATCH(DMI_PRODUCT_NAME, "X200CA"),
  349. },
  350. .driver_data = &quirk_asus_x200ca,
  351. },
  352. {
  353. .callback = dmi_matched,
  354. .ident = "ASUSTeK COMPUTER INC. UX330UAK",
  355. .matches = {
  356. DMI_MATCH(DMI_SYS_VENDOR, "ASUSTeK COMPUTER INC."),
  357. DMI_MATCH(DMI_PRODUCT_NAME, "UX330UAK"),
  358. },
  359. .driver_data = &quirk_asus_forceals,
  360. },
  361. {
  362. .callback = dmi_matched,
  363. .ident = "ASUSTeK COMPUTER INC. X550LB",
  364. .matches = {
  365. DMI_MATCH(DMI_SYS_VENDOR, "ASUSTeK COMPUTER INC."),
  366. DMI_MATCH(DMI_PRODUCT_NAME, "X550LB"),
  367. },
  368. .driver_data = &quirk_asus_x550lb,
  369. },
  370. {
  371. .callback = dmi_matched,
  372. .ident = "ASUSTeK COMPUTER INC. UX430UQ",
  373. .matches = {
  374. DMI_MATCH(DMI_SYS_VENDOR, "ASUSTeK COMPUTER INC."),
  375. DMI_MATCH(DMI_PRODUCT_NAME, "UX430UQ"),
  376. },
  377. .driver_data = &quirk_asus_forceals,
  378. },
  379. {
  380. .callback = dmi_matched,
  381. .ident = "ASUSTeK COMPUTER INC. UX430UNR",
  382. .matches = {
  383. DMI_MATCH(DMI_SYS_VENDOR, "ASUSTeK COMPUTER INC."),
  384. DMI_MATCH(DMI_PRODUCT_NAME, "UX430UNR"),
  385. },
  386. .driver_data = &quirk_asus_forceals,
  387. },
  388. {
  389. .callback = dmi_matched,
  390. .ident = "Asus Transformer T100TA / T100HA / T100CHI",
  391. .matches = {
  392. DMI_MATCH(DMI_SYS_VENDOR, "ASUSTeK COMPUTER INC."),
  393. /* Match *T100* */
  394. DMI_MATCH(DMI_PRODUCT_NAME, "T100"),
  395. },
  396. .driver_data = &quirk_asus_use_kbd_dock_devid,
  397. },
  398. {
  399. .callback = dmi_matched,
  400. .ident = "Asus Transformer T101HA",
  401. .matches = {
  402. DMI_MATCH(DMI_SYS_VENDOR, "ASUSTeK COMPUTER INC."),
  403. DMI_MATCH(DMI_PRODUCT_NAME, "T101HA"),
  404. },
  405. .driver_data = &quirk_asus_use_kbd_dock_devid,
  406. },
  407. {
  408. .callback = dmi_matched,
  409. .ident = "Asus Transformer T200TA",
  410. .matches = {
  411. DMI_MATCH(DMI_SYS_VENDOR, "ASUSTeK COMPUTER INC."),
  412. DMI_MATCH(DMI_PRODUCT_NAME, "T200TA"),
  413. },
  414. .driver_data = &quirk_asus_use_kbd_dock_devid,
  415. },
  416. {
  417. .callback = dmi_matched,
  418. .ident = "ASUS ZenBook Flip UX360",
  419. .matches = {
  420. DMI_MATCH(DMI_SYS_VENDOR, "ASUSTeK COMPUTER INC."),
  421. /* Match UX360* */
  422. DMI_MATCH(DMI_PRODUCT_NAME, "UX360"),
  423. },
  424. .driver_data = &quirk_asus_use_lid_flip_devid,
  425. },
  426. {
  427. .callback = dmi_matched,
  428. .ident = "ASUS TP200s / E205SA",
  429. .matches = {
  430. DMI_MATCH(DMI_SYS_VENDOR, "ASUSTeK COMPUTER INC."),
  431. DMI_MATCH(DMI_PRODUCT_NAME, "E205SA"),
  432. },
  433. .driver_data = &quirk_asus_use_lid_flip_devid,
  434. },
  435. {
  436. .callback = dmi_matched,
  437. .ident = "ASUS ROG FLOW X13",
  438. .matches = {
  439. DMI_MATCH(DMI_SYS_VENDOR, "ASUSTeK COMPUTER INC."),
  440. /* Match GV301** */
  441. DMI_MATCH(DMI_PRODUCT_NAME, "GV301"),
  442. },
  443. .driver_data = &quirk_asus_tablet_mode,
  444. },
  445. {
  446. .callback = dmi_matched,
  447. .ident = "ASUS ROG FLOW X16",
  448. .matches = {
  449. DMI_MATCH(DMI_SYS_VENDOR, "ASUSTeK COMPUTER INC."),
  450. DMI_MATCH(DMI_PRODUCT_NAME, "GV601R"),
  451. },
  452. .driver_data = &quirk_asus_tablet_mode,
  453. },
  454. {
  455. .callback = dmi_matched,
  456. .ident = "ASUS ROG FLOW X16",
  457. .matches = {
  458. DMI_MATCH(DMI_SYS_VENDOR, "ASUSTeK COMPUTER INC."),
  459. DMI_MATCH(DMI_PRODUCT_NAME, "GV601V"),
  460. },
  461. .driver_data = &quirk_asus_tablet_mode,
  462. },
  463. {
  464. .callback = dmi_matched,
  465. .ident = "ASUS VivoBook E410MA",
  466. .matches = {
  467. DMI_MATCH(DMI_SYS_VENDOR, "ASUSTeK COMPUTER INC."),
  468. DMI_MATCH(DMI_PRODUCT_NAME, "E410MA"),
  469. },
  470. .driver_data = &quirk_asus_ignore_fan,
  471. },
  472. {},
  473. };
  474. static void asus_nb_wmi_quirks(struct asus_wmi_driver *driver)
  475. {
  476. quirks = &quirk_asus_unknown;
  477. dmi_check_system(asus_quirks);
  478. driver->quirks = quirks;
  479. driver->panel_power = FB_BLANK_UNBLANK;
  480. /* overwrite the wapf setting if the wapf paramater is specified */
  481. if (wapf != -1)
  482. quirks->wapf = wapf;
  483. else
  484. wapf = quirks->wapf;
  485. if (tablet_mode_sw != -1)
  486. quirks->tablet_switch_mode = tablet_mode_sw;
  487. }
  488. static const struct key_entry asus_nb_wmi_keymap[] = {
  489. { KE_KEY, ASUS_WMI_BRN_DOWN, { KEY_BRIGHTNESSDOWN } },
  490. { KE_KEY, ASUS_WMI_BRN_UP, { KEY_BRIGHTNESSUP } },
  491. { KE_KEY, 0x2a, { KEY_SELECTIVE_SCREENSHOT } },
  492. { KE_IGNORE, 0x2b, }, /* PrintScreen (also send via PS/2) on newer models */
  493. { KE_IGNORE, 0x2c, }, /* CapsLock (also send via PS/2) on newer models */
  494. { KE_KEY, 0x30, { KEY_VOLUMEUP } },
  495. { KE_KEY, 0x31, { KEY_VOLUMEDOWN } },
  496. { KE_KEY, 0x32, { KEY_MUTE } },
  497. { KE_KEY, 0x33, { KEY_SCREENLOCK } },
  498. { KE_KEY, 0x35, { KEY_SCREENLOCK } },
  499. { KE_KEY, 0x38, { KEY_PROG3 } }, /* Armoury Crate */
  500. { KE_KEY, 0x40, { KEY_PREVIOUSSONG } },
  501. { KE_KEY, 0x41, { KEY_NEXTSONG } },
  502. { KE_KEY, 0x43, { KEY_STOPCD } }, /* Stop/Eject */
  503. { KE_KEY, 0x45, { KEY_PLAYPAUSE } },
  504. { KE_KEY, 0x4c, { KEY_MEDIA } }, /* WMP Key */
  505. { KE_KEY, 0x50, { KEY_EMAIL } },
  506. { KE_KEY, 0x51, { KEY_WWW } },
  507. { KE_KEY, 0x55, { KEY_CALC } },
  508. { KE_IGNORE, 0x57, }, /* Battery mode */
  509. { KE_IGNORE, 0x58, }, /* AC mode */
  510. { KE_KEY, 0x5C, { KEY_F15 } }, /* Power Gear key */
  511. { KE_KEY, 0x5D, { KEY_WLAN } }, /* Wireless console Toggle */
  512. { KE_KEY, 0x5E, { KEY_WLAN } }, /* Wireless console Enable */
  513. { KE_KEY, 0x5F, { KEY_WLAN } }, /* Wireless console Disable */
  514. { KE_KEY, 0x60, { KEY_TOUCHPAD_ON } },
  515. { KE_KEY, 0x61, { KEY_SWITCHVIDEOMODE } }, /* SDSP LCD only */
  516. { KE_KEY, 0x62, { KEY_SWITCHVIDEOMODE } }, /* SDSP CRT only */
  517. { KE_KEY, 0x63, { KEY_SWITCHVIDEOMODE } }, /* SDSP LCD + CRT */
  518. { KE_KEY, 0x64, { KEY_SWITCHVIDEOMODE } }, /* SDSP TV */
  519. { KE_KEY, 0x65, { KEY_SWITCHVIDEOMODE } }, /* SDSP LCD + TV */
  520. { KE_KEY, 0x66, { KEY_SWITCHVIDEOMODE } }, /* SDSP CRT + TV */
  521. { KE_KEY, 0x67, { KEY_SWITCHVIDEOMODE } }, /* SDSP LCD + CRT + TV */
  522. { KE_KEY, 0x6B, { KEY_TOUCHPAD_TOGGLE } },
  523. { KE_IGNORE, 0x6E, }, /* Low Battery notification */
  524. { KE_KEY, 0x71, { KEY_F13 } }, /* General-purpose button */
  525. { KE_IGNORE, 0x79, }, /* Charger type dectection notification */
  526. { KE_KEY, 0x7a, { KEY_ALS_TOGGLE } }, /* Ambient Light Sensor Toggle */
  527. { KE_IGNORE, 0x7B, }, /* Charger connect/disconnect notification */
  528. { KE_KEY, 0x7c, { KEY_MICMUTE } },
  529. { KE_KEY, 0x7D, { KEY_BLUETOOTH } }, /* Bluetooth Enable */
  530. { KE_KEY, 0x7E, { KEY_BLUETOOTH } }, /* Bluetooth Disable */
  531. { KE_KEY, 0x82, { KEY_CAMERA } },
  532. { KE_KEY, 0x85, { KEY_CAMERA } },
  533. { KE_KEY, 0x86, { KEY_PROG1 } }, /* MyASUS Key */
  534. { KE_KEY, 0x88, { KEY_RFKILL } }, /* Radio Toggle Key */
  535. { KE_KEY, 0x8A, { KEY_PROG1 } }, /* Color enhancement mode */
  536. { KE_KEY, 0x8C, { KEY_SWITCHVIDEOMODE } }, /* SDSP DVI only */
  537. { KE_KEY, 0x8D, { KEY_SWITCHVIDEOMODE } }, /* SDSP LCD + DVI */
  538. { KE_KEY, 0x8E, { KEY_SWITCHVIDEOMODE } }, /* SDSP CRT + DVI */
  539. { KE_KEY, 0x8F, { KEY_SWITCHVIDEOMODE } }, /* SDSP TV + DVI */
  540. { KE_KEY, 0x90, { KEY_SWITCHVIDEOMODE } }, /* SDSP LCD + CRT + DVI */
  541. { KE_KEY, 0x91, { KEY_SWITCHVIDEOMODE } }, /* SDSP LCD + TV + DVI */
  542. { KE_KEY, 0x92, { KEY_SWITCHVIDEOMODE } }, /* SDSP CRT + TV + DVI */
  543. { KE_KEY, 0x93, { KEY_SWITCHVIDEOMODE } }, /* SDSP LCD + CRT + TV + DVI */
  544. { KE_KEY, 0x95, { KEY_MEDIA } },
  545. { KE_KEY, 0x99, { KEY_PHONE } }, /* Conflicts with fan mode switch */
  546. { KE_KEY, 0xA0, { KEY_SWITCHVIDEOMODE } }, /* SDSP HDMI only */
  547. { KE_KEY, 0xA1, { KEY_SWITCHVIDEOMODE } }, /* SDSP LCD + HDMI */
  548. { KE_KEY, 0xA2, { KEY_SWITCHVIDEOMODE } }, /* SDSP CRT + HDMI */
  549. { KE_KEY, 0xA3, { KEY_SWITCHVIDEOMODE } }, /* SDSP TV + HDMI */
  550. { KE_KEY, 0xA4, { KEY_SWITCHVIDEOMODE } }, /* SDSP LCD + CRT + HDMI */
  551. { KE_KEY, 0xA5, { KEY_SWITCHVIDEOMODE } }, /* SDSP LCD + TV + HDMI */
  552. { KE_KEY, 0xA6, { KEY_SWITCHVIDEOMODE } }, /* SDSP CRT + TV + HDMI */
  553. { KE_KEY, 0xA7, { KEY_SWITCHVIDEOMODE } }, /* SDSP LCD + CRT + TV + HDMI */
  554. { KE_KEY, 0xAE, { KEY_FN_F5 } }, /* Fn+F5 fan mode on 2020+ */
  555. { KE_KEY, 0xB3, { KEY_PROG4 } }, /* AURA */
  556. { KE_KEY, 0xB5, { KEY_CALC } },
  557. { KE_IGNORE, 0xC0, }, /* External display connect/disconnect notification */
  558. { KE_KEY, 0xC4, { KEY_KBDILLUMUP } },
  559. { KE_KEY, 0xC5, { KEY_KBDILLUMDOWN } },
  560. { KE_IGNORE, 0xC6, }, /* Ambient Light Sensor notification */
  561. { KE_KEY, 0xFA, { KEY_PROG2 } }, /* Lid flip action */
  562. { KE_KEY, 0xBD, { KEY_PROG2 } }, /* Lid flip action on ROG xflow laptops */
  563. { KE_END, 0},
  564. };
  565. static struct asus_wmi_driver asus_nb_wmi_driver = {
  566. .name = ASUS_NB_WMI_FILE,
  567. .owner = THIS_MODULE,
  568. .event_guid = ASUS_NB_WMI_EVENT_GUID,
  569. .keymap = asus_nb_wmi_keymap,
  570. .input_name = "Asus WMI hotkeys",
  571. .input_phys = ASUS_NB_WMI_FILE "/input0",
  572. .detect_quirks = asus_nb_wmi_quirks,
  573. };
  574. static int __init asus_nb_wmi_init(void)
  575. {
  576. return asus_wmi_register_driver(&asus_nb_wmi_driver);
  577. }
  578. static void __exit asus_nb_wmi_exit(void)
  579. {
  580. asus_wmi_unregister_driver(&asus_nb_wmi_driver);
  581. }
  582. module_init(asus_nb_wmi_init);
  583. module_exit(asus_nb_wmi_exit);