blob: 228bffaa69c913e31205582082f4b112e857bb0d [file] [log] [blame]
Chris Kellyb3147862012-02-20 21:12:08 +00001/* -----------------------------------------------------------------------------
2 * Copyright (c) 2011 Ozmo Inc
3 * Released under the GNU General Public License Version 2 (GPLv2).
4 *
5 * This file implements the protocol specific parts of the USB service for a PD.
6 * -----------------------------------------------------------------------------
7 */
8#include <linux/init.h>
9#include <linux/module.h>
10#include <linux/timer.h>
11#include <linux/sched.h>
12#include <linux/netdevice.h>
13#include <linux/errno.h>
14#include <linux/input.h>
15#include <asm/unaligned.h>
Joe Perchesf724b582013-07-23 13:45:00 +010016#include "ozdbg.h"
Chris Kellyb3147862012-02-20 21:12:08 +000017#include "ozprotocol.h"
18#include "ozeltbuf.h"
19#include "ozpd.h"
20#include "ozproto.h"
21#include "ozusbif.h"
22#include "ozhcd.h"
Chris Kellyb3147862012-02-20 21:12:08 +000023#include "ozusbsvc.h"
Rupesh Gujare6e244a82013-08-13 18:24:22 +010024
Chris Kellyb3147862012-02-20 21:12:08 +000025#define MAX_ISOC_FIXED_DATA (253-sizeof(struct oz_isoc_fixed))
Rupesh Gujare6e244a82013-08-13 18:24:22 +010026
Rupesh Gujare4e7fb822013-08-23 16:11:02 +010027/*
Chris Kellyb3147862012-02-20 21:12:08 +000028 * Context: softirq
29 */
30static int oz_usb_submit_elt(struct oz_elt_buf *eb, struct oz_elt_info *ei,
31 struct oz_usb_ctx *usb_ctx, u8 strid, u8 isoc)
32{
33 int ret;
34 struct oz_elt *elt = (struct oz_elt *)ei->data;
35 struct oz_app_hdr *app_hdr = (struct oz_app_hdr *)(elt+1);
Rupesh Gujare18f81912013-08-13 18:24:21 +010036
Chris Kellyb3147862012-02-20 21:12:08 +000037 elt->type = OZ_ELT_APP_DATA;
38 ei->app_id = OZ_APPID_USB;
39 ei->length = elt->length + sizeof(struct oz_elt);
40 app_hdr->app_id = OZ_APPID_USB;
41 spin_lock_bh(&eb->lock);
42 if (isoc == 0) {
43 app_hdr->elt_seq_num = usb_ctx->tx_seq_num++;
44 if (usb_ctx->tx_seq_num == 0)
45 usb_ctx->tx_seq_num = 1;
46 }
47 ret = oz_queue_elt_info(eb, isoc, strid, ei);
48 if (ret)
49 oz_elt_info_free(eb, ei);
50 spin_unlock_bh(&eb->lock);
51 return ret;
52}
Rupesh Gujare6e244a82013-08-13 18:24:22 +010053
Rupesh Gujare4e7fb822013-08-23 16:11:02 +010054/*
Chris Kellyb3147862012-02-20 21:12:08 +000055 * Context: softirq
56 */
57int oz_usb_get_desc_req(void *hpd, u8 req_id, u8 req_type, u8 desc_type,
58 u8 index, u16 windex, int offset, int len)
59{
60 struct oz_usb_ctx *usb_ctx = (struct oz_usb_ctx *)hpd;
61 struct oz_pd *pd = usb_ctx->pd;
62 struct oz_elt *elt;
63 struct oz_get_desc_req *body;
64 struct oz_elt_buf *eb = &pd->elt_buff;
65 struct oz_elt_info *ei = oz_elt_info_alloc(&pd->elt_buff);
Rupesh Gujare18f81912013-08-13 18:24:21 +010066
Joe Perchesf724b582013-07-23 13:45:00 +010067 oz_dbg(ON, " req_type = 0x%x\n", req_type);
68 oz_dbg(ON, " desc_type = 0x%x\n", desc_type);
69 oz_dbg(ON, " index = 0x%x\n", index);
70 oz_dbg(ON, " windex = 0x%x\n", windex);
71 oz_dbg(ON, " offset = 0x%x\n", offset);
72 oz_dbg(ON, " len = 0x%x\n", len);
Chris Kellyb3147862012-02-20 21:12:08 +000073 if (len > 200)
74 len = 200;
Peter Huewe4d1b2fb2013-02-15 15:22:23 +010075 if (ei == NULL)
Chris Kellyb3147862012-02-20 21:12:08 +000076 return -1;
77 elt = (struct oz_elt *)ei->data;
78 elt->length = sizeof(struct oz_get_desc_req);
79 body = (struct oz_get_desc_req *)(elt+1);
80 body->type = OZ_GET_DESC_REQ;
81 body->req_id = req_id;
82 put_unaligned(cpu_to_le16(offset), &body->offset);
83 put_unaligned(cpu_to_le16(len), &body->size);
84 body->req_type = req_type;
85 body->desc_type = desc_type;
86 body->w_index = windex;
87 body->index = index;
88 return oz_usb_submit_elt(eb, ei, usb_ctx, 0, 0);
89}
Rupesh Gujare6e244a82013-08-13 18:24:22 +010090
Rupesh Gujare4e7fb822013-08-23 16:11:02 +010091/*
Chris Kellyb3147862012-02-20 21:12:08 +000092 * Context: tasklet
93 */
94static int oz_usb_set_config_req(void *hpd, u8 req_id, u8 index)
95{
96 struct oz_usb_ctx *usb_ctx = (struct oz_usb_ctx *)hpd;
97 struct oz_pd *pd = usb_ctx->pd;
98 struct oz_elt *elt;
99 struct oz_elt_buf *eb = &pd->elt_buff;
100 struct oz_elt_info *ei = oz_elt_info_alloc(&pd->elt_buff);
101 struct oz_set_config_req *body;
Rupesh Gujare18f81912013-08-13 18:24:21 +0100102
Peter Huewe4d1b2fb2013-02-15 15:22:23 +0100103 if (ei == NULL)
Chris Kellyb3147862012-02-20 21:12:08 +0000104 return -1;
105 elt = (struct oz_elt *)ei->data;
106 elt->length = sizeof(struct oz_set_config_req);
107 body = (struct oz_set_config_req *)(elt+1);
108 body->type = OZ_SET_CONFIG_REQ;
109 body->req_id = req_id;
110 body->index = index;
111 return oz_usb_submit_elt(eb, ei, usb_ctx, 0, 0);
112}
Rupesh Gujare6e244a82013-08-13 18:24:22 +0100113
Rupesh Gujare4e7fb822013-08-23 16:11:02 +0100114/*
Chris Kellyb3147862012-02-20 21:12:08 +0000115 * Context: tasklet
116 */
117static int oz_usb_set_interface_req(void *hpd, u8 req_id, u8 index, u8 alt)
118{
119 struct oz_usb_ctx *usb_ctx = (struct oz_usb_ctx *)hpd;
120 struct oz_pd *pd = usb_ctx->pd;
121 struct oz_elt *elt;
122 struct oz_elt_buf *eb = &pd->elt_buff;
123 struct oz_elt_info *ei = oz_elt_info_alloc(&pd->elt_buff);
124 struct oz_set_interface_req *body;
Rupesh Gujare18f81912013-08-13 18:24:21 +0100125
Peter Huewe4d1b2fb2013-02-15 15:22:23 +0100126 if (ei == NULL)
Chris Kellyb3147862012-02-20 21:12:08 +0000127 return -1;
128 elt = (struct oz_elt *)ei->data;
129 elt->length = sizeof(struct oz_set_interface_req);
130 body = (struct oz_set_interface_req *)(elt+1);
131 body->type = OZ_SET_INTERFACE_REQ;
132 body->req_id = req_id;
133 body->index = index;
134 body->alternative = alt;
135 return oz_usb_submit_elt(eb, ei, usb_ctx, 0, 0);
136}
Rupesh Gujare6e244a82013-08-13 18:24:22 +0100137
Rupesh Gujare4e7fb822013-08-23 16:11:02 +0100138/*
Chris Kellyb3147862012-02-20 21:12:08 +0000139 * Context: tasklet
140 */
141static int oz_usb_set_clear_feature_req(void *hpd, u8 req_id, u8 type,
142 u8 recipient, u8 index, __le16 feature)
143{
144 struct oz_usb_ctx *usb_ctx = (struct oz_usb_ctx *)hpd;
145 struct oz_pd *pd = usb_ctx->pd;
146 struct oz_elt *elt;
147 struct oz_elt_buf *eb = &pd->elt_buff;
148 struct oz_elt_info *ei = oz_elt_info_alloc(&pd->elt_buff);
149 struct oz_feature_req *body;
Rupesh Gujare18f81912013-08-13 18:24:21 +0100150
Peter Huewe4d1b2fb2013-02-15 15:22:23 +0100151 if (ei == NULL)
Chris Kellyb3147862012-02-20 21:12:08 +0000152 return -1;
153 elt = (struct oz_elt *)ei->data;
154 elt->length = sizeof(struct oz_feature_req);
155 body = (struct oz_feature_req *)(elt+1);
156 body->type = type;
157 body->req_id = req_id;
158 body->recipient = recipient;
159 body->index = index;
160 put_unaligned(feature, &body->feature);
161 return oz_usb_submit_elt(eb, ei, usb_ctx, 0, 0);
162}
Rupesh Gujare6e244a82013-08-13 18:24:22 +0100163
Rupesh Gujare4e7fb822013-08-23 16:11:02 +0100164/*
Chris Kellyb3147862012-02-20 21:12:08 +0000165 * Context: tasklet
166 */
167static int oz_usb_vendor_class_req(void *hpd, u8 req_id, u8 req_type,
Peter Huewedc7f5b32013-02-15 21:17:24 +0100168 u8 request, __le16 value, __le16 index, const u8 *data, int data_len)
Chris Kellyb3147862012-02-20 21:12:08 +0000169{
170 struct oz_usb_ctx *usb_ctx = (struct oz_usb_ctx *)hpd;
171 struct oz_pd *pd = usb_ctx->pd;
172 struct oz_elt *elt;
173 struct oz_elt_buf *eb = &pd->elt_buff;
174 struct oz_elt_info *ei = oz_elt_info_alloc(&pd->elt_buff);
175 struct oz_vendor_class_req *body;
Rupesh Gujare18f81912013-08-13 18:24:21 +0100176
Peter Huewe4d1b2fb2013-02-15 15:22:23 +0100177 if (ei == NULL)
Chris Kellyb3147862012-02-20 21:12:08 +0000178 return -1;
179 elt = (struct oz_elt *)ei->data;
180 elt->length = sizeof(struct oz_vendor_class_req) - 1 + data_len;
181 body = (struct oz_vendor_class_req *)(elt+1);
182 body->type = OZ_VENDOR_CLASS_REQ;
183 body->req_id = req_id;
184 body->req_type = req_type;
185 body->request = request;
186 put_unaligned(value, &body->value);
187 put_unaligned(index, &body->index);
188 if (data_len)
189 memcpy(body->data, data, data_len);
190 return oz_usb_submit_elt(eb, ei, usb_ctx, 0, 0);
191}
Rupesh Gujare6e244a82013-08-13 18:24:22 +0100192
Rupesh Gujare4e7fb822013-08-23 16:11:02 +0100193/*
Chris Kellyb3147862012-02-20 21:12:08 +0000194 * Context: tasklet
195 */
196int oz_usb_control_req(void *hpd, u8 req_id, struct usb_ctrlrequest *setup,
Peter Huewedc7f5b32013-02-15 21:17:24 +0100197 const u8 *data, int data_len)
Chris Kellyb3147862012-02-20 21:12:08 +0000198{
199 unsigned wvalue = le16_to_cpu(setup->wValue);
200 unsigned windex = le16_to_cpu(setup->wIndex);
201 unsigned wlength = le16_to_cpu(setup->wLength);
202 int rc = 0;
Rupesh Gujare18f81912013-08-13 18:24:21 +0100203
Chris Kellyb3147862012-02-20 21:12:08 +0000204 if ((setup->bRequestType & USB_TYPE_MASK) == USB_TYPE_STANDARD) {
205 switch (setup->bRequest) {
206 case USB_REQ_GET_DESCRIPTOR:
207 rc = oz_usb_get_desc_req(hpd, req_id,
208 setup->bRequestType, (u8)(wvalue>>8),
209 (u8)wvalue, setup->wIndex, 0, wlength);
210 break;
211 case USB_REQ_SET_CONFIGURATION:
212 rc = oz_usb_set_config_req(hpd, req_id, (u8)wvalue);
213 break;
214 case USB_REQ_SET_INTERFACE: {
215 u8 if_num = (u8)windex;
216 u8 alt = (u8)wvalue;
217 rc = oz_usb_set_interface_req(hpd, req_id,
218 if_num, alt);
219 }
220 break;
221 case USB_REQ_SET_FEATURE:
222 rc = oz_usb_set_clear_feature_req(hpd, req_id,
223 OZ_SET_FEATURE_REQ,
224 setup->bRequestType & 0xf, (u8)windex,
225 setup->wValue);
226 break;
227 case USB_REQ_CLEAR_FEATURE:
228 rc = oz_usb_set_clear_feature_req(hpd, req_id,
229 OZ_CLEAR_FEATURE_REQ,
230 setup->bRequestType & 0xf,
231 (u8)windex, setup->wValue);
232 break;
233 }
234 } else {
235 rc = oz_usb_vendor_class_req(hpd, req_id, setup->bRequestType,
236 setup->bRequest, setup->wValue, setup->wIndex,
237 data, data_len);
238 }
239 return rc;
240}
Rupesh Gujare6e244a82013-08-13 18:24:22 +0100241
Rupesh Gujare4e7fb822013-08-23 16:11:02 +0100242/*
Chris Kellyb3147862012-02-20 21:12:08 +0000243 * Context: softirq
244 */
245int oz_usb_send_isoc(void *hpd, u8 ep_num, struct urb *urb)
246{
247 struct oz_usb_ctx *usb_ctx = (struct oz_usb_ctx *)hpd;
248 struct oz_pd *pd = usb_ctx->pd;
249 struct oz_elt_buf *eb;
250 int i;
251 int hdr_size;
252 u8 *data;
253 struct usb_iso_packet_descriptor *desc;
254
255 if (pd->mode & OZ_F_ISOC_NO_ELTS) {
256 for (i = 0; i < urb->number_of_packets; i++) {
257 u8 *data;
258 desc = &urb->iso_frame_desc[i];
259 data = ((u8 *)urb->transfer_buffer)+desc->offset;
260 oz_send_isoc_unit(pd, ep_num, data, desc->length);
261 }
262 return 0;
263 }
264
265 hdr_size = sizeof(struct oz_isoc_fixed) - 1;
266 eb = &pd->elt_buff;
267 i = 0;
268 while (i < urb->number_of_packets) {
269 struct oz_elt_info *ei = oz_elt_info_alloc(eb);
270 struct oz_elt *elt;
271 struct oz_isoc_fixed *body;
272 int unit_count;
273 int unit_size;
274 int rem;
Peter Huewe4d1b2fb2013-02-15 15:22:23 +0100275 if (ei == NULL)
Chris Kellyb3147862012-02-20 21:12:08 +0000276 return -1;
277 rem = MAX_ISOC_FIXED_DATA;
278 elt = (struct oz_elt *)ei->data;
279 body = (struct oz_isoc_fixed *)(elt + 1);
280 body->type = OZ_USB_ENDPOINT_DATA;
281 body->endpoint = ep_num;
282 body->format = OZ_DATA_F_ISOC_FIXED;
283 unit_size = urb->iso_frame_desc[i].length;
284 body->unit_size = (u8)unit_size;
285 data = ((u8 *)(elt+1)) + hdr_size;
286 unit_count = 0;
287 while (i < urb->number_of_packets) {
288 desc = &urb->iso_frame_desc[i];
289 if ((unit_size == desc->length) &&
290 (desc->length <= rem)) {
291 memcpy(data, ((u8 *)urb->transfer_buffer) +
292 desc->offset, unit_size);
293 data += unit_size;
294 rem -= unit_size;
295 unit_count++;
296 desc->status = 0;
297 desc->actual_length = desc->length;
298 i++;
299 } else {
300 break;
301 }
302 }
303 elt->length = hdr_size + MAX_ISOC_FIXED_DATA - rem;
304 /* Store the number of units in body->frame_number for the
305 * moment. This field will be correctly determined before
306 * the element is sent. */
307 body->frame_number = (u8)unit_count;
308 oz_usb_submit_elt(eb, ei, usb_ctx, ep_num,
309 pd->mode & OZ_F_ISOC_ANYTIME);
310 }
311 return 0;
312}
Rupesh Gujare6e244a82013-08-13 18:24:22 +0100313
Rupesh Gujare4e7fb822013-08-23 16:11:02 +0100314/*
Chris Kellyb3147862012-02-20 21:12:08 +0000315 * Context: softirq-serialized
316 */
Peter Huewea7f74c32013-02-15 21:17:22 +0100317static void oz_usb_handle_ep_data(struct oz_usb_ctx *usb_ctx,
Chris Kellyb3147862012-02-20 21:12:08 +0000318 struct oz_usb_hdr *usb_hdr, int len)
319{
320 struct oz_data *data_hdr = (struct oz_data *)usb_hdr;
Rupesh Gujare18f81912013-08-13 18:24:21 +0100321
Chris Kellyb3147862012-02-20 21:12:08 +0000322 switch (data_hdr->format) {
323 case OZ_DATA_F_MULTIPLE_FIXED: {
324 struct oz_multiple_fixed *body =
325 (struct oz_multiple_fixed *)data_hdr;
326 u8 *data = body->data;
327 int n = (len - sizeof(struct oz_multiple_fixed)+1)
328 / body->unit_size;
329 while (n--) {
330 oz_hcd_data_ind(usb_ctx->hport, body->endpoint,
331 data, body->unit_size);
332 data += body->unit_size;
333 }
334 }
335 break;
336 case OZ_DATA_F_ISOC_FIXED: {
337 struct oz_isoc_fixed *body =
338 (struct oz_isoc_fixed *)data_hdr;
339 int data_len = len-sizeof(struct oz_isoc_fixed)+1;
340 int unit_size = body->unit_size;
341 u8 *data = body->data;
342 int count;
343 int i;
344 if (!unit_size)
345 break;
346 count = data_len/unit_size;
347 for (i = 0; i < count; i++) {
348 oz_hcd_data_ind(usb_ctx->hport,
349 body->endpoint, data, unit_size);
350 data += unit_size;
351 }
352 }
353 break;
354 }
355
356}
Rupesh Gujare6e244a82013-08-13 18:24:22 +0100357
Rupesh Gujare4e7fb822013-08-23 16:11:02 +0100358/*
Chris Kellyb3147862012-02-20 21:12:08 +0000359 * This is called when the PD has received a USB element. The type of element
360 * is determined and is then passed to an appropriate handler function.
361 * Context: softirq-serialized
362 */
363void oz_usb_rx(struct oz_pd *pd, struct oz_elt *elt)
364{
365 struct oz_usb_hdr *usb_hdr = (struct oz_usb_hdr *)(elt + 1);
366 struct oz_usb_ctx *usb_ctx;
367
368 spin_lock_bh(&pd->app_lock[OZ_APPID_USB-1]);
369 usb_ctx = (struct oz_usb_ctx *)pd->app_ctx[OZ_APPID_USB-1];
370 if (usb_ctx)
371 oz_usb_get(usb_ctx);
372 spin_unlock_bh(&pd->app_lock[OZ_APPID_USB-1]);
Peter Huewe4d1b2fb2013-02-15 15:22:23 +0100373 if (usb_ctx == NULL)
Chris Kellyb3147862012-02-20 21:12:08 +0000374 return; /* Context has gone so nothing to do. */
375 if (usb_ctx->stopped)
376 goto done;
377 /* If sequence number is non-zero then check it is not a duplicate.
378 * Zero sequence numbers are always accepted.
379 */
380 if (usb_hdr->elt_seq_num != 0) {
381 if (((usb_ctx->rx_seq_num - usb_hdr->elt_seq_num) & 0x80) == 0)
382 /* Reject duplicate element. */
383 goto done;
384 }
385 usb_ctx->rx_seq_num = usb_hdr->elt_seq_num;
386 switch (usb_hdr->type) {
387 case OZ_GET_DESC_RSP: {
388 struct oz_get_desc_rsp *body =
389 (struct oz_get_desc_rsp *)usb_hdr;
390 int data_len = elt->length -
391 sizeof(struct oz_get_desc_rsp) + 1;
392 u16 offs = le16_to_cpu(get_unaligned(&body->offset));
393 u16 total_size =
394 le16_to_cpu(get_unaligned(&body->total_size));
Joe Perchesf724b582013-07-23 13:45:00 +0100395 oz_dbg(ON, "USB_REQ_GET_DESCRIPTOR - cnf\n");
Chris Kellyb3147862012-02-20 21:12:08 +0000396 oz_hcd_get_desc_cnf(usb_ctx->hport, body->req_id,
397 body->rcode, body->data,
398 data_len, offs, total_size);
399 }
400 break;
401 case OZ_SET_CONFIG_RSP: {
402 struct oz_set_config_rsp *body =
403 (struct oz_set_config_rsp *)usb_hdr;
404 oz_hcd_control_cnf(usb_ctx->hport, body->req_id,
Peter Huewe4d1b2fb2013-02-15 15:22:23 +0100405 body->rcode, NULL, 0);
Chris Kellyb3147862012-02-20 21:12:08 +0000406 }
407 break;
408 case OZ_SET_INTERFACE_RSP: {
409 struct oz_set_interface_rsp *body =
410 (struct oz_set_interface_rsp *)usb_hdr;
411 oz_hcd_control_cnf(usb_ctx->hport,
Peter Huewe4d1b2fb2013-02-15 15:22:23 +0100412 body->req_id, body->rcode, NULL, 0);
Chris Kellyb3147862012-02-20 21:12:08 +0000413 }
414 break;
415 case OZ_VENDOR_CLASS_RSP: {
416 struct oz_vendor_class_rsp *body =
417 (struct oz_vendor_class_rsp *)usb_hdr;
418 oz_hcd_control_cnf(usb_ctx->hport, body->req_id,
419 body->rcode, body->data, elt->length-
420 sizeof(struct oz_vendor_class_rsp)+1);
421 }
422 break;
423 case OZ_USB_ENDPOINT_DATA:
424 oz_usb_handle_ep_data(usb_ctx, usb_hdr, elt->length);
425 break;
426 }
427done:
428 oz_usb_put(usb_ctx);
429}
Rupesh Gujare6e244a82013-08-13 18:24:22 +0100430
Rupesh Gujare4e7fb822013-08-23 16:11:02 +0100431/*
Chris Kellyb3147862012-02-20 21:12:08 +0000432 * Context: softirq, process
433 */
434void oz_usb_farewell(struct oz_pd *pd, u8 ep_num, u8 *data, u8 len)
435{
436 struct oz_usb_ctx *usb_ctx;
Rupesh Gujare18f81912013-08-13 18:24:21 +0100437
Chris Kellyb3147862012-02-20 21:12:08 +0000438 spin_lock_bh(&pd->app_lock[OZ_APPID_USB-1]);
439 usb_ctx = (struct oz_usb_ctx *)pd->app_ctx[OZ_APPID_USB-1];
440 if (usb_ctx)
441 oz_usb_get(usb_ctx);
442 spin_unlock_bh(&pd->app_lock[OZ_APPID_USB-1]);
Peter Huewe4d1b2fb2013-02-15 15:22:23 +0100443 if (usb_ctx == NULL)
Chris Kellyb3147862012-02-20 21:12:08 +0000444 return; /* Context has gone so nothing to do. */
445 if (!usb_ctx->stopped) {
Joe Perchesf724b582013-07-23 13:45:00 +0100446 oz_dbg(ON, "Farewell indicated ep = 0x%x\n", ep_num);
Chris Kellyb3147862012-02-20 21:12:08 +0000447 oz_hcd_data_ind(usb_ctx->hport, ep_num, data, len);
448 }
449 oz_usb_put(usb_ctx);
450}