2  * USB Cypress M8 driver
 
   5  *          Lonnie Mendez (dignome@gmail.com) 
 
   6  *      Copyright (C) 2003,2004
 
   7  *          Neil Whelchel (koyama@firstlight.net)
 
   9  *      This program is free software; you can redistribute it and/or modify
 
  10  *      it under the terms of the GNU General Public License as published by
 
  11  *      the Free Software Foundation; either version 2 of the License, or
 
  12  *      (at your option) any later version.
 
  14  * See Documentation/usb/usb-serial.txt for more information on using this driver
 
  16  * See http://geocities.com/i0xox0i for information on this driver and the
 
  17  * earthmate usb device.
 
  19  *  Lonnie Mendez <dignome@gmail.com>
 
  21  *      Fixed problem where setting or retreiving the serial config would fail with
 
  22  *      EPIPE.  Removed CRTS toggling so the driver behaves more like other usbserial
 
  23  *      adapters.  Issued new interval of 1ms instead of the default 10ms.  As a
 
  24  *      result, transfer speed has been substantially increased.  From avg. 850bps to
 
  25  *      avg. 3300bps.  initial termios has also been modified.  Cleaned up code and
 
  26  *      formatting issues so it is more readable.  Replaced the C++ style comments.
 
  28  *  Lonnie Mendez <dignome@gmail.com>
 
  30  *      Incorporated write buffering from pl2303 driver.  Fixed bug with line
 
  31  *      handling so both lines are raised in cypress_open. (was dropping rts)
 
  32  *      Various code cleanups made as well along with other misc bug fixes.
 
  34  *  Lonnie Mendez <dignome@gmail.com>
 
  36  *      Driver modified to support dynamic line settings.  Various improvments
 
  41  *      Driver first released.
 
  45 /* Thanks to Neil Whelchel for writing the first cypress m8 implementation for linux. */
 
  46 /* Thanks to cypress for providing references for the hid reports. */
 
  47 /* Thanks to Jiang Zhang for providing links and for general help. */
 
  48 /* Code originates and was built up from ftdi_sio, belkin, pl2303 and others. */
 
  51 #include <linux/kernel.h>
 
  52 #include <linux/errno.h>
 
  53 #include <linux/init.h>
 
  54 #include <linux/slab.h>
 
  55 #include <linux/tty.h>
 
  56 #include <linux/tty_driver.h>
 
  57 #include <linux/tty_flip.h>
 
  58 #include <linux/module.h>
 
  59 #include <linux/moduleparam.h>
 
  60 #include <linux/spinlock.h>
 
  61 #include <linux/usb.h>
 
  62 #include <linux/usb/serial.h>
 
  63 #include <linux/serial.h>
 
  64 #include <linux/delay.h>
 
  65 #include <asm/uaccess.h>
 
  67 #include "cypress_m8.h"
 
  70 #ifdef CONFIG_USB_SERIAL_DEBUG
 
  81 #define DRIVER_VERSION "v1.09"
 
  82 #define DRIVER_AUTHOR "Lonnie Mendez <dignome@gmail.com>, Neil Whelchel <koyama@firstlight.net>"
 
  83 #define DRIVER_DESC "Cypress USB to Serial Driver"
 
  85 /* write buffer size defines */
 
  86 #define CYPRESS_BUF_SIZE        1024
 
  87 #define CYPRESS_CLOSING_WAIT    (30*HZ)
 
  89 static struct usb_device_id id_table_earthmate [] = {
 
  90         { USB_DEVICE(VENDOR_ID_DELORME, PRODUCT_ID_EARTHMATEUSB) },
 
  91         { USB_DEVICE(VENDOR_ID_DELORME, PRODUCT_ID_EARTHMATEUSB_LT20) },
 
  92         { }                                             /* Terminating entry */
 
  95 static struct usb_device_id id_table_cyphidcomrs232 [] = {
 
  96         { USB_DEVICE(VENDOR_ID_CYPRESS, PRODUCT_ID_CYPHIDCOM) },
 
  97         { }                                             /* Terminating entry */
 
 100 static struct usb_device_id id_table_nokiaca42v2 [] = {
 
 101         { USB_DEVICE(VENDOR_ID_DAZZLE, PRODUCT_ID_CA42) },
 
 102         { }                                             /* Terminating entry */
 
 105 static struct usb_device_id id_table_combined [] = {
 
 106         { USB_DEVICE(VENDOR_ID_DELORME, PRODUCT_ID_EARTHMATEUSB) },
 
 107         { USB_DEVICE(VENDOR_ID_DELORME, PRODUCT_ID_EARTHMATEUSB_LT20) },
 
 108         { USB_DEVICE(VENDOR_ID_CYPRESS, PRODUCT_ID_CYPHIDCOM) },
 
 109         { USB_DEVICE(VENDOR_ID_DAZZLE, PRODUCT_ID_CA42) },
 
 110         { }                                             /* Terminating entry */
 
 113 MODULE_DEVICE_TABLE (usb, id_table_combined);
 
 115 static struct usb_driver cypress_driver = {
 
 117         .probe =        usb_serial_probe,
 
 118         .disconnect =   usb_serial_disconnect,
 
 119         .id_table =     id_table_combined,
 
 123 struct cypress_private {
 
 124         spinlock_t lock;                   /* private lock */
 
 125         int chiptype;                      /* identifier of device, for quirks/etc */
 
 126         int bytes_in;                      /* used for statistics */
 
 127         int bytes_out;                     /* used for statistics */
 
 128         int cmd_count;                     /* used for statistics */
 
 129         int cmd_ctrl;                      /* always set this to 1 before issuing a command */
 
 130         struct cypress_buf *buf;           /* write buffer */
 
 131         int write_urb_in_use;              /* write urb in use indicator */
 
 132         int write_urb_interval;            /* interval to use for write urb */
 
 133         int read_urb_interval;             /* interval to use for read urb */
 
 134         int comm_is_ok;                    /* true if communication is (still) ok */
 
 135         int termios_initialized;
 
 136         __u8 line_control;                 /* holds dtr / rts value */
 
 137         __u8 current_status;               /* received from last read - info on dsr,cts,cd,ri,etc */
 
 138         __u8 current_config;               /* stores the current configuration byte */
 
 139         __u8 rx_flags;                     /* throttling - used from whiteheat/ftdi_sio */
 
 140         int baud_rate;                     /* stores current baud rate in integer form */
 
 141         int cbr_mask;                      /* stores current baud rate in masked form */
 
 142         int isthrottled;                   /* if throttled, discard reads */
 
 143         wait_queue_head_t delta_msr_wait;  /* used for TIOCMIWAIT */
 
 144         char prev_status, diff_status;     /* used for TIOCMIWAIT */
 
 145         /* we pass a pointer to this as the arguement sent to cypress_set_termios old_termios */
 
 146         struct ktermios tmp_termios;       /* stores the old termios settings */
 
 149 /* write buffer structure */
 
 151         unsigned int    buf_size;
 
 157 /* function prototypes for the Cypress USB to serial device */
 
 158 static int  cypress_earthmate_startup   (struct usb_serial *serial);
 
 159 static int  cypress_hidcom_startup      (struct usb_serial *serial);
 
 160 static int  cypress_ca42v2_startup      (struct usb_serial *serial);
 
 161 static void cypress_shutdown            (struct usb_serial *serial);
 
 162 static int  cypress_open                (struct usb_serial_port *port, struct file *filp);
 
 163 static void cypress_close               (struct usb_serial_port *port, struct file *filp);
 
 164 static int  cypress_write               (struct usb_serial_port *port, const unsigned char *buf, int count);
 
 165 static void cypress_send                (struct usb_serial_port *port);
 
 166 static int  cypress_write_room          (struct usb_serial_port *port);
 
 167 static int  cypress_ioctl               (struct usb_serial_port *port, struct file * file, unsigned int cmd, unsigned long arg);
 
 168 static void cypress_set_termios         (struct usb_serial_port *port, struct ktermios * old);
 
 169 static int  cypress_tiocmget            (struct usb_serial_port *port, struct file *file);
 
 170 static int  cypress_tiocmset            (struct usb_serial_port *port, struct file *file, unsigned int set, unsigned int clear);
 
 171 static int  cypress_chars_in_buffer     (struct usb_serial_port *port);
 
 172 static void cypress_throttle            (struct usb_serial_port *port);
 
 173 static void cypress_unthrottle          (struct usb_serial_port *port);
 
 174 static void cypress_set_dead            (struct usb_serial_port *port);
 
 175 static void cypress_read_int_callback   (struct urb *urb);
 
 176 static void cypress_write_int_callback  (struct urb *urb);
 
 177 /* baud helper functions */
 
 178 static int       mask_to_rate           (unsigned mask);
 
 179 static unsigned  rate_to_mask           (int rate);
 
 180 /* write buffer functions */
 
 181 static struct cypress_buf *cypress_buf_alloc(unsigned int size);
 
 182 static void               cypress_buf_free(struct cypress_buf *cb);
 
 183 static void               cypress_buf_clear(struct cypress_buf *cb);
 
 184 static unsigned int       cypress_buf_data_avail(struct cypress_buf *cb);
 
 185 static unsigned int       cypress_buf_space_avail(struct cypress_buf *cb);
 
 186 static unsigned int       cypress_buf_put(struct cypress_buf *cb, const char *buf, unsigned int count);
 
 187 static unsigned int       cypress_buf_get(struct cypress_buf *cb, char *buf, unsigned int count);
 
 190 static struct usb_serial_driver cypress_earthmate_device = {
 
 192                 .owner =                THIS_MODULE,
 
 195         .description =                  "DeLorme Earthmate USB",
 
 196         .usb_driver =                   &cypress_driver,
 
 197         .id_table =                     id_table_earthmate,
 
 198         .num_interrupt_in =             1,
 
 199         .num_interrupt_out =            1,
 
 200         .num_bulk_in =                  NUM_DONT_CARE,
 
 201         .num_bulk_out =                 NUM_DONT_CARE,
 
 203         .attach =                       cypress_earthmate_startup,
 
 204         .shutdown =                     cypress_shutdown,
 
 205         .open =                         cypress_open,
 
 206         .close =                        cypress_close,
 
 207         .write =                        cypress_write,
 
 208         .write_room =                   cypress_write_room,
 
 209         .ioctl =                        cypress_ioctl,
 
 210         .set_termios =                  cypress_set_termios,
 
 211         .tiocmget =                     cypress_tiocmget,
 
 212         .tiocmset =                     cypress_tiocmset,
 
 213         .chars_in_buffer =              cypress_chars_in_buffer,
 
 214         .throttle =                     cypress_throttle,
 
 215         .unthrottle =                   cypress_unthrottle,
 
 216         .read_int_callback =            cypress_read_int_callback,
 
 217         .write_int_callback =           cypress_write_int_callback,
 
 220 static struct usb_serial_driver cypress_hidcom_device = {
 
 222                 .owner =                THIS_MODULE,
 
 225         .description =                  "HID->COM RS232 Adapter",
 
 226         .usb_driver =                   &cypress_driver,
 
 227         .id_table =                     id_table_cyphidcomrs232,
 
 228         .num_interrupt_in =             1,
 
 229         .num_interrupt_out =            1,
 
 230         .num_bulk_in =                  NUM_DONT_CARE,
 
 231         .num_bulk_out =                 NUM_DONT_CARE,
 
 233         .attach =                       cypress_hidcom_startup,
 
 234         .shutdown =                     cypress_shutdown,
 
 235         .open =                         cypress_open,
 
 236         .close =                        cypress_close,
 
 237         .write =                        cypress_write,
 
 238         .write_room =                   cypress_write_room,
 
 239         .ioctl =                        cypress_ioctl,
 
 240         .set_termios =                  cypress_set_termios,
 
 241         .tiocmget =                     cypress_tiocmget,
 
 242         .tiocmset =                     cypress_tiocmset,
 
 243         .chars_in_buffer =              cypress_chars_in_buffer,
 
 244         .throttle =                     cypress_throttle,
 
 245         .unthrottle =                   cypress_unthrottle,
 
 246         .read_int_callback =            cypress_read_int_callback,
 
 247         .write_int_callback =           cypress_write_int_callback,
 
 250 static struct usb_serial_driver cypress_ca42v2_device = {
 
 252                 .owner =                THIS_MODULE,
 
 253                 .name =                 "nokiaca42v2",
 
 255         .description =                  "Nokia CA-42 V2 Adapter",
 
 256         .usb_driver =                   &cypress_driver,
 
 257         .id_table =                     id_table_nokiaca42v2,
 
 258         .num_interrupt_in =             1,
 
 259         .num_interrupt_out =            1,
 
 260         .num_bulk_in =                  NUM_DONT_CARE,
 
 261         .num_bulk_out =                 NUM_DONT_CARE,
 
 263         .attach =                       cypress_ca42v2_startup,
 
 264         .shutdown =                     cypress_shutdown,
 
 265         .open =                         cypress_open,
 
 266         .close =                        cypress_close,
 
 267         .write =                        cypress_write,
 
 268         .write_room =                   cypress_write_room,
 
 269         .ioctl =                        cypress_ioctl,
 
 270         .set_termios =                  cypress_set_termios,
 
 271         .tiocmget =                     cypress_tiocmget,
 
 272         .tiocmset =                     cypress_tiocmset,
 
 273         .chars_in_buffer =              cypress_chars_in_buffer,
 
 274         .throttle =                     cypress_throttle,
 
 275         .unthrottle =                   cypress_unthrottle,
 
 276         .read_int_callback =            cypress_read_int_callback,
 
 277         .write_int_callback =           cypress_write_int_callback,
 
 280 /*****************************************************************************
 
 281  * Cypress serial helper functions
 
 282  *****************************************************************************/
 
 285 /* This function can either set or retrieve the current serial line settings */
 
 286 static int cypress_serial_control (struct usb_serial_port *port, unsigned baud_mask, int data_bits, int stop_bits,
 
 287                                    int parity_enable, int parity_type, int reset, int cypress_request_type)
 
 289         int new_baudrate = 0, retval = 0, tries = 0;
 
 290         struct cypress_private *priv;
 
 291         __u8 feature_buffer[8];
 
 294         dbg("%s", __FUNCTION__);
 
 296         priv = usb_get_serial_port_data(port);
 
 298         if (!priv->comm_is_ok)
 
 301         switch(cypress_request_type) {
 
 302                 case CYPRESS_SET_CONFIG:
 
 305                          * The general purpose firmware for the Cypress M8 allows for a maximum speed
 
 306                          * of 57600bps (I have no idea whether DeLorme chose to use the general purpose
 
 307                          * firmware or not), if you need to modify this speed setting for your own
 
 308                          * project please add your own chiptype and modify the code likewise.  The
 
 309                          * Cypress HID->COM device will work successfully up to 115200bps (but the
 
 310                          * actual throughput is around 3kBps).
 
 312                         if (baud_mask != priv->cbr_mask) {
 
 313                                 dbg("%s - baud rate is changing", __FUNCTION__);
 
 314                                 if ( priv->chiptype == CT_EARTHMATE ) {
 
 315                                         /* 300 and 600 baud rates are supported under the generic firmware,
 
 316                                          * but are not used with NMEA and SiRF protocols */
 
 318                                         if ( (baud_mask == B300) || (baud_mask == B600) ) {
 
 319                                                 err("%s - failed setting baud rate, unsupported speed",
 
 321                                                 new_baudrate = priv->baud_rate;
 
 322                                         } else if ( (new_baudrate = mask_to_rate(baud_mask)) == -1) {
 
 323                                                 err("%s - failed setting baud rate, unsupported speed",
 
 325                                                 new_baudrate = priv->baud_rate;
 
 327                                 } else if (priv->chiptype == CT_CYPHIDCOM) {
 
 328                                         if ( (new_baudrate = mask_to_rate(baud_mask)) == -1) {
 
 329                                                 err("%s - failed setting baud rate, unsupported speed",
 
 331                                                 new_baudrate = priv->baud_rate;
 
 333                                 } else if (priv->chiptype == CT_CA42V2) {
 
 334                                         if ( (new_baudrate = mask_to_rate(baud_mask)) == -1) {
 
 335                                                 err("%s - failed setting baud rate, unsupported speed",
 
 337                                                 new_baudrate = priv->baud_rate;
 
 339                                 } else if (priv->chiptype == CT_GENERIC) {
 
 340                                         if ( (new_baudrate = mask_to_rate(baud_mask)) == -1) {
 
 341                                                 err("%s - failed setting baud rate, unsupported speed",
 
 343                                                 new_baudrate = priv->baud_rate;
 
 346                                         info("%s - please define your chiptype", __FUNCTION__);
 
 347                                         new_baudrate = priv->baud_rate;
 
 349                         } else {  /* baud rate not changing, keep the old */
 
 350                                 new_baudrate = priv->baud_rate;
 
 352                         dbg("%s - baud rate is being sent as %d", __FUNCTION__, new_baudrate);
 
 354                         memset(feature_buffer, 0, 8);
 
 355                         /* fill the feature_buffer with new configuration */
 
 356                         *((u_int32_t *)feature_buffer) = new_baudrate;
 
 358                         feature_buffer[4] |= data_bits;   /* assign data bits in 2 bit space ( max 3 ) */
 
 360                         feature_buffer[4] |= (stop_bits << 3);   /* assign stop bits in 1 bit space */
 
 361                         feature_buffer[4] |= (parity_enable << 4);   /* assign parity flag in 1 bit space */
 
 362                         feature_buffer[4] |= (parity_type << 5);   /* assign parity type in 1 bit space */
 
 364                         feature_buffer[4] |= (reset << 7);   /* assign reset at end of byte, 1 bit space */
 
 366                         dbg("%s - device is being sent this feature report:", __FUNCTION__);
 
 367                         dbg("%s - %02X - %02X - %02X - %02X - %02X", __FUNCTION__, feature_buffer[0], feature_buffer[1],
 
 368                             feature_buffer[2], feature_buffer[3], feature_buffer[4]);
 
 371                         retval = usb_control_msg (port->serial->dev, usb_sndctrlpipe(port->serial->dev, 0),
 
 372                                                   HID_REQ_SET_REPORT, USB_DIR_OUT | USB_RECIP_INTERFACE | USB_TYPE_CLASS,
 
 373                                                           0x0300, 0, feature_buffer, 8, 500);
 
 378                         } while (retval != 8 && retval != -ENODEV);
 
 381                                 err("%s - failed sending serial line settings - %d", __FUNCTION__, retval);
 
 382                                 cypress_set_dead(port);
 
 384                                 spin_lock_irqsave(&priv->lock, flags);
 
 385                                 priv->baud_rate = new_baudrate;
 
 386                                 priv->cbr_mask = baud_mask;
 
 387                                 priv->current_config = feature_buffer[4];
 
 388                                 spin_unlock_irqrestore(&priv->lock, flags);
 
 391                 case CYPRESS_GET_CONFIG:
 
 392                         dbg("%s - retreiving serial line settings", __FUNCTION__);
 
 393                         /* set initial values in feature buffer */
 
 394                         memset(feature_buffer, 0, 8);
 
 397                         retval = usb_control_msg (port->serial->dev, usb_rcvctrlpipe(port->serial->dev, 0),
 
 398                                                   HID_REQ_GET_REPORT, USB_DIR_IN | USB_RECIP_INTERFACE | USB_TYPE_CLASS,
 
 399                                                           0x0300, 0, feature_buffer, 8, 500);
 
 404                         } while (retval != 5 && retval != -ENODEV);
 
 407                                 err("%s - failed to retrieve serial line settings - %d", __FUNCTION__, retval);
 
 408                                 cypress_set_dead(port);
 
 411                                 spin_lock_irqsave(&priv->lock, flags);
 
 413                                 /* store the config in one byte, and later use bit masks to check values */
 
 414                                 priv->current_config = feature_buffer[4];
 
 415                                 priv->baud_rate = *((u_int32_t *)feature_buffer);
 
 417                                 if ( (priv->cbr_mask = rate_to_mask(priv->baud_rate)) == 0x40)
 
 418                                         dbg("%s - failed setting the baud mask (not defined)", __FUNCTION__);
 
 419                                 spin_unlock_irqrestore(&priv->lock, flags);
 
 422         spin_lock_irqsave(&priv->lock, flags);
 
 424         spin_unlock_irqrestore(&priv->lock, flags);
 
 427 } /* cypress_serial_control */
 
 430 static void cypress_set_dead(struct usb_serial_port *port)
 
 432         struct cypress_private *priv = usb_get_serial_port_data(port);
 
 435         spin_lock_irqsave(&priv->lock, flags);
 
 436         if (!priv->comm_is_ok) {
 
 437                 spin_unlock_irqrestore(&priv->lock, flags);
 
 440         priv->comm_is_ok = 0;
 
 441         spin_unlock_irqrestore(&priv->lock, flags);
 
 443         err("cypress_m8 suspending failing port %d - interval might be too short",
 
 448 /* given a baud mask, it will return integer baud on success */
 
 449 static int mask_to_rate (unsigned mask)
 
 454                 case B0: rate = 0; break;
 
 455                 case B300: rate = 300; break;
 
 456                 case B600: rate = 600; break;
 
 457                 case B1200: rate = 1200; break;
 
 458                 case B2400: rate = 2400; break;
 
 459                 case B4800: rate = 4800; break;
 
 460                 case B9600: rate = 9600; break;
 
 461                 case B19200: rate = 19200; break;
 
 462                 case B38400: rate = 38400; break;
 
 463                 case B57600: rate = 57600; break;
 
 464                 case B115200: rate = 115200; break;
 
 472 static unsigned rate_to_mask (int rate)
 
 477                 case 0: mask = B0; break;
 
 478                 case 300: mask = B300; break;
 
 479                 case 600: mask = B600; break;
 
 480                 case 1200: mask = B1200; break;
 
 481                 case 2400: mask = B2400; break;
 
 482                 case 4800: mask = B4800; break;
 
 483                 case 9600: mask = B9600; break;
 
 484                 case 19200: mask = B19200; break;
 
 485                 case 38400: mask = B38400; break;
 
 486                 case 57600: mask = B57600; break;
 
 487                 case 115200: mask = B115200; break;
 
 488                 default: mask = 0x40;
 
 493 /*****************************************************************************
 
 494  * Cypress serial driver functions
 
 495  *****************************************************************************/
 
 498 static int generic_startup (struct usb_serial *serial)
 
 500         struct cypress_private *priv;
 
 501         struct usb_serial_port *port = serial->port[0];
 
 503         dbg("%s - port %d", __FUNCTION__, port->number);
 
 505         priv = kzalloc(sizeof (struct cypress_private), GFP_KERNEL);
 
 509         priv->comm_is_ok = !0;
 
 510         spin_lock_init(&priv->lock);
 
 511         priv->buf = cypress_buf_alloc(CYPRESS_BUF_SIZE);
 
 512         if (priv->buf == NULL) {
 
 516         init_waitqueue_head(&priv->delta_msr_wait);
 
 518         usb_reset_configuration (serial->dev);
 
 521         priv->line_control = 0;
 
 522         priv->termios_initialized = 0;
 
 524         priv->cbr_mask = B300;
 
 526                 priv->write_urb_interval = interval;
 
 527                 priv->read_urb_interval = interval;
 
 528                 dbg("%s - port %d read & write intervals forced to %d",
 
 529                     __FUNCTION__,port->number,interval);
 
 531                 priv->write_urb_interval = port->interrupt_out_urb->interval;
 
 532                 priv->read_urb_interval = port->interrupt_in_urb->interval;
 
 533                 dbg("%s - port %d intervals: read=%d write=%d",
 
 534                     __FUNCTION__,port->number,
 
 535                     priv->read_urb_interval,priv->write_urb_interval);
 
 537         usb_set_serial_port_data(port, priv);
 
 543 static int cypress_earthmate_startup (struct usb_serial *serial)
 
 545         struct cypress_private *priv;
 
 547         dbg("%s", __FUNCTION__);
 
 549         if (generic_startup(serial)) {
 
 550                 dbg("%s - Failed setting up port %d", __FUNCTION__,
 
 551                                 serial->port[0]->number);
 
 555         priv = usb_get_serial_port_data(serial->port[0]);
 
 556         priv->chiptype = CT_EARTHMATE;
 
 559 } /* cypress_earthmate_startup */
 
 562 static int cypress_hidcom_startup (struct usb_serial *serial)
 
 564         struct cypress_private *priv;
 
 566         dbg("%s", __FUNCTION__);
 
 568         if (generic_startup(serial)) {
 
 569                 dbg("%s - Failed setting up port %d", __FUNCTION__,
 
 570                                 serial->port[0]->number);
 
 574         priv = usb_get_serial_port_data(serial->port[0]);
 
 575         priv->chiptype = CT_CYPHIDCOM;
 
 578 } /* cypress_hidcom_startup */
 
 581 static int cypress_ca42v2_startup (struct usb_serial *serial)
 
 583         struct cypress_private *priv;
 
 585         dbg("%s", __FUNCTION__);
 
 587         if (generic_startup(serial)) {
 
 588                 dbg("%s - Failed setting up port %d", __FUNCTION__,
 
 589                                 serial->port[0]->number);
 
 593         priv = usb_get_serial_port_data(serial->port[0]);
 
 594         priv->chiptype = CT_CA42V2;
 
 597 } /* cypress_ca42v2_startup */
 
 600 static void cypress_shutdown (struct usb_serial *serial)
 
 602         struct cypress_private *priv;
 
 604         dbg ("%s - port %d", __FUNCTION__, serial->port[0]->number);
 
 606         /* all open ports are closed at this point */
 
 608         priv = usb_get_serial_port_data(serial->port[0]);
 
 611                 cypress_buf_free(priv->buf);
 
 613                 usb_set_serial_port_data(serial->port[0], NULL);
 
 618 static int cypress_open (struct usb_serial_port *port, struct file *filp)
 
 620         struct cypress_private *priv = usb_get_serial_port_data(port);
 
 621         struct usb_serial *serial = port->serial;
 
 625         dbg("%s - port %d", __FUNCTION__, port->number);
 
 627         if (!priv->comm_is_ok)
 
 630         /* clear halts before open */
 
 631         usb_clear_halt(serial->dev, 0x81);
 
 632         usb_clear_halt(serial->dev, 0x02);
 
 634         spin_lock_irqsave(&priv->lock, flags);
 
 635         /* reset read/write statistics */
 
 640         spin_unlock_irqrestore(&priv->lock, flags);
 
 642         /* setting to zero could cause data loss */
 
 643         port->tty->low_latency = 1;
 
 645         /* raise both lines and set termios */
 
 646         spin_lock_irqsave(&priv->lock, flags);
 
 647         priv->line_control = CONTROL_DTR | CONTROL_RTS;
 
 649         spin_unlock_irqrestore(&priv->lock, flags);
 
 650         result = cypress_write(port, NULL, 0);
 
 653                 dev_err(&port->dev, "%s - failed setting the control lines - error %d\n", __FUNCTION__, result);
 
 656                 dbg("%s - success setting the control lines", __FUNCTION__);    
 
 658         cypress_set_termios(port, &priv->tmp_termios);
 
 660         /* setup the port and start reading from the device */
 
 661         if(!port->interrupt_in_urb){
 
 662                 err("%s - interrupt_in_urb is empty!", __FUNCTION__);
 
 666         usb_fill_int_urb(port->interrupt_in_urb, serial->dev,
 
 667                 usb_rcvintpipe(serial->dev, port->interrupt_in_endpointAddress),
 
 668                 port->interrupt_in_urb->transfer_buffer, port->interrupt_in_urb->transfer_buffer_length,
 
 669                 cypress_read_int_callback, port, priv->read_urb_interval);
 
 670         result = usb_submit_urb(port->interrupt_in_urb, GFP_KERNEL);
 
 673                 dev_err(&port->dev, "%s - failed submitting read urb, error %d\n", __FUNCTION__, result);
 
 674                 cypress_set_dead(port);
 
 681 static void cypress_close(struct usb_serial_port *port, struct file * filp)
 
 683         struct cypress_private *priv = usb_get_serial_port_data(port);
 
 684         unsigned int c_cflag;
 
 690         dbg("%s - port %d", __FUNCTION__, port->number);
 
 692         /* wait for data to drain from buffer */
 
 693         spin_lock_irqsave(&priv->lock, flags);
 
 694         timeout = CYPRESS_CLOSING_WAIT;
 
 695         init_waitqueue_entry(&wait, current);
 
 696         add_wait_queue(&port->tty->write_wait, &wait);
 
 698                 set_current_state(TASK_INTERRUPTIBLE);
 
 699                 if (cypress_buf_data_avail(priv->buf) == 0
 
 700                 || timeout == 0 || signal_pending(current)
 
 701                 || !usb_get_intfdata(port->serial->interface))
 
 703                 spin_unlock_irqrestore(&priv->lock, flags);
 
 704                 timeout = schedule_timeout(timeout);
 
 705                 spin_lock_irqsave(&priv->lock, flags);
 
 707         set_current_state(TASK_RUNNING);
 
 708         remove_wait_queue(&port->tty->write_wait, &wait);
 
 709         /* clear out any remaining data in the buffer */
 
 710         cypress_buf_clear(priv->buf);
 
 711         spin_unlock_irqrestore(&priv->lock, flags);
 
 713         /* wait for characters to drain from device */
 
 714         bps = tty_get_baud_rate(port->tty);
 
 716                 timeout = max((HZ*2560)/bps,HZ/10);
 
 719         schedule_timeout_interruptible(timeout);
 
 721         dbg("%s - stopping urbs", __FUNCTION__);
 
 722         usb_kill_urb (port->interrupt_in_urb);
 
 723         usb_kill_urb (port->interrupt_out_urb);
 
 726                 c_cflag = port->tty->termios->c_cflag;
 
 727                 if (c_cflag & HUPCL) {
 
 728                         /* drop dtr and rts */
 
 729                         priv = usb_get_serial_port_data(port);
 
 730                         spin_lock_irqsave(&priv->lock, flags);
 
 731                         priv->line_control = 0;
 
 733                         spin_unlock_irqrestore(&priv->lock, flags);
 
 734                         cypress_write(port, NULL, 0);
 
 739                 dev_info (&port->dev, "Statistics: %d Bytes In | %d Bytes Out | %d Commands Issued\n",
 
 740                           priv->bytes_in, priv->bytes_out, priv->cmd_count);
 
 741 } /* cypress_close */
 
 744 static int cypress_write(struct usb_serial_port *port, const unsigned char *buf, int count)
 
 746         struct cypress_private *priv = usb_get_serial_port_data(port);
 
 749         dbg("%s - port %d, %d bytes", __FUNCTION__, port->number, count);
 
 751         /* line control commands, which need to be executed immediately,
 
 752            are not put into the buffer for obvious reasons.
 
 754         if (priv->cmd_ctrl) {
 
 762         spin_lock_irqsave(&priv->lock, flags);
 
 763         count = cypress_buf_put(priv->buf, buf, count);
 
 764         spin_unlock_irqrestore(&priv->lock, flags);
 
 770 } /* cypress_write */
 
 773 static void cypress_send(struct usb_serial_port *port)
 
 775         int count = 0, result, offset, actual_size;
 
 776         struct cypress_private *priv = usb_get_serial_port_data(port);
 
 779         if (!priv->comm_is_ok)
 
 782         dbg("%s - port %d", __FUNCTION__, port->number);
 
 783         dbg("%s - interrupt out size is %d", __FUNCTION__, port->interrupt_out_size);
 
 785         spin_lock_irqsave(&priv->lock, flags);
 
 786         if (priv->write_urb_in_use) {
 
 787                 dbg("%s - can't write, urb in use", __FUNCTION__);
 
 788                 spin_unlock_irqrestore(&priv->lock, flags);
 
 791         spin_unlock_irqrestore(&priv->lock, flags);
 
 794         memset(port->interrupt_out_urb->transfer_buffer, 0, port->interrupt_out_size);
 
 796         spin_lock_irqsave(&priv->lock, flags);
 
 797         switch (port->interrupt_out_size) {
 
 799                         /* this is for the CY7C64013... */
 
 801                         port->interrupt_out_buffer[0] = priv->line_control;
 
 804                         /* this is for the CY7C63743... */
 
 806                         port->interrupt_out_buffer[0] = priv->line_control;
 
 809                         dbg("%s - wrong packet size", __FUNCTION__);
 
 810                         spin_unlock_irqrestore(&priv->lock, flags);
 
 814         if (priv->line_control & CONTROL_RESET)
 
 815                 priv->line_control &= ~CONTROL_RESET;
 
 817         if (priv->cmd_ctrl) {
 
 819                 dbg("%s - line control command being issued", __FUNCTION__);
 
 820                 spin_unlock_irqrestore(&priv->lock, flags);
 
 823                 spin_unlock_irqrestore(&priv->lock, flags);
 
 825         count = cypress_buf_get(priv->buf, &port->interrupt_out_buffer[offset],
 
 826                                 port->interrupt_out_size-offset);
 
 832         switch (port->interrupt_out_size) {
 
 834                         port->interrupt_out_buffer[1] = count;
 
 837                         port->interrupt_out_buffer[0] |= count;
 
 840         dbg("%s - count is %d", __FUNCTION__, count);
 
 843         spin_lock_irqsave(&priv->lock, flags);
 
 844         priv->write_urb_in_use = 1;
 
 845         spin_unlock_irqrestore(&priv->lock, flags);
 
 850                 actual_size = count + (port->interrupt_out_size == 32 ? 2 : 1);
 
 852         usb_serial_debug_data(debug, &port->dev, __FUNCTION__, port->interrupt_out_size,
 
 853                               port->interrupt_out_urb->transfer_buffer);
 
 855         usb_fill_int_urb(port->interrupt_out_urb, port->serial->dev,
 
 856                 usb_sndintpipe(port->serial->dev, port->interrupt_out_endpointAddress),
 
 857                 port->interrupt_out_buffer, port->interrupt_out_size,
 
 858                 cypress_write_int_callback, port, priv->write_urb_interval);
 
 859         result = usb_submit_urb (port->interrupt_out_urb, GFP_ATOMIC);
 
 861                 dev_err(&port->dev, "%s - failed submitting write urb, error %d\n", __FUNCTION__,
 
 863                 priv->write_urb_in_use = 0;
 
 864                 cypress_set_dead(port);
 
 867         spin_lock_irqsave(&priv->lock, flags);
 
 868         if (priv->cmd_ctrl) {
 
 871         priv->bytes_out += count; /* do not count the line control and size bytes */
 
 872         spin_unlock_irqrestore(&priv->lock, flags);
 
 874         usb_serial_port_softint(port);
 
 878 /* returns how much space is available in the soft buffer */
 
 879 static int cypress_write_room(struct usb_serial_port *port)
 
 881         struct cypress_private *priv = usb_get_serial_port_data(port);
 
 885         dbg("%s - port %d", __FUNCTION__, port->number);
 
 887         spin_lock_irqsave(&priv->lock, flags);
 
 888         room = cypress_buf_space_avail(priv->buf);
 
 889         spin_unlock_irqrestore(&priv->lock, flags);
 
 891         dbg("%s - returns %d", __FUNCTION__, room);
 
 896 static int cypress_tiocmget (struct usb_serial_port *port, struct file *file)
 
 898         struct cypress_private *priv = usb_get_serial_port_data(port);
 
 899         __u8 status, control;
 
 900         unsigned int result = 0;
 
 903         dbg("%s - port %d", __FUNCTION__, port->number);
 
 905         spin_lock_irqsave(&priv->lock, flags);
 
 906         control = priv->line_control;
 
 907         status = priv->current_status;
 
 908         spin_unlock_irqrestore(&priv->lock, flags);
 
 910         result = ((control & CONTROL_DTR)        ? TIOCM_DTR : 0)
 
 911                 | ((control & CONTROL_RTS)       ? TIOCM_RTS : 0)
 
 912                 | ((status & UART_CTS)        ? TIOCM_CTS : 0)
 
 913                 | ((status & UART_DSR)        ? TIOCM_DSR : 0)
 
 914                 | ((status & UART_RI)         ? TIOCM_RI  : 0)
 
 915                 | ((status & UART_CD)         ? TIOCM_CD  : 0);
 
 917         dbg("%s - result = %x", __FUNCTION__, result);
 
 923 static int cypress_tiocmset (struct usb_serial_port *port, struct file *file,
 
 924                                unsigned int set, unsigned int clear)
 
 926         struct cypress_private *priv = usb_get_serial_port_data(port);
 
 929         dbg("%s - port %d", __FUNCTION__, port->number);
 
 931         spin_lock_irqsave(&priv->lock, flags);
 
 933                 priv->line_control |= CONTROL_RTS;
 
 935                 priv->line_control |= CONTROL_DTR;
 
 936         if (clear & TIOCM_RTS)
 
 937                 priv->line_control &= ~CONTROL_RTS;
 
 938         if (clear & TIOCM_DTR)
 
 939                 priv->line_control &= ~CONTROL_DTR;
 
 940         spin_unlock_irqrestore(&priv->lock, flags);
 
 943         return cypress_write(port, NULL, 0);
 
 947 static int cypress_ioctl (struct usb_serial_port *port, struct file * file, unsigned int cmd, unsigned long arg)
 
 949         struct cypress_private *priv = usb_get_serial_port_data(port);
 
 951         dbg("%s - port %d, cmd 0x%.4x", __FUNCTION__, port->number, cmd);
 
 955                         if (copy_to_user((void __user *)arg, port->tty->termios, sizeof(struct ktermios))) {
 
 961                         if (copy_from_user(port->tty->termios, (void __user *)arg, sizeof(struct ktermios))) {
 
 964                         /* here we need to call cypress_set_termios to invoke the new settings */
 
 965                         cypress_set_termios(port, &priv->tmp_termios);
 
 968                 /* This code comes from drivers/char/serial.c and ftdi_sio.c */
 
 970                         while (priv != NULL) {
 
 971                                 interruptible_sleep_on(&priv->delta_msr_wait);
 
 972                                 /* see if a signal did it */
 
 973                                 if (signal_pending(current))
 
 976                                         char diff = priv->diff_status;
 
 979                                                 return -EIO; /* no change => error */
 
 982                                         /* consume all events */
 
 983                                         priv->diff_status = 0;
 
 985                                         /* return 0 if caller wanted to know about these bits */
 
 986                                         if ( ((arg & TIOCM_RNG) && (diff & UART_RI)) ||
 
 987                                              ((arg & TIOCM_DSR) && (diff & UART_DSR)) ||
 
 988                                              ((arg & TIOCM_CD) && (diff & UART_CD)) ||
 
 989                                              ((arg & TIOCM_CTS) && (diff & UART_CTS)) ) {
 
 992                                         /* otherwise caller can't care less about what happened,
 
 993                                          * and so we continue to wait for more events.
 
1003         dbg("%s - arg not supported - it was 0x%04x - check include/asm/ioctls.h", __FUNCTION__, cmd);
 
1005         return -ENOIOCTLCMD;
 
1006 } /* cypress_ioctl */
 
1009 static void cypress_set_termios (struct usb_serial_port *port,
 
1010                 struct ktermios *old_termios)
 
1012         struct cypress_private *priv = usb_get_serial_port_data(port);
 
1013         struct tty_struct *tty;
 
1014         int data_bits, stop_bits, parity_type, parity_enable;
 
1015         unsigned cflag, iflag, baud_mask;
 
1016         unsigned long flags;
 
1020         dbg("%s - port %d", __FUNCTION__, port->number);
 
1023         if ((!tty) || (!tty->termios)) {
 
1024                 dbg("%s - no tty structures", __FUNCTION__);
 
1028         spin_lock_irqsave(&priv->lock, flags);
 
1029         if (!priv->termios_initialized) {
 
1030                 if (priv->chiptype == CT_EARTHMATE) {
 
1031                         *(tty->termios) = tty_std_termios;
 
1032                         tty->termios->c_cflag = B4800 | CS8 | CREAD | HUPCL |
 
1034                 } else if (priv->chiptype == CT_CYPHIDCOM) {
 
1035                         *(tty->termios) = tty_std_termios;
 
1036                         tty->termios->c_cflag = B9600 | CS8 | CREAD | HUPCL |
 
1038                 } else if (priv->chiptype == CT_CA42V2) {
 
1039                         *(tty->termios) = tty_std_termios;
 
1040                         tty->termios->c_cflag = B9600 | CS8 | CREAD | HUPCL |
 
1043                 priv->termios_initialized = 1;
 
1045         spin_unlock_irqrestore(&priv->lock, flags);
 
1047         cflag = tty->termios->c_cflag;
 
1048         iflag = tty->termios->c_iflag;
 
1050         /* check if there are new settings */
 
1052                 if ((cflag != old_termios->c_cflag) ||
 
1053                         (RELEVANT_IFLAG(iflag) !=
 
1054                          RELEVANT_IFLAG(old_termios->c_iflag))) {
 
1055                         dbg("%s - attempting to set new termios settings",
 
1057                         /* should make a copy of this in case something goes
 
1058                          * wrong in the function, we can restore it */
 
1059                         spin_lock_irqsave(&priv->lock, flags);
 
1060                         priv->tmp_termios = *(tty->termios);
 
1061                         spin_unlock_irqrestore(&priv->lock, flags);
 
1063                         dbg("%s - nothing to do, exiting", __FUNCTION__);
 
1069         /* set number of data bits, parity, stop bits */
 
1070         /* when parity is disabled the parity type bit is ignored */
 
1072         /* 1 means 2 stop bits, 0 means 1 stop bit */
 
1073         stop_bits = cflag & CSTOPB ? 1 : 0;
 
1075         if (cflag & PARENB) {
 
1077                 /* 1 means odd parity, 0 means even parity */
 
1078                 parity_type = cflag & PARODD ? 1 : 0;
 
1080                 parity_enable = parity_type = 0;
 
1082         if (cflag & CSIZE) {
 
1083                 switch (cflag & CSIZE) {
 
1097                                 err("%s - CSIZE was set, but not CS5-CS8",
 
1104         spin_lock_irqsave(&priv->lock, flags);
 
1105         oldlines = priv->line_control;
 
1106         if ((cflag & CBAUD) == B0) {
 
1107                 /* drop dtr and rts */
 
1108                 dbg("%s - dropping the lines, baud rate 0bps", __FUNCTION__);
 
1110                 priv->line_control &= ~(CONTROL_DTR | CONTROL_RTS);
 
1112                 baud_mask = (cflag & CBAUD);
 
1115                                 dbg("%s - setting baud 300bps", __FUNCTION__);
 
1118                                 dbg("%s - setting baud 600bps", __FUNCTION__);
 
1121                                 dbg("%s - setting baud 1200bps", __FUNCTION__);
 
1124                                 dbg("%s - setting baud 2400bps", __FUNCTION__);
 
1127                                 dbg("%s - setting baud 4800bps", __FUNCTION__);
 
1130                                 dbg("%s - setting baud 9600bps", __FUNCTION__);
 
1133                                 dbg("%s - setting baud 19200bps", __FUNCTION__);
 
1136                                 dbg("%s - setting baud 38400bps", __FUNCTION__);
 
1139                                 dbg("%s - setting baud 57600bps", __FUNCTION__);
 
1142                                 dbg("%s - setting baud 115200bps", __FUNCTION__);
 
1145                                 dbg("%s - unknown masked baud rate", __FUNCTION__);
 
1147                 priv->line_control = (CONTROL_DTR | CONTROL_RTS);
 
1149         spin_unlock_irqrestore(&priv->lock, flags);
 
1151         dbg("%s - sending %d stop_bits, %d parity_enable, %d parity_type, "
 
1152                         "%d data_bits (+5)", __FUNCTION__, stop_bits,
 
1153                         parity_enable, parity_type, data_bits);
 
1155         cypress_serial_control(port, baud_mask, data_bits, stop_bits,
 
1156                         parity_enable, parity_type, 0, CYPRESS_SET_CONFIG);
 
1158         /* we perform a CYPRESS_GET_CONFIG so that the current settings are
 
1159          * filled into the private structure this should confirm that all is
 
1160          * working if it returns what we just set */
 
1161         cypress_serial_control(port, 0, 0, 0, 0, 0, 0, CYPRESS_GET_CONFIG);
 
1163         /* Here we can define custom tty settings for devices; the main tty
 
1164          * termios flag base comes from empeg.c */
 
1166         spin_lock_irqsave(&priv->lock, flags);
 
1167         if ( (priv->chiptype == CT_EARTHMATE) && (priv->baud_rate == 4800) ) {
 
1168                 dbg("Using custom termios settings for a baud rate of "
 
1170                 /* define custom termios settings for NMEA protocol */
 
1172                 tty->termios->c_iflag /* input modes - */
 
1173                         &= ~(IGNBRK  /* disable ignore break */
 
1174                         | BRKINT     /* disable break causes interrupt */
 
1175                         | PARMRK     /* disable mark parity errors */
 
1176                         | ISTRIP     /* disable clear high bit of input char */
 
1177                         | INLCR      /* disable translate NL to CR */
 
1178                         | IGNCR      /* disable ignore CR */
 
1179                         | ICRNL      /* disable translate CR to NL */
 
1180                         | IXON);     /* disable enable XON/XOFF flow control */
 
1182                 tty->termios->c_oflag /* output modes */
 
1183                         &= ~OPOST;    /* disable postprocess output char */
 
1185                 tty->termios->c_lflag /* line discipline modes */
 
1186                         &= ~(ECHO     /* disable echo input characters */
 
1187                         | ECHONL      /* disable echo new line */
 
1188                         | ICANON      /* disable erase, kill, werase, and rprnt
 
1189                                          special characters */
 
1190                         | ISIG        /* disable interrupt, quit, and suspend
 
1191                                          special characters */
 
1192                         | IEXTEN);    /* disable non-POSIX special characters */
 
1193         } /* CT_CYPHIDCOM: Application should handle this for device */
 
1195         linechange = (priv->line_control != oldlines);
 
1196         spin_unlock_irqrestore(&priv->lock, flags);
 
1198         /* if necessary, set lines */
 
1201                 cypress_write(port, NULL, 0);
 
1203 } /* cypress_set_termios */
 
1206 /* returns amount of data still left in soft buffer */
 
1207 static int cypress_chars_in_buffer(struct usb_serial_port *port)
 
1209         struct cypress_private *priv = usb_get_serial_port_data(port);
 
1211         unsigned long flags;
 
1213         dbg("%s - port %d", __FUNCTION__, port->number);
 
1215         spin_lock_irqsave(&priv->lock, flags);
 
1216         chars = cypress_buf_data_avail(priv->buf);
 
1217         spin_unlock_irqrestore(&priv->lock, flags);
 
1219         dbg("%s - returns %d", __FUNCTION__, chars);
 
1224 static void cypress_throttle (struct usb_serial_port *port)
 
1226         struct cypress_private *priv = usb_get_serial_port_data(port);
 
1227         unsigned long flags;
 
1229         dbg("%s - port %d", __FUNCTION__, port->number);
 
1231         spin_lock_irqsave(&priv->lock, flags);
 
1232         priv->rx_flags = THROTTLED;
 
1233         spin_unlock_irqrestore(&priv->lock, flags);
 
1237 static void cypress_unthrottle (struct usb_serial_port *port)
 
1239         struct cypress_private *priv = usb_get_serial_port_data(port);
 
1240         int actually_throttled, result;
 
1241         unsigned long flags;
 
1243         dbg("%s - port %d", __FUNCTION__, port->number);
 
1245         spin_lock_irqsave(&priv->lock, flags);
 
1246         actually_throttled = priv->rx_flags & ACTUALLY_THROTTLED;
 
1248         spin_unlock_irqrestore(&priv->lock, flags);
 
1250         if (!priv->comm_is_ok)
 
1253         if (actually_throttled) {
 
1254                 port->interrupt_in_urb->dev = port->serial->dev;
 
1256                 result = usb_submit_urb(port->interrupt_in_urb, GFP_ATOMIC);
 
1258                         dev_err(&port->dev, "%s - failed submitting read urb, "
 
1259                                         "error %d\n", __FUNCTION__, result);
 
1260                         cypress_set_dead(port);
 
1266 static void cypress_read_int_callback(struct urb *urb)
 
1268         struct usb_serial_port *port = (struct usb_serial_port *)urb->context;
 
1269         struct cypress_private *priv = usb_get_serial_port_data(port);
 
1270         struct tty_struct *tty;
 
1271         unsigned char *data = urb->transfer_buffer;
 
1272         unsigned long flags;
 
1273         char tty_flag = TTY_NORMAL;
 
1279         dbg("%s - port %d", __FUNCTION__, port->number);
 
1281         switch (urb->status) {
 
1282         case 0: /* success */
 
1287                 /* precursor to disconnect so just go away */
 
1290                 usb_clear_halt(port->serial->dev,0x81);
 
1293                 /* something ugly is going on... */
 
1294                 dev_err(&urb->dev->dev,"%s - unexpected nonzero read status received: %d\n",
 
1295                         __FUNCTION__,urb->status);
 
1296                 cypress_set_dead(port);
 
1300         spin_lock_irqsave(&priv->lock, flags);
 
1301         if (priv->rx_flags & THROTTLED) {
 
1302                 dbg("%s - now throttling", __FUNCTION__);
 
1303                 priv->rx_flags |= ACTUALLY_THROTTLED;
 
1304                 spin_unlock_irqrestore(&priv->lock, flags);
 
1307         spin_unlock_irqrestore(&priv->lock, flags);
 
1311                 dbg("%s - bad tty pointer - exiting", __FUNCTION__);
 
1315         spin_lock_irqsave(&priv->lock, flags);
 
1316         switch(urb->actual_length) {
 
1318                         /* This is for the CY7C64013... */
 
1319                         priv->current_status = data[0] & 0xF8;
 
1320                         bytes = data[1] + 2;
 
1326                         /* This is for the CY7C63743... */
 
1327                         priv->current_status = data[0] & 0xF8;
 
1328                         bytes = (data[0] & 0x07) + 1;
 
1334                         dbg("%s - wrong packet size - received %d bytes",
 
1335                                         __FUNCTION__, urb->actual_length);
 
1336                         spin_unlock_irqrestore(&priv->lock, flags);
 
1339         spin_unlock_irqrestore(&priv->lock, flags);
 
1341         usb_serial_debug_data (debug, &port->dev, __FUNCTION__,
 
1342                         urb->actual_length, data);
 
1344         spin_lock_irqsave(&priv->lock, flags);
 
1345         /* check to see if status has changed */
 
1347                 if (priv->current_status != priv->prev_status) {
 
1348                         priv->diff_status |= priv->current_status ^
 
1350                         wake_up_interruptible(&priv->delta_msr_wait);
 
1351                         priv->prev_status = priv->current_status;
 
1354         spin_unlock_irqrestore(&priv->lock, flags);
 
1356         /* hangup, as defined in acm.c... this might be a bad place for it
 
1358         if (tty && !(tty->termios->c_cflag & CLOCAL) &&
 
1359                         !(priv->current_status & UART_CD)) {
 
1360                 dbg("%s - calling hangup", __FUNCTION__);
 
1365         /* There is one error bit... I'm assuming it is a parity error
 
1366          * indicator as the generic firmware will set this bit to 1 if a
 
1367          * parity error occurs.
 
1368          * I can not find reference to any other error events. */
 
1369         spin_lock_irqsave(&priv->lock, flags);
 
1370         if (priv->current_status & CYP_ERROR) {
 
1371                 spin_unlock_irqrestore(&priv->lock, flags);
 
1372                 tty_flag = TTY_PARITY;
 
1373                 dbg("%s - Parity Error detected", __FUNCTION__);
 
1375                 spin_unlock_irqrestore(&priv->lock, flags);
 
1377         /* process read if there is data other than line status */
 
1378         if (tty && (bytes > i)) {
 
1379                 bytes = tty_buffer_request_room(tty, bytes);
 
1380                 for (; i < bytes ; ++i) {
 
1381                         dbg("pushing byte number %d - %d - %c", i, data[i],
 
1383                         tty_insert_flip_char(tty, data[i], tty_flag);
 
1385                 tty_flip_buffer_push(port->tty);
 
1388         spin_lock_irqsave(&priv->lock, flags);
 
1389         /* control and status byte(s) are also counted */
 
1390         priv->bytes_in += bytes;
 
1391         spin_unlock_irqrestore(&priv->lock, flags);
 
1395         /* Continue trying to always read... unless the port has closed. */
 
1397         if (port->open_count > 0 && priv->comm_is_ok) {
 
1398                 usb_fill_int_urb(port->interrupt_in_urb, port->serial->dev,
 
1399                                 usb_rcvintpipe(port->serial->dev,
 
1400                                         port->interrupt_in_endpointAddress),
 
1401                                 port->interrupt_in_urb->transfer_buffer,
 
1402                                 port->interrupt_in_urb->transfer_buffer_length,
 
1403                                 cypress_read_int_callback, port, priv->read_urb_interval);
 
1404                 result = usb_submit_urb(port->interrupt_in_urb, GFP_ATOMIC);
 
1406                         dev_err(&urb->dev->dev, "%s - failed resubmitting "
 
1407                                         "read urb, error %d\n", __FUNCTION__,
 
1409                         cypress_set_dead(port);
 
1414 } /* cypress_read_int_callback */
 
1417 static void cypress_write_int_callback(struct urb *urb)
 
1419         struct usb_serial_port *port = (struct usb_serial_port *)urb->context;
 
1420         struct cypress_private *priv = usb_get_serial_port_data(port);
 
1423         dbg("%s - port %d", __FUNCTION__, port->number);
 
1425         switch (urb->status) {
 
1432                         /* this urb is terminated, clean up */
 
1433                         dbg("%s - urb shutting down with status: %d", __FUNCTION__, urb->status);
 
1434                         priv->write_urb_in_use = 0;
 
1436                 case -EPIPE: /* no break needed; clear halt and resubmit */
 
1437                         if (!priv->comm_is_ok)
 
1439                         usb_clear_halt(port->serial->dev, 0x02);
 
1440                         /* error in the urb, so we have to resubmit it */
 
1441                         dbg("%s - nonzero write bulk status received: %d", __FUNCTION__, urb->status);
 
1442                         port->interrupt_out_urb->transfer_buffer_length = 1;
 
1443                         port->interrupt_out_urb->dev = port->serial->dev;
 
1444                         result = usb_submit_urb(port->interrupt_out_urb, GFP_ATOMIC);
 
1447                         dev_err(&urb->dev->dev, "%s - failed resubmitting write urb, error %d\n",
 
1448                                 __FUNCTION__, result);
 
1449                         cypress_set_dead(port);
 
1452                         dev_err(&urb->dev->dev,"%s - unexpected nonzero write status received: %d\n",
 
1453                                 __FUNCTION__,urb->status);
 
1454                         cypress_set_dead(port);
 
1458         priv->write_urb_in_use = 0;
 
1460         /* send any buffered data */
 
1465 /*****************************************************************************
 
1466  * Write buffer functions - buffering code from pl2303 used
 
1467  *****************************************************************************/
 
1472  * Allocate a circular buffer and all associated memory.
 
1475 static struct cypress_buf *cypress_buf_alloc(unsigned int size)
 
1478         struct cypress_buf *cb;
 
1484         cb = kmalloc(sizeof(struct cypress_buf), GFP_KERNEL);
 
1488         cb->buf_buf = kmalloc(size, GFP_KERNEL);
 
1489         if (cb->buf_buf == NULL) {
 
1494         cb->buf_size = size;
 
1495         cb->buf_get = cb->buf_put = cb->buf_buf;
 
1505  * Free the buffer and all associated memory.
 
1508 static void cypress_buf_free(struct cypress_buf *cb)
 
1520  * Clear out all data in the circular buffer.
 
1523 static void cypress_buf_clear(struct cypress_buf *cb)
 
1526                 cb->buf_get = cb->buf_put;
 
1527                 /* equivalent to a get of all data available */
 
1532  * cypress_buf_data_avail
 
1534  * Return the number of bytes of data available in the circular
 
1538 static unsigned int cypress_buf_data_avail(struct cypress_buf *cb)
 
1541                 return ((cb->buf_size + cb->buf_put - cb->buf_get) % cb->buf_size);
 
1548  * cypress_buf_space_avail
 
1550  * Return the number of bytes of space available in the circular
 
1554 static unsigned int cypress_buf_space_avail(struct cypress_buf *cb)
 
1557                 return ((cb->buf_size + cb->buf_get - cb->buf_put - 1) % cb->buf_size);
 
1566  * Copy data data from a user buffer and put it into the circular buffer.
 
1567  * Restrict to the amount of space available.
 
1569  * Return the number of bytes copied.
 
1572 static unsigned int cypress_buf_put(struct cypress_buf *cb, const char *buf,
 
1582         len  = cypress_buf_space_avail(cb);
 
1589         len = cb->buf_buf + cb->buf_size - cb->buf_put;
 
1591                 memcpy(cb->buf_put, buf, len);
 
1592                 memcpy(cb->buf_buf, buf+len, count - len);
 
1593                 cb->buf_put = cb->buf_buf + count - len;
 
1595                 memcpy(cb->buf_put, buf, count);
 
1597                         cb->buf_put += count;
 
1598                 else /* count == len */
 
1599                         cb->buf_put = cb->buf_buf;
 
1610  * Get data from the circular buffer and copy to the given buffer.
 
1611  * Restrict to the amount of data available.
 
1613  * Return the number of bytes copied.
 
1616 static unsigned int cypress_buf_get(struct cypress_buf *cb, char *buf,
 
1626         len = cypress_buf_data_avail(cb);
 
1633         len = cb->buf_buf + cb->buf_size - cb->buf_get;
 
1635                 memcpy(buf, cb->buf_get, len);
 
1636                 memcpy(buf+len, cb->buf_buf, count - len);
 
1637                 cb->buf_get = cb->buf_buf + count - len;
 
1639                 memcpy(buf, cb->buf_get, count);
 
1641                         cb->buf_get += count;
 
1642                 else /* count == len */
 
1643                         cb->buf_get = cb->buf_buf;
 
1650 /*****************************************************************************
 
1652  *****************************************************************************/
 
1654 static int __init cypress_init(void)
 
1658         dbg("%s", __FUNCTION__);
 
1660         retval = usb_serial_register(&cypress_earthmate_device);
 
1662                 goto failed_em_register;
 
1663         retval = usb_serial_register(&cypress_hidcom_device);
 
1665                 goto failed_hidcom_register;
 
1666         retval = usb_serial_register(&cypress_ca42v2_device);
 
1668                 goto failed_ca42v2_register;
 
1669         retval = usb_register(&cypress_driver);
 
1671                 goto failed_usb_register;
 
1673         info(DRIVER_DESC " " DRIVER_VERSION);
 
1676 failed_usb_register:
 
1677         usb_serial_deregister(&cypress_ca42v2_device);
 
1678 failed_ca42v2_register:
 
1679         usb_serial_deregister(&cypress_hidcom_device);
 
1680 failed_hidcom_register:
 
1681         usb_serial_deregister(&cypress_earthmate_device);
 
1687 static void __exit cypress_exit (void)
 
1689         dbg("%s", __FUNCTION__);
 
1691         usb_deregister (&cypress_driver);
 
1692         usb_serial_deregister (&cypress_earthmate_device);
 
1693         usb_serial_deregister (&cypress_hidcom_device);
 
1694         usb_serial_deregister (&cypress_ca42v2_device);
 
1698 module_init(cypress_init);
 
1699 module_exit(cypress_exit);
 
1701 MODULE_AUTHOR( DRIVER_AUTHOR );
 
1702 MODULE_DESCRIPTION( DRIVER_DESC );
 
1703 MODULE_VERSION( DRIVER_VERSION );
 
1704 MODULE_LICENSE("GPL");
 
1706 module_param(debug, bool, S_IRUGO | S_IWUSR);
 
1707 MODULE_PARM_DESC(debug, "Debug enabled or not");
 
1708 module_param(stats, bool, S_IRUGO | S_IWUSR);
 
1709 MODULE_PARM_DESC(stats, "Enable statistics or not");
 
1710 module_param(interval, int, S_IRUGO | S_IWUSR);
 
1711 MODULE_PARM_DESC(interval, "Overrides interrupt interval");