pxrc.c 6.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303
  1. // SPDX-License-Identifier: GPL-2.0
  2. /*
  3. * Driver for Phoenix RC Flight Controller Adapter
  4. *
  5. * Copyright (C) 2018 Marcus Folkesson <marcus.folkesson@gmail.com>
  6. *
  7. */
  8. #include <linux/kernel.h>
  9. #include <linux/errno.h>
  10. #include <linux/slab.h>
  11. #include <linux/module.h>
  12. #include <linux/uaccess.h>
  13. #include <linux/usb.h>
  14. #include <linux/usb/input.h>
  15. #include <linux/mutex.h>
  16. #include <linux/input.h>
  17. #define PXRC_VENDOR_ID (0x1781)
  18. #define PXRC_PRODUCT_ID (0x0898)
  19. static const struct usb_device_id pxrc_table[] = {
  20. { USB_DEVICE(PXRC_VENDOR_ID, PXRC_PRODUCT_ID) },
  21. { }
  22. };
  23. MODULE_DEVICE_TABLE(usb, pxrc_table);
  24. struct pxrc {
  25. struct input_dev *input;
  26. struct usb_device *udev;
  27. struct usb_interface *intf;
  28. struct urb *urb;
  29. struct mutex pm_mutex;
  30. bool is_open;
  31. __u8 epaddr;
  32. char phys[64];
  33. unsigned char *data;
  34. size_t bsize;
  35. };
  36. static void pxrc_usb_irq(struct urb *urb)
  37. {
  38. struct pxrc *pxrc = urb->context;
  39. int error;
  40. switch (urb->status) {
  41. case 0:
  42. /* success */
  43. break;
  44. case -ETIME:
  45. /* this urb is timing out */
  46. dev_dbg(&pxrc->intf->dev,
  47. "%s - urb timed out - was the device unplugged?\n",
  48. __func__);
  49. return;
  50. case -ECONNRESET:
  51. case -ENOENT:
  52. case -ESHUTDOWN:
  53. case -EPIPE:
  54. /* this urb is terminated, clean up */
  55. dev_dbg(&pxrc->intf->dev, "%s - urb shutting down with status: %d\n",
  56. __func__, urb->status);
  57. return;
  58. default:
  59. dev_dbg(&pxrc->intf->dev, "%s - nonzero urb status received: %d\n",
  60. __func__, urb->status);
  61. goto exit;
  62. }
  63. if (urb->actual_length == 8) {
  64. input_report_abs(pxrc->input, ABS_X, pxrc->data[0]);
  65. input_report_abs(pxrc->input, ABS_Y, pxrc->data[2]);
  66. input_report_abs(pxrc->input, ABS_RX, pxrc->data[3]);
  67. input_report_abs(pxrc->input, ABS_RY, pxrc->data[4]);
  68. input_report_abs(pxrc->input, ABS_RUDDER, pxrc->data[5]);
  69. input_report_abs(pxrc->input, ABS_THROTTLE, pxrc->data[6]);
  70. input_report_abs(pxrc->input, ABS_MISC, pxrc->data[7]);
  71. input_report_key(pxrc->input, BTN_A, pxrc->data[1]);
  72. }
  73. exit:
  74. /* Resubmit to fetch new fresh URBs */
  75. error = usb_submit_urb(urb, GFP_ATOMIC);
  76. if (error && error != -EPERM)
  77. dev_err(&pxrc->intf->dev,
  78. "%s - usb_submit_urb failed with result: %d",
  79. __func__, error);
  80. }
  81. static int pxrc_open(struct input_dev *input)
  82. {
  83. struct pxrc *pxrc = input_get_drvdata(input);
  84. int retval;
  85. mutex_lock(&pxrc->pm_mutex);
  86. retval = usb_submit_urb(pxrc->urb, GFP_KERNEL);
  87. if (retval) {
  88. dev_err(&pxrc->intf->dev,
  89. "%s - usb_submit_urb failed, error: %d\n",
  90. __func__, retval);
  91. retval = -EIO;
  92. goto out;
  93. }
  94. pxrc->is_open = true;
  95. out:
  96. mutex_unlock(&pxrc->pm_mutex);
  97. return retval;
  98. }
  99. static void pxrc_close(struct input_dev *input)
  100. {
  101. struct pxrc *pxrc = input_get_drvdata(input);
  102. mutex_lock(&pxrc->pm_mutex);
  103. usb_kill_urb(pxrc->urb);
  104. pxrc->is_open = false;
  105. mutex_unlock(&pxrc->pm_mutex);
  106. }
  107. static int pxrc_usb_init(struct pxrc *pxrc)
  108. {
  109. struct usb_endpoint_descriptor *epirq;
  110. unsigned int pipe;
  111. int retval;
  112. /* Set up the endpoint information */
  113. /* This device only has an interrupt endpoint */
  114. retval = usb_find_common_endpoints(pxrc->intf->cur_altsetting,
  115. NULL, NULL, &epirq, NULL);
  116. if (retval) {
  117. dev_err(&pxrc->intf->dev,
  118. "Could not find endpoint\n");
  119. goto error;
  120. }
  121. pxrc->bsize = usb_endpoint_maxp(epirq);
  122. pxrc->epaddr = epirq->bEndpointAddress;
  123. pxrc->data = devm_kmalloc(&pxrc->intf->dev, pxrc->bsize, GFP_KERNEL);
  124. if (!pxrc->data) {
  125. retval = -ENOMEM;
  126. goto error;
  127. }
  128. usb_set_intfdata(pxrc->intf, pxrc);
  129. usb_make_path(pxrc->udev, pxrc->phys, sizeof(pxrc->phys));
  130. strlcat(pxrc->phys, "/input0", sizeof(pxrc->phys));
  131. pxrc->urb = usb_alloc_urb(0, GFP_KERNEL);
  132. if (!pxrc->urb) {
  133. retval = -ENOMEM;
  134. goto error;
  135. }
  136. pipe = usb_rcvintpipe(pxrc->udev, pxrc->epaddr),
  137. usb_fill_int_urb(pxrc->urb, pxrc->udev, pipe, pxrc->data, pxrc->bsize,
  138. pxrc_usb_irq, pxrc, 1);
  139. error:
  140. return retval;
  141. }
  142. static int pxrc_input_init(struct pxrc *pxrc)
  143. {
  144. pxrc->input = devm_input_allocate_device(&pxrc->intf->dev);
  145. if (pxrc->input == NULL) {
  146. dev_err(&pxrc->intf->dev, "couldn't allocate input device\n");
  147. return -ENOMEM;
  148. }
  149. pxrc->input->name = "PXRC Flight Controller Adapter";
  150. pxrc->input->phys = pxrc->phys;
  151. usb_to_input_id(pxrc->udev, &pxrc->input->id);
  152. pxrc->input->open = pxrc_open;
  153. pxrc->input->close = pxrc_close;
  154. input_set_capability(pxrc->input, EV_KEY, BTN_A);
  155. input_set_abs_params(pxrc->input, ABS_X, 0, 255, 0, 0);
  156. input_set_abs_params(pxrc->input, ABS_Y, 0, 255, 0, 0);
  157. input_set_abs_params(pxrc->input, ABS_RX, 0, 255, 0, 0);
  158. input_set_abs_params(pxrc->input, ABS_RY, 0, 255, 0, 0);
  159. input_set_abs_params(pxrc->input, ABS_RUDDER, 0, 255, 0, 0);
  160. input_set_abs_params(pxrc->input, ABS_THROTTLE, 0, 255, 0, 0);
  161. input_set_abs_params(pxrc->input, ABS_MISC, 0, 255, 0, 0);
  162. input_set_drvdata(pxrc->input, pxrc);
  163. return input_register_device(pxrc->input);
  164. }
  165. static int pxrc_probe(struct usb_interface *intf,
  166. const struct usb_device_id *id)
  167. {
  168. struct pxrc *pxrc;
  169. int retval;
  170. pxrc = devm_kzalloc(&intf->dev, sizeof(*pxrc), GFP_KERNEL);
  171. if (!pxrc)
  172. return -ENOMEM;
  173. mutex_init(&pxrc->pm_mutex);
  174. pxrc->udev = usb_get_dev(interface_to_usbdev(intf));
  175. pxrc->intf = intf;
  176. retval = pxrc_usb_init(pxrc);
  177. if (retval)
  178. goto error;
  179. retval = pxrc_input_init(pxrc);
  180. if (retval)
  181. goto err_free_urb;
  182. return 0;
  183. err_free_urb:
  184. usb_free_urb(pxrc->urb);
  185. error:
  186. return retval;
  187. }
  188. static void pxrc_disconnect(struct usb_interface *intf)
  189. {
  190. struct pxrc *pxrc = usb_get_intfdata(intf);
  191. usb_free_urb(pxrc->urb);
  192. usb_set_intfdata(intf, NULL);
  193. }
  194. static int pxrc_suspend(struct usb_interface *intf, pm_message_t message)
  195. {
  196. struct pxrc *pxrc = usb_get_intfdata(intf);
  197. mutex_lock(&pxrc->pm_mutex);
  198. if (pxrc->is_open)
  199. usb_kill_urb(pxrc->urb);
  200. mutex_unlock(&pxrc->pm_mutex);
  201. return 0;
  202. }
  203. static int pxrc_resume(struct usb_interface *intf)
  204. {
  205. struct pxrc *pxrc = usb_get_intfdata(intf);
  206. int retval = 0;
  207. mutex_lock(&pxrc->pm_mutex);
  208. if (pxrc->is_open && usb_submit_urb(pxrc->urb, GFP_KERNEL) < 0)
  209. retval = -EIO;
  210. mutex_unlock(&pxrc->pm_mutex);
  211. return retval;
  212. }
  213. static int pxrc_pre_reset(struct usb_interface *intf)
  214. {
  215. struct pxrc *pxrc = usb_get_intfdata(intf);
  216. mutex_lock(&pxrc->pm_mutex);
  217. usb_kill_urb(pxrc->urb);
  218. return 0;
  219. }
  220. static int pxrc_post_reset(struct usb_interface *intf)
  221. {
  222. struct pxrc *pxrc = usb_get_intfdata(intf);
  223. int retval = 0;
  224. if (pxrc->is_open && usb_submit_urb(pxrc->urb, GFP_KERNEL) < 0)
  225. retval = -EIO;
  226. mutex_unlock(&pxrc->pm_mutex);
  227. return retval;
  228. }
  229. static int pxrc_reset_resume(struct usb_interface *intf)
  230. {
  231. return pxrc_resume(intf);
  232. }
  233. static struct usb_driver pxrc_driver = {
  234. .name = "pxrc",
  235. .probe = pxrc_probe,
  236. .disconnect = pxrc_disconnect,
  237. .id_table = pxrc_table,
  238. .suspend = pxrc_suspend,
  239. .resume = pxrc_resume,
  240. .pre_reset = pxrc_pre_reset,
  241. .post_reset = pxrc_post_reset,
  242. .reset_resume = pxrc_reset_resume,
  243. };
  244. module_usb_driver(pxrc_driver);
  245. MODULE_AUTHOR("Marcus Folkesson <marcus.folkesson@gmail.com>");
  246. MODULE_DESCRIPTION("PhoenixRC Flight Controller Adapter");
  247. MODULE_LICENSE("GPL v2");