xref: /linux/drivers/media/usb/go7007/go7007-loader.c (revision 75bf465f0bc33e9b776a46d6a1b9b990f5fb7c37)
1*1802d0beSThomas Gleixner // SPDX-License-Identifier: GPL-2.0-only
2b11869dbSPete Eberlein /*
3b11869dbSPete Eberlein  * Copyright (C) 2008 Sensoray Company Inc.
4b11869dbSPete Eberlein  */
5b11869dbSPete Eberlein 
6b11869dbSPete Eberlein #include <linux/module.h>
75a0e3ad6STejun Heo #include <linux/slab.h>
8b11869dbSPete Eberlein #include <linux/usb.h>
9c046981fSHans Verkuil #include <linux/firmware.h>
10c046981fSHans Verkuil #include <cypress_firmware.h>
11b11869dbSPete Eberlein 
120ee3d4d5SHans Verkuil struct fw_config {
130ee3d4d5SHans Verkuil 	u16 vendor;
140ee3d4d5SHans Verkuil 	u16 product;
150ee3d4d5SHans Verkuil 	const char * const fw_name1;
160ee3d4d5SHans Verkuil 	const char * const fw_name2;
170ee3d4d5SHans Verkuil };
18b11869dbSPete Eberlein 
1904ad3a64SEbru Akagunduz static struct fw_config fw_configs[] = {
200ee3d4d5SHans Verkuil 	{ 0x1943, 0xa250, "go7007/s2250-1.fw", "go7007/s2250-2.fw" },
210ee3d4d5SHans Verkuil 	{ 0x093b, 0xa002, "go7007/px-m402u.fw", NULL },
220ee3d4d5SHans Verkuil 	{ 0x093b, 0xa004, "go7007/px-tv402u.fw", NULL },
230ee3d4d5SHans Verkuil 	{ 0x0eb1, 0x6666, "go7007/lr192.fw", NULL },
240ee3d4d5SHans Verkuil 	{ 0x0eb1, 0x6668, "go7007/wis-startrek.fw", NULL },
250ee3d4d5SHans Verkuil 	{ 0, 0, NULL, NULL }
260ee3d4d5SHans Verkuil };
270ee3d4d5SHans Verkuil MODULE_FIRMWARE("go7007/s2250-1.fw");
280ee3d4d5SHans Verkuil MODULE_FIRMWARE("go7007/s2250-2.fw");
290ee3d4d5SHans Verkuil MODULE_FIRMWARE("go7007/px-m402u.fw");
300ee3d4d5SHans Verkuil MODULE_FIRMWARE("go7007/px-tv402u.fw");
310ee3d4d5SHans Verkuil MODULE_FIRMWARE("go7007/lr192.fw");
320ee3d4d5SHans Verkuil MODULE_FIRMWARE("go7007/wis-startrek.fw");
33b11869dbSPete Eberlein 
34e129c974SHans Verkuil static int go7007_loader_probe(struct usb_interface *interface,
35b11869dbSPete Eberlein 				const struct usb_device_id *id)
36b11869dbSPete Eberlein {
37b11869dbSPete Eberlein 	struct usb_device *usbdev;
38b11869dbSPete Eberlein 	const struct firmware *fw;
390ee3d4d5SHans Verkuil 	u16 vendor, product;
400ee3d4d5SHans Verkuil 	const char *fw1, *fw2;
410ee3d4d5SHans Verkuil 	int ret;
420ee3d4d5SHans Verkuil 	int i;
43b11869dbSPete Eberlein 
44b11869dbSPete Eberlein 	usbdev = usb_get_dev(interface_to_usbdev(interface));
450ee3d4d5SHans Verkuil 	if (!usbdev)
460ee3d4d5SHans Verkuil 		goto failed2;
47b11869dbSPete Eberlein 
48b11869dbSPete Eberlein 	if (usbdev->descriptor.bNumConfigurations != 1) {
49b11558a3SYAMANE Toshiaki 		dev_err(&interface->dev, "can't handle multiple config\n");
5050c88544SAlexey Khoroshilov 		goto failed2;
51b11869dbSPete Eberlein 	}
52b11869dbSPete Eberlein 
530ee3d4d5SHans Verkuil 	vendor = le16_to_cpu(usbdev->descriptor.idVendor);
540ee3d4d5SHans Verkuil 	product = le16_to_cpu(usbdev->descriptor.idProduct);
550ee3d4d5SHans Verkuil 
560ee3d4d5SHans Verkuil 	for (i = 0; fw_configs[i].fw_name1; i++)
570ee3d4d5SHans Verkuil 		if (fw_configs[i].vendor == vendor &&
580ee3d4d5SHans Verkuil 		    fw_configs[i].product == product)
59b11869dbSPete Eberlein 			break;
60b11869dbSPete Eberlein 
610ee3d4d5SHans Verkuil 	/* Should never happen */
620ee3d4d5SHans Verkuil 	if (fw_configs[i].fw_name1 == NULL)
630ee3d4d5SHans Verkuil 		goto failed2;
64b11869dbSPete Eberlein 
650ee3d4d5SHans Verkuil 	fw1 = fw_configs[i].fw_name1;
660ee3d4d5SHans Verkuil 	fw2 = fw_configs[i].fw_name2;
67535ec049SJoe Perches 
680ee3d4d5SHans Verkuil 	dev_info(&interface->dev, "loading firmware %s\n", fw1);
69b11869dbSPete Eberlein 
700ee3d4d5SHans Verkuil 	if (request_firmware(&fw, fw1, &usbdev->dev)) {
71b11558a3SYAMANE Toshiaki 		dev_err(&interface->dev,
720ee3d4d5SHans Verkuil 			"unable to load firmware from file \"%s\"\n", fw1);
73b11869dbSPete Eberlein 		goto failed2;
74b11869dbSPete Eberlein 	}
7579a63c60SHans Verkuil 	ret = cypress_load_firmware(usbdev, fw, CYPRESS_FX2);
76b11869dbSPete Eberlein 	release_firmware(fw);
77b11869dbSPete Eberlein 	if (0 != ret) {
78b11558a3SYAMANE Toshiaki 		dev_err(&interface->dev, "loader download failed\n");
79b11869dbSPete Eberlein 		goto failed2;
80b11869dbSPete Eberlein 	}
81b11869dbSPete Eberlein 
820ee3d4d5SHans Verkuil 	if (fw2 == NULL)
830ee3d4d5SHans Verkuil 		return 0;
840ee3d4d5SHans Verkuil 
850ee3d4d5SHans Verkuil 	if (request_firmware(&fw, fw2, &usbdev->dev)) {
86b11558a3SYAMANE Toshiaki 		dev_err(&interface->dev,
870ee3d4d5SHans Verkuil 			"unable to load firmware from file \"%s\"\n", fw2);
88b11869dbSPete Eberlein 		goto failed2;
89b11869dbSPete Eberlein 	}
9079a63c60SHans Verkuil 	ret = cypress_load_firmware(usbdev, fw, CYPRESS_FX2);
91b11869dbSPete Eberlein 	release_firmware(fw);
92b11869dbSPete Eberlein 	if (0 != ret) {
93e129c974SHans Verkuil 		dev_err(&interface->dev, "firmware download failed\n");
94b11869dbSPete Eberlein 		goto failed2;
95b11869dbSPete Eberlein 	}
96b11869dbSPete Eberlein 	return 0;
97b11869dbSPete Eberlein 
98b11869dbSPete Eberlein failed2:
9950c88544SAlexey Khoroshilov 	usb_put_dev(usbdev);
100b11558a3SYAMANE Toshiaki 	dev_err(&interface->dev, "probe failed\n");
1010ee3d4d5SHans Verkuil 	return -ENODEV;
102b11869dbSPete Eberlein }
103b11869dbSPete Eberlein 
104e129c974SHans Verkuil static void go7007_loader_disconnect(struct usb_interface *interface)
105b11869dbSPete Eberlein {
106e129c974SHans Verkuil 	dev_info(&interface->dev, "disconnect\n");
10750c88544SAlexey Khoroshilov 	usb_put_dev(interface_to_usbdev(interface));
108b11869dbSPete Eberlein 	usb_set_intfdata(interface, NULL);
109b11869dbSPete Eberlein }
110b11869dbSPete Eberlein 
111e129c974SHans Verkuil static const struct usb_device_id go7007_loader_ids[] = {
112b11869dbSPete Eberlein 	{ USB_DEVICE(0x1943, 0xa250) },
1130ee3d4d5SHans Verkuil 	{ USB_DEVICE(0x093b, 0xa002) },
1140ee3d4d5SHans Verkuil 	{ USB_DEVICE(0x093b, 0xa004) },
1150ee3d4d5SHans Verkuil 	{ USB_DEVICE(0x0eb1, 0x6666) },
1160ee3d4d5SHans Verkuil 	{ USB_DEVICE(0x0eb1, 0x6668) },
117b11869dbSPete Eberlein 	{}                          /* Terminating entry */
118b11869dbSPete Eberlein };
119b11869dbSPete Eberlein 
120e129c974SHans Verkuil MODULE_DEVICE_TABLE(usb, go7007_loader_ids);
121b11869dbSPete Eberlein 
122e129c974SHans Verkuil static struct usb_driver go7007_loader_driver = {
123e129c974SHans Verkuil 	.name		= "go7007-loader",
124e129c974SHans Verkuil 	.probe		= go7007_loader_probe,
125e129c974SHans Verkuil 	.disconnect	= go7007_loader_disconnect,
126e129c974SHans Verkuil 	.id_table	= go7007_loader_ids,
127b11869dbSPete Eberlein };
128b11869dbSPete Eberlein 
129e129c974SHans Verkuil module_usb_driver(go7007_loader_driver);
130832b6a89SPete Eberlein 
131832b6a89SPete Eberlein MODULE_AUTHOR("");
1320ee3d4d5SHans Verkuil MODULE_DESCRIPTION("firmware loader for go7007-usb");
133832b6a89SPete Eberlein MODULE_LICENSE("GPL v2");
134