blob: aa4361f29ccfda4799db503004de6b092fe994f8 [file] [log] [blame]
David Brownell61d8bae2008-06-19 18:18:50 -07001/*
2 * f_serial.c - generic USB serial function driver
3 *
4 * Copyright (C) 2003 Al Borchers (alborchers@steinerpoint.com)
5 * Copyright (C) 2008 by David Brownell
6 * Copyright (C) 2008 by Nokia Corporation
7 *
8 * This software is distributed under the terms of the GNU General
9 * Public License ("GPL") as published by the Free Software Foundation,
10 * either version 2 of that License or (at your option) any later version.
11 */
12
Tejun Heo5a0e3ad2010-03-24 17:04:11 +090013#include <linux/slab.h>
David Brownell61d8bae2008-06-19 18:18:50 -070014#include <linux/kernel.h>
15#include <linux/device.h>
Hemant Kumarbffd4142011-11-03 13:20:40 -070016#include <mach/usb_gadget_xport.h>
David Brownell61d8bae2008-06-19 18:18:50 -070017
18#include "u_serial.h"
19#include "gadget_chips.h"
20
21
22/*
23 * This function packages a simple "generic serial" port with no real
24 * control mechanisms, just raw data transfer over two bulk endpoints.
25 *
26 * Because it's not standardized, this isn't as interoperable as the
27 * CDC ACM driver. However, for many purposes it's just as functional
28 * if you can arrange appropriate host side drivers.
29 */
Hemant Kumarbffd4142011-11-03 13:20:40 -070030#define GSERIAL_NO_PORTS 2
David Brownell61d8bae2008-06-19 18:18:50 -070031
32struct gser_descs {
33 struct usb_endpoint_descriptor *in;
34 struct usb_endpoint_descriptor *out;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -070035#ifdef CONFIG_MODEM_SUPPORT
36 struct usb_endpoint_descriptor *notify;
37#endif
David Brownell61d8bae2008-06-19 18:18:50 -070038};
39
40struct f_gser {
41 struct gserial port;
42 u8 data_id;
43 u8 port_num;
44
David Brownell61d8bae2008-06-19 18:18:50 -070045 struct gser_descs fs;
David Brownell61d8bae2008-06-19 18:18:50 -070046 struct gser_descs hs;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -070047 u8 online;
48 enum transport_type transport;
49
50#ifdef CONFIG_MODEM_SUPPORT
51 u8 pending;
52 spinlock_t lock;
53 struct usb_ep *notify;
54 struct usb_endpoint_descriptor *notify_desc;
55 struct usb_request *notify_req;
56
57 struct usb_cdc_line_coding port_line_coding;
58
59 /* SetControlLineState request */
60 u16 port_handshake_bits;
61#define ACM_CTRL_RTS (1 << 1) /* unused with full duplex */
62#define ACM_CTRL_DTR (1 << 0) /* host is ready for data r/w */
63
64 /* SerialState notification */
65 u16 serial_state;
66#define ACM_CTRL_OVERRUN (1 << 6)
67#define ACM_CTRL_PARITY (1 << 5)
68#define ACM_CTRL_FRAMING (1 << 4)
69#define ACM_CTRL_RI (1 << 3)
70#define ACM_CTRL_BRK (1 << 2)
71#define ACM_CTRL_DSR (1 << 1)
72#define ACM_CTRL_DCD (1 << 0)
73#endif
David Brownell61d8bae2008-06-19 18:18:50 -070074};
75
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -070076static unsigned int no_tty_ports;
77static unsigned int no_sdio_ports;
78static unsigned int no_smd_ports;
79static unsigned int nr_ports;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -070080
81static struct port_info {
82 enum transport_type transport;
83 unsigned port_num;
84 unsigned client_port_num;
85} gserial_ports[GSERIAL_NO_PORTS];
86
87static inline bool is_transport_sdio(enum transport_type t)
88{
Hemant Kumarbffd4142011-11-03 13:20:40 -070089 if (t == USB_GADGET_XPORT_SDIO)
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -070090 return 1;
91 return 0;
92}
93
David Brownell61d8bae2008-06-19 18:18:50 -070094static inline struct f_gser *func_to_gser(struct usb_function *f)
95{
96 return container_of(f, struct f_gser, port.func);
97}
98
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -070099#ifdef CONFIG_MODEM_SUPPORT
100static inline struct f_gser *port_to_gser(struct gserial *p)
101{
102 return container_of(p, struct f_gser, port);
103}
104#define GS_LOG2_NOTIFY_INTERVAL 5 /* 1 << 5 == 32 msec */
105#define GS_NOTIFY_MAXPACKET 10 /* notification + 2 bytes */
106#endif
David Brownell61d8bae2008-06-19 18:18:50 -0700107/*-------------------------------------------------------------------------*/
108
109/* interface descriptor: */
110
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700111static struct usb_interface_descriptor gser_interface_desc = {
David Brownell61d8bae2008-06-19 18:18:50 -0700112 .bLength = USB_DT_INTERFACE_SIZE,
113 .bDescriptorType = USB_DT_INTERFACE,
114 /* .bInterfaceNumber = DYNAMIC */
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700115#ifdef CONFIG_MODEM_SUPPORT
116 .bNumEndpoints = 3,
117#else
David Brownell61d8bae2008-06-19 18:18:50 -0700118 .bNumEndpoints = 2,
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700119#endif
David Brownell61d8bae2008-06-19 18:18:50 -0700120 .bInterfaceClass = USB_CLASS_VENDOR_SPEC,
121 .bInterfaceSubClass = 0,
122 .bInterfaceProtocol = 0,
123 /* .iInterface = DYNAMIC */
124};
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700125#ifdef CONFIG_MODEM_SUPPORT
126static struct usb_cdc_header_desc gser_header_desc = {
127 .bLength = sizeof(gser_header_desc),
128 .bDescriptorType = USB_DT_CS_INTERFACE,
129 .bDescriptorSubType = USB_CDC_HEADER_TYPE,
130 .bcdCDC = __constant_cpu_to_le16(0x0110),
131};
David Brownell61d8bae2008-06-19 18:18:50 -0700132
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700133static struct usb_cdc_call_mgmt_descriptor
134gser_call_mgmt_descriptor = {
135 .bLength = sizeof(gser_call_mgmt_descriptor),
136 .bDescriptorType = USB_DT_CS_INTERFACE,
137 .bDescriptorSubType = USB_CDC_CALL_MANAGEMENT_TYPE,
138 .bmCapabilities = 0,
139 /* .bDataInterface = DYNAMIC */
140};
141
142static struct usb_cdc_acm_descriptor gser_descriptor = {
143 .bLength = sizeof(gser_descriptor),
144 .bDescriptorType = USB_DT_CS_INTERFACE,
145 .bDescriptorSubType = USB_CDC_ACM_TYPE,
146 .bmCapabilities = USB_CDC_CAP_LINE,
147};
148
149static struct usb_cdc_union_desc gser_union_desc = {
150 .bLength = sizeof(gser_union_desc),
151 .bDescriptorType = USB_DT_CS_INTERFACE,
152 .bDescriptorSubType = USB_CDC_UNION_TYPE,
153 /* .bMasterInterface0 = DYNAMIC */
154 /* .bSlaveInterface0 = DYNAMIC */
155};
156#endif
David Brownell61d8bae2008-06-19 18:18:50 -0700157/* full speed support: */
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700158#ifdef CONFIG_MODEM_SUPPORT
159static struct usb_endpoint_descriptor gser_fs_notify_desc = {
160 .bLength = USB_DT_ENDPOINT_SIZE,
161 .bDescriptorType = USB_DT_ENDPOINT,
162 .bEndpointAddress = USB_DIR_IN,
163 .bmAttributes = USB_ENDPOINT_XFER_INT,
164 .wMaxPacketSize = __constant_cpu_to_le16(GS_NOTIFY_MAXPACKET),
165 .bInterval = 1 << GS_LOG2_NOTIFY_INTERVAL,
166};
167#endif
David Brownell61d8bae2008-06-19 18:18:50 -0700168
Manu Gautama4d993f2011-08-30 18:25:55 +0530169static struct usb_endpoint_descriptor gser_fs_in_desc = {
David Brownell61d8bae2008-06-19 18:18:50 -0700170 .bLength = USB_DT_ENDPOINT_SIZE,
171 .bDescriptorType = USB_DT_ENDPOINT,
172 .bEndpointAddress = USB_DIR_IN,
173 .bmAttributes = USB_ENDPOINT_XFER_BULK,
174};
175
Manu Gautama4d993f2011-08-30 18:25:55 +0530176static struct usb_endpoint_descriptor gser_fs_out_desc = {
David Brownell61d8bae2008-06-19 18:18:50 -0700177 .bLength = USB_DT_ENDPOINT_SIZE,
178 .bDescriptorType = USB_DT_ENDPOINT,
179 .bEndpointAddress = USB_DIR_OUT,
180 .bmAttributes = USB_ENDPOINT_XFER_BULK,
181};
182
Manu Gautama4d993f2011-08-30 18:25:55 +0530183static struct usb_descriptor_header *gser_fs_function[] = {
David Brownell61d8bae2008-06-19 18:18:50 -0700184 (struct usb_descriptor_header *) &gser_interface_desc,
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700185#ifdef CONFIG_MODEM_SUPPORT
186 (struct usb_descriptor_header *) &gser_header_desc,
187 (struct usb_descriptor_header *) &gser_call_mgmt_descriptor,
188 (struct usb_descriptor_header *) &gser_descriptor,
189 (struct usb_descriptor_header *) &gser_union_desc,
190 (struct usb_descriptor_header *) &gser_fs_notify_desc,
191#endif
David Brownell61d8bae2008-06-19 18:18:50 -0700192 (struct usb_descriptor_header *) &gser_fs_in_desc,
193 (struct usb_descriptor_header *) &gser_fs_out_desc,
194 NULL,
195};
196
197/* high speed support: */
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700198#ifdef CONFIG_MODEM_SUPPORT
199static struct usb_endpoint_descriptor gser_hs_notify_desc = {
200 .bLength = USB_DT_ENDPOINT_SIZE,
201 .bDescriptorType = USB_DT_ENDPOINT,
202 .bEndpointAddress = USB_DIR_IN,
203 .bmAttributes = USB_ENDPOINT_XFER_INT,
204 .wMaxPacketSize = __constant_cpu_to_le16(GS_NOTIFY_MAXPACKET),
205 .bInterval = GS_LOG2_NOTIFY_INTERVAL+4,
206};
207#endif
David Brownell61d8bae2008-06-19 18:18:50 -0700208
Manu Gautama4d993f2011-08-30 18:25:55 +0530209static struct usb_endpoint_descriptor gser_hs_in_desc = {
David Brownell61d8bae2008-06-19 18:18:50 -0700210 .bLength = USB_DT_ENDPOINT_SIZE,
211 .bDescriptorType = USB_DT_ENDPOINT,
212 .bmAttributes = USB_ENDPOINT_XFER_BULK,
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700213 .wMaxPacketSize = __constant_cpu_to_le16(512),
David Brownell61d8bae2008-06-19 18:18:50 -0700214};
215
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700216static struct usb_endpoint_descriptor gser_hs_out_desc = {
David Brownell61d8bae2008-06-19 18:18:50 -0700217 .bLength = USB_DT_ENDPOINT_SIZE,
218 .bDescriptorType = USB_DT_ENDPOINT,
219 .bmAttributes = USB_ENDPOINT_XFER_BULK,
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700220 .wMaxPacketSize = __constant_cpu_to_le16(512),
David Brownell61d8bae2008-06-19 18:18:50 -0700221};
222
Manu Gautama4d993f2011-08-30 18:25:55 +0530223static struct usb_descriptor_header *gser_hs_function[] = {
David Brownell61d8bae2008-06-19 18:18:50 -0700224 (struct usb_descriptor_header *) &gser_interface_desc,
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700225#ifdef CONFIG_MODEM_SUPPORT
226 (struct usb_descriptor_header *) &gser_header_desc,
227 (struct usb_descriptor_header *) &gser_call_mgmt_descriptor,
228 (struct usb_descriptor_header *) &gser_descriptor,
229 (struct usb_descriptor_header *) &gser_union_desc,
230 (struct usb_descriptor_header *) &gser_hs_notify_desc,
231#endif
David Brownell61d8bae2008-06-19 18:18:50 -0700232 (struct usb_descriptor_header *) &gser_hs_in_desc,
233 (struct usb_descriptor_header *) &gser_hs_out_desc,
234 NULL,
235};
236
237/* string descriptors: */
238
239static struct usb_string gser_string_defs[] = {
240 [0].s = "Generic Serial",
241 { } /* end of list */
242};
243
244static struct usb_gadget_strings gser_string_table = {
245 .language = 0x0409, /* en-us */
246 .strings = gser_string_defs,
247};
248
249static struct usb_gadget_strings *gser_strings[] = {
250 &gser_string_table,
251 NULL,
252};
253
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700254static char *transport_to_str(enum transport_type t)
255{
256 switch (t) {
Hemant Kumarbffd4142011-11-03 13:20:40 -0700257 case USB_GADGET_XPORT_TTY:
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700258 return "TTY";
Hemant Kumarbffd4142011-11-03 13:20:40 -0700259 case USB_GADGET_XPORT_SDIO:
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700260 return "SDIO";
Hemant Kumarbffd4142011-11-03 13:20:40 -0700261 case USB_GADGET_XPORT_SMD:
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700262 return "SMD";
Hemant Kumarbffd4142011-11-03 13:20:40 -0700263 default:
264 return "NONE";
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700265 }
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700266}
267
Manu Gautama4d993f2011-08-30 18:25:55 +0530268static enum transport_type serial_str_to_transport(const char *name)
269{
270 if (!strcasecmp("SDIO", name))
Hemant Kumarbffd4142011-11-03 13:20:40 -0700271 return USB_GADGET_XPORT_SDIO;
Manu Gautama4d993f2011-08-30 18:25:55 +0530272 if (!strcasecmp("SMD", name))
Hemant Kumarbffd4142011-11-03 13:20:40 -0700273 return USB_GADGET_XPORT_SMD;
Manu Gautama4d993f2011-08-30 18:25:55 +0530274
Hemant Kumarbffd4142011-11-03 13:20:40 -0700275 return USB_GADGET_XPORT_TTY;
Manu Gautama4d993f2011-08-30 18:25:55 +0530276}
277
278
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700279static int gport_setup(struct usb_configuration *c)
280{
281 int ret = 0;
282
283 pr_debug("%s: no_tty_ports:%u no_sdio_ports: %u nr_ports:%u\n",
284 __func__, no_tty_ports, no_sdio_ports, nr_ports);
285
286 if (no_tty_ports)
287 ret = gserial_setup(c->cdev->gadget, no_tty_ports);
288 if (no_sdio_ports)
289 ret = gsdio_setup(c->cdev->gadget, no_sdio_ports);
290 if (no_smd_ports)
291 ret = gsmd_setup(c->cdev->gadget, no_smd_ports);
292
293 return ret;
294}
Manu Gautama4d993f2011-08-30 18:25:55 +0530295
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700296static int gport_connect(struct f_gser *gser)
297{
298 unsigned port_num;
299
300 pr_debug("%s: transport:%s f_gser:%p gserial:%p port_num:%d\n",
301 __func__, transport_to_str(gser->transport),
302 gser, &gser->port, gser->port_num);
303
304 port_num = gserial_ports[gser->port_num].client_port_num;
305
306 switch (gser->transport) {
Hemant Kumarbffd4142011-11-03 13:20:40 -0700307 case USB_GADGET_XPORT_TTY:
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700308 gserial_connect(&gser->port, port_num);
309 break;
Hemant Kumarbffd4142011-11-03 13:20:40 -0700310 case USB_GADGET_XPORT_SDIO:
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700311 gsdio_connect(&gser->port, port_num);
312 break;
Hemant Kumarbffd4142011-11-03 13:20:40 -0700313 case USB_GADGET_XPORT_SMD:
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700314 gsmd_connect(&gser->port, port_num);
315 break;
316 default:
317 pr_err("%s: Un-supported transport: %s\n", __func__,
318 transport_to_str(gser->transport));
319 return -ENODEV;
320 }
321
322 return 0;
323}
324
325static int gport_disconnect(struct f_gser *gser)
326{
327 unsigned port_num;
328
329 pr_debug("%s: transport:%s f_gser:%p gserial:%p port_num:%d\n",
330 __func__, transport_to_str(gser->transport),
331 gser, &gser->port, gser->port_num);
332
333 port_num = gserial_ports[gser->port_num].client_port_num;
334
335 switch (gser->transport) {
Hemant Kumarbffd4142011-11-03 13:20:40 -0700336 case USB_GADGET_XPORT_TTY:
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700337 gserial_disconnect(&gser->port);
338 break;
Hemant Kumarbffd4142011-11-03 13:20:40 -0700339 case USB_GADGET_XPORT_SDIO:
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700340 gsdio_disconnect(&gser->port, port_num);
341 break;
Hemant Kumarbffd4142011-11-03 13:20:40 -0700342 case USB_GADGET_XPORT_SMD:
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700343 gsmd_disconnect(&gser->port, port_num);
344 break;
345 default:
346 pr_err("%s: Un-supported transport:%s\n", __func__,
347 transport_to_str(gser->transport));
348 return -ENODEV;
349 }
350
351 return 0;
352}
353
354#ifdef CONFIG_MODEM_SUPPORT
355static void gser_complete_set_line_coding(struct usb_ep *ep,
356 struct usb_request *req)
357{
358 struct f_gser *gser = ep->driver_data;
359 struct usb_composite_dev *cdev = gser->port.func.config->cdev;
360
361 if (req->status != 0) {
362 DBG(cdev, "gser ttyGS%d completion, err %d\n",
363 gser->port_num, req->status);
364 return;
365 }
366
367 /* normal completion */
368 if (req->actual != sizeof(gser->port_line_coding)) {
369 DBG(cdev, "gser ttyGS%d short resp, len %d\n",
370 gser->port_num, req->actual);
371 usb_ep_set_halt(ep);
372 } else {
373 struct usb_cdc_line_coding *value = req->buf;
374 gser->port_line_coding = *value;
375 }
376}
David Brownell61d8bae2008-06-19 18:18:50 -0700377/*-------------------------------------------------------------------------*/
378
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700379static int
380gser_setup(struct usb_function *f, const struct usb_ctrlrequest *ctrl)
381{
382 struct f_gser *gser = func_to_gser(f);
383 struct usb_composite_dev *cdev = f->config->cdev;
384 struct usb_request *req = cdev->req;
385 int value = -EOPNOTSUPP;
386 u16 w_index = le16_to_cpu(ctrl->wIndex);
387 u16 w_value = le16_to_cpu(ctrl->wValue);
388 u16 w_length = le16_to_cpu(ctrl->wLength);
389
390 switch ((ctrl->bRequestType << 8) | ctrl->bRequest) {
391
392 /* SET_LINE_CODING ... just read and save what the host sends */
393 case ((USB_DIR_OUT | USB_TYPE_CLASS | USB_RECIP_INTERFACE) << 8)
394 | USB_CDC_REQ_SET_LINE_CODING:
395 if (w_length != sizeof(struct usb_cdc_line_coding))
396 goto invalid;
397
398 value = w_length;
399 cdev->gadget->ep0->driver_data = gser;
400 req->complete = gser_complete_set_line_coding;
401 break;
402
403 /* GET_LINE_CODING ... return what host sent, or initial value */
404 case ((USB_DIR_IN | USB_TYPE_CLASS | USB_RECIP_INTERFACE) << 8)
405 | USB_CDC_REQ_GET_LINE_CODING:
406 value = min_t(unsigned, w_length,
407 sizeof(struct usb_cdc_line_coding));
408 memcpy(req->buf, &gser->port_line_coding, value);
409 break;
410
411 /* SET_CONTROL_LINE_STATE ... save what the host sent */
412 case ((USB_DIR_OUT | USB_TYPE_CLASS | USB_RECIP_INTERFACE) << 8)
413 | USB_CDC_REQ_SET_CONTROL_LINE_STATE:
414
415 value = 0;
416 gser->port_handshake_bits = w_value;
417 if (gser->port.notify_modem) {
418 unsigned port_num =
419 gserial_ports[gser->port_num].client_port_num;
420
421 gser->port.notify_modem(&gser->port,
422 port_num, w_value);
423 }
424 break;
425
426 default:
427invalid:
428 DBG(cdev, "invalid control req%02x.%02x v%04x i%04x l%d\n",
429 ctrl->bRequestType, ctrl->bRequest,
430 w_value, w_index, w_length);
431 }
432
433 /* respond with data transfer or status phase? */
434 if (value >= 0) {
435 DBG(cdev, "gser ttyGS%d req%02x.%02x v%04x i%04x l%d\n",
436 gser->port_num, ctrl->bRequestType, ctrl->bRequest,
437 w_value, w_index, w_length);
438 req->zero = 0;
439 req->length = value;
440 value = usb_ep_queue(cdev->gadget->ep0, req, GFP_ATOMIC);
441 if (value < 0)
442 ERROR(cdev, "gser response on ttyGS%d, err %d\n",
443 gser->port_num, value);
444 }
445
446 /* device either stalls (value < 0) or reports success */
447 return value;
448}
449#endif
David Brownell61d8bae2008-06-19 18:18:50 -0700450static int gser_set_alt(struct usb_function *f, unsigned intf, unsigned alt)
451{
452 struct f_gser *gser = func_to_gser(f);
453 struct usb_composite_dev *cdev = f->config->cdev;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700454 int rc = 0;
David Brownell61d8bae2008-06-19 18:18:50 -0700455
456 /* we know alt == 0, so this is an activation or a reset */
457
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700458#ifdef CONFIG_MODEM_SUPPORT
459 if (gser->notify->driver_data) {
460 DBG(cdev, "reset generic ctl ttyGS%d\n", gser->port_num);
461 usb_ep_disable(gser->notify);
David Brownell61d8bae2008-06-19 18:18:50 -0700462 }
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700463 gser->notify_desc = ep_choose(cdev->gadget,
464 gser->hs.notify,
465 gser->fs.notify);
466 rc = usb_ep_enable(gser->notify, gser->notify_desc);
467 if (rc) {
468 ERROR(cdev, "can't enable %s, result %d\n",
469 gser->notify->name, rc);
470 return rc;
471 }
472 gser->notify->driver_data = gser;
473#endif
474
475 if (gser->port.in->driver_data) {
476 DBG(cdev, "reset generic data ttyGS%d\n", gser->port_num);
477 gport_disconnect(gser);
478 } else {
479 DBG(cdev, "activate generic data ttyGS%d\n", gser->port_num);
480 }
481 gser->port.in_desc = ep_choose(cdev->gadget,
482 gser->hs.in, gser->fs.in);
483 gser->port.out_desc = ep_choose(cdev->gadget,
484 gser->hs.out, gser->fs.out);
485
486 gport_connect(gser);
487
488 gser->online = 1;
489 return rc;
David Brownell61d8bae2008-06-19 18:18:50 -0700490}
491
492static void gser_disable(struct usb_function *f)
493{
494 struct f_gser *gser = func_to_gser(f);
495 struct usb_composite_dev *cdev = f->config->cdev;
496
497 DBG(cdev, "generic ttyGS%d deactivated\n", gser->port_num);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700498
499 gport_disconnect(gser);
500
501#ifdef CONFIG_MODEM_SUPPORT
502 usb_ep_fifo_flush(gser->notify);
503 usb_ep_disable(gser->notify);
504#endif
505 gser->online = 0;
506}
507#ifdef CONFIG_MODEM_SUPPORT
508static int gser_notify(struct f_gser *gser, u8 type, u16 value,
509 void *data, unsigned length)
510{
511 struct usb_ep *ep = gser->notify;
512 struct usb_request *req;
513 struct usb_cdc_notification *notify;
514 const unsigned len = sizeof(*notify) + length;
515 void *buf;
516 int status;
517 struct usb_composite_dev *cdev = gser->port.func.config->cdev;
518
519 req = gser->notify_req;
520 gser->notify_req = NULL;
521 gser->pending = false;
522
523 req->length = len;
524 notify = req->buf;
525 buf = notify + 1;
526
527 notify->bmRequestType = USB_DIR_IN | USB_TYPE_CLASS
528 | USB_RECIP_INTERFACE;
529 notify->bNotificationType = type;
530 notify->wValue = cpu_to_le16(value);
531 notify->wIndex = cpu_to_le16(gser->data_id);
532 notify->wLength = cpu_to_le16(length);
533 memcpy(buf, data, length);
534
535 status = usb_ep_queue(ep, req, GFP_ATOMIC);
536 if (status < 0) {
537 ERROR(cdev, "gser ttyGS%d can't notify serial state, %d\n",
538 gser->port_num, status);
539 gser->notify_req = req;
540 }
541
542 return status;
David Brownell61d8bae2008-06-19 18:18:50 -0700543}
544
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700545static int gser_notify_serial_state(struct f_gser *gser)
546{
547 int status;
548 unsigned long flags;
549 struct usb_composite_dev *cdev = gser->port.func.config->cdev;
550
551 spin_lock_irqsave(&gser->lock, flags);
552 if (gser->notify_req) {
553 DBG(cdev, "gser ttyGS%d serial state %04x\n",
554 gser->port_num, gser->serial_state);
555 status = gser_notify(gser, USB_CDC_NOTIFY_SERIAL_STATE,
556 0, &gser->serial_state,
557 sizeof(gser->serial_state));
558 } else {
559 gser->pending = true;
560 status = 0;
561 }
562 spin_unlock_irqrestore(&gser->lock, flags);
563 return status;
564}
565
566static void gser_notify_complete(struct usb_ep *ep, struct usb_request *req)
567{
568 struct f_gser *gser = req->context;
569 u8 doit = false;
570 unsigned long flags;
571
572 /* on this call path we do NOT hold the port spinlock,
573 * which is why ACM needs its own spinlock
574 */
575 spin_lock_irqsave(&gser->lock, flags);
576 if (req->status != -ESHUTDOWN)
577 doit = gser->pending;
578 gser->notify_req = req;
579 spin_unlock_irqrestore(&gser->lock, flags);
580
581 if (doit && gser->online)
582 gser_notify_serial_state(gser);
583}
584static void gser_connect(struct gserial *port)
585{
586 struct f_gser *gser = port_to_gser(port);
587
588 gser->serial_state |= ACM_CTRL_DSR | ACM_CTRL_DCD;
589 gser_notify_serial_state(gser);
590}
591
592unsigned int gser_get_dtr(struct gserial *port)
593{
594 struct f_gser *gser = port_to_gser(port);
595
596 if (gser->port_handshake_bits & ACM_CTRL_DTR)
597 return 1;
598 else
599 return 0;
600}
601
602unsigned int gser_get_rts(struct gserial *port)
603{
604 struct f_gser *gser = port_to_gser(port);
605
606 if (gser->port_handshake_bits & ACM_CTRL_RTS)
607 return 1;
608 else
609 return 0;
610}
611
612unsigned int gser_send_carrier_detect(struct gserial *port, unsigned int yes)
613{
614 struct f_gser *gser = port_to_gser(port);
615 u16 state;
616
617 state = gser->serial_state;
618 state &= ~ACM_CTRL_DCD;
619 if (yes)
620 state |= ACM_CTRL_DCD;
621
622 gser->serial_state = state;
623 return gser_notify_serial_state(gser);
624
625}
626
627unsigned int gser_send_ring_indicator(struct gserial *port, unsigned int yes)
628{
629 struct f_gser *gser = port_to_gser(port);
630 u16 state;
631
632 state = gser->serial_state;
633 state &= ~ACM_CTRL_RI;
634 if (yes)
635 state |= ACM_CTRL_RI;
636
637 gser->serial_state = state;
638 return gser_notify_serial_state(gser);
639
640}
641static void gser_disconnect(struct gserial *port)
642{
643 struct f_gser *gser = port_to_gser(port);
644
645 gser->serial_state &= ~(ACM_CTRL_DSR | ACM_CTRL_DCD);
646 gser_notify_serial_state(gser);
647}
648
649static int gser_send_break(struct gserial *port, int duration)
650{
651 struct f_gser *gser = port_to_gser(port);
652 u16 state;
653
654 state = gser->serial_state;
655 state &= ~ACM_CTRL_BRK;
656 if (duration)
657 state |= ACM_CTRL_BRK;
658
659 gser->serial_state = state;
660 return gser_notify_serial_state(gser);
661}
662
663static int gser_send_modem_ctrl_bits(struct gserial *port, int ctrl_bits)
664{
665 struct f_gser *gser = port_to_gser(port);
666
667 gser->serial_state = ctrl_bits;
668
669 return gser_notify_serial_state(gser);
670}
671#endif
David Brownell61d8bae2008-06-19 18:18:50 -0700672/*-------------------------------------------------------------------------*/
673
674/* serial function driver setup/binding */
675
Manu Gautama4d993f2011-08-30 18:25:55 +0530676static int
David Brownell61d8bae2008-06-19 18:18:50 -0700677gser_bind(struct usb_configuration *c, struct usb_function *f)
678{
679 struct usb_composite_dev *cdev = c->cdev;
680 struct f_gser *gser = func_to_gser(f);
681 int status;
682 struct usb_ep *ep;
683
684 /* allocate instance-specific interface IDs */
685 status = usb_interface_id(c, f);
686 if (status < 0)
687 goto fail;
688 gser->data_id = status;
689 gser_interface_desc.bInterfaceNumber = status;
690
691 status = -ENODEV;
692
693 /* allocate instance-specific endpoints */
694 ep = usb_ep_autoconfig(cdev->gadget, &gser_fs_in_desc);
695 if (!ep)
696 goto fail;
697 gser->port.in = ep;
698 ep->driver_data = cdev; /* claim */
699
700 ep = usb_ep_autoconfig(cdev->gadget, &gser_fs_out_desc);
701 if (!ep)
702 goto fail;
703 gser->port.out = ep;
704 ep->driver_data = cdev; /* claim */
705
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700706#ifdef CONFIG_MODEM_SUPPORT
707 ep = usb_ep_autoconfig(cdev->gadget, &gser_fs_notify_desc);
708 if (!ep)
709 goto fail;
710 gser->notify = ep;
711 ep->driver_data = cdev; /* claim */
712 /* allocate notification */
713 gser->notify_req = gs_alloc_req(ep,
714 sizeof(struct usb_cdc_notification) + 2,
715 GFP_KERNEL);
716 if (!gser->notify_req)
717 goto fail;
718
719 gser->notify_req->complete = gser_notify_complete;
720 gser->notify_req->context = gser;
721#endif
722
David Brownell61d8bae2008-06-19 18:18:50 -0700723 /* copy descriptors, and track endpoint copies */
724 f->descriptors = usb_copy_descriptors(gser_fs_function);
725
Pavankumar Kondetif0f95d82011-09-23 11:38:57 +0530726 if (!f->descriptors)
727 goto fail;
728
David Brownell61d8bae2008-06-19 18:18:50 -0700729 gser->fs.in = usb_find_endpoint(gser_fs_function,
730 f->descriptors, &gser_fs_in_desc);
731 gser->fs.out = usb_find_endpoint(gser_fs_function,
732 f->descriptors, &gser_fs_out_desc);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700733#ifdef CONFIG_MODEM_SUPPORT
734 gser->fs.notify = usb_find_endpoint(gser_fs_function,
735 f->descriptors, &gser_fs_notify_desc);
736#endif
David Brownell61d8bae2008-06-19 18:18:50 -0700737
738
739 /* support all relevant hardware speeds... we expect that when
740 * hardware is dual speed, all bulk-capable endpoints work at
741 * both speeds
742 */
743 if (gadget_is_dualspeed(c->cdev->gadget)) {
744 gser_hs_in_desc.bEndpointAddress =
745 gser_fs_in_desc.bEndpointAddress;
746 gser_hs_out_desc.bEndpointAddress =
747 gser_fs_out_desc.bEndpointAddress;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700748#ifdef CONFIG_MODEM_SUPPORT
749 gser_hs_notify_desc.bEndpointAddress =
750 gser_fs_notify_desc.bEndpointAddress;
751#endif
David Brownell61d8bae2008-06-19 18:18:50 -0700752
753 /* copy descriptors, and track endpoint copies */
754 f->hs_descriptors = usb_copy_descriptors(gser_hs_function);
755
Pavankumar Kondetif0f95d82011-09-23 11:38:57 +0530756 if (!f->hs_descriptors)
757 goto fail;
758
David Brownell61d8bae2008-06-19 18:18:50 -0700759 gser->hs.in = usb_find_endpoint(gser_hs_function,
760 f->hs_descriptors, &gser_hs_in_desc);
761 gser->hs.out = usb_find_endpoint(gser_hs_function,
762 f->hs_descriptors, &gser_hs_out_desc);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700763#ifdef CONFIG_MODEM_SUPPORT
764 gser->hs.notify = usb_find_endpoint(gser_hs_function,
765 f->hs_descriptors, &gser_hs_notify_desc);
766#endif
David Brownell61d8bae2008-06-19 18:18:50 -0700767 }
768
769 DBG(cdev, "generic ttyGS%d: %s speed IN/%s OUT/%s\n",
770 gser->port_num,
771 gadget_is_dualspeed(c->cdev->gadget) ? "dual" : "full",
772 gser->port.in->name, gser->port.out->name);
773 return 0;
774
775fail:
Pavankumar Kondetif0f95d82011-09-23 11:38:57 +0530776 if (f->descriptors)
777 usb_free_descriptors(f->descriptors);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700778#ifdef CONFIG_MODEM_SUPPORT
779 if (gser->notify_req)
780 gs_free_req(gser->notify, gser->notify_req);
781
782 /* we might as well release our claims on endpoints */
783 if (gser->notify)
784 gser->notify->driver_data = NULL;
785#endif
David Brownell61d8bae2008-06-19 18:18:50 -0700786 /* we might as well release our claims on endpoints */
787 if (gser->port.out)
788 gser->port.out->driver_data = NULL;
789 if (gser->port.in)
790 gser->port.in->driver_data = NULL;
791
792 ERROR(cdev, "%s: can't bind, err %d\n", f->name, status);
793
794 return status;
795}
796
797static void
798gser_unbind(struct usb_configuration *c, struct usb_function *f)
799{
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700800#ifdef CONFIG_MODEM_SUPPORT
801 struct f_gser *gser = func_to_gser(f);
802#endif
David Brownell61d8bae2008-06-19 18:18:50 -0700803 if (gadget_is_dualspeed(c->cdev->gadget))
804 usb_free_descriptors(f->hs_descriptors);
805 usb_free_descriptors(f->descriptors);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700806#ifdef CONFIG_MODEM_SUPPORT
807 gs_free_req(gser->notify, gser->notify_req);
808#endif
David Brownell61d8bae2008-06-19 18:18:50 -0700809 kfree(func_to_gser(f));
810}
811
812/**
813 * gser_bind_config - add a generic serial function to a configuration
814 * @c: the configuration to support the serial instance
815 * @port_num: /dev/ttyGS* port this interface will use
816 * Context: single threaded during gadget setup
817 *
818 * Returns zero on success, else negative errno.
819 *
820 * Caller must have called @gserial_setup() with enough ports to
821 * handle all the ones it binds. Caller is also responsible
822 * for calling @gserial_cleanup() before module unload.
823 */
Manu Gautama4d993f2011-08-30 18:25:55 +0530824int gser_bind_config(struct usb_configuration *c, u8 port_num)
David Brownell61d8bae2008-06-19 18:18:50 -0700825{
826 struct f_gser *gser;
827 int status;
828
829 /* REVISIT might want instance-specific strings to help
830 * distinguish instances ...
831 */
832
833 /* maybe allocate device-global string ID */
834 if (gser_string_defs[0].id == 0) {
835 status = usb_string_id(c->cdev);
836 if (status < 0)
837 return status;
838 gser_string_defs[0].id = status;
839 }
840
841 /* allocate and initialize one new instance */
842 gser = kzalloc(sizeof *gser, GFP_KERNEL);
843 if (!gser)
844 return -ENOMEM;
845
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700846#ifdef CONFIG_MODEM_SUPPORT
847 spin_lock_init(&gser->lock);
848#endif
David Brownell61d8bae2008-06-19 18:18:50 -0700849 gser->port_num = port_num;
850
851 gser->port.func.name = "gser";
852 gser->port.func.strings = gser_strings;
853 gser->port.func.bind = gser_bind;
854 gser->port.func.unbind = gser_unbind;
855 gser->port.func.set_alt = gser_set_alt;
856 gser->port.func.disable = gser_disable;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700857 gser->transport = gserial_ports[port_num].transport;
858#ifdef CONFIG_MODEM_SUPPORT
859 /* We support only two ports for now */
860 if (port_num == 0)
861 gser->port.func.name = "modem";
862 else
863 gser->port.func.name = "nmea";
864 gser->port.func.setup = gser_setup;
865 gser->port.connect = gser_connect;
866 gser->port.get_dtr = gser_get_dtr;
867 gser->port.get_rts = gser_get_rts;
868 gser->port.send_carrier_detect = gser_send_carrier_detect;
869 gser->port.send_ring_indicator = gser_send_ring_indicator;
870 gser->port.send_modem_ctrl_bits = gser_send_modem_ctrl_bits;
871 gser->port.disconnect = gser_disconnect;
872 gser->port.send_break = gser_send_break;
873#endif
David Brownell61d8bae2008-06-19 18:18:50 -0700874
875 status = usb_add_function(c, &gser->port.func);
876 if (status)
877 kfree(gser);
878 return status;
879}
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700880
Manu Gautama4d993f2011-08-30 18:25:55 +0530881/**
882 * gserial_init_port - bind a gserial_port to its transport
883 */
884static int gserial_init_port(int port_num, const char *name)
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700885{
Manu Gautama4d993f2011-08-30 18:25:55 +0530886 enum transport_type transport;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700887
Manu Gautama4d993f2011-08-30 18:25:55 +0530888 if (port_num >= GSERIAL_NO_PORTS)
889 return -ENODEV;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700890
Manu Gautama4d993f2011-08-30 18:25:55 +0530891 transport = serial_str_to_transport(name);
892 pr_debug("%s, port:%d, transport:%s\n", __func__,
893 port_num, transport_to_str(transport));
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700894
Manu Gautama4d993f2011-08-30 18:25:55 +0530895 gserial_ports[port_num].transport = transport;
896 gserial_ports[port_num].port_num = port_num;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700897
Manu Gautama4d993f2011-08-30 18:25:55 +0530898 switch (transport) {
Hemant Kumarbffd4142011-11-03 13:20:40 -0700899 case USB_GADGET_XPORT_TTY:
Manu Gautama4d993f2011-08-30 18:25:55 +0530900 gserial_ports[port_num].client_port_num = no_tty_ports;
901 no_tty_ports++;
902 break;
Hemant Kumarbffd4142011-11-03 13:20:40 -0700903 case USB_GADGET_XPORT_SDIO:
Manu Gautama4d993f2011-08-30 18:25:55 +0530904 gserial_ports[port_num].client_port_num = no_sdio_ports;
905 no_sdio_ports++;
906 break;
Hemant Kumarbffd4142011-11-03 13:20:40 -0700907 case USB_GADGET_XPORT_SMD:
Manu Gautama4d993f2011-08-30 18:25:55 +0530908 gserial_ports[port_num].client_port_num = no_smd_ports;
909 no_smd_ports++;
910 break;
911 default:
912 pr_err("%s: Un-supported transport transport: %u\n",
913 __func__, gserial_ports[port_num].transport);
914 return -ENODEV;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700915 }
916
Manu Gautama4d993f2011-08-30 18:25:55 +0530917 nr_ports++;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700918
919 return 0;
920}