2005-10-08 11:20:51 +00:00
# ifndef FISH_IO_H
# define FISH_IO_H
2012-03-07 08:54:01 +00:00
# include <vector>
2012-03-04 10:35:30 +00:00
# include <tr1/memory>
using std : : tr1 : : shared_ptr ;
2005-10-08 11:20:51 +00:00
/**
Describes what type of IO operation an io_data_t represents
*/
2012-12-31 17:11:46 +00:00
enum io_mode_t
2005-10-08 11:20:51 +00:00
{
2013-01-15 09:39:20 +00:00
IO_FILE , IO_PIPE , IO_FD , IO_BUFFER , IO_CLOSE
2012-03-04 10:35:30 +00:00
} ;
2005-10-08 11:20:51 +00:00
/** Represents an FD redirection */
2012-03-04 10:35:30 +00:00
class io_data_t
2005-10-08 11:20:51 +00:00
{
2012-03-04 10:35:30 +00:00
private :
2013-01-07 15:04:55 +00:00
/* No assignment or copying allowed */
io_data_t ( const io_data_t & rhs ) ;
2012-08-05 20:24:33 +00:00
void operator = ( const io_data_t & rhs ) ;
2013-01-04 10:03:41 +00:00
2013-01-15 09:39:20 +00:00
protected :
io_data_t ( io_mode_t m , int f ) :
io_mode ( m ) ,
2013-01-16 03:55:57 +00:00
fd ( f )
2013-01-15 09:39:20 +00:00
{
}
2012-03-04 10:35:30 +00:00
public :
2012-11-19 00:30:30 +00:00
/** Type of redirect */
2013-01-17 06:46:02 +00:00
const io_mode_t io_mode ;
2012-11-19 00:30:30 +00:00
/** FD to redirect */
int fd ;
2013-01-15 09:39:20 +00:00
virtual void print ( ) const = 0 ;
virtual ~ io_data_t ( ) = 0 ;
2012-02-10 02:43:36 +00:00
} ;
2013-01-09 08:02:04 +00:00
class io_close_t : public io_data_t
{
public :
io_close_t ( int f ) :
io_data_t ( IO_CLOSE , f )
{
}
virtual void print ( ) const ;
} ;
2013-01-15 07:37:33 +00:00
class io_fd_t : public io_data_t
{
public :
/** fd to redirect specified fd to */
int old_fd ;
/** Whether to close old_fd */
int close_old ;
virtual void print ( ) const ;
io_fd_t ( int f , int old , bool close = false ) :
io_data_t ( IO_FD , f ) ,
old_fd ( old ) ,
close_old ( close )
{
}
} ;
2013-01-15 08:18:03 +00:00
class io_file_t : public io_data_t
{
public :
/** Filename, malloc'd. This needs to be used after fork, so don't use wcstring here. */
const char * filename_cstr ;
/** file creation flags to send to open */
int flags ;
/** Convenience to set filename_cstr via wcstring */
void set_filename ( const wcstring & str )
{
free ( ( void * ) filename_cstr ) ;
filename_cstr = wcs2str ( str . c_str ( ) ) ;
}
virtual void print ( ) const ;
io_file_t ( int f , const char * fname = NULL , int fl = 0 ) :
io_data_t ( IO_FILE , f ) ,
filename_cstr ( fname ? strdup ( fname ) : NULL ) ,
flags ( fl )
{
}
virtual ~ io_file_t ( )
{
free ( ( void * ) filename_cstr ) ;
}
} ;
2005-10-08 11:20:51 +00:00
2013-01-15 09:31:36 +00:00
class io_pipe_t : public io_data_t
{
2013-01-17 06:46:02 +00:00
protected :
io_pipe_t ( io_mode_t m , int f , bool i ) :
io_data_t ( m , f ) ,
pipe_fd ( ) ,
is_input ( i )
{
}
2013-01-15 09:31:36 +00:00
public :
int pipe_fd [ 2 ] ;
2013-01-16 03:55:57 +00:00
bool is_input ;
2013-01-15 09:31:36 +00:00
virtual void print ( ) const ;
2013-01-16 03:55:57 +00:00
io_pipe_t ( int f , bool i ) :
2013-01-15 09:31:36 +00:00
io_data_t ( IO_PIPE , f ) ,
2013-01-16 03:55:57 +00:00
pipe_fd ( ) ,
is_input ( i )
2013-01-15 09:31:36 +00:00
{
}
} ;
class io_buffer_t : public io_pipe_t
2013-01-15 08:44:31 +00:00
{
private :
/** buffer to save output in */
2013-01-17 07:46:10 +00:00
shared_ptr < std : : vector < char > > out_buffer ;
2013-01-15 08:44:31 +00:00
2013-01-16 03:55:57 +00:00
io_buffer_t ( int f , bool i ) :
2013-01-17 06:46:02 +00:00
io_pipe_t ( IO_BUFFER , f , i ) ,
2013-01-17 07:46:10 +00:00
out_buffer ( )
2013-01-15 08:44:31 +00:00
{
}
2013-01-15 09:07:30 +00:00
public :
virtual void print ( ) const ;
2013-01-16 03:30:18 +00:00
virtual ~ io_buffer_t ( ) ;
2013-01-15 09:02:46 +00:00
2013-01-17 07:46:10 +00:00
/** Function to create the output buffer */
void out_buffer_create ( )
{
out_buffer . reset ( new std : : vector < char > ) ;
}
2013-01-15 08:44:31 +00:00
/** Function to append to the buffer */
void out_buffer_append ( const char * ptr , size_t count )
{
2013-01-17 07:46:10 +00:00
assert ( out_buffer . get ( ) ! = NULL ) ;
2013-01-15 08:44:31 +00:00
out_buffer - > insert ( out_buffer - > end ( ) , ptr , ptr + count ) ;
}
/** Function to get a pointer to the buffer */
char * out_buffer_ptr ( void )
{
2013-01-17 07:46:10 +00:00
assert ( out_buffer . get ( ) ! = NULL ) ;
2013-01-15 08:44:31 +00:00
return out_buffer - > empty ( ) ? NULL : & out_buffer - > at ( 0 ) ;
}
const char * out_buffer_ptr ( void ) const
{
2013-01-17 07:46:10 +00:00
assert ( out_buffer . get ( ) ! = NULL ) ;
2013-01-15 08:44:31 +00:00
return out_buffer - > empty ( ) ? NULL : & out_buffer - > at ( 0 ) ;
}
/** Function to get the size of the buffer */
size_t out_buffer_size ( void ) const
{
2013-01-17 07:46:10 +00:00
assert ( out_buffer . get ( ) ! = NULL ) ;
2013-01-15 08:44:31 +00:00
return out_buffer - > size ( ) ;
}
2013-01-15 09:07:30 +00:00
2013-01-15 09:10:40 +00:00
/**
Close output pipe , and read from input pipe until eof .
*/
void read ( ) ;
2013-01-15 09:07:30 +00:00
/**
Create a IO_BUFFER type io redirection , complete with a pipe and a
vector < char > for output . The default file descriptor used is 1 for
output buffering and 0 for input buffering .
\ param is_input set this parameter to zero if the buffer should be
used to buffer the output of a command , or non - zero to buffer the
input to a command .
*/
static io_buffer_t * create ( bool is_input ) ;
2013-01-15 08:44:31 +00:00
} ;
2013-01-07 15:04:55 +00:00
class io_chain_t : public std : : vector < shared_ptr < io_data_t > >
2012-11-19 00:30:30 +00:00
{
2012-08-15 07:57:56 +00:00
public :
io_chain_t ( ) ;
2013-01-07 15:04:55 +00:00
io_chain_t ( const shared_ptr < io_data_t > & ) ;
2012-11-18 10:23:22 +00:00
2013-01-07 15:04:55 +00:00
void remove ( const shared_ptr < const io_data_t > & element ) ;
2012-08-15 07:57:56 +00:00
io_chain_t duplicate ( ) const ;
2012-08-19 21:09:39 +00:00
void duplicate_prepend ( const io_chain_t & src ) ;
2012-08-15 07:57:56 +00:00
void destroy ( ) ;
2012-11-18 10:23:22 +00:00
2013-01-07 15:04:55 +00:00
shared_ptr < const io_data_t > get_io_for_fd ( int fd ) const ;
shared_ptr < io_data_t > get_io_for_fd ( int fd ) ;
2012-11-18 10:23:22 +00:00
2012-08-15 07:57:56 +00:00
} ;
2005-10-08 11:20:51 +00:00
/**
Remove the specified io redirection from the chain
*/
2013-01-07 15:04:55 +00:00
void io_remove ( io_chain_t & list , const shared_ptr < const io_data_t > & element ) ;
2005-10-08 11:20:51 +00:00
2012-08-15 07:57:56 +00:00
/** Return a shallow copy of the specified chain of redirections that contains only the applicable redirections. That is, if there's multiple redirections for the same fd, only the second one is included. */
io_chain_t io_unique ( const io_chain_t & chain ) ;
2012-08-19 21:09:39 +00:00
/** Prepends a copy of the specified 'src' chain of redirections to 'dst.' Uses operator new. */
2012-11-19 00:30:30 +00:00
void io_duplicate_prepend ( const io_chain_t & src , io_chain_t & dst ) ;
2012-08-15 07:57:56 +00:00
/** Destroys an io_chain */
void io_chain_destroy ( io_chain_t & chain ) ;
2005-10-08 11:20:51 +00:00
/**
2007-04-16 20:10:53 +00:00
Return the last io redirection in the chain for the specified file descriptor .
2005-10-08 11:20:51 +00:00
*/
2013-01-07 15:04:55 +00:00
shared_ptr < const io_data_t > io_chain_get ( const io_chain_t & src , int fd ) ;
shared_ptr < io_data_t > io_chain_get ( io_chain_t & src , int fd ) ;
2005-10-08 11:20:51 +00:00
2012-08-15 07:57:56 +00:00
/** Print debug information about the specified IO redirection chain to stderr. */
2012-11-19 00:30:30 +00:00
void io_print ( const io_chain_t & chain ) ;
2006-05-14 22:29:05 +00:00
2005-10-08 11:20:51 +00:00
# endif