2005-09-20 13:26:39 +00:00
/*
2008-01-12 19:18:48 +00:00
Copyright ( C ) 2005 - 2008 Axel Liljencrantz
2005-09-20 13:26:39 +00:00
2006-11-01 14:47:47 +00:00
This program is free software ; you can redistribute it and / or modify
it under the terms of the GNU General Public License version 2 as
published by the Free Software Foundation .
2005-09-20 13:26:39 +00:00
This program is distributed in the hope that it will be useful ,
but WITHOUT ANY WARRANTY ; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE . See the
GNU General Public License for more details .
You should have received a copy of the GNU General Public License
along with this program ; if not , write to the Free Software
Foundation , Inc . , 59 Temple Place - Suite 330 , Boston , MA 02111 - 1307 , USA .
*/
2008-01-12 19:21:35 +00:00
/** \file fish.c
2005-09-20 13:26:39 +00:00
The main loop of < tt > fish < / tt > .
*/
# include "config.h"
2006-02-28 13:17:16 +00:00
2005-09-20 13:26:39 +00:00
# include <stdlib.h>
# include <stdio.h>
# include <wchar.h>
# include <string.h>
# include <unistd.h>
# include <errno.h>
# include <unistd.h>
# include <termios.h>
# include <fcntl.h>
2012-07-08 22:20:39 +00:00
# include <sys/param.h>
2005-09-20 13:26:39 +00:00
# ifdef HAVE_GETOPT_H
# include <getopt.h>
# endif
# include <locale.h>
# include <signal.h>
2006-02-28 13:17:16 +00:00
# include "fallback.h"
2005-09-20 13:26:39 +00:00
# include "util.h"
2006-02-28 13:17:16 +00:00
2005-09-20 13:26:39 +00:00
# include "common.h"
# include "reader.h"
# include "builtin.h"
# include "function.h"
# include "complete.h"
# include "wutil.h"
# include "env.h"
# include "sanity.h"
# include "proc.h"
# include "parser.h"
# include "expand.h"
# include "intern.h"
2005-10-05 22:37:08 +00:00
# include "exec.h"
# include "event.h"
2005-10-15 00:51:26 +00:00
# include "output.h"
2006-04-19 23:42:11 +00:00
# include "history.h"
2006-10-19 11:50:23 +00:00
# include "path.h"
2005-09-20 13:26:39 +00:00
2012-07-08 22:20:39 +00:00
/* PATH_MAX may not exist */
# ifndef PATH_MAX
# define PATH_MAX 1024
# endif
2006-05-18 13:00:39 +00:00
/**
The string describing the single - character options accepted by the main fish binary
*/
2010-10-03 03:46:26 +00:00
# define GETOPT_STRING "+hilnvc:p:d:"
2006-03-10 13:38:09 +00:00
2012-07-08 22:20:39 +00:00
static bool has_suffix ( const std : : string & path , const char * suffix , bool ignore_case )
{
size_t pathlen = path . size ( ) , suffixlen = strlen ( suffix ) ;
return pathlen > = suffixlen & & ! ( ignore_case ? strcasecmp : strcmp ) ( path . c_str ( ) + pathlen - suffixlen , suffix ) ;
}
/* Modifies the given path by calling realpath. Returns true if realpath succeeded, false otherwise */
static bool get_realpath ( std : : string & path )
{
char buff [ PATH_MAX ] , * ptr ;
if ( ( ptr = realpath ( path . c_str ( ) , buff ) ) )
{
path = ptr ;
}
return ptr ! = NULL ;
}
/* OS X function for getting the executable path */
extern " C " {
int _NSGetExecutablePath ( char * buf , uint32_t * bufsize ) ;
}
/* Return the path to the current executable. This needs to be realpath'd. */
static std : : string get_executable_path ( const char * argv0 )
{
char buff [ PATH_MAX ] ;
# if __APPLE__
{
/* Returns 0 on success, -1 if the buffer is too small */
uint32_t buffSize = sizeof buff ;
if ( 0 = = _NSGetExecutablePath ( buff , & buffSize ) )
return std : : string ( buff ) ;
/* Loop until we're big enough */
char * mbuff = ( char * ) malloc ( buffSize ) ;
while ( 0 > _NSGetExecutablePath ( mbuff , & buffSize ) )
mbuff = ( char * ) realloc ( mbuff , buffSize ) ;
/* Return the string */
std : : string result = mbuff ;
free ( mbuff ) ;
return result ;
}
# endif
{
/* On other Unixes, try /proc directory. This might be worth breaking out into macros. */
if ( 0 < readlink ( " /proc/self/exe " , buff , sizeof buff ) | | // Linux
0 < readlink ( " /proc/curproc/file " , buff , sizeof buff ) | | // BSD
0 < readlink ( " /proc/self/path/a.out " , buff , sizeof buff ) ) // Solaris
{
return std : : string ( buff ) ;
}
}
/* Just return argv0, which probably won't work (i.e. it's not an absolute path or a path relative to the working directory, but instead something the caller found via $PATH). We'll eventually fall back to the compile time paths. */
return std : : string ( argv0 ? argv0 : " " ) ;
}
static struct config_paths_t determine_config_directory_paths ( const char * argv0 )
{
struct config_paths_t paths ;
bool done = false ;
std : : string exec_path = get_executable_path ( argv0 ) ;
if ( get_realpath ( exec_path ) )
{
# if __APPLE__
/* On OS X, maybe we're an app bundle, and should use the bundle's files. Since we don't link CF, use this lame approach to test it: see if the resolved path ends with /Contents/MacOS/fish, case insensitive since HFS+ usually is.
*/
if ( ! done )
{
const char * suffix = " /Contents/MacOS/fish " ;
const size_t suffixlen = strlen ( suffix ) ;
if ( has_suffix ( exec_path , suffix , true ) )
{
/* Looks like we're a bundle. Cut the string at the / prefixing /Contents... and then the rest */
wcstring wide_resolved_path = str2wcstring ( exec_path ) ;
wide_resolved_path . resize ( exec_path . size ( ) - suffixlen ) ;
wide_resolved_path . append ( L " /Contents/Resources/ " ) ;
/* Append share, etc, doc */
paths . data = wide_resolved_path + L " share/fish " ;
paths . sysconf = wide_resolved_path + L " etc/fish " ;
2012-07-08 22:42:47 +00:00
paths . doc = wide_resolved_path + L " doc/fish " ;
2012-07-08 22:20:39 +00:00
/* But the bin_dir is the resolved_path, minus fish (aka the MacOS directory) */
paths . bin = str2wcstring ( exec_path ) ;
paths . bin . resize ( paths . bin . size ( ) - strlen ( " /fish " ) ) ;
done = true ;
}
}
# endif
if ( ! done )
{
/* The next check is that we are in a reloctable directory tree like this:
bin / fish
etc / fish
share / fish
Check it !
*/
const char * suffix = " /bin/fish " ;
if ( has_suffix ( exec_path , suffix , false ) )
{
wcstring base_path = str2wcstring ( exec_path ) ;
base_path . resize ( base_path . size ( ) - strlen ( suffix ) ) ;
paths . data = base_path + L " /share/fish " ;
paths . sysconf = base_path + L " /etc/fish " ;
2012-07-08 22:42:47 +00:00
paths . doc = base_path + L " /share/doc/fish " ;
2012-07-08 22:20:39 +00:00
paths . bin = base_path + L " /bin " ;
struct stat buf ;
if ( 0 = = wstat ( paths . data , & buf ) & & 0 = = wstat ( paths . sysconf , & buf ) )
{
done = true ;
}
}
}
}
if ( ! done )
{
/* Fall back to what got compiled in. */
paths . data = L " " DATADIR " /fish " ;
paths . sysconf = L " " SYSCONFDIR " /fish " ;
paths . doc = L " " DATADIR " /doc/fish " ;
paths . bin = L " " PREFIX " /bin " ;
done = true ;
}
return paths ;
}
2005-09-20 13:26:39 +00:00
/**
2012-07-08 22:20:39 +00:00
Parse init files . exec_path is the path of fish executable as determined by argv [ 0 ] .
2005-09-20 13:26:39 +00:00
*/
2012-07-08 22:20:39 +00:00
static int read_init ( const struct config_paths_t & paths )
2005-09-20 13:26:39 +00:00
{
2012-01-23 05:57:30 +00:00
parser_t & parser = parser_t : : principal_parser ( ) ;
2012-07-08 22:20:39 +00:00
parser . eval ( L " builtin . " + paths . data + L " /config.fish 2>/dev/null " , 0 , TOP ) ;
parser . eval ( L " builtin . " + paths . sysconf + L " /config.fish 2>/dev/null " , 0 , TOP ) ;
2011-12-27 03:18:46 +00:00
2006-10-19 11:50:23 +00:00
/*
We need to get the configuration directory before we can source the user configuration file
*/
2012-02-08 06:44:10 +00:00
wcstring config_dir ;
2007-04-11 13:18:23 +00:00
/*
2012-02-08 06:44:10 +00:00
If path_get_config returns false then we have no configuration directory
2007-04-11 13:18:23 +00:00
and no custom config to load .
*/
2012-02-08 06:44:10 +00:00
if ( path_get_config ( config_dir ) )
2007-04-11 13:18:23 +00:00
{
2012-02-08 06:44:10 +00:00
wcstring config_dir_escaped = escape_string ( config_dir , 1 ) ;
wcstring eval_buff = format_string ( L " builtin . %ls/config.fish 2>/dev/null " , config_dir_escaped . c_str ( ) ) ;
2012-03-06 23:12:37 +00:00
parser . eval ( eval_buff , 0 , TOP ) ;
2007-04-11 13:18:23 +00:00
}
2011-12-27 03:18:46 +00:00
2005-09-20 13:26:39 +00:00
return 1 ;
}
2008-01-13 16:47:47 +00:00
/**
2008-01-09 01:23:38 +00:00
Parse the argument list , return the index of the first non - switch
arguments .
*/
2012-01-15 06:32:45 +00:00
static int fish_parse_opt ( int argc , char * * argv , const char * * cmd_ptr )
2005-09-20 13:26:39 +00:00
{
int my_optind ;
2006-10-25 20:54:43 +00:00
int force_interactive = 0 ;
2011-12-27 03:18:46 +00:00
2005-09-20 13:26:39 +00:00
while ( 1 )
{
static struct option
long_options [ ] =
{
{
2011-12-27 03:18:46 +00:00
" command " , required_argument , 0 , ' c '
2005-09-20 13:26:39 +00:00
}
,
2006-02-22 15:41:52 +00:00
{
2011-12-27 03:18:46 +00:00
" debug-level " , required_argument , 0 , ' d '
2006-02-22 15:41:52 +00:00
}
,
2005-09-20 13:26:39 +00:00
{
2011-12-27 03:18:46 +00:00
" interactive " , no_argument , 0 , ' i '
2005-09-20 13:26:39 +00:00
}
,
2006-02-03 00:37:54 +00:00
{
2011-12-27 03:18:46 +00:00
" login " , no_argument , 0 , ' l '
2006-02-03 00:37:54 +00:00
}
,
2006-03-18 01:04:59 +00:00
{
2011-12-27 03:18:46 +00:00
" no-execute " , no_argument , 0 , ' n '
2006-03-18 01:04:59 +00:00
}
,
2005-09-20 13:26:39 +00:00
{
2011-12-27 03:18:46 +00:00
" profile " , required_argument , 0 , ' p '
2005-09-20 13:26:39 +00:00
}
,
{
2011-12-27 03:18:46 +00:00
" help " , no_argument , 0 , ' h '
2005-09-20 13:26:39 +00:00
}
,
{
2011-12-27 03:18:46 +00:00
" version " , no_argument , 0 , ' v '
2005-09-20 13:26:39 +00:00
}
,
2011-12-27 03:18:46 +00:00
{
0 , 0 , 0 , 0
2005-09-20 13:26:39 +00:00
}
}
;
2011-12-27 03:18:46 +00:00
2005-09-20 13:26:39 +00:00
int opt_index = 0 ;
2011-12-27 03:18:46 +00:00
2005-09-20 13:26:39 +00:00
int opt = getopt_long ( argc ,
2011-12-27 03:18:46 +00:00
argv ,
2006-05-18 13:00:39 +00:00
GETOPT_STRING ,
2011-12-27 03:18:46 +00:00
long_options ,
2005-09-20 13:26:39 +00:00
& opt_index ) ;
2011-12-27 03:18:46 +00:00
2005-09-20 13:26:39 +00:00
if ( opt = = - 1 )
break ;
2011-12-27 03:18:46 +00:00
2005-09-20 13:26:39 +00:00
switch ( opt )
{
case 0 :
2006-07-22 10:16:51 +00:00
{
2005-09-20 13:26:39 +00:00
break ;
2006-07-22 10:16:51 +00:00
}
2011-12-27 03:18:46 +00:00
case ' c ' :
2006-07-22 10:16:51 +00:00
{
2011-12-27 03:18:46 +00:00
* cmd_ptr = optarg ;
2005-09-20 13:26:39 +00:00
is_interactive_session = 0 ;
break ;
2006-07-22 10:16:51 +00:00
}
2011-12-27 03:18:46 +00:00
case ' d ' :
2006-02-22 15:41:52 +00:00
{
char * end ;
2007-01-09 13:41:17 +00:00
int tmp ;
errno = 0 ;
tmp = strtol ( optarg , & end , 10 ) ;
2011-12-27 03:18:46 +00:00
2007-01-09 13:41:17 +00:00
if ( tmp > = 0 & & tmp < = 10 & & ! * end & & ! errno )
2006-02-22 15:41:52 +00:00
{
debug_level = tmp ;
}
else
{
debug ( 0 , _ ( L " Invalid value '%s' for debug level switch " ) , optarg ) ;
2012-02-28 23:11:46 +00:00
exit_without_destructors ( 1 ) ;
2006-02-22 15:41:52 +00:00
}
break ;
}
2011-12-27 03:18:46 +00:00
2005-09-20 13:26:39 +00:00
case ' h ' :
2006-07-22 10:16:51 +00:00
{
2007-01-17 13:12:46 +00:00
* cmd_ptr = " __fish_print_help fish " ;
2005-09-20 13:26:39 +00:00
break ;
2006-07-22 10:16:51 +00:00
}
2011-12-27 03:18:46 +00:00
2005-09-20 13:26:39 +00:00
case ' i ' :
2006-07-22 10:16:51 +00:00
{
2005-09-20 13:26:39 +00:00
force_interactive = 1 ;
2011-12-27 03:18:46 +00:00
break ;
2006-07-22 10:16:51 +00:00
}
2011-12-27 03:18:46 +00:00
2006-02-03 00:37:54 +00:00
case ' l ' :
2006-07-22 10:16:51 +00:00
{
2006-02-03 00:37:54 +00:00
is_login = 1 ;
2011-12-27 03:18:46 +00:00
break ;
2006-07-22 10:16:51 +00:00
}
2011-12-27 03:18:46 +00:00
2006-03-18 01:04:59 +00:00
case ' n ' :
2006-07-22 10:16:51 +00:00
{
2006-03-18 01:04:59 +00:00
no_exec = 1 ;
2011-12-27 03:18:46 +00:00
break ;
2006-07-22 10:16:51 +00:00
}
2011-12-27 03:18:46 +00:00
2005-09-20 13:26:39 +00:00
case ' p ' :
2006-07-22 10:16:51 +00:00
{
2005-09-20 13:26:39 +00:00
profile = optarg ;
2011-12-27 03:18:46 +00:00
break ;
2006-07-22 10:16:51 +00:00
}
2011-12-27 03:18:46 +00:00
2005-09-20 13:26:39 +00:00
case ' v ' :
2006-07-22 10:16:51 +00:00
{
2011-12-27 03:18:46 +00:00
fwprintf ( stderr ,
_ ( L " %s, version %s \n " ) ,
2005-09-20 13:26:39 +00:00
PACKAGE_NAME ,
PACKAGE_VERSION ) ;
2012-02-28 23:11:46 +00:00
exit_without_destructors ( 0 ) ;
2006-07-22 10:16:51 +00:00
}
2011-12-27 03:18:46 +00:00
2005-09-20 13:26:39 +00:00
case ' ? ' :
2006-07-22 10:16:51 +00:00
{
2012-02-28 23:11:46 +00:00
exit_without_destructors ( 1 ) ;
2006-07-22 10:16:51 +00:00
}
2011-12-27 03:18:46 +00:00
}
2005-09-20 13:26:39 +00:00
}
my_optind = optind ;
2011-12-27 03:18:46 +00:00
2006-02-19 01:14:32 +00:00
is_login | = ( strcmp ( argv [ 0 ] , " -fish " ) = = 0 ) ;
2011-12-27 03:18:46 +00:00
2008-01-09 01:23:38 +00:00
/*
We are an interactive session if we have not been given an
explicit command to execute , _and_ stdin is a tty .
*/
2006-10-25 20:54:43 +00:00
is_interactive_session & = ( * cmd_ptr = = 0 ) ;
2005-09-20 13:26:39 +00:00
is_interactive_session & = ( my_optind = = argc ) ;
2011-12-27 03:18:46 +00:00
is_interactive_session & = isatty ( STDIN_FILENO ) ;
2008-01-09 01:23:38 +00:00
/*
We are also an interactive session if we have are forced -
*/
2006-02-03 00:37:54 +00:00
is_interactive_session | = force_interactive ;
2006-01-04 12:51:02 +00:00
2006-10-25 20:54:43 +00:00
return my_optind ;
}
/**
Calls a bunch of init functions , parses the init files and then
parses commands from stdin or files , depending on arguments
*/
2012-03-06 23:12:37 +00:00
extern int g_fork_count ;
2006-10-25 20:54:43 +00:00
int main ( int argc , char * * argv )
2012-03-06 23:12:37 +00:00
{
2012-01-02 21:40:03 +00:00
struct stat tmp ;
2006-10-25 20:54:43 +00:00
int res = 1 ;
2012-01-15 06:32:45 +00:00
const char * cmd = 0 ;
2006-10-25 20:54:43 +00:00
int my_optind = 0 ;
2012-01-05 21:58:48 +00:00
set_main_thread ( ) ;
2012-02-28 02:43:24 +00:00
setup_fork_guards ( ) ;
2006-10-25 20:54:43 +00:00
wsetlocale ( LC_ALL , L " " ) ;
is_interactive_session = 1 ;
program_name = L " fish " ;
2012-01-03 18:54:06 +00:00
stat ( " ----------FISH_HIT_MAIN---------- " , & tmp ) ;
2006-11-11 10:48:40 +00:00
2006-10-25 20:54:43 +00:00
my_optind = fish_parse_opt ( argc , argv , & cmd ) ;
2006-03-18 01:04:59 +00:00
/*
No - exec is prohibited when in interactive mode
*/
if ( is_interactive_session & & no_exec )
{
debug ( 1 , _ ( L " Can not use the no-execute mode when running an interactive session " ) ) ;
no_exec = 0 ;
}
2012-07-18 17:50:38 +00:00
const struct config_paths_t paths = determine_config_directory_paths ( argv [ 0 ] ) ;
2012-07-08 22:20:39 +00:00
2011-12-27 03:18:46 +00:00
proc_init ( ) ;
event_init ( ) ;
2005-11-02 15:41:59 +00:00
wutil_init ( ) ;
2012-01-23 04:47:13 +00:00
//parser_init();
2005-09-20 13:26:39 +00:00
builtin_init ( ) ;
function_init ( ) ;
2012-07-18 17:50:38 +00:00
env_init ( & paths ) ;
2005-10-25 09:39:45 +00:00
reader_init ( ) ;
2006-04-19 09:55:02 +00:00
history_init ( ) ;
2006-03-10 13:38:09 +00:00
2012-01-23 05:40:08 +00:00
parser_t & parser = parser_t : : principal_parser ( ) ;
2012-03-06 23:51:48 +00:00
if ( g_log_forks )
printf ( " %d: g_fork_count: %d \n " , __LINE__ , g_fork_count ) ;
2012-03-06 23:12:37 +00:00
2012-07-08 22:20:39 +00:00
if ( read_init ( paths ) )
2005-09-20 13:26:39 +00:00
{
if ( cmd ! = 0 )
{
wchar_t * cmd_wcs = str2wcs ( cmd ) ;
2012-01-23 04:47:13 +00:00
res = parser . eval ( cmd_wcs , 0 , TOP ) ;
2005-09-20 13:26:39 +00:00
free ( cmd_wcs ) ;
2006-05-14 10:16:23 +00:00
reader_exit ( 0 , 0 ) ;
2005-09-20 13:26:39 +00:00
}
else
{
if ( my_optind = = argc )
{
2010-09-18 12:26:54 +00:00
res = reader_read ( STDIN_FILENO , 0 ) ;
2005-09-20 13:26:39 +00:00
}
else
{
2011-12-27 03:18:46 +00:00
char * * ptr ;
2010-10-03 03:46:26 +00:00
char * file = * ( argv + ( my_optind + + ) ) ;
2011-12-27 03:18:46 +00:00
int i ;
2005-10-19 12:07:44 +00:00
int fd ;
2006-02-02 15:23:56 +00:00
wchar_t * rel_filename , * abs_filename ;
2012-03-02 08:27:40 +00:00
2005-10-19 12:07:44 +00:00
if ( ( fd = open ( file , O_RDONLY ) ) = = - 1 )
2005-09-20 13:26:39 +00:00
{
wperror ( L " open " ) ;
return 1 ;
}
2012-03-02 08:27:40 +00:00
// OK to not do this atomically since we cannot have gone multithreaded yet
set_cloexec ( fd ) ;
2010-10-03 03:46:26 +00:00
if ( * ( argv + my_optind ) )
2005-09-20 13:26:39 +00:00
{
2012-02-22 19:07:34 +00:00
wcstring sb ;
2010-10-03 03:46:26 +00:00
for ( i = 1 , ptr = argv + my_optind ; * ptr ; i + + , ptr + + )
2005-09-20 13:26:39 +00:00
{
if ( i ! = 1 )
2012-02-22 19:07:34 +00:00
sb . append ( ARRAY_SEP_STR ) ;
sb . append ( str2wcstring ( * ptr ) ) ;
2005-09-20 13:26:39 +00:00
}
2011-12-27 03:18:46 +00:00
2012-02-22 19:07:34 +00:00
env_set ( L " argv " , sb . c_str ( ) , 0 ) ;
2005-09-20 13:26:39 +00:00
}
2006-02-02 15:23:56 +00:00
rel_filename = str2wcs ( file ) ;
2006-02-03 00:38:55 +00:00
abs_filename = wrealpath ( rel_filename , 0 ) ;
2007-09-08 22:24:53 +00:00
2006-02-03 00:38:55 +00:00
if ( ! abs_filename )
2007-09-08 22:24:53 +00:00
{
2006-02-03 00:38:55 +00:00
abs_filename = wcsdup ( rel_filename ) ;
2007-09-08 22:24:53 +00:00
}
2006-02-02 15:23:56 +00:00
reader_push_current_filename ( intern ( abs_filename ) ) ;
free ( rel_filename ) ;
free ( abs_filename ) ;
2006-02-03 00:37:54 +00:00
2007-04-25 18:30:02 +00:00
res = reader_read ( fd , 0 ) ;
2005-09-20 13:26:39 +00:00
if ( res )
{
2011-12-27 03:18:46 +00:00
debug ( 1 ,
_ ( L " Error while reading file %ls \n " ) ,
2007-09-08 22:24:53 +00:00
reader_current_filename ( ) ? reader_current_filename ( ) : _ ( L " Standard input " ) ) ;
2011-12-27 03:18:46 +00:00
}
2006-02-02 15:23:56 +00:00
reader_pop_current_filename ( ) ;
2005-09-20 13:26:39 +00:00
}
}
}
2011-12-27 03:18:46 +00:00
2006-05-14 10:16:23 +00:00
proc_fire_event ( L " PROCESS_EXIT " , EVENT_EXIT , getpid ( ) , res ) ;
2011-12-27 03:18:46 +00:00
2006-04-19 09:55:02 +00:00
history_destroy ( ) ;
2005-09-20 13:26:39 +00:00
proc_destroy ( ) ;
builtin_destroy ( ) ;
reader_destroy ( ) ;
2012-01-23 05:40:08 +00:00
parser . destroy ( ) ;
2005-09-20 13:26:39 +00:00
wutil_destroy ( ) ;
2005-10-05 22:37:08 +00:00
event_destroy ( ) ;
2011-12-27 03:18:46 +00:00
2006-07-22 10:16:51 +00:00
env_destroy ( ) ;
2011-12-27 03:18:46 +00:00
2012-03-06 23:51:48 +00:00
if ( g_log_forks )
printf ( " %d: g_fork_count: %d \n " , __LINE__ , g_fork_count ) ;
2012-03-06 23:12:37 +00:00
2011-12-27 03:18:46 +00:00
return res ? STATUS_UNKNOWN_COMMAND : proc_get_last_status ( ) ;
2005-09-20 13:26:39 +00:00
}