blob: f660bb198c65534a6cbe8183d3f5d0a30a532eb1 [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 */
Chris Kellyb3147862012-02-20 21:12:08 +00008#include <linux/module.h>
9#include <linux/timer.h>
10#include <linux/sched.h>
11#include <linux/netdevice.h>
12#include <linux/errno.h>
13#include <linux/input.h>
14#include <asm/unaligned.h>
Joe Perchesf724b582013-07-23 13:45:00 +010015#include "ozdbg.h"
Chris Kellyb3147862012-02-20 21:12:08 +000016#include "ozprotocol.h"
17#include "ozeltbuf.h"
18#include "ozpd.h"
19#include "ozproto.h"
20#include "ozusbif.h"
21#include "ozhcd.h"
Chris Kellyb3147862012-02-20 21:12:08 +000022#include "ozusbsvc.h"
Rupesh Gujare6e244a82013-08-13 18:24:22 +010023
Chris Kellyb3147862012-02-20 21:12:08 +000024#define MAX_ISOC_FIXED_DATA (253-sizeof(struct oz_isoc_fixed))
Rupesh Gujare6e244a82013-08-13 18:24:22 +010025
Rupesh Gujare4e7fb822013-08-23 16:11:02 +010026/*
Chris Kellyb3147862012-02-20 21:12:08 +000027 * Context: softirq
28 */
29static int oz_usb_submit_elt(struct oz_elt_buf *eb, struct oz_elt_info *ei,
30 struct oz_usb_ctx *usb_ctx, u8 strid, u8 isoc)
31{
32 int ret;
33 struct oz_elt *elt = (struct oz_elt *)ei->data;
34 struct oz_app_hdr *app_hdr = (struct oz_app_hdr *)(elt+1);
Rupesh Gujare18f81912013-08-13 18:24:21 +010035
Chris Kellyb3147862012-02-20 21:12:08 +000036 elt->type = OZ_ELT_APP_DATA;
37 ei->app_id = OZ_APPID_USB;
38 ei->length = elt->length + sizeof(struct oz_elt);
39 app_hdr->app_id = OZ_APPID_USB;
40 spin_lock_bh(&eb->lock);
41 if (isoc == 0) {
42 app_hdr->elt_seq_num = usb_ctx->tx_seq_num++;
43 if (usb_ctx->tx_seq_num == 0)
44 usb_ctx->tx_seq_num = 1;
45 }
46 ret = oz_queue_elt_info(eb, isoc, strid, ei);
47 if (ret)
48 oz_elt_info_free(eb, ei);
49 spin_unlock_bh(&eb->lock);
50 return ret;
51}
Rupesh Gujare6e244a82013-08-13 18:24:22 +010052
Rupesh Gujare4e7fb822013-08-23 16:11:02 +010053/*
Chris Kellyb3147862012-02-20 21:12:08 +000054 * Context: softirq
55 */
56int oz_usb_get_desc_req(void *hpd, u8 req_id, u8 req_type, u8 desc_type,
Himangi Saraogi45b1fe52014-03-12 22:20:32 +053057 u8 index, __le16 windex, int offset, int len)
Chris Kellyb3147862012-02-20 21:12:08 +000058{
Tapasweni Pathakb3d43a32014-10-30 17:02:57 +053059 struct oz_usb_ctx *usb_ctx = hpd;
Chris Kellyb3147862012-02-20 21:12:08 +000060 struct oz_pd *pd = usb_ctx->pd;
61 struct oz_elt *elt;
62 struct oz_get_desc_req *body;
63 struct oz_elt_buf *eb = &pd->elt_buff;
64 struct oz_elt_info *ei = oz_elt_info_alloc(&pd->elt_buff);
Rupesh Gujare18f81912013-08-13 18:24:21 +010065
Joe Perchesf724b582013-07-23 13:45:00 +010066 oz_dbg(ON, " req_type = 0x%x\n", req_type);
67 oz_dbg(ON, " desc_type = 0x%x\n", desc_type);
68 oz_dbg(ON, " index = 0x%x\n", index);
69 oz_dbg(ON, " windex = 0x%x\n", windex);
70 oz_dbg(ON, " offset = 0x%x\n", offset);
71 oz_dbg(ON, " len = 0x%x\n", len);
Chris Kellyb3147862012-02-20 21:12:08 +000072 if (len > 200)
73 len = 200;
Peter Huewe4d1b2fb2013-02-15 15:22:23 +010074 if (ei == NULL)
Chris Kellyb3147862012-02-20 21:12:08 +000075 return -1;
76 elt = (struct oz_elt *)ei->data;
77 elt->length = sizeof(struct oz_get_desc_req);
78 body = (struct oz_get_desc_req *)(elt+1);
79 body->type = OZ_GET_DESC_REQ;
80 body->req_id = req_id;
81 put_unaligned(cpu_to_le16(offset), &body->offset);
82 put_unaligned(cpu_to_le16(len), &body->size);
83 body->req_type = req_type;
84 body->desc_type = desc_type;
85 body->w_index = windex;
86 body->index = index;
87 return oz_usb_submit_elt(eb, ei, usb_ctx, 0, 0);
88}
Rupesh Gujare6e244a82013-08-13 18:24:22 +010089
Rupesh Gujare4e7fb822013-08-23 16:11:02 +010090/*
Chris Kellyb3147862012-02-20 21:12:08 +000091 * Context: tasklet
92 */
93static int oz_usb_set_config_req(void *hpd, u8 req_id, u8 index)
94{
Tapasweni Pathakb3d43a32014-10-30 17:02:57 +053095 struct oz_usb_ctx *usb_ctx = hpd;
Chris Kellyb3147862012-02-20 21:12:08 +000096 struct oz_pd *pd = usb_ctx->pd;
97 struct oz_elt *elt;
98 struct oz_elt_buf *eb = &pd->elt_buff;
99 struct oz_elt_info *ei = oz_elt_info_alloc(&pd->elt_buff);
100 struct oz_set_config_req *body;
Rupesh Gujare18f81912013-08-13 18:24:21 +0100101
Peter Huewe4d1b2fb2013-02-15 15:22:23 +0100102 if (ei == NULL)
Chris Kellyb3147862012-02-20 21:12:08 +0000103 return -1;
104 elt = (struct oz_elt *)ei->data;
105 elt->length = sizeof(struct oz_set_config_req);
106 body = (struct oz_set_config_req *)(elt+1);
107 body->type = OZ_SET_CONFIG_REQ;
108 body->req_id = req_id;
109 body->index = index;
110 return oz_usb_submit_elt(eb, ei, usb_ctx, 0, 0);
111}
Rupesh Gujare6e244a82013-08-13 18:24:22 +0100112
Rupesh Gujare4e7fb822013-08-23 16:11:02 +0100113/*
Chris Kellyb3147862012-02-20 21:12:08 +0000114 * Context: tasklet
115 */
116static int oz_usb_set_interface_req(void *hpd, u8 req_id, u8 index, u8 alt)
117{
Tapasweni Pathakb3d43a32014-10-30 17:02:57 +0530118 struct oz_usb_ctx *usb_ctx = hpd;
Chris Kellyb3147862012-02-20 21:12:08 +0000119 struct oz_pd *pd = usb_ctx->pd;
120 struct oz_elt *elt;
121 struct oz_elt_buf *eb = &pd->elt_buff;
122 struct oz_elt_info *ei = oz_elt_info_alloc(&pd->elt_buff);
123 struct oz_set_interface_req *body;
Rupesh Gujare18f81912013-08-13 18:24:21 +0100124
Peter Huewe4d1b2fb2013-02-15 15:22:23 +0100125 if (ei == NULL)
Chris Kellyb3147862012-02-20 21:12:08 +0000126 return -1;
127 elt = (struct oz_elt *)ei->data;
128 elt->length = sizeof(struct oz_set_interface_req);
129 body = (struct oz_set_interface_req *)(elt+1);
130 body->type = OZ_SET_INTERFACE_REQ;
131 body->req_id = req_id;
132 body->index = index;
133 body->alternative = alt;
134 return oz_usb_submit_elt(eb, ei, usb_ctx, 0, 0);
135}
Rupesh Gujare6e244a82013-08-13 18:24:22 +0100136
Rupesh Gujare4e7fb822013-08-23 16:11:02 +0100137/*
Chris Kellyb3147862012-02-20 21:12:08 +0000138 * Context: tasklet
139 */
140static int oz_usb_set_clear_feature_req(void *hpd, u8 req_id, u8 type,
141 u8 recipient, u8 index, __le16 feature)
142{
Tapasweni Pathakb3d43a32014-10-30 17:02:57 +0530143 struct oz_usb_ctx *usb_ctx = hpd;
Chris Kellyb3147862012-02-20 21:12:08 +0000144 struct oz_pd *pd = usb_ctx->pd;
145 struct oz_elt *elt;
146 struct oz_elt_buf *eb = &pd->elt_buff;
147 struct oz_elt_info *ei = oz_elt_info_alloc(&pd->elt_buff);
148 struct oz_feature_req *body;
Rupesh Gujare18f81912013-08-13 18:24:21 +0100149
Peter Huewe4d1b2fb2013-02-15 15:22:23 +0100150 if (ei == NULL)
Chris Kellyb3147862012-02-20 21:12:08 +0000151 return -1;
152 elt = (struct oz_elt *)ei->data;
153 elt->length = sizeof(struct oz_feature_req);
154 body = (struct oz_feature_req *)(elt+1);
155 body->type = type;
156 body->req_id = req_id;
157 body->recipient = recipient;
158 body->index = index;
159 put_unaligned(feature, &body->feature);
160 return oz_usb_submit_elt(eb, ei, usb_ctx, 0, 0);
161}
Rupesh Gujare6e244a82013-08-13 18:24:22 +0100162
Rupesh Gujare4e7fb822013-08-23 16:11:02 +0100163/*
Chris Kellyb3147862012-02-20 21:12:08 +0000164 * Context: tasklet
165 */
166static int oz_usb_vendor_class_req(void *hpd, u8 req_id, u8 req_type,
Peter Huewedc7f5b32013-02-15 21:17:24 +0100167 u8 request, __le16 value, __le16 index, const u8 *data, int data_len)
Chris Kellyb3147862012-02-20 21:12:08 +0000168{
Tapasweni Pathakb3d43a32014-10-30 17:02:57 +0530169 struct oz_usb_ctx *usb_ctx = hpd;
Chris Kellyb3147862012-02-20 21:12:08 +0000170 struct oz_pd *pd = usb_ctx->pd;
171 struct oz_elt *elt;
172 struct oz_elt_buf *eb = &pd->elt_buff;
173 struct oz_elt_info *ei = oz_elt_info_alloc(&pd->elt_buff);
174 struct oz_vendor_class_req *body;
Rupesh Gujare18f81912013-08-13 18:24:21 +0100175
Peter Huewe4d1b2fb2013-02-15 15:22:23 +0100176 if (ei == NULL)
Chris Kellyb3147862012-02-20 21:12:08 +0000177 return -1;
178 elt = (struct oz_elt *)ei->data;
179 elt->length = sizeof(struct oz_vendor_class_req) - 1 + data_len;
180 body = (struct oz_vendor_class_req *)(elt+1);
181 body->type = OZ_VENDOR_CLASS_REQ;
182 body->req_id = req_id;
183 body->req_type = req_type;
184 body->request = request;
185 put_unaligned(value, &body->value);
186 put_unaligned(index, &body->index);
187 if (data_len)
188 memcpy(body->data, data, data_len);
189 return oz_usb_submit_elt(eb, ei, usb_ctx, 0, 0);
190}
Rupesh Gujare6e244a82013-08-13 18:24:22 +0100191
Rupesh Gujare4e7fb822013-08-23 16:11:02 +0100192/*
Chris Kellyb3147862012-02-20 21:12:08 +0000193 * Context: tasklet
194 */
195int oz_usb_control_req(void *hpd, u8 req_id, struct usb_ctrlrequest *setup,
Peter Huewedc7f5b32013-02-15 21:17:24 +0100196 const u8 *data, int data_len)
Chris Kellyb3147862012-02-20 21:12:08 +0000197{
198 unsigned wvalue = le16_to_cpu(setup->wValue);
199 unsigned windex = le16_to_cpu(setup->wIndex);
200 unsigned wlength = le16_to_cpu(setup->wLength);
201 int rc = 0;
Rupesh Gujare18f81912013-08-13 18:24:21 +0100202
Chris Kellyb3147862012-02-20 21:12:08 +0000203 if ((setup->bRequestType & USB_TYPE_MASK) == USB_TYPE_STANDARD) {
204 switch (setup->bRequest) {
205 case USB_REQ_GET_DESCRIPTOR:
206 rc = oz_usb_get_desc_req(hpd, req_id,
207 setup->bRequestType, (u8)(wvalue>>8),
208 (u8)wvalue, setup->wIndex, 0, wlength);
209 break;
210 case USB_REQ_SET_CONFIGURATION:
211 rc = oz_usb_set_config_req(hpd, req_id, (u8)wvalue);
212 break;
213 case USB_REQ_SET_INTERFACE: {
214 u8 if_num = (u8)windex;
215 u8 alt = (u8)wvalue;
Adrian Nicoarace6880e2014-09-08 14:39:58 -0400216
Chris Kellyb3147862012-02-20 21:12:08 +0000217 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{
Tapasweni Pathakb3d43a32014-10-30 17:02:57 +0530247 struct oz_usb_ctx *usb_ctx = hpd;
Chris Kellyb3147862012-02-20 21:12:08 +0000248 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;
Adrian Nicoarace6880e2014-09-08 14:39:58 -0400258
Chris Kellyb3147862012-02-20 21:12:08 +0000259 desc = &urb->iso_frame_desc[i];
260 data = ((u8 *)urb->transfer_buffer)+desc->offset;
261 oz_send_isoc_unit(pd, ep_num, data, desc->length);
262 }
263 return 0;
264 }
265
266 hdr_size = sizeof(struct oz_isoc_fixed) - 1;
267 eb = &pd->elt_buff;
268 i = 0;
269 while (i < urb->number_of_packets) {
270 struct oz_elt_info *ei = oz_elt_info_alloc(eb);
271 struct oz_elt *elt;
272 struct oz_isoc_fixed *body;
273 int unit_count;
274 int unit_size;
275 int rem;
Adrian Nicoarace6880e2014-09-08 14:39:58 -0400276
Peter Huewe4d1b2fb2013-02-15 15:22:23 +0100277 if (ei == NULL)
Chris Kellyb3147862012-02-20 21:12:08 +0000278 return -1;
279 rem = MAX_ISOC_FIXED_DATA;
280 elt = (struct oz_elt *)ei->data;
281 body = (struct oz_isoc_fixed *)(elt + 1);
282 body->type = OZ_USB_ENDPOINT_DATA;
283 body->endpoint = ep_num;
284 body->format = OZ_DATA_F_ISOC_FIXED;
285 unit_size = urb->iso_frame_desc[i].length;
286 body->unit_size = (u8)unit_size;
287 data = ((u8 *)(elt+1)) + hdr_size;
288 unit_count = 0;
289 while (i < urb->number_of_packets) {
290 desc = &urb->iso_frame_desc[i];
291 if ((unit_size == desc->length) &&
292 (desc->length <= rem)) {
293 memcpy(data, ((u8 *)urb->transfer_buffer) +
294 desc->offset, unit_size);
295 data += unit_size;
296 rem -= unit_size;
297 unit_count++;
298 desc->status = 0;
299 desc->actual_length = desc->length;
300 i++;
301 } else {
302 break;
303 }
304 }
305 elt->length = hdr_size + MAX_ISOC_FIXED_DATA - rem;
306 /* Store the number of units in body->frame_number for the
307 * moment. This field will be correctly determined before
308 * the element is sent. */
309 body->frame_number = (u8)unit_count;
310 oz_usb_submit_elt(eb, ei, usb_ctx, ep_num,
311 pd->mode & OZ_F_ISOC_ANYTIME);
312 }
313 return 0;
314}
Rupesh Gujare6e244a82013-08-13 18:24:22 +0100315
Rupesh Gujare4e7fb822013-08-23 16:11:02 +0100316/*
Chris Kellyb3147862012-02-20 21:12:08 +0000317 * Context: softirq-serialized
318 */
Peter Huewea7f74c32013-02-15 21:17:22 +0100319static void oz_usb_handle_ep_data(struct oz_usb_ctx *usb_ctx,
Chris Kellyb3147862012-02-20 21:12:08 +0000320 struct oz_usb_hdr *usb_hdr, int len)
321{
322 struct oz_data *data_hdr = (struct oz_data *)usb_hdr;
Rupesh Gujare18f81912013-08-13 18:24:21 +0100323
Chris Kellyb3147862012-02-20 21:12:08 +0000324 switch (data_hdr->format) {
325 case OZ_DATA_F_MULTIPLE_FIXED: {
326 struct oz_multiple_fixed *body =
327 (struct oz_multiple_fixed *)data_hdr;
328 u8 *data = body->data;
Jason A. Donenfeld9a590292015-05-29 13:07:01 +0200329 unsigned int n;
330 if (!body->unit_size ||
331 len < sizeof(struct oz_multiple_fixed) - 1)
Jason A. Donenfeld04bf4642015-05-29 13:07:00 +0200332 break;
Jason A. Donenfeld9a590292015-05-29 13:07:01 +0200333 n = (len - (sizeof(struct oz_multiple_fixed) - 1))
Chris Kellyb3147862012-02-20 21:12:08 +0000334 / body->unit_size;
335 while (n--) {
336 oz_hcd_data_ind(usb_ctx->hport, body->endpoint,
337 data, body->unit_size);
338 data += body->unit_size;
339 }
340 }
341 break;
342 case OZ_DATA_F_ISOC_FIXED: {
343 struct oz_isoc_fixed *body =
344 (struct oz_isoc_fixed *)data_hdr;
345 int data_len = len-sizeof(struct oz_isoc_fixed)+1;
346 int unit_size = body->unit_size;
347 u8 *data = body->data;
348 int count;
349 int i;
Adrian Nicoarace6880e2014-09-08 14:39:58 -0400350
Chris Kellyb3147862012-02-20 21:12:08 +0000351 if (!unit_size)
352 break;
353 count = data_len/unit_size;
354 for (i = 0; i < count; i++) {
355 oz_hcd_data_ind(usb_ctx->hport,
356 body->endpoint, data, unit_size);
357 data += unit_size;
358 }
359 }
360 break;
361 }
362
363}
Rupesh Gujare6e244a82013-08-13 18:24:22 +0100364
Rupesh Gujare4e7fb822013-08-23 16:11:02 +0100365/*
Chris Kellyb3147862012-02-20 21:12:08 +0000366 * This is called when the PD has received a USB element. The type of element
367 * is determined and is then passed to an appropriate handler function.
368 * Context: softirq-serialized
369 */
370void oz_usb_rx(struct oz_pd *pd, struct oz_elt *elt)
371{
372 struct oz_usb_hdr *usb_hdr = (struct oz_usb_hdr *)(elt + 1);
373 struct oz_usb_ctx *usb_ctx;
374
Christoph Jaegera9686e72014-08-04 14:54:52 +0200375 spin_lock_bh(&pd->app_lock[OZ_APPID_USB]);
376 usb_ctx = (struct oz_usb_ctx *)pd->app_ctx[OZ_APPID_USB];
Chris Kellyb3147862012-02-20 21:12:08 +0000377 if (usb_ctx)
378 oz_usb_get(usb_ctx);
Christoph Jaegera9686e72014-08-04 14:54:52 +0200379 spin_unlock_bh(&pd->app_lock[OZ_APPID_USB]);
Peter Huewe4d1b2fb2013-02-15 15:22:23 +0100380 if (usb_ctx == NULL)
Chris Kellyb3147862012-02-20 21:12:08 +0000381 return; /* Context has gone so nothing to do. */
382 if (usb_ctx->stopped)
383 goto done;
384 /* If sequence number is non-zero then check it is not a duplicate.
385 * Zero sequence numbers are always accepted.
386 */
387 if (usb_hdr->elt_seq_num != 0) {
388 if (((usb_ctx->rx_seq_num - usb_hdr->elt_seq_num) & 0x80) == 0)
389 /* Reject duplicate element. */
390 goto done;
391 }
392 usb_ctx->rx_seq_num = usb_hdr->elt_seq_num;
393 switch (usb_hdr->type) {
394 case OZ_GET_DESC_RSP: {
395 struct oz_get_desc_rsp *body =
396 (struct oz_get_desc_rsp *)usb_hdr;
Jason A. Donenfeldd114b9f2015-05-29 13:06:58 +0200397 u16 offs, total_size;
398 u8 data_len;
399
400 if (elt->length < sizeof(struct oz_get_desc_rsp) - 1)
401 break;
402 data_len = elt->length -
403 (sizeof(struct oz_get_desc_rsp) - 1);
404 offs = le16_to_cpu(get_unaligned(&body->offset));
405 total_size =
Chris Kellyb3147862012-02-20 21:12:08 +0000406 le16_to_cpu(get_unaligned(&body->total_size));
Joe Perchesf724b582013-07-23 13:45:00 +0100407 oz_dbg(ON, "USB_REQ_GET_DESCRIPTOR - cnf\n");
Chris Kellyb3147862012-02-20 21:12:08 +0000408 oz_hcd_get_desc_cnf(usb_ctx->hport, body->req_id,
409 body->rcode, body->data,
410 data_len, offs, total_size);
411 }
412 break;
413 case OZ_SET_CONFIG_RSP: {
414 struct oz_set_config_rsp *body =
415 (struct oz_set_config_rsp *)usb_hdr;
416 oz_hcd_control_cnf(usb_ctx->hport, body->req_id,
Peter Huewe4d1b2fb2013-02-15 15:22:23 +0100417 body->rcode, NULL, 0);
Chris Kellyb3147862012-02-20 21:12:08 +0000418 }
419 break;
420 case OZ_SET_INTERFACE_RSP: {
421 struct oz_set_interface_rsp *body =
422 (struct oz_set_interface_rsp *)usb_hdr;
423 oz_hcd_control_cnf(usb_ctx->hport,
Peter Huewe4d1b2fb2013-02-15 15:22:23 +0100424 body->req_id, body->rcode, NULL, 0);
Chris Kellyb3147862012-02-20 21:12:08 +0000425 }
426 break;
427 case OZ_VENDOR_CLASS_RSP: {
428 struct oz_vendor_class_rsp *body =
429 (struct oz_vendor_class_rsp *)usb_hdr;
430 oz_hcd_control_cnf(usb_ctx->hport, body->req_id,
431 body->rcode, body->data, elt->length-
432 sizeof(struct oz_vendor_class_rsp)+1);
433 }
434 break;
435 case OZ_USB_ENDPOINT_DATA:
436 oz_usb_handle_ep_data(usb_ctx, usb_hdr, elt->length);
437 break;
438 }
439done:
440 oz_usb_put(usb_ctx);
441}
Rupesh Gujare6e244a82013-08-13 18:24:22 +0100442
Rupesh Gujare4e7fb822013-08-23 16:11:02 +0100443/*
Chris Kellyb3147862012-02-20 21:12:08 +0000444 * Context: softirq, process
445 */
446void oz_usb_farewell(struct oz_pd *pd, u8 ep_num, u8 *data, u8 len)
447{
448 struct oz_usb_ctx *usb_ctx;
Rupesh Gujare18f81912013-08-13 18:24:21 +0100449
Christoph Jaegera9686e72014-08-04 14:54:52 +0200450 spin_lock_bh(&pd->app_lock[OZ_APPID_USB]);
451 usb_ctx = (struct oz_usb_ctx *)pd->app_ctx[OZ_APPID_USB];
Chris Kellyb3147862012-02-20 21:12:08 +0000452 if (usb_ctx)
453 oz_usb_get(usb_ctx);
Christoph Jaegera9686e72014-08-04 14:54:52 +0200454 spin_unlock_bh(&pd->app_lock[OZ_APPID_USB]);
Peter Huewe4d1b2fb2013-02-15 15:22:23 +0100455 if (usb_ctx == NULL)
Chris Kellyb3147862012-02-20 21:12:08 +0000456 return; /* Context has gone so nothing to do. */
457 if (!usb_ctx->stopped) {
Joe Perchesf724b582013-07-23 13:45:00 +0100458 oz_dbg(ON, "Farewell indicated ep = 0x%x\n", ep_num);
Chris Kellyb3147862012-02-20 21:12:08 +0000459 oz_hcd_data_ind(usb_ctx->hport, ep_num, data, len);
460 }
461 oz_usb_put(usb_ctx);
462}