asus-nb-wmi.c 20 KB

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