162306a36Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0-or-later
262306a36Sopenharmony_ci/*
362306a36Sopenharmony_ci * Copyright (C) 2015 Karol Kosik <karo9@interia.eu>
462306a36Sopenharmony_ci *		 2015 Samsung Electronics
562306a36Sopenharmony_ci * Author:	 Igor Kotrasinski <i.kotrasinsk@samsung.com>
662306a36Sopenharmony_ci *
762306a36Sopenharmony_ci * Based on tools/usb/usbip/libsrc/usbip_host_driver.c, which is:
862306a36Sopenharmony_ci * Copyright (C) 2011 matt mooney <mfm@muteddisk.com>
962306a36Sopenharmony_ci *               2005-2007 Takahiro Hirofuchi
1062306a36Sopenharmony_ci */
1162306a36Sopenharmony_ci
1262306a36Sopenharmony_ci#include <fcntl.h>
1362306a36Sopenharmony_ci#include <string.h>
1462306a36Sopenharmony_ci#include <linux/usb/ch9.h>
1562306a36Sopenharmony_ci
1662306a36Sopenharmony_ci#include <unistd.h>
1762306a36Sopenharmony_ci
1862306a36Sopenharmony_ci#include "usbip_host_common.h"
1962306a36Sopenharmony_ci#include "usbip_device_driver.h"
2062306a36Sopenharmony_ci
2162306a36Sopenharmony_ci#undef  PROGNAME
2262306a36Sopenharmony_ci#define PROGNAME "libusbip"
2362306a36Sopenharmony_ci
2462306a36Sopenharmony_ci#define copy_descr_attr16(dev, descr, attr)			\
2562306a36Sopenharmony_ci		((dev)->attr = le16toh((descr)->attr))		\
2662306a36Sopenharmony_ci
2762306a36Sopenharmony_ci#define copy_descr_attr(dev, descr, attr)			\
2862306a36Sopenharmony_ci		((dev)->attr = (descr)->attr)			\
2962306a36Sopenharmony_ci
3062306a36Sopenharmony_ci#define ARRAY_SIZE(arr) (sizeof(arr) / sizeof((arr)[0]))
3162306a36Sopenharmony_ci
3262306a36Sopenharmony_cistatic struct {
3362306a36Sopenharmony_ci	enum usb_device_speed speed;
3462306a36Sopenharmony_ci	const char *name;
3562306a36Sopenharmony_ci} speed_names[] = {
3662306a36Sopenharmony_ci	{
3762306a36Sopenharmony_ci		.speed = USB_SPEED_UNKNOWN,
3862306a36Sopenharmony_ci		.name = "UNKNOWN",
3962306a36Sopenharmony_ci	},
4062306a36Sopenharmony_ci	{
4162306a36Sopenharmony_ci		.speed = USB_SPEED_LOW,
4262306a36Sopenharmony_ci		.name = "low-speed",
4362306a36Sopenharmony_ci	},
4462306a36Sopenharmony_ci	{
4562306a36Sopenharmony_ci		.speed = USB_SPEED_FULL,
4662306a36Sopenharmony_ci		.name = "full-speed",
4762306a36Sopenharmony_ci	},
4862306a36Sopenharmony_ci	{
4962306a36Sopenharmony_ci		.speed = USB_SPEED_HIGH,
5062306a36Sopenharmony_ci		.name = "high-speed",
5162306a36Sopenharmony_ci	},
5262306a36Sopenharmony_ci	{
5362306a36Sopenharmony_ci		.speed = USB_SPEED_WIRELESS,
5462306a36Sopenharmony_ci		.name = "wireless",
5562306a36Sopenharmony_ci	},
5662306a36Sopenharmony_ci	{
5762306a36Sopenharmony_ci		.speed = USB_SPEED_SUPER,
5862306a36Sopenharmony_ci		.name = "super-speed",
5962306a36Sopenharmony_ci	},
6062306a36Sopenharmony_ci};
6162306a36Sopenharmony_ci
6262306a36Sopenharmony_cistatic
6362306a36Sopenharmony_ciint read_usb_vudc_device(struct udev_device *sdev, struct usbip_usb_device *dev)
6462306a36Sopenharmony_ci{
6562306a36Sopenharmony_ci	const char *path, *name;
6662306a36Sopenharmony_ci	char filepath[SYSFS_PATH_MAX];
6762306a36Sopenharmony_ci	struct usb_device_descriptor descr;
6862306a36Sopenharmony_ci	unsigned int i;
6962306a36Sopenharmony_ci	FILE *fd = NULL;
7062306a36Sopenharmony_ci	struct udev_device *plat;
7162306a36Sopenharmony_ci	const char *speed;
7262306a36Sopenharmony_ci	size_t ret;
7362306a36Sopenharmony_ci
7462306a36Sopenharmony_ci	plat = udev_device_get_parent(sdev);
7562306a36Sopenharmony_ci	path = udev_device_get_syspath(plat);
7662306a36Sopenharmony_ci	snprintf(filepath, SYSFS_PATH_MAX, "%s/%s",
7762306a36Sopenharmony_ci		 path, VUDC_DEVICE_DESCR_FILE);
7862306a36Sopenharmony_ci	fd = fopen(filepath, "r");
7962306a36Sopenharmony_ci	if (!fd)
8062306a36Sopenharmony_ci		return -1;
8162306a36Sopenharmony_ci	ret = fread((char *) &descr, sizeof(descr), 1, fd);
8262306a36Sopenharmony_ci	if (ret != 1) {
8362306a36Sopenharmony_ci		err("Cannot read vudc device descr file: %s", strerror(errno));
8462306a36Sopenharmony_ci		goto err;
8562306a36Sopenharmony_ci	}
8662306a36Sopenharmony_ci	fclose(fd);
8762306a36Sopenharmony_ci
8862306a36Sopenharmony_ci	copy_descr_attr(dev, &descr, bDeviceClass);
8962306a36Sopenharmony_ci	copy_descr_attr(dev, &descr, bDeviceSubClass);
9062306a36Sopenharmony_ci	copy_descr_attr(dev, &descr, bDeviceProtocol);
9162306a36Sopenharmony_ci	copy_descr_attr(dev, &descr, bNumConfigurations);
9262306a36Sopenharmony_ci	copy_descr_attr16(dev, &descr, idVendor);
9362306a36Sopenharmony_ci	copy_descr_attr16(dev, &descr, idProduct);
9462306a36Sopenharmony_ci	copy_descr_attr16(dev, &descr, bcdDevice);
9562306a36Sopenharmony_ci
9662306a36Sopenharmony_ci	strncpy(dev->path, path, SYSFS_PATH_MAX - 1);
9762306a36Sopenharmony_ci	dev->path[SYSFS_PATH_MAX - 1] = '\0';
9862306a36Sopenharmony_ci
9962306a36Sopenharmony_ci	dev->speed = USB_SPEED_UNKNOWN;
10062306a36Sopenharmony_ci	speed = udev_device_get_sysattr_value(sdev, "current_speed");
10162306a36Sopenharmony_ci	if (speed) {
10262306a36Sopenharmony_ci		for (i = 0; i < ARRAY_SIZE(speed_names); i++) {
10362306a36Sopenharmony_ci			if (!strcmp(speed_names[i].name, speed)) {
10462306a36Sopenharmony_ci				dev->speed = speed_names[i].speed;
10562306a36Sopenharmony_ci				break;
10662306a36Sopenharmony_ci			}
10762306a36Sopenharmony_ci		}
10862306a36Sopenharmony_ci	}
10962306a36Sopenharmony_ci
11062306a36Sopenharmony_ci	/* Only used for user output, little sense to output them in general */
11162306a36Sopenharmony_ci	dev->bNumInterfaces = 0;
11262306a36Sopenharmony_ci	dev->bConfigurationValue = 0;
11362306a36Sopenharmony_ci	dev->busnum = 0;
11462306a36Sopenharmony_ci
11562306a36Sopenharmony_ci	name = udev_device_get_sysname(plat);
11662306a36Sopenharmony_ci	strncpy(dev->busid, name, SYSFS_BUS_ID_SIZE - 1);
11762306a36Sopenharmony_ci	dev->busid[SYSFS_BUS_ID_SIZE - 1] = '\0';
11862306a36Sopenharmony_ci	return 0;
11962306a36Sopenharmony_cierr:
12062306a36Sopenharmony_ci	fclose(fd);
12162306a36Sopenharmony_ci	return -1;
12262306a36Sopenharmony_ci}
12362306a36Sopenharmony_ci
12462306a36Sopenharmony_cistatic int is_my_device(struct udev_device *dev)
12562306a36Sopenharmony_ci{
12662306a36Sopenharmony_ci	const char *driver;
12762306a36Sopenharmony_ci
12862306a36Sopenharmony_ci	driver = udev_device_get_property_value(dev, "USB_UDC_NAME");
12962306a36Sopenharmony_ci	return driver != NULL && !strcmp(driver, USBIP_DEVICE_DRV_NAME);
13062306a36Sopenharmony_ci}
13162306a36Sopenharmony_ci
13262306a36Sopenharmony_cistatic int usbip_device_driver_open(struct usbip_host_driver *hdriver)
13362306a36Sopenharmony_ci{
13462306a36Sopenharmony_ci	int ret;
13562306a36Sopenharmony_ci
13662306a36Sopenharmony_ci	hdriver->ndevs = 0;
13762306a36Sopenharmony_ci	INIT_LIST_HEAD(&hdriver->edev_list);
13862306a36Sopenharmony_ci
13962306a36Sopenharmony_ci	ret = usbip_generic_driver_open(hdriver);
14062306a36Sopenharmony_ci	if (ret)
14162306a36Sopenharmony_ci		err("please load " USBIP_CORE_MOD_NAME ".ko and "
14262306a36Sopenharmony_ci		    USBIP_DEVICE_DRV_NAME ".ko!");
14362306a36Sopenharmony_ci
14462306a36Sopenharmony_ci	return ret;
14562306a36Sopenharmony_ci}
14662306a36Sopenharmony_ci
14762306a36Sopenharmony_cistruct usbip_host_driver device_driver = {
14862306a36Sopenharmony_ci	.edev_list = LIST_HEAD_INIT(device_driver.edev_list),
14962306a36Sopenharmony_ci	.udev_subsystem = "udc",
15062306a36Sopenharmony_ci	.ops = {
15162306a36Sopenharmony_ci		.open = usbip_device_driver_open,
15262306a36Sopenharmony_ci		.close = usbip_generic_driver_close,
15362306a36Sopenharmony_ci		.refresh_device_list = usbip_generic_refresh_device_list,
15462306a36Sopenharmony_ci		.get_device = usbip_generic_get_device,
15562306a36Sopenharmony_ci		.read_device = read_usb_vudc_device,
15662306a36Sopenharmony_ci		.is_my_device = is_my_device,
15762306a36Sopenharmony_ci	},
15862306a36Sopenharmony_ci};
159