2006-07-10 04:45:14 -07:00
/*
2007-10-16 01:26:41 -07:00
* Copyright ( C ) 2001 - 2007 Jeff Dike ( jdike @ { linux . intel , addtoit } . com )
2005-04-16 15:20:36 -07:00
* Licensed under the GPL
*/
# include <errno.h>
2007-10-16 01:26:41 -07:00
# include <fcntl.h>
# include <termios.h>
2005-04-16 15:20:36 -07:00
# include "chan_user.h"
2007-10-16 01:26:41 -07:00
# include "kern_constants.h"
2005-04-16 15:20:36 -07:00
# include "os.h"
2006-10-19 23:28:20 -07:00
# include "um_malloc.h"
2007-10-16 01:26:41 -07:00
# include "user.h"
2005-04-16 15:20:36 -07:00
struct tty_chan {
char * dev ;
int raw ;
struct termios tt ;
} ;
2006-09-27 01:50:33 -07:00
static void * tty_chan_init ( char * str , int device , const struct chan_opts * opts )
2005-04-16 15:20:36 -07:00
{
struct tty_chan * data ;
2007-10-16 01:26:41 -07:00
if ( * str ! = ' : ' ) {
printk ( UM_KERN_ERR " tty_init : channel type 'tty' must specify "
2005-04-16 15:20:36 -07:00
" a device \n " ) ;
2006-07-10 04:45:14 -07:00
return NULL ;
2005-04-16 15:20:36 -07:00
}
str + + ;
2007-07-15 23:38:56 -07:00
data = kmalloc ( sizeof ( * data ) , UM_GFP_KERNEL ) ;
2007-10-16 01:26:41 -07:00
if ( data = = NULL )
2006-07-10 04:45:14 -07:00
return NULL ;
2005-04-16 15:20:36 -07:00
* data = ( ( struct tty_chan ) { . dev = str ,
. raw = opts - > raw } ) ;
2006-07-10 04:45:14 -07:00
return data ;
2005-04-16 15:20:36 -07:00
}
static int tty_open ( int input , int output , int primary , void * d ,
char * * dev_out )
{
struct tty_chan * data = d ;
2007-10-16 01:26:41 -07:00
int fd , err , mode = 0 ;
if ( input & & output )
mode = O_RDWR ;
else if ( input )
mode = O_RDONLY ;
else if ( output )
mode = O_WRONLY ;
2005-04-16 15:20:36 -07:00
2007-10-16 01:26:41 -07:00
fd = open ( data - > dev , mode ) ;
if ( fd < 0 )
return - errno ;
2006-07-10 04:45:14 -07:00
2007-10-16 01:26:41 -07:00
if ( data - > raw ) {
2005-04-16 15:20:36 -07:00
CATCH_EINTR ( err = tcgetattr ( fd , & data - > tt ) ) ;
2007-10-16 01:26:41 -07:00
if ( err )
2006-07-10 04:45:14 -07:00
return err ;
2005-04-16 15:20:36 -07:00
err = raw ( fd ) ;
2007-10-16 01:26:41 -07:00
if ( err )
2006-07-10 04:45:14 -07:00
return err ;
2005-04-16 15:20:36 -07:00
}
* dev_out = data - > dev ;
2006-07-10 04:45:14 -07:00
return fd ;
2005-04-16 15:20:36 -07:00
}
2006-09-27 01:50:33 -07:00
const struct chan_ops tty_ops = {
2005-04-16 15:20:36 -07:00
. type = " tty " ,
. init = tty_chan_init ,
. open = tty_open ,
. close = generic_close ,
. read = generic_read ,
. write = generic_write ,
2005-11-13 16:07:10 -08:00
. console_write = generic_console_write ,
2005-04-16 15:20:36 -07:00
. window_size = generic_window_size ,
. free = generic_free ,
. winch = 0 ,
} ;