1 // SPDX-License-Identifier: GPL-2.0-or-later
3 * Copyright (C) 2015 Karol Kosik <karo9@interia.eu>
4 * 2015 Samsung Electronics
5 * Author: Igor Kotrasinski <i.kotrasinsk@samsung.com>
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
14 #include <linux/usb/ch9.h>
18 #include "usbip_host_common.h"
19 #include "usbip_device_driver.h"
22 #define PROGNAME "libusbip"
24 #define copy_descr_attr16(dev, descr, attr) \
25 ((dev)->attr = le16toh((descr)->attr)) \
27 #define copy_descr_attr(dev, descr, attr) \
28 ((dev)->attr = (descr)->attr) \
30 #define ARRAY_SIZE(arr) (sizeof(arr) / sizeof((arr)[0]))
33 enum usb_device_speed speed;
37 .speed = USB_SPEED_UNKNOWN,
41 .speed = USB_SPEED_LOW,
45 .speed = USB_SPEED_FULL,
49 .speed = USB_SPEED_HIGH,
53 .speed = USB_SPEED_WIRELESS,
57 .speed = USB_SPEED_SUPER,
58 .name = "super-speed",
63 int read_usb_vudc_device(struct udev_device *sdev, struct usbip_usb_device *dev)
65 const char *path, *name;
66 char filepath[SYSFS_PATH_MAX];
67 struct usb_device_descriptor descr;
70 struct udev_device *plat;
74 plat = udev_device_get_parent(sdev);
75 path = udev_device_get_syspath(plat);
76 snprintf(filepath, SYSFS_PATH_MAX, "%s/%s",
77 path, VUDC_DEVICE_DESCR_FILE);
78 fd = fopen(filepath, "r");
81 ret = fread((char *) &descr, sizeof(descr), 1, fd);
86 copy_descr_attr(dev, &descr, bDeviceClass);
87 copy_descr_attr(dev, &descr, bDeviceSubClass);
88 copy_descr_attr(dev, &descr, bDeviceProtocol);
89 copy_descr_attr(dev, &descr, bNumConfigurations);
90 copy_descr_attr16(dev, &descr, idVendor);
91 copy_descr_attr16(dev, &descr, idProduct);
92 copy_descr_attr16(dev, &descr, bcdDevice);
94 strncpy(dev->path, path, SYSFS_PATH_MAX - 1);
95 dev->path[SYSFS_PATH_MAX - 1] = '\0';
97 dev->speed = USB_SPEED_UNKNOWN;
98 speed = udev_device_get_sysattr_value(sdev, "current_speed");
100 for (i = 0; i < ARRAY_SIZE(speed_names); i++) {
101 if (!strcmp(speed_names[i].name, speed)) {
102 dev->speed = speed_names[i].speed;
108 /* Only used for user output, little sense to output them in general */
109 dev->bNumInterfaces = 0;
110 dev->bConfigurationValue = 0;
113 name = udev_device_get_sysname(plat);
114 strncpy(dev->busid, name, SYSFS_BUS_ID_SIZE - 1);
115 dev->busid[SYSFS_BUS_ID_SIZE - 1] = '\0';
122 static int is_my_device(struct udev_device *dev)
126 driver = udev_device_get_property_value(dev, "USB_UDC_NAME");
127 return driver != NULL && !strcmp(driver, USBIP_DEVICE_DRV_NAME);
130 static int usbip_device_driver_open(struct usbip_host_driver *hdriver)
135 INIT_LIST_HEAD(&hdriver->edev_list);
137 ret = usbip_generic_driver_open(hdriver);
139 err("please load " USBIP_CORE_MOD_NAME ".ko and "
140 USBIP_DEVICE_DRV_NAME ".ko!");
145 struct usbip_host_driver device_driver = {
146 .edev_list = LIST_HEAD_INIT(device_driver.edev_list),
147 .udev_subsystem = "udc",
149 .open = usbip_device_driver_open,
150 .close = usbip_generic_driver_close,
151 .refresh_device_list = usbip_generic_refresh_device_list,
152 .get_device = usbip_generic_get_device,
153 .read_device = read_usb_vudc_device,
154 .is_my_device = is_my_device,