usbip_device_driver.c 3.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158
  1. // SPDX-License-Identifier: GPL-2.0-or-later
  2. /*
  3. * Copyright (C) 2015 Karol Kosik <karo9@interia.eu>
  4. * 2015 Samsung Electronics
  5. * Author: Igor Kotrasinski <i.kotrasinsk@samsung.com>
  6. *
  7. * Based on tools/usb/usbip/libsrc/usbip_host_driver.c, which is:
  8. * Copyright (C) 2011 matt mooney <mfm@muteddisk.com>
  9. * 2005-2007 Takahiro Hirofuchi
  10. */
  11. #include <fcntl.h>
  12. #include <string.h>
  13. #include <linux/usb/ch9.h>
  14. #include <unistd.h>
  15. #include "usbip_host_common.h"
  16. #include "usbip_device_driver.h"
  17. #undef PROGNAME
  18. #define PROGNAME "libusbip"
  19. #define copy_descr_attr16(dev, descr, attr) \
  20. ((dev)->attr = le16toh((descr)->attr)) \
  21. #define copy_descr_attr(dev, descr, attr) \
  22. ((dev)->attr = (descr)->attr) \
  23. #define ARRAY_SIZE(arr) (sizeof(arr) / sizeof((arr)[0]))
  24. static struct {
  25. enum usb_device_speed speed;
  26. const char *name;
  27. } speed_names[] = {
  28. {
  29. .speed = USB_SPEED_UNKNOWN,
  30. .name = "UNKNOWN",
  31. },
  32. {
  33. .speed = USB_SPEED_LOW,
  34. .name = "low-speed",
  35. },
  36. {
  37. .speed = USB_SPEED_FULL,
  38. .name = "full-speed",
  39. },
  40. {
  41. .speed = USB_SPEED_HIGH,
  42. .name = "high-speed",
  43. },
  44. {
  45. .speed = USB_SPEED_WIRELESS,
  46. .name = "wireless",
  47. },
  48. {
  49. .speed = USB_SPEED_SUPER,
  50. .name = "super-speed",
  51. },
  52. };
  53. static
  54. int read_usb_vudc_device(struct udev_device *sdev, struct usbip_usb_device *dev)
  55. {
  56. const char *path, *name;
  57. char filepath[SYSFS_PATH_MAX];
  58. struct usb_device_descriptor descr;
  59. unsigned int i;
  60. FILE *fd = NULL;
  61. struct udev_device *plat;
  62. const char *speed;
  63. size_t ret;
  64. plat = udev_device_get_parent(sdev);
  65. path = udev_device_get_syspath(plat);
  66. snprintf(filepath, SYSFS_PATH_MAX, "%s/%s",
  67. path, VUDC_DEVICE_DESCR_FILE);
  68. fd = fopen(filepath, "r");
  69. if (!fd)
  70. return -1;
  71. ret = fread((char *) &descr, sizeof(descr), 1, fd);
  72. if (ret != 1) {
  73. err("Cannot read vudc device descr file: %s", strerror(errno));
  74. goto err;
  75. }
  76. fclose(fd);
  77. copy_descr_attr(dev, &descr, bDeviceClass);
  78. copy_descr_attr(dev, &descr, bDeviceSubClass);
  79. copy_descr_attr(dev, &descr, bDeviceProtocol);
  80. copy_descr_attr(dev, &descr, bNumConfigurations);
  81. copy_descr_attr16(dev, &descr, idVendor);
  82. copy_descr_attr16(dev, &descr, idProduct);
  83. copy_descr_attr16(dev, &descr, bcdDevice);
  84. strncpy(dev->path, path, SYSFS_PATH_MAX - 1);
  85. dev->path[SYSFS_PATH_MAX - 1] = '\0';
  86. dev->speed = USB_SPEED_UNKNOWN;
  87. speed = udev_device_get_sysattr_value(sdev, "current_speed");
  88. if (speed) {
  89. for (i = 0; i < ARRAY_SIZE(speed_names); i++) {
  90. if (!strcmp(speed_names[i].name, speed)) {
  91. dev->speed = speed_names[i].speed;
  92. break;
  93. }
  94. }
  95. }
  96. /* Only used for user output, little sense to output them in general */
  97. dev->bNumInterfaces = 0;
  98. dev->bConfigurationValue = 0;
  99. dev->busnum = 0;
  100. name = udev_device_get_sysname(plat);
  101. strncpy(dev->busid, name, SYSFS_BUS_ID_SIZE - 1);
  102. dev->busid[SYSFS_BUS_ID_SIZE - 1] = '\0';
  103. return 0;
  104. err:
  105. fclose(fd);
  106. return -1;
  107. }
  108. static int is_my_device(struct udev_device *dev)
  109. {
  110. const char *driver;
  111. driver = udev_device_get_property_value(dev, "USB_UDC_NAME");
  112. return driver != NULL && !strcmp(driver, USBIP_DEVICE_DRV_NAME);
  113. }
  114. static int usbip_device_driver_open(struct usbip_host_driver *hdriver)
  115. {
  116. int ret;
  117. hdriver->ndevs = 0;
  118. INIT_LIST_HEAD(&hdriver->edev_list);
  119. ret = usbip_generic_driver_open(hdriver);
  120. if (ret)
  121. err("please load " USBIP_CORE_MOD_NAME ".ko and "
  122. USBIP_DEVICE_DRV_NAME ".ko!");
  123. return ret;
  124. }
  125. struct usbip_host_driver device_driver = {
  126. .edev_list = LIST_HEAD_INIT(device_driver.edev_list),
  127. .udev_subsystem = "udc",
  128. .ops = {
  129. .open = usbip_device_driver_open,
  130. .close = usbip_generic_driver_close,
  131. .refresh_device_list = usbip_generic_refresh_device_list,
  132. .get_device = usbip_generic_get_device,
  133. .read_device = read_usb_vudc_device,
  134. .is_my_device = is_my_device,
  135. },
  136. };