blob: a442352d7b61ed59e195132c4c579680554d3ec0 [file] [log] [blame]
Linus Torvalds1da177e2005-04-16 15:20:36 -07001/*
2 Keyspan USB to Serial Converter driver
Alan Coxdeb91682008-07-22 11:13:08 +01003
Linus Torvalds1da177e2005-04-16 15:20:36 -07004 (C) Copyright (C) 2000-2001 Hugh Blemings <hugh@blemings.org>
5 (C) Copyright (C) 2002 Greg Kroah-Hartman <greg@kroah.com>
Alan Coxdeb91682008-07-22 11:13:08 +01006
Linus Torvalds1da177e2005-04-16 15:20:36 -07007 This program is free software; you can redistribute it and/or modify
8 it under the terms of the GNU General Public License as published by
9 the Free Software Foundation; either version 2 of the License, or
10 (at your option) any later version.
11
Justin P. Mattock631dd1a2010-10-18 11:03:14 +020012 See http://blemings.org/hugh/keyspan.html for more information.
Alan Coxdeb91682008-07-22 11:13:08 +010013
Linus Torvalds1da177e2005-04-16 15:20:36 -070014 Code in this driver inspired by and in a number of places taken
15 from Brian Warner's original Keyspan-PDA driver.
16
17 This driver has been put together with the support of Innosys, Inc.
18 and Keyspan, Inc the manufacturers of the Keyspan USB-serial products.
19 Thanks Guys :)
Alan Coxdeb91682008-07-22 11:13:08 +010020
Linus Torvalds1da177e2005-04-16 15:20:36 -070021 Thanks to Paulus for miscellaneous tidy ups, some largish chunks
22 of much nicer and/or completely new code and (perhaps most uniquely)
23 having the patience to sit down and explain why and where he'd changed
Alan Coxdeb91682008-07-22 11:13:08 +010024 stuff.
25
26 Tip 'o the hat to IBM (and previously Linuxcare :) for supporting
Linus Torvalds1da177e2005-04-16 15:20:36 -070027 staff in their work on open source projects.
28
29 Change History
30
31 2003sep04 LPM (Keyspan) add support for new single port product USA19HS.
32 Improve setup message handling for all devices.
33
34 Wed Feb 19 22:00:00 PST 2003 (Jeffrey S. Laing <keyspan@jsl.com>)
35 Merged the current (1/31/03) Keyspan code with the current (2.4.21-pre4)
36 Linux source tree. The Linux tree lacked support for the 49WLC and
37 others. The Keyspan patches didn't work with the current kernel.
38
39 2003jan30 LPM add support for the 49WLC and MPR
40
41 Wed Apr 25 12:00:00 PST 2002 (Keyspan)
42 Started with Hugh Blemings' code dated Jan 17, 2002. All adapters
43 now supported (including QI and QW). Modified port open, port
44 close, and send setup() logic to fix various data and endpoint
45 synchronization bugs and device LED status bugs. Changed keyspan_
46 write_room() to accurately return transmit buffer availability.
47 Changed forwardingLength from 1 to 16 for all adapters.
48
49 Fri Oct 12 16:45:00 EST 2001
50 Preliminary USA-19QI and USA-28 support (both test OK for me, YMMV)
51
52 Wed Apr 25 12:00:00 PST 2002 (Keyspan)
53 Started with Hugh Blemings' code dated Jan 17, 2002. All adapters
54 now supported (including QI and QW). Modified port open, port
55 close, and send setup() logic to fix various data and endpoint
56 synchronization bugs and device LED status bugs. Changed keyspan_
57 write_room() to accurately return transmit buffer availability.
58 Changed forwardingLength from 1 to 16 for all adapters.
59
60 Fri Oct 12 16:45:00 EST 2001
61 Preliminary USA-19QI and USA-28 support (both test OK for me, YMMV)
62
63 Mon Oct 8 14:29:00 EST 2001 hugh
64 Fixed bug that prevented mulitport devices operating correctly
65 if they weren't the first unit attached.
66
67 Sat Oct 6 12:31:21 EST 2001 hugh
68 Added support for USA-28XA and -28XB, misc cleanups, break support
69 for usa26 based models thanks to David Gibson.
70
71 Thu May 31 11:56:42 PDT 2001 gkh
72 switched from using spinlock to a semaphore
Alan Coxdeb91682008-07-22 11:13:08 +010073
Linus Torvalds1da177e2005-04-16 15:20:36 -070074 (04/08/2001) gb
75 Identify version on module load.
Alan Coxdeb91682008-07-22 11:13:08 +010076
Linus Torvalds1da177e2005-04-16 15:20:36 -070077 (11/01/2000) Adam J. Richter
78 usb_device_id table support.
Alan Coxdeb91682008-07-22 11:13:08 +010079
Linus Torvalds1da177e2005-04-16 15:20:36 -070080 Tue Oct 10 23:15:33 EST 2000 Hugh
81 Merged Paul's changes with my USA-49W mods. Work in progress
82 still...
Alan Coxdeb91682008-07-22 11:13:08 +010083
Linus Torvalds1da177e2005-04-16 15:20:36 -070084 Wed Jul 19 14:00:42 EST 2000 gkh
85 Added module_init and module_exit functions to handle the fact that
86 this driver is a loadable module now.
Alan Coxdeb91682008-07-22 11:13:08 +010087
Linus Torvalds1da177e2005-04-16 15:20:36 -070088 Tue Jul 18 16:14:52 EST 2000 Hugh
89 Basic character input/output for USA-19 now mostly works,
90 fixed at 9600 baud for the moment.
91
92 Sat Jul 8 11:11:48 EST 2000 Hugh
93 First public release - nothing works except the firmware upload.
94 Tested on PPC and x86 architectures, seems to behave...
95*/
96
97
Linus Torvalds1da177e2005-04-16 15:20:36 -070098#include <linux/kernel.h>
99#include <linux/jiffies.h>
100#include <linux/errno.h>
101#include <linux/init.h>
102#include <linux/slab.h>
103#include <linux/tty.h>
104#include <linux/tty_driver.h>
105#include <linux/tty_flip.h>
106#include <linux/module.h>
107#include <linux/spinlock.h>
David Woodhouse2971c572008-05-30 14:04:03 +0300108#include <linux/firmware.h>
109#include <linux/ihex.h>
Alan Coxdeb91682008-07-22 11:13:08 +0100110#include <linux/uaccess.h>
Linus Torvalds1da177e2005-04-16 15:20:36 -0700111#include <linux/usb.h>
Greg Kroah-Hartmana9698882006-07-11 21:22:58 -0700112#include <linux/usb/serial.h>
Linus Torvalds1da177e2005-04-16 15:20:36 -0700113#include "keyspan.h"
114
115static int debug;
116
117/*
118 * Version Information
119 */
Lucy McCoy0ca12682007-05-18 12:10:41 -0700120#define DRIVER_VERSION "v1.1.5"
Linus Torvalds1da177e2005-04-16 15:20:36 -0700121#define DRIVER_AUTHOR "Hugh Blemings <hugh@misc.nu"
122#define DRIVER_DESC "Keyspan USB to Serial Converter Driver"
123
124#define INSTAT_BUFLEN 32
125#define GLOCONT_BUFLEN 64
Lucy McCoy0ca12682007-05-18 12:10:41 -0700126#define INDAT49W_BUFLEN 512
Linus Torvalds1da177e2005-04-16 15:20:36 -0700127
128 /* Per device and per port private data */
129struct keyspan_serial_private {
130 const struct keyspan_device_details *device_details;
131
132 struct urb *instat_urb;
133 char instat_buf[INSTAT_BUFLEN];
134
Alan Coxdeb91682008-07-22 11:13:08 +0100135 /* added to support 49wg, where data from all 4 ports comes in
136 on 1 EP and high-speed supported */
Lucy McCoy0ca12682007-05-18 12:10:41 -0700137 struct urb *indat_urb;
138 char indat_buf[INDAT49W_BUFLEN];
139
Linus Torvalds1da177e2005-04-16 15:20:36 -0700140 /* XXX this one probably will need a lock */
141 struct urb *glocont_urb;
142 char glocont_buf[GLOCONT_BUFLEN];
Alan Coxdeb91682008-07-22 11:13:08 +0100143 char ctrl_buf[8]; /* for EP0 control message */
Linus Torvalds1da177e2005-04-16 15:20:36 -0700144};
145
146struct keyspan_port_private {
147 /* Keep track of which input & output endpoints to use */
148 int in_flip;
149 int out_flip;
150
151 /* Keep duplicate of device details in each port
152 structure as well - simplifies some of the
153 callback functions etc. */
154 const struct keyspan_device_details *device_details;
155
156 /* Input endpoints and buffer for this port */
157 struct urb *in_urbs[2];
158 char in_buffer[2][64];
159 /* Output endpoints and buffer for this port */
160 struct urb *out_urbs[2];
161 char out_buffer[2][64];
162
163 /* Input ack endpoint */
164 struct urb *inack_urb;
165 char inack_buffer[1];
166
167 /* Output control endpoint */
168 struct urb *outcont_urb;
169 char outcont_buffer[64];
170
171 /* Settings for the port */
172 int baud;
173 int old_baud;
174 unsigned int cflag;
175 unsigned int old_cflag;
176 enum {flow_none, flow_cts, flow_xon} flow_control;
177 int rts_state; /* Handshaking pins (outputs) */
178 int dtr_state;
179 int cts_state; /* Handshaking pins (inputs) */
180 int dsr_state;
181 int dcd_state;
182 int ri_state;
183 int break_on;
184
185 unsigned long tx_start_time[2];
186 int resend_cont; /* need to resend control packet */
187};
188
Linus Torvalds1da177e2005-04-16 15:20:36 -0700189/* Include Keyspan message headers. All current Keyspan Adapters
Lucy McCoy0ca12682007-05-18 12:10:41 -0700190 make use of one of five message formats which are referred
Alan Coxdeb91682008-07-22 11:13:08 +0100191 to as USA-26, USA-28, USA-49, USA-90, USA-67 by Keyspan and
192 within this driver. */
Linus Torvalds1da177e2005-04-16 15:20:36 -0700193#include "keyspan_usa26msg.h"
194#include "keyspan_usa28msg.h"
195#include "keyspan_usa49msg.h"
196#include "keyspan_usa90msg.h"
Lucy McCoy0ca12682007-05-18 12:10:41 -0700197#include "keyspan_usa67msg.h"
Alan Coxdeb91682008-07-22 11:13:08 +0100198
Linus Torvalds1da177e2005-04-16 15:20:36 -0700199
200/* Functions used by new usb-serial code. */
Alan Coxdeb91682008-07-22 11:13:08 +0100201static int __init keyspan_init(void)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700202{
203 int retval;
204 retval = usb_serial_register(&keyspan_pre_device);
205 if (retval)
206 goto failed_pre_device_register;
207 retval = usb_serial_register(&keyspan_1port_device);
208 if (retval)
209 goto failed_1port_device_register;
210 retval = usb_serial_register(&keyspan_2port_device);
211 if (retval)
212 goto failed_2port_device_register;
213 retval = usb_serial_register(&keyspan_4port_device);
214 if (retval)
215 goto failed_4port_device_register;
216 retval = usb_register(&keyspan_driver);
Alan Coxdeb91682008-07-22 11:13:08 +0100217 if (retval)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700218 goto failed_usb_register;
219
Greg Kroah-Hartmanc197a8d2008-08-18 13:21:04 -0700220 printk(KERN_INFO KBUILD_MODNAME ": " DRIVER_VERSION ":"
221 DRIVER_DESC "\n");
Linus Torvalds1da177e2005-04-16 15:20:36 -0700222
223 return 0;
224failed_usb_register:
225 usb_serial_deregister(&keyspan_4port_device);
226failed_4port_device_register:
227 usb_serial_deregister(&keyspan_2port_device);
228failed_2port_device_register:
229 usb_serial_deregister(&keyspan_1port_device);
230failed_1port_device_register:
231 usb_serial_deregister(&keyspan_pre_device);
232failed_pre_device_register:
233 return retval;
234}
235
Alan Coxdeb91682008-07-22 11:13:08 +0100236static void __exit keyspan_exit(void)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700237{
Alan Coxdeb91682008-07-22 11:13:08 +0100238 usb_deregister(&keyspan_driver);
239 usb_serial_deregister(&keyspan_pre_device);
240 usb_serial_deregister(&keyspan_1port_device);
241 usb_serial_deregister(&keyspan_2port_device);
242 usb_serial_deregister(&keyspan_4port_device);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700243}
244
245module_init(keyspan_init);
246module_exit(keyspan_exit);
247
Alan Cox95da3102008-07-22 11:09:07 +0100248static void keyspan_break_ctl(struct tty_struct *tty, int break_state)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700249{
Alan Cox95da3102008-07-22 11:09:07 +0100250 struct usb_serial_port *port = tty->driver_data;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700251 struct keyspan_port_private *p_priv;
252
Alan Coxdeb91682008-07-22 11:13:08 +0100253 dbg("%s", __func__);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700254
255 p_priv = usb_get_serial_port_data(port);
256
257 if (break_state == -1)
258 p_priv->break_on = 1;
259 else
260 p_priv->break_on = 0;
261
262 keyspan_send_setup(port, 0);
263}
264
265
Alan Coxdeb91682008-07-22 11:13:08 +0100266static void keyspan_set_termios(struct tty_struct *tty,
Alan Cox95da3102008-07-22 11:09:07 +0100267 struct usb_serial_port *port, struct ktermios *old_termios)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700268{
269 int baud_rate, device_port;
270 struct keyspan_port_private *p_priv;
271 const struct keyspan_device_details *d_details;
272 unsigned int cflag;
273
Harvey Harrison441b62c2008-03-03 16:08:34 -0800274 dbg("%s", __func__);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700275
276 p_priv = usb_get_serial_port_data(port);
277 d_details = p_priv->device_details;
Alan Cox74240b02007-10-18 01:24:20 -0700278 cflag = tty->termios->c_cflag;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700279 device_port = port->number - port->serial->minor;
280
281 /* Baud rate calculation takes baud rate as an integer
282 so other rates can be generated if desired. */
Alan Cox74240b02007-10-18 01:24:20 -0700283 baud_rate = tty_get_baud_rate(tty);
Alan Coxdeb91682008-07-22 11:13:08 +0100284 /* If no match or invalid, don't change */
Alan Cox74240b02007-10-18 01:24:20 -0700285 if (d_details->calculate_baud_rate(baud_rate, d_details->baudclk,
Linus Torvalds1da177e2005-04-16 15:20:36 -0700286 NULL, NULL, NULL, device_port) == KEYSPAN_BAUD_RATE_OK) {
287 /* FIXME - more to do here to ensure rate changes cleanly */
Alan Cox74240b02007-10-18 01:24:20 -0700288 /* FIXME - calcuate exact rate from divisor ? */
Linus Torvalds1da177e2005-04-16 15:20:36 -0700289 p_priv->baud = baud_rate;
Alan Cox74240b02007-10-18 01:24:20 -0700290 } else
291 baud_rate = tty_termios_baud_rate(old_termios);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700292
Alan Cox74240b02007-10-18 01:24:20 -0700293 tty_encode_baud_rate(tty, baud_rate, baud_rate);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700294 /* set CTS/RTS handshake etc. */
295 p_priv->cflag = cflag;
296 p_priv->flow_control = (cflag & CRTSCTS)? flow_cts: flow_none;
297
Alan Cox74240b02007-10-18 01:24:20 -0700298 /* Mark/Space not supported */
299 tty->termios->c_cflag &= ~CMSPAR;
300
Linus Torvalds1da177e2005-04-16 15:20:36 -0700301 keyspan_send_setup(port, 0);
302}
303
Alan Cox60b33c12011-02-14 16:26:14 +0000304static int keyspan_tiocmget(struct tty_struct *tty)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700305{
Alan Cox95da3102008-07-22 11:09:07 +0100306 struct usb_serial_port *port = tty->driver_data;
307 struct keyspan_port_private *p_priv = usb_get_serial_port_data(port);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700308 unsigned int value;
Alan Coxdeb91682008-07-22 11:13:08 +0100309
Linus Torvalds1da177e2005-04-16 15:20:36 -0700310 value = ((p_priv->rts_state) ? TIOCM_RTS : 0) |
311 ((p_priv->dtr_state) ? TIOCM_DTR : 0) |
312 ((p_priv->cts_state) ? TIOCM_CTS : 0) |
313 ((p_priv->dsr_state) ? TIOCM_DSR : 0) |
314 ((p_priv->dcd_state) ? TIOCM_CAR : 0) |
Alan Coxdeb91682008-07-22 11:13:08 +0100315 ((p_priv->ri_state) ? TIOCM_RNG : 0);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700316
317 return value;
318}
319
Alan Cox20b9d172011-02-14 16:26:50 +0000320static int keyspan_tiocmset(struct tty_struct *tty,
Linus Torvalds1da177e2005-04-16 15:20:36 -0700321 unsigned int set, unsigned int clear)
322{
Alan Cox95da3102008-07-22 11:09:07 +0100323 struct usb_serial_port *port = tty->driver_data;
324 struct keyspan_port_private *p_priv = usb_get_serial_port_data(port);
Alan Coxdeb91682008-07-22 11:13:08 +0100325
Linus Torvalds1da177e2005-04-16 15:20:36 -0700326 if (set & TIOCM_RTS)
327 p_priv->rts_state = 1;
328 if (set & TIOCM_DTR)
329 p_priv->dtr_state = 1;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700330 if (clear & TIOCM_RTS)
331 p_priv->rts_state = 0;
332 if (clear & TIOCM_DTR)
333 p_priv->dtr_state = 0;
334 keyspan_send_setup(port, 0);
335 return 0;
336}
337
Alan Cox95da3102008-07-22 11:09:07 +0100338/* Write function is similar for the four protocols used
339 with only a minor change for usa90 (usa19hs) required */
340static int keyspan_write(struct tty_struct *tty,
341 struct usb_serial_port *port, const unsigned char *buf, int count)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700342{
343 struct keyspan_port_private *p_priv;
344 const struct keyspan_device_details *d_details;
345 int flip;
346 int left, todo;
347 struct urb *this_urb;
Alan Coxdeb91682008-07-22 11:13:08 +0100348 int err, maxDataLen, dataOffset;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700349
350 p_priv = usb_get_serial_port_data(port);
351 d_details = p_priv->device_details;
352
353 if (d_details->msg_format == msg_usa90) {
Alan Coxdeb91682008-07-22 11:13:08 +0100354 maxDataLen = 64;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700355 dataOffset = 0;
356 } else {
357 maxDataLen = 63;
358 dataOffset = 1;
359 }
Alan Coxdeb91682008-07-22 11:13:08 +0100360
Linus Torvalds1da177e2005-04-16 15:20:36 -0700361 dbg("%s - for port %d (%d chars), flip=%d",
Harvey Harrison441b62c2008-03-03 16:08:34 -0800362 __func__, port->number, count, p_priv->out_flip);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700363
364 for (left = count; left > 0; left -= todo) {
365 todo = left;
366 if (todo > maxDataLen)
367 todo = maxDataLen;
368
369 flip = p_priv->out_flip;
Alan Coxdeb91682008-07-22 11:13:08 +0100370
Linus Torvalds1da177e2005-04-16 15:20:36 -0700371 /* Check we have a valid urb/endpoint before we use it... */
Alan Coxdeb91682008-07-22 11:13:08 +0100372 this_urb = p_priv->out_urbs[flip];
373 if (this_urb == NULL) {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700374 /* no bulk out, so return 0 bytes written */
Harvey Harrison441b62c2008-03-03 16:08:34 -0800375 dbg("%s - no output urb :(", __func__);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700376 return count;
377 }
378
Alan Coxdeb91682008-07-22 11:13:08 +0100379 dbg("%s - endpoint %d flip %d",
380 __func__, usb_pipeendpoint(this_urb->pipe), flip);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700381
382 if (this_urb->status == -EINPROGRESS) {
Alan Coxdeb91682008-07-22 11:13:08 +0100383 if (time_before(jiffies,
384 p_priv->tx_start_time[flip] + 10 * HZ))
Linus Torvalds1da177e2005-04-16 15:20:36 -0700385 break;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700386 usb_unlink_urb(this_urb);
387 break;
388 }
389
Alan Coxdeb91682008-07-22 11:13:08 +0100390 /* First byte in buffer is "last flag" (except for usa19hx)
391 - unused so for now so set to zero */
Linus Torvalds1da177e2005-04-16 15:20:36 -0700392 ((char *)this_urb->transfer_buffer)[0] = 0;
393
Alan Coxdeb91682008-07-22 11:13:08 +0100394 memcpy(this_urb->transfer_buffer + dataOffset, buf, todo);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700395 buf += todo;
396
397 /* send the data out the bulk port */
398 this_urb->transfer_buffer_length = todo + dataOffset;
399
Linus Torvalds1da177e2005-04-16 15:20:36 -0700400 this_urb->dev = port->serial->dev;
Alan Coxdeb91682008-07-22 11:13:08 +0100401 err = usb_submit_urb(this_urb, GFP_ATOMIC);
402 if (err != 0)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700403 dbg("usb_submit_urb(write bulk) failed (%d)", err);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700404 p_priv->tx_start_time[flip] = jiffies;
405
406 /* Flip for next time if usa26 or usa28 interface
407 (not used on usa49) */
408 p_priv->out_flip = (flip + 1) & d_details->outdat_endp_flip;
409 }
410
411 return count - left;
412}
413
David Howells7d12e782006-10-05 14:55:46 +0100414static void usa26_indat_callback(struct urb *urb)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700415{
416 int i, err;
417 int endpoint;
418 struct usb_serial_port *port;
419 struct tty_struct *tty;
420 unsigned char *data = urb->transfer_buffer;
Greg Kroah-Hartman95b93452007-06-15 15:44:13 -0700421 int status = urb->status;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700422
Alan Coxdeb91682008-07-22 11:13:08 +0100423 dbg("%s", __func__);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700424
425 endpoint = usb_pipeendpoint(urb->pipe);
426
Greg Kroah-Hartman95b93452007-06-15 15:44:13 -0700427 if (status) {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700428 dbg("%s - nonzero status: %x on endpoint %d.",
Harvey Harrison441b62c2008-03-03 16:08:34 -0800429 __func__, status, endpoint);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700430 return;
431 }
432
Ming Leicdc97792008-02-24 18:41:47 +0800433 port = urb->context;
Alan Cox4a90f092008-10-13 10:39:46 +0100434 tty = tty_port_tty_get(&port->port);
Alan Coxa5569a52008-01-21 17:18:24 -0800435 if (tty && urb->actual_length) {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700436 /* 0x80 bit is error flag */
437 if ((data[0] & 0x80) == 0) {
Alan Coxdeb91682008-07-22 11:13:08 +0100438 /* no errors on individual bytes, only
439 possible overrun err */
Linus Torvalds1da177e2005-04-16 15:20:36 -0700440 if (data[0] & RXERROR_OVERRUN)
Alan Coxdeb91682008-07-22 11:13:08 +0100441 err = TTY_OVERRUN;
442 else
443 err = 0;
444 for (i = 1; i < urb->actual_length ; ++i)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700445 tty_insert_flip_char(tty, data[i], err);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700446 } else {
447 /* some bytes had errors, every byte has status */
Harvey Harrison441b62c2008-03-03 16:08:34 -0800448 dbg("%s - RX error!!!!", __func__);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700449 for (i = 0; i + 1 < urb->actual_length; i += 2) {
450 int stat = data[i], flag = 0;
451 if (stat & RXERROR_OVERRUN)
452 flag |= TTY_OVERRUN;
453 if (stat & RXERROR_FRAMING)
454 flag |= TTY_FRAME;
455 if (stat & RXERROR_PARITY)
456 flag |= TTY_PARITY;
457 /* XXX should handle break (0x10) */
458 tty_insert_flip_char(tty, data[i+1], flag);
459 }
460 }
461 tty_flip_buffer_push(tty);
462 }
Alan Cox4a90f092008-10-13 10:39:46 +0100463 tty_kref_put(tty);
Alan Coxdeb91682008-07-22 11:13:08 +0100464
465 /* Resubmit urb so we continue receiving */
Linus Torvalds1da177e2005-04-16 15:20:36 -0700466 urb->dev = port->serial->dev;
Alan Stern1f871582010-02-17 10:05:47 -0500467 err = usb_submit_urb(urb, GFP_ATOMIC);
468 if (err != 0)
469 dbg("%s - resubmit read urb failed. (%d)", __func__, err);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700470}
471
Alan Coxdeb91682008-07-22 11:13:08 +0100472/* Outdat handling is common for all devices */
David Howells7d12e782006-10-05 14:55:46 +0100473static void usa2x_outdat_callback(struct urb *urb)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700474{
475 struct usb_serial_port *port;
476 struct keyspan_port_private *p_priv;
477
Ming Leicdc97792008-02-24 18:41:47 +0800478 port = urb->context;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700479 p_priv = usb_get_serial_port_data(port);
Alan Coxdeb91682008-07-22 11:13:08 +0100480 dbg("%s - urb %d", __func__, urb == p_priv->out_urbs[1]);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700481
Alan Stern1f871582010-02-17 10:05:47 -0500482 usb_serial_port_softint(port);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700483}
484
David Howells7d12e782006-10-05 14:55:46 +0100485static void usa26_inack_callback(struct urb *urb)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700486{
Alan Coxdeb91682008-07-22 11:13:08 +0100487 dbg("%s", __func__);
488
Linus Torvalds1da177e2005-04-16 15:20:36 -0700489}
490
David Howells7d12e782006-10-05 14:55:46 +0100491static void usa26_outcont_callback(struct urb *urb)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700492{
493 struct usb_serial_port *port;
494 struct keyspan_port_private *p_priv;
495
Ming Leicdc97792008-02-24 18:41:47 +0800496 port = urb->context;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700497 p_priv = usb_get_serial_port_data(port);
498
499 if (p_priv->resend_cont) {
Alan Coxdeb91682008-07-22 11:13:08 +0100500 dbg("%s - sending setup", __func__);
501 keyspan_usa26_send_setup(port->serial, port,
502 p_priv->resend_cont - 1);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700503 }
504}
505
David Howells7d12e782006-10-05 14:55:46 +0100506static void usa26_instat_callback(struct urb *urb)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700507{
508 unsigned char *data = urb->transfer_buffer;
509 struct keyspan_usa26_portStatusMessage *msg;
510 struct usb_serial *serial;
511 struct usb_serial_port *port;
512 struct keyspan_port_private *p_priv;
Alan Cox4a90f092008-10-13 10:39:46 +0100513 struct tty_struct *tty;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700514 int old_dcd_state, err;
Greg Kroah-Hartman95b93452007-06-15 15:44:13 -0700515 int status = urb->status;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700516
Ming Leicdc97792008-02-24 18:41:47 +0800517 serial = urb->context;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700518
Greg Kroah-Hartman95b93452007-06-15 15:44:13 -0700519 if (status) {
Harvey Harrison441b62c2008-03-03 16:08:34 -0800520 dbg("%s - nonzero status: %x", __func__, status);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700521 return;
522 }
523 if (urb->actual_length != 9) {
Harvey Harrison441b62c2008-03-03 16:08:34 -0800524 dbg("%s - %d byte report??", __func__, urb->actual_length);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700525 goto exit;
526 }
527
528 msg = (struct keyspan_usa26_portStatusMessage *)data;
529
530#if 0
531 dbg("%s - port status: port %d cts %d dcd %d dsr %d ri %d toff %d txoff %d rxen %d cr %d",
Harvey Harrison441b62c2008-03-03 16:08:34 -0800532 __func__, msg->port, msg->hskia_cts, msg->gpia_dcd, msg->dsr, msg->ri, msg->_txOff,
Linus Torvalds1da177e2005-04-16 15:20:36 -0700533 msg->_txXoff, msg->rxEnabled, msg->controlResponse);
534#endif
535
536 /* Now do something useful with the data */
537
538
Alan Coxdeb91682008-07-22 11:13:08 +0100539 /* Check port number from message and retrieve private data */
Linus Torvalds1da177e2005-04-16 15:20:36 -0700540 if (msg->port >= serial->num_ports) {
Alan Coxdeb91682008-07-22 11:13:08 +0100541 dbg("%s - Unexpected port number %d", __func__, msg->port);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700542 goto exit;
543 }
544 port = serial->port[msg->port];
545 p_priv = usb_get_serial_port_data(port);
Alan Coxdeb91682008-07-22 11:13:08 +0100546
Linus Torvalds1da177e2005-04-16 15:20:36 -0700547 /* Update handshaking pin state information */
548 old_dcd_state = p_priv->dcd_state;
549 p_priv->cts_state = ((msg->hskia_cts) ? 1 : 0);
550 p_priv->dsr_state = ((msg->dsr) ? 1 : 0);
551 p_priv->dcd_state = ((msg->gpia_dcd) ? 1 : 0);
552 p_priv->ri_state = ((msg->ri) ? 1 : 0);
553
Alan Cox4a90f092008-10-13 10:39:46 +0100554 if (old_dcd_state != p_priv->dcd_state) {
555 tty = tty_port_tty_get(&port->port);
556 if (tty && !C_CLOCAL(tty))
557 tty_hangup(tty);
558 tty_kref_put(tty);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700559 }
Alan Coxdeb91682008-07-22 11:13:08 +0100560
Linus Torvalds1da177e2005-04-16 15:20:36 -0700561 /* Resubmit urb so we continue receiving */
562 urb->dev = serial->dev;
Alan Coxdeb91682008-07-22 11:13:08 +0100563 err = usb_submit_urb(urb, GFP_ATOMIC);
564 if (err != 0)
Harvey Harrison441b62c2008-03-03 16:08:34 -0800565 dbg("%s - resubmit read urb failed. (%d)", __func__, err);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700566exit: ;
567}
568
David Howells7d12e782006-10-05 14:55:46 +0100569static void usa26_glocont_callback(struct urb *urb)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700570{
Alan Coxdeb91682008-07-22 11:13:08 +0100571 dbg("%s", __func__);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700572}
573
574
David Howells7d12e782006-10-05 14:55:46 +0100575static void usa28_indat_callback(struct urb *urb)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700576{
Alan Coxf035a8a2008-07-22 11:13:32 +0100577 int err;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700578 struct usb_serial_port *port;
579 struct tty_struct *tty;
580 unsigned char *data;
581 struct keyspan_port_private *p_priv;
Greg Kroah-Hartman95b93452007-06-15 15:44:13 -0700582 int status = urb->status;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700583
Alan Coxdeb91682008-07-22 11:13:08 +0100584 dbg("%s", __func__);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700585
Ming Leicdc97792008-02-24 18:41:47 +0800586 port = urb->context;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700587 p_priv = usb_get_serial_port_data(port);
588 data = urb->transfer_buffer;
589
590 if (urb != p_priv->in_urbs[p_priv->in_flip])
591 return;
592
593 do {
Greg Kroah-Hartman95b93452007-06-15 15:44:13 -0700594 if (status) {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700595 dbg("%s - nonzero status: %x on endpoint %d.",
Harvey Harrison441b62c2008-03-03 16:08:34 -0800596 __func__, status, usb_pipeendpoint(urb->pipe));
Linus Torvalds1da177e2005-04-16 15:20:36 -0700597 return;
598 }
599
Ming Leicdc97792008-02-24 18:41:47 +0800600 port = urb->context;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700601 p_priv = usb_get_serial_port_data(port);
602 data = urb->transfer_buffer;
603
Alan Cox4a90f092008-10-13 10:39:46 +0100604 tty =tty_port_tty_get(&port->port);
605 if (tty && urb->actual_length) {
Alan Coxf035a8a2008-07-22 11:13:32 +0100606 tty_insert_flip_string(tty, data, urb->actual_length);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700607 tty_flip_buffer_push(tty);
608 }
Alan Cox4a90f092008-10-13 10:39:46 +0100609 tty_kref_put(tty);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700610
611 /* Resubmit urb so we continue receiving */
612 urb->dev = port->serial->dev;
Alan Stern1f871582010-02-17 10:05:47 -0500613 err = usb_submit_urb(urb, GFP_ATOMIC);
614 if (err != 0)
615 dbg("%s - resubmit read urb failed. (%d)",
616 __func__, err);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700617 p_priv->in_flip ^= 1;
618
619 urb = p_priv->in_urbs[p_priv->in_flip];
620 } while (urb->status != -EINPROGRESS);
621}
622
David Howells7d12e782006-10-05 14:55:46 +0100623static void usa28_inack_callback(struct urb *urb)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700624{
Alan Coxdeb91682008-07-22 11:13:08 +0100625 dbg("%s", __func__);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700626}
627
David Howells7d12e782006-10-05 14:55:46 +0100628static void usa28_outcont_callback(struct urb *urb)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700629{
630 struct usb_serial_port *port;
631 struct keyspan_port_private *p_priv;
632
Ming Leicdc97792008-02-24 18:41:47 +0800633 port = urb->context;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700634 p_priv = usb_get_serial_port_data(port);
635
636 if (p_priv->resend_cont) {
Alan Coxdeb91682008-07-22 11:13:08 +0100637 dbg("%s - sending setup", __func__);
638 keyspan_usa28_send_setup(port->serial, port,
639 p_priv->resend_cont - 1);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700640 }
641}
642
David Howells7d12e782006-10-05 14:55:46 +0100643static void usa28_instat_callback(struct urb *urb)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700644{
645 int err;
646 unsigned char *data = urb->transfer_buffer;
647 struct keyspan_usa28_portStatusMessage *msg;
648 struct usb_serial *serial;
649 struct usb_serial_port *port;
650 struct keyspan_port_private *p_priv;
Alan Cox4a90f092008-10-13 10:39:46 +0100651 struct tty_struct *tty;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700652 int old_dcd_state;
Greg Kroah-Hartman95b93452007-06-15 15:44:13 -0700653 int status = urb->status;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700654
Ming Leicdc97792008-02-24 18:41:47 +0800655 serial = urb->context;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700656
Greg Kroah-Hartman95b93452007-06-15 15:44:13 -0700657 if (status) {
Harvey Harrison441b62c2008-03-03 16:08:34 -0800658 dbg("%s - nonzero status: %x", __func__, status);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700659 return;
660 }
661
662 if (urb->actual_length != sizeof(struct keyspan_usa28_portStatusMessage)) {
Harvey Harrison441b62c2008-03-03 16:08:34 -0800663 dbg("%s - bad length %d", __func__, urb->actual_length);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700664 goto exit;
665 }
666
Harvey Harrison441b62c2008-03-03 16:08:34 -0800667 /*dbg("%s %x %x %x %x %x %x %x %x %x %x %x %x", __func__
Linus Torvalds1da177e2005-04-16 15:20:36 -0700668 data[0], data[1], data[2], data[3], data[4], data[5],
669 data[6], data[7], data[8], data[9], data[10], data[11]);*/
Alan Coxdeb91682008-07-22 11:13:08 +0100670
671 /* Now do something useful with the data */
Linus Torvalds1da177e2005-04-16 15:20:36 -0700672 msg = (struct keyspan_usa28_portStatusMessage *)data;
673
Alan Coxdeb91682008-07-22 11:13:08 +0100674 /* Check port number from message and retrieve private data */
Linus Torvalds1da177e2005-04-16 15:20:36 -0700675 if (msg->port >= serial->num_ports) {
Alan Coxdeb91682008-07-22 11:13:08 +0100676 dbg("%s - Unexpected port number %d", __func__, msg->port);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700677 goto exit;
678 }
679 port = serial->port[msg->port];
680 p_priv = usb_get_serial_port_data(port);
Alan Coxdeb91682008-07-22 11:13:08 +0100681
Linus Torvalds1da177e2005-04-16 15:20:36 -0700682 /* Update handshaking pin state information */
683 old_dcd_state = p_priv->dcd_state;
684 p_priv->cts_state = ((msg->cts) ? 1 : 0);
685 p_priv->dsr_state = ((msg->dsr) ? 1 : 0);
686 p_priv->dcd_state = ((msg->dcd) ? 1 : 0);
687 p_priv->ri_state = ((msg->ri) ? 1 : 0);
688
Alan Cox4a90f092008-10-13 10:39:46 +0100689 if( old_dcd_state != p_priv->dcd_state && old_dcd_state) {
690 tty = tty_port_tty_get(&port->port);
691 if (tty && !C_CLOCAL(tty))
692 tty_hangup(tty);
693 tty_kref_put(tty);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700694 }
695
696 /* Resubmit urb so we continue receiving */
697 urb->dev = serial->dev;
Alan Coxdeb91682008-07-22 11:13:08 +0100698 err = usb_submit_urb(urb, GFP_ATOMIC);
699 if (err != 0)
Harvey Harrison441b62c2008-03-03 16:08:34 -0800700 dbg("%s - resubmit read urb failed. (%d)", __func__, err);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700701exit: ;
702}
703
David Howells7d12e782006-10-05 14:55:46 +0100704static void usa28_glocont_callback(struct urb *urb)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700705{
Alan Coxdeb91682008-07-22 11:13:08 +0100706 dbg("%s", __func__);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700707}
708
709
David Howells7d12e782006-10-05 14:55:46 +0100710static void usa49_glocont_callback(struct urb *urb)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700711{
712 struct usb_serial *serial;
713 struct usb_serial_port *port;
714 struct keyspan_port_private *p_priv;
715 int i;
716
Alan Coxdeb91682008-07-22 11:13:08 +0100717 dbg("%s", __func__);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700718
Ming Leicdc97792008-02-24 18:41:47 +0800719 serial = urb->context;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700720 for (i = 0; i < serial->num_ports; ++i) {
721 port = serial->port[i];
722 p_priv = usb_get_serial_port_data(port);
723
724 if (p_priv->resend_cont) {
Alan Coxdeb91682008-07-22 11:13:08 +0100725 dbg("%s - sending setup", __func__);
726 keyspan_usa49_send_setup(serial, port,
727 p_priv->resend_cont - 1);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700728 break;
729 }
730 }
731}
732
733 /* This is actually called glostat in the Keyspan
734 doco */
David Howells7d12e782006-10-05 14:55:46 +0100735static void usa49_instat_callback(struct urb *urb)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700736{
737 int err;
738 unsigned char *data = urb->transfer_buffer;
739 struct keyspan_usa49_portStatusMessage *msg;
740 struct usb_serial *serial;
741 struct usb_serial_port *port;
742 struct keyspan_port_private *p_priv;
743 int old_dcd_state;
Greg Kroah-Hartman95b93452007-06-15 15:44:13 -0700744 int status = urb->status;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700745
Alan Coxdeb91682008-07-22 11:13:08 +0100746 dbg("%s", __func__);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700747
Ming Leicdc97792008-02-24 18:41:47 +0800748 serial = urb->context;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700749
Greg Kroah-Hartman95b93452007-06-15 15:44:13 -0700750 if (status) {
Harvey Harrison441b62c2008-03-03 16:08:34 -0800751 dbg("%s - nonzero status: %x", __func__, status);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700752 return;
753 }
754
Alan Coxdeb91682008-07-22 11:13:08 +0100755 if (urb->actual_length !=
756 sizeof(struct keyspan_usa49_portStatusMessage)) {
Harvey Harrison441b62c2008-03-03 16:08:34 -0800757 dbg("%s - bad length %d", __func__, urb->actual_length);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700758 goto exit;
759 }
760
Harvey Harrison441b62c2008-03-03 16:08:34 -0800761 /*dbg(" %x %x %x %x %x %x %x %x %x %x %x", __func__,
Linus Torvalds1da177e2005-04-16 15:20:36 -0700762 data[0], data[1], data[2], data[3], data[4], data[5],
763 data[6], data[7], data[8], data[9], data[10]);*/
Alan Coxdeb91682008-07-22 11:13:08 +0100764
765 /* Now do something useful with the data */
Linus Torvalds1da177e2005-04-16 15:20:36 -0700766 msg = (struct keyspan_usa49_portStatusMessage *)data;
767
Alan Coxdeb91682008-07-22 11:13:08 +0100768 /* Check port number from message and retrieve private data */
Linus Torvalds1da177e2005-04-16 15:20:36 -0700769 if (msg->portNumber >= serial->num_ports) {
Alan Coxdeb91682008-07-22 11:13:08 +0100770 dbg("%s - Unexpected port number %d",
771 __func__, msg->portNumber);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700772 goto exit;
773 }
774 port = serial->port[msg->portNumber];
775 p_priv = usb_get_serial_port_data(port);
Alan Coxdeb91682008-07-22 11:13:08 +0100776
Linus Torvalds1da177e2005-04-16 15:20:36 -0700777 /* Update handshaking pin state information */
778 old_dcd_state = p_priv->dcd_state;
779 p_priv->cts_state = ((msg->cts) ? 1 : 0);
780 p_priv->dsr_state = ((msg->dsr) ? 1 : 0);
781 p_priv->dcd_state = ((msg->dcd) ? 1 : 0);
782 p_priv->ri_state = ((msg->ri) ? 1 : 0);
783
Alan Cox4a90f092008-10-13 10:39:46 +0100784 if (old_dcd_state != p_priv->dcd_state && old_dcd_state) {
785 struct tty_struct *tty = tty_port_tty_get(&port->port);
786 if (tty && !C_CLOCAL(tty))
787 tty_hangup(tty);
788 tty_kref_put(tty);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700789 }
790
Alan Coxdeb91682008-07-22 11:13:08 +0100791 /* Resubmit urb so we continue receiving */
Linus Torvalds1da177e2005-04-16 15:20:36 -0700792 urb->dev = serial->dev;
793
Alan Coxdeb91682008-07-22 11:13:08 +0100794 err = usb_submit_urb(urb, GFP_ATOMIC);
795 if (err != 0)
Harvey Harrison441b62c2008-03-03 16:08:34 -0800796 dbg("%s - resubmit read urb failed. (%d)", __func__, err);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700797exit: ;
798}
799
David Howells7d12e782006-10-05 14:55:46 +0100800static void usa49_inack_callback(struct urb *urb)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700801{
Alan Coxdeb91682008-07-22 11:13:08 +0100802 dbg("%s", __func__);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700803}
804
David Howells7d12e782006-10-05 14:55:46 +0100805static void usa49_indat_callback(struct urb *urb)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700806{
807 int i, err;
808 int endpoint;
809 struct usb_serial_port *port;
810 struct tty_struct *tty;
811 unsigned char *data = urb->transfer_buffer;
Greg Kroah-Hartman95b93452007-06-15 15:44:13 -0700812 int status = urb->status;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700813
Alan Coxdeb91682008-07-22 11:13:08 +0100814 dbg("%s", __func__);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700815
816 endpoint = usb_pipeendpoint(urb->pipe);
817
Greg Kroah-Hartman95b93452007-06-15 15:44:13 -0700818 if (status) {
Harvey Harrison441b62c2008-03-03 16:08:34 -0800819 dbg("%s - nonzero status: %x on endpoint %d.", __func__,
Greg Kroah-Hartman95b93452007-06-15 15:44:13 -0700820 status, endpoint);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700821 return;
822 }
823
Ming Leicdc97792008-02-24 18:41:47 +0800824 port = urb->context;
Alan Cox4a90f092008-10-13 10:39:46 +0100825 tty = tty_port_tty_get(&port->port);
Alan Cox3004e532008-01-03 16:59:04 +0000826 if (tty && urb->actual_length) {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700827 /* 0x80 bit is error flag */
828 if ((data[0] & 0x80) == 0) {
829 /* no error on any byte */
Alan Coxf035a8a2008-07-22 11:13:32 +0100830 tty_insert_flip_string(tty, data + 1,
831 urb->actual_length - 1);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700832 } else {
833 /* some bytes had errors, every byte has status */
834 for (i = 0; i + 1 < urb->actual_length; i += 2) {
835 int stat = data[i], flag = 0;
836 if (stat & RXERROR_OVERRUN)
837 flag |= TTY_OVERRUN;
838 if (stat & RXERROR_FRAMING)
839 flag |= TTY_FRAME;
840 if (stat & RXERROR_PARITY)
841 flag |= TTY_PARITY;
842 /* XXX should handle break (0x10) */
843 tty_insert_flip_char(tty, data[i+1], flag);
844 }
845 }
846 tty_flip_buffer_push(tty);
847 }
Alan Cox4a90f092008-10-13 10:39:46 +0100848 tty_kref_put(tty);
Alan Coxdeb91682008-07-22 11:13:08 +0100849
850 /* Resubmit urb so we continue receiving */
Linus Torvalds1da177e2005-04-16 15:20:36 -0700851 urb->dev = port->serial->dev;
Alan Stern1f871582010-02-17 10:05:47 -0500852 err = usb_submit_urb(urb, GFP_ATOMIC);
853 if (err != 0)
854 dbg("%s - resubmit read urb failed. (%d)", __func__, err);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700855}
856
Lucy McCoy0ca12682007-05-18 12:10:41 -0700857static void usa49wg_indat_callback(struct urb *urb)
858{
859 int i, len, x, err;
860 struct usb_serial *serial;
861 struct usb_serial_port *port;
862 struct tty_struct *tty;
863 unsigned char *data = urb->transfer_buffer;
Greg Kroah-Hartman95b93452007-06-15 15:44:13 -0700864 int status = urb->status;
Lucy McCoy0ca12682007-05-18 12:10:41 -0700865
Alan Coxdeb91682008-07-22 11:13:08 +0100866 dbg("%s", __func__);
Lucy McCoy0ca12682007-05-18 12:10:41 -0700867
868 serial = urb->context;
869
Greg Kroah-Hartman95b93452007-06-15 15:44:13 -0700870 if (status) {
Harvey Harrison441b62c2008-03-03 16:08:34 -0800871 dbg("%s - nonzero status: %x", __func__, status);
Lucy McCoy0ca12682007-05-18 12:10:41 -0700872 return;
873 }
874
875 /* inbound data is in the form P#, len, status, data */
876 i = 0;
877 len = 0;
878
879 if (urb->actual_length) {
880 while (i < urb->actual_length) {
881
882 /* Check port number from message*/
883 if (data[i] >= serial->num_ports) {
Alan Coxdeb91682008-07-22 11:13:08 +0100884 dbg("%s - Unexpected port number %d",
Harvey Harrison441b62c2008-03-03 16:08:34 -0800885 __func__, data[i]);
Lucy McCoy0ca12682007-05-18 12:10:41 -0700886 return;
887 }
888 port = serial->port[data[i++]];
Alan Cox4a90f092008-10-13 10:39:46 +0100889 tty = tty_port_tty_get(&port->port);
Lucy McCoy0ca12682007-05-18 12:10:41 -0700890 len = data[i++];
891
892 /* 0x80 bit is error flag */
893 if ((data[i] & 0x80) == 0) {
894 /* no error on any byte */
895 i++;
896 for (x = 1; x < len ; ++x)
Alan Stern1f871582010-02-17 10:05:47 -0500897 tty_insert_flip_char(tty, data[i++], 0);
Lucy McCoy0ca12682007-05-18 12:10:41 -0700898 } else {
899 /*
900 * some bytes had errors, every byte has status
901 */
902 for (x = 0; x + 1 < len; x += 2) {
903 int stat = data[i], flag = 0;
904 if (stat & RXERROR_OVERRUN)
905 flag |= TTY_OVERRUN;
906 if (stat & RXERROR_FRAMING)
907 flag |= TTY_FRAME;
908 if (stat & RXERROR_PARITY)
909 flag |= TTY_PARITY;
910 /* XXX should handle break (0x10) */
Alan Stern1f871582010-02-17 10:05:47 -0500911 tty_insert_flip_char(tty,
Lucy McCoy0ca12682007-05-18 12:10:41 -0700912 data[i+1], flag);
913 i += 2;
914 }
915 }
Alan Stern1f871582010-02-17 10:05:47 -0500916 tty_flip_buffer_push(tty);
Alan Cox4a90f092008-10-13 10:39:46 +0100917 tty_kref_put(tty);
Lucy McCoy0ca12682007-05-18 12:10:41 -0700918 }
919 }
920
921 /* Resubmit urb so we continue receiving */
922 urb->dev = serial->dev;
923
924 err = usb_submit_urb(urb, GFP_ATOMIC);
925 if (err != 0)
Harvey Harrison441b62c2008-03-03 16:08:34 -0800926 dbg("%s - resubmit read urb failed. (%d)", __func__, err);
Lucy McCoy0ca12682007-05-18 12:10:41 -0700927}
928
Linus Torvalds1da177e2005-04-16 15:20:36 -0700929/* not used, usa-49 doesn't have per-port control endpoints */
Lucy McCoy0ca12682007-05-18 12:10:41 -0700930static void usa49_outcont_callback(struct urb *urb)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700931{
Alan Coxdeb91682008-07-22 11:13:08 +0100932 dbg("%s", __func__);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700933}
934
Lucy McCoy0ca12682007-05-18 12:10:41 -0700935static void usa90_indat_callback(struct urb *urb)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700936{
937 int i, err;
938 int endpoint;
939 struct usb_serial_port *port;
940 struct keyspan_port_private *p_priv;
941 struct tty_struct *tty;
942 unsigned char *data = urb->transfer_buffer;
Greg Kroah-Hartman95b93452007-06-15 15:44:13 -0700943 int status = urb->status;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700944
Alan Coxdeb91682008-07-22 11:13:08 +0100945 dbg("%s", __func__);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700946
947 endpoint = usb_pipeendpoint(urb->pipe);
948
Greg Kroah-Hartman95b93452007-06-15 15:44:13 -0700949 if (status) {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700950 dbg("%s - nonzero status: %x on endpoint %d.",
Harvey Harrison441b62c2008-03-03 16:08:34 -0800951 __func__, status, endpoint);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700952 return;
953 }
954
Ming Leicdc97792008-02-24 18:41:47 +0800955 port = urb->context;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700956 p_priv = usb_get_serial_port_data(port);
957
Linus Torvalds1da177e2005-04-16 15:20:36 -0700958 if (urb->actual_length) {
Alan Cox4a90f092008-10-13 10:39:46 +0100959 tty = tty_port_tty_get(&port->port);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700960 /* if current mode is DMA, looks like usa28 format
Alan Coxdeb91682008-07-22 11:13:08 +0100961 otherwise looks like usa26 data format */
Linus Torvalds1da177e2005-04-16 15:20:36 -0700962
Alan Coxf035a8a2008-07-22 11:13:32 +0100963 if (p_priv->baud > 57600)
964 tty_insert_flip_string(tty, data, urb->actual_length);
965 else {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700966 /* 0x80 bit is error flag */
967 if ((data[0] & 0x80) == 0) {
Alan Coxdeb91682008-07-22 11:13:08 +0100968 /* no errors on individual bytes, only
969 possible overrun err*/
Linus Torvalds1da177e2005-04-16 15:20:36 -0700970 if (data[0] & RXERROR_OVERRUN)
Alan Coxdeb91682008-07-22 11:13:08 +0100971 err = TTY_OVERRUN;
972 else
973 err = 0;
974 for (i = 1; i < urb->actual_length ; ++i)
975 tty_insert_flip_char(tty, data[i],
976 err);
977 } else {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700978 /* some bytes had errors, every byte has status */
Harvey Harrison441b62c2008-03-03 16:08:34 -0800979 dbg("%s - RX error!!!!", __func__);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700980 for (i = 0; i + 1 < urb->actual_length; i += 2) {
981 int stat = data[i], flag = 0;
982 if (stat & RXERROR_OVERRUN)
983 flag |= TTY_OVERRUN;
984 if (stat & RXERROR_FRAMING)
985 flag |= TTY_FRAME;
986 if (stat & RXERROR_PARITY)
987 flag |= TTY_PARITY;
988 /* XXX should handle break (0x10) */
Alan Coxdeb91682008-07-22 11:13:08 +0100989 tty_insert_flip_char(tty, data[i+1],
990 flag);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700991 }
992 }
993 }
994 tty_flip_buffer_push(tty);
Alan Cox4a90f092008-10-13 10:39:46 +0100995 tty_kref_put(tty);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700996 }
Alan Coxdeb91682008-07-22 11:13:08 +0100997
Linus Torvalds1da177e2005-04-16 15:20:36 -0700998 /* Resubmit urb so we continue receiving */
999 urb->dev = port->serial->dev;
Alan Stern1f871582010-02-17 10:05:47 -05001000 err = usb_submit_urb(urb, GFP_ATOMIC);
1001 if (err != 0)
1002 dbg("%s - resubmit read urb failed. (%d)", __func__, err);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001003}
1004
1005
David Howells7d12e782006-10-05 14:55:46 +01001006static void usa90_instat_callback(struct urb *urb)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001007{
1008 unsigned char *data = urb->transfer_buffer;
1009 struct keyspan_usa90_portStatusMessage *msg;
1010 struct usb_serial *serial;
1011 struct usb_serial_port *port;
1012 struct keyspan_port_private *p_priv;
Alan Cox4a90f092008-10-13 10:39:46 +01001013 struct tty_struct *tty;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001014 int old_dcd_state, err;
Greg Kroah-Hartman95b93452007-06-15 15:44:13 -07001015 int status = urb->status;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001016
Ming Leicdc97792008-02-24 18:41:47 +08001017 serial = urb->context;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001018
Greg Kroah-Hartman95b93452007-06-15 15:44:13 -07001019 if (status) {
Harvey Harrison441b62c2008-03-03 16:08:34 -08001020 dbg("%s - nonzero status: %x", __func__, status);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001021 return;
1022 }
1023 if (urb->actual_length < 14) {
Harvey Harrison441b62c2008-03-03 16:08:34 -08001024 dbg("%s - %d byte report??", __func__, urb->actual_length);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001025 goto exit;
1026 }
1027
1028 msg = (struct keyspan_usa90_portStatusMessage *)data;
1029
1030 /* Now do something useful with the data */
1031
1032 port = serial->port[0];
1033 p_priv = usb_get_serial_port_data(port);
Alan Coxdeb91682008-07-22 11:13:08 +01001034
Linus Torvalds1da177e2005-04-16 15:20:36 -07001035 /* Update handshaking pin state information */
1036 old_dcd_state = p_priv->dcd_state;
1037 p_priv->cts_state = ((msg->cts) ? 1 : 0);
1038 p_priv->dsr_state = ((msg->dsr) ? 1 : 0);
1039 p_priv->dcd_state = ((msg->dcd) ? 1 : 0);
1040 p_priv->ri_state = ((msg->ri) ? 1 : 0);
1041
Alan Cox4a90f092008-10-13 10:39:46 +01001042 if (old_dcd_state != p_priv->dcd_state && old_dcd_state) {
1043 tty = tty_port_tty_get(&port->port);
1044 if (tty && !C_CLOCAL(tty))
1045 tty_hangup(tty);
1046 tty_kref_put(tty);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001047 }
Alan Coxdeb91682008-07-22 11:13:08 +01001048
Linus Torvalds1da177e2005-04-16 15:20:36 -07001049 /* Resubmit urb so we continue receiving */
1050 urb->dev = serial->dev;
Alan Coxdeb91682008-07-22 11:13:08 +01001051 err = usb_submit_urb(urb, GFP_ATOMIC);
1052 if (err != 0)
Harvey Harrison441b62c2008-03-03 16:08:34 -08001053 dbg("%s - resubmit read urb failed. (%d)", __func__, err);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001054exit:
1055 ;
1056}
1057
David Howells7d12e782006-10-05 14:55:46 +01001058static void usa90_outcont_callback(struct urb *urb)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001059{
1060 struct usb_serial_port *port;
1061 struct keyspan_port_private *p_priv;
1062
Ming Leicdc97792008-02-24 18:41:47 +08001063 port = urb->context;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001064 p_priv = usb_get_serial_port_data(port);
1065
1066 if (p_priv->resend_cont) {
Alan Coxdeb91682008-07-22 11:13:08 +01001067 dbg("%s - sending setup", __func__);
1068 keyspan_usa90_send_setup(port->serial, port,
1069 p_priv->resend_cont - 1);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001070 }
1071}
1072
Lucy McCoy0ca12682007-05-18 12:10:41 -07001073/* Status messages from the 28xg */
1074static void usa67_instat_callback(struct urb *urb)
1075{
1076 int err;
1077 unsigned char *data = urb->transfer_buffer;
1078 struct keyspan_usa67_portStatusMessage *msg;
1079 struct usb_serial *serial;
1080 struct usb_serial_port *port;
1081 struct keyspan_port_private *p_priv;
1082 int old_dcd_state;
Greg Kroah-Hartman95b93452007-06-15 15:44:13 -07001083 int status = urb->status;
Lucy McCoy0ca12682007-05-18 12:10:41 -07001084
Alan Coxdeb91682008-07-22 11:13:08 +01001085 dbg("%s", __func__);
Lucy McCoy0ca12682007-05-18 12:10:41 -07001086
1087 serial = urb->context;
1088
Greg Kroah-Hartman95b93452007-06-15 15:44:13 -07001089 if (status) {
Harvey Harrison441b62c2008-03-03 16:08:34 -08001090 dbg("%s - nonzero status: %x", __func__, status);
Lucy McCoy0ca12682007-05-18 12:10:41 -07001091 return;
1092 }
1093
Alan Coxdeb91682008-07-22 11:13:08 +01001094 if (urb->actual_length !=
1095 sizeof(struct keyspan_usa67_portStatusMessage)) {
Harvey Harrison441b62c2008-03-03 16:08:34 -08001096 dbg("%s - bad length %d", __func__, urb->actual_length);
Lucy McCoy0ca12682007-05-18 12:10:41 -07001097 return;
1098 }
1099
1100
1101 /* Now do something useful with the data */
1102 msg = (struct keyspan_usa67_portStatusMessage *)data;
1103
1104 /* Check port number from message and retrieve private data */
1105 if (msg->port >= serial->num_ports) {
Alan Coxdeb91682008-07-22 11:13:08 +01001106 dbg("%s - Unexpected port number %d", __func__, msg->port);
Lucy McCoy0ca12682007-05-18 12:10:41 -07001107 return;
1108 }
1109
1110 port = serial->port[msg->port];
1111 p_priv = usb_get_serial_port_data(port);
1112
1113 /* Update handshaking pin state information */
1114 old_dcd_state = p_priv->dcd_state;
1115 p_priv->cts_state = ((msg->hskia_cts) ? 1 : 0);
1116 p_priv->dcd_state = ((msg->gpia_dcd) ? 1 : 0);
1117
Alan Cox4a90f092008-10-13 10:39:46 +01001118 if (old_dcd_state != p_priv->dcd_state && old_dcd_state) {
1119 struct tty_struct *tty = tty_port_tty_get(&port->port);
1120 if (tty && !C_CLOCAL(tty))
1121 tty_hangup(tty);
1122 tty_kref_put(tty);
Lucy McCoy0ca12682007-05-18 12:10:41 -07001123 }
1124
1125 /* Resubmit urb so we continue receiving */
1126 urb->dev = serial->dev;
1127 err = usb_submit_urb(urb, GFP_ATOMIC);
1128 if (err != 0)
Harvey Harrison441b62c2008-03-03 16:08:34 -08001129 dbg("%s - resubmit read urb failed. (%d)", __func__, err);
Lucy McCoy0ca12682007-05-18 12:10:41 -07001130}
1131
1132static void usa67_glocont_callback(struct urb *urb)
1133{
1134 struct usb_serial *serial;
1135 struct usb_serial_port *port;
1136 struct keyspan_port_private *p_priv;
1137 int i;
1138
Alan Coxdeb91682008-07-22 11:13:08 +01001139 dbg("%s", __func__);
Lucy McCoy0ca12682007-05-18 12:10:41 -07001140
1141 serial = urb->context;
1142 for (i = 0; i < serial->num_ports; ++i) {
1143 port = serial->port[i];
1144 p_priv = usb_get_serial_port_data(port);
1145
1146 if (p_priv->resend_cont) {
Alan Coxdeb91682008-07-22 11:13:08 +01001147 dbg("%s - sending setup", __func__);
Lucy McCoy0ca12682007-05-18 12:10:41 -07001148 keyspan_usa67_send_setup(serial, port,
1149 p_priv->resend_cont - 1);
1150 break;
1151 }
1152 }
1153}
1154
Alan Cox95da3102008-07-22 11:09:07 +01001155static int keyspan_write_room(struct tty_struct *tty)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001156{
Alan Cox95da3102008-07-22 11:09:07 +01001157 struct usb_serial_port *port = tty->driver_data;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001158 struct keyspan_port_private *p_priv;
1159 const struct keyspan_device_details *d_details;
1160 int flip;
1161 int data_len;
1162 struct urb *this_urb;
1163
Harvey Harrison441b62c2008-03-03 16:08:34 -08001164 dbg("%s", __func__);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001165 p_priv = usb_get_serial_port_data(port);
1166 d_details = p_priv->device_details;
1167
Alan Coxa5b6f602008-04-08 17:16:06 +01001168 /* FIXME: locking */
Linus Torvalds1da177e2005-04-16 15:20:36 -07001169 if (d_details->msg_format == msg_usa90)
Alan Coxdeb91682008-07-22 11:13:08 +01001170 data_len = 64;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001171 else
1172 data_len = 63;
1173
1174 flip = p_priv->out_flip;
1175
1176 /* Check both endpoints to see if any are available. */
Alan Coxdeb91682008-07-22 11:13:08 +01001177 this_urb = p_priv->out_urbs[flip];
1178 if (this_urb != NULL) {
Linus Torvalds1da177e2005-04-16 15:20:36 -07001179 if (this_urb->status != -EINPROGRESS)
Alan Coxdeb91682008-07-22 11:13:08 +01001180 return data_len;
1181 flip = (flip + 1) & d_details->outdat_endp_flip;
1182 this_urb = p_priv->out_urbs[flip];
1183 if (this_urb != NULL) {
Linus Torvalds1da177e2005-04-16 15:20:36 -07001184 if (this_urb->status != -EINPROGRESS)
Alan Coxdeb91682008-07-22 11:13:08 +01001185 return data_len;
1186 }
Linus Torvalds1da177e2005-04-16 15:20:36 -07001187 }
Alan Coxa5b6f602008-04-08 17:16:06 +01001188 return 0;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001189}
1190
1191
Alan Coxa509a7e2009-09-19 13:13:26 -07001192static int keyspan_open(struct tty_struct *tty, struct usb_serial_port *port)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001193{
Andrew Mortonf78ba152007-11-28 16:21:54 -08001194 struct keyspan_port_private *p_priv;
1195 struct keyspan_serial_private *s_priv;
1196 struct usb_serial *serial = port->serial;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001197 const struct keyspan_device_details *d_details;
1198 int i, err;
Andrew Mortonf78ba152007-11-28 16:21:54 -08001199 int baud_rate, device_port;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001200 struct urb *urb;
Alan Cox95da3102008-07-22 11:09:07 +01001201 unsigned int cflag = 0;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001202
1203 s_priv = usb_get_serial_data(serial);
1204 p_priv = usb_get_serial_port_data(port);
1205 d_details = p_priv->device_details;
Borislav Petkov7eea4362007-11-14 17:00:39 -08001206
Harvey Harrison441b62c2008-03-03 16:08:34 -08001207 dbg("%s - port%d.", __func__, port->number);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001208
1209 /* Set some sane defaults */
1210 p_priv->rts_state = 1;
1211 p_priv->dtr_state = 1;
1212 p_priv->baud = 9600;
1213
1214 /* force baud and lcr to be set on open */
1215 p_priv->old_baud = 0;
1216 p_priv->old_cflag = 0;
1217
1218 p_priv->out_flip = 0;
1219 p_priv->in_flip = 0;
1220
1221 /* Reset low level data toggle and start reading from endpoints */
1222 for (i = 0; i < 2; i++) {
Alan Coxdeb91682008-07-22 11:13:08 +01001223 urb = p_priv->in_urbs[i];
1224 if (urb == NULL)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001225 continue;
1226 urb->dev = serial->dev;
1227
Alan Coxdeb91682008-07-22 11:13:08 +01001228 /* make sure endpoint data toggle is synchronized
1229 with the device */
Linus Torvalds1da177e2005-04-16 15:20:36 -07001230 usb_clear_halt(urb->dev, urb->pipe);
Alan Coxdeb91682008-07-22 11:13:08 +01001231 err = usb_submit_urb(urb, GFP_KERNEL);
1232 if (err != 0)
1233 dbg("%s - submit urb %d failed (%d)",
1234 __func__, i, err);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001235 }
1236
1237 /* Reset low level data toggle on out endpoints */
1238 for (i = 0; i < 2; i++) {
Alan Coxdeb91682008-07-22 11:13:08 +01001239 urb = p_priv->out_urbs[i];
1240 if (urb == NULL)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001241 continue;
1242 urb->dev = serial->dev;
Alan Coxdeb91682008-07-22 11:13:08 +01001243 /* usb_settoggle(urb->dev, usb_pipeendpoint(urb->pipe),
1244 usb_pipeout(urb->pipe), 0); */
Linus Torvalds1da177e2005-04-16 15:20:36 -07001245 }
1246
Andrew Mortonf78ba152007-11-28 16:21:54 -08001247 /* get the terminal config for the setup message now so we don't
1248 * need to send 2 of them */
1249
Andrew Mortonf78ba152007-11-28 16:21:54 -08001250 device_port = port->number - port->serial->minor;
Alan Cox95da3102008-07-22 11:09:07 +01001251 if (tty) {
1252 cflag = tty->termios->c_cflag;
1253 /* Baud rate calculation takes baud rate as an integer
1254 so other rates can be generated if desired. */
1255 baud_rate = tty_get_baud_rate(tty);
1256 /* If no match or invalid, leave as default */
1257 if (baud_rate >= 0
1258 && d_details->calculate_baud_rate(baud_rate, d_details->baudclk,
1259 NULL, NULL, NULL, device_port) == KEYSPAN_BAUD_RATE_OK) {
1260 p_priv->baud = baud_rate;
1261 }
Andrew Mortonf78ba152007-11-28 16:21:54 -08001262 }
Andrew Mortonf78ba152007-11-28 16:21:54 -08001263 /* set CTS/RTS handshake etc. */
1264 p_priv->cflag = cflag;
1265 p_priv->flow_control = (cflag & CRTSCTS)? flow_cts: flow_none;
1266
1267 keyspan_send_setup(port, 1);
Alan Coxdeb91682008-07-22 11:13:08 +01001268 /* mdelay(100); */
1269 /* keyspan_set_termios(port, NULL); */
Andrew Mortonf78ba152007-11-28 16:21:54 -08001270
Alan Coxa5b6f602008-04-08 17:16:06 +01001271 return 0;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001272}
1273
1274static inline void stop_urb(struct urb *urb)
1275{
Greg Kroah-Hartman242cf672005-07-29 16:11:07 -04001276 if (urb && urb->status == -EINPROGRESS)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001277 usb_kill_urb(urb);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001278}
1279
Alan Cox335f8512009-06-11 12:26:29 +01001280static void keyspan_dtr_rts(struct usb_serial_port *port, int on)
1281{
1282 struct keyspan_port_private *p_priv = usb_get_serial_port_data(port);
1283
1284 p_priv->rts_state = on;
1285 p_priv->dtr_state = on;
1286 keyspan_send_setup(port, 0);
1287}
1288
1289static void keyspan_close(struct usb_serial_port *port)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001290{
1291 int i;
1292 struct usb_serial *serial = port->serial;
1293 struct keyspan_serial_private *s_priv;
1294 struct keyspan_port_private *p_priv;
1295
Harvey Harrison441b62c2008-03-03 16:08:34 -08001296 dbg("%s", __func__);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001297 s_priv = usb_get_serial_data(serial);
1298 p_priv = usb_get_serial_port_data(port);
Alan Coxdeb91682008-07-22 11:13:08 +01001299
Linus Torvalds1da177e2005-04-16 15:20:36 -07001300 p_priv->rts_state = 0;
1301 p_priv->dtr_state = 0;
Alan Coxdeb91682008-07-22 11:13:08 +01001302
Linus Torvalds1da177e2005-04-16 15:20:36 -07001303 if (serial->dev) {
1304 keyspan_send_setup(port, 2);
1305 /* pilot-xfer seems to work best with this delay */
1306 mdelay(100);
Alan Coxdeb91682008-07-22 11:13:08 +01001307 /* keyspan_set_termios(port, NULL); */
Linus Torvalds1da177e2005-04-16 15:20:36 -07001308 }
1309
1310 /*while (p_priv->outcont_urb->status == -EINPROGRESS) {
Harvey Harrison441b62c2008-03-03 16:08:34 -08001311 dbg("%s - urb in progress", __func__);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001312 }*/
1313
1314 p_priv->out_flip = 0;
1315 p_priv->in_flip = 0;
1316
1317 if (serial->dev) {
1318 /* Stop reading/writing urbs */
1319 stop_urb(p_priv->inack_urb);
1320 /* stop_urb(p_priv->outcont_urb); */
1321 for (i = 0; i < 2; i++) {
1322 stop_urb(p_priv->in_urbs[i]);
1323 stop_urb(p_priv->out_urbs[i]);
1324 }
1325 }
Linus Torvalds1da177e2005-04-16 15:20:36 -07001326}
1327
Alan Coxdeb91682008-07-22 11:13:08 +01001328/* download the firmware to a pre-renumeration device */
1329static int keyspan_fake_startup(struct usb_serial *serial)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001330{
1331 int response;
David Woodhouse2971c572008-05-30 14:04:03 +03001332 const struct ihex_binrec *record;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001333 char *fw_name;
David Woodhouse2971c572008-05-30 14:04:03 +03001334 const struct firmware *fw;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001335
1336 dbg("Keyspan startup version %04x product %04x",
1337 le16_to_cpu(serial->dev->descriptor.bcdDevice),
1338 le16_to_cpu(serial->dev->descriptor.idProduct));
Alan Coxdeb91682008-07-22 11:13:08 +01001339
1340 if ((le16_to_cpu(serial->dev->descriptor.bcdDevice) & 0x8000)
1341 != 0x8000) {
Linus Torvalds1da177e2005-04-16 15:20:36 -07001342 dbg("Firmware already loaded. Quitting.");
Alan Coxdeb91682008-07-22 11:13:08 +01001343 return 1;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001344 }
1345
1346 /* Select firmware image on the basis of idProduct */
1347 switch (le16_to_cpu(serial->dev->descriptor.idProduct)) {
1348 case keyspan_usa28_pre_product_id:
David Woodhouse2971c572008-05-30 14:04:03 +03001349 fw_name = "keyspan/usa28.fw";
Linus Torvalds1da177e2005-04-16 15:20:36 -07001350 break;
1351
1352 case keyspan_usa28x_pre_product_id:
David Woodhouse2971c572008-05-30 14:04:03 +03001353 fw_name = "keyspan/usa28x.fw";
Linus Torvalds1da177e2005-04-16 15:20:36 -07001354 break;
1355
1356 case keyspan_usa28xa_pre_product_id:
David Woodhouse2971c572008-05-30 14:04:03 +03001357 fw_name = "keyspan/usa28xa.fw";
Linus Torvalds1da177e2005-04-16 15:20:36 -07001358 break;
1359
1360 case keyspan_usa28xb_pre_product_id:
David Woodhouse2971c572008-05-30 14:04:03 +03001361 fw_name = "keyspan/usa28xb.fw";
Linus Torvalds1da177e2005-04-16 15:20:36 -07001362 break;
1363
1364 case keyspan_usa19_pre_product_id:
David Woodhouse2971c572008-05-30 14:04:03 +03001365 fw_name = "keyspan/usa19.fw";
Linus Torvalds1da177e2005-04-16 15:20:36 -07001366 break;
Alan Coxdeb91682008-07-22 11:13:08 +01001367
Linus Torvalds1da177e2005-04-16 15:20:36 -07001368 case keyspan_usa19qi_pre_product_id:
David Woodhouse2971c572008-05-30 14:04:03 +03001369 fw_name = "keyspan/usa19qi.fw";
Linus Torvalds1da177e2005-04-16 15:20:36 -07001370 break;
Alan Coxdeb91682008-07-22 11:13:08 +01001371
Linus Torvalds1da177e2005-04-16 15:20:36 -07001372 case keyspan_mpr_pre_product_id:
David Woodhouse2971c572008-05-30 14:04:03 +03001373 fw_name = "keyspan/mpr.fw";
Linus Torvalds1da177e2005-04-16 15:20:36 -07001374 break;
1375
1376 case keyspan_usa19qw_pre_product_id:
David Woodhouse2971c572008-05-30 14:04:03 +03001377 fw_name = "keyspan/usa19qw.fw";
Linus Torvalds1da177e2005-04-16 15:20:36 -07001378 break;
Alan Coxdeb91682008-07-22 11:13:08 +01001379
Linus Torvalds1da177e2005-04-16 15:20:36 -07001380 case keyspan_usa18x_pre_product_id:
David Woodhouse2971c572008-05-30 14:04:03 +03001381 fw_name = "keyspan/usa18x.fw";
Linus Torvalds1da177e2005-04-16 15:20:36 -07001382 break;
Alan Coxdeb91682008-07-22 11:13:08 +01001383
Linus Torvalds1da177e2005-04-16 15:20:36 -07001384 case keyspan_usa19w_pre_product_id:
David Woodhouse2971c572008-05-30 14:04:03 +03001385 fw_name = "keyspan/usa19w.fw";
Linus Torvalds1da177e2005-04-16 15:20:36 -07001386 break;
Alan Coxdeb91682008-07-22 11:13:08 +01001387
Linus Torvalds1da177e2005-04-16 15:20:36 -07001388 case keyspan_usa49w_pre_product_id:
David Woodhouse2971c572008-05-30 14:04:03 +03001389 fw_name = "keyspan/usa49w.fw";
Linus Torvalds1da177e2005-04-16 15:20:36 -07001390 break;
1391
1392 case keyspan_usa49wlc_pre_product_id:
David Woodhouse2971c572008-05-30 14:04:03 +03001393 fw_name = "keyspan/usa49wlc.fw";
Linus Torvalds1da177e2005-04-16 15:20:36 -07001394 break;
1395
1396 default:
David Woodhouse2971c572008-05-30 14:04:03 +03001397 dev_err(&serial->dev->dev, "Unknown product ID (%04x)\n",
1398 le16_to_cpu(serial->dev->descriptor.idProduct));
1399 return 1;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001400 }
1401
David Woodhouse2971c572008-05-30 14:04:03 +03001402 if (request_ihex_firmware(&fw, fw_name, &serial->dev->dev)) {
Linus Torvalds1da177e2005-04-16 15:20:36 -07001403 dev_err(&serial->dev->dev, "Required keyspan firmware image (%s) unavailable.\n", fw_name);
1404 return(1);
1405 }
1406
1407 dbg("Uploading Keyspan %s firmware.", fw_name);
1408
1409 /* download the firmware image */
1410 response = ezusb_set_reset(serial, 1);
1411
David Woodhouse2971c572008-05-30 14:04:03 +03001412 record = (const struct ihex_binrec *)fw->data;
1413
1414 while (record) {
1415 response = ezusb_writememory(serial, be32_to_cpu(record->addr),
Linus Torvalds1da177e2005-04-16 15:20:36 -07001416 (unsigned char *)record->data,
David Woodhouse2971c572008-05-30 14:04:03 +03001417 be16_to_cpu(record->len), 0xa0);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001418 if (response < 0) {
Alan Coxdeb91682008-07-22 11:13:08 +01001419 dev_err(&serial->dev->dev, "ezusb_writememory failed for Keyspan firmware (%d %04X %p %d)\n",
David Woodhouse2971c572008-05-30 14:04:03 +03001420 response, be32_to_cpu(record->addr),
1421 record->data, be16_to_cpu(record->len));
Linus Torvalds1da177e2005-04-16 15:20:36 -07001422 break;
1423 }
David Woodhouse2971c572008-05-30 14:04:03 +03001424 record = ihex_next_binrec(record);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001425 }
David Woodhouse2971c572008-05-30 14:04:03 +03001426 release_firmware(fw);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001427 /* bring device out of reset. Renumeration will occur in a
1428 moment and the new device will bind to the real driver */
1429 response = ezusb_set_reset(serial, 0);
1430
1431 /* we don't want this device to have a driver assigned to it. */
Alan Coxdeb91682008-07-22 11:13:08 +01001432 return 1;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001433}
1434
1435/* Helper functions used by keyspan_setup_urbs */
Rainer Weikusatfdcba532007-01-03 15:36:25 +01001436static struct usb_endpoint_descriptor const *find_ep(struct usb_serial const *serial,
1437 int endpoint)
1438{
1439 struct usb_host_interface *iface_desc;
1440 struct usb_endpoint_descriptor *ep;
1441 int i;
1442
1443 iface_desc = serial->interface->cur_altsetting;
1444 for (i = 0; i < iface_desc->desc.bNumEndpoints; ++i) {
1445 ep = &iface_desc->endpoint[i].desc;
1446 if (ep->bEndpointAddress == endpoint)
1447 return ep;
1448 }
1449 dev_warn(&serial->interface->dev, "found no endpoint descriptor for "
1450 "endpoint %x\n", endpoint);
1451 return NULL;
1452}
1453
Alan Coxdeb91682008-07-22 11:13:08 +01001454static struct urb *keyspan_setup_urb(struct usb_serial *serial, int endpoint,
Linus Torvalds1da177e2005-04-16 15:20:36 -07001455 int dir, void *ctx, char *buf, int len,
David Howells7d12e782006-10-05 14:55:46 +01001456 void (*callback)(struct urb *))
Linus Torvalds1da177e2005-04-16 15:20:36 -07001457{
1458 struct urb *urb;
Rainer Weikusatfdcba532007-01-03 15:36:25 +01001459 struct usb_endpoint_descriptor const *ep_desc;
1460 char const *ep_type_name;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001461
1462 if (endpoint == -1)
1463 return NULL; /* endpoint not needed */
1464
Alan Coxdeb91682008-07-22 11:13:08 +01001465 dbg("%s - alloc for endpoint %d.", __func__, endpoint);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001466 urb = usb_alloc_urb(0, GFP_KERNEL); /* No ISO */
1467 if (urb == NULL) {
Alan Coxdeb91682008-07-22 11:13:08 +01001468 dbg("%s - alloc for endpoint %d failed.", __func__, endpoint);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001469 return NULL;
1470 }
1471
Lucy McCoy0ca12682007-05-18 12:10:41 -07001472 if (endpoint == 0) {
1473 /* control EP filled in when used */
1474 return urb;
1475 }
1476
Rainer Weikusatfdcba532007-01-03 15:36:25 +01001477 ep_desc = find_ep(serial, endpoint);
1478 if (!ep_desc) {
1479 /* leak the urb, something's wrong and the callers don't care */
1480 return urb;
1481 }
1482 if (usb_endpoint_xfer_int(ep_desc)) {
1483 ep_type_name = "INT";
1484 usb_fill_int_urb(urb, serial->dev,
1485 usb_sndintpipe(serial->dev, endpoint) | dir,
1486 buf, len, callback, ctx,
1487 ep_desc->bInterval);
1488 } else if (usb_endpoint_xfer_bulk(ep_desc)) {
1489 ep_type_name = "BULK";
1490 usb_fill_bulk_urb(urb, serial->dev,
1491 usb_sndbulkpipe(serial->dev, endpoint) | dir,
1492 buf, len, callback, ctx);
1493 } else {
1494 dev_warn(&serial->interface->dev,
1495 "unsupported endpoint type %x\n",
Julia Lawall2e0fe702008-12-29 11:22:14 +01001496 usb_endpoint_type(ep_desc));
Rainer Weikusatfdcba532007-01-03 15:36:25 +01001497 usb_free_urb(urb);
1498 return NULL;
1499 }
Linus Torvalds1da177e2005-04-16 15:20:36 -07001500
Rainer Weikusatfdcba532007-01-03 15:36:25 +01001501 dbg("%s - using urb %p for %s endpoint %x",
1502 __func__, urb, ep_type_name, endpoint);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001503 return urb;
1504}
1505
1506static struct callbacks {
David Howells7d12e782006-10-05 14:55:46 +01001507 void (*instat_callback)(struct urb *);
1508 void (*glocont_callback)(struct urb *);
1509 void (*indat_callback)(struct urb *);
1510 void (*outdat_callback)(struct urb *);
1511 void (*inack_callback)(struct urb *);
1512 void (*outcont_callback)(struct urb *);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001513} keyspan_callbacks[] = {
1514 {
1515 /* msg_usa26 callbacks */
1516 .instat_callback = usa26_instat_callback,
1517 .glocont_callback = usa26_glocont_callback,
1518 .indat_callback = usa26_indat_callback,
1519 .outdat_callback = usa2x_outdat_callback,
1520 .inack_callback = usa26_inack_callback,
1521 .outcont_callback = usa26_outcont_callback,
1522 }, {
1523 /* msg_usa28 callbacks */
1524 .instat_callback = usa28_instat_callback,
1525 .glocont_callback = usa28_glocont_callback,
1526 .indat_callback = usa28_indat_callback,
1527 .outdat_callback = usa2x_outdat_callback,
1528 .inack_callback = usa28_inack_callback,
1529 .outcont_callback = usa28_outcont_callback,
1530 }, {
1531 /* msg_usa49 callbacks */
1532 .instat_callback = usa49_instat_callback,
1533 .glocont_callback = usa49_glocont_callback,
1534 .indat_callback = usa49_indat_callback,
1535 .outdat_callback = usa2x_outdat_callback,
1536 .inack_callback = usa49_inack_callback,
1537 .outcont_callback = usa49_outcont_callback,
1538 }, {
1539 /* msg_usa90 callbacks */
1540 .instat_callback = usa90_instat_callback,
Alan Coxdeb91682008-07-22 11:13:08 +01001541 .glocont_callback = usa28_glocont_callback,
Linus Torvalds1da177e2005-04-16 15:20:36 -07001542 .indat_callback = usa90_indat_callback,
1543 .outdat_callback = usa2x_outdat_callback,
1544 .inack_callback = usa28_inack_callback,
1545 .outcont_callback = usa90_outcont_callback,
Lucy McCoy0ca12682007-05-18 12:10:41 -07001546 }, {
1547 /* msg_usa67 callbacks */
1548 .instat_callback = usa67_instat_callback,
1549 .glocont_callback = usa67_glocont_callback,
1550 .indat_callback = usa26_indat_callback,
1551 .outdat_callback = usa2x_outdat_callback,
1552 .inack_callback = usa26_inack_callback,
1553 .outcont_callback = usa26_outcont_callback,
Linus Torvalds1da177e2005-04-16 15:20:36 -07001554 }
1555};
1556
1557 /* Generic setup urbs function that uses
1558 data in device_details */
1559static void keyspan_setup_urbs(struct usb_serial *serial)
1560{
1561 int i, j;
1562 struct keyspan_serial_private *s_priv;
1563 const struct keyspan_device_details *d_details;
1564 struct usb_serial_port *port;
1565 struct keyspan_port_private *p_priv;
1566 struct callbacks *cback;
1567 int endp;
1568
Alan Coxdeb91682008-07-22 11:13:08 +01001569 dbg("%s", __func__);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001570
1571 s_priv = usb_get_serial_data(serial);
1572 d_details = s_priv->device_details;
1573
Alan Coxdeb91682008-07-22 11:13:08 +01001574 /* Setup values for the various callback routines */
Linus Torvalds1da177e2005-04-16 15:20:36 -07001575 cback = &keyspan_callbacks[d_details->msg_format];
1576
Alan Coxdeb91682008-07-22 11:13:08 +01001577 /* Allocate and set up urbs for each one that is in use,
1578 starting with instat endpoints */
Linus Torvalds1da177e2005-04-16 15:20:36 -07001579 s_priv->instat_urb = keyspan_setup_urb
1580 (serial, d_details->instat_endpoint, USB_DIR_IN,
1581 serial, s_priv->instat_buf, INSTAT_BUFLEN,
1582 cback->instat_callback);
1583
Lucy McCoy0ca12682007-05-18 12:10:41 -07001584 s_priv->indat_urb = keyspan_setup_urb
1585 (serial, d_details->indat_endpoint, USB_DIR_IN,
1586 serial, s_priv->indat_buf, INDAT49W_BUFLEN,
1587 usa49wg_indat_callback);
1588
Linus Torvalds1da177e2005-04-16 15:20:36 -07001589 s_priv->glocont_urb = keyspan_setup_urb
1590 (serial, d_details->glocont_endpoint, USB_DIR_OUT,
1591 serial, s_priv->glocont_buf, GLOCONT_BUFLEN,
1592 cback->glocont_callback);
1593
Alan Coxdeb91682008-07-22 11:13:08 +01001594 /* Setup endpoints for each port specific thing */
1595 for (i = 0; i < d_details->num_ports; i++) {
Linus Torvalds1da177e2005-04-16 15:20:36 -07001596 port = serial->port[i];
1597 p_priv = usb_get_serial_port_data(port);
1598
1599 /* Do indat endpoints first, once for each flip */
1600 endp = d_details->indat_endpoints[i];
1601 for (j = 0; j <= d_details->indat_endp_flip; ++j, ++endp) {
1602 p_priv->in_urbs[j] = keyspan_setup_urb
1603 (serial, endp, USB_DIR_IN, port,
1604 p_priv->in_buffer[j], 64,
1605 cback->indat_callback);
1606 }
1607 for (; j < 2; ++j)
1608 p_priv->in_urbs[j] = NULL;
1609
1610 /* outdat endpoints also have flip */
1611 endp = d_details->outdat_endpoints[i];
1612 for (j = 0; j <= d_details->outdat_endp_flip; ++j, ++endp) {
1613 p_priv->out_urbs[j] = keyspan_setup_urb
1614 (serial, endp, USB_DIR_OUT, port,
1615 p_priv->out_buffer[j], 64,
1616 cback->outdat_callback);
1617 }
1618 for (; j < 2; ++j)
1619 p_priv->out_urbs[j] = NULL;
1620
1621 /* inack endpoint */
1622 p_priv->inack_urb = keyspan_setup_urb
1623 (serial, d_details->inack_endpoints[i], USB_DIR_IN,
1624 port, p_priv->inack_buffer, 1, cback->inack_callback);
1625
1626 /* outcont endpoint */
1627 p_priv->outcont_urb = keyspan_setup_urb
1628 (serial, d_details->outcont_endpoints[i], USB_DIR_OUT,
1629 port, p_priv->outcont_buffer, 64,
1630 cback->outcont_callback);
Alan Coxdeb91682008-07-22 11:13:08 +01001631 }
Linus Torvalds1da177e2005-04-16 15:20:36 -07001632}
1633
1634/* usa19 function doesn't require prescaler */
1635static int keyspan_usa19_calc_baud(u32 baud_rate, u32 baudclk, u8 *rate_hi,
1636 u8 *rate_low, u8 *prescaler, int portnum)
1637{
1638 u32 b16, /* baud rate times 16 (actual rate used internally) */
Alan Coxdeb91682008-07-22 11:13:08 +01001639 div, /* divisor */
Linus Torvalds1da177e2005-04-16 15:20:36 -07001640 cnt; /* inverse of divisor (programmed into 8051) */
Linus Torvalds1da177e2005-04-16 15:20:36 -07001641
Alan Coxdeb91682008-07-22 11:13:08 +01001642 dbg("%s - %d.", __func__, baud_rate);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001643
Alan Coxdeb91682008-07-22 11:13:08 +01001644 /* prevent divide by zero... */
1645 b16 = baud_rate * 16L;
1646 if (b16 == 0)
1647 return KEYSPAN_INVALID_BAUD_RATE;
1648 /* Any "standard" rate over 57k6 is marginal on the USA-19
1649 as we run out of divisor resolution. */
1650 if (baud_rate > 57600)
1651 return KEYSPAN_INVALID_BAUD_RATE;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001652
Alan Coxdeb91682008-07-22 11:13:08 +01001653 /* calculate the divisor and the counter (its inverse) */
1654 div = baudclk / b16;
1655 if (div == 0)
1656 return KEYSPAN_INVALID_BAUD_RATE;
1657 else
Linus Torvalds1da177e2005-04-16 15:20:36 -07001658 cnt = 0 - div;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001659
Alan Coxdeb91682008-07-22 11:13:08 +01001660 if (div > 0xffff)
1661 return KEYSPAN_INVALID_BAUD_RATE;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001662
Alan Coxdeb91682008-07-22 11:13:08 +01001663 /* return the counter values if non-null */
1664 if (rate_low)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001665 *rate_low = (u8) (cnt & 0xff);
Alan Coxdeb91682008-07-22 11:13:08 +01001666 if (rate_hi)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001667 *rate_hi = (u8) ((cnt >> 8) & 0xff);
Alan Coxdeb91682008-07-22 11:13:08 +01001668 if (rate_low && rate_hi)
1669 dbg("%s - %d %02x %02x.",
1670 __func__, baud_rate, *rate_hi, *rate_low);
1671 return KEYSPAN_BAUD_RATE_OK;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001672}
1673
1674/* usa19hs function doesn't require prescaler */
1675static int keyspan_usa19hs_calc_baud(u32 baud_rate, u32 baudclk, u8 *rate_hi,
1676 u8 *rate_low, u8 *prescaler, int portnum)
1677{
1678 u32 b16, /* baud rate times 16 (actual rate used internally) */
Alan Coxdeb91682008-07-22 11:13:08 +01001679 div; /* divisor */
Linus Torvalds1da177e2005-04-16 15:20:36 -07001680
Alan Coxdeb91682008-07-22 11:13:08 +01001681 dbg("%s - %d.", __func__, baud_rate);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001682
Alan Coxdeb91682008-07-22 11:13:08 +01001683 /* prevent divide by zero... */
1684 b16 = baud_rate * 16L;
1685 if (b16 == 0)
1686 return KEYSPAN_INVALID_BAUD_RATE;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001687
Alan Coxdeb91682008-07-22 11:13:08 +01001688 /* calculate the divisor */
1689 div = baudclk / b16;
1690 if (div == 0)
1691 return KEYSPAN_INVALID_BAUD_RATE;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001692
Alan Coxdeb91682008-07-22 11:13:08 +01001693 if (div > 0xffff)
1694 return KEYSPAN_INVALID_BAUD_RATE;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001695
Alan Coxdeb91682008-07-22 11:13:08 +01001696 /* return the counter values if non-null */
1697 if (rate_low)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001698 *rate_low = (u8) (div & 0xff);
Alan Coxdeb91682008-07-22 11:13:08 +01001699
1700 if (rate_hi)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001701 *rate_hi = (u8) ((div >> 8) & 0xff);
Alan Coxdeb91682008-07-22 11:13:08 +01001702
1703 if (rate_low && rate_hi)
1704 dbg("%s - %d %02x %02x.",
1705 __func__, baud_rate, *rate_hi, *rate_low);
1706
1707 return KEYSPAN_BAUD_RATE_OK;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001708}
1709
1710static int keyspan_usa19w_calc_baud(u32 baud_rate, u32 baudclk, u8 *rate_hi,
1711 u8 *rate_low, u8 *prescaler, int portnum)
1712{
1713 u32 b16, /* baud rate times 16 (actual rate used internally) */
1714 clk, /* clock with 13/8 prescaler */
Alan Coxdeb91682008-07-22 11:13:08 +01001715 div, /* divisor using 13/8 prescaler */
Linus Torvalds1da177e2005-04-16 15:20:36 -07001716 res, /* resulting baud rate using 13/8 prescaler */
1717 diff, /* error using 13/8 prescaler */
1718 smallest_diff;
1719 u8 best_prescaler;
1720 int i;
1721
Alan Coxdeb91682008-07-22 11:13:08 +01001722 dbg("%s - %d.", __func__, baud_rate);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001723
Alan Coxdeb91682008-07-22 11:13:08 +01001724 /* prevent divide by zero */
1725 b16 = baud_rate * 16L;
1726 if (b16 == 0)
1727 return KEYSPAN_INVALID_BAUD_RATE;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001728
Alan Coxdeb91682008-07-22 11:13:08 +01001729 /* Calculate prescaler by trying them all and looking
1730 for best fit */
1731
1732 /* start with largest possible difference */
Linus Torvalds1da177e2005-04-16 15:20:36 -07001733 smallest_diff = 0xffffffff;
1734
1735 /* 0 is an invalid prescaler, used as a flag */
1736 best_prescaler = 0;
1737
Alan Coxdeb91682008-07-22 11:13:08 +01001738 for (i = 8; i <= 0xff; ++i) {
Linus Torvalds1da177e2005-04-16 15:20:36 -07001739 clk = (baudclk * 8) / (u32) i;
Alan Coxdeb91682008-07-22 11:13:08 +01001740
1741 div = clk / b16;
1742 if (div == 0)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001743 continue;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001744
1745 res = clk / div;
Alan Coxdeb91682008-07-22 11:13:08 +01001746 diff = (res > b16) ? (res-b16) : (b16-res);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001747
Alan Coxdeb91682008-07-22 11:13:08 +01001748 if (diff < smallest_diff) {
Linus Torvalds1da177e2005-04-16 15:20:36 -07001749 best_prescaler = i;
1750 smallest_diff = diff;
1751 }
1752 }
1753
Alan Coxdeb91682008-07-22 11:13:08 +01001754 if (best_prescaler == 0)
1755 return KEYSPAN_INVALID_BAUD_RATE;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001756
1757 clk = (baudclk * 8) / (u32) best_prescaler;
1758 div = clk / b16;
1759
Alan Coxdeb91682008-07-22 11:13:08 +01001760 /* return the divisor and prescaler if non-null */
1761 if (rate_low)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001762 *rate_low = (u8) (div & 0xff);
Alan Coxdeb91682008-07-22 11:13:08 +01001763 if (rate_hi)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001764 *rate_hi = (u8) ((div >> 8) & 0xff);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001765 if (prescaler) {
1766 *prescaler = best_prescaler;
Harvey Harrison441b62c2008-03-03 16:08:34 -08001767 /* dbg("%s - %d %d", __func__, *prescaler, div); */
Linus Torvalds1da177e2005-04-16 15:20:36 -07001768 }
Alan Coxdeb91682008-07-22 11:13:08 +01001769 return KEYSPAN_BAUD_RATE_OK;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001770}
1771
1772 /* USA-28 supports different maximum baud rates on each port */
1773static int keyspan_usa28_calc_baud(u32 baud_rate, u32 baudclk, u8 *rate_hi,
1774 u8 *rate_low, u8 *prescaler, int portnum)
1775{
1776 u32 b16, /* baud rate times 16 (actual rate used internally) */
Alan Coxdeb91682008-07-22 11:13:08 +01001777 div, /* divisor */
Linus Torvalds1da177e2005-04-16 15:20:36 -07001778 cnt; /* inverse of divisor (programmed into 8051) */
1779
Alan Coxdeb91682008-07-22 11:13:08 +01001780 dbg("%s - %d.", __func__, baud_rate);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001781
1782 /* prevent divide by zero */
Alan Coxdeb91682008-07-22 11:13:08 +01001783 b16 = baud_rate * 16L;
1784 if (b16 == 0)
1785 return KEYSPAN_INVALID_BAUD_RATE;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001786
Alan Coxdeb91682008-07-22 11:13:08 +01001787 /* calculate the divisor and the counter (its inverse) */
1788 div = KEYSPAN_USA28_BAUDCLK / b16;
1789 if (div == 0)
1790 return KEYSPAN_INVALID_BAUD_RATE;
1791 else
Linus Torvalds1da177e2005-04-16 15:20:36 -07001792 cnt = 0 - div;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001793
Alan Coxdeb91682008-07-22 11:13:08 +01001794 /* check for out of range, based on portnum,
1795 and return result */
1796 if (portnum == 0) {
1797 if (div > 0xffff)
1798 return KEYSPAN_INVALID_BAUD_RATE;
1799 } else {
1800 if (portnum == 1) {
1801 if (div > 0xff)
1802 return KEYSPAN_INVALID_BAUD_RATE;
1803 } else
1804 return KEYSPAN_INVALID_BAUD_RATE;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001805 }
1806
1807 /* return the counter values if not NULL
1808 (port 1 will ignore retHi) */
Alan Coxdeb91682008-07-22 11:13:08 +01001809 if (rate_low)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001810 *rate_low = (u8) (cnt & 0xff);
Alan Coxdeb91682008-07-22 11:13:08 +01001811 if (rate_hi)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001812 *rate_hi = (u8) ((cnt >> 8) & 0xff);
Alan Coxdeb91682008-07-22 11:13:08 +01001813 dbg("%s - %d OK.", __func__, baud_rate);
1814 return KEYSPAN_BAUD_RATE_OK;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001815}
1816
1817static int keyspan_usa26_send_setup(struct usb_serial *serial,
1818 struct usb_serial_port *port,
1819 int reset_port)
1820{
Alan Coxdeb91682008-07-22 11:13:08 +01001821 struct keyspan_usa26_portControlMessage msg;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001822 struct keyspan_serial_private *s_priv;
1823 struct keyspan_port_private *p_priv;
1824 const struct keyspan_device_details *d_details;
1825 int outcont_urb;
1826 struct urb *this_urb;
1827 int device_port, err;
1828
Alan Coxdeb91682008-07-22 11:13:08 +01001829 dbg("%s reset=%d", __func__, reset_port);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001830
1831 s_priv = usb_get_serial_data(serial);
1832 p_priv = usb_get_serial_port_data(port);
1833 d_details = s_priv->device_details;
1834 device_port = port->number - port->serial->minor;
1835
1836 outcont_urb = d_details->outcont_endpoints[port->number];
1837 this_urb = p_priv->outcont_urb;
1838
Harvey Harrison441b62c2008-03-03 16:08:34 -08001839 dbg("%s - endpoint %d", __func__, usb_pipeendpoint(this_urb->pipe));
Linus Torvalds1da177e2005-04-16 15:20:36 -07001840
1841 /* Make sure we have an urb then send the message */
1842 if (this_urb == NULL) {
Harvey Harrison441b62c2008-03-03 16:08:34 -08001843 dbg("%s - oops no urb.", __func__);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001844 return -1;
1845 }
1846
1847 /* Save reset port val for resend.
Lucy McCoy0ca12682007-05-18 12:10:41 -07001848 Don't overwrite resend for open/close condition. */
1849 if ((reset_port + 1) > p_priv->resend_cont)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001850 p_priv->resend_cont = reset_port + 1;
1851 if (this_urb->status == -EINPROGRESS) {
Alan Coxdeb91682008-07-22 11:13:08 +01001852 /* dbg("%s - already writing", __func__); */
Linus Torvalds1da177e2005-04-16 15:20:36 -07001853 mdelay(5);
Alan Coxdeb91682008-07-22 11:13:08 +01001854 return -1;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001855 }
1856
Alan Coxdeb91682008-07-22 11:13:08 +01001857 memset(&msg, 0, sizeof(struct keyspan_usa26_portControlMessage));
1858
1859 /* Only set baud rate if it's changed */
Linus Torvalds1da177e2005-04-16 15:20:36 -07001860 if (p_priv->old_baud != p_priv->baud) {
1861 p_priv->old_baud = p_priv->baud;
1862 msg.setClocking = 0xff;
1863 if (d_details->calculate_baud_rate
1864 (p_priv->baud, d_details->baudclk, &msg.baudHi,
Alan Coxdeb91682008-07-22 11:13:08 +01001865 &msg.baudLo, &msg.prescaler, device_port) == KEYSPAN_INVALID_BAUD_RATE) {
1866 dbg("%s - Invalid baud rate %d requested, using 9600.",
1867 __func__, p_priv->baud);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001868 msg.baudLo = 0;
1869 msg.baudHi = 125; /* Values for 9600 baud */
1870 msg.prescaler = 10;
1871 }
1872 msg.setPrescaler = 0xff;
1873 }
1874
1875 msg.lcr = (p_priv->cflag & CSTOPB)? STOPBITS_678_2: STOPBITS_5678_1;
1876 switch (p_priv->cflag & CSIZE) {
1877 case CS5:
1878 msg.lcr |= USA_DATABITS_5;
1879 break;
1880 case CS6:
1881 msg.lcr |= USA_DATABITS_6;
1882 break;
1883 case CS7:
1884 msg.lcr |= USA_DATABITS_7;
1885 break;
1886 case CS8:
1887 msg.lcr |= USA_DATABITS_8;
1888 break;
1889 }
1890 if (p_priv->cflag & PARENB) {
1891 /* note USA_PARITY_NONE == 0 */
1892 msg.lcr |= (p_priv->cflag & PARODD)?
Alan Coxdeb91682008-07-22 11:13:08 +01001893 USA_PARITY_ODD : USA_PARITY_EVEN;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001894 }
1895 msg.setLcr = 0xff;
1896
1897 msg.ctsFlowControl = (p_priv->flow_control == flow_cts);
1898 msg.xonFlowControl = 0;
1899 msg.setFlowControl = 0xff;
1900 msg.forwardingLength = 16;
1901 msg.xonChar = 17;
1902 msg.xoffChar = 19;
1903
1904 /* Opening port */
1905 if (reset_port == 1) {
1906 msg._txOn = 1;
1907 msg._txOff = 0;
1908 msg.txFlush = 0;
1909 msg.txBreak = 0;
1910 msg.rxOn = 1;
1911 msg.rxOff = 0;
1912 msg.rxFlush = 1;
1913 msg.rxForward = 0;
1914 msg.returnStatus = 0;
1915 msg.resetDataToggle = 0xff;
1916 }
1917
1918 /* Closing port */
1919 else if (reset_port == 2) {
1920 msg._txOn = 0;
1921 msg._txOff = 1;
1922 msg.txFlush = 0;
1923 msg.txBreak = 0;
1924 msg.rxOn = 0;
1925 msg.rxOff = 1;
1926 msg.rxFlush = 1;
1927 msg.rxForward = 0;
1928 msg.returnStatus = 0;
1929 msg.resetDataToggle = 0;
1930 }
1931
1932 /* Sending intermediate configs */
1933 else {
Alan Coxdeb91682008-07-22 11:13:08 +01001934 msg._txOn = (!p_priv->break_on);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001935 msg._txOff = 0;
1936 msg.txFlush = 0;
1937 msg.txBreak = (p_priv->break_on);
1938 msg.rxOn = 0;
1939 msg.rxOff = 0;
1940 msg.rxFlush = 0;
1941 msg.rxForward = 0;
1942 msg.returnStatus = 0;
1943 msg.resetDataToggle = 0x0;
1944 }
1945
Alan Coxdeb91682008-07-22 11:13:08 +01001946 /* Do handshaking outputs */
Linus Torvalds1da177e2005-04-16 15:20:36 -07001947 msg.setTxTriState_setRts = 0xff;
1948 msg.txTriState_rts = p_priv->rts_state;
1949
1950 msg.setHskoa_setDtr = 0xff;
1951 msg.hskoa_dtr = p_priv->dtr_state;
Alan Coxdeb91682008-07-22 11:13:08 +01001952
Linus Torvalds1da177e2005-04-16 15:20:36 -07001953 p_priv->resend_cont = 0;
Alan Coxdeb91682008-07-22 11:13:08 +01001954 memcpy(this_urb->transfer_buffer, &msg, sizeof(msg));
1955
Linus Torvalds1da177e2005-04-16 15:20:36 -07001956 /* send the data out the device on control endpoint */
1957 this_urb->transfer_buffer_length = sizeof(msg);
1958
1959 this_urb->dev = serial->dev;
Alan Coxdeb91682008-07-22 11:13:08 +01001960 err = usb_submit_urb(this_urb, GFP_ATOMIC);
1961 if (err != 0)
Harvey Harrison441b62c2008-03-03 16:08:34 -08001962 dbg("%s - usb_submit_urb(setup) failed (%d)", __func__, err);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001963#if 0
1964 else {
Harvey Harrison441b62c2008-03-03 16:08:34 -08001965 dbg("%s - usb_submit_urb(%d) OK %d bytes (end %d)", __func__
Linus Torvalds1da177e2005-04-16 15:20:36 -07001966 outcont_urb, this_urb->transfer_buffer_length,
1967 usb_pipeendpoint(this_urb->pipe));
1968 }
1969#endif
1970
Alan Coxa5b6f602008-04-08 17:16:06 +01001971 return 0;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001972}
1973
1974static int keyspan_usa28_send_setup(struct usb_serial *serial,
1975 struct usb_serial_port *port,
1976 int reset_port)
1977{
Alan Coxdeb91682008-07-22 11:13:08 +01001978 struct keyspan_usa28_portControlMessage msg;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001979 struct keyspan_serial_private *s_priv;
1980 struct keyspan_port_private *p_priv;
1981 const struct keyspan_device_details *d_details;
1982 struct urb *this_urb;
1983 int device_port, err;
1984
Alan Coxdeb91682008-07-22 11:13:08 +01001985 dbg("%s", __func__);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001986
1987 s_priv = usb_get_serial_data(serial);
1988 p_priv = usb_get_serial_port_data(port);
1989 d_details = s_priv->device_details;
1990 device_port = port->number - port->serial->minor;
1991
1992 /* only do something if we have a bulk out endpoint */
Alan Coxdeb91682008-07-22 11:13:08 +01001993 this_urb = p_priv->outcont_urb;
1994 if (this_urb == NULL) {
Harvey Harrison441b62c2008-03-03 16:08:34 -08001995 dbg("%s - oops no urb.", __func__);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001996 return -1;
1997 }
1998
1999 /* Save reset port val for resend.
Lucy McCoy0ca12682007-05-18 12:10:41 -07002000 Don't overwrite resend for open/close condition. */
2001 if ((reset_port + 1) > p_priv->resend_cont)
Linus Torvalds1da177e2005-04-16 15:20:36 -07002002 p_priv->resend_cont = reset_port + 1;
2003 if (this_urb->status == -EINPROGRESS) {
Alan Coxdeb91682008-07-22 11:13:08 +01002004 dbg("%s already writing", __func__);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002005 mdelay(5);
Alan Coxdeb91682008-07-22 11:13:08 +01002006 return -1;
Linus Torvalds1da177e2005-04-16 15:20:36 -07002007 }
2008
Alan Coxdeb91682008-07-22 11:13:08 +01002009 memset(&msg, 0, sizeof(struct keyspan_usa28_portControlMessage));
Linus Torvalds1da177e2005-04-16 15:20:36 -07002010
2011 msg.setBaudRate = 1;
2012 if (d_details->calculate_baud_rate(p_priv->baud, d_details->baudclk,
Alan Coxdeb91682008-07-22 11:13:08 +01002013 &msg.baudHi, &msg.baudLo, NULL, device_port) == KEYSPAN_INVALID_BAUD_RATE) {
2014 dbg("%s - Invalid baud rate requested %d.",
2015 __func__, p_priv->baud);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002016 msg.baudLo = 0xff;
2017 msg.baudHi = 0xb2; /* Values for 9600 baud */
2018 }
2019
2020 /* If parity is enabled, we must calculate it ourselves. */
2021 msg.parity = 0; /* XXX for now */
2022
2023 msg.ctsFlowControl = (p_priv->flow_control == flow_cts);
2024 msg.xonFlowControl = 0;
2025
Alan Coxdeb91682008-07-22 11:13:08 +01002026 /* Do handshaking outputs, DTR is inverted relative to RTS */
Linus Torvalds1da177e2005-04-16 15:20:36 -07002027 msg.rts = p_priv->rts_state;
2028 msg.dtr = p_priv->dtr_state;
2029
2030 msg.forwardingLength = 16;
2031 msg.forwardMs = 10;
2032 msg.breakThreshold = 45;
2033 msg.xonChar = 17;
2034 msg.xoffChar = 19;
2035
2036 /*msg.returnStatus = 1;
2037 msg.resetDataToggle = 0xff;*/
2038 /* Opening port */
2039 if (reset_port == 1) {
2040 msg._txOn = 1;
2041 msg._txOff = 0;
2042 msg.txFlush = 0;
2043 msg.txForceXoff = 0;
2044 msg.txBreak = 0;
2045 msg.rxOn = 1;
2046 msg.rxOff = 0;
2047 msg.rxFlush = 1;
2048 msg.rxForward = 0;
2049 msg.returnStatus = 0;
2050 msg.resetDataToggle = 0xff;
2051 }
2052 /* Closing port */
2053 else if (reset_port == 2) {
2054 msg._txOn = 0;
2055 msg._txOff = 1;
2056 msg.txFlush = 0;
2057 msg.txForceXoff = 0;
2058 msg.txBreak = 0;
2059 msg.rxOn = 0;
2060 msg.rxOff = 1;
2061 msg.rxFlush = 1;
2062 msg.rxForward = 0;
2063 msg.returnStatus = 0;
2064 msg.resetDataToggle = 0;
2065 }
2066 /* Sending intermediate configs */
2067 else {
Alan Coxdeb91682008-07-22 11:13:08 +01002068 msg._txOn = (!p_priv->break_on);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002069 msg._txOff = 0;
2070 msg.txFlush = 0;
2071 msg.txForceXoff = 0;
2072 msg.txBreak = (p_priv->break_on);
2073 msg.rxOn = 0;
2074 msg.rxOff = 0;
2075 msg.rxFlush = 0;
2076 msg.rxForward = 0;
2077 msg.returnStatus = 0;
2078 msg.resetDataToggle = 0x0;
2079 }
2080
2081 p_priv->resend_cont = 0;
Alan Coxdeb91682008-07-22 11:13:08 +01002082 memcpy(this_urb->transfer_buffer, &msg, sizeof(msg));
Linus Torvalds1da177e2005-04-16 15:20:36 -07002083
2084 /* send the data out the device on control endpoint */
2085 this_urb->transfer_buffer_length = sizeof(msg);
2086
2087 this_urb->dev = serial->dev;
Alan Coxdeb91682008-07-22 11:13:08 +01002088 err = usb_submit_urb(this_urb, GFP_ATOMIC);
2089 if (err != 0)
Harvey Harrison441b62c2008-03-03 16:08:34 -08002090 dbg("%s - usb_submit_urb(setup) failed", __func__);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002091#if 0
2092 else {
Harvey Harrison441b62c2008-03-03 16:08:34 -08002093 dbg("%s - usb_submit_urb(setup) OK %d bytes", __func__,
Linus Torvalds1da177e2005-04-16 15:20:36 -07002094 this_urb->transfer_buffer_length);
2095 }
2096#endif
2097
Alan Coxa5b6f602008-04-08 17:16:06 +01002098 return 0;
Linus Torvalds1da177e2005-04-16 15:20:36 -07002099}
2100
2101static int keyspan_usa49_send_setup(struct usb_serial *serial,
2102 struct usb_serial_port *port,
2103 int reset_port)
2104{
Lucy McCoy0ca12682007-05-18 12:10:41 -07002105 struct keyspan_usa49_portControlMessage msg;
2106 struct usb_ctrlrequest *dr = NULL;
Linus Torvalds1da177e2005-04-16 15:20:36 -07002107 struct keyspan_serial_private *s_priv;
2108 struct keyspan_port_private *p_priv;
2109 const struct keyspan_device_details *d_details;
Linus Torvalds1da177e2005-04-16 15:20:36 -07002110 struct urb *this_urb;
2111 int err, device_port;
2112
Alan Coxdeb91682008-07-22 11:13:08 +01002113 dbg("%s", __func__);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002114
2115 s_priv = usb_get_serial_data(serial);
2116 p_priv = usb_get_serial_port_data(port);
2117 d_details = s_priv->device_details;
2118
Linus Torvalds1da177e2005-04-16 15:20:36 -07002119 this_urb = s_priv->glocont_urb;
2120
Lucy McCoy0ca12682007-05-18 12:10:41 -07002121 /* Work out which port within the device is being setup */
Linus Torvalds1da177e2005-04-16 15:20:36 -07002122 device_port = port->number - port->serial->minor;
2123
Huzaifa Sidhpurwalad8661502011-02-21 12:58:44 +05302124 /* Make sure we have an urb then send the message */
Linus Torvalds1da177e2005-04-16 15:20:36 -07002125 if (this_urb == NULL) {
Harvey Harrison441b62c2008-03-03 16:08:34 -08002126 dbg("%s - oops no urb for port %d.", __func__, port->number);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002127 return -1;
2128 }
2129
Huzaifa Sidhpurwalad8661502011-02-21 12:58:44 +05302130 dbg("%s - endpoint %d port %d (%d)",
2131 __func__, usb_pipeendpoint(this_urb->pipe),
2132 port->number, device_port);
2133
Linus Torvalds1da177e2005-04-16 15:20:36 -07002134 /* Save reset port val for resend.
Lucy McCoy0ca12682007-05-18 12:10:41 -07002135 Don't overwrite resend for open/close condition. */
2136 if ((reset_port + 1) > p_priv->resend_cont)
Linus Torvalds1da177e2005-04-16 15:20:36 -07002137 p_priv->resend_cont = reset_port + 1;
Lucy McCoy0ca12682007-05-18 12:10:41 -07002138
Linus Torvalds1da177e2005-04-16 15:20:36 -07002139 if (this_urb->status == -EINPROGRESS) {
Alan Coxdeb91682008-07-22 11:13:08 +01002140 /* dbg("%s - already writing", __func__); */
Linus Torvalds1da177e2005-04-16 15:20:36 -07002141 mdelay(5);
Alan Coxdeb91682008-07-22 11:13:08 +01002142 return -1;
Linus Torvalds1da177e2005-04-16 15:20:36 -07002143 }
2144
Alan Coxdeb91682008-07-22 11:13:08 +01002145 memset(&msg, 0, sizeof(struct keyspan_usa49_portControlMessage));
Linus Torvalds1da177e2005-04-16 15:20:36 -07002146
2147 /*msg.portNumber = port->number;*/
2148 msg.portNumber = device_port;
Alan Coxdeb91682008-07-22 11:13:08 +01002149
2150 /* Only set baud rate if it's changed */
Linus Torvalds1da177e2005-04-16 15:20:36 -07002151 if (p_priv->old_baud != p_priv->baud) {
2152 p_priv->old_baud = p_priv->baud;
2153 msg.setClocking = 0xff;
2154 if (d_details->calculate_baud_rate
2155 (p_priv->baud, d_details->baudclk, &msg.baudHi,
Alan Coxdeb91682008-07-22 11:13:08 +01002156 &msg.baudLo, &msg.prescaler, device_port) == KEYSPAN_INVALID_BAUD_RATE) {
2157 dbg("%s - Invalid baud rate %d requested, using 9600.",
2158 __func__, p_priv->baud);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002159 msg.baudLo = 0;
2160 msg.baudHi = 125; /* Values for 9600 baud */
2161 msg.prescaler = 10;
2162 }
Alan Coxdeb91682008-07-22 11:13:08 +01002163 /* msg.setPrescaler = 0xff; */
Linus Torvalds1da177e2005-04-16 15:20:36 -07002164 }
2165
2166 msg.lcr = (p_priv->cflag & CSTOPB)? STOPBITS_678_2: STOPBITS_5678_1;
2167 switch (p_priv->cflag & CSIZE) {
2168 case CS5:
2169 msg.lcr |= USA_DATABITS_5;
2170 break;
2171 case CS6:
2172 msg.lcr |= USA_DATABITS_6;
2173 break;
2174 case CS7:
2175 msg.lcr |= USA_DATABITS_7;
2176 break;
2177 case CS8:
2178 msg.lcr |= USA_DATABITS_8;
2179 break;
2180 }
2181 if (p_priv->cflag & PARENB) {
2182 /* note USA_PARITY_NONE == 0 */
2183 msg.lcr |= (p_priv->cflag & PARODD)?
Alan Coxdeb91682008-07-22 11:13:08 +01002184 USA_PARITY_ODD : USA_PARITY_EVEN;
Linus Torvalds1da177e2005-04-16 15:20:36 -07002185 }
2186 msg.setLcr = 0xff;
2187
2188 msg.ctsFlowControl = (p_priv->flow_control == flow_cts);
2189 msg.xonFlowControl = 0;
2190 msg.setFlowControl = 0xff;
Alan Coxdeb91682008-07-22 11:13:08 +01002191
Linus Torvalds1da177e2005-04-16 15:20:36 -07002192 msg.forwardingLength = 16;
2193 msg.xonChar = 17;
2194 msg.xoffChar = 19;
2195
Alan Coxdeb91682008-07-22 11:13:08 +01002196 /* Opening port */
Linus Torvalds1da177e2005-04-16 15:20:36 -07002197 if (reset_port == 1) {
2198 msg._txOn = 1;
2199 msg._txOff = 0;
2200 msg.txFlush = 0;
2201 msg.txBreak = 0;
2202 msg.rxOn = 1;
2203 msg.rxOff = 0;
2204 msg.rxFlush = 1;
2205 msg.rxForward = 0;
2206 msg.returnStatus = 0;
2207 msg.resetDataToggle = 0xff;
2208 msg.enablePort = 1;
2209 msg.disablePort = 0;
2210 }
2211 /* Closing port */
2212 else if (reset_port == 2) {
2213 msg._txOn = 0;
2214 msg._txOff = 1;
2215 msg.txFlush = 0;
2216 msg.txBreak = 0;
2217 msg.rxOn = 0;
2218 msg.rxOff = 1;
2219 msg.rxFlush = 1;
2220 msg.rxForward = 0;
2221 msg.returnStatus = 0;
2222 msg.resetDataToggle = 0;
2223 msg.enablePort = 0;
2224 msg.disablePort = 1;
2225 }
2226 /* Sending intermediate configs */
2227 else {
Alan Coxdeb91682008-07-22 11:13:08 +01002228 msg._txOn = (!p_priv->break_on);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002229 msg._txOff = 0;
2230 msg.txFlush = 0;
2231 msg.txBreak = (p_priv->break_on);
2232 msg.rxOn = 0;
2233 msg.rxOff = 0;
2234 msg.rxFlush = 0;
2235 msg.rxForward = 0;
2236 msg.returnStatus = 0;
2237 msg.resetDataToggle = 0x0;
2238 msg.enablePort = 0;
2239 msg.disablePort = 0;
2240 }
2241
Alan Coxdeb91682008-07-22 11:13:08 +01002242 /* Do handshaking outputs */
Linus Torvalds1da177e2005-04-16 15:20:36 -07002243 msg.setRts = 0xff;
2244 msg.rts = p_priv->rts_state;
2245
2246 msg.setDtr = 0xff;
2247 msg.dtr = p_priv->dtr_state;
Alan Coxdeb91682008-07-22 11:13:08 +01002248
Linus Torvalds1da177e2005-04-16 15:20:36 -07002249 p_priv->resend_cont = 0;
Linus Torvalds1da177e2005-04-16 15:20:36 -07002250
Alan Coxdeb91682008-07-22 11:13:08 +01002251 /* if the device is a 49wg, we send control message on usb
2252 control EP 0 */
Lucy McCoy0ca12682007-05-18 12:10:41 -07002253
2254 if (d_details->product_id == keyspan_usa49wg_product_id) {
2255 dr = (void *)(s_priv->ctrl_buf);
2256 dr->bRequestType = USB_TYPE_VENDOR | USB_DIR_OUT;
2257 dr->bRequest = 0xB0; /* 49wg control message */;
2258 dr->wValue = 0;
2259 dr->wIndex = 0;
2260 dr->wLength = cpu_to_le16(sizeof(msg));
2261
Alan Coxdeb91682008-07-22 11:13:08 +01002262 memcpy(s_priv->glocont_buf, &msg, sizeof(msg));
Lucy McCoy0ca12682007-05-18 12:10:41 -07002263
Alan Coxdeb91682008-07-22 11:13:08 +01002264 usb_fill_control_urb(this_urb, serial->dev,
2265 usb_sndctrlpipe(serial->dev, 0),
2266 (unsigned char *)dr, s_priv->glocont_buf,
2267 sizeof(msg), usa49_glocont_callback, serial);
Lucy McCoy0ca12682007-05-18 12:10:41 -07002268
2269 } else {
2270 memcpy(this_urb->transfer_buffer, &msg, sizeof(msg));
Alan Coxdeb91682008-07-22 11:13:08 +01002271
Lucy McCoy0ca12682007-05-18 12:10:41 -07002272 /* send the data out the device on control endpoint */
2273 this_urb->transfer_buffer_length = sizeof(msg);
2274
2275 this_urb->dev = serial->dev;
2276 }
Alan Coxdeb91682008-07-22 11:13:08 +01002277 err = usb_submit_urb(this_urb, GFP_ATOMIC);
2278 if (err != 0)
Harvey Harrison441b62c2008-03-03 16:08:34 -08002279 dbg("%s - usb_submit_urb(setup) failed (%d)", __func__, err);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002280#if 0
2281 else {
Harvey Harrison441b62c2008-03-03 16:08:34 -08002282 dbg("%s - usb_submit_urb(%d) OK %d bytes (end %d)", __func__,
Lucy McCoy0ca12682007-05-18 12:10:41 -07002283 outcont_urb, this_urb->transfer_buffer_length,
2284 usb_pipeendpoint(this_urb->pipe));
Linus Torvalds1da177e2005-04-16 15:20:36 -07002285 }
2286#endif
2287
Alan Coxa5b6f602008-04-08 17:16:06 +01002288 return 0;
Linus Torvalds1da177e2005-04-16 15:20:36 -07002289}
2290
2291static int keyspan_usa90_send_setup(struct usb_serial *serial,
2292 struct usb_serial_port *port,
2293 int reset_port)
2294{
Alan Coxdeb91682008-07-22 11:13:08 +01002295 struct keyspan_usa90_portControlMessage msg;
Linus Torvalds1da177e2005-04-16 15:20:36 -07002296 struct keyspan_serial_private *s_priv;
2297 struct keyspan_port_private *p_priv;
2298 const struct keyspan_device_details *d_details;
2299 struct urb *this_urb;
2300 int err;
2301 u8 prescaler;
2302
Alan Coxdeb91682008-07-22 11:13:08 +01002303 dbg("%s", __func__);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002304
2305 s_priv = usb_get_serial_data(serial);
2306 p_priv = usb_get_serial_port_data(port);
2307 d_details = s_priv->device_details;
2308
2309 /* only do something if we have a bulk out endpoint */
Alan Coxdeb91682008-07-22 11:13:08 +01002310 this_urb = p_priv->outcont_urb;
2311 if (this_urb == NULL) {
Harvey Harrison441b62c2008-03-03 16:08:34 -08002312 dbg("%s - oops no urb.", __func__);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002313 return -1;
2314 }
2315
2316 /* Save reset port val for resend.
2317 Don't overwrite resend for open/close condition. */
2318 if ((reset_port + 1) > p_priv->resend_cont)
2319 p_priv->resend_cont = reset_port + 1;
2320 if (this_urb->status == -EINPROGRESS) {
Alan Coxdeb91682008-07-22 11:13:08 +01002321 dbg("%s already writing", __func__);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002322 mdelay(5);
Alan Coxdeb91682008-07-22 11:13:08 +01002323 return -1;
Linus Torvalds1da177e2005-04-16 15:20:36 -07002324 }
2325
Alan Coxdeb91682008-07-22 11:13:08 +01002326 memset(&msg, 0, sizeof(struct keyspan_usa90_portControlMessage));
Linus Torvalds1da177e2005-04-16 15:20:36 -07002327
Alan Coxdeb91682008-07-22 11:13:08 +01002328 /* Only set baud rate if it's changed */
Linus Torvalds1da177e2005-04-16 15:20:36 -07002329 if (p_priv->old_baud != p_priv->baud) {
2330 p_priv->old_baud = p_priv->baud;
2331 msg.setClocking = 0x01;
2332 if (d_details->calculate_baud_rate
2333 (p_priv->baud, d_details->baudclk, &msg.baudHi,
Alan Coxdeb91682008-07-22 11:13:08 +01002334 &msg.baudLo, &prescaler, 0) == KEYSPAN_INVALID_BAUD_RATE) {
2335 dbg("%s - Invalid baud rate %d requested, using 9600.",
2336 __func__, p_priv->baud);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002337 p_priv->baud = 9600;
Alan Coxdeb91682008-07-22 11:13:08 +01002338 d_details->calculate_baud_rate(p_priv->baud, d_details->baudclk,
Linus Torvalds1da177e2005-04-16 15:20:36 -07002339 &msg.baudHi, &msg.baudLo, &prescaler, 0);
2340 }
2341 msg.setRxMode = 1;
2342 msg.setTxMode = 1;
2343 }
2344
2345 /* modes must always be correctly specified */
Alan Coxdeb91682008-07-22 11:13:08 +01002346 if (p_priv->baud > 57600) {
Linus Torvalds1da177e2005-04-16 15:20:36 -07002347 msg.rxMode = RXMODE_DMA;
2348 msg.txMode = TXMODE_DMA;
Alan Coxdeb91682008-07-22 11:13:08 +01002349 } else {
Linus Torvalds1da177e2005-04-16 15:20:36 -07002350 msg.rxMode = RXMODE_BYHAND;
2351 msg.txMode = TXMODE_BYHAND;
2352 }
2353
2354 msg.lcr = (p_priv->cflag & CSTOPB)? STOPBITS_678_2: STOPBITS_5678_1;
2355 switch (p_priv->cflag & CSIZE) {
2356 case CS5:
2357 msg.lcr |= USA_DATABITS_5;
2358 break;
2359 case CS6:
2360 msg.lcr |= USA_DATABITS_6;
2361 break;
2362 case CS7:
2363 msg.lcr |= USA_DATABITS_7;
2364 break;
2365 case CS8:
2366 msg.lcr |= USA_DATABITS_8;
2367 break;
2368 }
2369 if (p_priv->cflag & PARENB) {
2370 /* note USA_PARITY_NONE == 0 */
2371 msg.lcr |= (p_priv->cflag & PARODD)?
Alan Coxdeb91682008-07-22 11:13:08 +01002372 USA_PARITY_ODD : USA_PARITY_EVEN;
Linus Torvalds1da177e2005-04-16 15:20:36 -07002373 }
2374 if (p_priv->old_cflag != p_priv->cflag) {
2375 p_priv->old_cflag = p_priv->cflag;
2376 msg.setLcr = 0x01;
2377 }
2378
2379 if (p_priv->flow_control == flow_cts)
2380 msg.txFlowControl = TXFLOW_CTS;
2381 msg.setTxFlowControl = 0x01;
2382 msg.setRxFlowControl = 0x01;
Alan Coxdeb91682008-07-22 11:13:08 +01002383
Linus Torvalds1da177e2005-04-16 15:20:36 -07002384 msg.rxForwardingLength = 16;
Alan Coxdeb91682008-07-22 11:13:08 +01002385 msg.rxForwardingTimeout = 16;
Linus Torvalds1da177e2005-04-16 15:20:36 -07002386 msg.txAckSetting = 0;
2387 msg.xonChar = 17;
2388 msg.xoffChar = 19;
2389
Alan Coxdeb91682008-07-22 11:13:08 +01002390 /* Opening port */
Linus Torvalds1da177e2005-04-16 15:20:36 -07002391 if (reset_port == 1) {
2392 msg.portEnabled = 1;
2393 msg.rxFlush = 1;
2394 msg.txBreak = (p_priv->break_on);
2395 }
2396 /* Closing port */
Alan Coxdeb91682008-07-22 11:13:08 +01002397 else if (reset_port == 2)
Linus Torvalds1da177e2005-04-16 15:20:36 -07002398 msg.portEnabled = 0;
Linus Torvalds1da177e2005-04-16 15:20:36 -07002399 /* Sending intermediate configs */
2400 else {
Alan Stern1f871582010-02-17 10:05:47 -05002401 msg.portEnabled = 1;
Linus Torvalds1da177e2005-04-16 15:20:36 -07002402 msg.txBreak = (p_priv->break_on);
2403 }
2404
Alan Coxdeb91682008-07-22 11:13:08 +01002405 /* Do handshaking outputs */
Linus Torvalds1da177e2005-04-16 15:20:36 -07002406 msg.setRts = 0x01;
2407 msg.rts = p_priv->rts_state;
2408
2409 msg.setDtr = 0x01;
2410 msg.dtr = p_priv->dtr_state;
Alan Coxdeb91682008-07-22 11:13:08 +01002411
Linus Torvalds1da177e2005-04-16 15:20:36 -07002412 p_priv->resend_cont = 0;
Alan Coxdeb91682008-07-22 11:13:08 +01002413 memcpy(this_urb->transfer_buffer, &msg, sizeof(msg));
2414
Linus Torvalds1da177e2005-04-16 15:20:36 -07002415 /* send the data out the device on control endpoint */
2416 this_urb->transfer_buffer_length = sizeof(msg);
2417
2418 this_urb->dev = serial->dev;
Alan Coxdeb91682008-07-22 11:13:08 +01002419 err = usb_submit_urb(this_urb, GFP_ATOMIC);
2420 if (err != 0)
Harvey Harrison441b62c2008-03-03 16:08:34 -08002421 dbg("%s - usb_submit_urb(setup) failed (%d)", __func__, err);
Alan Coxa5b6f602008-04-08 17:16:06 +01002422 return 0;
Linus Torvalds1da177e2005-04-16 15:20:36 -07002423}
2424
Lucy McCoy0ca12682007-05-18 12:10:41 -07002425static int keyspan_usa67_send_setup(struct usb_serial *serial,
2426 struct usb_serial_port *port,
2427 int reset_port)
2428{
2429 struct keyspan_usa67_portControlMessage msg;
2430 struct keyspan_serial_private *s_priv;
2431 struct keyspan_port_private *p_priv;
2432 const struct keyspan_device_details *d_details;
2433 struct urb *this_urb;
2434 int err, device_port;
2435
Alan Coxdeb91682008-07-22 11:13:08 +01002436 dbg("%s", __func__);
Lucy McCoy0ca12682007-05-18 12:10:41 -07002437
2438 s_priv = usb_get_serial_data(serial);
2439 p_priv = usb_get_serial_port_data(port);
2440 d_details = s_priv->device_details;
2441
2442 this_urb = s_priv->glocont_urb;
2443
2444 /* Work out which port within the device is being setup */
2445 device_port = port->number - port->serial->minor;
2446
2447 /* Make sure we have an urb then send the message */
2448 if (this_urb == NULL) {
Harvey Harrison441b62c2008-03-03 16:08:34 -08002449 dbg("%s - oops no urb for port %d.", __func__,
Lucy McCoy0ca12682007-05-18 12:10:41 -07002450 port->number);
2451 return -1;
2452 }
2453
2454 /* Save reset port val for resend.
2455 Don't overwrite resend for open/close condition. */
2456 if ((reset_port + 1) > p_priv->resend_cont)
2457 p_priv->resend_cont = reset_port + 1;
2458 if (this_urb->status == -EINPROGRESS) {
Alan Coxdeb91682008-07-22 11:13:08 +01002459 /* dbg("%s - already writing", __func__); */
Lucy McCoy0ca12682007-05-18 12:10:41 -07002460 mdelay(5);
Alan Coxdeb91682008-07-22 11:13:08 +01002461 return -1;
Lucy McCoy0ca12682007-05-18 12:10:41 -07002462 }
2463
2464 memset(&msg, 0, sizeof(struct keyspan_usa67_portControlMessage));
2465
2466 msg.port = device_port;
2467
2468 /* Only set baud rate if it's changed */
2469 if (p_priv->old_baud != p_priv->baud) {
2470 p_priv->old_baud = p_priv->baud;
2471 msg.setClocking = 0xff;
2472 if (d_details->calculate_baud_rate
2473 (p_priv->baud, d_details->baudclk, &msg.baudHi,
Alan Coxdeb91682008-07-22 11:13:08 +01002474 &msg.baudLo, &msg.prescaler, device_port) == KEYSPAN_INVALID_BAUD_RATE) {
2475 dbg("%s - Invalid baud rate %d requested, using 9600.",
2476 __func__, p_priv->baud);
Lucy McCoy0ca12682007-05-18 12:10:41 -07002477 msg.baudLo = 0;
2478 msg.baudHi = 125; /* Values for 9600 baud */
2479 msg.prescaler = 10;
2480 }
2481 msg.setPrescaler = 0xff;
2482 }
2483
2484 msg.lcr = (p_priv->cflag & CSTOPB) ? STOPBITS_678_2 : STOPBITS_5678_1;
2485 switch (p_priv->cflag & CSIZE) {
2486 case CS5:
2487 msg.lcr |= USA_DATABITS_5;
2488 break;
2489 case CS6:
2490 msg.lcr |= USA_DATABITS_6;
2491 break;
2492 case CS7:
2493 msg.lcr |= USA_DATABITS_7;
2494 break;
2495 case CS8:
2496 msg.lcr |= USA_DATABITS_8;
2497 break;
2498 }
2499 if (p_priv->cflag & PARENB) {
2500 /* note USA_PARITY_NONE == 0 */
2501 msg.lcr |= (p_priv->cflag & PARODD)?
Alan Coxdeb91682008-07-22 11:13:08 +01002502 USA_PARITY_ODD : USA_PARITY_EVEN;
Lucy McCoy0ca12682007-05-18 12:10:41 -07002503 }
2504 msg.setLcr = 0xff;
2505
2506 msg.ctsFlowControl = (p_priv->flow_control == flow_cts);
2507 msg.xonFlowControl = 0;
2508 msg.setFlowControl = 0xff;
2509 msg.forwardingLength = 16;
2510 msg.xonChar = 17;
2511 msg.xoffChar = 19;
2512
2513 if (reset_port == 1) {
2514 /* Opening port */
2515 msg._txOn = 1;
2516 msg._txOff = 0;
2517 msg.txFlush = 0;
2518 msg.txBreak = 0;
2519 msg.rxOn = 1;
2520 msg.rxOff = 0;
2521 msg.rxFlush = 1;
2522 msg.rxForward = 0;
2523 msg.returnStatus = 0;
2524 msg.resetDataToggle = 0xff;
2525 } else if (reset_port == 2) {
2526 /* Closing port */
2527 msg._txOn = 0;
2528 msg._txOff = 1;
2529 msg.txFlush = 0;
2530 msg.txBreak = 0;
2531 msg.rxOn = 0;
2532 msg.rxOff = 1;
2533 msg.rxFlush = 1;
2534 msg.rxForward = 0;
2535 msg.returnStatus = 0;
2536 msg.resetDataToggle = 0;
2537 } else {
2538 /* Sending intermediate configs */
Alan Coxdeb91682008-07-22 11:13:08 +01002539 msg._txOn = (!p_priv->break_on);
Lucy McCoy0ca12682007-05-18 12:10:41 -07002540 msg._txOff = 0;
2541 msg.txFlush = 0;
2542 msg.txBreak = (p_priv->break_on);
2543 msg.rxOn = 0;
2544 msg.rxOff = 0;
2545 msg.rxFlush = 0;
2546 msg.rxForward = 0;
2547 msg.returnStatus = 0;
2548 msg.resetDataToggle = 0x0;
2549 }
2550
2551 /* Do handshaking outputs */
2552 msg.setTxTriState_setRts = 0xff;
2553 msg.txTriState_rts = p_priv->rts_state;
2554
2555 msg.setHskoa_setDtr = 0xff;
2556 msg.hskoa_dtr = p_priv->dtr_state;
2557
2558 p_priv->resend_cont = 0;
2559
2560 memcpy(this_urb->transfer_buffer, &msg, sizeof(msg));
2561
2562 /* send the data out the device on control endpoint */
2563 this_urb->transfer_buffer_length = sizeof(msg);
2564 this_urb->dev = serial->dev;
2565
2566 err = usb_submit_urb(this_urb, GFP_ATOMIC);
2567 if (err != 0)
Harvey Harrison441b62c2008-03-03 16:08:34 -08002568 dbg("%s - usb_submit_urb(setup) failed (%d)", __func__,
Lucy McCoy0ca12682007-05-18 12:10:41 -07002569 err);
Alan Coxa5b6f602008-04-08 17:16:06 +01002570 return 0;
Lucy McCoy0ca12682007-05-18 12:10:41 -07002571}
2572
Linus Torvalds1da177e2005-04-16 15:20:36 -07002573static void keyspan_send_setup(struct usb_serial_port *port, int reset_port)
2574{
2575 struct usb_serial *serial = port->serial;
2576 struct keyspan_serial_private *s_priv;
2577 const struct keyspan_device_details *d_details;
2578
Alan Coxdeb91682008-07-22 11:13:08 +01002579 dbg("%s", __func__);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002580
2581 s_priv = usb_get_serial_data(serial);
2582 d_details = s_priv->device_details;
2583
2584 switch (d_details->msg_format) {
2585 case msg_usa26:
2586 keyspan_usa26_send_setup(serial, port, reset_port);
2587 break;
2588 case msg_usa28:
2589 keyspan_usa28_send_setup(serial, port, reset_port);
2590 break;
2591 case msg_usa49:
2592 keyspan_usa49_send_setup(serial, port, reset_port);
2593 break;
2594 case msg_usa90:
2595 keyspan_usa90_send_setup(serial, port, reset_port);
2596 break;
Lucy McCoy0ca12682007-05-18 12:10:41 -07002597 case msg_usa67:
2598 keyspan_usa67_send_setup(serial, port, reset_port);
2599 break;
Linus Torvalds1da177e2005-04-16 15:20:36 -07002600 }
2601}
2602
2603
2604/* Gets called by the "real" driver (ie once firmware is loaded
2605 and renumeration has taken place. */
Alan Coxdeb91682008-07-22 11:13:08 +01002606static int keyspan_startup(struct usb_serial *serial)
Linus Torvalds1da177e2005-04-16 15:20:36 -07002607{
2608 int i, err;
2609 struct usb_serial_port *port;
2610 struct keyspan_serial_private *s_priv;
2611 struct keyspan_port_private *p_priv;
2612 const struct keyspan_device_details *d_details;
2613
Harvey Harrison441b62c2008-03-03 16:08:34 -08002614 dbg("%s", __func__);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002615
2616 for (i = 0; (d_details = keyspan_devices[i]) != NULL; ++i)
Alan Coxdeb91682008-07-22 11:13:08 +01002617 if (d_details->product_id ==
2618 le16_to_cpu(serial->dev->descriptor.idProduct))
Linus Torvalds1da177e2005-04-16 15:20:36 -07002619 break;
2620 if (d_details == NULL) {
Alan Coxdeb91682008-07-22 11:13:08 +01002621 dev_err(&serial->dev->dev, "%s - unknown product id %x\n",
2622 __func__, le16_to_cpu(serial->dev->descriptor.idProduct));
Linus Torvalds1da177e2005-04-16 15:20:36 -07002623 return 1;
2624 }
2625
2626 /* Setup private data for serial driver */
Eric Sesterhenn80b6ca42006-02-27 21:29:43 +01002627 s_priv = kzalloc(sizeof(struct keyspan_serial_private), GFP_KERNEL);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002628 if (!s_priv) {
Alan Coxdeb91682008-07-22 11:13:08 +01002629 dbg("%s - kmalloc for keyspan_serial_private failed.",
2630 __func__);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002631 return -ENOMEM;
2632 }
Linus Torvalds1da177e2005-04-16 15:20:36 -07002633
2634 s_priv->device_details = d_details;
2635 usb_set_serial_data(serial, s_priv);
2636
2637 /* Now setup per port private data */
2638 for (i = 0; i < serial->num_ports; i++) {
2639 port = serial->port[i];
Alan Coxdeb91682008-07-22 11:13:08 +01002640 p_priv = kzalloc(sizeof(struct keyspan_port_private),
2641 GFP_KERNEL);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002642 if (!p_priv) {
Harvey Harrison441b62c2008-03-03 16:08:34 -08002643 dbg("%s - kmalloc for keyspan_port_private (%d) failed!.", __func__, i);
Alan Coxdeb91682008-07-22 11:13:08 +01002644 return 1;
Linus Torvalds1da177e2005-04-16 15:20:36 -07002645 }
Linus Torvalds1da177e2005-04-16 15:20:36 -07002646 p_priv->device_details = d_details;
2647 usb_set_serial_port_data(port, p_priv);
2648 }
2649
2650 keyspan_setup_urbs(serial);
2651
Lucy McCoy0ca12682007-05-18 12:10:41 -07002652 if (s_priv->instat_urb != NULL) {
2653 s_priv->instat_urb->dev = serial->dev;
2654 err = usb_submit_urb(s_priv->instat_urb, GFP_KERNEL);
2655 if (err != 0)
Harvey Harrison441b62c2008-03-03 16:08:34 -08002656 dbg("%s - submit instat urb failed %d", __func__,
Lucy McCoy0ca12682007-05-18 12:10:41 -07002657 err);
2658 }
2659 if (s_priv->indat_urb != NULL) {
2660 s_priv->indat_urb->dev = serial->dev;
2661 err = usb_submit_urb(s_priv->indat_urb, GFP_KERNEL);
2662 if (err != 0)
Harvey Harrison441b62c2008-03-03 16:08:34 -08002663 dbg("%s - submit indat urb failed %d", __func__,
Lucy McCoy0ca12682007-05-18 12:10:41 -07002664 err);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002665 }
Alan Coxdeb91682008-07-22 11:13:08 +01002666
Alan Coxa5b6f602008-04-08 17:16:06 +01002667 return 0;
Linus Torvalds1da177e2005-04-16 15:20:36 -07002668}
2669
Alan Sternf9c99bb2009-06-02 11:53:55 -04002670static void keyspan_disconnect(struct usb_serial *serial)
Linus Torvalds1da177e2005-04-16 15:20:36 -07002671{
2672 int i, j;
2673 struct usb_serial_port *port;
2674 struct keyspan_serial_private *s_priv;
2675 struct keyspan_port_private *p_priv;
2676
Harvey Harrison441b62c2008-03-03 16:08:34 -08002677 dbg("%s", __func__);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002678
2679 s_priv = usb_get_serial_data(serial);
2680
2681 /* Stop reading/writing urbs */
2682 stop_urb(s_priv->instat_urb);
2683 stop_urb(s_priv->glocont_urb);
Lucy McCoy0ca12682007-05-18 12:10:41 -07002684 stop_urb(s_priv->indat_urb);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002685 for (i = 0; i < serial->num_ports; ++i) {
2686 port = serial->port[i];
2687 p_priv = usb_get_serial_port_data(port);
2688 stop_urb(p_priv->inack_urb);
2689 stop_urb(p_priv->outcont_urb);
2690 for (j = 0; j < 2; j++) {
2691 stop_urb(p_priv->in_urbs[j]);
2692 stop_urb(p_priv->out_urbs[j]);
2693 }
2694 }
2695
2696 /* Now free them */
Mariusz Kozlowski1cadc132006-11-08 15:36:34 +01002697 usb_free_urb(s_priv->instat_urb);
Lucy McCoy0ca12682007-05-18 12:10:41 -07002698 usb_free_urb(s_priv->indat_urb);
Mariusz Kozlowski1cadc132006-11-08 15:36:34 +01002699 usb_free_urb(s_priv->glocont_urb);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002700 for (i = 0; i < serial->num_ports; ++i) {
2701 port = serial->port[i];
2702 p_priv = usb_get_serial_port_data(port);
Mariusz Kozlowski1cadc132006-11-08 15:36:34 +01002703 usb_free_urb(p_priv->inack_urb);
2704 usb_free_urb(p_priv->outcont_urb);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002705 for (j = 0; j < 2; j++) {
Mariusz Kozlowski1cadc132006-11-08 15:36:34 +01002706 usb_free_urb(p_priv->in_urbs[j]);
2707 usb_free_urb(p_priv->out_urbs[j]);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002708 }
2709 }
Alan Sternf9c99bb2009-06-02 11:53:55 -04002710}
2711
2712static void keyspan_release(struct usb_serial *serial)
2713{
2714 int i;
2715 struct usb_serial_port *port;
2716 struct keyspan_serial_private *s_priv;
2717
2718 dbg("%s", __func__);
2719
2720 s_priv = usb_get_serial_data(serial);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002721
2722 /* dbg("Freeing serial->private."); */
2723 kfree(s_priv);
2724
2725 /* dbg("Freeing port->private."); */
2726 /* Now free per port private data */
2727 for (i = 0; i < serial->num_ports; i++) {
2728 port = serial->port[i];
2729 kfree(usb_get_serial_port_data(port));
2730 }
2731}
2732
Alan Coxdeb91682008-07-22 11:13:08 +01002733MODULE_AUTHOR(DRIVER_AUTHOR);
2734MODULE_DESCRIPTION(DRIVER_DESC);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002735MODULE_LICENSE("GPL");
2736
David Woodhouse2971c572008-05-30 14:04:03 +03002737MODULE_FIRMWARE("keyspan/usa28.fw");
2738MODULE_FIRMWARE("keyspan/usa28x.fw");
2739MODULE_FIRMWARE("keyspan/usa28xa.fw");
2740MODULE_FIRMWARE("keyspan/usa28xb.fw");
2741MODULE_FIRMWARE("keyspan/usa19.fw");
2742MODULE_FIRMWARE("keyspan/usa19qi.fw");
2743MODULE_FIRMWARE("keyspan/mpr.fw");
2744MODULE_FIRMWARE("keyspan/usa19qw.fw");
2745MODULE_FIRMWARE("keyspan/usa18x.fw");
2746MODULE_FIRMWARE("keyspan/usa19w.fw");
2747MODULE_FIRMWARE("keyspan/usa49w.fw");
2748MODULE_FIRMWARE("keyspan/usa49wlc.fw");
2749
Linus Torvalds1da177e2005-04-16 15:20:36 -07002750module_param(debug, bool, S_IRUGO | S_IWUSR);
2751MODULE_PARM_DESC(debug, "Debug enabled or not");
2752