Path: blob/master/tools/usb/usbip/libsrc/usbip_device_driver.c
26288 views
// SPDX-License-Identifier: GPL-2.0-or-later1/*2* Copyright (C) 2015 Karol Kosik <[email protected]>3* 2015 Samsung Electronics4* Author: Igor Kotrasinski <[email protected]>5*6* Based on tools/usb/usbip/libsrc/usbip_host_driver.c, which is:7* Copyright (C) 2011 matt mooney <[email protected]>8* 2005-2007 Takahiro Hirofuchi9*/1011#include <fcntl.h>12#include <string.h>13#include <linux/usb/ch9.h>1415#include <unistd.h>1617#include "usbip_host_common.h"18#include "usbip_device_driver.h"1920#undef PROGNAME21#define PROGNAME "libusbip"2223#define copy_descr_attr16(dev, descr, attr) \24((dev)->attr = le16toh((descr)->attr)) \2526#define copy_descr_attr(dev, descr, attr) \27((dev)->attr = (descr)->attr) \2829#define ARRAY_SIZE(arr) (sizeof(arr) / sizeof((arr)[0]))3031static struct {32enum usb_device_speed speed;33const char *name;34} speed_names[] = {35{36.speed = USB_SPEED_UNKNOWN,37.name = "UNKNOWN",38},39{40.speed = USB_SPEED_LOW,41.name = "low-speed",42},43{44.speed = USB_SPEED_FULL,45.name = "full-speed",46},47{48.speed = USB_SPEED_HIGH,49.name = "high-speed",50},51{52.speed = USB_SPEED_WIRELESS,53.name = "wireless",54},55{56.speed = USB_SPEED_SUPER,57.name = "super-speed",58},59};6061static62int read_usb_vudc_device(struct udev_device *sdev, struct usbip_usb_device *dev)63{64const char *path, *name;65char filepath[SYSFS_PATH_MAX];66struct usb_device_descriptor descr;67unsigned int i;68FILE *fd = NULL;69struct udev_device *plat;70const char *speed;71size_t ret;7273plat = udev_device_get_parent(sdev);74path = udev_device_get_syspath(plat);75snprintf(filepath, SYSFS_PATH_MAX, "%s/%s",76path, VUDC_DEVICE_DESCR_FILE);77fd = fopen(filepath, "r");78if (!fd)79return -1;80ret = fread((char *) &descr, sizeof(descr), 1, fd);81if (ret != 1) {82err("Cannot read vudc device descr file: %s", strerror(errno));83goto err;84}85fclose(fd);8687copy_descr_attr(dev, &descr, bDeviceClass);88copy_descr_attr(dev, &descr, bDeviceSubClass);89copy_descr_attr(dev, &descr, bDeviceProtocol);90copy_descr_attr(dev, &descr, bNumConfigurations);91copy_descr_attr16(dev, &descr, idVendor);92copy_descr_attr16(dev, &descr, idProduct);93copy_descr_attr16(dev, &descr, bcdDevice);9495strncpy(dev->path, path, SYSFS_PATH_MAX - 1);96dev->path[SYSFS_PATH_MAX - 1] = '\0';9798dev->speed = USB_SPEED_UNKNOWN;99speed = udev_device_get_sysattr_value(sdev, "current_speed");100if (speed) {101for (i = 0; i < ARRAY_SIZE(speed_names); i++) {102if (!strcmp(speed_names[i].name, speed)) {103dev->speed = speed_names[i].speed;104break;105}106}107}108109/* Only used for user output, little sense to output them in general */110dev->bNumInterfaces = 0;111dev->bConfigurationValue = 0;112dev->busnum = 0;113114name = udev_device_get_sysname(plat);115strncpy(dev->busid, name, SYSFS_BUS_ID_SIZE - 1);116dev->busid[SYSFS_BUS_ID_SIZE - 1] = '\0';117return 0;118err:119fclose(fd);120return -1;121}122123static int is_my_device(struct udev_device *dev)124{125const char *driver;126127driver = udev_device_get_property_value(dev, "USB_UDC_NAME");128return driver != NULL && !strcmp(driver, USBIP_DEVICE_DRV_NAME);129}130131static int usbip_device_driver_open(struct usbip_host_driver *hdriver)132{133int ret;134135hdriver->ndevs = 0;136INIT_LIST_HEAD(&hdriver->edev_list);137138ret = usbip_generic_driver_open(hdriver);139if (ret)140err("please load " USBIP_CORE_MOD_NAME ".ko and "141USBIP_DEVICE_DRV_NAME ".ko!");142143return ret;144}145146struct usbip_host_driver device_driver = {147.edev_list = LIST_HEAD_INIT(device_driver.edev_list),148.udev_subsystem = "udc",149.ops = {150.open = usbip_device_driver_open,151.close = usbip_generic_driver_close,152.refresh_device_list = usbip_generic_refresh_device_list,153.get_device = usbip_generic_get_device,154.read_device = read_usb_vudc_device,155.is_my_device = is_my_device,156},157};158159160