2  * IPWireless 3G PCMCIA Network Driver
 
   5  *   by Stephen Blackheath <stephen@blacksapphire.com>,
 
   6  *      Ben Martel <benm@symmetric.co.nz>
 
   8  * Copyrighted as follows:
 
   9  *   Copyright (C) 2004 by Symmetric Systems Ltd (NZ)
 
  11  * Various driver changes and rewrites, port to new kernels
 
  12  *   Copyright (C) 2006-2007 Jiri Kosina
 
  14  * Misc code cleanups and updates
 
  15  *   Copyright (C) 2007 David Sterba
 
  18 #include <linux/init.h>
 
  19 #include <linux/kernel.h>
 
  20 #include <linux/module.h>
 
  21 #include <linux/mutex.h>
 
  22 #include <linux/ppp_defs.h>
 
  24 #include <linux/if_ppp.h>
 
  25 #include <linux/sched.h>
 
  26 #include <linux/serial.h>
 
  27 #include <linux/slab.h>
 
  28 #include <linux/tty.h>
 
  29 #include <linux/tty_driver.h>
 
  30 #include <linux/tty_flip.h>
 
  31 #include <linux/uaccess.h>
 
  32 #include <linux/version.h>
 
  39 #define IPWIRELESS_PCMCIA_START         (0)
 
  40 #define IPWIRELESS_PCMCIA_MINORS        (24)
 
  41 #define IPWIRELESS_PCMCIA_MINOR_RANGE   (8)
 
  43 #define TTYTYPE_MODEM    (0)
 
  44 #define TTYTYPE_MONITOR  (1)
 
  45 #define TTYTYPE_RAS_RAW  (2)
 
  49         struct ipw_hardware *hardware;
 
  50         unsigned int channel_idx;
 
  51         unsigned int secondary_channel_idx;
 
  53         struct ipw_network *network;
 
  54         struct tty_struct *linux_tty;
 
  56         unsigned int control_lines;
 
  57         struct mutex ipw_tty_mutex;
 
  62 static struct ipw_tty *ttys[IPWIRELESS_PCMCIA_MINORS];
 
  64 static struct tty_driver *ipw_tty_driver;
 
  66 static char *tty_type_name(int tty_type)
 
  68         static char *channel_names[] = {
 
  74         return channel_names[tty_type];
 
  77 static void report_registering(struct ipw_tty *tty)
 
  79         char *iftype = tty_type_name(tty->tty_type);
 
  81         printk(KERN_INFO IPWIRELESS_PCCARD_NAME
 
  82                ": registering %s device ttyIPWp%d\n", iftype, tty->index);
 
  85 static void report_deregistering(struct ipw_tty *tty)
 
  87         char *iftype = tty_type_name(tty->tty_type);
 
  89         printk(KERN_INFO IPWIRELESS_PCCARD_NAME
 
  90                ": deregistering %s device ttyIPWp%d\n", iftype,
 
  94 static struct ipw_tty *get_tty(int minor)
 
  96         if (minor < ipw_tty_driver->minor_start
 
  97                         || minor >= ipw_tty_driver->minor_start +
 
  98                         IPWIRELESS_PCMCIA_MINORS)
 
 101                 int minor_offset = minor - ipw_tty_driver->minor_start;
 
 104                  * The 'ras_raw' channel is only available when 'loopback' mode
 
 106                  * Number of minor starts with 16 (_RANGE * _RAS_RAW).
 
 108                 if (!ipwireless_loopback &&
 
 110                                  IPWIRELESS_PCMCIA_MINOR_RANGE * TTYTYPE_RAS_RAW)
 
 113                 return ttys[minor_offset];
 
 117 static int ipw_open(struct tty_struct *linux_tty, struct file *filp)
 
 119         int minor = linux_tty->index;
 
 120         struct ipw_tty *tty = get_tty(minor);
 
 125         mutex_lock(&tty->ipw_tty_mutex);
 
 128                 mutex_unlock(&tty->ipw_tty_mutex);
 
 131         if (tty->open_count == 0)
 
 132                 tty->tx_bytes_queued = 0;
 
 136         tty->linux_tty = linux_tty;
 
 137         linux_tty->driver_data = tty;
 
 138         linux_tty->low_latency = 1;
 
 140         if (tty->tty_type == TTYTYPE_MODEM)
 
 141                 ipwireless_ppp_open(tty->network);
 
 143         mutex_unlock(&tty->ipw_tty_mutex);
 
 148 static void do_ipw_close(struct ipw_tty *tty)
 
 152         if (tty->open_count == 0) {
 
 153                 struct tty_struct *linux_tty = tty->linux_tty;
 
 155                 if (linux_tty != NULL) {
 
 156                         tty->linux_tty = NULL;
 
 157                         linux_tty->driver_data = NULL;
 
 159                         if (tty->tty_type == TTYTYPE_MODEM)
 
 160                                 ipwireless_ppp_close(tty->network);
 
 165 static void ipw_hangup(struct tty_struct *linux_tty)
 
 167         struct ipw_tty *tty = linux_tty->driver_data;
 
 172         mutex_lock(&tty->ipw_tty_mutex);
 
 173         if (tty->open_count == 0) {
 
 174                 mutex_unlock(&tty->ipw_tty_mutex);
 
 180         mutex_unlock(&tty->ipw_tty_mutex);
 
 183 static void ipw_close(struct tty_struct *linux_tty, struct file *filp)
 
 185         ipw_hangup(linux_tty);
 
 188 /* Take data received from hardware, and send it out the tty */
 
 189 void ipwireless_tty_received(struct ipw_tty *tty, unsigned char *data,
 
 192         struct tty_struct *linux_tty;
 
 195         mutex_lock(&tty->ipw_tty_mutex);
 
 196         linux_tty = tty->linux_tty;
 
 197         if (linux_tty == NULL) {
 
 198                 mutex_unlock(&tty->ipw_tty_mutex);
 
 202         if (!tty->open_count) {
 
 203                 mutex_unlock(&tty->ipw_tty_mutex);
 
 206         mutex_unlock(&tty->ipw_tty_mutex);
 
 208         work = tty_insert_flip_string(linux_tty, data, length);
 
 211                 printk(KERN_DEBUG IPWIRELESS_PCCARD_NAME
 
 212                                 ": %d chars not inserted to flip buffer!\n",
 
 216          * This may sleep if ->low_latency is set
 
 219                 tty_flip_buffer_push(linux_tty);
 
 222 static void ipw_write_packet_sent_callback(void *callback_data,
 
 223                                            unsigned int packet_length)
 
 225         struct ipw_tty *tty = callback_data;
 
 228          * Packet has been sent, so we subtract the number of bytes from our
 
 229          * tally of outstanding TX bytes.
 
 231         tty->tx_bytes_queued -= packet_length;
 
 234 static int ipw_write(struct tty_struct *linux_tty,
 
 235                      const unsigned char *buf, int count)
 
 237         struct ipw_tty *tty = linux_tty->driver_data;
 
 243         mutex_lock(&tty->ipw_tty_mutex);
 
 244         if (!tty->open_count) {
 
 245                 mutex_unlock(&tty->ipw_tty_mutex);
 
 249         room = IPWIRELESS_TX_QUEUE_SIZE - tty->tx_bytes_queued;
 
 252         /* Don't allow caller to write any more than we have room for */
 
 257                 mutex_unlock(&tty->ipw_tty_mutex);
 
 261         ret = ipwireless_send_packet(tty->hardware, IPW_CHANNEL_RAS,
 
 262                                (unsigned char *) buf, count,
 
 263                                ipw_write_packet_sent_callback, tty);
 
 265                 mutex_unlock(&tty->ipw_tty_mutex);
 
 269         tty->tx_bytes_queued += count;
 
 270         mutex_unlock(&tty->ipw_tty_mutex);
 
 275 static int ipw_write_room(struct tty_struct *linux_tty)
 
 277         struct ipw_tty *tty = linux_tty->driver_data;
 
 283         if (!tty->open_count)
 
 286         room = IPWIRELESS_TX_QUEUE_SIZE - tty->tx_bytes_queued;
 
 293 static int ipwireless_get_serial_info(struct ipw_tty *tty,
 
 294                                       struct serial_struct __user *retinfo)
 
 296         struct serial_struct tmp;
 
 301         memset(&tmp, 0, sizeof(tmp));
 
 302         tmp.type = PORT_UNKNOWN;
 
 303         tmp.line = tty->index;
 
 307         tmp.baud_base = 115200;
 
 309         tmp.closing_wait = 0;
 
 310         tmp.custom_divisor = 0;
 
 312         if (copy_to_user(retinfo, &tmp, sizeof(*retinfo)))
 
 318 static int ipw_chars_in_buffer(struct tty_struct *linux_tty)
 
 320         struct ipw_tty *tty = linux_tty->driver_data;
 
 325         if (!tty->open_count)
 
 328         return tty->tx_bytes_queued;
 
 331 static int get_control_lines(struct ipw_tty *tty)
 
 333         unsigned int my = tty->control_lines;
 
 334         unsigned int out = 0;
 
 336         if (my & IPW_CONTROL_LINE_RTS)
 
 338         if (my & IPW_CONTROL_LINE_DTR)
 
 340         if (my & IPW_CONTROL_LINE_CTS)
 
 342         if (my & IPW_CONTROL_LINE_DSR)
 
 344         if (my & IPW_CONTROL_LINE_DCD)
 
 350 static int set_control_lines(struct ipw_tty *tty, unsigned int set,
 
 355         if (set & TIOCM_RTS) {
 
 356                 ret = ipwireless_set_RTS(tty->hardware, tty->channel_idx, 1);
 
 359                 if (tty->secondary_channel_idx != -1) {
 
 360                         ret = ipwireless_set_RTS(tty->hardware,
 
 361                                           tty->secondary_channel_idx, 1);
 
 366         if (set & TIOCM_DTR) {
 
 367                 ret = ipwireless_set_DTR(tty->hardware, tty->channel_idx, 1);
 
 370                 if (tty->secondary_channel_idx != -1) {
 
 371                         ret = ipwireless_set_DTR(tty->hardware,
 
 372                                           tty->secondary_channel_idx, 1);
 
 377         if (clear & TIOCM_RTS) {
 
 378                 ret = ipwireless_set_RTS(tty->hardware, tty->channel_idx, 0);
 
 379                 if (tty->secondary_channel_idx != -1) {
 
 380                         ret = ipwireless_set_RTS(tty->hardware,
 
 381                                           tty->secondary_channel_idx, 0);
 
 386         if (clear & TIOCM_DTR) {
 
 387                 ret = ipwireless_set_DTR(tty->hardware, tty->channel_idx, 0);
 
 388                 if (tty->secondary_channel_idx != -1) {
 
 389                         ret = ipwireless_set_DTR(tty->hardware,
 
 390                                           tty->secondary_channel_idx, 0);
 
 398 static int ipw_tiocmget(struct tty_struct *linux_tty, struct file *file)
 
 400         struct ipw_tty *tty = linux_tty->driver_data;
 
 405         if (!tty->open_count)
 
 408         return get_control_lines(tty);
 
 412 ipw_tiocmset(struct tty_struct *linux_tty, struct file *file,
 
 413              unsigned int set, unsigned int clear)
 
 415         struct ipw_tty *tty = linux_tty->driver_data;
 
 420         if (!tty->open_count)
 
 423         return set_control_lines(tty, set, clear);
 
 426 static int ipw_ioctl(struct tty_struct *linux_tty, struct file *file,
 
 427                      unsigned int cmd, unsigned long arg)
 
 429         struct ipw_tty *tty = linux_tty->driver_data;
 
 434         if (!tty->open_count)
 
 439                 return ipwireless_get_serial_info(tty, (void __user *) arg);
 
 442                 return 0;       /* Keeps the PCMCIA scripts happy. */
 
 445         if (tty->tty_type == TTYTYPE_MODEM) {
 
 449                                 int chan = ipwireless_ppp_channel_index(
 
 454                                 if (put_user(chan, (int __user *) arg))
 
 461                                 int unit = ipwireless_ppp_unit_number(
 
 466                                 if (put_user(unit, (int __user *) arg))
 
 473                         return n_tty_ioctl(linux_tty, file, cmd, arg);
 
 476                         return n_tty_ioctl(linux_tty, file, cmd, arg);
 
 482                                 if (put_user(val, (int __user *) arg))
 
 492 static int add_tty(dev_node_t *nodesp, int j,
 
 493                     struct ipw_hardware *hardware,
 
 494                     struct ipw_network *network, int channel_idx,
 
 495                     int secondary_channel_idx, int tty_type)
 
 497         ttys[j] = kzalloc(sizeof(struct ipw_tty), GFP_KERNEL);
 
 501         ttys[j]->hardware = hardware;
 
 502         ttys[j]->channel_idx = channel_idx;
 
 503         ttys[j]->secondary_channel_idx = secondary_channel_idx;
 
 504         ttys[j]->network = network;
 
 505         ttys[j]->tty_type = tty_type;
 
 506         mutex_init(&ttys[j]->ipw_tty_mutex);
 
 508         tty_register_device(ipw_tty_driver, j, NULL);
 
 509         ipwireless_associate_network_tty(network, channel_idx, ttys[j]);
 
 511         if (secondary_channel_idx != -1)
 
 512                 ipwireless_associate_network_tty(network,
 
 513                                                  secondary_channel_idx,
 
 515         if (nodesp != NULL) {
 
 516                 sprintf(nodesp->dev_name, "ttyIPWp%d", j);
 
 517                 nodesp->major = ipw_tty_driver->major;
 
 518                 nodesp->minor = j + ipw_tty_driver->minor_start;
 
 520         if (get_tty(j + ipw_tty_driver->minor_start) == ttys[j])
 
 521                 report_registering(ttys[j]);
 
 525 struct ipw_tty *ipwireless_tty_create(struct ipw_hardware *hardware,
 
 526                                       struct ipw_network *network,
 
 531         for (i = 0; i < IPWIRELESS_PCMCIA_MINOR_RANGE; i++) {
 
 534                 for (j = i; j < IPWIRELESS_PCMCIA_MINORS;
 
 535                                 j += IPWIRELESS_PCMCIA_MINOR_RANGE)
 
 536                         if (ttys[j] != NULL) {
 
 544                         if (add_tty(&nodes[0], j, hardware, network,
 
 545                                         IPW_CHANNEL_DIALLER, IPW_CHANNEL_RAS,
 
 549                         j += IPWIRELESS_PCMCIA_MINOR_RANGE;
 
 550                         if (add_tty(&nodes[1], j, hardware, network,
 
 551                                         IPW_CHANNEL_DIALLER, -1,
 
 555                         j += IPWIRELESS_PCMCIA_MINOR_RANGE;
 
 556                         if (add_tty(NULL, j, hardware, network,
 
 561                         nodes[0].next = &nodes[1];
 
 562                         nodes[1].next = NULL;
 
 571  * Must be called before ipwireless_network_free().
 
 573 void ipwireless_tty_free(struct ipw_tty *tty)
 
 576         struct ipw_network *network = ttys[tty->index]->network;
 
 578         for (j = tty->index; j < IPWIRELESS_PCMCIA_MINORS;
 
 579                         j += IPWIRELESS_PCMCIA_MINOR_RANGE) {
 
 580                 struct ipw_tty *ttyj = ttys[j];
 
 583                         mutex_lock(&ttyj->ipw_tty_mutex);
 
 584                         if (get_tty(j + ipw_tty_driver->minor_start) == ttyj)
 
 585                                 report_deregistering(ttyj);
 
 587                         if (ttyj->linux_tty != NULL) {
 
 588                                 mutex_unlock(&ttyj->ipw_tty_mutex);
 
 589                                 tty_hangup(ttyj->linux_tty);
 
 590                                 /* Wait till the tty_hangup has completed */
 
 591                                 flush_scheduled_work();
 
 592                                 mutex_lock(&ttyj->ipw_tty_mutex);
 
 594                         while (ttyj->open_count)
 
 596                         ipwireless_disassociate_network_ttys(network,
 
 598                         tty_unregister_device(ipw_tty_driver, j);
 
 600                         mutex_unlock(&ttyj->ipw_tty_mutex);
 
 606 static struct tty_operations tty_ops = {
 
 609         .hangup = ipw_hangup,
 
 611         .write_room = ipw_write_room,
 
 613         .chars_in_buffer = ipw_chars_in_buffer,
 
 614         .tiocmget = ipw_tiocmget,
 
 615         .tiocmset = ipw_tiocmset,
 
 618 int ipwireless_tty_init(void)
 
 622         ipw_tty_driver = alloc_tty_driver(IPWIRELESS_PCMCIA_MINORS);
 
 626         ipw_tty_driver->owner = THIS_MODULE;
 
 627         ipw_tty_driver->driver_name = IPWIRELESS_PCCARD_NAME;
 
 628         ipw_tty_driver->name = "ttyIPWp";
 
 629         ipw_tty_driver->major = 0;
 
 630         ipw_tty_driver->minor_start = IPWIRELESS_PCMCIA_START;
 
 631         ipw_tty_driver->type = TTY_DRIVER_TYPE_SERIAL;
 
 632         ipw_tty_driver->subtype = SERIAL_TYPE_NORMAL;
 
 633         ipw_tty_driver->flags = TTY_DRIVER_REAL_RAW | TTY_DRIVER_DYNAMIC_DEV;
 
 634         ipw_tty_driver->init_termios = tty_std_termios;
 
 635         ipw_tty_driver->init_termios.c_cflag =
 
 636             B9600 | CS8 | CREAD | HUPCL | CLOCAL;
 
 637         ipw_tty_driver->init_termios.c_ispeed = 9600;
 
 638         ipw_tty_driver->init_termios.c_ospeed = 9600;
 
 639         tty_set_operations(ipw_tty_driver, &tty_ops);
 
 640         result = tty_register_driver(ipw_tty_driver);
 
 642                 printk(KERN_ERR IPWIRELESS_PCCARD_NAME
 
 643                        ": failed to register tty driver\n");
 
 644                 put_tty_driver(ipw_tty_driver);
 
 651 void ipwireless_tty_release(void)
 
 655         ret = tty_unregister_driver(ipw_tty_driver);
 
 656         put_tty_driver(ipw_tty_driver);
 
 658                 printk(KERN_ERR IPWIRELESS_PCCARD_NAME
 
 659                         ": tty_unregister_driver failed with code %d\n", ret);
 
 662 int ipwireless_tty_is_modem(struct ipw_tty *tty)
 
 664         return tty->tty_type == TTYTYPE_MODEM;
 
 668 ipwireless_tty_notify_control_line_change(struct ipw_tty *tty,
 
 669                                           unsigned int channel_idx,
 
 670                                           unsigned int control_lines,
 
 671                                           unsigned int changed_mask)
 
 673         unsigned int old_control_lines = tty->control_lines;
 
 675         tty->control_lines = (tty->control_lines & ~changed_mask)
 
 676                 | (control_lines & changed_mask);
 
 679          * If DCD is de-asserted, we close the tty so pppd can tell that we
 
 682         if ((old_control_lines & IPW_CONTROL_LINE_DCD)
 
 683                         && !(tty->control_lines & IPW_CONTROL_LINE_DCD)
 
 685                 tty_hangup(tty->linux_tty);