support Quanta and Blackberry modes
[project/usbmode.git] / switch.c
index 2c33d9c..fe25e80 100644 (file)
--- a/switch.c
+++ b/switch.c
@@ -11,6 +11,7 @@ enum {
        DATA_RELEASE_DELAY,
        DATA_CONFIG,
        DATA_ALT,
+       DATA_DEV_CLASS,
        __DATA_MAX
 };
 
@@ -151,6 +152,20 @@ static void handle_huaweinew(struct usbdev_data *data, struct blob_attr **tb)
        send_messages(data, msgs, ARRAY_SIZE(msgs));
 }
 
+static void handle_option(struct usbdev_data *data, struct blob_attr **tb)
+{
+       static struct msg_entry msgs[] = {
+               {
+                       "\x55\x53\x42\x43\x12\x34\x56\x78\x00\x00\x00\x00\x00\x00\x06\x01"
+                       "\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00", 31
+               }
+       };
+
+       detach_driver(data);
+       data->need_response = false;
+       send_messages(data, msgs, ARRAY_SIZE(msgs));
+}
+
 static void handle_standardeject(struct usbdev_data *data, struct blob_attr **tb)
 {
        static struct msg_entry msgs[] = {
@@ -160,6 +175,12 @@ static void handle_standardeject(struct usbdev_data *data, struct blob_attr **tb
                }, {
                        "\x55\x53\x42\x43\x12\x34\x56\x79\x00\x00\x00\x00\x00\x00\x06\x1b"
                        "\x00\x00\x00\x02\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00", 31
+               }, {
+                       "\x55\x53\x42\x43\x12\x34\x56\x78\x00\x00\x00\x00\x00\x01\x06\x1e"
+                       "\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00", 31
+               }, {
+                       "\x55\x53\x42\x43\x12\x34\x56\x79\x00\x00\x00\x00\x00\x01\x06\x1b"
+                       "\x00\x00\x00\x02\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00", 31
                }
        };
 
@@ -354,6 +375,23 @@ static void handle_mbim(struct usbdev_data *data, struct blob_attr **tb)
        }
 }
 
+static void handle_quanta(struct usbdev_data *data, struct blob_attr **tb)
+{
+       int type = LIBUSB_REQUEST_TYPE_VENDOR | LIBUSB_RECIPIENT_DEVICE | LIBUSB_ENDPOINT_IN;
+
+       detach_driver(data);
+       send_control_packet(data, type, 0xff, 0, 0, 8);
+}
+
+static void handle_blackberry(struct usbdev_data *data, struct blob_attr **tb)
+{
+       int type = LIBUSB_REQUEST_TYPE_VENDOR | LIBUSB_RECIPIENT_DEVICE | LIBUSB_ENDPOINT_IN;
+
+       detach_driver(data);
+       send_control_packet(data, type, 0xb1, 0x0000, 0, 8);
+       send_control_packet(data, type, 0xa9, 0x000e, 0, 8);
+}
+
 static void set_alt_setting(struct usbdev_data *data, int setting)
 {
        if (libusb_claim_interface(data->devh, data->interface))
@@ -377,6 +415,9 @@ enum {
        MODE_MOBILE_ACTION,
        MODE_CISCO,
        MODE_MBIM,
+       MODE_OPTION,
+       MODE_QUANTA,
+       MODE_BLACKBERRY,
        __MODE_MAX
 };
 
@@ -397,6 +438,9 @@ static const struct {
        [MODE_MOBILE_ACTION] = { "MobileAction", handle_mobile_action },
        [MODE_CISCO] = { "Cisco", handle_cisco },
        [MODE_MBIM] = { "MBIM", handle_mbim },
+       [MODE_OPTION] = { "Option", handle_option },
+       [MODE_QUANTA] = { "Quanta", handle_quanta },
+       [MODE_BLACKBERRY] = { "Blackberry", handle_blackberry },
 };
 
 void handle_switch(struct usbdev_data *data)
@@ -410,12 +454,17 @@ void handle_switch(struct usbdev_data *data)
                [DATA_RESPONSE] = { .name = "response", .type = BLOBMSG_TYPE_BOOL },
                [DATA_CONFIG] = { .name = "config", .type = BLOBMSG_TYPE_INT32 },
                [DATA_ALT] = { .name = "alt", .type = BLOBMSG_TYPE_INT32 },
+               [DATA_DEV_CLASS] = { .name = "t_class", .type = BLOBMSG_TYPE_INT32 },
        };
        struct blob_attr *tb[__DATA_MAX];
        int mode = MODE_GENERIC;
+       int t_class = 0;
 
        blobmsg_parse(data_policy, __DATA_MAX, tb, blobmsg_data(data->info), blobmsg_data_len(data->info));
 
+       if (tb[DATA_DEV_CLASS])
+               t_class = blobmsg_get_u32(tb[DATA_DEV_CLASS]);
+
        if (tb[DATA_INTERFACE])
                data->interface = blobmsg_get_u32(tb[DATA_INTERFACE]);
 
@@ -431,6 +480,9 @@ void handle_switch(struct usbdev_data *data)
        if (tb[DATA_RESPONSE])
                data->need_response = blobmsg_get_bool(tb[DATA_RESPONSE]);
 
+       if (t_class > 0 && data->dev_class != t_class)
+               return;
+
        if (tb[DATA_MODE]) {
                const char *modestr;
                int i;