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
);
83 err("Cannot read vudc device descr file: %s", strerror(errno
));
88 copy_descr_attr(dev
, &descr
, bDeviceClass
);
89 copy_descr_attr(dev
, &descr
, bDeviceSubClass
);
90 copy_descr_attr(dev
, &descr
, bDeviceProtocol
);
91 copy_descr_attr(dev
, &descr
, bNumConfigurations
);
92 copy_descr_attr16(dev
, &descr
, idVendor
);
93 copy_descr_attr16(dev
, &descr
, idProduct
);
94 copy_descr_attr16(dev
, &descr
, bcdDevice
);
96 strncpy(dev
->path
, path
, SYSFS_PATH_MAX
- 1);
97 dev
->path
[SYSFS_PATH_MAX
- 1] = '\0';
99 dev
->speed
= USB_SPEED_UNKNOWN
;
100 speed
= udev_device_get_sysattr_value(sdev
, "current_speed");
102 for (i
= 0; i
< ARRAY_SIZE(speed_names
); i
++) {
103 if (!strcmp(speed_names
[i
].name
, speed
)) {
104 dev
->speed
= speed_names
[i
].speed
;
110 /* Only used for user output, little sense to output them in general */
111 dev
->bNumInterfaces
= 0;
112 dev
->bConfigurationValue
= 0;
115 name
= udev_device_get_sysname(plat
);
116 strncpy(dev
->busid
, name
, SYSFS_BUS_ID_SIZE
- 1);
117 dev
->busid
[SYSFS_BUS_ID_SIZE
- 1] = '\0';
124 static int is_my_device(struct udev_device
*dev
)
128 driver
= udev_device_get_property_value(dev
, "USB_UDC_NAME");
129 return driver
!= NULL
&& !strcmp(driver
, USBIP_DEVICE_DRV_NAME
);
132 static int usbip_device_driver_open(struct usbip_host_driver
*hdriver
)
137 INIT_LIST_HEAD(&hdriver
->edev_list
);
139 ret
= usbip_generic_driver_open(hdriver
);
141 err("please load " USBIP_CORE_MOD_NAME
".ko and "
142 USBIP_DEVICE_DRV_NAME
".ko!");
147 struct usbip_host_driver device_driver
= {
148 .edev_list
= LIST_HEAD_INIT(device_driver
.edev_list
),
149 .udev_subsystem
= "udc",
151 .open
= usbip_device_driver_open
,
152 .close
= usbip_generic_driver_close
,
153 .refresh_device_list
= usbip_generic_refresh_device_list
,
154 .get_device
= usbip_generic_get_device
,
155 .read_device
= read_usb_vudc_device
,
156 .is_my_device
= is_my_device
,