2005-10-08 19:20:51 +08:00
# ifndef FISH_IO_H
# define FISH_IO_H
2012-03-07 16:54:01 +08:00
# include <vector>
2012-03-04 18:35:30 +08:00
# include <tr1/memory>
using std : : tr1 : : shared_ptr ;
2005-10-08 19:20:51 +08:00
/**
Describes what type of IO operation an io_data_t represents
*/
2013-01-01 01:11:46 +08:00
enum io_mode_t
2005-10-08 19:20:51 +08:00
{
2013-01-01 01:12:47 +08:00
IO_INVALID , IO_FILE , IO_PIPE , IO_FD , IO_BUFFER , IO_CLOSE
2012-03-04 18:35:30 +08:00
} ;
2005-10-08 19:20:51 +08:00
/** Represents an FD redirection */
2012-03-04 18:35:30 +08:00
class io_data_t
2005-10-08 19:20:51 +08:00
{
2012-03-04 18:35:30 +08:00
private :
2012-03-10 03:28:24 +08:00
/** buffer to save output in for IO_BUFFER. Note that in the original fish, the buffer was a pointer to a buffer_t stored in the param2 union down below, and when an io_data_t was duplicated the pointer was copied so that two io_data_ts referenced the same buffer. It's not clear to me how this was ever cleaned up correctly. But it's important that they share the same buffer for reasons I don't yet understand either. We can get correct sharing and cleanup with shared_ptr. */
2012-03-04 18:35:30 +08:00
shared_ptr < std : : vector < char > > out_buffer ;
2013-01-07 23:04:55 +08:00
/* No assignment or copying allowed */
io_data_t ( const io_data_t & rhs ) ;
2012-08-06 04:24:33 +08:00
void operator = ( const io_data_t & rhs ) ;
2013-01-04 18:03:41 +08:00
2012-03-04 18:35:30 +08:00
public :
2012-11-19 08:30:30 +08:00
/** Type of redirect */
2013-01-01 01:11:46 +08:00
io_mode_t io_mode ;
2012-11-19 08:30:30 +08:00
/** FD to redirect */
int fd ;
/**
Type - specific parameter for redirection
*/
union
{
/** Fds for IO_PIPE and for IO_BUFFER */
int pipe_fd [ 2 ] ;
/** fd to redirect specified fd to, for IO_FD */
int old_fd ;
} param1 ;
/** Second type-specific paramter for redirection */
union
{
/** file creation flags to send to open for IO_FILE */
int flags ;
/** Whether to close old_fd for IO_FD */
int close_old ;
} param2 ;
2012-11-18 18:23:22 +08:00
2012-03-09 15:21:07 +08:00
/** Filename IO_FILE. malloc'd. This needs to be used after fork, so don't use wcstring here. */
const char * filename_cstr ;
2012-11-18 18:23:22 +08:00
2012-03-09 15:21:07 +08:00
/** Convenience to set filename_cstr via wcstring */
2012-11-19 08:30:30 +08:00
void set_filename ( const wcstring & str )
{
2012-03-09 15:21:07 +08:00
free ( ( void * ) filename_cstr ) ;
filename_cstr = wcs2str ( str . c_str ( ) ) ;
}
2012-03-04 18:35:30 +08:00
/** Function to create the output buffer */
2012-11-19 08:30:30 +08:00
void out_buffer_create ( )
{
2012-03-04 18:35:30 +08:00
out_buffer . reset ( new std : : vector < char > ) ;
}
2012-11-18 18:23:22 +08:00
2012-03-04 18:35:30 +08:00
/** Function to append to the buffer */
2012-11-19 08:30:30 +08:00
void out_buffer_append ( const char * ptr , size_t count )
{
2012-11-18 18:23:22 +08:00
assert ( out_buffer . get ( ) ! = NULL ) ;
2012-03-04 18:35:30 +08:00
out_buffer - > insert ( out_buffer - > end ( ) , ptr , ptr + count ) ;
}
2012-11-18 18:23:22 +08:00
2012-03-04 18:35:30 +08:00
/** Function to get a pointer to the buffer */
2012-11-19 08:30:30 +08:00
char * out_buffer_ptr ( void )
{
2012-03-04 18:35:30 +08:00
assert ( out_buffer . get ( ) ! = NULL ) ;
2012-08-15 15:57:56 +08:00
return out_buffer - > empty ( ) ? NULL : & out_buffer - > at ( 0 ) ;
}
2012-11-18 18:23:22 +08:00
2012-11-19 08:30:30 +08:00
const char * out_buffer_ptr ( void ) const
{
2012-08-15 15:57:56 +08:00
assert ( out_buffer . get ( ) ! = NULL ) ;
return out_buffer - > empty ( ) ? NULL : & out_buffer - > at ( 0 ) ;
2012-03-04 18:35:30 +08:00
}
2012-11-18 18:23:22 +08:00
2012-03-04 18:35:30 +08:00
/** Function to get the size of the buffer */
2012-11-19 08:30:30 +08:00
size_t out_buffer_size ( void ) const
{
2012-03-04 18:35:30 +08:00
assert ( out_buffer . get ( ) ! = NULL ) ;
return out_buffer - > size ( ) ;
}
2006-08-13 09:38:03 +08:00
2012-11-19 08:30:30 +08:00
/** Set to true if this is an input io redirection */
bool is_input ;
2012-11-18 18:23:22 +08:00
2012-08-06 04:24:33 +08:00
io_data_t ( ) :
out_buffer ( ) ,
2013-01-01 01:12:47 +08:00
io_mode ( IO_INVALID ) ,
2012-08-06 04:24:33 +08:00
fd ( 0 ) ,
param1 ( ) ,
param2 ( ) ,
filename_cstr ( NULL ) ,
2012-08-15 15:57:56 +08:00
is_input ( 0 )
2012-03-09 15:21:07 +08:00
{
}
2012-11-18 18:23:22 +08:00
2012-11-19 08:30:30 +08:00
~ io_data_t ( )
{
2012-03-09 15:21:07 +08:00
free ( ( void * ) filename_cstr ) ;
}
2012-02-10 10:43:36 +08:00
} ;
2005-10-08 19:20:51 +08:00
2013-01-07 23:04:55 +08:00
class io_chain_t : public std : : vector < shared_ptr < io_data_t > >
2012-11-19 08:30:30 +08:00
{
2012-08-15 15:57:56 +08:00
public :
io_chain_t ( ) ;
2013-01-07 23:04:55 +08:00
io_chain_t ( const shared_ptr < io_data_t > & ) ;
2012-11-18 18:23:22 +08:00
2013-01-07 23:04:55 +08:00
void remove ( const shared_ptr < const io_data_t > & element ) ;
2012-08-15 15:57:56 +08:00
io_chain_t duplicate ( ) const ;
2012-08-20 05:09:39 +08:00
void duplicate_prepend ( const io_chain_t & src ) ;
2012-08-15 15:57:56 +08:00
void destroy ( ) ;
2012-11-18 18:23:22 +08:00
2013-01-07 23:04:55 +08: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 18:23:22 +08:00
2012-08-15 15:57:56 +08:00
} ;
2005-10-08 19:20:51 +08:00
/**
Remove the specified io redirection from the chain
*/
2013-01-07 23:04:55 +08:00
void io_remove ( io_chain_t & list , const shared_ptr < const io_data_t > & element ) ;
2005-10-08 19:20:51 +08:00
2012-08-15 15:57:56 +08: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-20 05:09:39 +08:00
/** Prepends a copy of the specified 'src' chain of redirections to 'dst.' Uses operator new. */
2012-11-19 08:30:30 +08:00
void io_duplicate_prepend ( const io_chain_t & src , io_chain_t & dst ) ;
2012-08-15 15:57:56 +08:00
/** Destroys an io_chain */
void io_chain_destroy ( io_chain_t & chain ) ;
2005-10-08 19:20:51 +08:00
/**
2007-04-17 04:10:53 +08:00
Return the last io redirection in the chain for the specified file descriptor .
2005-10-08 19:20:51 +08:00
*/
2013-01-07 23:04:55 +08: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 19:20:51 +08:00
/**
Free all resources used by a IO_BUFFER type io redirection .
*/
2013-01-07 23:04:55 +08:00
void io_buffer_destroy ( const shared_ptr < io_data_t > & io_buffer ) ;
2005-10-08 19:20:51 +08:00
/**
Create a IO_BUFFER type io redirection , complete with a pipe and a
2012-03-09 15:21:07 +08:00
vector < char > for output . The default file descriptor used is 1 for
2006-08-13 09:38:03 +08:00
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 .
2005-10-08 19:20:51 +08:00
*/
2012-11-19 08:30:30 +08:00
io_data_t * io_buffer_create ( bool is_input ) ;
2005-10-08 19:20:51 +08:00
/**
Close output pipe , and read from input pipe until eof .
*/
2012-11-19 08:30:30 +08:00
void io_buffer_read ( io_data_t * d ) ;
2005-10-08 19:20:51 +08:00
2012-08-15 15:57:56 +08:00
/** Print debug information about the specified IO redirection chain to stderr. */
2012-11-19 08:30:30 +08:00
void io_print ( const io_chain_t & chain ) ;
2006-05-15 06:29:05 +08:00
2005-10-08 19:20:51 +08:00
# endif