📄 sa1100.c
字号:
/* * linux/drivers/char/serial_sa1100.c * * Driver for SA11x0 serial ports * * Based on drivers/char/serial.c, by Linus Torvalds, Theodore Ts'o. * * Copyright (C) 2000 Deep Blue Solutions Ltd. * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 2 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program; if not, write to the Free Software * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA * * $Id: sa1100.c,v 1.14.2.4 2002/10/24 09:53:25 rmk Exp $ * */#include <linux/config.h>#include <linux/module.h>#include <linux/errno.h>#include <linux/signal.h>#include <linux/sched.h>#include <linux/interrupt.h>#include <linux/tty.h>#include <linux/tty_flip.h>#include <linux/major.h>#include <linux/string.h>#include <linux/fcntl.h>#include <linux/ptrace.h>#include <linux/ioport.h>#include <linux/mm.h>#include <linux/slab.h>#include <linux/init.h>#include <linux/circ_buf.h>#include <linux/serial.h>#include <linux/console.h>#include <linux/sysrq.h>#include <asm/system.h>#include <asm/io.h>#include <asm/irq.h>#include <asm/uaccess.h>#include <asm/bitops.h>#include <asm/hardware.h>#include <asm/mach/serial_sa1100.h>#if defined(CONFIG_SERIAL_SA1100_CONSOLE) && defined(CONFIG_MAGIC_SYSRQ)#define SUPPORT_SYSRQ#endif#include <linux/serial_core.h>/* We've been assigned a range on the "Low-density serial ports" major */#define SERIAL_SA1100_MAJOR 204#define CALLOUT_SA1100_MAJOR 205#define MINOR_START 5#define NR_PORTS 3#define SA1100_ISR_PASS_LIMIT 256/* * Convert from ignore_status_mask or read_status_mask to UTSR[01] */#define SM_TO_UTSR0(x) ((x) & 0xff)#define SM_TO_UTSR1(x) ((x) >> 8)#define UTSR0_TO_SM(x) ((x))#define UTSR1_TO_SM(x) ((x) << 8)#define UART_GET_UTCR0(port) __raw_readl((port)->membase + UTCR0)#define UART_GET_UTCR1(port) __raw_readl((port)->membase + UTCR1)#define UART_GET_UTCR2(port) __raw_readl((port)->membase + UTCR2)#define UART_GET_UTCR3(port) __raw_readl((port)->membase + UTCR3)#define UART_GET_UTSR0(port) __raw_readl((port)->membase + UTSR0)#define UART_GET_UTSR1(port) __raw_readl((port)->membase + UTSR1)#define UART_GET_CHAR(port) __raw_readl((port)->membase + UTDR)#define UART_PUT_UTCR0(port,v) __raw_writel((v),(port)->membase + UTCR0)#define UART_PUT_UTCR1(port,v) __raw_writel((v),(port)->membase + UTCR1)#define UART_PUT_UTCR2(port,v) __raw_writel((v),(port)->membase + UTCR2)#define UART_PUT_UTCR3(port,v) __raw_writel((v),(port)->membase + UTCR3)#define UART_PUT_UTSR0(port,v) __raw_writel((v),(port)->membase + UTSR0)#define UART_PUT_UTSR1(port,v) __raw_writel((v),(port)->membase + UTSR1)#define UART_PUT_CHAR(port,v) __raw_writel((v),(port)->membase + UTDR)/* * This is the size of our serial port register set. */#define UART_PORT_SIZE 0x24static struct tty_driver normal, callout;static struct tty_struct *sa1100_table[NR_PORTS];static struct termios *sa1100_termios[NR_PORTS], *sa1100_termios_locked[NR_PORTS];static int (*sa1100_open)(struct uart_port *, struct uart_info *);static void (*sa1100_close)(struct uart_port *, struct uart_info *);#ifdef SUPPORT_SYSRQstatic struct console sa1100_console;#endif/* * interrupts disabled on entry */static void sa1100_stop_tx(struct uart_port *port, u_int from_tty){ u32 utcr3 = UART_GET_UTCR3(port); UART_PUT_UTCR3(port, utcr3 & ~UTCR3_TIE); port->read_status_mask &= ~UTSR0_TO_SM(UTSR0_TFS);}/* * interrupts may not be disabled on entry */static void sa1100_start_tx(struct uart_port *port, u_int nonempty, u_int from_tty){ if (nonempty) { unsigned long flags; u32 utcr3; local_irq_save(flags); utcr3 = UART_GET_UTCR3(port); port->read_status_mask |= UTSR0_TO_SM(UTSR0_TFS); UART_PUT_UTCR3(port, utcr3 | UTCR3_TIE); local_irq_restore(flags); }}/* * Interrupts enabled */static void sa1100_stop_rx(struct uart_port *port){ u32 utcr3 = UART_GET_UTCR3(port); UART_PUT_UTCR3(port, utcr3 & ~UTCR3_RIE);}/* * No modem control lines */static void sa1100_enable_ms(struct uart_port *port){}static voidsa1100_rx_chars(struct uart_info *info, struct pt_regs *regs){ struct tty_struct *tty = info->tty; unsigned int status, ch, flg, ignored = 0; struct uart_port *port = info->port; status = UTSR1_TO_SM(UART_GET_UTSR1(port)) | UTSR0_TO_SM(UART_GET_UTSR0(port)); while (status & UTSR1_TO_SM(UTSR1_RNE)) { ch = UART_GET_CHAR(port); if (tty->flip.count >= TTY_FLIPBUF_SIZE) goto ignore_char; port->icount.rx++; flg = TTY_NORMAL; /* * note that the error handling code is * out of the main execution path */ if (status & UTSR1_TO_SM(UTSR1_PRE | UTSR1_FRE | UTSR1_ROR)) goto handle_error; if (uart_handle_sysrq_char(info, ch, regs)) goto ignore_char; error_return: *tty->flip.flag_buf_ptr++ = flg; *tty->flip.char_buf_ptr++ = ch; tty->flip.count++; ignore_char: status = UTSR1_TO_SM(UART_GET_UTSR1(port)) | UTSR0_TO_SM(UART_GET_UTSR0(port)); }out: tty_flip_buffer_push(tty); return;handle_error: if (status & UTSR1_TO_SM(UTSR1_PRE)) port->icount.parity++; else if (status & UTSR1_TO_SM(UTSR1_FRE)) port->icount.frame++; if (status & UTSR1_TO_SM(UTSR1_ROR)) port->icount.overrun++; if (status & port->ignore_status_mask) { if (++ignored > 100) goto out; goto ignore_char; } status &= port->read_status_mask; if (status & UTSR1_TO_SM(UTSR1_PRE)) flg = TTY_PARITY; else if (status & UTSR1_TO_SM(UTSR1_FRE)) flg = TTY_FRAME; if (status & UTSR1_TO_SM(UTSR1_ROR)) { /* * overrun does *not* affect the character * we read from the FIFO */ *tty->flip.flag_buf_ptr++ = flg; *tty->flip.char_buf_ptr++ = ch; tty->flip.count++; if (tty->flip.count >= TTY_FLIPBUF_SIZE) goto ignore_char; ch = 0; flg = TTY_OVERRUN; }#ifdef SUPPORT_SYSRQ info->sysrq = 0;#endif goto error_return;}static void sa1100_tx_chars(struct uart_info *info){ struct uart_port *port = info->port; if (port->x_char) { UART_PUT_CHAR(port, port->x_char); port->icount.tx++; port->x_char = 0; return; } if (info->xmit.head == info->xmit.tail || info->tty->stopped || info->tty->hw_stopped) { sa1100_stop_tx(info->port, 0); return; } /* * Tried using FIFO (not checking TNF) for fifo fill: * still had the '4 bytes repeated' problem. */ while (UART_GET_UTSR1(port) & UTSR1_TNF) { UART_PUT_CHAR(port, info->xmit.buf[info->xmit.tail]); info->xmit.tail = (info->xmit.tail + 1) & (UART_XMIT_SIZE - 1); port->icount.tx++; if (info->xmit.head == info->xmit.tail) break; } if (CIRC_CNT(info->xmit.head, info->xmit.tail, UART_XMIT_SIZE) < WAKEUP_CHARS) uart_event(info, EVT_WRITE_WAKEUP); if (info->xmit.head == info->xmit.tail) sa1100_stop_tx(info->port, 0);}static void sa1100_int(int irq, void *dev_id, struct pt_regs *regs){ struct uart_info *info = dev_id; struct uart_port *port = info->port; unsigned int status, pass_counter = 0; status = UART_GET_UTSR0(port); status &= (SM_TO_UTSR0(port->read_status_mask) | ~UTSR0_TFS); do { if (status & (UTSR0_RFS | UTSR0_RID)) { /* Clear the receiver idle bit, if set */ if (status & UTSR0_RID) UART_PUT_UTSR0(port, UTSR0_RID); sa1100_rx_chars(info, regs); } /* Clear the relevent break bits */ if (status & (UTSR0_RBB | UTSR0_REB)) UART_PUT_UTSR0(port, status & (UTSR0_RBB | UTSR0_REB)); if (status & UTSR0_RBB) port->icount.brk++; if (status & UTSR0_REB) {#ifdef SUPPORT_SYSRQ if (port->line == sa1100_console.index && !info->sysrq) { info->sysrq = jiffies + HZ*5; }#endif } if (status & UTSR0_TFS) sa1100_tx_chars(info); if (pass_counter++ > SA1100_ISR_PASS_LIMIT) break; status = UART_GET_UTSR0(port); status &= (SM_TO_UTSR0(port->read_status_mask) | ~UTSR0_TFS); } while (status & (UTSR0_TFS | UTSR0_RFS | UTSR0_RID));}/* * Return TIOCSER_TEMT when transmitter is not busy. */static u_int sa1100_tx_empty(struct uart_port *port){ return UART_GET_UTSR1(port) & UTSR1_TBY ? 0 : TIOCSER_TEMT;}static u_int sa1100_get_mctrl(struct uart_port *port){ return TIOCM_CTS | TIOCM_DSR | TIOCM_CAR;}static void sa1100_set_mctrl(struct uart_port *port, u_int mctrl){}/* * Interrupts always disabled. */static void sa1100_break_ctl(struct uart_port *port, int break_state){ u_int utcr3; utcr3 = UART_GET_UTCR3(port); if (break_state == -1) utcr3 |= UTCR3_BRK; else utcr3 &= ~UTCR3_BRK; UART_PUT_UTCR3(port, utcr3);}static int sa1100_startup(struct uart_port *port, struct uart_info *info){ int retval; /* * Allocate the IRQ */ retval = request_irq(port->irq, sa1100_int, 0, "serial_sa1100", info); if (retval) return retval; /* * If there is a specific "open" function (to register * control line interrupts) */ if (sa1100_open) { retval = sa1100_open(port, info); if (retval) { free_irq(port->irq, info); return retval; } } /* * Finally, clear and enable interrupts */ UART_PUT_UTSR0(port, -1); UART_PUT_UTCR3(port, UTCR3_RXE | UTCR3_TXE | UTCR3_RIE); return 0;}static void sa1100_shutdown(struct uart_port *port, struct uart_info *info){ /* * Free the interrupt */ free_irq(port->irq, info); /* * If there is a specific "close" function (to unregister * control line interrupts) */ if (sa1100_close) sa1100_close(port, info); /* * Disable all interrupts, port and break condition. */ UART_PUT_UTCR3(port, 0);}static void sa1100_change_speed(struct uart_port *port, u_int cflag, u_int iflag, u_int quot){ unsigned long flags; u_int utcr0, old_utcr3;
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -