2005-09-20 21:26:39 +08:00
/** \file parser.c
The fish parser . Contains functions for parsing code .
*/
# include <stdlib.h>
# include <stdio.h>
# include <wchar.h>
# include <sys/types.h>
# include <sys/stat.h>
# include <unistd.h>
# include <errno.h>
# include <fcntl.h>
# include <termios.h>
# include <pwd.h>
# include <dirent.h>
# include <signal.h>
# include "config.h"
# include "util.h"
# include "common.h"
# include "wutil.h"
# include "proc.h"
# include "parser.h"
# include "tokenizer.h"
# include "exec.h"
# include "wildcard.h"
# include "function.h"
# include "builtin.h"
# include "builtin_help.h"
# include "env.h"
# include "expand.h"
# include "reader.h"
# include "sanity.h"
# include "env_universal.h"
2005-10-06 06:37:08 +08:00
# include "event.h"
2006-01-04 20:51:02 +08:00
# include "translate.h"
2005-09-20 21:26:39 +08:00
/**
Maximum number of block levels in code . This is not the same as
maximum recursion depth , this only has to do with how many block
levels are legal in the source code , not at evaluation .
*/
# define BLOCK_MAX_COUNT 64
/**
Maximum number of function calls , i . e . recursion depth .
*/
# define MAX_RECURSION_DEPTH 128
2005-10-24 23:26:25 +08:00
/**
Message about reporting bugs , used on weird internal error to
hopefully get them to report stuff .
*/
2006-01-04 20:51:02 +08:00
# define BUGREPORT_MSG _( L"If this error can be reproduced, please send a bug report to %s.")
2005-10-09 19:48:16 +08:00
2005-09-20 21:26:39 +08:00
/**
Error message for improper use of the exec builtin
*/
2006-01-04 20:51:02 +08:00
# define EXEC_ERR_MSG _(L"This command can not be used in a pipeline")
2005-09-20 21:26:39 +08:00
/**
Error message for tokenizer error . The tokenizer message is
appended to this message .
*/
2006-01-04 20:51:02 +08:00
# define TOK_ERR_MSG _( L"Tokenizer error: '%ls'")
2005-09-20 21:26:39 +08:00
/**
Error message for short circut command error .
*/
2006-01-04 20:51:02 +08:00
# define COND_ERR_MSG _( L"Short circut command requires additional command")
2005-09-20 21:26:39 +08:00
/**
Error message on reaching maximum recusrion depth
*/
2006-01-04 20:51:02 +08:00
# define RECURSION_ERR_MSG _( L"Maximum recursion depth reached. Accidental infinite loop?")
2005-09-20 21:26:39 +08:00
2005-10-08 10:00:08 +08:00
/**
Error message used when the end of a block can ' t be located
*/
2006-01-11 22:17:35 +08:00
# define BLOCK_END_ERR_MSG _( L"Could not locate end of block. The 'end' command is missing, misspelled or a ';' is missing.")
2005-10-08 08:23:49 +08:00
2005-09-20 21:26:39 +08:00
/**
Error message on reaching maximum number of block calls
*/
2006-01-04 20:51:02 +08:00
# define BLOCK_ERR_MSG _( L"Maximum number of nested blocks reached.")
2005-09-20 21:26:39 +08:00
/**
2005-12-14 11:39:39 +08:00
Error message when a non - string token is found when expecting a command name
2005-09-20 21:26:39 +08:00
*/
2006-01-11 22:17:35 +08:00
# define CMD_ERR_MSG _( L"Expected a command name, got token of type '%ls'")
2005-09-20 21:26:39 +08:00
2006-01-06 00:02:28 +08:00
/**
Error message when a non - string token is found when expecting a command name
*/
2006-01-11 22:17:35 +08:00
# define CMD_OR_ERR_MSG _( L"Expected a command name, got token of type '%ls'. Did you mean 'COMMAND; or COMMAND'? For more information on the 'or' builtin command, see the help section for 'or' by typing 'help or'.")
2006-01-06 00:02:28 +08:00
/**
Error message when a non - string token is found when expecting a command name
*/
2006-01-11 22:17:35 +08:00
# define CMD_AND_ERR_MSG _( L"Expected a command name, got token of type '%ls'. Did you mean 'COMMAND; and COMMAND'? For more information on the 'and' builtin command, see the help section for 'and' by typing 'help and'.")
2006-01-06 00:02:28 +08:00
2005-09-20 21:26:39 +08:00
/**
2005-12-14 11:39:39 +08:00
Error message when encountering an illegal command name
2005-09-20 21:26:39 +08:00
*/
2006-01-11 22:17:35 +08:00
# define ILLEGAL_CMD_ERR_MSG _( L"Illegal command name '%ls'")
2005-09-20 21:26:39 +08:00
2005-12-04 00:43:56 +08:00
/**
Error message for wildcards with no matches
*/
2006-01-04 20:51:02 +08:00
# define WILDCARD_ERR_MSG _( L"Warning: No match for wildcard '%ls'. The command will not be executed.")
2005-12-08 00:06:47 +08:00
2005-12-14 02:28:03 +08:00
/**
Error when using case builtin outside of switch block
*/
2006-01-04 20:51:02 +08:00
# define INVALID_CASE_ERR_MSG _( L"'case' builtin not inside of switch block")
2005-12-14 02:28:03 +08:00
/**
Error when using loop control builtins ( break or continue ) outside of loop
*/
2006-01-04 20:51:02 +08:00
# define INVALID_LOOP_ERR_MSG _( L"Loop control command while not inside of loop" )
2005-12-14 02:28:03 +08:00
/**
Error when using else builtin outside of if block
*/
2006-01-04 20:51:02 +08:00
# define INVALID_ELSE_ERR_MSG _( L"'else' builtin not inside of if block" )
2005-12-14 02:28:03 +08:00
2005-12-14 04:11:21 +08:00
/**
Error when using end builtin outside of block
*/
2006-01-04 20:51:02 +08:00
# define INVALID_END_ERR_MSG _( L"'end' command outside of block")
2005-12-14 04:11:21 +08:00
2005-12-08 00:06:47 +08:00
/**
Error message for Posix - style assignment
*/
2006-01-11 22:17:35 +08:00
# define COMMAND_ASSIGN_ERR_MSG _( L"Unknown command '%ls'. Did you mean 'set VARIABLE VALUE'? For information on setting variable values, see the help section on the set command by typing 'help set'.")
2005-12-04 00:43:56 +08:00
2005-12-14 09:07:12 +08:00
/**
Error for invalid redirection token
*/
2006-01-04 20:51:02 +08:00
# define REDIRECT_TOKEN_ERR_MSG _( L"Expected redirection specification, got token of type '%ls'")
2005-12-14 11:39:39 +08:00
/**
Error when encountering redirection without a command
*/
2006-01-04 20:51:02 +08:00
# define INVALID_REDIRECTION_ERR_MSG _( L"Encountered redirection when expecting a command name. Fish does not allow a redirection operation before a command.")
/**
Error for evaluating null pointer
*/
# define EVAL_NULL_ERR_MSG _( L"Tried to evaluate null pointer." )
/**
Error for evaluating in illegal scope
*/
2006-01-11 22:17:35 +08:00
# define INVALID_SCOPE_ERR_MSG _( L"Tried to evaluate commands using invalid block type '%ls'" )
2006-01-04 20:51:02 +08:00
2005-12-14 11:39:39 +08:00
/**
Error for wrong token type
*/
2006-01-04 20:51:02 +08:00
# define UNEXPECTED_TOKEN_ERR_MSG _( L"Unexpected token of type '%ls'")
/**
Unexpected error in parser_get_filename ( )
*/
# define MISSING_COMMAND_ERR_MSG _( L"Error while searching for command '%ls'" )
/**
While block description
*/
# define WHILE_BLOCK _( L"'while' block" )
/**
For block description
*/
# define FOR_BLOCK _( L"'for' block" )
/**
If block description
*/
# define IF_BLOCK _( L"'if' conditional block" )
/**
function definition block description
*/
# define FUNCTION_DEF_BLOCK _( L"function definition block" )
/**
Function invocation block description
*/
# define FUNCTION_CALL_BLOCK _( L"function invocation block" )
/**
Switch block description
*/
# define SWITCH_BLOCK _( L"'switch' block" )
/**
Fake block description
*/
# define FAKE_BLOCK _( L"unexecutable block" )
/**
Top block description
*/
# define TOP_BLOCK _( L"global root block" )
/**
Command substitution block description
*/
# define SUBST_BLOCK _( L"command substitution block" )
/**
Begin block description
*/
2006-01-11 22:17:35 +08:00
# define BEGIN_BLOCK _( L"'begin' unconditional block" )
2006-01-04 20:51:02 +08:00
/**
Unknown block description
*/
# define UNKNOWN_BLOCK _( L"unknown / invalid block" )
2005-12-14 09:07:12 +08:00
2005-09-20 21:26:39 +08:00
/** Last error code */
int error_code ;
2005-12-12 06:21:01 +08:00
event_block_t * global_event_block = 0 ;
2005-09-20 21:26:39 +08:00
/** Position of last error */
static int err_pos ;
/** Description of last error */
static wchar_t err_str [ 256 ] ;
/** Pointer to the current tokenizer */
static tokenizer * current_tokenizer ;
/** String for representing the current line */
2006-01-12 20:54:57 +08:00
static string_buffer_t * lineinfo = 0 ;
2005-09-20 21:26:39 +08:00
/** This is the position of the beginning of the currently parsed command */
static int current_tokenizer_pos ;
/** The current innermost block */
block_t * current_block = 0 ;
/** List of called functions, used to help prevent infinite recursion */
static array_list_t forbidden_function ;
/**
String index where the current job started .
*/
static int job_start_pos ;
io_data_t * block_io ;
/**
List of all profiling data
*/
static array_list_t profile_data ;
2005-10-08 10:00:08 +08:00
/**
Keeps track of how many recursive eval calls have been made . Eval
doesn ' t call itself directly , recursion happens on blocks and on
command substitutions .
*/
2005-09-20 21:26:39 +08:00
static int eval_level = - 1 ;
static int parse_job ( process_t * p ,
job_t * j ,
tokenizer * tok ) ;
2005-10-08 10:00:08 +08:00
/**
Struct used to keep track of profiling data for a command
*/
2005-09-20 21:26:39 +08:00
typedef struct
{
2005-10-24 23:26:25 +08:00
/**
2005-12-04 00:43:56 +08:00
Time spent executing the specified command , including parse time for nested blocks .
2005-10-24 23:26:25 +08:00
*/
int exec ;
/**
2005-12-04 00:43:56 +08:00
Time spent parsing the specified command , including execution time for command substitutions .
2005-10-24 23:26:25 +08:00
*/
int parse ;
/**
2005-12-04 00:43:56 +08:00
The block level of the specified command . nested blocks and command substitutions both increase the block level .
2005-10-24 23:26:25 +08:00
*/
int level ;
/**
2005-12-04 00:43:56 +08:00
If the execution of this command was skipped .
2005-10-24 23:26:25 +08:00
*/
int skipped ;
/**
2005-12-04 00:43:56 +08:00
The command string .
2005-10-24 23:26:25 +08:00
*/
2005-09-20 21:26:39 +08:00
wchar_t * cmd ;
} profile_element_t ;
int block_count ( block_t * b )
{
2005-12-31 00:29:19 +08:00
2005-09-20 21:26:39 +08:00
if ( b = = 0 )
return 0 ;
return ( block_count ( b - > outer ) + 1 ) ;
}
void parser_push_block ( int type )
{
2005-12-12 06:21:01 +08:00
block_t * new = calloc ( 1 , sizeof ( block_t ) ) ;
2005-09-20 21:26:39 +08:00
2005-12-12 23:51:04 +08:00
debug ( 3 , L " Block push %ls %d \n " , parser_get_block_desc ( type ) , block_count ( current_block ) + 1 ) ;
2005-09-20 21:26:39 +08:00
new - > outer = current_block ;
new - > type = ( current_block & & current_block - > skip ) ? FAKE : type ;
2005-09-25 03:31:17 +08:00
/*
New blocks should be skipped if the outer block is skipped ,
2005-12-09 10:41:16 +08:00
except TOP ans SUBST block , which open up new environments . Fake
blocks should always be skipped . Rather complicated . . . : - (
2005-09-25 03:31:17 +08:00
*/
2005-09-20 21:26:39 +08:00
new - > skip = current_block ? current_block - > skip : 0 ;
2005-09-25 03:31:17 +08:00
if ( type = = TOP | | type = = SUBST )
new - > skip = 0 ;
2005-12-09 10:41:16 +08:00
if ( type = = FAKE )
new - > skip = 1 ;
2005-09-25 03:31:17 +08:00
2005-10-15 08:51:26 +08:00
new - > job = 0 ;
2005-09-20 21:26:39 +08:00
new - > loop_status = LOOP_NORMAL ;
current_block = new ;
if ( ( new - > type ! = FUNCTION_DEF ) & &
( new - > type ! = FAKE ) & &
( new - > type ! = TOP ) )
{
env_push ( type = = FUNCTION_CALL ) ;
}
}
void parser_pop_block ( )
{
2005-12-12 23:51:04 +08:00
debug ( 3 , L " Block pop %ls %d \n " , parser_get_block_desc ( current_block - > type ) , block_count ( current_block ) - 1 ) ;
2005-12-12 06:21:01 +08:00
event_block_t * eb , * eb_next ;
2005-09-20 21:26:39 +08:00
if ( ( current_block - > type ! = FUNCTION_DEF ) & &
( current_block - > type ! = FAKE ) & &
( current_block - > type ! = TOP ) )
{
env_pop ( ) ;
}
switch ( current_block - > type )
{
case FOR :
{
2005-10-12 03:31:16 +08:00
free ( current_block - > param1 . for_variable ) ;
al_foreach ( & current_block - > param2 . for_vars ,
2005-09-20 21:26:39 +08:00
( void ( * ) ( const void * ) ) & free ) ;
2005-10-12 03:31:16 +08:00
al_destroy ( & current_block - > param2 . for_vars ) ;
2005-09-20 21:26:39 +08:00
break ;
}
case SWITCH :
{
2005-10-12 03:31:16 +08:00
free ( current_block - > param1 . switch_value ) ;
2005-09-20 21:26:39 +08:00
break ;
}
case FUNCTION_DEF :
{
2005-10-12 03:31:16 +08:00
free ( current_block - > param1 . function_name ) ;
free ( current_block - > param2 . function_description ) ;
al_foreach ( current_block - > param4 . function_events ,
2005-10-06 06:37:08 +08:00
( void ( * ) ( const void * ) ) & event_free ) ;
2005-10-12 03:31:16 +08:00
al_destroy ( current_block - > param4 . function_events ) ;
free ( current_block - > param4 . function_events ) ;
2005-09-20 21:26:39 +08:00
break ;
}
}
2005-12-12 06:21:01 +08:00
for ( eb = current_block - > first_event_block ; eb ; eb = eb_next )
{
eb_next = eb - > next ;
free ( eb ) ;
}
2005-09-20 21:26:39 +08:00
block_t * old = current_block ;
current_block = current_block - > outer ;
free ( old ) ;
}
2006-01-04 20:51:02 +08:00
const wchar_t * parser_get_block_desc ( int block )
2005-09-20 21:26:39 +08:00
{
switch ( block )
{
case WHILE :
2006-01-04 20:51:02 +08:00
return WHILE_BLOCK ;
2005-09-20 21:26:39 +08:00
case FOR :
2006-01-04 20:51:02 +08:00
return FOR_BLOCK ;
2005-09-20 21:26:39 +08:00
case IF :
2006-01-04 20:51:02 +08:00
return IF_BLOCK ;
2005-09-20 21:26:39 +08:00
case FUNCTION_DEF :
2006-01-04 20:51:02 +08:00
return FUNCTION_DEF_BLOCK ;
2005-09-20 21:26:39 +08:00
case FUNCTION_CALL :
2006-01-04 20:51:02 +08:00
return FUNCTION_CALL_BLOCK ;
2005-09-20 21:26:39 +08:00
case SWITCH :
2006-01-04 20:51:02 +08:00
return SWITCH_BLOCK ;
2005-09-20 21:26:39 +08:00
case FAKE :
2006-01-04 20:51:02 +08:00
return FAKE_BLOCK ;
2005-09-20 21:26:39 +08:00
case TOP :
2006-01-04 20:51:02 +08:00
return TOP_BLOCK ;
2005-09-20 21:26:39 +08:00
case SUBST :
2006-01-04 20:51:02 +08:00
return SUBST_BLOCK ;
2005-09-20 21:26:39 +08:00
case BEGIN :
2006-01-04 20:51:02 +08:00
return BEGIN_BLOCK ;
2005-09-20 21:26:39 +08:00
default :
2006-01-04 20:51:02 +08:00
return UNKNOWN_BLOCK ;
2005-09-20 21:26:39 +08:00
}
}
2005-09-25 19:25:42 +08:00
int parser_skip_arguments ( const wchar_t * cmd )
2005-09-20 21:26:39 +08:00
{
2005-09-25 19:25:42 +08:00
2005-12-14 09:07:12 +08:00
return contains_str ( cmd ,
2005-09-25 19:25:42 +08:00
L " else " ,
L " begin " ,
2005-10-01 02:28:26 +08:00
( void * ) 0 ) ;
2005-09-20 21:26:39 +08:00
}
2005-09-25 19:25:42 +08:00
int parser_is_subcommand ( const wchar_t * cmd )
{
2005-10-01 02:28:26 +08:00
2005-09-25 19:25:42 +08:00
return parser_skip_arguments ( cmd ) | |
contains_str ( cmd ,
L " command " ,
L " builtin " ,
L " while " ,
L " exec " ,
L " if " ,
L " and " ,
L " or " ,
L " not " ,
2005-10-01 02:28:26 +08:00
( void * ) 0 ) ;
2005-09-25 19:25:42 +08:00
}
2005-09-20 21:26:39 +08:00
/**
Test if the specified string is command that opens a new block
*/
static int parser_is_block ( wchar_t * word )
{
return contains_str ( word ,
L " for " ,
L " while " ,
L " if " ,
L " function " ,
L " switch " ,
L " begin " ,
2005-10-01 02:28:26 +08:00
( void * ) 0 ) ;
2005-09-20 21:26:39 +08:00
}
int parser_is_reserved ( wchar_t * word )
{
return parser_is_block ( word ) | |
parser_is_subcommand ( word ) | |
contains_str ( word ,
L " end " ,
L " case " ,
L " else " ,
L " return " ,
L " continue " ,
L " break " ,
2005-10-01 02:28:26 +08:00
( void * ) 0 ) ;
2005-09-20 21:26:39 +08:00
}
int parser_is_pipe_forbidden ( wchar_t * word )
{
return contains_str ( word ,
L " exec " ,
L " case " ,
L " break " ,
L " return " ,
L " continue " ,
2005-10-01 02:28:26 +08:00
( void * ) 0 ) ;
2005-09-20 21:26:39 +08:00
}
2005-10-08 10:00:08 +08:00
/**
Search the text for the end of the current block
*/
2005-09-20 21:26:39 +08:00
static const wchar_t * parser_find_end ( const wchar_t * buff )
{
tokenizer tok ;
int had_cmd = 0 ;
int count = 0 ;
int error = 0 ;
int mark = 0 ;
for ( tok_init ( & tok , buff , 0 ) ;
tok_has_next ( & tok ) & & ! error ;
tok_next ( & tok ) )
{
int last_type = tok_last_type ( & tok ) ;
switch ( last_type )
{
case TOK_STRING :
{
if ( ! had_cmd )
{
if ( wcscmp ( tok_last ( & tok ) , L " end " ) = = 0 )
{
count - - ;
}
else if ( parser_is_block ( tok_last ( & tok ) ) )
{
count + + ;
}
if ( count < 0 )
{
error = 1 ;
}
had_cmd = 1 ;
}
break ;
}
case TOK_END :
{
had_cmd = 0 ;
break ;
}
case TOK_PIPE :
case TOK_BACKGROUND :
{
if ( had_cmd )
{
had_cmd = 0 ;
}
else
{
error = 1 ;
}
break ;
}
case TOK_ERROR :
error = 1 ;
break ;
default :
break ;
}
if ( ! count )
{
tok_next ( & tok ) ;
mark = tok_get_pos ( & tok ) ;
break ;
}
}
tok_destroy ( & tok ) ;
if ( ! count & & ! error ) {
return buff + mark ;
}
return 0 ;
}
wchar_t * parser_cdpath_get ( wchar_t * dir )
{
wchar_t * res = 0 ;
if ( ! dir )
return 0 ;
if ( dir [ 0 ] = = L ' / ' )
{
struct stat buf ;
if ( wstat ( dir , & buf ) = = 0 )
{
if ( S_ISDIR ( buf . st_mode ) )
{
res = wcsdup ( dir ) ;
}
}
}
else
{
2005-12-14 04:10:36 +08:00
wchar_t * path ;
wchar_t * path_cpy ;
wchar_t * nxt_path ;
wchar_t * state ;
wchar_t * whole_path ;
2005-09-20 21:26:39 +08:00
2005-12-14 04:10:36 +08:00
path = env_get ( L " CDPATH " ) ;
if ( ! path | | ! wcslen ( path ) )
2005-09-20 21:26:39 +08:00
{
path = L " . " ;
}
2005-12-14 04:10:36 +08:00
nxt_path = path ;
path_cpy = wcsdup ( path ) ;
if ( ! path_cpy )
2005-09-20 21:26:39 +08:00
{
die_mem ( ) ;
}
2005-10-14 19:40:33 +08:00
2005-09-20 21:26:39 +08:00
for ( nxt_path = wcstok ( path_cpy , ARRAY_SEP_STR , & state ) ;
nxt_path ! = 0 ;
nxt_path = wcstok ( 0 , ARRAY_SEP_STR , & state ) )
{
wchar_t * expanded_path = expand_tilde ( wcsdup ( nxt_path ) ) ;
// debug( 2, L"woot %ls\n", expanded_path );
int path_len = wcslen ( expanded_path ) ;
if ( path_len = = 0 )
{
free ( expanded_path ) ;
continue ;
}
whole_path =
wcsdupcat2 ( expanded_path ,
( expanded_path [ path_len - 1 ] ! = L ' / ' ) ? L " / " : L " " ,
dir , 0 ) ;
free ( expanded_path ) ;
struct stat buf ;
if ( wstat ( whole_path , & buf ) = = 0 )
{
if ( S_ISDIR ( buf . st_mode ) )
{
res = whole_path ;
break ;
}
}
free ( whole_path ) ;
}
free ( path_cpy ) ;
}
return res ;
}
void parser_forbid_function ( wchar_t * function )
{
/*
if ( function )
debug ( 2 , L " Forbid %ls \n " , function ) ;
*/
al_push ( & forbidden_function , function ? wcsdup ( function ) : 0 ) ;
}
void parser_allow_function ( )
{
/*
if ( al_peek ( & forbidden_function ) )
debug ( 2 , L " Allow %ls \n " , al_peek ( & forbidden_function ) ) ;
*/
free ( ( void * ) al_pop ( & forbidden_function ) ) ;
}
2005-12-08 00:06:47 +08:00
void error ( int ec , int p , const wchar_t * str , . . . )
2005-09-20 21:26:39 +08:00
{
2005-12-08 00:06:47 +08:00
va_list va ;
2005-09-20 21:26:39 +08:00
error_code = ec ;
err_pos = p ;
2005-12-08 00:06:47 +08:00
va_start ( va , str ) ;
vswprintf ( err_str , 256 , str , va ) ;
va_end ( va ) ;
2005-09-20 21:26:39 +08:00
}
wchar_t * get_filename ( const wchar_t * cmd )
{
wchar_t * path ;
2005-12-12 07:39:39 +08:00
debug ( 2 , L " get_filename( '%ls' ) " , cmd ) ;
2005-09-20 21:26:39 +08:00
if ( wcschr ( cmd , ' / ' ) ! = 0 )
{
if ( waccess ( cmd , X_OK ) = = 0 )
{
struct stat buff ;
wstat ( cmd , & buff ) ;
if ( S_ISREG ( buff . st_mode ) )
return wcsdup ( cmd ) ;
else
return 0 ;
}
}
else
{
path = env_get ( L " PATH " ) ;
if ( path ! = 0 )
{
/*
Allocate string long enough to hold the whole command
*/
wchar_t * new_cmd = malloc ( sizeof ( wchar_t ) * ( wcslen ( cmd ) + wcslen ( path ) + 2 ) ) ;
/*
We tokenize a copy of the path , since strtok modifies
its arguments
*/
wchar_t * path_cpy = wcsdup ( path ) ;
wchar_t * nxt_path = path ;
wchar_t * state ;
if ( ( new_cmd = = 0 ) | | ( path_cpy = = 0 ) )
{
die_mem ( ) ;
}
for ( nxt_path = wcstok ( path_cpy , ARRAY_SEP_STR , & state ) ;
nxt_path ! = 0 ;
nxt_path = wcstok ( 0 , ARRAY_SEP_STR , & state ) )
{
int path_len = wcslen ( nxt_path ) ;
wcscpy ( new_cmd , nxt_path ) ;
if ( new_cmd [ path_len - 1 ] ! = ' / ' )
{
new_cmd [ path_len + + ] = ' / ' ;
}
wcscpy ( & new_cmd [ path_len ] , cmd ) ;
if ( waccess ( new_cmd , X_OK ) = = 0 )
{
struct stat buff ;
if ( wstat ( new_cmd , & buff ) = = - 1 )
{
if ( errno ! = EACCES )
wperror ( L " stat " ) ;
continue ;
}
if ( S_ISREG ( buff . st_mode ) )
{
free ( path_cpy ) ;
return new_cmd ;
}
}
else
{
switch ( errno )
{
case ENOENT :
case ENAMETOOLONG :
case EACCES :
case ENOTDIR :
break ;
default :
debug ( 1 ,
2006-01-04 20:51:02 +08:00
MISSING_COMMAND_ERR_MSG ,
2005-09-20 21:26:39 +08:00
new_cmd ) ;
wperror ( L " access " ) ;
}
}
}
free ( path_cpy ) ;
free ( new_cmd ) ;
}
}
return 0 ;
}
void parser_init ( )
{
if ( profile )
{
al_init ( & profile_data ) ;
}
al_init ( & forbidden_function ) ;
}
void print_profile ( array_list_t * p ,
int pos ,
FILE * out )
{
profile_element_t * me , * prev ;
int i ;
int my_time ;
if ( pos > = al_get_count ( p ) )
return ;
me = ( profile_element_t * ) al_get ( p , pos ) ;
if ( ! me - > skipped )
{
my_time = me - > parse + me - > exec ;
for ( i = pos + 1 ; i < al_get_count ( p ) ; i + + )
{
prev = ( profile_element_t * ) al_get ( p , i ) ;
if ( prev - > skipped )
continue ;
if ( prev - > level < = me - > level )
break ;
if ( prev - > level > me - > level + 1 )
continue ;
my_time - = prev - > parse ;
my_time - = prev - > exec ;
}
if ( me - > cmd )
{
fwprintf ( out , L " %d \t %d \t " , my_time , me - > parse + me - > exec ) ;
for ( i = 0 ; i < me - > level ; i + + )
{
fwprintf ( out , L " - " ) ;
}
fwprintf ( out , L " > %ls \n " , me - > cmd ) ;
}
}
print_profile ( p , pos + 1 , out ) ;
free ( me - > cmd ) ;
free ( me ) ;
}
void parser_destroy ( )
{
if ( profile )
{
/*
Save profiling information
*/
FILE * f = fopen ( profile , " w " ) ;
if ( ! f )
{
debug ( 1 ,
2006-01-04 20:51:02 +08:00
_ ( L " Could not write profiling information to file '%s' " ) ,
2005-09-20 21:26:39 +08:00
profile ) ;
}
else
{
fwprintf ( f ,
2006-01-04 20:51:02 +08:00
_ ( L " Time \t Sum \t Command \n " ) ,
2005-09-20 21:26:39 +08:00
al_get_count ( & profile_data ) ) ;
print_profile ( & profile_data , 0 , f ) ;
fclose ( f ) ;
}
al_destroy ( & profile_data ) ;
}
al_destroy ( & forbidden_function ) ;
}
2005-10-08 10:00:08 +08:00
/**
Print error message if an error has occured while parsing
*/
2005-09-20 21:26:39 +08:00
static void print_errors ( )
{
if ( error_code )
{
int tmp ;
debug ( 0 , L " %ls " , err_str ) ;
tmp = current_tokenizer_pos ;
current_tokenizer_pos = err_pos ;
fwprintf ( stderr , L " %ls " , parser_current_line ( ) ) ;
current_tokenizer_pos = tmp ;
}
}
int eval_args ( const wchar_t * line , array_list_t * args )
{
tokenizer tok ;
/*
eval_args may be called while evaulating another command , so we
save the previous tokenizer and restore it on exit
*/
tokenizer * previous_tokenizer = current_tokenizer ;
int previous_pos = current_tokenizer_pos ;
int do_loop = 1 ;
tok_init ( & tok , line , 0 ) ;
current_tokenizer = & tok ;
error_code = 0 ;
for ( ; do_loop & & tok_has_next ( & tok ) ; tok_next ( & tok ) )
{
current_tokenizer_pos = tok_get_pos ( & tok ) ;
switch ( tok_last_type ( & tok ) )
{
case TOK_STRING :
2005-12-04 00:43:56 +08:00
switch ( expand_string ( wcsdup ( tok_last ( & tok ) ) , args , 0 ) )
2005-09-20 21:26:39 +08:00
{
2005-12-04 00:43:56 +08:00
case EXPAND_ERROR :
{
err_pos = tok_get_pos ( & tok ) ;
do_loop = 0 ;
break ;
}
default :
{
break ;
}
2005-09-20 21:26:39 +08:00
}
break ;
case TOK_END :
break ;
case TOK_ERROR :
{
2005-12-08 00:06:47 +08:00
error ( SYNTAX_ERROR ,
tok_get_pos ( & tok ) ,
TOK_ERR_MSG ,
tok_last ( & tok ) ) ;
2005-09-20 21:26:39 +08:00
do_loop = 0 ;
break ;
}
default :
2005-12-08 00:06:47 +08:00
error ( SYNTAX_ERROR ,
tok_get_pos ( & tok ) ,
2005-12-14 11:39:39 +08:00
UNEXPECTED_TOKEN_ERR_MSG ,
2005-12-08 00:06:47 +08:00
tok_get_desc ( tok_last_type ( & tok ) ) ) ;
2005-09-20 21:26:39 +08:00
do_loop = 0 ;
break ;
}
}
print_errors ( ) ;
tok_destroy ( & tok ) ;
current_tokenizer = previous_tokenizer ;
current_tokenizer_pos = previous_pos ;
return 1 ;
}
wchar_t * parser_current_line ( )
{
int lineno = 1 ;
wchar_t * file = reader_current_filename ( ) ;
wchar_t * whole_str = tok_string ( current_tokenizer ) ;
wchar_t * line = whole_str ;
wchar_t * line_end ;
int i ;
int offset ;
int current_line_pos = current_tokenizer_pos ;
2006-01-12 20:54:57 +08:00
2005-09-20 21:26:39 +08:00
if ( ! line )
return L " " ;
2006-01-12 20:54:57 +08:00
if ( ! lineinfo )
{
lineinfo = malloc ( sizeof ( string_buffer_t ) ) ;
sb_init ( lineinfo ) ;
}
sb_clear ( lineinfo ) ;
2005-09-20 21:26:39 +08:00
/*
Calculate line number , line offset , etc .
*/
for ( i = 0 ; i < current_tokenizer_pos ; i + + )
{
if ( whole_str [ i ] = = L ' \n ' )
{
lineno + + ;
current_line_pos = current_tokenizer_pos - i - 1 ;
line = & whole_str [ i + 1 ] ;
}
}
/*
Copy current line from whole string
*/
line_end = wcschr ( line , L ' \n ' ) ;
if ( ! line_end )
line_end = line + wcslen ( line ) ;
line = wcsndup ( line , line_end - line ) ;
2006-01-12 20:54:57 +08:00
debug ( 4 , L " Current pos %d, line pos %d, file_length %d, is_interactive %d \n " , current_tokenizer_pos , current_line_pos , wcslen ( whole_str ) , is_interactive ) ;
2005-09-20 21:26:39 +08:00
if ( ! is_interactive )
{
2006-01-12 20:54:57 +08:00
sb_printf ( lineinfo ,
_ ( L " %ls (line %d): " ) ,
file ,
lineno ) ;
offset = my_wcswidth ( ( wchar_t * ) lineinfo - > buff ) ;
2005-09-20 21:26:39 +08:00
}
else
{
offset = 0 ;
}
2005-12-14 09:07:12 +08:00
/*
Skip printing character position if we are in interactive mode
and the error was on the first character of the line
*/
2006-01-12 20:54:57 +08:00
if ( ! is_interactive | | ( current_line_pos ! = 0 ) )
{
sb_printf ( lineinfo ,
L " %ls \n %*c^ \n " ,
line ,
offset + current_line_pos ,
L ' ' ) ;
}
2005-09-20 21:26:39 +08:00
free ( line ) ;
2006-01-12 20:54:57 +08:00
return ( wchar_t * ) lineinfo - > buff ;
2005-09-20 21:26:39 +08:00
}
int parser_get_pos ( )
{
return tok_get_pos ( current_tokenizer ) ;
}
int parser_get_job_pos ( )
{
return job_start_pos ;
}
void parser_set_pos ( int p )
{
tok_set_pos ( current_tokenizer , p ) ;
}
const wchar_t * parser_get_buffer ( )
{
return tok_string ( current_tokenizer ) ;
}
int parser_is_help ( wchar_t * s , int min_match )
{
int len = wcslen ( s ) ;
return ( wcscmp ( L " -h " , s ) = = 0 ) | |
( len > = 3 & & ( wcsncmp ( L " --help " , s , len ) = = 0 ) ) ;
}
/**
Parse options for the specified job
\ param p the process to parse options for
\ param j the job to which the process belongs to
\ param tok the tokenizer to read options from
\ param args the argument list to insert options into
*/
static void parse_job_main_loop ( process_t * p ,
job_t * j ,
tokenizer * tok ,
array_list_t * args )
{
int is_finished = 0 ;
int proc_is_count = 0 ;
2005-12-04 00:43:56 +08:00
int matched_wildcard = 0 , unmatched_wildcard = 0 ;
wchar_t * unmatched = 0 ;
int unmatched_pos = 0 ;
2005-09-20 21:26:39 +08:00
/*
Test if this is the ' count ' command . We need to special case
count , since it should display a help message on ' count . h ' ,
but not on ' set foo - h ; count $ foo ' .
*/
if ( p - > actual_cmd )
{
wchar_t * woot = wcsrchr ( p - > actual_cmd , L ' / ' ) ;
if ( ! woot )
woot = p - > actual_cmd ;
else
woot + + ;
proc_is_count = wcscmp ( woot , L " count " ) = = 0 ;
}
while ( 1 )
{
/* debug( 2, L"Read token %ls\n", wcsdup(tok_last( tok )) ); */
switch ( tok_last_type ( tok ) )
{
case TOK_PIPE :
if ( ( p - > type = = INTERNAL_EXEC ) )
{
error ( SYNTAX_ERROR ,
2005-12-08 00:06:47 +08:00
tok_get_pos ( tok ) ,
EXEC_ERR_MSG ) ;
2005-09-20 21:26:39 +08:00
return ;
}
2005-10-07 22:08:57 +08:00
p - > pipe_fd = wcstol ( tok_last ( tok ) , 0 , 10 ) ;
2005-09-20 21:26:39 +08:00
p - > argv = list_to_char_arr ( args ) ;
p - > next = calloc ( 1 , sizeof ( process_t ) ) ;
if ( p - > next = = 0 )
{
die_mem ( ) ;
}
tok_next ( tok ) ;
if ( ! parse_job ( p - > next , j , tok ) )
{
/*
2005-09-25 03:31:17 +08:00
Don ' t do anything on failiure . parse_job will notice the error flag beeing set
2005-09-20 21:26:39 +08:00
*/
2005-09-25 03:31:17 +08:00
2005-09-20 21:26:39 +08:00
}
is_finished = 1 ;
break ;
case TOK_BACKGROUND :
j - > fg = 0 ;
2006-01-18 20:42:48 +08:00
j - > terminal = 0 ;
2005-09-20 21:26:39 +08:00
case TOK_END :
{
p - > argv = list_to_char_arr ( args ) ;
if ( tok_has_next ( tok ) )
tok_next ( tok ) ;
is_finished = 1 ;
break ;
}
case TOK_STRING :
{
int skip = 0 ;
if ( current_block - > skip )
{
2005-09-25 03:31:17 +08:00
/*
If this command should be skipped , we do not expand the arguments
*/
2005-09-20 21:26:39 +08:00
skip = 1 ;
2005-09-25 03:31:17 +08:00
/*
But if this is in fact a case statement , then it should be evaluated
*/
2005-09-20 21:26:39 +08:00
if ( ( current_block - > type = = SWITCH ) & &
( wcscmp ( al_get ( args , 0 ) , L " case " ) = = 0 ) & &
2005-09-25 03:31:17 +08:00
p - > type = = INTERNAL_BUILTIN )
2005-09-20 21:26:39 +08:00
{
skip = 0 ;
}
}
if ( ! skip )
{
if ( proc_is_count & &
( al_get_count ( args ) = = 1 ) & &
( parser_is_help ( tok_last ( tok ) , 0 ) ) )
{
/*
Display help for count
*/
p - > type = INTERNAL_BUILTIN ;
wcscpy ( p - > actual_cmd , L " count " ) ;
}
2005-09-25 03:31:17 +08:00
2005-12-04 00:43:56 +08:00
switch ( expand_string ( wcsdup ( tok_last ( tok ) ) , args , 0 ) )
2005-09-20 21:26:39 +08:00
{
2005-12-04 00:43:56 +08:00
case EXPAND_ERROR :
2005-09-20 21:26:39 +08:00
{
2005-12-04 00:43:56 +08:00
err_pos = tok_get_pos ( tok ) ;
if ( error_code = = 0 )
{
2005-12-08 00:06:47 +08:00
error ( SYNTAX_ERROR ,
tok_get_pos ( tok ) ,
2006-01-04 20:51:02 +08:00
_ ( L " Could not expand string '%ls' " ) ,
2005-12-08 00:06:47 +08:00
tok_last ( tok ) ) ;
2005-12-04 00:43:56 +08:00
}
break ;
2005-09-20 21:26:39 +08:00
}
2005-12-04 00:43:56 +08:00
case EXPAND_WILDCARD_NO_MATCH :
{
unmatched_wildcard = 1 ;
if ( ! unmatched )
{
unmatched = wcsdup ( tok_last ( tok ) ) ;
unmatched_pos = tok_get_pos ( tok ) ;
}
break ;
}
case EXPAND_WILDCARD_MATCH :
{
matched_wildcard = 1 ;
break ;
}
case EXPAND_OK :
{
break ;
}
2005-09-20 21:26:39 +08:00
}
2005-12-04 00:43:56 +08:00
2005-09-20 21:26:39 +08:00
}
break ;
}
case TOK_REDIRECT_OUT :
case TOK_REDIRECT_IN :
case TOK_REDIRECT_APPEND :
case TOK_REDIRECT_FD :
{
int type = tok_last_type ( tok ) ;
io_data_t * new_io ;
wchar_t * target = 0 ;
/*
Don ' t check redirections in skipped part
2005-12-14 09:07:12 +08:00
Otherwise , bogus errors may be the result . ( Do check
that token is string , though )
2005-09-20 21:26:39 +08:00
*/
if ( current_block - > skip )
{
tok_next ( tok ) ;
2005-12-14 09:07:12 +08:00
if ( tok_last_type ( tok ) ! = TOK_STRING )
{
error ( SYNTAX_ERROR ,
tok_get_pos ( tok ) ,
REDIRECT_TOKEN_ERR_MSG ,
tok_get_desc ( tok_last_type ( tok ) ) ) ;
}
2005-09-20 21:26:39 +08:00
break ;
}
new_io = calloc ( 1 , sizeof ( io_data_t ) ) ;
if ( ! new_io )
die_mem ( ) ;
new_io - > fd = wcstol ( tok_last ( tok ) ,
0 ,
10 ) ;
tok_next ( tok ) ;
switch ( tok_last_type ( tok ) )
{
case TOK_STRING :
{
target = expand_one ( wcsdup ( tok_last ( tok ) ) , 0 ) ;
if ( target = = 0 & & error_code = = 0 )
{
2005-12-08 00:06:47 +08:00
error ( SYNTAX_ERROR ,
tok_get_pos ( tok ) ,
2005-12-14 09:07:12 +08:00
REDIRECT_TOKEN_ERR_MSG ,
2005-12-08 00:06:47 +08:00
tok_last ( tok ) ) ;
2005-09-20 21:26:39 +08:00
}
}
break ;
default :
2005-12-08 00:06:47 +08:00
error ( SYNTAX_ERROR ,
tok_get_pos ( tok ) ,
2005-12-14 11:39:39 +08:00
REDIRECT_TOKEN_ERR_MSG ,
2005-12-08 00:06:47 +08:00
tok_get_desc ( tok_last_type ( tok ) ) ) ;
2005-09-20 21:26:39 +08:00
}
if ( target = = 0 | | wcslen ( target ) = = 0 )
{
if ( error_code = = 0 )
error ( SYNTAX_ERROR ,
2005-12-08 00:06:47 +08:00
tok_get_pos ( tok ) ,
2006-01-04 20:51:02 +08:00
_ ( L " Invalid IO redirection " ) ) ;
2005-09-20 21:26:39 +08:00
tok_next ( tok ) ;
}
else
{
switch ( type )
{
case TOK_REDIRECT_APPEND :
new_io - > io_mode = IO_FILE ;
2005-10-12 03:31:16 +08:00
new_io - > param2 . flags = O_CREAT | O_APPEND | O_WRONLY ;
new_io - > param1 . filename = target ;
2005-09-20 21:26:39 +08:00
break ;
case TOK_REDIRECT_OUT :
new_io - > io_mode = IO_FILE ;
2005-10-12 03:31:16 +08:00
new_io - > param2 . flags = O_CREAT | O_WRONLY | O_TRUNC ;
new_io - > param1 . filename = target ;
2005-09-20 21:26:39 +08:00
break ;
case TOK_REDIRECT_IN :
new_io - > io_mode = IO_FILE ;
2005-10-12 03:31:16 +08:00
new_io - > param2 . flags = O_RDONLY ;
new_io - > param1 . filename = target ;
2005-09-20 21:26:39 +08:00
break ;
case TOK_REDIRECT_FD :
if ( wcscmp ( target , L " - " ) = = 0 )
{
new_io - > io_mode = IO_CLOSE ;
}
else
{
new_io - > io_mode = IO_FD ;
2005-10-12 03:31:16 +08:00
new_io - > param1 . old_fd = wcstol ( target ,
2005-12-14 09:07:12 +08:00
0 ,
10 ) ;
2005-10-12 03:31:16 +08:00
if ( ( new_io - > param1 . old_fd < 0 ) | |
( new_io - > param1 . old_fd > 10 ) )
2005-09-20 21:26:39 +08:00
{
2005-12-08 00:06:47 +08:00
error ( SYNTAX_ERROR ,
tok_get_pos ( tok ) ,
2006-01-04 20:51:02 +08:00
_ ( L " Requested redirection to something that is not a file descriptor %ls " ) ,
2005-12-08 00:06:47 +08:00
target ) ;
2005-09-20 21:26:39 +08:00
tok_next ( tok ) ;
}
free ( target ) ;
}
break ;
}
}
j - > io = io_add ( j - > io , new_io ) ;
}
break ;
case TOK_ERROR :
{
2005-12-08 00:06:47 +08:00
error ( SYNTAX_ERROR ,
tok_get_pos ( tok ) ,
TOK_ERR_MSG ,
tok_last ( tok ) ) ;
2005-09-20 21:26:39 +08:00
return ;
}
default :
2005-12-08 00:06:47 +08:00
error ( SYNTAX_ERROR ,
tok_get_pos ( tok ) ,
2005-12-14 11:39:39 +08:00
UNEXPECTED_TOKEN_ERR_MSG ,
2005-12-08 00:06:47 +08:00
tok_get_desc ( tok_last_type ( tok ) ) ) ;
2005-09-20 21:26:39 +08:00
tok_next ( tok ) ;
break ;
}
if ( ( is_finished ) | | ( error_code ! = 0 ) )
break ;
tok_next ( tok ) ;
}
2005-12-04 00:43:56 +08:00
if ( ! error_code )
{
if ( unmatched_wildcard & & ! matched_wildcard )
{
2005-12-09 10:41:16 +08:00
j - > wildcard_error = 1 ;
proc_set_last_status ( 1 ) ;
if ( is_interactive & & ! is_block )
{
int tmp ;
debug ( 1 , WILDCARD_ERR_MSG , unmatched ) ;
tmp = current_tokenizer_pos ;
current_tokenizer_pos = unmatched_pos ;
fwprintf ( stderr , L " %ls " , parser_current_line ( ) ) ;
current_tokenizer_pos = tmp ;
}
2005-12-04 00:43:56 +08:00
}
}
free ( unmatched ) ;
2005-09-20 21:26:39 +08:00
return ;
}
/**
Fully parse a single job . Does not call exec on it .
\ param p The process structure that should be used to represent the first process in the job .
\ param j The job structure to contain the parsed job
\ param tok tokenizer to read from
\ return 1 on success , 0 on error
*/
static int parse_job ( process_t * p ,
job_t * j ,
tokenizer * tok )
{
array_list_t args ; // The list that will become the argc array for the program
int use_function = 1 ; // May functions be considered when checking what action this command represents
int use_builtin = 1 ; // May builtins be considered when checking what action this command represents
int is_new_block = 0 ; // Does this command create a new block?
block_t * prev_block = current_block ;
2005-12-12 07:39:39 +08:00
debug ( 2 , L " begin parse_job() \n " ) ;
2005-09-20 21:26:39 +08:00
al_init ( & args ) ;
current_tokenizer_pos = tok_get_pos ( tok ) ;
while ( al_get_count ( & args ) = = 0 )
{
wchar_t * nxt = 0 ;
switch ( tok_last_type ( tok ) )
{
case TOK_STRING :
2005-12-04 00:43:56 +08:00
{
2005-09-20 21:26:39 +08:00
nxt = expand_one ( wcsdup ( tok_last ( tok ) ) ,
EXPAND_SKIP_SUBSHELL | EXPAND_SKIP_VARIABLES ) ;
2005-12-12 07:39:39 +08:00
debug ( 2 , L " command '%ls' -> '%ls' " , tok_last ( tok ) , nxt ? nxt : L " 0 " ) ;
2005-09-20 21:26:39 +08:00
if ( nxt = = 0 )
{
2005-12-08 00:06:47 +08:00
error ( SYNTAX_ERROR ,
tok_get_pos ( tok ) ,
2005-12-14 11:39:39 +08:00
ILLEGAL_CMD_ERR_MSG ,
2005-12-08 00:06:47 +08:00
tok_last ( tok ) ) ;
2005-09-20 21:26:39 +08:00
al_destroy ( & args ) ;
return 0 ;
}
break ;
2005-12-04 00:43:56 +08:00
}
2005-09-20 21:26:39 +08:00
case TOK_ERROR :
{
2005-12-08 00:06:47 +08:00
error ( SYNTAX_ERROR ,
tok_get_pos ( tok ) ,
TOK_ERR_MSG ,
tok_last ( tok ) ) ;
2005-09-20 21:26:39 +08:00
al_destroy ( & args ) ;
return 0 ;
}
2006-01-06 00:02:28 +08:00
case TOK_PIPE :
{
wchar_t * str = tok_string ( tok ) ;
if ( tok_get_pos ( tok ) > 0 & & str [ tok_get_pos ( tok ) - 1 ] = = L ' | ' )
{
error ( SYNTAX_ERROR ,
tok_get_pos ( tok ) ,
CMD_OR_ERR_MSG ,
tok_get_desc ( tok_last_type ( tok ) ) ) ;
}
else
{
error ( SYNTAX_ERROR ,
tok_get_pos ( tok ) ,
CMD_ERR_MSG ,
tok_get_desc ( tok_last_type ( tok ) ) ) ;
}
al_destroy ( & args ) ;
return 0 ;
}
2005-09-20 21:26:39 +08:00
default :
2005-12-04 00:43:56 +08:00
{
2005-12-08 00:06:47 +08:00
error ( SYNTAX_ERROR ,
tok_get_pos ( tok ) ,
2005-12-14 11:39:39 +08:00
CMD_ERR_MSG ,
2005-12-08 00:06:47 +08:00
tok_get_desc ( tok_last_type ( tok ) ) ) ;
2005-09-20 21:26:39 +08:00
al_destroy ( & args ) ;
return 0 ;
2005-12-04 00:43:56 +08:00
}
2005-09-20 21:26:39 +08:00
}
2005-12-04 00:43:56 +08:00
2005-09-20 21:26:39 +08:00
int mark = tok_get_pos ( tok ) ;
if ( wcscmp ( L " command " , nxt ) = = 0 )
{
tok_next ( tok ) ;
if ( parser_is_help ( tok_last ( tok ) , 0 ) )
{
tok_set_pos ( tok , mark ) ;
}
else
{
use_function = 0 ;
use_builtin = 0 ;
free ( nxt ) ;
continue ;
}
}
else if ( wcscmp ( L " builtin " , nxt ) = = 0 )
{
tok_next ( tok ) ;
if ( tok_last ( tok ) [ 0 ] = = L ' - ' )
{
tok_set_pos ( tok , mark ) ;
}
else
{
use_function = 0 ;
free ( nxt ) ;
continue ;
}
}
else if ( wcscmp ( L " not " , nxt ) = = 0 )
{
tok_next ( tok ) ;
if ( tok_last ( tok ) [ 0 ] = = L ' - ' )
{
tok_set_pos ( tok , mark ) ;
}
else
{
j - > negate = 1 - j - > negate ;
free ( nxt ) ;
continue ;
}
}
else if ( wcscmp ( L " and " , nxt ) = = 0 )
{
tok_next ( tok ) ;
if ( tok_last ( tok ) [ 0 ] = = L ' - ' )
{
tok_set_pos ( tok , mark ) ;
}
else
{
2006-01-05 23:37:53 +08:00
j - > skip = proc_get_last_status ( ) ;
2005-09-20 21:26:39 +08:00
free ( nxt ) ;
continue ;
}
}
else if ( wcscmp ( L " or " , nxt ) = = 0 )
{
tok_next ( tok ) ;
if ( tok_last ( tok ) [ 0 ] = = L ' - ' )
{
tok_set_pos ( tok , mark ) ;
}
else
{
2006-01-05 23:37:53 +08:00
j - > skip = ! proc_get_last_status ( ) ;
2005-09-20 21:26:39 +08:00
free ( nxt ) ;
continue ;
}
}
else if ( wcscmp ( L " exec " , nxt ) = = 0 )
{
if ( p ! = j - > first_process )
{
error ( SYNTAX_ERROR ,
2005-12-08 00:06:47 +08:00
tok_get_pos ( tok ) ,
EXEC_ERR_MSG ) ;
2005-09-20 21:26:39 +08:00
al_destroy ( & args ) ;
free ( nxt ) ;
return 0 ;
}
tok_next ( tok ) ;
if ( tok_last ( tok ) [ 0 ] = = L ' - ' )
{
tok_set_pos ( tok , mark ) ;
}
else
{
use_function = 0 ;
use_builtin = 0 ;
p - > type = INTERNAL_EXEC ;
free ( nxt ) ;
continue ;
}
}
else if ( wcscmp ( L " while " , nxt ) = = 0 )
{
int new_block = 0 ;
tok_next ( tok ) ;
2005-12-04 00:43:56 +08:00
if ( ( current_block - > type ! = WHILE ) )
2005-09-20 21:26:39 +08:00
{
new_block = 1 ;
}
2005-10-12 03:31:16 +08:00
else if ( current_block - > param1 . while_state = = WHILE_TEST_AGAIN )
2005-09-20 21:26:39 +08:00
{
2005-10-12 03:31:16 +08:00
current_block - > param1 . while_state = WHILE_TEST_FIRST ;
2005-09-20 21:26:39 +08:00
}
else
{
new_block = 1 ;
}
if ( new_block )
{
parser_push_block ( WHILE ) ;
2005-10-12 03:31:16 +08:00
current_block - > param1 . while_state = WHILE_TEST_FIRST ;
2005-09-20 21:26:39 +08:00
current_block - > tok_pos = mark ;
}
free ( nxt ) ;
is_new_block = 1 ;
continue ;
}
else if ( wcscmp ( L " if " , nxt ) = = 0 )
{
tok_next ( tok ) ;
parser_push_block ( IF ) ;
2005-10-12 03:31:16 +08:00
current_block - > param1 . if_state = 0 ;
2005-09-20 21:26:39 +08:00
current_block - > tok_pos = mark ;
free ( nxt ) ;
is_new_block = 1 ;
continue ;
}
if ( use_function )
{
int nxt_forbidden ;
wchar_t * forbid ;
forbid = ( wchar_t * ) ( al_get_count ( & forbidden_function ) ? al_peek ( & forbidden_function ) : 0 ) ;
nxt_forbidden = forbid & & ( wcscmp ( forbid , nxt ) = = 0 ) ;
/*
Make feeble attempt to avoid infinite recursion . Will at
least catch some accidental infinite recursion calls .
*/
if ( function_exists ( nxt ) & & ! nxt_forbidden )
{
/*
Check if we have reached the maximum recursion depth
*/
if ( al_get_count ( & forbidden_function ) > MAX_RECURSION_DEPTH )
{
error ( SYNTAX_ERROR ,
2005-12-08 00:06:47 +08:00
tok_get_pos ( tok ) ,
RECURSION_ERR_MSG ) ;
2005-09-20 21:26:39 +08:00
}
else
{
p - > type = INTERNAL_FUNCTION ;
}
}
}
al_push ( & args , nxt ) ;
}
if ( error_code = = 0 )
{
if ( ! p - > type )
{
if ( use_builtin & &
builtin_exists ( ( wchar_t * ) al_get ( & args , 0 ) ) )
{
p - > type = INTERNAL_BUILTIN ;
is_new_block = parser_is_block ( ( wchar_t * ) al_get ( & args , 0 ) ) ;
}
}
if ( ! p - > type | | ( p - > type = = INTERNAL_EXEC ) )
{
/*
If we are not executing the current block , allow
non - existent commands .
*/
if ( current_block - > skip )
{
p - > actual_cmd = wcsdup ( L " " ) ;
}
else
{
p - > actual_cmd = get_filename ( ( wchar_t * ) al_get ( & args , 0 ) ) ;
2005-12-12 07:39:39 +08:00
debug ( 2 , L " filename '%ls' -> '%ls' " , ( wchar_t * ) al_get ( & args , 0 ) , p - > actual_cmd ? p - > actual_cmd : L " 0 " ) ;
2005-09-20 21:26:39 +08:00
/*
Check if the specified command exists
*/
if ( p - > actual_cmd = = 0 )
{
/*
That is not a command ! Test if it is a
directory , in which case , we use ' cd ' as the
implicit command .
*/
wchar_t * pp =
parser_cdpath_get ( ( wchar_t * ) al_get ( & args , 0 ) ) ;
if ( pp )
{
wchar_t * tmp ;
free ( pp ) ;
tmp = ( wchar_t * ) al_get ( & args , 0 ) ;
al_truncate ( & args , 0 ) ;
al_push ( & args , wcsdup ( L " cd " ) ) ;
al_push ( & args , tmp ) ;
/*
If we have defined a wrapper around cd , use it ,
otherwise use the cd builtin
*/
if ( function_exists ( L " cd " ) )
p - > type = INTERNAL_FUNCTION ;
else
p - > type = INTERNAL_BUILTIN ;
}
else
{
2005-12-08 00:06:47 +08:00
if ( wcschr ( ( wchar_t * ) al_get ( & args , 0 ) , L ' = ' ) )
{
error ( EVAL_ERROR ,
tok_get_pos ( tok ) ,
COMMAND_ASSIGN_ERR_MSG ,
( wchar_t * ) al_get ( & args , 0 ) ) ;
}
else
{
error ( EVAL_ERROR ,
tok_get_pos ( tok ) ,
2006-01-04 20:51:02 +08:00
_ ( L " Unknown command '%ls' " ) ,
2005-12-08 00:06:47 +08:00
( wchar_t * ) al_get ( & args , 0 ) ) ;
}
2005-09-20 21:26:39 +08:00
}
}
}
}
}
if ( is_new_block )
{
const wchar_t * end = parser_find_end ( tok_string ( tok ) +
current_tokenizer_pos ) ;
tokenizer subtok ;
int make_sub_block = j - > first_process ! = p ;
if ( ! end )
{
error ( SYNTAX_ERROR ,
2005-12-08 00:06:47 +08:00
tok_get_pos ( tok ) ,
2005-12-14 11:39:39 +08:00
BLOCK_END_ERR_MSG ) ;
2005-09-20 21:26:39 +08:00
}
if ( ! make_sub_block )
{
tok_init ( & subtok , end , 0 ) ;
switch ( tok_last_type ( & subtok ) )
{
case TOK_END :
break ;
case TOK_REDIRECT_OUT :
case TOK_REDIRECT_APPEND :
case TOK_REDIRECT_IN :
case TOK_REDIRECT_FD :
case TOK_PIPE :
{
make_sub_block = 1 ;
break ;
}
default :
{
2005-10-08 08:23:49 +08:00
error ( SYNTAX_ERROR ,
2005-12-08 00:06:47 +08:00
current_tokenizer_pos ,
BLOCK_END_ERR_MSG ) ;
2005-09-20 21:26:39 +08:00
}
}
tok_destroy ( & subtok ) ;
}
if ( make_sub_block )
{
int end_pos = end - tok_string ( tok ) ;
wchar_t * sub_block = wcsndup ( tok_string ( tok ) + current_tokenizer_pos ,
end_pos - current_tokenizer_pos ) ;
p - > type = INTERNAL_BLOCK ;
free ( ( void * ) al_get ( & args , 0 ) ) ;
al_set ( & args , 0 , sub_block ) ;
tok_set_pos ( tok ,
end_pos ) ;
while ( prev_block ! = current_block )
2005-12-12 23:51:04 +08:00
{
2005-09-20 21:26:39 +08:00
parser_pop_block ( ) ;
2005-12-12 23:51:04 +08:00
}
2005-09-20 21:26:39 +08:00
}
else tok_next ( tok ) ;
}
else tok_next ( tok ) ;
if ( ! error_code )
2005-09-25 19:25:42 +08:00
{
if ( p - > type = = INTERNAL_BUILTIN & & parser_skip_arguments ( ( wchar_t * ) al_get ( & args , 0 ) ) )
{
p - > argv = list_to_char_arr ( & args ) ;
// tok_next(tok);
}
else
{
parse_job_main_loop ( p , j , tok , & args ) ;
}
}
2005-09-20 21:26:39 +08:00
if ( error_code )
{
/*
We don ' t know what the state of the args array and the argv
vector is on error , so we do an internal cleanup here .
*/
al_foreach ( & args ,
( void ( * ) ( const void * ) ) & free ) ;
free ( p - > argv ) ;
p - > argv = 0 ;
/*
Make sure the block stack is consistent
*/
while ( prev_block ! = current_block )
2005-12-12 23:51:04 +08:00
{
2005-09-20 21:26:39 +08:00
parser_pop_block ( ) ;
2005-12-12 23:51:04 +08:00
}
2005-09-20 21:26:39 +08:00
}
al_destroy ( & args ) ;
// debug( 2, L"end parse_job()\n" );
return ! error_code ;
}
/**
Do skipped execution of command . This means that only limited
execution of block level commands such as end and switch should be
preformed .
\ param j the job to execute
*/
static void skipped_exec ( job_t * j )
{
process_t * p ;
2005-12-09 10:41:16 +08:00
2005-09-20 21:26:39 +08:00
for ( p = j - > first_process ; p ; p = p - > next )
{
2005-12-09 10:41:16 +08:00
if ( p - > type = = INTERNAL_BUILTIN )
2005-09-20 21:26:39 +08:00
{
if ( ( wcscmp ( p - > argv [ 0 ] , L " for " ) = = 0 ) | |
( wcscmp ( p - > argv [ 0 ] , L " switch " ) = = 0 ) | |
2005-12-12 23:51:04 +08:00
( wcscmp ( p - > argv [ 0 ] , L " begin " ) = = 0 ) | |
2005-09-20 21:26:39 +08:00
( wcscmp ( p - > argv [ 0 ] , L " function " ) = = 0 ) )
{
parser_push_block ( FAKE ) ;
}
else if ( wcscmp ( p - > argv [ 0 ] , L " end " ) = = 0 )
{
if ( ! current_block - > outer - > skip )
{
exec ( j ) ;
return ;
}
parser_pop_block ( ) ;
}
else if ( wcscmp ( p - > argv [ 0 ] , L " else " ) = = 0 )
{
if ( ( current_block - > type = = IF ) & &
2005-10-12 03:31:16 +08:00
( current_block - > param1 . if_state ! = 0 ) )
2005-09-20 21:26:39 +08:00
{
exec ( j ) ;
return ;
}
}
else if ( wcscmp ( p - > argv [ 0 ] , L " case " ) = = 0 )
{
if ( ( current_block - > type = = SWITCH ) )
{
exec ( j ) ;
return ;
}
}
}
}
job_free ( j ) ;
}
/**
Evaluates a job from the specified tokenizer . First calls
parse_job to parse the job and then calls exec to execute it .
\ param tok The tokenizer to read tokens from
*/
static void eval_job ( tokenizer * tok )
{
job_t * j ;
int start_pos = job_start_pos = tok_get_pos ( tok ) ;
2005-10-14 19:40:33 +08:00
debug ( 2 , L " begin eval_job() " ) ;
2005-09-20 21:26:39 +08:00
long long t1 = 0 , t2 = 0 , t3 = 0 ;
profile_element_t * p = 0 ;
int skip = 0 ;
2005-10-08 10:00:08 +08:00
2005-09-20 21:26:39 +08:00
if ( profile )
{
p = malloc ( sizeof ( profile_element_t ) ) ;
p - > cmd = 0 ;
al_push ( & profile_data , p ) ;
p - > skipped = 1 ;
t1 = get_time ( ) ;
}
switch ( tok_last_type ( tok ) )
{
case TOK_STRING :
{
j = job_create ( ) ;
j - > command = 0 ;
j - > fg = 1 ;
j - > constructed = 0 ;
2005-10-12 03:23:43 +08:00
j - > skip_notification = is_subshell | | is_block | | is_event | | ( ! is_interactive ) ;
2006-01-18 20:42:48 +08:00
j - > terminal = is_interactive & & ! is_subshell ;
2005-09-20 21:26:39 +08:00
2006-01-18 20:42:48 +08:00
current_block - > job = j ;
2005-09-20 21:26:39 +08:00
if ( is_interactive )
{
if ( tcgetattr ( 0 , & j - > tmodes ) )
{
tok_next ( tok ) ;
wperror ( L " tcgetattr " ) ;
job_free ( j ) ;
break ;
}
}
2005-10-12 03:23:43 +08:00
2005-09-20 21:26:39 +08:00
j - > first_process = calloc ( 1 , sizeof ( process_t ) ) ;
if ( parse_job ( j - > first_process , j , tok ) & &
j - > first_process - > argv )
{
if ( job_start_pos < tok_get_pos ( tok ) )
{
int stop_pos = tok_get_pos ( tok ) ;
wchar_t * newline = wcschr ( tok_string ( tok ) + start_pos ,
L ' \n ' ) ;
if ( newline )
stop_pos = mini ( stop_pos , newline - tok_string ( tok ) ) ;
j - > command = wcsndup ( tok_string ( tok ) + start_pos ,
stop_pos - start_pos ) ;
}
else
j - > command = wcsdup ( L " " ) ;
if ( profile )
{
t2 = get_time ( ) ;
p - > cmd = wcsdup ( j - > command ) ;
p - > skipped = current_block - > skip ;
}
2005-10-15 08:51:26 +08:00
2005-09-20 21:26:39 +08:00
skip | = current_block - > skip ;
2005-12-09 10:41:16 +08:00
skip | = j - > wildcard_error ;
2006-01-05 23:37:53 +08:00
skip | = j - > skip ;
2005-09-20 21:26:39 +08:00
if ( ! skip )
{
2005-11-30 03:51:32 +08:00
int was_builtin = 0 ;
2005-12-01 23:45:44 +08:00
// if( j->first_process->type==INTERNAL_BUILTIN && !j->first_process->next)
// was_builtin = 1;
2005-11-30 03:51:32 +08:00
2005-09-20 21:26:39 +08:00
exec ( j ) ;
2005-11-30 03:51:32 +08:00
/* Only external commands require a new fishd barrier */
if ( ! was_builtin )
proc_had_barrier = 0 ;
2005-09-20 21:26:39 +08:00
}
else
{
skipped_exec ( j ) ;
}
if ( profile )
{
t3 = get_time ( ) ;
p - > level = eval_level ;
p - > parse = t2 - t1 ;
p - > exec = t3 - t2 ;
}
if ( current_block - > type = = WHILE )
{
2005-10-12 03:31:16 +08:00
switch ( current_block - > param1 . while_state )
2005-09-20 21:26:39 +08:00
{
case WHILE_TEST_FIRST :
{
current_block - > skip = proc_get_last_status ( ) ! = 0 ;
2005-10-12 03:31:16 +08:00
current_block - > param1 . while_state = WHILE_TESTED ;
2005-09-20 21:26:39 +08:00
}
break ;
}
}
if ( current_block - > type = = IF )
{
2005-10-12 03:31:16 +08:00
if ( ( ! current_block - > param1 . if_state ) & &
2005-09-20 21:26:39 +08:00
( ! current_block - > skip ) )
{
current_block - > skip = proc_get_last_status ( ) ! = 0 ;
2005-10-12 03:31:16 +08:00
current_block - > param1 . if_state + + ;
2005-09-20 21:26:39 +08:00
}
}
}
else
{
2005-09-25 03:31:17 +08:00
/*
2005-12-07 21:02:09 +08:00
This job could not be properly parsed . We free it instead , and set the status to 1.
2005-09-25 03:31:17 +08:00
*/
2005-09-20 21:26:39 +08:00
job_free ( j ) ;
2005-12-07 21:02:09 +08:00
proc_set_last_status ( 1 ) ;
2005-09-20 21:26:39 +08:00
}
2005-10-15 08:51:26 +08:00
current_block - > job = 0 ;
2005-09-20 21:26:39 +08:00
break ;
}
case TOK_END :
{
if ( tok_has_next ( tok ) )
tok_next ( tok ) ;
break ;
}
2006-01-06 00:02:28 +08:00
case TOK_BACKGROUND :
{
wchar_t * str = tok_string ( tok ) ;
if ( tok_get_pos ( tok ) > 0 & & str [ tok_get_pos ( tok ) - 1 ] = = L ' & ' )
{
error ( SYNTAX_ERROR ,
tok_get_pos ( tok ) ,
CMD_AND_ERR_MSG ,
tok_get_desc ( tok_last_type ( tok ) ) ) ;
}
else
{
error ( SYNTAX_ERROR ,
tok_get_pos ( tok ) ,
CMD_ERR_MSG ,
tok_get_desc ( tok_last_type ( tok ) ) ) ;
}
2006-01-09 07:00:49 +08:00
return ;
2006-01-06 00:02:28 +08:00
}
2005-09-20 21:26:39 +08:00
case TOK_ERROR :
{
2005-12-08 00:06:47 +08:00
error ( SYNTAX_ERROR ,
tok_get_pos ( tok ) ,
TOK_ERR_MSG ,
tok_last ( tok ) ) ;
2005-09-20 21:26:39 +08:00
return ;
}
2005-09-25 19:25:42 +08:00
2005-09-20 21:26:39 +08:00
default :
{
2005-12-08 00:06:47 +08:00
error ( SYNTAX_ERROR ,
tok_get_pos ( tok ) ,
2005-12-14 11:39:39 +08:00
CMD_ERR_MSG ,
2005-12-08 00:06:47 +08:00
tok_get_desc ( tok_last_type ( tok ) ) ) ;
2005-09-20 21:26:39 +08:00
return ;
}
}
2005-10-12 03:23:43 +08:00
job_reap ( 0 ) ;
2005-10-09 19:53:09 +08:00
// debug( 2, L"end eval_job()\n" );
2005-09-20 21:26:39 +08:00
}
int eval ( const wchar_t * cmd , io_data_t * io , int block_type )
{
int forbid_count ;
int code ;
tokenizer * previous_tokenizer = current_tokenizer ;
block_t * start_current_block = current_block ;
io_data_t * prev_io = block_io ;
block_io = io ;
2005-10-12 03:23:43 +08:00
job_reap ( 0 ) ;
2005-12-12 23:51:04 +08:00
debug ( 4 , L " eval: %ls " , cmd ) ;
2005-10-12 03:23:43 +08:00
2005-09-20 21:26:39 +08:00
if ( ! cmd )
{
debug ( 1 ,
2006-01-04 20:51:02 +08:00
EVAL_NULL_ERR_MSG ) ;
debug ( 1 ,
BUGREPORT_MSG ,
2005-10-09 19:48:16 +08:00
PACKAGE_BUGREPORT ) ;
2005-09-20 21:26:39 +08:00
return 1 ;
}
if ( ( block_type ! = TOP ) & &
( block_type ! = FUNCTION_CALL ) & &
( block_type ! = SUBST ) )
{
debug ( 1 ,
2006-01-04 20:51:02 +08:00
INVALID_SCOPE_ERR_MSG ,
parser_get_block_desc ( block_type ) ) ;
debug ( 1 ,
BUGREPORT_MSG ,
2005-10-09 19:48:16 +08:00
PACKAGE_BUGREPORT ) ;
2005-12-14 09:07:12 +08:00
return 1 ;
}
2005-10-09 19:48:16 +08:00
2005-09-20 21:26:39 +08:00
eval_level + + ;
current_tokenizer = malloc ( sizeof ( tokenizer ) ) ;
2005-10-09 19:48:16 +08:00
2005-09-20 21:26:39 +08:00
parser_push_block ( block_type ) ;
2005-10-09 19:48:16 +08:00
2005-09-20 21:26:39 +08:00
forbid_count = al_get_count ( & forbidden_function ) ;
2005-10-09 19:48:16 +08:00
2005-09-20 21:26:39 +08:00
tok_init ( current_tokenizer , cmd , 0 ) ;
error_code = 0 ;
2005-10-09 19:48:16 +08:00
2005-12-12 06:21:01 +08:00
event_fire ( 0 ) ;
2005-10-14 19:40:33 +08:00
2005-09-20 21:26:39 +08:00
while ( tok_has_next ( current_tokenizer ) & &
! error_code & &
! sanity_check ( ) & &
! exit_status ( ) )
2005-10-14 19:40:33 +08:00
{
2005-09-20 21:26:39 +08:00
eval_job ( current_tokenizer ) ;
2005-12-12 06:21:01 +08:00
event_fire ( 0 ) ;
2005-10-14 19:40:33 +08:00
}
2005-10-09 19:48:16 +08:00
2005-09-20 21:26:39 +08:00
int prev_block_type = current_block - > type ;
2005-12-12 23:51:04 +08:00
2005-09-20 21:26:39 +08:00
parser_pop_block ( ) ;
while ( start_current_block ! = current_block )
{
if ( current_block = = 0 )
{
debug ( 0 ,
2006-01-04 20:51:02 +08:00
_ ( L " End of block mismatch. Program terminating. " ) ) ;
debug ( 0 ,
2005-10-09 19:48:16 +08:00
BUGREPORT_MSG ,
PACKAGE_BUGREPORT ) ;
2005-09-20 21:26:39 +08:00
exit ( 1 ) ;
break ;
}
2005-10-15 08:51:26 +08:00
2005-09-20 21:26:39 +08:00
if ( ( ! error_code ) & & ( ! exit_status ( ) ) & & ( ! proc_get_last_status ( ) ) )
{
char * h ;
//debug( 2, L"Status %d\n", proc_get_last_status() );
2006-01-05 23:37:53 +08:00
debug ( 1 ,
L " %ls " , parser_get_block_desc ( current_block - > type ) ) ;
debug ( 1 ,
BLOCK_END_ERR_MSG ) ;
fwprintf ( stderr , L " %ls " , parser_current_line ( ) ) ;
2005-09-20 21:26:39 +08:00
2006-01-05 23:37:53 +08:00
h = builtin_help_get ( L " end " ) ;
if ( h )
fwprintf ( stderr , L " %s " , h ) ;
break ;
2005-09-20 21:26:39 +08:00
}
prev_block_type = current_block - > type ;
parser_pop_block ( ) ;
}
print_errors ( ) ;
tok_destroy ( current_tokenizer ) ;
free ( current_tokenizer ) ;
while ( forbid_count < al_get_count ( & forbidden_function ) )
parser_allow_function ( ) ;
current_tokenizer = previous_tokenizer ;
code = error_code ;
error_code = 0 ;
block_io = prev_io ;
eval_level - - ;
2005-10-12 03:23:43 +08:00
job_reap ( 0 ) ;
2005-09-20 21:26:39 +08:00
return code ;
}
int parser_test ( wchar_t * buff ,
int babble )
{
tokenizer tok ;
int had_cmd = 0 ;
int count = 0 ;
int err = 0 ;
tokenizer * previous_tokenizer = current_tokenizer ;
int previous_pos = current_tokenizer_pos ;
static int block_pos [ BLOCK_MAX_COUNT ] ;
static int block_type [ BLOCK_MAX_COUNT ] ;
int is_pipeline = 0 ;
int forbid_pipeline = 0 ;
int needs_cmd = 0 ;
int require_additional_commands = 0 ;
current_tokenizer = & tok ;
2005-10-08 10:00:08 +08:00
2005-09-20 21:26:39 +08:00
for ( tok_init ( & tok , buff , 0 ) ;
tok_has_next ( & tok ) & & ! err ;
tok_next ( & tok ) )
{
current_tokenizer_pos = tok_get_pos ( & tok ) ;
int last_type = tok_last_type ( & tok ) ;
switch ( last_type )
{
case TOK_STRING :
{
if ( ! had_cmd )
{
int mark = tok_get_pos ( & tok ) ;
had_cmd = 1 ;
if ( require_additional_commands )
{
if ( contains_str ( tok_last ( & tok ) ,
L " end " ,
0 ) )
{
err = 1 ;
if ( babble )
{
error ( SYNTAX_ERROR ,
2005-12-08 00:06:47 +08:00
tok_get_pos ( & tok ) ,
COND_ERR_MSG ) ;
2005-09-20 21:26:39 +08:00
print_errors ( ) ;
}
}
require_additional_commands - - ;
}
2005-09-25 19:25:42 +08:00
/*
Decrement block count on end command
*/
2005-09-20 21:26:39 +08:00
if ( wcscmp ( tok_last ( & tok ) , L " end " ) = = 0 )
{
tok_next ( & tok ) ;
count - - ;
tok_set_pos ( & tok , mark ) ;
}
2005-09-25 19:25:42 +08:00
/*
Handle block commands
*/
if ( parser_is_block ( tok_last ( & tok ) ) )
2005-09-20 21:26:39 +08:00
{
if ( count > = BLOCK_MAX_COUNT )
{
2005-12-08 00:06:47 +08:00
error ( SYNTAX_ERROR ,
tok_get_pos ( & tok ) ,
BLOCK_ERR_MSG ) ;
2005-09-20 21:26:39 +08:00
print_errors ( ) ;
}
else
{
if ( wcscmp ( tok_last ( & tok ) , L " while " ) = = 0 )
block_type [ count ] = WHILE ;
else if ( wcscmp ( tok_last ( & tok ) , L " for " ) = = 0 )
block_type [ count ] = FOR ;
else if ( wcscmp ( tok_last ( & tok ) , L " switch " ) = = 0 )
block_type [ count ] = SWITCH ;
else if ( wcscmp ( tok_last ( & tok ) , L " if " ) = = 0 )
block_type [ count ] = IF ;
else if ( wcscmp ( tok_last ( & tok ) , L " function " ) = = 0 )
block_type [ count ] = FUNCTION_DEF ;
2005-09-25 19:25:42 +08:00
else if ( wcscmp ( tok_last ( & tok ) , L " begin " ) = = 0 )
block_type [ count ] = BEGIN ;
2005-09-20 21:26:39 +08:00
else
block_type [ count ] = - 1 ;
// debug( 2, L"add block of type %d after cmd %ls\n", block_type[count], tok_last(&tok) );
block_pos [ count ] = current_tokenizer_pos ;
tok_next ( & tok ) ;
count + + ;
tok_set_pos ( & tok , mark ) ;
}
}
2005-09-25 19:25:42 +08:00
/*
If parser_is_subcommand is true , the command
accepts a second command as it ' s first
argument . If parser_skip_arguments is true , the
second argument is optional .
*/
if ( parser_is_subcommand ( tok_last ( & tok ) ) & & ! parser_skip_arguments ( tok_last ( & tok ) ) )
2005-09-20 21:26:39 +08:00
{
needs_cmd = 1 ;
had_cmd = 0 ;
}
2005-09-25 19:25:42 +08:00
/*
The short circut commands requires _two_ additional commands .
*/
2005-09-20 21:26:39 +08:00
if ( contains_str ( tok_last ( & tok ) ,
L " or " ,
L " and " ,
0 ) )
{
if ( is_pipeline )
{
err = 1 ;
if ( babble )
{
error ( SYNTAX_ERROR ,
2005-12-08 00:06:47 +08:00
tok_get_pos ( & tok ) ,
EXEC_ERR_MSG ) ;
2005-09-20 21:26:39 +08:00
print_errors ( ) ;
}
}
2006-01-05 23:37:53 +08:00
require_additional_commands = 1 ;
2005-09-20 21:26:39 +08:00
}
2005-09-25 19:25:42 +08:00
/*
There are a lot of situations where pipelines
are forbidden , inclusing when using the exec
builtin .
*/
2005-09-20 21:26:39 +08:00
if ( parser_is_pipe_forbidden ( tok_last ( & tok ) ) )
{
if ( is_pipeline )
{
err = 1 ;
if ( babble )
{
error ( SYNTAX_ERROR ,
2005-12-08 00:06:47 +08:00
tok_get_pos ( & tok ) ,
EXEC_ERR_MSG ) ;
2005-09-20 21:26:39 +08:00
print_errors ( ) ;
}
}
forbid_pipeline = 1 ;
}
2005-09-25 19:25:42 +08:00
/*
Test that the case builtin is only used in a switch block
*/
2005-09-20 21:26:39 +08:00
if ( wcscmp ( L " case " , tok_last ( & tok ) ) = = 0 )
{
if ( ! count | | block_type [ count - 1 ] ! = SWITCH )
{
err = 1 ;
// debug( 2, L"Error on block type %d\n", block_type[count-1] );
2005-12-14 09:07:12 +08:00
if ( babble )
{
error ( SYNTAX_ERROR ,
tok_get_pos ( & tok ) ,
2005-12-14 02:28:03 +08:00
INVALID_CASE_ERR_MSG ) ;
2005-12-14 09:07:12 +08:00
print_errors ( ) ;
}
}
2005-09-25 19:25:42 +08:00
}
/*
Test that break and continue are only used in loop blocks
*/
if ( wcscmp ( L " break " , tok_last ( & tok ) ) = = 0 | |
wcscmp ( L " continue " , tok_last ( & tok ) ) = = 0 )
2005-09-20 21:26:39 +08:00
{
int found_loop = 0 ;
int i ;
for ( i = count - 1 ; i > = 0 ; i - - )
{
if ( ( block_type [ i ] = = WHILE ) | |
( block_type [ i ] = = FOR ) )
{
found_loop = 1 ;
break ;
}
}
if ( ! found_loop )
{
err = 1 ;
2005-12-14 09:07:12 +08:00
if ( babble )
{
error ( SYNTAX_ERROR ,
tok_get_pos ( & tok ) ,
INVALID_LOOP_ERR_MSG ) ;
2005-12-08 00:06:47 +08:00
print_errors ( ) ;
2005-12-14 09:07:12 +08:00
}
}
2005-09-20 21:26:39 +08:00
}
2005-09-25 19:25:42 +08:00
/*
Test that else is only used in an if - block
*/
if ( wcscmp ( L " else " , tok_last ( & tok ) ) = = 0 )
2005-09-20 21:26:39 +08:00
{
if ( ! count | | block_type [ count - 1 ] ! = IF )
2005-12-14 09:07:12 +08:00
{
err = 1 ;
if ( babble )
{
error ( SYNTAX_ERROR ,
tok_get_pos ( & tok ) ,
2005-12-14 02:28:03 +08:00
INVALID_ELSE_ERR_MSG ) ;
2005-12-08 00:06:47 +08:00
print_errors ( ) ;
2005-12-14 09:07:12 +08:00
}
}
2005-09-20 21:26:39 +08:00
}
2005-09-25 19:25:42 +08:00
/*
Test that end is not used when not inside a blovk
*/
2005-09-20 21:26:39 +08:00
if ( count < 0 )
{
err = 1 ;
if ( babble )
{
error ( SYNTAX_ERROR ,
2005-12-08 00:06:47 +08:00
tok_get_pos ( & tok ) ,
2005-12-14 04:11:21 +08:00
INVALID_END_ERR_MSG ) ;
2005-09-20 21:26:39 +08:00
print_errors ( ) ;
}
}
}
break ;
}
case TOK_REDIRECT_OUT :
case TOK_REDIRECT_IN :
case TOK_REDIRECT_APPEND :
case TOK_REDIRECT_FD :
{
if ( ! had_cmd )
{
err = 1 ;
if ( babble )
{
error ( SYNTAX_ERROR ,
2005-12-08 00:06:47 +08:00
tok_get_pos ( & tok ) ,
2005-12-14 11:39:39 +08:00
INVALID_REDIRECTION_ERR_MSG ) ;
2005-09-20 21:26:39 +08:00
print_errors ( ) ;
}
}
break ;
}
case TOK_END :
{
if ( needs_cmd & & ! had_cmd )
{
err = 1 ;
2005-12-14 09:07:12 +08:00
if ( babble )
{
error ( SYNTAX_ERROR ,
tok_get_pos ( & tok ) ,
2005-12-14 11:39:39 +08:00
CMD_ERR_MSG ,
tok_get_desc ( tok_last_type ( & tok ) ) ) ;
2005-12-08 00:06:47 +08:00
print_errors ( ) ;
2005-12-14 09:07:12 +08:00
}
2005-09-20 21:26:39 +08:00
}
needs_cmd = 0 ;
had_cmd = 0 ;
is_pipeline = 0 ;
forbid_pipeline = 0 ;
break ;
}
case TOK_PIPE :
{
if ( forbid_pipeline )
{
err = 1 ;
if ( babble )
{
error ( SYNTAX_ERROR ,
2005-12-08 00:06:47 +08:00
tok_get_pos ( & tok ) ,
EXEC_ERR_MSG ) ;
2005-09-20 21:26:39 +08:00
print_errors ( ) ;
}
}
needs_cmd = 0 ;
is_pipeline = 1 ;
}
case TOK_BACKGROUND :
{
if ( needs_cmd & & ! had_cmd )
{
err = 1 ;
2005-12-14 09:07:12 +08:00
if ( babble )
{
error ( SYNTAX_ERROR ,
tok_get_pos ( & tok ) ,
2005-12-14 11:39:39 +08:00
CMD_ERR_MSG ,
tok_get_desc ( tok_last_type ( & tok ) ) ) ;
2005-12-08 00:06:47 +08:00
2005-12-14 09:07:12 +08:00
print_errors ( ) ;
}
2005-09-20 21:26:39 +08:00
}
if ( had_cmd )
{
had_cmd = 0 ;
}
break ;
}
case TOK_ERROR :
default :
err = 1 ;
if ( babble )
{
2005-12-08 00:06:47 +08:00
error ( SYNTAX_ERROR ,
tok_get_pos ( & tok ) ,
TOK_ERR_MSG ,
tok_last ( & tok ) ) ;
2005-09-20 21:26:39 +08:00
print_errors ( ) ;
//debug( 2, tok_last( &tok) );
}
break ;
}
}
if ( require_additional_commands )
{
err = 1 ;
if ( babble )
{
error ( SYNTAX_ERROR ,
2005-12-08 00:06:47 +08:00
tok_get_pos ( & tok ) ,
COND_ERR_MSG ) ;
2005-09-20 21:26:39 +08:00
print_errors ( ) ;
}
}
2005-10-14 19:40:33 +08:00
2005-09-20 21:26:39 +08:00
if ( babble & & count > 0 )
{
error ( SYNTAX_ERROR ,
2005-12-08 00:06:47 +08:00
block_pos [ count - 1 ] ,
2005-12-14 11:39:39 +08:00
BLOCK_END_ERR_MSG ) ;
2005-09-20 21:26:39 +08:00
print_errors ( ) ;
}
2005-10-14 19:40:33 +08:00
2005-09-20 21:26:39 +08:00
tok_destroy ( & tok ) ;
2005-10-14 19:40:33 +08:00
2005-09-20 21:26:39 +08:00
current_tokenizer = previous_tokenizer ;
current_tokenizer_pos = previous_pos ;
error_code = 0 ;
return err | ( ( count ! = 0 ) < < 1 ) ;
}