detach the kernel driver before mode switch
[project/usbmode.git] / switch.c
1 #include "switch.h"
2
3 enum {
4         DATA_MODE,
5         DATA_MSG,
6         DATA_INTERFACE,
7         __DATA_MAX
8 };
9
10 static void detach_driver(struct usbdev_data *data)
11 {
12         libusb_detach_kernel_driver(data->devh, data->interface);
13 }
14
15 static void handle_generic(struct usbdev_data *data, struct blob_attr **tb)
16 {
17         detach_driver(data);
18 }
19
20 static void handle_huawei(struct usbdev_data *data, struct blob_attr **tb)
21 {
22         /* TODO */
23 }
24
25 static void handle_sierra(struct usbdev_data *data, struct blob_attr **tb)
26 {
27         /* TODO */
28 }
29
30 static void handle_sony(struct usbdev_data *data, struct blob_attr **tb)
31 {
32         /* TODO */
33 }
34
35 static void handle_qisda(struct usbdev_data *data, struct blob_attr **tb)
36 {
37         /* TODO */
38 }
39
40 static void handle_gct(struct usbdev_data *data, struct blob_attr **tb)
41 {
42         detach_driver(data);
43         /* TODO */
44 }
45
46 static void handle_kobil(struct usbdev_data *data, struct blob_attr **tb)
47 {
48         detach_driver(data);
49         /* TODO */
50 }
51
52 static void handle_sequans(struct usbdev_data *data, struct blob_attr **tb)
53 {
54         /* TODO */
55 }
56
57 static void handle_mobile_action(struct usbdev_data *data, struct blob_attr **tb)
58 {
59         /* TODO */
60 }
61
62 static void handle_cisco(struct usbdev_data *data, struct blob_attr **tb)
63 {
64         detach_driver(data);
65         /* TODO */
66 }
67
68 enum {
69         MODE_GENERIC,
70         MODE_HUAWEI,
71         MODE_SIERRA,
72         MODE_SONY,
73         MODE_QISDA,
74         MODE_GCT,
75         MODE_KOBIL,
76         MODE_SEQUANS,
77         MODE_MOBILE_ACTION,
78         MODE_CISCO,
79         __MODE_MAX
80 };
81
82 static const struct {
83         const char *name;
84         void (*cb)(struct usbdev_data *data, struct blob_attr **tb);
85 } modeswitch_cb[__MODE_MAX] = {
86         [MODE_GENERIC] = { "Generic", handle_generic },
87         [MODE_HUAWEI] = { "Huawei", handle_huawei },
88         [MODE_SIERRA] = { "Sierra", handle_sierra },
89         [MODE_SONY] = { "Sony", handle_sony },
90         [MODE_QISDA] = { "Qisda", handle_qisda },
91         [MODE_GCT] = { "GCT", handle_gct },
92         [MODE_KOBIL] = { "Kobil", handle_kobil },
93         [MODE_SEQUANS] = { "Sequans", handle_sequans },
94         [MODE_MOBILE_ACTION] = { "MobileAction", handle_mobile_action },
95         [MODE_CISCO] = { "Cisco", handle_cisco },
96 };
97
98 void handle_switch(struct usbdev_data *data)
99 {
100         static const struct blobmsg_policy data_policy[__DATA_MAX] = {
101                 [DATA_MODE] = { .name = "mode", .type = BLOBMSG_TYPE_STRING },
102                 [DATA_MSG] = { .name = "msg", .type = BLOBMSG_TYPE_ARRAY },
103                 [DATA_INTERFACE] = { .name = "interface", .type = BLOBMSG_TYPE_INT32 },
104         };
105         struct blob_attr *tb[__DATA_MAX];
106         int mode = MODE_GENERIC;
107
108         blobmsg_parse(data_policy, __DATA_MAX, tb, blobmsg_data(data->info), blobmsg_data_len(data->info));
109
110         if (tb[DATA_INTERFACE])
111                 data->interface = blobmsg_get_u32(tb[DATA_INTERFACE]);
112
113         if (tb[DATA_MODE]) {
114                 const char *modestr;
115                 int i;
116
117                 modestr = blobmsg_data(tb[DATA_MODE]);
118                 for (i = 0; i < __MODE_MAX; i++) {
119                         if (strcmp(modeswitch_cb[i].name, modestr) != 0)
120                                 continue;
121
122                         mode = i;
123                         break;
124                 }
125         }
126
127         modeswitch_cb[mode].cb(data, tb);
128 }