hid-asus.c 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594
  1. /*
  2. * HID driver for Asus notebook built-in keyboard.
  3. * Fixes small logical maximum to match usage maximum.
  4. *
  5. * Currently supported devices are:
  6. * EeeBook X205TA
  7. * VivoBook E200HA
  8. *
  9. * Copyright (c) 2016 Yusuke Fujimaki <usk.fujimaki@gmail.com>
  10. *
  11. * This module based on hid-ortek by
  12. * Copyright (c) 2010 Johnathon Harris <jmharris@gmail.com>
  13. * Copyright (c) 2011 Jiri Kosina
  14. *
  15. * This module has been updated to add support for Asus i2c touchpad.
  16. *
  17. * Copyright (c) 2016 Brendan McGrath <redmcg@redmandi.dyndns.org>
  18. * Copyright (c) 2016 Victor Vlasenko <victor.vlasenko@sysgears.com>
  19. * Copyright (c) 2016 Frederik Wenigwieser <frederik.wenigwieser@gmail.com>
  20. */
  21. /*
  22. * This program is free software; you can redistribute it and/or modify it
  23. * under the terms of the GNU General Public License as published by the Free
  24. * Software Foundation; either version 2 of the License, or (at your option)
  25. * any later version.
  26. */
  27. #include <linux/hid.h>
  28. #include <linux/module.h>
  29. #include <linux/input/mt.h>
  30. #include "hid-ids.h"
  31. MODULE_AUTHOR("Yusuke Fujimaki <usk.fujimaki@gmail.com>");
  32. MODULE_AUTHOR("Brendan McGrath <redmcg@redmandi.dyndns.org>");
  33. MODULE_AUTHOR("Victor Vlasenko <victor.vlasenko@sysgears.com>");
  34. MODULE_AUTHOR("Frederik Wenigwieser <frederik.wenigwieser@gmail.com>");
  35. MODULE_DESCRIPTION("Asus HID Keyboard and TouchPad");
  36. #define FEATURE_REPORT_ID 0x0d
  37. #define INPUT_REPORT_ID 0x5d
  38. #define FEATURE_KBD_REPORT_ID 0x5a
  39. #define INPUT_REPORT_SIZE 28
  40. #define FEATURE_KBD_REPORT_SIZE 16
  41. #define SUPPORT_KBD_BACKLIGHT BIT(0)
  42. #define MAX_CONTACTS 5
  43. #define MAX_X 2794
  44. #define MAX_Y 1758
  45. #define MAX_TOUCH_MAJOR 8
  46. #define MAX_PRESSURE 128
  47. #define CONTACT_DATA_SIZE 5
  48. #define BTN_LEFT_MASK 0x01
  49. #define CONTACT_TOOL_TYPE_MASK 0x80
  50. #define CONTACT_X_MSB_MASK 0xf0
  51. #define CONTACT_Y_MSB_MASK 0x0f
  52. #define CONTACT_TOUCH_MAJOR_MASK 0x07
  53. #define CONTACT_PRESSURE_MASK 0x7f
  54. #define QUIRK_FIX_NOTEBOOK_REPORT BIT(0)
  55. #define QUIRK_NO_INIT_REPORTS BIT(1)
  56. #define QUIRK_SKIP_INPUT_MAPPING BIT(2)
  57. #define QUIRK_IS_MULTITOUCH BIT(3)
  58. #define QUIRK_NO_CONSUMER_USAGES BIT(4)
  59. #define QUIRK_USE_KBD_BACKLIGHT BIT(5)
  60. #define QUIRK_T100_KEYBOARD BIT(6)
  61. #define I2C_KEYBOARD_QUIRKS (QUIRK_FIX_NOTEBOOK_REPORT | \
  62. QUIRK_NO_INIT_REPORTS | \
  63. QUIRK_NO_CONSUMER_USAGES)
  64. #define I2C_TOUCHPAD_QUIRKS (QUIRK_NO_INIT_REPORTS | \
  65. QUIRK_SKIP_INPUT_MAPPING | \
  66. QUIRK_IS_MULTITOUCH)
  67. #define TRKID_SGN ((TRKID_MAX + 1) >> 1)
  68. struct asus_kbd_leds {
  69. struct led_classdev cdev;
  70. struct hid_device *hdev;
  71. struct work_struct work;
  72. unsigned int brightness;
  73. bool removed;
  74. };
  75. struct asus_drvdata {
  76. unsigned long quirks;
  77. struct input_dev *input;
  78. struct asus_kbd_leds *kbd_backlight;
  79. bool enable_backlight;
  80. };
  81. static void asus_report_contact_down(struct input_dev *input,
  82. int toolType, u8 *data)
  83. {
  84. int touch_major, pressure;
  85. int x = (data[0] & CONTACT_X_MSB_MASK) << 4 | data[1];
  86. int y = MAX_Y - ((data[0] & CONTACT_Y_MSB_MASK) << 8 | data[2]);
  87. if (toolType == MT_TOOL_PALM) {
  88. touch_major = MAX_TOUCH_MAJOR;
  89. pressure = MAX_PRESSURE;
  90. } else {
  91. touch_major = (data[3] >> 4) & CONTACT_TOUCH_MAJOR_MASK;
  92. pressure = data[4] & CONTACT_PRESSURE_MASK;
  93. }
  94. input_report_abs(input, ABS_MT_POSITION_X, x);
  95. input_report_abs(input, ABS_MT_POSITION_Y, y);
  96. input_report_abs(input, ABS_MT_TOUCH_MAJOR, touch_major);
  97. input_report_abs(input, ABS_MT_PRESSURE, pressure);
  98. }
  99. /* Required for Synaptics Palm Detection */
  100. static void asus_report_tool_width(struct input_dev *input)
  101. {
  102. struct input_mt *mt = input->mt;
  103. struct input_mt_slot *oldest;
  104. int oldid, count, i;
  105. oldest = NULL;
  106. oldid = mt->trkid;
  107. count = 0;
  108. for (i = 0; i < mt->num_slots; ++i) {
  109. struct input_mt_slot *ps = &mt->slots[i];
  110. int id = input_mt_get_value(ps, ABS_MT_TRACKING_ID);
  111. if (id < 0)
  112. continue;
  113. if ((id - oldid) & TRKID_SGN) {
  114. oldest = ps;
  115. oldid = id;
  116. }
  117. count++;
  118. }
  119. if (oldest) {
  120. input_report_abs(input, ABS_TOOL_WIDTH,
  121. input_mt_get_value(oldest, ABS_MT_TOUCH_MAJOR));
  122. }
  123. }
  124. static void asus_report_input(struct input_dev *input, u8 *data)
  125. {
  126. int i;
  127. u8 *contactData = data + 2;
  128. for (i = 0; i < MAX_CONTACTS; i++) {
  129. bool down = !!(data[1] & BIT(i+3));
  130. int toolType = contactData[3] & CONTACT_TOOL_TYPE_MASK ?
  131. MT_TOOL_PALM : MT_TOOL_FINGER;
  132. input_mt_slot(input, i);
  133. input_mt_report_slot_state(input, toolType, down);
  134. if (down) {
  135. asus_report_contact_down(input, toolType, contactData);
  136. contactData += CONTACT_DATA_SIZE;
  137. }
  138. }
  139. input_report_key(input, BTN_LEFT, data[1] & BTN_LEFT_MASK);
  140. asus_report_tool_width(input);
  141. input_mt_sync_frame(input);
  142. input_sync(input);
  143. }
  144. static int asus_raw_event(struct hid_device *hdev,
  145. struct hid_report *report, u8 *data, int size)
  146. {
  147. struct asus_drvdata *drvdata = hid_get_drvdata(hdev);
  148. if (drvdata->quirks & QUIRK_IS_MULTITOUCH &&
  149. data[0] == INPUT_REPORT_ID &&
  150. size == INPUT_REPORT_SIZE) {
  151. asus_report_input(drvdata->input, data);
  152. return 1;
  153. }
  154. return 0;
  155. }
  156. static int asus_kbd_set_report(struct hid_device *hdev, u8 *buf, size_t buf_size)
  157. {
  158. unsigned char *dmabuf;
  159. int ret;
  160. dmabuf = kmemdup(buf, buf_size, GFP_KERNEL);
  161. if (!dmabuf)
  162. return -ENOMEM;
  163. ret = hid_hw_raw_request(hdev, FEATURE_KBD_REPORT_ID, dmabuf,
  164. buf_size, HID_FEATURE_REPORT,
  165. HID_REQ_SET_REPORT);
  166. kfree(dmabuf);
  167. return ret;
  168. }
  169. static int asus_kbd_init(struct hid_device *hdev)
  170. {
  171. u8 buf[] = { FEATURE_KBD_REPORT_ID, 0x41, 0x53, 0x55, 0x53, 0x20, 0x54,
  172. 0x65, 0x63, 0x68, 0x2e, 0x49, 0x6e, 0x63, 0x2e, 0x00 };
  173. int ret;
  174. ret = asus_kbd_set_report(hdev, buf, sizeof(buf));
  175. if (ret < 0)
  176. hid_err(hdev, "Asus failed to send init command: %d\n", ret);
  177. return ret;
  178. }
  179. static int asus_kbd_get_functions(struct hid_device *hdev,
  180. unsigned char *kbd_func)
  181. {
  182. u8 buf[] = { FEATURE_KBD_REPORT_ID, 0x05, 0x20, 0x31, 0x00, 0x08 };
  183. u8 *readbuf;
  184. int ret;
  185. ret = asus_kbd_set_report(hdev, buf, sizeof(buf));
  186. if (ret < 0) {
  187. hid_err(hdev, "Asus failed to send configuration command: %d\n", ret);
  188. return ret;
  189. }
  190. readbuf = kzalloc(FEATURE_KBD_REPORT_SIZE, GFP_KERNEL);
  191. if (!readbuf)
  192. return -ENOMEM;
  193. ret = hid_hw_raw_request(hdev, FEATURE_KBD_REPORT_ID, readbuf,
  194. FEATURE_KBD_REPORT_SIZE, HID_FEATURE_REPORT,
  195. HID_REQ_GET_REPORT);
  196. if (ret < 0) {
  197. hid_err(hdev, "Asus failed to request functions: %d\n", ret);
  198. kfree(readbuf);
  199. return ret;
  200. }
  201. *kbd_func = readbuf[6];
  202. kfree(readbuf);
  203. return ret;
  204. }
  205. static void asus_kbd_backlight_set(struct led_classdev *led_cdev,
  206. enum led_brightness brightness)
  207. {
  208. struct asus_kbd_leds *led = container_of(led_cdev, struct asus_kbd_leds,
  209. cdev);
  210. if (led->brightness == brightness)
  211. return;
  212. led->brightness = brightness;
  213. schedule_work(&led->work);
  214. }
  215. static enum led_brightness asus_kbd_backlight_get(struct led_classdev *led_cdev)
  216. {
  217. struct asus_kbd_leds *led = container_of(led_cdev, struct asus_kbd_leds,
  218. cdev);
  219. return led->brightness;
  220. }
  221. static void asus_kbd_backlight_work(struct work_struct *work)
  222. {
  223. struct asus_kbd_leds *led = container_of(work, struct asus_kbd_leds, work);
  224. u8 buf[] = { FEATURE_KBD_REPORT_ID, 0xba, 0xc5, 0xc4, 0x00 };
  225. int ret;
  226. if (led->removed)
  227. return;
  228. buf[4] = led->brightness;
  229. ret = asus_kbd_set_report(led->hdev, buf, sizeof(buf));
  230. if (ret < 0)
  231. hid_err(led->hdev, "Asus failed to set keyboard backlight: %d\n", ret);
  232. }
  233. static int asus_kbd_register_leds(struct hid_device *hdev)
  234. {
  235. struct asus_drvdata *drvdata = hid_get_drvdata(hdev);
  236. unsigned char kbd_func;
  237. int ret;
  238. /* Initialize keyboard */
  239. ret = asus_kbd_init(hdev);
  240. if (ret < 0)
  241. return ret;
  242. /* Get keyboard functions */
  243. ret = asus_kbd_get_functions(hdev, &kbd_func);
  244. if (ret < 0)
  245. return ret;
  246. /* Check for backlight support */
  247. if (!(kbd_func & SUPPORT_KBD_BACKLIGHT))
  248. return -ENODEV;
  249. drvdata->kbd_backlight = devm_kzalloc(&hdev->dev,
  250. sizeof(struct asus_kbd_leds),
  251. GFP_KERNEL);
  252. if (!drvdata->kbd_backlight)
  253. return -ENOMEM;
  254. drvdata->kbd_backlight->removed = false;
  255. drvdata->kbd_backlight->brightness = 0;
  256. drvdata->kbd_backlight->hdev = hdev;
  257. drvdata->kbd_backlight->cdev.name = "asus::kbd_backlight";
  258. drvdata->kbd_backlight->cdev.max_brightness = 3;
  259. drvdata->kbd_backlight->cdev.brightness_set = asus_kbd_backlight_set;
  260. drvdata->kbd_backlight->cdev.brightness_get = asus_kbd_backlight_get;
  261. INIT_WORK(&drvdata->kbd_backlight->work, asus_kbd_backlight_work);
  262. ret = devm_led_classdev_register(&hdev->dev, &drvdata->kbd_backlight->cdev);
  263. if (ret < 0) {
  264. /* No need to have this still around */
  265. devm_kfree(&hdev->dev, drvdata->kbd_backlight);
  266. }
  267. return ret;
  268. }
  269. static int asus_input_configured(struct hid_device *hdev, struct hid_input *hi)
  270. {
  271. struct input_dev *input = hi->input;
  272. struct asus_drvdata *drvdata = hid_get_drvdata(hdev);
  273. if (drvdata->quirks & QUIRK_IS_MULTITOUCH) {
  274. int ret;
  275. input_set_abs_params(input, ABS_MT_POSITION_X, 0, MAX_X, 0, 0);
  276. input_set_abs_params(input, ABS_MT_POSITION_Y, 0, MAX_Y, 0, 0);
  277. input_set_abs_params(input, ABS_TOOL_WIDTH, 0, MAX_TOUCH_MAJOR, 0, 0);
  278. input_set_abs_params(input, ABS_MT_TOUCH_MAJOR, 0, MAX_TOUCH_MAJOR, 0, 0);
  279. input_set_abs_params(input, ABS_MT_PRESSURE, 0, MAX_PRESSURE, 0, 0);
  280. __set_bit(BTN_LEFT, input->keybit);
  281. __set_bit(INPUT_PROP_BUTTONPAD, input->propbit);
  282. ret = input_mt_init_slots(input, MAX_CONTACTS, INPUT_MT_POINTER);
  283. if (ret) {
  284. hid_err(hdev, "Asus input mt init slots failed: %d\n", ret);
  285. return ret;
  286. }
  287. }
  288. drvdata->input = input;
  289. if (drvdata->enable_backlight && asus_kbd_register_leds(hdev))
  290. hid_warn(hdev, "Failed to initialize backlight.\n");
  291. return 0;
  292. }
  293. #define asus_map_key_clear(c) hid_map_usage_clear(hi, usage, bit, \
  294. max, EV_KEY, (c))
  295. static int asus_input_mapping(struct hid_device *hdev,
  296. struct hid_input *hi, struct hid_field *field,
  297. struct hid_usage *usage, unsigned long **bit,
  298. int *max)
  299. {
  300. struct asus_drvdata *drvdata = hid_get_drvdata(hdev);
  301. if (drvdata->quirks & QUIRK_SKIP_INPUT_MAPPING) {
  302. /* Don't map anything from the HID report.
  303. * We do it all manually in asus_input_configured
  304. */
  305. return -1;
  306. }
  307. /* ASUS-specific keyboard hotkeys */
  308. if ((usage->hid & HID_USAGE_PAGE) == 0xff310000) {
  309. set_bit(EV_REP, hi->input->evbit);
  310. switch (usage->hid & HID_USAGE) {
  311. case 0x10: asus_map_key_clear(KEY_BRIGHTNESSDOWN); break;
  312. case 0x20: asus_map_key_clear(KEY_BRIGHTNESSUP); break;
  313. case 0x35: asus_map_key_clear(KEY_DISPLAY_OFF); break;
  314. case 0x6c: asus_map_key_clear(KEY_SLEEP); break;
  315. case 0x82: asus_map_key_clear(KEY_CAMERA); break;
  316. case 0x88: asus_map_key_clear(KEY_RFKILL); break;
  317. case 0xb5: asus_map_key_clear(KEY_CALC); break;
  318. case 0xc4: asus_map_key_clear(KEY_KBDILLUMUP); break;
  319. case 0xc5: asus_map_key_clear(KEY_KBDILLUMDOWN); break;
  320. /* ASUS touchpad toggle */
  321. case 0x6b: asus_map_key_clear(KEY_F21); break;
  322. /* ROG key */
  323. case 0x38: asus_map_key_clear(KEY_PROG1); break;
  324. /* Fn+C ASUS Splendid */
  325. case 0xba: asus_map_key_clear(KEY_PROG2); break;
  326. /* Fn+Space Power4Gear Hybrid */
  327. case 0x5c: asus_map_key_clear(KEY_PROG3); break;
  328. default:
  329. /* ASUS lazily declares 256 usages, ignore the rest,
  330. * as some make the keyboard appear as a pointer device. */
  331. return -1;
  332. }
  333. /*
  334. * Check and enable backlight only on devices with UsagePage ==
  335. * 0xff31 to avoid initializing the keyboard firmware multiple
  336. * times on devices with multiple HID descriptors but same
  337. * PID/VID.
  338. */
  339. if (drvdata->quirks & QUIRK_USE_KBD_BACKLIGHT)
  340. drvdata->enable_backlight = true;
  341. return 1;
  342. }
  343. if (drvdata->quirks & QUIRK_NO_CONSUMER_USAGES &&
  344. (usage->hid & HID_USAGE_PAGE) == HID_UP_CONSUMER) {
  345. switch (usage->hid & HID_USAGE) {
  346. case 0xe2: /* Mute */
  347. case 0xe9: /* Volume up */
  348. case 0xea: /* Volume down */
  349. return 0;
  350. default:
  351. /* Ignore dummy Consumer usages which make the
  352. * keyboard incorrectly appear as a pointer device.
  353. */
  354. return -1;
  355. }
  356. }
  357. return 0;
  358. }
  359. static int asus_start_multitouch(struct hid_device *hdev)
  360. {
  361. int ret;
  362. const unsigned char buf[] = { FEATURE_REPORT_ID, 0x00, 0x03, 0x01, 0x00 };
  363. unsigned char *dmabuf = kmemdup(buf, sizeof(buf), GFP_KERNEL);
  364. if (!dmabuf) {
  365. ret = -ENOMEM;
  366. hid_err(hdev, "Asus failed to alloc dma buf: %d\n", ret);
  367. return ret;
  368. }
  369. ret = hid_hw_raw_request(hdev, dmabuf[0], dmabuf, sizeof(buf),
  370. HID_FEATURE_REPORT, HID_REQ_SET_REPORT);
  371. kfree(dmabuf);
  372. if (ret != sizeof(buf)) {
  373. hid_err(hdev, "Asus failed to start multitouch: %d\n", ret);
  374. return ret;
  375. }
  376. return 0;
  377. }
  378. static int __maybe_unused asus_reset_resume(struct hid_device *hdev)
  379. {
  380. struct asus_drvdata *drvdata = hid_get_drvdata(hdev);
  381. if (drvdata->quirks & QUIRK_IS_MULTITOUCH)
  382. return asus_start_multitouch(hdev);
  383. return 0;
  384. }
  385. static int asus_probe(struct hid_device *hdev, const struct hid_device_id *id)
  386. {
  387. int ret;
  388. struct asus_drvdata *drvdata;
  389. drvdata = devm_kzalloc(&hdev->dev, sizeof(*drvdata), GFP_KERNEL);
  390. if (drvdata == NULL) {
  391. hid_err(hdev, "Can't alloc Asus descriptor\n");
  392. return -ENOMEM;
  393. }
  394. hid_set_drvdata(hdev, drvdata);
  395. drvdata->quirks = id->driver_data;
  396. if (drvdata->quirks & QUIRK_NO_INIT_REPORTS)
  397. hdev->quirks |= HID_QUIRK_NO_INIT_REPORTS;
  398. ret = hid_parse(hdev);
  399. if (ret) {
  400. hid_err(hdev, "Asus hid parse failed: %d\n", ret);
  401. return ret;
  402. }
  403. ret = hid_hw_start(hdev, HID_CONNECT_DEFAULT);
  404. if (ret) {
  405. hid_err(hdev, "Asus hw start failed: %d\n", ret);
  406. return ret;
  407. }
  408. if (!drvdata->input) {
  409. hid_err(hdev, "Asus input not registered\n");
  410. ret = -ENOMEM;
  411. goto err_stop_hw;
  412. }
  413. if (drvdata->quirks & QUIRK_IS_MULTITOUCH) {
  414. drvdata->input->name = "Asus TouchPad";
  415. } else {
  416. drvdata->input->name = "Asus Keyboard";
  417. }
  418. if (drvdata->quirks & QUIRK_IS_MULTITOUCH) {
  419. ret = asus_start_multitouch(hdev);
  420. if (ret)
  421. goto err_stop_hw;
  422. }
  423. return 0;
  424. err_stop_hw:
  425. hid_hw_stop(hdev);
  426. return ret;
  427. }
  428. static void asus_remove(struct hid_device *hdev)
  429. {
  430. struct asus_drvdata *drvdata = hid_get_drvdata(hdev);
  431. if (drvdata->kbd_backlight) {
  432. drvdata->kbd_backlight->removed = true;
  433. cancel_work_sync(&drvdata->kbd_backlight->work);
  434. }
  435. hid_hw_stop(hdev);
  436. }
  437. static __u8 *asus_report_fixup(struct hid_device *hdev, __u8 *rdesc,
  438. unsigned int *rsize)
  439. {
  440. struct asus_drvdata *drvdata = hid_get_drvdata(hdev);
  441. if (drvdata->quirks & QUIRK_FIX_NOTEBOOK_REPORT &&
  442. *rsize >= 56 && rdesc[54] == 0x25 && rdesc[55] == 0x65) {
  443. hid_info(hdev, "Fixing up Asus notebook report descriptor\n");
  444. rdesc[55] = 0xdd;
  445. }
  446. if (drvdata->quirks & QUIRK_T100_KEYBOARD &&
  447. *rsize == 76 && rdesc[73] == 0x81 && rdesc[74] == 0x01) {
  448. hid_info(hdev, "Fixing up Asus T100 keyb report descriptor\n");
  449. rdesc[74] &= ~HID_MAIN_ITEM_CONSTANT;
  450. }
  451. return rdesc;
  452. }
  453. static const struct hid_device_id asus_devices[] = {
  454. { HID_I2C_DEVICE(USB_VENDOR_ID_ASUSTEK,
  455. USB_DEVICE_ID_ASUSTEK_I2C_KEYBOARD), I2C_KEYBOARD_QUIRKS},
  456. { HID_I2C_DEVICE(USB_VENDOR_ID_ASUSTEK,
  457. USB_DEVICE_ID_ASUSTEK_I2C_TOUCHPAD), I2C_TOUCHPAD_QUIRKS },
  458. { HID_USB_DEVICE(USB_VENDOR_ID_ASUSTEK,
  459. USB_DEVICE_ID_ASUSTEK_ROG_KEYBOARD1) },
  460. { HID_USB_DEVICE(USB_VENDOR_ID_ASUSTEK,
  461. USB_DEVICE_ID_ASUSTEK_ROG_KEYBOARD2), QUIRK_USE_KBD_BACKLIGHT },
  462. { HID_USB_DEVICE(USB_VENDOR_ID_ASUSTEK,
  463. USB_DEVICE_ID_ASUSTEK_T100_KEYBOARD),
  464. QUIRK_T100_KEYBOARD | QUIRK_NO_CONSUMER_USAGES },
  465. { }
  466. };
  467. MODULE_DEVICE_TABLE(hid, asus_devices);
  468. static struct hid_driver asus_driver = {
  469. .name = "asus",
  470. .id_table = asus_devices,
  471. .report_fixup = asus_report_fixup,
  472. .probe = asus_probe,
  473. .remove = asus_remove,
  474. .input_mapping = asus_input_mapping,
  475. .input_configured = asus_input_configured,
  476. #ifdef CONFIG_PM
  477. .reset_resume = asus_reset_resume,
  478. #endif
  479. .raw_event = asus_raw_event
  480. };
  481. module_hid_driver(asus_driver);
  482. MODULE_LICENSE("GPL");