hid-zydacron.c 3.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208
  1. // SPDX-License-Identifier: GPL-2.0-or-later
  2. /*
  3. * HID driver for zydacron remote control
  4. *
  5. * Copyright (c) 2010 Don Prince <[email protected]>
  6. */
  7. /*
  8. */
  9. #include <linux/device.h>
  10. #include <linux/hid.h>
  11. #include <linux/module.h>
  12. #include "hid-ids.h"
  13. struct zc_device {
  14. struct input_dev *input_ep81;
  15. unsigned short last_key[4];
  16. };
  17. /*
  18. * Zydacron remote control has an invalid HID report descriptor,
  19. * that needs fixing before we can parse it.
  20. */
  21. static __u8 *zc_report_fixup(struct hid_device *hdev, __u8 *rdesc,
  22. unsigned int *rsize)
  23. {
  24. if (*rsize >= 253 &&
  25. rdesc[0x96] == 0xbc && rdesc[0x97] == 0xff &&
  26. rdesc[0xca] == 0xbc && rdesc[0xcb] == 0xff &&
  27. rdesc[0xe1] == 0xbc && rdesc[0xe2] == 0xff) {
  28. hid_info(hdev,
  29. "fixing up zydacron remote control report descriptor\n");
  30. rdesc[0x96] = rdesc[0xca] = rdesc[0xe1] = 0x0c;
  31. rdesc[0x97] = rdesc[0xcb] = rdesc[0xe2] = 0x00;
  32. }
  33. return rdesc;
  34. }
  35. #define zc_map_key_clear(c) \
  36. hid_map_usage_clear(hi, usage, bit, max, EV_KEY, (c))
  37. static int zc_input_mapping(struct hid_device *hdev, struct hid_input *hi,
  38. struct hid_field *field, struct hid_usage *usage,
  39. unsigned long **bit, int *max)
  40. {
  41. int i;
  42. struct zc_device *zc = hid_get_drvdata(hdev);
  43. zc->input_ep81 = hi->input;
  44. if ((usage->hid & HID_USAGE_PAGE) != HID_UP_CONSUMER)
  45. return 0;
  46. dbg_hid("zynacron input mapping event [0x%x]\n",
  47. usage->hid & HID_USAGE);
  48. switch (usage->hid & HID_USAGE) {
  49. /* report 2 */
  50. case 0x10:
  51. zc_map_key_clear(KEY_MODE);
  52. break;
  53. case 0x30:
  54. zc_map_key_clear(KEY_SCREEN);
  55. break;
  56. case 0x70:
  57. zc_map_key_clear(KEY_INFO);
  58. break;
  59. /* report 3 */
  60. case 0x04:
  61. zc_map_key_clear(KEY_RADIO);
  62. break;
  63. /* report 4 */
  64. case 0x0d:
  65. zc_map_key_clear(KEY_PVR);
  66. break;
  67. case 0x25:
  68. zc_map_key_clear(KEY_TV);
  69. break;
  70. case 0x47:
  71. zc_map_key_clear(KEY_AUDIO);
  72. break;
  73. case 0x49:
  74. zc_map_key_clear(KEY_AUX);
  75. break;
  76. case 0x4a:
  77. zc_map_key_clear(KEY_VIDEO);
  78. break;
  79. case 0x48:
  80. zc_map_key_clear(KEY_DVD);
  81. break;
  82. case 0x24:
  83. zc_map_key_clear(KEY_MENU);
  84. break;
  85. case 0x32:
  86. zc_map_key_clear(KEY_TEXT);
  87. break;
  88. default:
  89. return 0;
  90. }
  91. for (i = 0; i < 4; i++)
  92. zc->last_key[i] = 0;
  93. return 1;
  94. }
  95. static int zc_raw_event(struct hid_device *hdev, struct hid_report *report,
  96. u8 *data, int size)
  97. {
  98. struct zc_device *zc = hid_get_drvdata(hdev);
  99. int ret = 0;
  100. unsigned key;
  101. unsigned short index;
  102. if (report->id == data[0]) {
  103. /* break keys */
  104. for (index = 0; index < 4; index++) {
  105. key = zc->last_key[index];
  106. if (key) {
  107. input_event(zc->input_ep81, EV_KEY, key, 0);
  108. zc->last_key[index] = 0;
  109. }
  110. }
  111. key = 0;
  112. switch (report->id) {
  113. case 0x02:
  114. case 0x03:
  115. switch (data[1]) {
  116. case 0x10:
  117. key = KEY_MODE;
  118. index = 0;
  119. break;
  120. case 0x30:
  121. key = KEY_SCREEN;
  122. index = 1;
  123. break;
  124. case 0x70:
  125. key = KEY_INFO;
  126. index = 2;
  127. break;
  128. case 0x04:
  129. key = KEY_RADIO;
  130. index = 3;
  131. break;
  132. }
  133. if (key) {
  134. input_event(zc->input_ep81, EV_KEY, key, 1);
  135. zc->last_key[index] = key;
  136. }
  137. ret = 1;
  138. break;
  139. }
  140. }
  141. return ret;
  142. }
  143. static int zc_probe(struct hid_device *hdev, const struct hid_device_id *id)
  144. {
  145. int ret;
  146. struct zc_device *zc;
  147. zc = devm_kzalloc(&hdev->dev, sizeof(*zc), GFP_KERNEL);
  148. if (zc == NULL) {
  149. hid_err(hdev, "can't alloc descriptor\n");
  150. return -ENOMEM;
  151. }
  152. hid_set_drvdata(hdev, zc);
  153. ret = hid_parse(hdev);
  154. if (ret) {
  155. hid_err(hdev, "parse failed\n");
  156. return ret;
  157. }
  158. ret = hid_hw_start(hdev, HID_CONNECT_DEFAULT);
  159. if (ret) {
  160. hid_err(hdev, "hw start failed\n");
  161. return ret;
  162. }
  163. return 0;
  164. }
  165. static const struct hid_device_id zc_devices[] = {
  166. { HID_USB_DEVICE(USB_VENDOR_ID_ZYDACRON, USB_DEVICE_ID_ZYDACRON_REMOTE_CONTROL) },
  167. { }
  168. };
  169. MODULE_DEVICE_TABLE(hid, zc_devices);
  170. static struct hid_driver zc_driver = {
  171. .name = "zydacron",
  172. .id_table = zc_devices,
  173. .report_fixup = zc_report_fixup,
  174. .input_mapping = zc_input_mapping,
  175. .raw_event = zc_raw_event,
  176. .probe = zc_probe,
  177. };
  178. module_hid_driver(zc_driver);
  179. MODULE_LICENSE("GPL");