2  *  linux/drivers/char/clps711x.c
 
   4  *  Driver for CLPS711x serial ports
 
   6  *  Based on drivers/char/serial.c, by Linus Torvalds, Theodore Ts'o.
 
   8  *  Copyright 1999 ARM Limited
 
   9  *  Copyright (C) 2000 Deep Blue Solutions Ltd.
 
  11  * This program is free software; you can redistribute it and/or modify
 
  12  * it under the terms of the GNU General Public License as published by
 
  13  * the Free Software Foundation; either version 2 of the License, or
 
  14  * (at your option) any later version.
 
  16  * This program is distributed in the hope that it will be useful,
 
  17  * but WITHOUT ANY WARRANTY; without even the implied warranty of
 
  18  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 
  19  * GNU General Public License for more details.
 
  21  * You should have received a copy of the GNU General Public License
 
  22  * along with this program; if not, write to the Free Software
 
  23  * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
 
  25  *  $Id: clps711x.c,v 1.42 2002/07/28 10:03:28 rmk Exp $
 
  29 #if defined(CONFIG_SERIAL_CLPS711X_CONSOLE) && defined(CONFIG_MAGIC_SYSRQ)
 
  33 #include <linux/module.h>
 
  34 #include <linux/ioport.h>
 
  35 #include <linux/init.h>
 
  36 #include <linux/console.h>
 
  37 #include <linux/sysrq.h>
 
  38 #include <linux/spinlock.h>
 
  39 #include <linux/device.h>
 
  40 #include <linux/tty.h>
 
  41 #include <linux/tty_flip.h>
 
  42 #include <linux/serial_core.h>
 
  43 #include <linux/serial.h>
 
  45 #include <asm/hardware.h>
 
  48 #include <asm/hardware/clps7111.h>
 
  52 #define SERIAL_CLPS711X_MAJOR   204
 
  53 #define SERIAL_CLPS711X_MINOR   40
 
  54 #define SERIAL_CLPS711X_NR      UART_NR
 
  57  * We use the relevant SYSCON register as a base address for these ports.
 
  59 #define UBRLCR(port)            ((port)->iobase + UBRLCR1 - SYSCON1)
 
  60 #define UARTDR(port)            ((port)->iobase + UARTDR1 - SYSCON1)
 
  61 #define SYSFLG(port)            ((port)->iobase + SYSFLG1 - SYSCON1)
 
  62 #define SYSCON(port)            ((port)->iobase + SYSCON1 - SYSCON1)
 
  64 #define TX_IRQ(port)            ((port)->irq)
 
  65 #define RX_IRQ(port)            ((port)->irq + 1)
 
  67 #define UART_ANY_ERR            (UARTDR_FRMERR | UARTDR_PARERR | UARTDR_OVERR)
 
  69 #define tx_enabled(port)        ((port)->unused[0])
 
  71 static void clps711xuart_stop_tx(struct uart_port *port)
 
  73         if (tx_enabled(port)) {
 
  74                 disable_irq(TX_IRQ(port));
 
  79 static void clps711xuart_start_tx(struct uart_port *port)
 
  81         if (!tx_enabled(port)) {
 
  82                 enable_irq(TX_IRQ(port));
 
  87 static void clps711xuart_stop_rx(struct uart_port *port)
 
  89         disable_irq(RX_IRQ(port));
 
  92 static void clps711xuart_enable_ms(struct uart_port *port)
 
  96 static irqreturn_t clps711xuart_int_rx(int irq, void *dev_id)
 
  98         struct uart_port *port = dev_id;
 
  99         struct tty_struct *tty = port->info->tty;
 
 100         unsigned int status, ch, flg;
 
 102         status = clps_readl(SYSFLG(port));
 
 103         while (!(status & SYSFLG_URXFE)) {
 
 104                 ch = clps_readl(UARTDR(port));
 
 111                  * Note that the error handling code is
 
 112                  * out of the main execution path
 
 114                 if (unlikely(ch & UART_ANY_ERR)) {
 
 115                         if (ch & UARTDR_PARERR)
 
 116                                 port->icount.parity++;
 
 117                         else if (ch & UARTDR_FRMERR)
 
 118                                 port->icount.frame++;
 
 119                         if (ch & UARTDR_OVERR)
 
 120                                 port->icount.overrun++;
 
 122                         ch &= port->read_status_mask;
 
 124                         if (ch & UARTDR_PARERR)
 
 126                         else if (ch & UARTDR_FRMERR)
 
 134                 if (uart_handle_sysrq_char(port, ch))
 
 138                  * CHECK: does overrun affect the current character?
 
 139                  * ASSUMPTION: it does not.
 
 141                 uart_insert_char(port, ch, UARTDR_OVERR, ch, flg);
 
 144                 status = clps_readl(SYSFLG(port));
 
 146         tty_flip_buffer_push(tty);
 
 150 static irqreturn_t clps711xuart_int_tx(int irq, void *dev_id)
 
 152         struct uart_port *port = dev_id;
 
 153         struct circ_buf *xmit = &port->info->xmit;
 
 157                 clps_writel(port->x_char, UARTDR(port));
 
 162         if (uart_circ_empty(xmit) || uart_tx_stopped(port)) {
 
 163                 clps711xuart_stop_tx(port);
 
 167         count = port->fifosize >> 1;
 
 169                 clps_writel(xmit->buf[xmit->tail], UARTDR(port));
 
 170                 xmit->tail = (xmit->tail + 1) & (UART_XMIT_SIZE - 1);
 
 172                 if (uart_circ_empty(xmit))
 
 174         } while (--count > 0);
 
 176         if (uart_circ_chars_pending(xmit) < WAKEUP_CHARS)
 
 177                 uart_write_wakeup(port);
 
 179         if (uart_circ_empty(xmit))
 
 180                 clps711xuart_stop_tx(port);
 
 185 static unsigned int clps711xuart_tx_empty(struct uart_port *port)
 
 187         unsigned int status = clps_readl(SYSFLG(port));
 
 188         return status & SYSFLG_UBUSY ? 0 : TIOCSER_TEMT;
 
 191 static unsigned int clps711xuart_get_mctrl(struct uart_port *port)
 
 193         unsigned int port_addr;
 
 194         unsigned int result = 0;
 
 197         port_addr = SYSFLG(port);
 
 198         if (port_addr == SYSFLG1) {
 
 199                 status = clps_readl(SYSFLG1);
 
 200                 if (status & SYSFLG1_DCD)
 
 202                 if (status & SYSFLG1_DSR)
 
 204                 if (status & SYSFLG1_CTS)
 
 212 clps711xuart_set_mctrl_null(struct uart_port *port, unsigned int mctrl)
 
 216 static void clps711xuart_break_ctl(struct uart_port *port, int break_state)
 
 221         spin_lock_irqsave(&port->lock, flags);
 
 222         ubrlcr = clps_readl(UBRLCR(port));
 
 223         if (break_state == -1)
 
 224                 ubrlcr |= UBRLCR_BREAK;
 
 226                 ubrlcr &= ~UBRLCR_BREAK;
 
 227         clps_writel(ubrlcr, UBRLCR(port));
 
 228         spin_unlock_irqrestore(&port->lock, flags);
 
 231 static int clps711xuart_startup(struct uart_port *port)
 
 236         tx_enabled(port) = 1;
 
 241         retval = request_irq(TX_IRQ(port), clps711xuart_int_tx, 0,
 
 242                              "clps711xuart_tx", port);
 
 246         retval = request_irq(RX_IRQ(port), clps711xuart_int_rx, 0,
 
 247                              "clps711xuart_rx", port);
 
 249                 free_irq(TX_IRQ(port), port);
 
 256         syscon = clps_readl(SYSCON(port));
 
 257         syscon |= SYSCON_UARTEN;
 
 258         clps_writel(syscon, SYSCON(port));
 
 263 static void clps711xuart_shutdown(struct uart_port *port)
 
 265         unsigned int ubrlcr, syscon;
 
 270         free_irq(TX_IRQ(port), port);   /* TX interrupt */
 
 271         free_irq(RX_IRQ(port), port);   /* RX interrupt */
 
 276         syscon = clps_readl(SYSCON(port));
 
 277         syscon &= ~SYSCON_UARTEN;
 
 278         clps_writel(syscon, SYSCON(port));
 
 281          * disable break condition and fifos
 
 283         ubrlcr = clps_readl(UBRLCR(port));
 
 284         ubrlcr &= ~(UBRLCR_FIFOEN | UBRLCR_BREAK);
 
 285         clps_writel(ubrlcr, UBRLCR(port));
 
 289 clps711xuart_set_termios(struct uart_port *port, struct termios *termios,
 
 292         unsigned int ubrlcr, baud, quot;
 
 296          * We don't implement CREAD.
 
 298         termios->c_cflag |= CREAD;
 
 301          * Ask the core to calculate the divisor for us.
 
 303         baud = uart_get_baud_rate(port, termios, old, 0, port->uartclk/16); 
 
 304         quot = uart_get_divisor(port, baud);
 
 306         switch (termios->c_cflag & CSIZE) {
 
 308                 ubrlcr = UBRLCR_WRDLEN5;
 
 311                 ubrlcr = UBRLCR_WRDLEN6;
 
 314                 ubrlcr = UBRLCR_WRDLEN7;
 
 317                 ubrlcr = UBRLCR_WRDLEN8;
 
 320         if (termios->c_cflag & CSTOPB)
 
 321                 ubrlcr |= UBRLCR_XSTOP;
 
 322         if (termios->c_cflag & PARENB) {
 
 323                 ubrlcr |= UBRLCR_PRTEN;
 
 324                 if (!(termios->c_cflag & PARODD))
 
 325                         ubrlcr |= UBRLCR_EVENPRT;
 
 327         if (port->fifosize > 1)
 
 328                 ubrlcr |= UBRLCR_FIFOEN;
 
 330         spin_lock_irqsave(&port->lock, flags);
 
 333          * Update the per-port timeout.
 
 335         uart_update_timeout(port, termios->c_cflag, baud);
 
 337         port->read_status_mask = UARTDR_OVERR;
 
 338         if (termios->c_iflag & INPCK)
 
 339                 port->read_status_mask |= UARTDR_PARERR | UARTDR_FRMERR;
 
 342          * Characters to ignore
 
 344         port->ignore_status_mask = 0;
 
 345         if (termios->c_iflag & IGNPAR)
 
 346                 port->ignore_status_mask |= UARTDR_FRMERR | UARTDR_PARERR;
 
 347         if (termios->c_iflag & IGNBRK) {
 
 349                  * If we're ignoring parity and break indicators,
 
 350                  * ignore overruns to (for real raw support).
 
 352                 if (termios->c_iflag & IGNPAR)
 
 353                         port->ignore_status_mask |= UARTDR_OVERR;
 
 358         clps_writel(ubrlcr | quot, UBRLCR(port));
 
 360         spin_unlock_irqrestore(&port->lock, flags);
 
 363 static const char *clps711xuart_type(struct uart_port *port)
 
 365         return port->type == PORT_CLPS711X ? "CLPS711x" : NULL;
 
 369  * Configure/autoconfigure the port.
 
 371 static void clps711xuart_config_port(struct uart_port *port, int flags)
 
 373         if (flags & UART_CONFIG_TYPE)
 
 374                 port->type = PORT_CLPS711X;
 
 377 static void clps711xuart_release_port(struct uart_port *port)
 
 381 static int clps711xuart_request_port(struct uart_port *port)
 
 386 static struct uart_ops clps711x_pops = {
 
 387         .tx_empty       = clps711xuart_tx_empty,
 
 388         .set_mctrl      = clps711xuart_set_mctrl_null,
 
 389         .get_mctrl      = clps711xuart_get_mctrl,
 
 390         .stop_tx        = clps711xuart_stop_tx,
 
 391         .start_tx       = clps711xuart_start_tx,
 
 392         .stop_rx        = clps711xuart_stop_rx,
 
 393         .enable_ms      = clps711xuart_enable_ms,
 
 394         .break_ctl      = clps711xuart_break_ctl,
 
 395         .startup        = clps711xuart_startup,
 
 396         .shutdown       = clps711xuart_shutdown,
 
 397         .set_termios    = clps711xuart_set_termios,
 
 398         .type           = clps711xuart_type,
 
 399         .config_port    = clps711xuart_config_port,
 
 400         .release_port   = clps711xuart_release_port,
 
 401         .request_port   = clps711xuart_request_port,
 
 404 static struct uart_port clps711x_ports[UART_NR] = {
 
 407                 .irq            = IRQ_UTXINT1, /* IRQ_URXINT1, IRQ_UMSINT */
 
 410                 .ops            = &clps711x_pops,
 
 412                 .flags          = UPF_BOOT_AUTOCONF,
 
 416                 .irq            = IRQ_UTXINT2, /* IRQ_URXINT2 */
 
 419                 .ops            = &clps711x_pops,
 
 421                 .flags          = UPF_BOOT_AUTOCONF,
 
 425 #ifdef CONFIG_SERIAL_CLPS711X_CONSOLE
 
 426 static void clps711xuart_console_putchar(struct uart_port *port, int ch)
 
 428         while (clps_readl(SYSFLG(port)) & SYSFLG_UTXFF)
 
 430         clps_writel(ch, UARTDR(port));
 
 434  *      Print a string to the serial port trying not to disturb
 
 435  *      any possible real use of the port...
 
 437  *      The console_lock must be held when we get here.
 
 439  *      Note that this is called with interrupts already disabled
 
 442 clps711xuart_console_write(struct console *co, const char *s,
 
 445         struct uart_port *port = clps711x_ports + co->index;
 
 446         unsigned int status, syscon;
 
 449          *      Ensure that the port is enabled.
 
 451         syscon = clps_readl(SYSCON(port));
 
 452         clps_writel(syscon | SYSCON_UARTEN, SYSCON(port));
 
 454         uart_console_write(port, s, count, clps711xuart_console_putchar);
 
 457          *      Finally, wait for transmitter to become empty
 
 458          *      and restore the uart state.
 
 461                 status = clps_readl(SYSFLG(port));
 
 462         } while (status & SYSFLG_UBUSY);
 
 464         clps_writel(syscon, SYSCON(port));
 
 468 clps711xuart_console_get_options(struct uart_port *port, int *baud,
 
 469                                  int *parity, int *bits)
 
 471         if (clps_readl(SYSCON(port)) & SYSCON_UARTEN) {
 
 472                 unsigned int ubrlcr, quot;
 
 474                 ubrlcr = clps_readl(UBRLCR(port));
 
 477                 if (ubrlcr & UBRLCR_PRTEN) {
 
 478                         if (ubrlcr & UBRLCR_EVENPRT)
 
 484                 if ((ubrlcr & UBRLCR_WRDLEN_MASK) == UBRLCR_WRDLEN7)
 
 489                 quot = ubrlcr & UBRLCR_BAUD_MASK;
 
 490                 *baud = port->uartclk / (16 * (quot + 1));
 
 494 static int __init clps711xuart_console_setup(struct console *co, char *options)
 
 496         struct uart_port *port;
 
 503          * Check whether an invalid uart number has been specified, and
 
 504          * if so, search for the first available port that does have
 
 507         port = uart_get_console(clps711x_ports, UART_NR, co);
 
 510                 uart_parse_options(options, &baud, &parity, &bits, &flow);
 
 512                 clps711xuart_console_get_options(port, &baud, &parity, &bits);
 
 514         return uart_set_options(port, co, baud, parity, bits, flow);
 
 517 static struct uart_driver clps711x_reg;
 
 518 static struct console clps711x_console = {
 
 520         .write          = clps711xuart_console_write,
 
 521         .device         = uart_console_device,
 
 522         .setup          = clps711xuart_console_setup,
 
 523         .flags          = CON_PRINTBUFFER,
 
 525         .data           = &clps711x_reg,
 
 528 static int __init clps711xuart_console_init(void)
 
 530         register_console(&clps711x_console);
 
 533 console_initcall(clps711xuart_console_init);
 
 535 #define CLPS711X_CONSOLE        &clps711x_console
 
 537 #define CLPS711X_CONSOLE        NULL
 
 540 static struct uart_driver clps711x_reg = {
 
 541         .driver_name            = "ttyCL",
 
 543         .major                  = SERIAL_CLPS711X_MAJOR,
 
 544         .minor                  = SERIAL_CLPS711X_MINOR,
 
 547         .cons                   = CLPS711X_CONSOLE,
 
 550 static int __init clps711xuart_init(void)
 
 554         printk(KERN_INFO "Serial: CLPS711x driver $Revision: 1.42 $\n");
 
 556         ret = uart_register_driver(&clps711x_reg);
 
 560         for (i = 0; i < UART_NR; i++)
 
 561                 uart_add_one_port(&clps711x_reg, &clps711x_ports[i]);
 
 566 static void __exit clps711xuart_exit(void)
 
 570         for (i = 0; i < UART_NR; i++)
 
 571                 uart_remove_one_port(&clps711x_reg, &clps711x_ports[i]);
 
 573         uart_unregister_driver(&clps711x_reg);
 
 576 module_init(clps711xuart_init);
 
 577 module_exit(clps711xuart_exit);
 
 579 MODULE_AUTHOR("Deep Blue Solutions Ltd");
 
 580 MODULE_DESCRIPTION("CLPS-711x generic serial driver $Revision: 1.42 $");
 
 581 MODULE_LICENSE("GPL");
 
 582 MODULE_ALIAS_CHARDEV(SERIAL_CLPS711X_MAJOR, SERIAL_CLPS711X_MINOR);