2005-04-17 02:20:36 +04:00
/*
* Copyright ( C ) 2000 , 2001 Jeff Dike ( jdike @ karaya . com )
* Licensed under the GPL
*/
# include "linux/config.h"
# include "linux/posix_types.h"
# include "linux/tty.h"
# include "linux/tty_flip.h"
# include "linux/types.h"
# include "linux/major.h"
# include "linux/kdev_t.h"
# include "linux/console.h"
# include "linux/string.h"
# include "linux/sched.h"
# include "linux/list.h"
# include "linux/init.h"
# include "linux/interrupt.h"
# include "linux/slab.h"
# include "linux/hardirq.h"
# include "asm/current.h"
# include "asm/irq.h"
# include "stdio_console.h"
# include "line.h"
# include "chan_kern.h"
# include "user_util.h"
# include "kern_util.h"
# include "irq_user.h"
# include "mconsole_kern.h"
# include "init.h"
# define MAX_TTYS (16)
/* ----------------------------------------------------------------------------- */
/* Referenced only by tty_driver below - presumably it's locked correctly
* by the tty driver .
*/
static struct tty_driver * console_driver ;
void stdio_announce ( char * dev_name , int dev )
{
printk ( KERN_INFO " Virtual console %d assigned device '%s' \n " , dev ,
dev_name ) ;
}
static struct chan_opts opts = {
. announce = stdio_announce ,
. xterm_title = " Virtual Console #%d " ,
. raw = 1 ,
. tramp_stack = 0 ,
. in_kernel = 1 ,
} ;
static int con_config ( char * str ) ;
static int con_get_config ( char * dev , char * str , int size , char * * error_out ) ;
2005-06-26 01:55:25 +04:00
static int con_remove ( int n ) ;
2005-04-17 02:20:36 +04:00
static struct line_driver driver = {
. name = " UML console " ,
. device_name = " tty " ,
. devfs_name = " vc/ " ,
. major = TTY_MAJOR ,
. minor_start = 0 ,
. type = TTY_DRIVER_TYPE_CONSOLE ,
. subtype = SYSTEM_TYPE_CONSOLE ,
. read_irq = CONSOLE_IRQ ,
. read_irq_name = " console " ,
. write_irq = CONSOLE_WRITE_IRQ ,
. write_irq_name = " console-write " ,
. symlink_from = " ttys " ,
. symlink_to = " vc " ,
. mc = {
. name = " con " ,
. config = con_config ,
. get_config = con_get_config ,
2006-01-06 11:18:50 +03:00
. id = line_id ,
2005-04-17 02:20:36 +04:00
. remove = con_remove ,
} ,
} ;
static struct lines console_lines = LINES_INIT ( MAX_TTYS ) ;
/* The array is initialized by line_init, which is an initcall. The
* individual elements are protected by individual semaphores .
*/
struct line vts [ MAX_TTYS ] = { LINE_INIT ( CONFIG_CON_ZERO_CHAN , & driver ) ,
2006-01-06 11:18:50 +03:00
[ 1 . . . MAX_TTYS - 1 ] =
2005-04-17 02:20:36 +04:00
LINE_INIT ( CONFIG_CON_CHAN , & driver ) } ;
static int con_config ( char * str )
{
2006-01-06 11:18:51 +03:00
return line_config ( vts , ARRAY_SIZE ( vts ) , str ) ;
2005-04-17 02:20:36 +04:00
}
static int con_get_config ( char * dev , char * str , int size , char * * error_out )
{
2006-01-06 11:18:51 +03:00
return line_get_config ( dev , vts , ARRAY_SIZE ( vts ) , str , size , error_out ) ;
2005-04-17 02:20:36 +04:00
}
2005-06-26 01:55:25 +04:00
static int con_remove ( int n )
2005-04-17 02:20:36 +04:00
{
2006-01-06 11:18:51 +03:00
return line_remove ( vts , ARRAY_SIZE ( vts ) , n ) ;
2005-04-17 02:20:36 +04:00
}
static int con_open ( struct tty_struct * tty , struct file * filp )
{
return line_open ( vts , tty , & opts ) ;
}
static int con_init_done = 0 ;
static struct tty_operations console_ops = {
. open = con_open ,
. close = line_close ,
. write = line_write ,
2005-05-01 19:58:56 +04:00
. put_char = line_put_char ,
2006-01-06 11:18:50 +03:00
. write_room = line_write_room ,
2005-04-17 02:20:36 +04:00
. chars_in_buffer = line_chars_in_buffer ,
2005-05-01 19:58:56 +04:00
. flush_buffer = line_flush_buffer ,
. flush_chars = line_flush_chars ,
2005-04-17 02:20:36 +04:00
. set_termios = line_set_termios ,
. ioctl = line_ioctl ,
} ;
static void uml_console_write ( struct console * console , const char * string ,
2006-01-06 11:18:50 +03:00
unsigned len )
2005-04-17 02:20:36 +04:00
{
struct line * line = & vts [ console - > index ] ;
2005-05-01 19:58:56 +04:00
unsigned long flags ;
2005-04-17 02:20:36 +04:00
2005-05-01 19:58:56 +04:00
spin_lock_irqsave ( & line - > lock , flags ) ;
2005-04-17 02:20:36 +04:00
console_write_chan ( & line - > chan_list , string , len ) ;
2005-05-01 19:58:56 +04:00
spin_unlock_irqrestore ( & line - > lock , flags ) ;
2005-04-17 02:20:36 +04:00
}
static struct tty_driver * uml_console_device ( struct console * c , int * index )
{
* index = c - > index ;
return console_driver ;
}
static int uml_console_setup ( struct console * co , char * options )
{
struct line * line = & vts [ co - > index ] ;
2006-01-06 11:18:50 +03:00
return console_open_chan ( line , co , & opts ) ;
2005-04-17 02:20:36 +04:00
}
static struct console stdiocons = {
. name = " tty " ,
. write = uml_console_write ,
. device = uml_console_device ,
. setup = uml_console_setup ,
. flags = CON_PRINTBUFFER ,
. index = - 1 ,
2006-01-06 11:18:50 +03:00
. data = & vts ,
2005-04-17 02:20:36 +04:00
} ;
int stdio_init ( void )
{
char * new_title ;
console_driver = line_register_devfs ( & console_lines , & driver ,
& console_ops , vts ,
ARRAY_SIZE ( vts ) ) ;
2006-01-06 11:18:50 +03:00
if ( console_driver = = NULL )
2005-04-17 02:20:36 +04:00
return - 1 ;
printk ( KERN_INFO " Initialized stdio console driver \n " ) ;
2006-01-06 11:18:51 +03:00
lines_init ( vts , ARRAY_SIZE ( vts ) ) ;
2005-04-17 02:20:36 +04:00
new_title = add_xterm_umid ( opts . xterm_title ) ;
if ( new_title ! = NULL )
opts . xterm_title = new_title ;
con_init_done = 1 ;
register_console ( & stdiocons ) ;
2006-01-06 11:18:50 +03:00
return 0 ;
2005-04-17 02:20:36 +04:00
}
late_initcall ( stdio_init ) ;
static void console_exit ( void )
{
if ( ! con_init_done )
return ;
2006-01-06 11:18:51 +03:00
close_lines ( vts , ARRAY_SIZE ( vts ) ) ;
2005-04-17 02:20:36 +04:00
}
__uml_exitcall ( console_exit ) ;
static int console_chan_setup ( char * str )
{
2006-01-06 11:18:51 +03:00
return line_setup ( vts , ARRAY_SIZE ( vts ) , str , 1 ) ;
2005-04-17 02:20:36 +04:00
}
__setup ( " con " , console_chan_setup ) ;
__channel_help ( console_chan_setup , " con " ) ;