sn_console.c
来自「linux 内核源代码」· C语言 代码 · 共 1,086 行 · 第 1/2 页
C
1,086 行
/* * C-Brick Serial Port (and console) driver for SGI Altix machines. * * This driver is NOT suitable for talking to the l1-controller for * anything other than 'console activities' --- please use the l1 * driver for that. * * * Copyright (c) 2004-2006 Silicon Graphics, Inc. All Rights Reserved. * * This program is free software; you can redistribute it and/or modify it * under the terms of version 2 of the GNU General Public License * as published by the Free Software Foundation. * * This program is distributed in the hope that it would be useful, but * WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. * * Further, this software is distributed without any warranty that it is * free of the rightful claim of any third person regarding infringement * or the like. Any license provided herein, whether implied or * otherwise, applies only to this software file. Patent licenses, if * any, provided herein do not apply to combinations of this program with * other software, or any other product whatsoever. * * You should have received a copy of the GNU General Public * License along with this program; if not, write the Free Software * Foundation, Inc., 59 Temple Place - Suite 330, Boston MA 02111-1307, USA. * * Contact information: Silicon Graphics, Inc., 1500 Crittenden Lane, * Mountain View, CA 94043, or: * * http://www.sgi.com * * For further information regarding this notice, see: * * http://oss.sgi.com/projects/GenInfo/NoticeExplan */#include <linux/interrupt.h>#include <linux/tty.h>#include <linux/serial.h>#include <linux/console.h>#include <linux/module.h>#include <linux/sysrq.h>#include <linux/circ_buf.h>#include <linux/serial_reg.h>#include <linux/delay.h> /* for mdelay */#include <linux/miscdevice.h>#include <linux/serial_core.h>#include <asm/io.h>#include <asm/sn/simulator.h>#include <asm/sn/sn_sal.h>/* number of characters we can transmit to the SAL console at a time */#define SN_SAL_MAX_CHARS 120/* 64K, when we're asynch, it must be at least printk's LOG_BUF_LEN to * avoid losing chars, (always has to be a power of 2) */#define SN_SAL_BUFFER_SIZE (64 * (1 << 10))#define SN_SAL_UART_FIFO_DEPTH 16#define SN_SAL_UART_FIFO_SPEED_CPS 9600/10/* sn_transmit_chars() calling args */#define TRANSMIT_BUFFERED 0#define TRANSMIT_RAW 1/* To use dynamic numbers only and not use the assigned major and minor, * define the following.. */ /* #define USE_DYNAMIC_MINOR 1 *//* use dynamic minor number */#define USE_DYNAMIC_MINOR 0 /* Don't rely on misc_register dynamic minor *//* Device name we're using */#define DEVICE_NAME "ttySG"#define DEVICE_NAME_DYNAMIC "ttySG0" /* need full name for misc_register *//* The major/minor we are using, ignored for USE_DYNAMIC_MINOR */#define DEVICE_MAJOR 204#define DEVICE_MINOR 40#ifdef CONFIG_MAGIC_SYSRQstatic char sysrq_serial_str[] = "\eSYS";static char *sysrq_serial_ptr = sysrq_serial_str;static unsigned long sysrq_requested;#endif /* CONFIG_MAGIC_SYSRQ *//* * Port definition - this kinda drives it all */struct sn_cons_port { struct timer_list sc_timer; struct uart_port sc_port; struct sn_sal_ops { int (*sal_puts_raw) (const char *s, int len); int (*sal_puts) (const char *s, int len); int (*sal_getc) (void); int (*sal_input_pending) (void); void (*sal_wakeup_transmit) (struct sn_cons_port *, int); } *sc_ops; unsigned long sc_interrupt_timeout; int sc_is_asynch;};static struct sn_cons_port sal_console_port;static int sn_process_input;/* Only used if USE_DYNAMIC_MINOR is set to 1 */static struct miscdevice misc; /* used with misc_register for dynamic */extern void early_sn_setup(void);#undef DEBUG#ifdef DEBUGstatic int sn_debug_printf(const char *fmt, ...);#define DPRINTF(x...) sn_debug_printf(x)#else#define DPRINTF(x...) do { } while (0)#endif/* Prototypes */static int snt_hw_puts_raw(const char *, int);static int snt_hw_puts_buffered(const char *, int);static int snt_poll_getc(void);static int snt_poll_input_pending(void);static int snt_intr_getc(void);static int snt_intr_input_pending(void);static void sn_transmit_chars(struct sn_cons_port *, int);/* A table for polling: */static struct sn_sal_ops poll_ops = { .sal_puts_raw = snt_hw_puts_raw, .sal_puts = snt_hw_puts_raw, .sal_getc = snt_poll_getc, .sal_input_pending = snt_poll_input_pending};/* A table for interrupts enabled */static struct sn_sal_ops intr_ops = { .sal_puts_raw = snt_hw_puts_raw, .sal_puts = snt_hw_puts_buffered, .sal_getc = snt_intr_getc, .sal_input_pending = snt_intr_input_pending, .sal_wakeup_transmit = sn_transmit_chars};/* the console does output in two distinctly different ways: * synchronous (raw) and asynchronous (buffered). initally, early_printk * does synchronous output. any data written goes directly to the SAL * to be output (incidentally, it is internally buffered by the SAL) * after interrupts and timers are initialized and available for use, * the console init code switches to asynchronous output. this is * also the earliest opportunity to begin polling for console input. * after console initialization, console output and tty (serial port) * output is buffered and sent to the SAL asynchronously (either by * timer callback or by UART interrupt) *//* routines for running the console in polling mode *//** * snt_poll_getc - Get a character from the console in polling mode * */static int snt_poll_getc(void){ int ch; ia64_sn_console_getc(&ch); return ch;}/** * snt_poll_input_pending - Check if any input is waiting - polling mode. * */static int snt_poll_input_pending(void){ int status, input; status = ia64_sn_console_check(&input); return !status && input;}/* routines for an interrupt driven console (normal) *//** * snt_intr_getc - Get a character from the console, interrupt mode * */static int snt_intr_getc(void){ return ia64_sn_console_readc();}/** * snt_intr_input_pending - Check if input is pending, interrupt mode * */static int snt_intr_input_pending(void){ return ia64_sn_console_intr_status() & SAL_CONSOLE_INTR_RECV;}/* these functions are polled and interrupt *//** * snt_hw_puts_raw - Send raw string to the console, polled or interrupt mode * @s: String * @len: Length * */static int snt_hw_puts_raw(const char *s, int len){ /* this will call the PROM and not return until this is done */ return ia64_sn_console_putb(s, len);}/** * snt_hw_puts_buffered - Send string to console, polled or interrupt mode * @s: String * @len: Length * */static int snt_hw_puts_buffered(const char *s, int len){ /* queue data to the PROM */ return ia64_sn_console_xmit_chars((char *)s, len);}/* uart interface structs * These functions are associated with the uart_port that the serial core * infrastructure calls. * * Note: Due to how the console works, many routines are no-ops. *//** * snp_type - What type of console are we? * @port: Port to operate with (we ignore since we only have one port) * */static const char *snp_type(struct uart_port *port){ return ("SGI SN L1");}/** * snp_tx_empty - Is the transmitter empty? We pretend we're always empty * @port: Port to operate on (we ignore since we only have one port) * */static unsigned int snp_tx_empty(struct uart_port *port){ return 1;}/** * snp_stop_tx - stop the transmitter - no-op for us * @port: Port to operat eon - we ignore - no-op function * */static void snp_stop_tx(struct uart_port *port){}/** * snp_release_port - Free i/o and resources for port - no-op for us * @port: Port to operate on - we ignore - no-op function * */static void snp_release_port(struct uart_port *port){}/** * snp_enable_ms - Force modem status interrupts on - no-op for us * @port: Port to operate on - we ignore - no-op function * */static void snp_enable_ms(struct uart_port *port){}/** * snp_shutdown - shut down the port - free irq and disable - no-op for us * @port: Port to shut down - we ignore * */static void snp_shutdown(struct uart_port *port){}/** * snp_set_mctrl - set control lines (dtr, rts, etc) - no-op for our console * @port: Port to operate on - we ignore * @mctrl: Lines to set/unset - we ignore * */static void snp_set_mctrl(struct uart_port *port, unsigned int mctrl){}/** * snp_get_mctrl - get contorl line info, we just return a static value * @port: port to operate on - we only have one port so we ignore this * */static unsigned int snp_get_mctrl(struct uart_port *port){ return TIOCM_CAR | TIOCM_RNG | TIOCM_DSR | TIOCM_CTS;}/** * snp_stop_rx - Stop the receiver - we ignor ethis * @port: Port to operate on - we ignore * */static void snp_stop_rx(struct uart_port *port){}/** * snp_start_tx - Start transmitter * @port: Port to operate on * */static void snp_start_tx(struct uart_port *port){ if (sal_console_port.sc_ops->sal_wakeup_transmit) sal_console_port.sc_ops->sal_wakeup_transmit(&sal_console_port, TRANSMIT_BUFFERED);}/** * snp_break_ctl - handle breaks - ignored by us * @port: Port to operate on * @break_state: Break state * */static void snp_break_ctl(struct uart_port *port, int break_state){}/** * snp_startup - Start up the serial port - always return 0 (We're always on) * @port: Port to operate on * */static int snp_startup(struct uart_port *port){ return 0;}/** * snp_set_termios - set termios stuff - we ignore these * @port: port to operate on * @termios: New settings * @termios: Old * */static voidsnp_set_termios(struct uart_port *port, struct ktermios *termios, struct ktermios *old){}/** * snp_request_port - allocate resources for port - ignored by us * @port: port to operate on * */static int snp_request_port(struct uart_port *port){ return 0;}/** * snp_config_port - allocate resources, set up - we ignore, we're always on * @port: Port to operate on * @flags: flags used for port setup * */static void snp_config_port(struct uart_port *port, int flags){}/* Associate the uart functions above - given to serial core */static struct uart_ops sn_console_ops = { .tx_empty = snp_tx_empty, .set_mctrl = snp_set_mctrl, .get_mctrl = snp_get_mctrl, .stop_tx = snp_stop_tx, .start_tx = snp_start_tx, .stop_rx = snp_stop_rx, .enable_ms = snp_enable_ms, .break_ctl = snp_break_ctl, .startup = snp_startup, .shutdown = snp_shutdown, .set_termios = snp_set_termios, .pm = NULL, .type = snp_type, .release_port = snp_release_port, .request_port = snp_request_port, .config_port = snp_config_port, .verify_port = NULL,};/* End of uart struct functions and defines */#ifdef DEBUG/** * sn_debug_printf - close to hardware debugging printf * @fmt: printf format * * This is as "close to the metal" as we can get, used when the driver * itself may be broken. * */static int sn_debug_printf(const char *fmt, ...){ static char printk_buf[1024]; int printed_len; va_list args; va_start(args, fmt); printed_len = vsnprintf(printk_buf, sizeof(printk_buf), fmt, args); if (!sal_console_port.sc_ops) { sal_console_port.sc_ops = &poll_ops; early_sn_setup(); } sal_console_port.sc_ops->sal_puts_raw(printk_buf, printed_len); va_end(args); return printed_len;}#endif /* DEBUG *//* * Interrupt handling routines. *//** * sn_receive_chars - Grab characters, pass them to tty layer * @port: Port to operate on * @flags: irq flags * * Note: If we're not registered with the serial core infrastructure yet, * we don't try to send characters to it... * */static voidsn_receive_chars(struct sn_cons_port *port, unsigned long flags){ int ch; struct tty_struct *tty; if (!port) { printk(KERN_ERR "sn_receive_chars - port NULL so can't receieve\n"); return; } if (!port->sc_ops) { printk(KERN_ERR "sn_receive_chars - port->sc_ops NULL so can't receieve\n"); return; } if (port->sc_port.info) { /* The serial_core stuffs are initilized, use them */ tty = port->sc_port.info->tty; } else { /* Not registered yet - can't pass to tty layer. */ tty = NULL; } while (port->sc_ops->sal_input_pending()) { ch = port->sc_ops->sal_getc(); if (ch < 0) { printk(KERN_ERR "sn_console: An error occured while " "obtaining data from the console (0x%0x)\n", ch); break; }#ifdef CONFIG_MAGIC_SYSRQ if (sysrq_requested) { unsigned long sysrq_timeout = sysrq_requested + HZ*5; sysrq_requested = 0; if (ch && time_before(jiffies, sysrq_timeout)) { spin_unlock_irqrestore(&port->sc_port.lock, flags); handle_sysrq(ch, NULL); spin_lock_irqsave(&port->sc_port.lock, flags); /* ignore actual sysrq command char */ continue; } } if (ch == *sysrq_serial_ptr) { if (!(*++sysrq_serial_ptr)) { sysrq_requested = jiffies; sysrq_serial_ptr = sysrq_serial_str; } /* * ignore the whole sysrq string except for the * leading escape */ if (ch != '\e') continue; } else sysrq_serial_ptr = sysrq_serial_str;#endif /* CONFIG_MAGIC_SYSRQ */ /* record the character to pass up to the tty layer */ if (tty) { if(tty_insert_flip_char(tty, ch, TTY_NORMAL) == 0) break; } port->sc_port.icount.rx++; } if (tty) tty_flip_buffer_push(tty);}/** * sn_transmit_chars - grab characters from serial core, send off * @port: Port to operate on * @raw: Transmit raw or buffered * * Note: If we're early, before we're registered with serial core, the * writes are going through sn_sal_console_write because that's how * register_console has been set up. We currently could have asynch * polls calling this function due to sn_sal_switch_to_asynch but we can * ignore them until we register with the serial core stuffs. * */static void sn_transmit_chars(struct sn_cons_port *port, int raw){ int xmit_count, tail, head, loops, ii;
⌨️ 快捷键说明
复制代码Ctrl + C
搜索代码Ctrl + F
全屏模式F11
增大字号Ctrl + =
减小字号Ctrl + -
显示快捷键?