1 /* -----------------------------------------------------------------------------
2 * Copyright (c) 2011 Ozmo Inc
3 * Released under the GNU General Public License Version 2 (GPLv2).
5 * This file implements the protocol specific parts of the USB service for a PD.
6 * -----------------------------------------------------------------------------
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>
17 #include "ozprotocol.h"
26 /*------------------------------------------------------------------------------
28 #define MAX_ISOC_FIXED_DATA (253-sizeof(struct oz_isoc_fixed))
29 /*------------------------------------------------------------------------------
32 static int oz_usb_submit_elt(struct oz_elt_buf
*eb
, struct oz_elt_info
*ei
,
33 struct oz_usb_ctx
*usb_ctx
, u8 strid
, u8 isoc
)
36 struct oz_elt
*elt
= (struct oz_elt
*)ei
->data
;
37 struct oz_app_hdr
*app_hdr
= (struct oz_app_hdr
*)(elt
+1);
38 elt
->type
= OZ_ELT_APP_DATA
;
39 ei
->app_id
= OZ_APPID_USB
;
40 ei
->length
= elt
->length
+ sizeof(struct oz_elt
);
41 app_hdr
->app_id
= OZ_APPID_USB
;
42 spin_lock_bh(&eb
->lock
);
44 app_hdr
->elt_seq_num
= usb_ctx
->tx_seq_num
++;
45 if (usb_ctx
->tx_seq_num
== 0)
46 usb_ctx
->tx_seq_num
= 1;
48 ret
= oz_queue_elt_info(eb
, isoc
, strid
, ei
);
50 oz_elt_info_free(eb
, ei
);
51 spin_unlock_bh(&eb
->lock
);
54 /*------------------------------------------------------------------------------
57 int 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
)
60 struct oz_usb_ctx
*usb_ctx
= (struct oz_usb_ctx
*)hpd
;
61 struct oz_pd
*pd
= usb_ctx
->pd
;
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
);
66 oz_trace(" req_type = 0x%x\n", req_type
);
67 oz_trace(" desc_type = 0x%x\n", desc_type
);
68 oz_trace(" index = 0x%x\n", index
);
69 oz_trace(" windex = 0x%x\n", windex
);
70 oz_trace(" offset = 0x%x\n", offset
);
71 oz_trace(" len = 0x%x\n", len
);
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
;
87 return oz_usb_submit_elt(eb
, ei
, usb_ctx
, 0, 0);
89 /*------------------------------------------------------------------------------
92 static int oz_usb_set_config_req(void *hpd
, u8 req_id
, u8 index
)
94 struct oz_usb_ctx
*usb_ctx
= (struct oz_usb_ctx
*)hpd
;
95 struct oz_pd
*pd
= usb_ctx
->pd
;
97 struct oz_elt_buf
*eb
= &pd
->elt_buff
;
98 struct oz_elt_info
*ei
= oz_elt_info_alloc(&pd
->elt_buff
);
99 struct oz_set_config_req
*body
;
102 elt
= (struct oz_elt
*)ei
->data
;
103 elt
->length
= sizeof(struct oz_set_config_req
);
104 body
= (struct oz_set_config_req
*)(elt
+1);
105 body
->type
= OZ_SET_CONFIG_REQ
;
106 body
->req_id
= req_id
;
108 return oz_usb_submit_elt(eb
, ei
, usb_ctx
, 0, 0);
110 /*------------------------------------------------------------------------------
113 static int oz_usb_set_interface_req(void *hpd
, u8 req_id
, u8 index
, u8 alt
)
115 struct oz_usb_ctx
*usb_ctx
= (struct oz_usb_ctx
*)hpd
;
116 struct oz_pd
*pd
= usb_ctx
->pd
;
118 struct oz_elt_buf
*eb
= &pd
->elt_buff
;
119 struct oz_elt_info
*ei
= oz_elt_info_alloc(&pd
->elt_buff
);
120 struct oz_set_interface_req
*body
;
123 elt
= (struct oz_elt
*)ei
->data
;
124 elt
->length
= sizeof(struct oz_set_interface_req
);
125 body
= (struct oz_set_interface_req
*)(elt
+1);
126 body
->type
= OZ_SET_INTERFACE_REQ
;
127 body
->req_id
= req_id
;
129 body
->alternative
= alt
;
130 return oz_usb_submit_elt(eb
, ei
, usb_ctx
, 0, 0);
132 /*------------------------------------------------------------------------------
135 static int oz_usb_set_clear_feature_req(void *hpd
, u8 req_id
, u8 type
,
136 u8 recipient
, u8 index
, __le16 feature
)
138 struct oz_usb_ctx
*usb_ctx
= (struct oz_usb_ctx
*)hpd
;
139 struct oz_pd
*pd
= usb_ctx
->pd
;
141 struct oz_elt_buf
*eb
= &pd
->elt_buff
;
142 struct oz_elt_info
*ei
= oz_elt_info_alloc(&pd
->elt_buff
);
143 struct oz_feature_req
*body
;
146 elt
= (struct oz_elt
*)ei
->data
;
147 elt
->length
= sizeof(struct oz_feature_req
);
148 body
= (struct oz_feature_req
*)(elt
+1);
150 body
->req_id
= req_id
;
151 body
->recipient
= recipient
;
153 put_unaligned(feature
, &body
->feature
);
154 return oz_usb_submit_elt(eb
, ei
, usb_ctx
, 0, 0);
156 /*------------------------------------------------------------------------------
159 static int oz_usb_vendor_class_req(void *hpd
, u8 req_id
, u8 req_type
,
160 u8 request
, __le16 value
, __le16 index
, const u8
*data
, int data_len
)
162 struct oz_usb_ctx
*usb_ctx
= (struct oz_usb_ctx
*)hpd
;
163 struct oz_pd
*pd
= usb_ctx
->pd
;
165 struct oz_elt_buf
*eb
= &pd
->elt_buff
;
166 struct oz_elt_info
*ei
= oz_elt_info_alloc(&pd
->elt_buff
);
167 struct oz_vendor_class_req
*body
;
170 elt
= (struct oz_elt
*)ei
->data
;
171 elt
->length
= sizeof(struct oz_vendor_class_req
) - 1 + data_len
;
172 body
= (struct oz_vendor_class_req
*)(elt
+1);
173 body
->type
= OZ_VENDOR_CLASS_REQ
;
174 body
->req_id
= req_id
;
175 body
->req_type
= req_type
;
176 body
->request
= request
;
177 put_unaligned(value
, &body
->value
);
178 put_unaligned(index
, &body
->index
);
180 memcpy(body
->data
, data
, data_len
);
181 return oz_usb_submit_elt(eb
, ei
, usb_ctx
, 0, 0);
183 /*------------------------------------------------------------------------------
186 int oz_usb_control_req(void *hpd
, u8 req_id
, struct usb_ctrlrequest
*setup
,
187 const u8
*data
, int data_len
)
189 unsigned wvalue
= le16_to_cpu(setup
->wValue
);
190 unsigned windex
= le16_to_cpu(setup
->wIndex
);
191 unsigned wlength
= le16_to_cpu(setup
->wLength
);
193 oz_event_log(OZ_EVT_CTRL_REQ
, setup
->bRequest
, req_id
,
194 (void *)(((unsigned long)(setup
->wValue
))<<16 |
195 ((unsigned long)setup
->wIndex
)),
196 setup
->bRequestType
);
197 if ((setup
->bRequestType
& USB_TYPE_MASK
) == USB_TYPE_STANDARD
) {
198 switch (setup
->bRequest
) {
199 case USB_REQ_GET_DESCRIPTOR
:
200 rc
= oz_usb_get_desc_req(hpd
, req_id
,
201 setup
->bRequestType
, (u8
)(wvalue
>>8),
202 (u8
)wvalue
, setup
->wIndex
, 0, wlength
);
204 case USB_REQ_SET_CONFIGURATION
:
205 rc
= oz_usb_set_config_req(hpd
, req_id
, (u8
)wvalue
);
207 case USB_REQ_SET_INTERFACE
: {
208 u8 if_num
= (u8
)windex
;
210 rc
= oz_usb_set_interface_req(hpd
, req_id
,
214 case USB_REQ_SET_FEATURE
:
215 rc
= oz_usb_set_clear_feature_req(hpd
, req_id
,
217 setup
->bRequestType
& 0xf, (u8
)windex
,
220 case USB_REQ_CLEAR_FEATURE
:
221 rc
= oz_usb_set_clear_feature_req(hpd
, req_id
,
222 OZ_CLEAR_FEATURE_REQ
,
223 setup
->bRequestType
& 0xf,
224 (u8
)windex
, setup
->wValue
);
228 rc
= oz_usb_vendor_class_req(hpd
, req_id
, setup
->bRequestType
,
229 setup
->bRequest
, setup
->wValue
, setup
->wIndex
,
234 /*------------------------------------------------------------------------------
237 int oz_usb_send_isoc(void *hpd
, u8 ep_num
, struct urb
*urb
)
239 struct oz_usb_ctx
*usb_ctx
= (struct oz_usb_ctx
*)hpd
;
240 struct oz_pd
*pd
= usb_ctx
->pd
;
241 struct oz_elt_buf
*eb
;
245 struct usb_iso_packet_descriptor
*desc
;
247 if (pd
->mode
& OZ_F_ISOC_NO_ELTS
) {
248 for (i
= 0; i
< urb
->number_of_packets
; i
++) {
250 desc
= &urb
->iso_frame_desc
[i
];
251 data
= ((u8
*)urb
->transfer_buffer
)+desc
->offset
;
252 oz_send_isoc_unit(pd
, ep_num
, data
, desc
->length
);
257 hdr_size
= sizeof(struct oz_isoc_fixed
) - 1;
260 while (i
< urb
->number_of_packets
) {
261 struct oz_elt_info
*ei
= oz_elt_info_alloc(eb
);
263 struct oz_isoc_fixed
*body
;
269 rem
= MAX_ISOC_FIXED_DATA
;
270 elt
= (struct oz_elt
*)ei
->data
;
271 body
= (struct oz_isoc_fixed
*)(elt
+ 1);
272 body
->type
= OZ_USB_ENDPOINT_DATA
;
273 body
->endpoint
= ep_num
;
274 body
->format
= OZ_DATA_F_ISOC_FIXED
;
275 unit_size
= urb
->iso_frame_desc
[i
].length
;
276 body
->unit_size
= (u8
)unit_size
;
277 data
= ((u8
*)(elt
+1)) + hdr_size
;
279 while (i
< urb
->number_of_packets
) {
280 desc
= &urb
->iso_frame_desc
[i
];
281 if ((unit_size
== desc
->length
) &&
282 (desc
->length
<= rem
)) {
283 memcpy(data
, ((u8
*)urb
->transfer_buffer
) +
284 desc
->offset
, unit_size
);
289 desc
->actual_length
= desc
->length
;
295 elt
->length
= hdr_size
+ MAX_ISOC_FIXED_DATA
- rem
;
296 /* Store the number of units in body->frame_number for the
297 * moment. This field will be correctly determined before
298 * the element is sent. */
299 body
->frame_number
= (u8
)unit_count
;
300 oz_usb_submit_elt(eb
, ei
, usb_ctx
, ep_num
,
301 pd
->mode
& OZ_F_ISOC_ANYTIME
);
305 /*------------------------------------------------------------------------------
306 * Context: softirq-serialized
308 static void oz_usb_handle_ep_data(struct oz_usb_ctx
*usb_ctx
,
309 struct oz_usb_hdr
*usb_hdr
, int len
)
311 struct oz_data
*data_hdr
= (struct oz_data
*)usb_hdr
;
312 switch (data_hdr
->format
) {
313 case OZ_DATA_F_MULTIPLE_FIXED
: {
314 struct oz_multiple_fixed
*body
=
315 (struct oz_multiple_fixed
*)data_hdr
;
316 u8
*data
= body
->data
;
317 int n
= (len
- sizeof(struct oz_multiple_fixed
)+1)
320 oz_hcd_data_ind(usb_ctx
->hport
, body
->endpoint
,
321 data
, body
->unit_size
);
322 data
+= body
->unit_size
;
326 case OZ_DATA_F_ISOC_FIXED
: {
327 struct oz_isoc_fixed
*body
=
328 (struct oz_isoc_fixed
*)data_hdr
;
329 int data_len
= len
-sizeof(struct oz_isoc_fixed
)+1;
330 int unit_size
= body
->unit_size
;
331 u8
*data
= body
->data
;
336 count
= data_len
/unit_size
;
337 for (i
= 0; i
< count
; i
++) {
338 oz_hcd_data_ind(usb_ctx
->hport
,
339 body
->endpoint
, data
, unit_size
);
347 /*------------------------------------------------------------------------------
348 * This is called when the PD has received a USB element. The type of element
349 * is determined and is then passed to an appropriate handler function.
350 * Context: softirq-serialized
352 void oz_usb_rx(struct oz_pd
*pd
, struct oz_elt
*elt
)
354 struct oz_usb_hdr
*usb_hdr
= (struct oz_usb_hdr
*)(elt
+ 1);
355 struct oz_usb_ctx
*usb_ctx
;
357 spin_lock_bh(&pd
->app_lock
[OZ_APPID_USB
-1]);
358 usb_ctx
= (struct oz_usb_ctx
*)pd
->app_ctx
[OZ_APPID_USB
-1];
361 spin_unlock_bh(&pd
->app_lock
[OZ_APPID_USB
-1]);
363 return; /* Context has gone so nothing to do. */
364 if (usb_ctx
->stopped
)
366 /* If sequence number is non-zero then check it is not a duplicate.
367 * Zero sequence numbers are always accepted.
369 if (usb_hdr
->elt_seq_num
!= 0) {
370 if (((usb_ctx
->rx_seq_num
- usb_hdr
->elt_seq_num
) & 0x80) == 0)
371 /* Reject duplicate element. */
374 usb_ctx
->rx_seq_num
= usb_hdr
->elt_seq_num
;
375 switch (usb_hdr
->type
) {
376 case OZ_GET_DESC_RSP
: {
377 struct oz_get_desc_rsp
*body
=
378 (struct oz_get_desc_rsp
*)usb_hdr
;
379 int data_len
= elt
->length
-
380 sizeof(struct oz_get_desc_rsp
) + 1;
381 u16 offs
= le16_to_cpu(get_unaligned(&body
->offset
));
383 le16_to_cpu(get_unaligned(&body
->total_size
));
384 oz_trace("USB_REQ_GET_DESCRIPTOR - cnf\n");
385 oz_hcd_get_desc_cnf(usb_ctx
->hport
, body
->req_id
,
386 body
->rcode
, body
->data
,
387 data_len
, offs
, total_size
);
390 case OZ_SET_CONFIG_RSP
: {
391 struct oz_set_config_rsp
*body
=
392 (struct oz_set_config_rsp
*)usb_hdr
;
393 oz_hcd_control_cnf(usb_ctx
->hport
, body
->req_id
,
394 body
->rcode
, NULL
, 0);
397 case OZ_SET_INTERFACE_RSP
: {
398 struct oz_set_interface_rsp
*body
=
399 (struct oz_set_interface_rsp
*)usb_hdr
;
400 oz_hcd_control_cnf(usb_ctx
->hport
,
401 body
->req_id
, body
->rcode
, NULL
, 0);
404 case OZ_VENDOR_CLASS_RSP
: {
405 struct oz_vendor_class_rsp
*body
=
406 (struct oz_vendor_class_rsp
*)usb_hdr
;
407 oz_hcd_control_cnf(usb_ctx
->hport
, body
->req_id
,
408 body
->rcode
, body
->data
, elt
->length
-
409 sizeof(struct oz_vendor_class_rsp
)+1);
412 case OZ_USB_ENDPOINT_DATA
:
413 oz_usb_handle_ep_data(usb_ctx
, usb_hdr
, elt
->length
);
419 /*------------------------------------------------------------------------------
420 * Context: softirq, process
422 void oz_usb_farewell(struct oz_pd
*pd
, u8 ep_num
, u8
*data
, u8 len
)
424 struct oz_usb_ctx
*usb_ctx
;
425 spin_lock_bh(&pd
->app_lock
[OZ_APPID_USB
-1]);
426 usb_ctx
= (struct oz_usb_ctx
*)pd
->app_ctx
[OZ_APPID_USB
-1];
429 spin_unlock_bh(&pd
->app_lock
[OZ_APPID_USB
-1]);
431 return; /* Context has gone so nothing to do. */
432 if (!usb_ctx
->stopped
) {
433 oz_trace("Farewell indicated ep = 0x%x\n", ep_num
);
434 oz_hcd_data_ind(usb_ctx
->hport
, ep_num
, data
, len
);