|
@@ -22,86 +22,67 @@
|
|
|
#include <linux/firmware.h>
|
|
|
#include <cypress_firmware.h>
|
|
|
|
|
|
-#define S2250_LOADER_FIRMWARE "s2250_loader.fw"
|
|
|
-#define S2250_FIRMWARE "2250.fw"
|
|
|
-
|
|
|
-typedef struct device_extension_s {
|
|
|
- struct kref kref;
|
|
|
- int minor;
|
|
|
- struct usb_device *usbdev;
|
|
|
-} device_extension_t, *pdevice_extension_t;
|
|
|
-
|
|
|
-#define USB_go7007_loader_MAJOR 240
|
|
|
-#define USB_go7007_loader_MINOR_BASE 0
|
|
|
-#define MAX_DEVICES 256
|
|
|
-
|
|
|
-static pdevice_extension_t go7007_dev_table[MAX_DEVICES];
|
|
|
-static DEFINE_MUTEX(go7007_dev_table_mutex);
|
|
|
+struct fw_config {
|
|
|
+ u16 vendor;
|
|
|
+ u16 product;
|
|
|
+ const char * const fw_name1;
|
|
|
+ const char * const fw_name2;
|
|
|
+};
|
|
|
|
|
|
-#define to_go7007_loader_dev_common(d) container_of(d, device_extension_t, kref)
|
|
|
-static void go7007_loader_delete(struct kref *kref)
|
|
|
-{
|
|
|
- pdevice_extension_t s = to_go7007_loader_dev_common(kref);
|
|
|
- go7007_dev_table[s->minor] = NULL;
|
|
|
- kfree(s);
|
|
|
-}
|
|
|
+struct fw_config fw_configs[] = {
|
|
|
+ { 0x1943, 0xa250, "go7007/s2250-1.fw", "go7007/s2250-2.fw" },
|
|
|
+ { 0x093b, 0xa002, "go7007/px-m402u.fw", NULL },
|
|
|
+ { 0x093b, 0xa004, "go7007/px-tv402u.fw", NULL },
|
|
|
+ { 0x0eb1, 0x6666, "go7007/lr192.fw", NULL },
|
|
|
+ { 0x0eb1, 0x6668, "go7007/wis-startrek.fw", NULL },
|
|
|
+ { 0, 0, NULL, NULL }
|
|
|
+};
|
|
|
+MODULE_FIRMWARE("go7007/s2250-1.fw");
|
|
|
+MODULE_FIRMWARE("go7007/s2250-2.fw");
|
|
|
+MODULE_FIRMWARE("go7007/px-m402u.fw");
|
|
|
+MODULE_FIRMWARE("go7007/px-tv402u.fw");
|
|
|
+MODULE_FIRMWARE("go7007/lr192.fw");
|
|
|
+MODULE_FIRMWARE("go7007/wis-startrek.fw");
|
|
|
|
|
|
static int go7007_loader_probe(struct usb_interface *interface,
|
|
|
const struct usb_device_id *id)
|
|
|
{
|
|
|
struct usb_device *usbdev;
|
|
|
- int minor, ret;
|
|
|
- pdevice_extension_t s = NULL;
|
|
|
const struct firmware *fw;
|
|
|
+ u16 vendor, product;
|
|
|
+ const char *fw1, *fw2;
|
|
|
+ int ret;
|
|
|
+ int i;
|
|
|
|
|
|
usbdev = usb_get_dev(interface_to_usbdev(interface));
|
|
|
- if (!usbdev) {
|
|
|
- dev_err(&interface->dev, "Enter go7007_loader_probe failed\n");
|
|
|
- return -1;
|
|
|
- }
|
|
|
- dev_info(&interface->dev, "vendor id 0x%x, device id 0x%x devnum:%d\n",
|
|
|
- usbdev->descriptor.idVendor, usbdev->descriptor.idProduct,
|
|
|
- usbdev->devnum);
|
|
|
+ if (!usbdev)
|
|
|
+ goto failed2;
|
|
|
|
|
|
if (usbdev->descriptor.bNumConfigurations != 1) {
|
|
|
dev_err(&interface->dev, "can't handle multiple config\n");
|
|
|
- return -1;
|
|
|
- }
|
|
|
- mutex_lock(&go7007_dev_table_mutex);
|
|
|
-
|
|
|
- for (minor = 0; minor < MAX_DEVICES; minor++) {
|
|
|
- if (go7007_dev_table[minor] == NULL)
|
|
|
- break;
|
|
|
+ return -ENODEV;
|
|
|
}
|
|
|
|
|
|
- if (minor < 0 || minor >= MAX_DEVICES) {
|
|
|
- dev_err(&interface->dev, "Invalid minor: %d\n", minor);
|
|
|
- goto failed;
|
|
|
- }
|
|
|
-
|
|
|
- /* Allocate dev data structure */
|
|
|
- s = kmalloc(sizeof(device_extension_t), GFP_KERNEL);
|
|
|
- if (s == NULL)
|
|
|
- goto failed;
|
|
|
-
|
|
|
- go7007_dev_table[minor] = s;
|
|
|
+ vendor = le16_to_cpu(usbdev->descriptor.idVendor);
|
|
|
+ product = le16_to_cpu(usbdev->descriptor.idProduct);
|
|
|
|
|
|
- dev_info(&interface->dev,
|
|
|
- "Device %d on Bus %d Minor %d\n",
|
|
|
- usbdev->devnum, usbdev->bus->busnum, minor);
|
|
|
+ for (i = 0; fw_configs[i].fw_name1; i++)
|
|
|
+ if (fw_configs[i].vendor == vendor &&
|
|
|
+ fw_configs[i].product == product)
|
|
|
+ break;
|
|
|
|
|
|
- memset(s, 0, sizeof(device_extension_t));
|
|
|
- s->usbdev = usbdev;
|
|
|
- dev_info(&interface->dev, "loading go7007-loader\n");
|
|
|
+ /* Should never happen */
|
|
|
+ if (fw_configs[i].fw_name1 == NULL)
|
|
|
+ goto failed2;
|
|
|
|
|
|
- kref_init(&(s->kref));
|
|
|
+ fw1 = fw_configs[i].fw_name1;
|
|
|
+ fw2 = fw_configs[i].fw_name2;
|
|
|
|
|
|
- mutex_unlock(&go7007_dev_table_mutex);
|
|
|
+ dev_info(&interface->dev, "loading firmware %s\n", fw1);
|
|
|
|
|
|
- if (request_firmware(&fw, S2250_LOADER_FIRMWARE, &usbdev->dev)) {
|
|
|
+ if (request_firmware(&fw, fw1, &usbdev->dev)) {
|
|
|
dev_err(&interface->dev,
|
|
|
- "unable to load firmware from file \"%s\"\n",
|
|
|
- S2250_LOADER_FIRMWARE);
|
|
|
+ "unable to load firmware from file \"%s\"\n", fw1);
|
|
|
goto failed2;
|
|
|
}
|
|
|
ret = usbv2_cypress_load_firmware(usbdev, fw, CYPRESS_FX2);
|
|
@@ -111,10 +92,12 @@ static int go7007_loader_probe(struct usb_interface *interface,
|
|
|
goto failed2;
|
|
|
}
|
|
|
|
|
|
- if (request_firmware(&fw, S2250_FIRMWARE, &usbdev->dev)) {
|
|
|
+ if (fw2 == NULL)
|
|
|
+ return 0;
|
|
|
+
|
|
|
+ if (request_firmware(&fw, fw2, &usbdev->dev)) {
|
|
|
dev_err(&interface->dev,
|
|
|
- "unable to load firmware from file \"%s\"\n",
|
|
|
- S2250_FIRMWARE);
|
|
|
+ "unable to load firmware from file \"%s\"\n", fw2);
|
|
|
goto failed2;
|
|
|
}
|
|
|
ret = usbv2_cypress_load_firmware(usbdev, fw, CYPRESS_FX2);
|
|
@@ -123,31 +106,25 @@ static int go7007_loader_probe(struct usb_interface *interface,
|
|
|
dev_err(&interface->dev, "firmware download failed\n");
|
|
|
goto failed2;
|
|
|
}
|
|
|
-
|
|
|
- usb_set_intfdata(interface, s);
|
|
|
return 0;
|
|
|
|
|
|
-failed:
|
|
|
- mutex_unlock(&go7007_dev_table_mutex);
|
|
|
failed2:
|
|
|
- if (s)
|
|
|
- kref_put(&(s->kref), go7007_loader_delete);
|
|
|
-
|
|
|
dev_err(&interface->dev, "probe failed\n");
|
|
|
- return -1;
|
|
|
+ return -ENODEV;
|
|
|
}
|
|
|
|
|
|
static void go7007_loader_disconnect(struct usb_interface *interface)
|
|
|
{
|
|
|
- pdevice_extension_t s;
|
|
|
dev_info(&interface->dev, "disconnect\n");
|
|
|
- s = usb_get_intfdata(interface);
|
|
|
usb_set_intfdata(interface, NULL);
|
|
|
- kref_put(&(s->kref), go7007_loader_delete);
|
|
|
}
|
|
|
|
|
|
static const struct usb_device_id go7007_loader_ids[] = {
|
|
|
- {USB_DEVICE(0x1943, 0xa250)},
|
|
|
+ { USB_DEVICE(0x1943, 0xa250) },
|
|
|
+ { USB_DEVICE(0x093b, 0xa002) },
|
|
|
+ { USB_DEVICE(0x093b, 0xa004) },
|
|
|
+ { USB_DEVICE(0x0eb1, 0x6666) },
|
|
|
+ { USB_DEVICE(0x0eb1, 0x6668) },
|
|
|
{} /* Terminating entry */
|
|
|
};
|
|
|
|
|
@@ -163,7 +140,5 @@ static struct usb_driver go7007_loader_driver = {
|
|
|
module_usb_driver(go7007_loader_driver);
|
|
|
|
|
|
MODULE_AUTHOR("");
|
|
|
-MODULE_DESCRIPTION("firmware loader for go7007 USB devices");
|
|
|
+MODULE_DESCRIPTION("firmware loader for go7007-usb");
|
|
|
MODULE_LICENSE("GPL v2");
|
|
|
-MODULE_FIRMWARE(S2250_LOADER_FIRMWARE);
|
|
|
-MODULE_FIRMWARE(S2250_FIRMWARE);
|