blob: ef08fc5ce7cb60d2394abac868182bcd3fe7c58b [file] [log] [blame]
Bar Weinered1059e2013-04-16 11:23:55 +03001/*
2 * Copyright (c) 2011-2013, The Linux Foundation. All rights reserved.
3 *
4 * This program is free software; you can redistribute it and/or modify
5 * it under the terms of the GNU General Public License version 2 and
6 * only version 2 as published by the Free Software Foundation.
7 *
8 * This program is distributed in the hope that it will be useful,
9 * but WITHOUT ANY WARRANTY; without even the implied warranty of
10 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
11 * GNU General Public License for more details.
12 */
13
14#include <linux/slab.h>
15#include <linux/kernel.h>
16#include <linux/device.h>
17#include <linux/spinlock.h>
18
19#include <mach/usb_gadget_xport.h>
20
21#include "u_rmnet.h"
22#include "gadget_chips.h"
23
24#define GPS_NOTIFY_INTERVAL 5
25#define GPS_MAX_NOTIFY_SIZE 64
26
27
28#define ACM_CTRL_DTR (1 << 0)
29
30/* TODO: use separate structures for data and
31 * control paths
32 */
33struct f_gps {
34 struct grmnet port;
35 u8 port_num;
36 int ifc_id;
37 atomic_t online;
38 atomic_t ctrl_online;
39 struct usb_composite_dev *cdev;
40
41 spinlock_t lock;
42
43 /* usb eps */
44 struct usb_ep *notify;
45 struct usb_request *notify_req;
46
47 /* control info */
48 struct list_head cpkt_resp_q;
49 atomic_t notify_count;
50 unsigned long cpkts_len;
51};
52
53static struct gps_ports {
54 enum transport_type ctrl_xport;
55 struct f_gps *port;
56} gps_port;
57
58static struct usb_interface_descriptor gps_interface_desc = {
59 .bLength = USB_DT_INTERFACE_SIZE,
60 .bDescriptorType = USB_DT_INTERFACE,
61 .bNumEndpoints = 1,
62 .bInterfaceClass = USB_CLASS_VENDOR_SPEC,
63 .bInterfaceSubClass = USB_CLASS_VENDOR_SPEC,
64 .bInterfaceProtocol = USB_CLASS_VENDOR_SPEC,
65 /* .iInterface = DYNAMIC */
66};
67
68/* Full speed support */
69static struct usb_endpoint_descriptor gps_fs_notify_desc = {
70 .bLength = USB_DT_ENDPOINT_SIZE,
71 .bDescriptorType = USB_DT_ENDPOINT,
72 .bEndpointAddress = USB_DIR_IN,
73 .bmAttributes = USB_ENDPOINT_XFER_INT,
74 .wMaxPacketSize = __constant_cpu_to_le16(GPS_MAX_NOTIFY_SIZE),
75 .bInterval = 1 << GPS_NOTIFY_INTERVAL,
76};
77
78static struct usb_descriptor_header *gps_fs_function[] = {
79 (struct usb_descriptor_header *) &gps_interface_desc,
80 (struct usb_descriptor_header *) &gps_fs_notify_desc,
81 NULL,
82};
83
84/* High speed support */
85static struct usb_endpoint_descriptor gps_hs_notify_desc = {
86 .bLength = USB_DT_ENDPOINT_SIZE,
87 .bDescriptorType = USB_DT_ENDPOINT,
88 .bEndpointAddress = USB_DIR_IN,
89 .bmAttributes = USB_ENDPOINT_XFER_INT,
90 .wMaxPacketSize = __constant_cpu_to_le16(GPS_MAX_NOTIFY_SIZE),
91 .bInterval = GPS_NOTIFY_INTERVAL + 4,
92};
93
94static struct usb_descriptor_header *gps_hs_function[] = {
95 (struct usb_descriptor_header *) &gps_interface_desc,
96 (struct usb_descriptor_header *) &gps_hs_notify_desc,
97 NULL,
98};
99
100/* Super speed support */
101static struct usb_endpoint_descriptor gps_ss_notify_desc = {
102 .bLength = USB_DT_ENDPOINT_SIZE,
103 .bDescriptorType = USB_DT_ENDPOINT,
104 .bEndpointAddress = USB_DIR_IN,
105 .bmAttributes = USB_ENDPOINT_XFER_INT,
106 .wMaxPacketSize = __constant_cpu_to_le16(GPS_MAX_NOTIFY_SIZE),
107 .bInterval = GPS_NOTIFY_INTERVAL + 4,
108};
109
110static struct usb_ss_ep_comp_descriptor gps_ss_notify_comp_desc = {
111 .bLength = sizeof gps_ss_notify_comp_desc,
112 .bDescriptorType = USB_DT_SS_ENDPOINT_COMP,
113
114 /* the following 3 values can be tweaked if necessary */
115 /* .bMaxBurst = 0, */
116 /* .bmAttributes = 0, */
117 .wBytesPerInterval = cpu_to_le16(GPS_MAX_NOTIFY_SIZE),
118};
119
120static struct usb_descriptor_header *gps_ss_function[] = {
121 (struct usb_descriptor_header *) &gps_interface_desc,
122 (struct usb_descriptor_header *) &gps_ss_notify_desc,
123 (struct usb_descriptor_header *) &gps_ss_notify_comp_desc,
124 NULL,
125};
126
127/* String descriptors */
128
129static struct usb_string gps_string_defs[] = {
130 [0].s = "GPS",
131 { } /* end of list */
132};
133
134static struct usb_gadget_strings gps_string_table = {
135 .language = 0x0409, /* en-us */
136 .strings = gps_string_defs,
137};
138
139static struct usb_gadget_strings *gps_strings[] = {
140 &gps_string_table,
141 NULL,
142};
143
144static void gps_ctrl_response_available(struct f_gps *dev);
145
146/* ------- misc functions --------------------*/
147
148static inline struct f_gps *func_to_gps(struct usb_function *f)
149{
150 return container_of(f, struct f_gps, port.func);
151}
152
153static inline struct f_gps *port_to_gps(struct grmnet *r)
154{
155 return container_of(r, struct f_gps, port);
156}
157
158static struct usb_request *
159gps_alloc_req(struct usb_ep *ep, unsigned len, gfp_t flags)
160{
161 struct usb_request *req;
162
163 req = usb_ep_alloc_request(ep, flags);
164 if (!req)
165 return ERR_PTR(-ENOMEM);
166
167 req->buf = kmalloc(len, flags);
168 if (!req->buf) {
169 usb_ep_free_request(ep, req);
170 return ERR_PTR(-ENOMEM);
171 }
172
173 req->length = len;
174
175 return req;
176}
177
178void gps_free_req(struct usb_ep *ep, struct usb_request *req)
179{
180 kfree(req->buf);
181 usb_ep_free_request(ep, req);
182}
183
184static struct rmnet_ctrl_pkt *gps_alloc_ctrl_pkt(unsigned len, gfp_t flags)
185{
186 struct rmnet_ctrl_pkt *pkt;
187
188 pkt = kzalloc(sizeof(struct rmnet_ctrl_pkt), flags);
189 if (!pkt)
190 return ERR_PTR(-ENOMEM);
191
192 pkt->buf = kmalloc(len, flags);
193 if (!pkt->buf) {
194 kfree(pkt);
195 return ERR_PTR(-ENOMEM);
196 }
197 pkt->len = len;
198
199 return pkt;
200}
201
202static void gps_free_ctrl_pkt(struct rmnet_ctrl_pkt *pkt)
203{
204 kfree(pkt->buf);
205 kfree(pkt);
206}
207
208/* -------------------------------------------*/
209
210static int gps_gport_setup(void)
211{
212 u8 base;
213 int res;
214
215 res = gsmd_ctrl_setup(GPS_CTRL_CLIENT, 1, &base);
216 gps_port.port->port_num += base;
217 return res;
218}
219
220static int gport_ctrl_connect(struct f_gps *dev)
221{
222 return gsmd_ctrl_connect(&dev->port, dev->port_num);
223}
224
225static int gport_gps_disconnect(struct f_gps *dev)
226{
227 gsmd_ctrl_disconnect(&dev->port, dev->port_num);
228 return 0;
229}
230
231static void gps_unbind(struct usb_configuration *c, struct usb_function *f)
232{
233 struct f_gps *dev = func_to_gps(f);
234
235 pr_debug("%s: portno:%d\n", __func__, dev->port_num);
236
237 if (gadget_is_superspeed(c->cdev->gadget))
238 usb_free_descriptors(f->ss_descriptors);
239 if (gadget_is_dualspeed(c->cdev->gadget))
240 usb_free_descriptors(f->hs_descriptors);
241 usb_free_descriptors(f->descriptors);
242
243 gps_free_req(dev->notify, dev->notify_req);
244
245 kfree(f->name);
246}
247
248static void gps_purge_responses(struct f_gps *dev)
249{
250 unsigned long flags;
251 struct rmnet_ctrl_pkt *cpkt;
252
253 pr_debug("%s: port#%d\n", __func__, dev->port_num);
254
255 spin_lock_irqsave(&dev->lock, flags);
256 while (!list_empty(&dev->cpkt_resp_q)) {
257 cpkt = list_first_entry(&dev->cpkt_resp_q,
258 struct rmnet_ctrl_pkt, list);
259
260 list_del(&cpkt->list);
261 rmnet_free_ctrl_pkt(cpkt);
262 }
263 atomic_set(&dev->notify_count, 0);
264 spin_unlock_irqrestore(&dev->lock, flags);
265}
266
267static void gps_suspend(struct usb_function *f)
268{
269 struct f_gps *dev = func_to_gps(f);
270 gps_purge_responses(dev);
271
272}
273
274static void gps_disable(struct usb_function *f)
275{
276 struct f_gps *dev = func_to_gps(f);
277
278 usb_ep_disable(dev->notify);
279 dev->notify->driver_data = NULL;
280
281 atomic_set(&dev->online, 0);
282
283 gps_purge_responses(dev);
284
285 gport_gps_disconnect(dev);
286}
287
288static int
289gps_set_alt(struct usb_function *f, unsigned intf, unsigned alt)
290{
291 struct f_gps *dev = func_to_gps(f);
292 struct usb_composite_dev *cdev = dev->cdev;
293 int ret;
294 struct list_head *cpkt;
295
296 pr_debug("%s:dev:%p\n", __func__, dev);
297
298 if (dev->notify->driver_data)
299 usb_ep_disable(dev->notify);
300
301 ret = config_ep_by_speed(cdev->gadget, f, dev->notify);
302 if (ret) {
303 dev->notify->desc = NULL;
304 ERROR(cdev, "config_ep_by_speed failes for ep %s, result %d\n",
305 dev->notify->name, ret);
306 return ret;
307 }
308 ret = usb_ep_enable(dev->notify);
309
310 if (ret) {
311 pr_err("%s: usb ep#%s enable failed, err#%d\n",
312 __func__, dev->notify->name, ret);
313 return ret;
314 }
315 dev->notify->driver_data = dev;
316
317 ret = gport_ctrl_connect(dev);
318
319 atomic_set(&dev->online, 1);
320
321 /* In case notifications were aborted, but there are pending control
322 packets in the response queue, re-add the notifications */
323 list_for_each(cpkt, &dev->cpkt_resp_q)
324 gps_ctrl_response_available(dev);
325
326 return ret;
327}
328
329static void gps_ctrl_response_available(struct f_gps *dev)
330{
331 struct usb_request *req = dev->notify_req;
332 struct usb_cdc_notification *event;
333 unsigned long flags;
334 int ret;
335 struct rmnet_ctrl_pkt *cpkt;
336
337 pr_debug("%s:dev:%p\n", __func__, dev);
338
339 spin_lock_irqsave(&dev->lock, flags);
340 if (!atomic_read(&dev->online) || !req || !req->buf) {
341 spin_unlock_irqrestore(&dev->lock, flags);
342 return;
343 }
344
345 if (atomic_inc_return(&dev->notify_count) != 1) {
346 spin_unlock_irqrestore(&dev->lock, flags);
347 return;
348 }
349
350 event = req->buf;
351 event->bmRequestType = USB_DIR_IN | USB_TYPE_CLASS
352 | USB_RECIP_INTERFACE;
353 event->bNotificationType = USB_CDC_NOTIFY_RESPONSE_AVAILABLE;
354 event->wValue = cpu_to_le16(0);
355 event->wIndex = cpu_to_le16(dev->ifc_id);
356 event->wLength = cpu_to_le16(0);
357 spin_unlock_irqrestore(&dev->lock, flags);
358
359 ret = usb_ep_queue(dev->notify, dev->notify_req, GFP_ATOMIC);
360 if (ret) {
361 spin_lock_irqsave(&dev->lock, flags);
362 if (!list_empty(&dev->cpkt_resp_q)) {
363 atomic_dec(&dev->notify_count);
364 cpkt = list_first_entry(&dev->cpkt_resp_q,
365 struct rmnet_ctrl_pkt, list);
366 list_del(&cpkt->list);
367 gps_free_ctrl_pkt(cpkt);
368 }
369 spin_unlock_irqrestore(&dev->lock, flags);
370 pr_debug("ep enqueue error %d\n", ret);
371 }
372}
373
374static void gps_connect(struct grmnet *gr)
375{
376 struct f_gps *dev;
377
378 if (!gr) {
379 pr_err("%s: Invalid grmnet:%p\n", __func__, gr);
380 return;
381 }
382
383 dev = port_to_gps(gr);
384
385 atomic_set(&dev->ctrl_online, 1);
386}
387
388static void gps_disconnect(struct grmnet *gr)
389{
390 struct f_gps *dev;
391 struct usb_cdc_notification *event;
392 int status;
393
394 if (!gr) {
395 pr_err("%s: Invalid grmnet:%p\n", __func__, gr);
396 return;
397 }
398
399 dev = port_to_gps(gr);
400
401 atomic_set(&dev->ctrl_online, 0);
402
403 if (!atomic_read(&dev->online)) {
404 pr_debug("%s: nothing to do\n", __func__);
405 return;
406 }
407
408 usb_ep_fifo_flush(dev->notify);
409
410 event = dev->notify_req->buf;
411 event->bmRequestType = USB_DIR_IN | USB_TYPE_CLASS
412 | USB_RECIP_INTERFACE;
413 event->bNotificationType = USB_CDC_NOTIFY_NETWORK_CONNECTION;
414 event->wValue = cpu_to_le16(0);
415 event->wIndex = cpu_to_le16(dev->ifc_id);
416 event->wLength = cpu_to_le16(0);
417
418 status = usb_ep_queue(dev->notify, dev->notify_req, GFP_ATOMIC);
419 if (status < 0) {
420 if (!atomic_read(&dev->online))
421 return;
422 pr_err("%s: gps notify ep enqueue error %d\n",
423 __func__, status);
424 }
425
426 gps_purge_responses(dev);
427}
428
429static int
430gps_send_cpkt_response(void *gr, void *buf, size_t len)
431{
432 struct f_gps *dev;
433 struct rmnet_ctrl_pkt *cpkt;
434 unsigned long flags;
435
436 if (!gr || !buf) {
437 pr_err("%s: Invalid grmnet/buf, grmnet:%p buf:%p\n",
438 __func__, gr, buf);
439 return -ENODEV;
440 }
441 cpkt = gps_alloc_ctrl_pkt(len, GFP_ATOMIC);
442 if (IS_ERR(cpkt)) {
443 pr_err("%s: Unable to allocate ctrl pkt\n", __func__);
444 return -ENOMEM;
445 }
446 memcpy(cpkt->buf, buf, len);
447 cpkt->len = len;
448
449 dev = port_to_gps(gr);
450
451 pr_debug("%s: dev:%p\n", __func__, dev);
452
453 if (!atomic_read(&dev->online) || !atomic_read(&dev->ctrl_online)) {
454 gps_free_ctrl_pkt(cpkt);
455 return 0;
456 }
457
458 spin_lock_irqsave(&dev->lock, flags);
459 list_add_tail(&cpkt->list, &dev->cpkt_resp_q);
460 spin_unlock_irqrestore(&dev->lock, flags);
461
462 gps_ctrl_response_available(dev);
463
464 return 0;
465}
466
467static void
468gps_cmd_complete(struct usb_ep *ep, struct usb_request *req)
469{
470 struct f_gps *dev = req->context;
471 struct usb_composite_dev *cdev;
472
473 if (!dev) {
474 pr_err("%s: dev is null\n", __func__);
475 return;
476 }
477
478 pr_debug("%s: dev:%p\n", __func__, dev);
479
480 cdev = dev->cdev;
481
482 if (dev->port.send_encap_cmd)
483 dev->port.send_encap_cmd(dev->port_num, req->buf, req->actual);
484}
485
486static void gps_notify_complete(struct usb_ep *ep, struct usb_request *req)
487{
488 struct f_gps *dev = req->context;
489 int status = req->status;
490 unsigned long flags;
491 struct rmnet_ctrl_pkt *cpkt;
492
493 pr_debug("%s: dev:%p port#%d\n", __func__, dev, dev->port_num);
494
495 switch (status) {
496 case -ECONNRESET:
497 case -ESHUTDOWN:
498 /* connection gone */
499 atomic_set(&dev->notify_count, 0);
500 break;
501 default:
502 pr_err("gps notify ep error %d\n", status);
503 /* FALLTHROUGH */
504 case 0:
505 if (!atomic_read(&dev->ctrl_online))
506 break;
507
508 if (atomic_dec_and_test(&dev->notify_count))
509 break;
510
511 status = usb_ep_queue(dev->notify, req, GFP_ATOMIC);
512 if (status) {
513 spin_lock_irqsave(&dev->lock, flags);
514 if (!list_empty(&dev->cpkt_resp_q)) {
515 atomic_dec(&dev->notify_count);
516 cpkt = list_first_entry(&dev->cpkt_resp_q,
517 struct rmnet_ctrl_pkt, list);
518 list_del(&cpkt->list);
519 gps_free_ctrl_pkt(cpkt);
520 }
521 spin_unlock_irqrestore(&dev->lock, flags);
522 pr_debug("ep enqueue error %d\n", status);
523 }
524 break;
525 }
526}
527
528static int
529gps_setup(struct usb_function *f, const struct usb_ctrlrequest *ctrl)
530{
531 struct f_gps *dev = func_to_gps(f);
532 struct usb_composite_dev *cdev = dev->cdev;
533 struct usb_request *req = cdev->req;
534 u16 w_index = le16_to_cpu(ctrl->wIndex);
535 u16 w_value = le16_to_cpu(ctrl->wValue);
536 u16 w_length = le16_to_cpu(ctrl->wLength);
537 int ret = -EOPNOTSUPP;
538
539 pr_debug("%s:dev:%p\n", __func__, dev);
540
541 if (!atomic_read(&dev->online)) {
542 pr_debug("%s: usb cable is not connected\n", __func__);
543 return -ENOTCONN;
544 }
545
546 switch ((ctrl->bRequestType << 8) | ctrl->bRequest) {
547 case ((USB_DIR_OUT | USB_TYPE_CLASS | USB_RECIP_INTERFACE) << 8)
548 | USB_CDC_SEND_ENCAPSULATED_COMMAND:
549 ret = w_length;
550 req->complete = gps_cmd_complete;
551 req->context = dev;
552 break;
553
554 case ((USB_DIR_IN | USB_TYPE_CLASS | USB_RECIP_INTERFACE) << 8)
555 | USB_CDC_GET_ENCAPSULATED_RESPONSE:
556 if (w_value)
557 goto invalid;
558 else {
559 unsigned len;
560 struct rmnet_ctrl_pkt *cpkt;
561
562 spin_lock(&dev->lock);
563 if (list_empty(&dev->cpkt_resp_q)) {
564 pr_err("%s: ctrl resp queue empty", __func__);
565 spin_unlock(&dev->lock);
566 goto invalid;
567 }
568
569 cpkt = list_first_entry(&dev->cpkt_resp_q,
570 struct rmnet_ctrl_pkt, list);
571 list_del(&cpkt->list);
572 spin_unlock(&dev->lock);
573
574 len = min_t(unsigned, w_length, cpkt->len);
575 memcpy(req->buf, cpkt->buf, len);
576 ret = len;
577
578 gps_free_ctrl_pkt(cpkt);
579 }
580 break;
581 case ((USB_DIR_OUT | USB_TYPE_CLASS | USB_RECIP_INTERFACE) << 8)
582 | USB_CDC_REQ_SET_CONTROL_LINE_STATE:
583 if (dev->port.notify_modem)
584 dev->port.notify_modem(&dev->port,
585 dev->port_num, w_value);
586 ret = 0;
587
588 break;
589 default:
590
591invalid:
592 DBG(cdev, "invalid control req%02x.%02x v%04x i%04x l%d\n",
593 ctrl->bRequestType, ctrl->bRequest,
594 w_value, w_index, w_length);
595 }
596
597 /* respond with data transfer or status phase? */
598 if (ret >= 0) {
599 VDBG(cdev, "gps req%02x.%02x v%04x i%04x l%d\n",
600 ctrl->bRequestType, ctrl->bRequest,
601 w_value, w_index, w_length);
602 req->zero = (ret < w_length);
603 req->length = ret;
604 ret = usb_ep_queue(cdev->gadget->ep0, req, GFP_ATOMIC);
605 if (ret < 0)
606 ERROR(cdev, "gps ep0 enqueue err %d\n", ret);
607 }
608
609 return ret;
610}
611
612static int gps_bind(struct usb_configuration *c, struct usb_function *f)
613{
614 struct f_gps *dev = func_to_gps(f);
615 struct usb_ep *ep;
616 struct usb_composite_dev *cdev = c->cdev;
617 int ret = -ENODEV;
618
619 dev->ifc_id = usb_interface_id(c, f);
620 if (dev->ifc_id < 0) {
621 pr_err("%s: unable to allocate ifc id, err:%d",
622 __func__, dev->ifc_id);
623 return dev->ifc_id;
624 }
625 gps_interface_desc.bInterfaceNumber = dev->ifc_id;
626
627 dev->port.in = NULL;
628 dev->port.out = NULL;
629
630 ep = usb_ep_autoconfig(cdev->gadget, &gps_fs_notify_desc);
631 if (!ep) {
632 pr_err("%s: usb epnotify autoconfig failed\n", __func__);
633 ret = -ENODEV;
634 goto ep_auto_notify_fail;
635 }
636 dev->notify = ep;
637 ep->driver_data = cdev;
638
639 dev->notify_req = gps_alloc_req(ep,
640 sizeof(struct usb_cdc_notification),
641 GFP_KERNEL);
642 if (IS_ERR(dev->notify_req)) {
643 pr_err("%s: unable to allocate memory for notify req\n",
644 __func__);
645 ret = -ENOMEM;
646 goto ep_notify_alloc_fail;
647 }
648
649 dev->notify_req->complete = gps_notify_complete;
650 dev->notify_req->context = dev;
651
652 ret = -ENOMEM;
653 f->descriptors = usb_copy_descriptors(gps_fs_function);
654
655 if (!f->descriptors)
656 goto fail;
657
658 if (gadget_is_dualspeed(cdev->gadget)) {
659 gps_hs_notify_desc.bEndpointAddress =
660 gps_fs_notify_desc.bEndpointAddress;
661
662 /* copy descriptors, and track endpoint copies */
663 f->hs_descriptors = usb_copy_descriptors(gps_hs_function);
664
665 if (!f->hs_descriptors)
666 goto fail;
667 }
668
669 if (gadget_is_superspeed(cdev->gadget)) {
670 gps_ss_notify_desc.bEndpointAddress =
671 gps_fs_notify_desc.bEndpointAddress;
672
673 /* copy descriptors, and track endpoint copies */
674 f->ss_descriptors = usb_copy_descriptors(gps_ss_function);
675
676 if (!f->ss_descriptors)
677 goto fail;
678 }
679
680 pr_info("%s: GPS(%d) %s Speed\n",
681 __func__, dev->port_num,
682 gadget_is_dualspeed(cdev->gadget) ? "dual" : "full");
683
684 return 0;
685
686fail:
687 if (f->ss_descriptors)
688 usb_free_descriptors(f->ss_descriptors);
689 if (f->hs_descriptors)
690 usb_free_descriptors(f->hs_descriptors);
691 if (f->descriptors)
692 usb_free_descriptors(f->descriptors);
693 if (dev->notify_req)
694 gps_free_req(dev->notify, dev->notify_req);
695ep_notify_alloc_fail:
696 dev->notify->driver_data = NULL;
697 dev->notify = NULL;
698ep_auto_notify_fail:
699 return ret;
700}
701
702static int gps_bind_config(struct usb_configuration *c)
703{
704 int status;
705 struct f_gps *dev;
706 struct usb_function *f;
707 unsigned long flags;
708
709 pr_debug("%s: usb config:%p\n", __func__, c);
710
711 if (gps_string_defs[0].id == 0) {
712 status = usb_string_id(c->cdev);
713 if (status < 0) {
714 pr_err("%s: failed to get string id, err:%d\n",
715 __func__, status);
716 return status;
717 }
718 gps_string_defs[0].id = status;
719 }
720
721 dev = gps_port.port;
722
723 spin_lock_irqsave(&dev->lock, flags);
724 dev->cdev = c->cdev;
725 f = &dev->port.func;
726 f->name = kasprintf(GFP_ATOMIC, "gps");
727 spin_unlock_irqrestore(&dev->lock, flags);
728 if (!f->name) {
729 pr_err("%s: cannot allocate memory for name\n", __func__);
730 return -ENOMEM;
731 }
732
733 f->strings = gps_strings;
734 f->bind = gps_bind;
735 f->unbind = gps_unbind;
736 f->disable = gps_disable;
737 f->set_alt = gps_set_alt;
738 f->setup = gps_setup;
739 f->suspend = gps_suspend;
740 dev->port.send_cpkt_response = gps_send_cpkt_response;
741 dev->port.disconnect = gps_disconnect;
742 dev->port.connect = gps_connect;
743
744 status = usb_add_function(c, f);
745 if (status) {
746 pr_err("%s: usb add function failed: %d\n",
747 __func__, status);
748 kfree(f->name);
749 return status;
750 }
751
752 pr_debug("%s: complete\n", __func__);
753
754 return status;
755}
756
757static void gps_cleanup(void)
758{
759 kfree(gps_port.port);
760}
761
762static int gps_init_port(void)
763{
764 struct f_gps *dev;
765
766 dev = kzalloc(sizeof(struct f_gps), GFP_KERNEL);
767 if (!dev) {
768 pr_err("%s: Unable to allocate gps device\n", __func__);
769 return -ENOMEM;
770 }
771
772 spin_lock_init(&dev->lock);
773 INIT_LIST_HEAD(&dev->cpkt_resp_q);
774 dev->port_num = 0;
775
776 gps_port.port = dev;
777 gps_port.ctrl_xport = USB_GADGET_XPORT_SMD;
778
779 return 0;
780}