2  *      This program is free software; you can redistribute it and/or
 
   3  *      modify it under the terms of the GNU General Public License
 
   4  *      as published by the Free Software Foundation; either version
 
   5  *      2 of the License, or (at your option) any later version.
 
   7  *      Original driver code supplied by Multi-Tech
 
  10  *      1/9/98  alan@redhat.com         Merge to 2.0.x kernel tree
 
  11  *                                      Obtain and use official major/minors
 
  12  *                                      Loader switched to a misc device
 
  13  *                                      (fixed range check bug as a side effect)
 
  15  *      9/12/98 alan@redhat.com         Rough port to 2.1.x
 
  17  *      10/6/99 sameer                  Merged the ISA and PCI drivers to
 
  18  *                                      a new unified driver.
 
  20  *      3/9/99  sameer                  Added support for ISI4616 cards.
 
  22  *      16/9/99 sameer                  We do not force RTS low anymore.
 
  23  *                                      This is to prevent the firmware
 
  24  *                                      from getting confused.
 
  26  *      26/10/99 sameer                 Cosmetic changes:The driver now
 
  27  *                                      dumps the Port Count information
 
  28  *                                      along with I/O address and IRQ.
 
  30  *      13/12/99 sameer                 Fixed the problem with IRQ sharing.
 
  32  *      10/5/00  sameer                 Fixed isicom_shutdown_board()
 
  33  *                                      to not lower DTR on all the ports
 
  34  *                                      when the last port on the card is
 
  37  *      10/5/00  sameer                 Signal mask setup command added
 
  38  *                                      to  isicom_setup_port and
 
  39  *                                      isicom_shutdown_port.
 
  41  *      24/5/00  sameer                 The driver is now SMP aware.
 
  44  *      27/11/00 Vinayak P Risbud       Fixed the Driver Crash Problem
 
  47  *      03/01/01  anil .s               Added support for resetting the
 
  48  *                                      internal modems on ISI cards.
 
  50  *      08/02/01  anil .s               Upgraded the driver for kernel
 
  53  *      11/04/01  Kevin                 Fixed firmware load problem with
 
  56  *      30/04/01  anil .s               Fixed the remote login through
 
  57  *                                      ISI port problem. Now the link
 
  58  *                                      does not go down before password
 
  61  *      03/05/01  anil .s               Fixed the problem with IRQ sharing
 
  62  *                                      among ISI-PCI cards.
 
  64  *      03/05/01  anil .s               Added support to display the version
 
  65  *                                      info during insmod as well as module
 
  68  *      10/05/01  anil .s               Done the modifications to the source
 
  69  *                                      file and Install script so that the
 
  70  *                                      same installation can be used for
 
  71  *                                      2.2.x and 2.4.x kernel.
 
  73  *      06/06/01  anil .s               Now we drop both dtr and rts during
 
  74  *                                      shutdown_port as well as raise them
 
  75  *                                      during isicom_config_port.
 
  77  *      09/06/01 acme@conectiva.com.br  use capable, not suser, do
 
  78  *                                      restore_flags on failure in
 
  79  *                                      isicom_send_break, verify put_user
 
  82  *      11/02/03  ranjeeth              Added support for 230 Kbps and 460 Kbps
 
  83  *                                      Baud index extended to 21
 
  85  *      20/03/03  ranjeeth              Made to work for Linux Advanced server.
 
  86  *                                      Taken care of license warning.
 
  88  *      10/12/03  Ravindra              Made to work for Fedora Core 1 of
 
  89  *                                      Red Hat Distribution
 
  91  *      06/01/05  Alan Cox              Merged the ISI and base kernel strands
 
  92  *                                      into a single 2.6 driver
 
  94  *      ***********************************************************
 
  96  *      To use this driver you also need the support package. You
 
  97  *      can find this in RPM format on
 
  98  *              ftp://ftp.linux.org.uk/pub/linux/alan
 
 100  *      You can find the original tools for this direct from Multitech
 
 101  *              ftp://ftp.multitech.com/ISI-Cards/
 
 103  *      Having installed the cards the module options (/etc/modprobe.conf)
 
 105  *      options isicom   io=card1,card2,card3,card4 irq=card1,card2,card3,card4
 
 107  *      Omit those entries for boards you don't have installed.
 
 111  *              64-bit verification
 
 114 #include <linux/module.h>
 
 115 #include <linux/firmware.h>
 
 116 #include <linux/kernel.h>
 
 117 #include <linux/tty.h>
 
 118 #include <linux/tty_flip.h>
 
 119 #include <linux/termios.h>
 
 120 #include <linux/fs.h>
 
 121 #include <linux/sched.h>
 
 122 #include <linux/serial.h>
 
 123 #include <linux/mm.h>
 
 124 #include <linux/interrupt.h>
 
 125 #include <linux/timer.h>
 
 126 #include <linux/delay.h>
 
 127 #include <linux/ioport.h>
 
 129 #include <asm/uaccess.h>
 
 131 #include <asm/system.h>
 
 133 #include <linux/pci.h>
 
 135 #include <linux/isicom.h>
 
 137 #define InterruptTheCard(base) outw(0, (base) + 0xc)
 
 138 #define ClearInterrupt(base) inw((base) + 0x0a)
 
 141 #define pr_dbg(str...) printk(KERN_DEBUG "ISICOM: " str)
 
 142 #define isicom_paranoia_check(a, b, c) __isicom_paranoia_check((a), (b), (c))
 
 144 #define pr_dbg(str...) do { } while (0)
 
 145 #define isicom_paranoia_check(a, b, c) 0
 
 148 static int isicom_probe(struct pci_dev *, const struct pci_device_id *);
 
 149 static void __devexit isicom_remove(struct pci_dev *);
 
 151 static struct pci_device_id isicom_pci_tbl[] = {
 
 152         { PCI_DEVICE(VENDOR_ID, 0x2028) },
 
 153         { PCI_DEVICE(VENDOR_ID, 0x2051) },
 
 154         { PCI_DEVICE(VENDOR_ID, 0x2052) },
 
 155         { PCI_DEVICE(VENDOR_ID, 0x2053) },
 
 156         { PCI_DEVICE(VENDOR_ID, 0x2054) },
 
 157         { PCI_DEVICE(VENDOR_ID, 0x2055) },
 
 158         { PCI_DEVICE(VENDOR_ID, 0x2056) },
 
 159         { PCI_DEVICE(VENDOR_ID, 0x2057) },
 
 160         { PCI_DEVICE(VENDOR_ID, 0x2058) },
 
 163 MODULE_DEVICE_TABLE(pci, isicom_pci_tbl);
 
 165 static struct pci_driver isicom_driver = {
 
 167         .id_table       = isicom_pci_tbl,
 
 168         .probe          = isicom_probe,
 
 169         .remove         = __devexit_p(isicom_remove)
 
 172 static int prev_card = 3;       /*      start servicing isi_card[0]     */
 
 173 static struct tty_driver *isicom_normal;
 
 175 static struct timer_list tx;
 
 176 static char re_schedule = 1;
 
 178 static void isicom_tx(unsigned long _data);
 
 179 static void isicom_start(struct tty_struct *tty);
 
 181 /*   baud index mappings from linux defns to isi */
 
 183 static signed char linuxb_to_isib[] = {
 
 184         -1, 0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 11, 13, 15, 16, 17, 18, 19
 
 190         unsigned char           port_count;
 
 191         unsigned short          status;
 
 192         unsigned short          port_status; /* each bit for each port */
 
 193         unsigned short          shift_count;
 
 194         struct isi_port         * ports;
 
 197         spinlock_t              card_lock; /* Card wide lock 11/5/00 -sameer */
 
 202         unsigned short          magic;
 
 210         struct isi_board        * card;
 
 211         struct tty_struct       * tty;
 
 212         wait_queue_head_t       close_wait;
 
 213         wait_queue_head_t       open_wait;
 
 214         struct work_struct      hangup_tq;
 
 215         struct work_struct      bh_tqueue;
 
 216         unsigned char           * xmit_buf;
 
 222 static struct isi_board isi_card[BOARD_COUNT];
 
 223 static struct isi_port  isi_ports[PORT_COUNT];
 
 226  *      Locking functions for card level locking. We need to own both
 
 227  *      the kernel lock for the card and have the card in a position that
 
 231 static int lock_card(struct isi_board *card)
 
 234         unsigned long base = card->base;
 
 236         for (retries = 0; retries < 100; retries++) {
 
 237                 spin_lock_irqsave(&card->card_lock, card->flags);
 
 238                 if (inw(base + 0xe) & 0x1) {
 
 241                         spin_unlock_irqrestore(&card->card_lock, card->flags);
 
 242                         udelay(1000);   /* 1ms */
 
 245         printk(KERN_WARNING "ISICOM: Failed to lock Card (0x%lx)\n",
 
 248         return 0;       /* Failed to acquire the card! */
 
 251 static int lock_card_at_interrupt(struct isi_board *card)
 
 253         unsigned char           retries;
 
 254         unsigned long base = card->base;
 
 256         for (retries = 0; retries < 200; retries++) {
 
 257                 spin_lock_irqsave(&card->card_lock, card->flags);
 
 259                 if (inw(base + 0xe) & 0x1)
 
 262                         spin_unlock_irqrestore(&card->card_lock, card->flags);
 
 264         /* Failing in interrupt is an acceptable event */
 
 265         return 0;       /* Failed to acquire the card! */
 
 268 static void unlock_card(struct isi_board *card)
 
 270         spin_unlock_irqrestore(&card->card_lock, card->flags);
 
 274  *  ISI Card specific ops ...
 
 277 static void raise_dtr(struct isi_port *port)
 
 279         struct isi_board *card = port->card;
 
 280         unsigned long base = card->base;
 
 281         u16 channel = port->channel;
 
 283         if (!lock_card(card))
 
 286         outw(0x8000 | (channel << card->shift_count) | 0x02, base);
 
 288         InterruptTheCard(base);
 
 289         port->status |= ISI_DTR;
 
 293 static inline void drop_dtr(struct isi_port *port)
 
 295         struct isi_board *card = port->card;
 
 296         unsigned long base = card->base;
 
 297         u16 channel = port->channel;
 
 299         if (!lock_card(card))
 
 302         outw(0x8000 | (channel << card->shift_count) | 0x02, base);
 
 304         InterruptTheCard(base);
 
 305         port->status &= ~ISI_DTR;
 
 309 static inline void raise_rts(struct isi_port *port)
 
 311         struct isi_board *card = port->card;
 
 312         unsigned long base = card->base;
 
 313         u16 channel = port->channel;
 
 315         if (!lock_card(card))
 
 318         outw(0x8000 | (channel << card->shift_count) | 0x02, base);
 
 320         InterruptTheCard(base);
 
 321         port->status |= ISI_RTS;
 
 324 static inline void drop_rts(struct isi_port *port)
 
 326         struct isi_board *card = port->card;
 
 327         unsigned long base = card->base;
 
 328         u16 channel = port->channel;
 
 330         if (!lock_card(card))
 
 333         outw(0x8000 | (channel << card->shift_count) | 0x02, base);
 
 335         InterruptTheCard(base);
 
 336         port->status &= ~ISI_RTS;
 
 340 static inline void raise_dtr_rts(struct isi_port *port)
 
 342         struct isi_board *card = port->card;
 
 343         unsigned long base = card->base;
 
 344         u16 channel = port->channel;
 
 346         if (!lock_card(card))
 
 349         outw(0x8000 | (channel << card->shift_count) | 0x02, base);
 
 351         InterruptTheCard(base);
 
 352         port->status |= (ISI_DTR | ISI_RTS);
 
 356 static void drop_dtr_rts(struct isi_port *port)
 
 358         struct isi_board *card = port->card;
 
 359         unsigned long base = card->base;
 
 360         u16 channel = port->channel;
 
 362         if (!lock_card(card))
 
 365         outw(0x8000 | (channel << card->shift_count) | 0x02, base);
 
 367         InterruptTheCard(base);
 
 368         port->status &= ~(ISI_RTS | ISI_DTR);
 
 372 static inline void kill_queue(struct isi_port *port, short queue)
 
 374         struct isi_board *card = port->card;
 
 375         unsigned long base = card->base;
 
 376         u16 channel = port->channel;
 
 378         if (!lock_card(card))
 
 381         outw(0x8000 | (channel << card->shift_count) | 0x02, base);
 
 382         outw((queue << 8) | 0x06, base);
 
 383         InterruptTheCard(base);
 
 388  *      ISICOM Driver specific routines ...
 
 392 static inline int __isicom_paranoia_check(struct isi_port const *port,
 
 393         char *name, const char *routine)
 
 396                 printk(KERN_WARNING "ISICOM: Warning: bad isicom magic for "
 
 397                         "dev %s in %s.\n", name, routine);
 
 400         if (port->magic != ISICOM_MAGIC) {
 
 401                 printk(KERN_WARNING "ISICOM: Warning: NULL isicom port for "
 
 402                         "dev %s in %s.\n", name, routine);
 
 412  *      We shovel data into the card buffers on a regular basis. The card
 
 413  *      will do the rest of the work for us.
 
 416 static void isicom_tx(unsigned long _data)
 
 418         short count = (BOARD_COUNT-1), card, base;
 
 419         short txcount, wrd, residue, word_count, cnt;
 
 420         struct isi_port *port;
 
 421         struct tty_struct *tty;
 
 423         /*      find next active board  */
 
 424         card = (prev_card + 1) & 0x0003;
 
 426                 if (isi_card[card].status & BOARD_ACTIVE)
 
 428                 card = (card + 1) & 0x0003;
 
 430         if (!(isi_card[card].status & BOARD_ACTIVE))
 
 435         count = isi_card[card].port_count;
 
 436         port = isi_card[card].ports;
 
 437         base = isi_card[card].base;
 
 438         for (;count > 0;count--, port++) {
 
 439                 if (!lock_card_at_interrupt(&isi_card[card]))
 
 441                 /* port not active or tx disabled to force flow control */
 
 442                 if (!(port->flags & ASYNC_INITIALIZED) ||
 
 443                                 !(port->status & ISI_TXOK))
 
 444                         unlock_card(&isi_card[card]);
 
 451                         unlock_card(&isi_card[card]);
 
 455                 txcount = min_t(short, TX_SIZE, port->xmit_cnt);
 
 456                 if (txcount <= 0 || tty->stopped || tty->hw_stopped) {
 
 457                         unlock_card(&isi_card[card]);
 
 460                 if (!(inw(base + 0x02) & (1 << port->channel))) {
 
 461                         unlock_card(&isi_card[card]);
 
 464                 pr_dbg("txing %d bytes, port%d.\n", txcount,
 
 466                 outw((port->channel << isi_card[card].shift_count) | txcount,
 
 471                         cnt = min_t(int, txcount, (SERIAL_XMIT_SIZE
 
 473                         if (residue == YES) {
 
 476                                         wrd |= (port->xmit_buf[port->xmit_tail]
 
 478                                         port->xmit_tail = (port->xmit_tail + 1)
 
 479                                                 & (SERIAL_XMIT_SIZE - 1);
 
 490                         word_count = cnt >> 1;
 
 491                         outsw(base, port->xmit_buf+port->xmit_tail,word_count);
 
 492                         port->xmit_tail = (port->xmit_tail
 
 493                                 + (word_count << 1)) & (SERIAL_XMIT_SIZE - 1);
 
 494                         txcount -= (word_count << 1);
 
 495                         port->xmit_cnt -= (word_count << 1);
 
 498                                 wrd = port->xmit_buf[port->xmit_tail];
 
 499                                 port->xmit_tail = (port->xmit_tail + 1)
 
 500                                         & (SERIAL_XMIT_SIZE - 1);
 
 506                 InterruptTheCard(base);
 
 507                 if (port->xmit_cnt <= 0)
 
 508                         port->status &= ~ISI_TXOK;
 
 509                 if (port->xmit_cnt <= WAKEUP_CHARS)
 
 510                         schedule_work(&port->bh_tqueue);
 
 511                 unlock_card(&isi_card[card]);
 
 514         /*      schedule another tx for hopefully in about 10ms */
 
 522         tx.expires = jiffies + HZ/100;
 
 524         tx.function = isicom_tx;
 
 530 /*      Interrupt handlers      */
 
 533 static void isicom_bottomhalf(void *data)
 
 535         struct isi_port *port = (struct isi_port *) data;
 
 536         struct tty_struct *tty = port->tty;
 
 542         wake_up_interruptible(&tty->write_wait);
 
 546  *      Main interrupt handler routine
 
 549 static irqreturn_t isicom_interrupt(int irq, void *dev_id)
 
 551         struct isi_board *card = dev_id;
 
 552         struct isi_port *port;
 
 553         struct tty_struct *tty;
 
 555         u16 header, word_count, count, channel;
 
 559         if (!card || !(card->status & FIRMWARE_LOADED))
 
 563         spin_lock(&card->card_lock);
 
 565         if (card->isa == NO) {
 
 567                  * disable any interrupts from the PCI card and lower the
 
 570                 outw(0x8000, base+0x04);
 
 571                 ClearInterrupt(base);
 
 574         inw(base);              /* get the dummy word out */
 
 576         channel = (header & 0x7800) >> card->shift_count;
 
 577         byte_count = header & 0xff;
 
 579         if (channel + 1 > card->port_count) {
 
 580                 printk(KERN_WARNING "ISICOM: isicom_interrupt(0x%lx): "
 
 581                         "%d(channel) > port_count.\n", base, channel+1);
 
 583                         ClearInterrupt(base);
 
 585                         outw(0x0000, base+0x04); /* enable interrupts */
 
 586                 spin_unlock(&card->card_lock);
 
 589         port = card->ports + channel;
 
 590         if (!(port->flags & ASYNC_INITIALIZED)) {
 
 592                         ClearInterrupt(base);
 
 594                         outw(0x0000, base+0x04); /* enable interrupts */
 
 600                 word_count = byte_count >> 1;
 
 601                 while(byte_count > 1) {
 
 605                 if (byte_count & 0x01)
 
 607                 if (card->isa == YES)
 
 608                         ClearInterrupt(base);
 
 610                         outw(0x0000, base+0x04); /* enable interrupts */
 
 611                 spin_unlock(&card->card_lock);
 
 615         if (header & 0x8000) {          /* Status Packet */
 
 617                 switch(header & 0xff) {
 
 618                 case 0: /* Change in EIA signals */
 
 619                         if (port->flags & ASYNC_CHECK_CD) {
 
 620                                 if (port->status & ISI_DCD) {
 
 621                                         if (!(header & ISI_DCD)) {
 
 622                                         /* Carrier has been lost  */
 
 623                                                 pr_dbg("interrupt: DCD->low.\n"
 
 625                                                 port->status &= ~ISI_DCD;
 
 626                                                 schedule_work(&port->hangup_tq);
 
 628                                 } else if (header & ISI_DCD) {
 
 629                                 /* Carrier has been detected */
 
 630                                         pr_dbg("interrupt: DCD->high.\n");
 
 631                                         port->status |= ISI_DCD;
 
 632                                         wake_up_interruptible(&port->open_wait);
 
 635                                 if (header & ISI_DCD)
 
 636                                         port->status |= ISI_DCD;
 
 638                                         port->status &= ~ISI_DCD;
 
 641                         if (port->flags & ASYNC_CTS_FLOW) {
 
 642                                 if (port->tty->hw_stopped) {
 
 643                                         if (header & ISI_CTS) {
 
 644                                                 port->tty->hw_stopped = 0;
 
 646                                                 port->status |= (ISI_TXOK
 
 648                                                 schedule_work(&port->bh_tqueue);
 
 650                                 } else if (!(header & ISI_CTS)) {
 
 651                                         port->tty->hw_stopped = 1;
 
 653                                         port->status &= ~(ISI_TXOK | ISI_CTS);
 
 656                                 if (header & ISI_CTS)
 
 657                                         port->status |= ISI_CTS;
 
 659                                         port->status &= ~ISI_CTS;
 
 662                         if (header & ISI_DSR)
 
 663                                 port->status |= ISI_DSR;
 
 665                                 port->status &= ~ISI_DSR;
 
 668                                 port->status |= ISI_RI;
 
 670                                 port->status &= ~ISI_RI;
 
 674                 case 1: /* Received Break !!! */
 
 675                         tty_insert_flip_char(tty, 0, TTY_BREAK);
 
 676                         if (port->flags & ASYNC_SAK)
 
 678                         tty_flip_buffer_push(tty);
 
 681                 case 2: /* Statistics            */
 
 682                         pr_dbg("isicom_interrupt: stats!!!.\n");
 
 686                         pr_dbg("Intr: Unknown code in status packet.\n");
 
 689         } else {                                /* Data   Packet */
 
 691                 count = tty_prepare_flip_string(tty, &rp, byte_count & ~1);
 
 692                 pr_dbg("Intr: Can rx %d of %d bytes.\n", count, byte_count);
 
 693                 word_count = count >> 1;
 
 694                 insw(base, rp, word_count);
 
 695                 byte_count -= (word_count << 1);
 
 696                 if (count & 0x0001) {
 
 697                         tty_insert_flip_char(tty,  inw(base) & 0xff,
 
 701                 if (byte_count > 0) {
 
 702                         pr_dbg("Intr(0x%lx:%d): Flip buffer overflow! dropping "
 
 703                                 "bytes...\n", base, channel + 1);
 
 704                         while(byte_count > 0) { /* drain out unread xtra data */
 
 709                 tty_flip_buffer_push(tty);
 
 711         if (card->isa == YES)
 
 712                 ClearInterrupt(base);
 
 714                 outw(0x0000, base+0x04); /* enable interrupts */
 
 719 static void isicom_config_port(struct isi_port *port)
 
 721         struct isi_board *card = port->card;
 
 722         struct tty_struct *tty;
 
 724         unsigned long base = card->base;
 
 725         u16 channel_setup, channel = port->channel,
 
 726                 shift_count = card->shift_count;
 
 727         unsigned char flow_ctrl;
 
 729         if (!(tty = port->tty) || !tty->termios)
 
 732         if (baud & CBAUDEX) {
 
 735                 /*  if CBAUDEX bit is on and the baud is set to either 50 or 75
 
 736                  *  then the card is programmed for 57.6Kbps or 115Kbps
 
 740                 if (baud < 1 || baud > 2)
 
 741                         port->tty->termios->c_cflag &= ~CBAUDEX;
 
 747                 /*  the ASYNC_SPD_HI and ASYNC_SPD_VHI options are set
 
 748                  *  by the set_serial_info ioctl ... this is done by
 
 749                  *  the 'setserial' utility.
 
 752                 if ((port->flags & ASYNC_SPD_MASK) == ASYNC_SPD_HI)
 
 753                         baud++; /*  57.6 Kbps */
 
 754                 if ((port->flags & ASYNC_SPD_MASK) == ASYNC_SPD_VHI)
 
 755                         baud +=2; /*  115  Kbps */
 
 757         if (linuxb_to_isib[baud] == -1) {
 
 765         if (lock_card(card)) {
 
 766                 outw(0x8000 | (channel << shift_count) |0x03, base);
 
 767                 outw(linuxb_to_isib[baud] << 8 | 0x03, base);
 
 769                 switch(C_CSIZE(tty)) {
 
 771                         channel_setup |= ISICOM_CS5;
 
 774                         channel_setup |= ISICOM_CS6;
 
 777                         channel_setup |= ISICOM_CS7;
 
 780                         channel_setup |= ISICOM_CS8;
 
 785                         channel_setup |= ISICOM_2SB;
 
 787                         channel_setup |= ISICOM_EVPAR;
 
 789                                 channel_setup |= ISICOM_ODPAR;
 
 791                 outw(channel_setup, base);
 
 792                 InterruptTheCard(base);
 
 796                 port->flags &= ~ASYNC_CHECK_CD;
 
 798                 port->flags |= ASYNC_CHECK_CD;
 
 800         /* flow control settings ...*/
 
 802         port->flags &= ~ASYNC_CTS_FLOW;
 
 803         if (C_CRTSCTS(tty)) {
 
 804                 port->flags |= ASYNC_CTS_FLOW;
 
 805                 flow_ctrl |= ISICOM_CTSRTS;
 
 808                 flow_ctrl |= ISICOM_RESPOND_XONXOFF;
 
 810                 flow_ctrl |= ISICOM_INITIATE_XONXOFF;
 
 812         if (lock_card(card)) {
 
 813                 outw(0x8000 | (channel << shift_count) |0x04, base);
 
 814                 outw(flow_ctrl << 8 | 0x05, base);
 
 815                 outw((STOP_CHAR(tty)) << 8 | (START_CHAR(tty)), base);
 
 816                 InterruptTheCard(base);
 
 820         /*      rx enabled -> enable port for rx on the card    */
 
 822                 card->port_status |= (1 << channel);
 
 823                 outw(card->port_status, base + 0x02);
 
 829 static inline void isicom_setup_board(struct isi_board *bp)
 
 832         struct isi_port *port;
 
 835         spin_lock_irqsave(&bp->card_lock, flags);
 
 836         if (bp->status & BOARD_ACTIVE) {
 
 837                 spin_unlock_irqrestore(&bp->card_lock, flags);
 
 841         bp->status |= BOARD_ACTIVE;
 
 842         spin_unlock_irqrestore(&bp->card_lock, flags);
 
 843         for (channel = 0; channel < bp->port_count; channel++, port++)
 
 848 static int isicom_setup_port(struct isi_port *port)
 
 850         struct isi_board *card = port->card;
 
 853         if (port->flags & ASYNC_INITIALIZED) {
 
 856         if (!port->xmit_buf) {
 
 859                 if (!(page = get_zeroed_page(GFP_KERNEL)))
 
 862                 if (port->xmit_buf) {
 
 866                 port->xmit_buf = (unsigned char *) page;
 
 869         spin_lock_irqsave(&card->card_lock, flags);
 
 871                 clear_bit(TTY_IO_ERROR, &port->tty->flags);
 
 872         if (port->count == 1)
 
 875         port->xmit_cnt = port->xmit_head = port->xmit_tail = 0;
 
 877         /*      discard any residual data       */
 
 878         kill_queue(port, ISICOM_KILLTX | ISICOM_KILLRX);
 
 880         isicom_config_port(port);
 
 881         port->flags |= ASYNC_INITIALIZED;
 
 882         spin_unlock_irqrestore(&card->card_lock, flags);
 
 887 static int block_til_ready(struct tty_struct *tty, struct file *filp,
 
 888         struct isi_port *port)
 
 890         struct isi_board *card = port->card;
 
 891         int do_clocal = 0, retval;
 
 893         DECLARE_WAITQUEUE(wait, current);
 
 895         /* block if port is in the process of being closed */
 
 897         if (tty_hung_up_p(filp) || port->flags & ASYNC_CLOSING) {
 
 898                 pr_dbg("block_til_ready: close in progress.\n");
 
 899                 interruptible_sleep_on(&port->close_wait);
 
 900                 if (port->flags & ASYNC_HUP_NOTIFY)
 
 906         /* if non-blocking mode is set ... */
 
 908         if ((filp->f_flags & O_NONBLOCK) ||
 
 909                         (tty->flags & (1 << TTY_IO_ERROR))) {
 
 910                 pr_dbg("block_til_ready: non-block mode.\n");
 
 911                 port->flags |= ASYNC_NORMAL_ACTIVE;
 
 918         /* block waiting for DCD to be asserted, and while
 
 919                                                 callout dev is busy */
 
 921         add_wait_queue(&port->open_wait, &wait);
 
 923         spin_lock_irqsave(&card->card_lock, flags);
 
 924         if (!tty_hung_up_p(filp))
 
 926         port->blocked_open++;
 
 927         spin_unlock_irqrestore(&card->card_lock, flags);
 
 932                 set_current_state(TASK_INTERRUPTIBLE);
 
 933                 if (tty_hung_up_p(filp) || !(port->flags & ASYNC_INITIALIZED)) {
 
 934                         if (port->flags & ASYNC_HUP_NOTIFY)
 
 937                                 retval = -ERESTARTSYS;
 
 940                 if (!(port->flags & ASYNC_CLOSING) &&
 
 941                                 (do_clocal || (port->status & ISI_DCD))) {
 
 944                 if (signal_pending(current)) {
 
 945                         retval = -ERESTARTSYS;
 
 950         set_current_state(TASK_RUNNING);
 
 951         remove_wait_queue(&port->open_wait, &wait);
 
 952         spin_lock_irqsave(&card->card_lock, flags);
 
 953         if (!tty_hung_up_p(filp))
 
 955         port->blocked_open--;
 
 956         spin_unlock_irqrestore(&card->card_lock, flags);
 
 959         port->flags |= ASYNC_NORMAL_ACTIVE;
 
 963 static int isicom_open(struct tty_struct *tty, struct file *filp)
 
 965         struct isi_port *port;
 
 966         struct isi_board *card;
 
 967         unsigned int line, board;
 
 971         if (line < 0 || line > PORT_COUNT-1)
 
 974         card = &isi_card[board];
 
 976         if (!(card->status & FIRMWARE_LOADED))
 
 979         /*  open on a port greater than the port count for the card !!! */
 
 980         if (line > ((board * 16) + card->port_count - 1))
 
 983         port = &isi_ports[line];
 
 984         if (isicom_paranoia_check(port, tty->name, "isicom_open"))
 
 987         isicom_setup_board(card);
 
 990         tty->driver_data = port;
 
 992         if ((error = isicom_setup_port(port))!=0)
 
 994         if ((error = block_til_ready(tty, filp, port))!=0)
 
1002 static inline void isicom_shutdown_board(struct isi_board *bp)
 
1004         unsigned long flags;
 
1006         spin_lock_irqsave(&bp->card_lock, flags);
 
1007         if (bp->status & BOARD_ACTIVE) {
 
1008                 bp->status &= ~BOARD_ACTIVE;
 
1010         spin_unlock_irqrestore(&bp->card_lock, flags);
 
1013 static void isicom_shutdown_port(struct isi_port *port)
 
1015         struct isi_board *card = port->card;
 
1016         struct tty_struct *tty;
 
1017         unsigned long flags;
 
1021         spin_lock_irqsave(&card->card_lock, flags);
 
1022         if (!(port->flags & ASYNC_INITIALIZED)) {
 
1023                 spin_unlock_irqrestore(&card->card_lock, flags);
 
1026         if (port->xmit_buf) {
 
1027                 free_page((unsigned long) port->xmit_buf);
 
1028                 port->xmit_buf = NULL;
 
1030         port->flags &= ~ASYNC_INITIALIZED;
 
1031         /* 3rd October 2000 : Vinayak P Risbud */
 
1033         spin_unlock_irqrestore(&card->card_lock, flags);
 
1035         /*Fix done by Anil .S on 30-04-2001
 
1036         remote login through isi port has dtr toggle problem
 
1037         due to which the carrier drops before the password prompt
 
1038         appears on the remote end. Now we drop the dtr only if the
 
1039         HUPCL(Hangup on close) flag is set for the tty*/
 
1042                 /* drop dtr on this port */
 
1045         /* any other port uninits  */
 
1047                 set_bit(TTY_IO_ERROR, &tty->flags);
 
1049         if (--card->count < 0) {
 
1050                 pr_dbg("isicom_shutdown_port: bad board(0x%lx) count %d.\n",
 
1051                         card->base, card->count);
 
1055         /* last port was closed, shutdown that boad too */
 
1058                         isicom_shutdown_board(card);
 
1062 static void isicom_close(struct tty_struct *tty, struct file *filp)
 
1064         struct isi_port *port = tty->driver_data;
 
1065         struct isi_board *card;
 
1066         unsigned long flags;
 
1071         if (isicom_paranoia_check(port, tty->name, "isicom_close"))
 
1074         pr_dbg("Close start!!!.\n");
 
1076         spin_lock_irqsave(&card->card_lock, flags);
 
1077         if (tty_hung_up_p(filp)) {
 
1078                 spin_unlock_irqrestore(&card->card_lock, flags);
 
1082         if (tty->count == 1 && port->count != 1) {
 
1083                 printk(KERN_WARNING "ISICOM:(0x%lx) isicom_close: bad port "
 
1084                         "count tty->count = 1 port count = %d.\n",
 
1085                         card->base, port->count);
 
1088         if (--port->count < 0) {
 
1089                 printk(KERN_WARNING "ISICOM:(0x%lx) isicom_close: bad port "
 
1090                         "count for channel%d = %d", card->base, port->channel,
 
1096                 spin_unlock_irqrestore(&card->card_lock, flags);
 
1099         port->flags |= ASYNC_CLOSING;
 
1101         spin_unlock_irqrestore(&card->card_lock, flags);
 
1103         if (port->closing_wait != ASYNC_CLOSING_WAIT_NONE)
 
1104                 tty_wait_until_sent(tty, port->closing_wait);
 
1105         /* indicate to the card that no more data can be received
 
1107         spin_lock_irqsave(&card->card_lock, flags);
 
1108         if (port->flags & ASYNC_INITIALIZED) {
 
1109                 card->port_status &= ~(1 << port->channel);
 
1110                 outw(card->port_status, card->base + 0x02);
 
1112         isicom_shutdown_port(port);
 
1113         spin_unlock_irqrestore(&card->card_lock, flags);
 
1115         if (tty->driver->flush_buffer)
 
1116                 tty->driver->flush_buffer(tty);
 
1117         tty_ldisc_flush(tty);
 
1119         spin_lock_irqsave(&card->card_lock, flags);
 
1122         if (port->blocked_open) {
 
1123                 spin_unlock_irqrestore(&card->card_lock, flags);
 
1124                 if (port->close_delay) {
 
1125                         pr_dbg("scheduling until time out.\n");
 
1126                         msleep_interruptible(
 
1127                                 jiffies_to_msecs(port->close_delay));
 
1129                 spin_lock_irqsave(&card->card_lock, flags);
 
1130                 wake_up_interruptible(&port->open_wait);
 
1132         port->flags &= ~(ASYNC_NORMAL_ACTIVE | ASYNC_CLOSING);
 
1133         wake_up_interruptible(&port->close_wait);
 
1134         spin_unlock_irqrestore(&card->card_lock, flags);
 
1138 static int isicom_write(struct tty_struct *tty, const unsigned char *buf,
 
1141         struct isi_port *port = tty->driver_data;
 
1142         struct isi_board *card = port->card;
 
1143         unsigned long flags;
 
1146         if (isicom_paranoia_check(port, tty->name, "isicom_write"))
 
1149         if (!port->xmit_buf)
 
1152         spin_lock_irqsave(&card->card_lock, flags);
 
1155                 cnt = min_t(int, count, min(SERIAL_XMIT_SIZE - port->xmit_cnt
 
1156                                 - 1, SERIAL_XMIT_SIZE - port->xmit_head));
 
1160                 memcpy(port->xmit_buf + port->xmit_head, buf, cnt);
 
1161                 port->xmit_head = (port->xmit_head + cnt) & (SERIAL_XMIT_SIZE
 
1163                 port->xmit_cnt += cnt;
 
1168         if (port->xmit_cnt && !tty->stopped && !tty->hw_stopped)
 
1169                 port->status |= ISI_TXOK;
 
1170         spin_unlock_irqrestore(&card->card_lock, flags);
 
1174 /* put_char et all */
 
1175 static void isicom_put_char(struct tty_struct *tty, unsigned char ch)
 
1177         struct isi_port *port = tty->driver_data;
 
1178         struct isi_board *card = port->card;
 
1179         unsigned long flags;
 
1181         if (isicom_paranoia_check(port, tty->name, "isicom_put_char"))
 
1184         if (!port->xmit_buf)
 
1187         spin_lock_irqsave(&card->card_lock, flags);
 
1188         if (port->xmit_cnt >= SERIAL_XMIT_SIZE - 1) {
 
1189                 spin_unlock_irqrestore(&card->card_lock, flags);
 
1193         port->xmit_buf[port->xmit_head++] = ch;
 
1194         port->xmit_head &= (SERIAL_XMIT_SIZE - 1);
 
1196         spin_unlock_irqrestore(&card->card_lock, flags);
 
1199 /* flush_chars et all */
 
1200 static void isicom_flush_chars(struct tty_struct *tty)
 
1202         struct isi_port *port = tty->driver_data;
 
1204         if (isicom_paranoia_check(port, tty->name, "isicom_flush_chars"))
 
1207         if (port->xmit_cnt <= 0 || tty->stopped || tty->hw_stopped ||
 
1211         /* this tells the transmitter to consider this port for
 
1212            data output to the card ... that's the best we can do. */
 
1213         port->status |= ISI_TXOK;
 
1216 /* write_room et all */
 
1217 static int isicom_write_room(struct tty_struct *tty)
 
1219         struct isi_port *port = tty->driver_data;
 
1222         if (isicom_paranoia_check(port, tty->name, "isicom_write_room"))
 
1225         free = SERIAL_XMIT_SIZE - port->xmit_cnt - 1;
 
1231 /* chars_in_buffer et all */
 
1232 static int isicom_chars_in_buffer(struct tty_struct *tty)
 
1234         struct isi_port *port = tty->driver_data;
 
1235         if (isicom_paranoia_check(port, tty->name, "isicom_chars_in_buffer"))
 
1237         return port->xmit_cnt;
 
1241 static inline void isicom_send_break(struct isi_port *port,
 
1242         unsigned long length)
 
1244         struct isi_board *card = port->card;
 
1245         unsigned long base = card->base;
 
1247         if (!lock_card(card))
 
1250         outw(0x8000 | ((port->channel) << (card->shift_count)) | 0x3, base);
 
1251         outw((length & 0xff) << 8 | 0x00, base);
 
1252         outw((length & 0xff00), base);
 
1253         InterruptTheCard(base);
 
1258 static int isicom_tiocmget(struct tty_struct *tty, struct file *file)
 
1260         struct isi_port *port = tty->driver_data;
 
1261         /* just send the port status */
 
1262         u16 status = port->status;
 
1264         if (isicom_paranoia_check(port, tty->name, "isicom_ioctl"))
 
1267         return  ((status & ISI_RTS) ? TIOCM_RTS : 0) |
 
1268                 ((status & ISI_DTR) ? TIOCM_DTR : 0) |
 
1269                 ((status & ISI_DCD) ? TIOCM_CAR : 0) |
 
1270                 ((status & ISI_DSR) ? TIOCM_DSR : 0) |
 
1271                 ((status & ISI_CTS) ? TIOCM_CTS : 0) |
 
1272                 ((status & ISI_RI ) ? TIOCM_RI  : 0);
 
1275 static int isicom_tiocmset(struct tty_struct *tty, struct file *file,
 
1276         unsigned int set, unsigned int clear)
 
1278         struct isi_port *port = tty->driver_data;
 
1280         if (isicom_paranoia_check(port, tty->name, "isicom_ioctl"))
 
1283         if (set & TIOCM_RTS)
 
1285         if (set & TIOCM_DTR)
 
1288         if (clear & TIOCM_RTS)
 
1290         if (clear & TIOCM_DTR)
 
1296 static int isicom_set_serial_info(struct isi_port *port,
 
1297         struct serial_struct __user *info)
 
1299         struct serial_struct newinfo;
 
1302         if (copy_from_user(&newinfo, info, sizeof(newinfo)))
 
1305         reconfig_port = ((port->flags & ASYNC_SPD_MASK) !=
 
1306                 (newinfo.flags & ASYNC_SPD_MASK));
 
1308         if (!capable(CAP_SYS_ADMIN)) {
 
1309                 if ((newinfo.close_delay != port->close_delay) ||
 
1310                                 (newinfo.closing_wait != port->closing_wait) ||
 
1311                                 ((newinfo.flags & ~ASYNC_USR_MASK) !=
 
1312                                 (port->flags & ~ASYNC_USR_MASK)))
 
1314                 port->flags = ((port->flags & ~ ASYNC_USR_MASK) |
 
1315                                 (newinfo.flags & ASYNC_USR_MASK));
 
1318                 port->close_delay = newinfo.close_delay;
 
1319                 port->closing_wait = newinfo.closing_wait;
 
1320                 port->flags = ((port->flags & ~ASYNC_FLAGS) |
 
1321                                 (newinfo.flags & ASYNC_FLAGS));
 
1323         if (reconfig_port) {
 
1324                 isicom_config_port(port);
 
1329 static int isicom_get_serial_info(struct isi_port *port,
 
1330         struct serial_struct __user *info)
 
1332         struct serial_struct out_info;
 
1334         memset(&out_info, 0, sizeof(out_info));
 
1335 /*      out_info.type = ? */
 
1336         out_info.line = port - isi_ports;
 
1337         out_info.port = port->card->base;
 
1338         out_info.irq = port->card->irq;
 
1339         out_info.flags = port->flags;
 
1340 /*      out_info.baud_base = ? */
 
1341         out_info.close_delay = port->close_delay;
 
1342         out_info.closing_wait = port->closing_wait;
 
1343         if (copy_to_user(info, &out_info, sizeof(out_info)))
 
1348 static int isicom_ioctl(struct tty_struct *tty, struct file *filp,
 
1349         unsigned int cmd, unsigned long arg)
 
1351         struct isi_port *port = tty->driver_data;
 
1352         void __user *argp = (void __user *)arg;
 
1355         if (isicom_paranoia_check(port, tty->name, "isicom_ioctl"))
 
1360                 retval = tty_check_change(tty);
 
1363                 tty_wait_until_sent(tty, 0);
 
1365                         isicom_send_break(port, HZ/4);
 
1369                 retval = tty_check_change(tty);
 
1372                 tty_wait_until_sent(tty, 0);
 
1373                 isicom_send_break(port, arg ? arg * (HZ/10) : HZ/4);
 
1377                 return put_user(C_CLOCAL(tty) ? 1 : 0,
 
1378                                 (unsigned long __user *)argp);
 
1381                 if (get_user(arg, (unsigned long __user *) argp))
 
1383                 tty->termios->c_cflag =
 
1384                         ((tty->termios->c_cflag & ~CLOCAL) |
 
1385                         (arg ? CLOCAL : 0));
 
1389                 return isicom_get_serial_info(port, argp);
 
1392                 return isicom_set_serial_info(port, argp);
 
1395                 return -ENOIOCTLCMD;
 
1400 /* set_termios et all */
 
1401 static void isicom_set_termios(struct tty_struct *tty,
 
1402         struct termios *old_termios)
 
1404         struct isi_port *port = tty->driver_data;
 
1406         if (isicom_paranoia_check(port, tty->name, "isicom_set_termios"))
 
1409         if (tty->termios->c_cflag == old_termios->c_cflag &&
 
1410                         tty->termios->c_iflag == old_termios->c_iflag)
 
1413         isicom_config_port(port);
 
1415         if ((old_termios->c_cflag & CRTSCTS) &&
 
1416                         !(tty->termios->c_cflag & CRTSCTS)) {
 
1417                 tty->hw_stopped = 0;
 
1422 /* throttle et all */
 
1423 static void isicom_throttle(struct tty_struct *tty)
 
1425         struct isi_port *port = tty->driver_data;
 
1426         struct isi_board *card = port->card;
 
1428         if (isicom_paranoia_check(port, tty->name, "isicom_throttle"))
 
1431         /* tell the card that this port cannot handle any more data for now */
 
1432         card->port_status &= ~(1 << port->channel);
 
1433         outw(card->port_status, card->base + 0x02);
 
1436 /* unthrottle et all */
 
1437 static void isicom_unthrottle(struct tty_struct *tty)
 
1439         struct isi_port *port = tty->driver_data;
 
1440         struct isi_board *card = port->card;
 
1442         if (isicom_paranoia_check(port, tty->name, "isicom_unthrottle"))
 
1445         /* tell the card that this port is ready to accept more data */
 
1446         card->port_status |= (1 << port->channel);
 
1447         outw(card->port_status, card->base + 0x02);
 
1451 static void isicom_stop(struct tty_struct *tty)
 
1453         struct isi_port *port = tty->driver_data;
 
1455         if (isicom_paranoia_check(port, tty->name, "isicom_stop"))
 
1458         /* this tells the transmitter not to consider this port for
 
1459            data output to the card. */
 
1460         port->status &= ~ISI_TXOK;
 
1464 static void isicom_start(struct tty_struct *tty)
 
1466         struct isi_port *port = tty->driver_data;
 
1468         if (isicom_paranoia_check(port, tty->name, "isicom_start"))
 
1471         /* this tells the transmitter to consider this port for
 
1472            data output to the card. */
 
1473         port->status |= ISI_TXOK;
 
1477 static void do_isicom_hangup(void *data)
 
1479         struct isi_port *port = data;
 
1480         struct tty_struct *tty;
 
1487 static void isicom_hangup(struct tty_struct *tty)
 
1489         struct isi_port *port = tty->driver_data;
 
1491         if (isicom_paranoia_check(port, tty->name, "isicom_hangup"))
 
1494         isicom_shutdown_port(port);
 
1496         port->flags &= ~ASYNC_NORMAL_ACTIVE;
 
1498         wake_up_interruptible(&port->open_wait);
 
1501 /* flush_buffer et all */
 
1502 static void isicom_flush_buffer(struct tty_struct *tty)
 
1504         struct isi_port *port = tty->driver_data;
 
1505         struct isi_board *card = port->card;
 
1506         unsigned long flags;
 
1508         if (isicom_paranoia_check(port, tty->name, "isicom_flush_buffer"))
 
1511         spin_lock_irqsave(&card->card_lock, flags);
 
1512         port->xmit_cnt = port->xmit_head = port->xmit_tail = 0;
 
1513         spin_unlock_irqrestore(&card->card_lock, flags);
 
1515         wake_up_interruptible(&tty->write_wait);
 
1520  * Driver init and deinit functions
 
1523 static int __devinit isicom_register_ioregion(struct pci_dev *pdev,
 
1524         const unsigned int index)
 
1526         struct isi_board *board = pci_get_drvdata(pdev);
 
1531         if (!request_region(board->base, 16, ISICOM_NAME)) {
 
1532                 dev_dbg(&pdev->dev, "I/O Region 0x%lx-0x%lx is busy. Card%d "
 
1533                         "will be disabled.\n", board->base, board->base + 15,
 
1541 static void isicom_unregister_ioregion(struct pci_dev *pdev)
 
1543         struct isi_board *board = pci_get_drvdata(pdev);
 
1548         release_region(board->base, 16);
 
1549         dev_dbg(&pdev->dev, "I/O Region 0x%lx-0x%lx released.\n",
 
1550                 board->base, board->base + 15);
 
1554 static const struct tty_operations isicom_ops = {
 
1555         .open                   = isicom_open,
 
1556         .close                  = isicom_close,
 
1557         .write                  = isicom_write,
 
1558         .put_char               = isicom_put_char,
 
1559         .flush_chars            = isicom_flush_chars,
 
1560         .write_room             = isicom_write_room,
 
1561         .chars_in_buffer        = isicom_chars_in_buffer,
 
1562         .ioctl                  = isicom_ioctl,
 
1563         .set_termios            = isicom_set_termios,
 
1564         .throttle               = isicom_throttle,
 
1565         .unthrottle             = isicom_unthrottle,
 
1566         .stop                   = isicom_stop,
 
1567         .start                  = isicom_start,
 
1568         .hangup                 = isicom_hangup,
 
1569         .flush_buffer           = isicom_flush_buffer,
 
1570         .tiocmget               = isicom_tiocmget,
 
1571         .tiocmset               = isicom_tiocmset,
 
1574 static int __devinit isicom_register_tty_driver(void)
 
1576         int error = -ENOMEM;
 
1578         /* tty driver structure initialization */
 
1579         isicom_normal = alloc_tty_driver(PORT_COUNT);
 
1583         isicom_normal->owner                    = THIS_MODULE;
 
1584         isicom_normal->name                     = "ttyM";
 
1585         isicom_normal->major                    = ISICOM_NMAJOR;
 
1586         isicom_normal->minor_start              = 0;
 
1587         isicom_normal->type                     = TTY_DRIVER_TYPE_SERIAL;
 
1588         isicom_normal->subtype                  = SERIAL_TYPE_NORMAL;
 
1589         isicom_normal->init_termios             = tty_std_termios;
 
1590         isicom_normal->init_termios.c_cflag     = B9600 | CS8 | CREAD | HUPCL |
 
1592         isicom_normal->flags                    = TTY_DRIVER_REAL_RAW;
 
1593         tty_set_operations(isicom_normal, &isicom_ops);
 
1595         if ((error = tty_register_driver(isicom_normal))) {
 
1596                 pr_dbg("Couldn't register the dialin driver, error=%d\n",
 
1598                 put_tty_driver(isicom_normal);
 
1604 static void isicom_unregister_tty_driver(void)
 
1608         if ((error = tty_unregister_driver(isicom_normal)))
 
1609                 pr_dbg("couldn't unregister normal driver, error=%d.\n", error);
 
1611         put_tty_driver(isicom_normal);
 
1614 static int __devinit isicom_register_isr(struct pci_dev *pdev,
 
1615         const unsigned int index)
 
1617         struct isi_board *board = pci_get_drvdata(pdev);
 
1618         unsigned long irqflags = IRQF_DISABLED;
 
1619         int retval = -EINVAL;
 
1624         if (board->isa == NO)
 
1625                 irqflags |= IRQF_SHARED;
 
1627         retval = request_irq(board->irq, isicom_interrupt, irqflags,
 
1628                 ISICOM_NAME, board);
 
1630                 dev_warn(&pdev->dev, "Could not install handler at Irq %d. "
 
1631                         "Card%d will be disabled.\n", board->irq, index + 1);
 
1638 static int __devinit reset_card(struct pci_dev *pdev,
 
1639         const unsigned int card, unsigned int *signature)
 
1641         struct isi_board *board = pci_get_drvdata(pdev);
 
1642         unsigned long base = board->base;
 
1643         unsigned int portcount = 0;
 
1646         dev_dbg(&pdev->dev, "ISILoad:Resetting Card%d at 0x%lx\n", card + 1,
 
1653         outw(0, base + 0x8); /* Reset */
 
1657         *signature = inw(base + 0x4) & 0xff;
 
1659         if (board->isa == YES) {
 
1660                 if (!(inw(base + 0xe) & 0x1) || (inw(base + 0x2))) {
 
1661                         dev_dbg(&pdev->dev, "base+0x2=0x%lx, base+0xe=0x%lx\n",
 
1662                                 inw(base + 0x2), inw(base + 0xe));
 
1663                         dev_err(&pdev->dev, "ISILoad:ISA Card%d reset failure "
 
1664                                 "(Possible bad I/O Port Address 0x%lx).\n",
 
1670                 portcount = inw(base + 0x2);
 
1671                 if (!(inw(base + 0xe) & 0x1) || ((portcount != 0) &&
 
1672                                 (portcount != 4) && (portcount != 8))) {
 
1673                         dev_dbg(&pdev->dev, "base+0x2=0x%lx, base+0xe=0x%lx\n",
 
1674                                 inw(base + 0x2), inw(base + 0xe));
 
1675                         dev_err(&pdev->dev, "ISILoad:PCI Card%d reset failure "
 
1676                                 "(Possible bad I/O Port Address 0x%lx).\n",
 
1683         switch (*signature) {
 
1687                 board->port_count = (board->isa == NO && portcount == 4) ? 4 :
 
1689                 board->shift_count = 12;
 
1692                 board->port_count = 16;
 
1693                 board->shift_count = 11;
 
1696                 dev_warn(&pdev->dev, "ISILoad:Card%d reset failure (Possible "
 
1697                         "bad I/O Port Address 0x%lx).\n", card + 1, base);
 
1698                 dev_dbg(&pdev->dev, "Sig=0x%lx\n", signature);
 
1701         dev_info(&pdev->dev, "-Done\n");
 
1707 static inline int WaitTillCardIsFree(u16 base)
 
1709         unsigned long count = 0;
 
1711         while (!(inw(base + 0xe) & 0x1) && count++ < 100)
 
1714         return !(inw(base + 0xe) & 0x1);
 
1717 static int __devinit load_firmware(struct pci_dev *pdev,
 
1718         const unsigned int index, const unsigned int signature)
 
1720         struct isi_board *board = pci_get_drvdata(pdev);
 
1721         const struct firmware *fw;
 
1722         unsigned long base = board->base;
 
1724         u16 word_count, status;
 
1735         switch (signature) {
 
1737                 name = "isi608.bin";
 
1740                 name = "isi608em.bin";
 
1743                 name = "isi616em.bin";
 
1746                 name = "isi4608.bin";
 
1749                 name = "isi4616.bin";
 
1752                 dev_err(&pdev->dev, "Unknown signature.\n");
 
1756         retval = request_firmware(&fw, name, &pdev->dev);
 
1762         for (frame = (struct stframe *)fw->data;
 
1763                         frame < (struct stframe *)(fw->data + fw->size);
 
1764                         frame = (struct stframe *)((u8 *)(frame + 1) +
 
1766                 if (WaitTillCardIsFree(base))
 
1769                 outw(0xf0, base);       /* start upload sequence */
 
1771                 outw(frame->addr, base); /* lsb of address */
 
1773                 word_count = frame->count / 2 + frame->count % 2;
 
1774                 outw(word_count, base);
 
1775                 InterruptTheCard(base);
 
1777                 udelay(100); /* 0x2f */
 
1779                 if (WaitTillCardIsFree(base))
 
1782                 if ((status = inw(base + 0x4)) != 0) {
 
1783                         dev_warn(&pdev->dev, "Card%d rejected load header:\n"
 
1784                                 "Address:0x%x\nCount:0x%x\nStatus:0x%x\n",
 
1785                                 index + 1, frame->addr, frame->count, status);
 
1788                 outsw(base, frame->data, word_count);
 
1790                 InterruptTheCard(base);
 
1792                 udelay(50); /* 0x0f */
 
1794                 if (WaitTillCardIsFree(base))
 
1797                 if ((status = inw(base + 0x4)) != 0) {
 
1798                         dev_err(&pdev->dev, "Card%d got out of sync.Card "
 
1799                                 "Status:0x%x\n", index + 1, status);
 
1804 /* XXX: should we test it by reading it back and comparing with original like
 
1805  * in load firmware package? */
 
1806         for (frame = (struct stframe *)fw->data;
 
1807                         frame < (struct stframe *)(fw->data + fw->size);
 
1808                         frame = (struct stframe *)((u8 *)(frame + 1) +
 
1810                 if (WaitTillCardIsFree(base))
 
1813                 outw(0xf1, base); /* start download sequence */
 
1815                 outw(frame->addr, base); /* lsb of address */
 
1817                 word_count = (frame->count >> 1) + frame->count % 2;
 
1818                 outw(word_count + 1, base);
 
1819                 InterruptTheCard(base);
 
1821                 udelay(50); /* 0xf */
 
1823                 if (WaitTillCardIsFree(base))
 
1826                 if ((status = inw(base + 0x4)) != 0) {
 
1827                         dev_warn(&pdev->dev, "Card%d rejected verify header:\n"
 
1828                                 "Address:0x%x\nCount:0x%x\nStatus: 0x%x\n",
 
1829                                 index + 1, frame->addr, frame->count, status);
 
1833                 data = kmalloc(word_count * 2, GFP_KERNEL);
 
1835                 insw(base, data, word_count);
 
1836                 InterruptTheCard(base);
 
1838                 for (a = 0; a < frame->count; a++)
 
1839                         if (data[a] != frame->data[a]) {
 
1841                                 dev_err(&pdev->dev, "Card%d, firmware upload "
 
1842                                         "failed\n", index + 1);
 
1847                 udelay(50); /* 0xf */
 
1849                 if (WaitTillCardIsFree(base))
 
1852                 if ((status = inw(base + 0x4)) != 0) {
 
1853                         dev_err(&pdev->dev, "Card%d verify got out of sync. "
 
1854                                 "Card Status:0x%x\n", index + 1, status);
 
1860         if (WaitTillCardIsFree(base))
 
1867         InterruptTheCard(base);
 
1868         outw(0x0, base + 0x4); /* for ISI4608 cards */
 
1870         board->status |= FIRMWARE_LOADED;
 
1874         release_firmware(fw);
 
1880  *      Insmod can set static symbols so keep these static
 
1886 static int __devinit isicom_probe(struct pci_dev *pdev,
 
1887         const struct pci_device_id *ent)
 
1889         unsigned int ioaddr, signature, index;
 
1890         int retval = -EPERM;
 
1892         struct isi_board *board = NULL;
 
1894         if (card >= BOARD_COUNT)
 
1897         ioaddr = pci_resource_start(pdev, 3);
 
1898         /* i.e at offset 0x1c in the PCI configuration register space. */
 
1900         dev_info(&pdev->dev, "ISI PCI Card(Device ID 0x%x)\n", ent->device);
 
1902         /* allot the first empty slot in the array */
 
1903         for (index = 0; index < BOARD_COUNT; index++)
 
1904                 if (isi_card[index].base == 0) {
 
1905                         board = &isi_card[index];
 
1909         board->base = ioaddr;
 
1910         board->irq = pciirq;
 
1914         pci_set_drvdata(pdev, board);
 
1916         retval = isicom_register_ioregion(pdev, index);
 
1920         retval = isicom_register_isr(pdev, index);
 
1924         retval = reset_card(pdev, index, &signature);
 
1928         retval = load_firmware(pdev, index, signature);
 
1935         free_irq(board->irq, board);
 
1937         isicom_unregister_ioregion(pdev);
 
1943 static void __devexit isicom_remove(struct pci_dev *pdev)
 
1945         struct isi_board *board = pci_get_drvdata(pdev);
 
1947         free_irq(board->irq, board);
 
1948         isicom_unregister_ioregion(pdev);
 
1951 static int __devinit isicom_setup(void)
 
1953         int retval, idx, channel;
 
1954         struct isi_port *port;
 
1957         memset(isi_ports, 0, sizeof(isi_ports));
 
1959         for(idx = 0; idx < BOARD_COUNT; idx++) {
 
1960                 port = &isi_ports[idx * 16];
 
1961                 isi_card[idx].ports = port;
 
1962                 spin_lock_init(&isi_card[idx].card_lock);
 
1963                 for (channel = 0; channel < 16; channel++, port++) {
 
1964                         port->magic = ISICOM_MAGIC;
 
1965                         port->card = &isi_card[idx];
 
1966                         port->channel = channel;
 
1967                         port->close_delay = 50 * HZ/100;
 
1968                         port->closing_wait = 3000 * HZ/100;
 
1969                         INIT_WORK(&port->hangup_tq, do_isicom_hangup, port);
 
1970                         INIT_WORK(&port->bh_tqueue, isicom_bottomhalf, port);
 
1972                         init_waitqueue_head(&port->open_wait);
 
1973                         init_waitqueue_head(&port->close_wait);
 
1976                 isi_card[idx].base = 0;
 
1977                 isi_card[idx].irq = 0;
 
1982                 if (irq[idx] == 2 || irq[idx] == 3 || irq[idx] == 4     ||
 
1983                                 irq[idx] == 5   || irq[idx] == 7        ||
 
1984                                 irq[idx] == 10  || irq[idx] == 11       ||
 
1985                                 irq[idx] == 12  || irq[idx] == 15) {
 
1986                         printk(KERN_ERR "ISICOM: ISA not supported yet.\n");
 
1990                         printk(KERN_ERR "ISICOM: Irq %d unsupported. "
 
1991                                 "Disabling Card%d...\n", irq[idx], idx + 1);
 
1994         retval = isicom_register_tty_driver();
 
1998         retval = pci_register_driver(&isicom_driver);
 
2000                 printk(KERN_ERR "ISICOM: Unable to register pci driver.\n");
 
2005         tx.expires = jiffies + 1;
 
2007         tx.function = isicom_tx;
 
2013         isicom_unregister_tty_driver();
 
2018 static void __exit isicom_exit(void)
 
2020         unsigned int index = 0;
 
2024         while (re_schedule != 2 && index++ < 100)
 
2027         pci_unregister_driver(&isicom_driver);
 
2028         isicom_unregister_tty_driver();
 
2031 module_init(isicom_setup);
 
2032 module_exit(isicom_exit);
 
2034 MODULE_AUTHOR("MultiTech");
 
2035 MODULE_DESCRIPTION("Driver for the ISI series of cards by MultiTech");
 
2036 MODULE_LICENSE("GPL");
 
2037 module_param_array(io, int, NULL, 0);
 
2038 MODULE_PARM_DESC(io, "I/O ports for the cards");
 
2039 module_param_array(irq, int, NULL, 0);
 
2040 MODULE_PARM_DESC(irq, "Interrupts for the cards");