/** \file parser.c The fish parser. Contains functions for parsing code. */ #include #include #include #include #include #include #include #include #include #include #include #include #include "config.h" #include "fallback.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 "env.h" #include "expand.h" #include "reader.h" #include "sanity.h" #include "env_universal.h" #include "event.h" #include "intern.h" #include "parse_util.h" #include "halloc.h" #include "halloc_util.h" /** 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 /** Message about reporting bugs, used on weird internal error to hopefully get them to report stuff. */ #define BUGREPORT_MSG _( L"If this error can be reproduced, please send a bug report to %s.") /** Error message for improper use of the exec builtin */ #define EXEC_ERR_MSG _(L"This command can not be used in a pipeline") /** Error message for tokenizer error. The tokenizer message is appended to this message. */ #define TOK_ERR_MSG _( L"Tokenizer error: '%ls'") /** Error message for short circuit command error. */ #define COND_ERR_MSG _( L"An additional command is required" ) /** Error message on reaching maximum recusrion depth */ #define RECURSION_ERR_MSG _( L"Maximum recursion depth reached. Accidental infinite loop?") /** Error message used when the end of a block can't be located */ #define BLOCK_END_ERR_MSG _( L"Could not locate end of block. The 'end' command is missing, misspelled or a ';' is missing.") /** Error message on reaching maximum number of block calls */ #define BLOCK_ERR_MSG _( L"Maximum number of nested blocks reached.") /** Error message when a non-string token is found when expecting a command name */ #define CMD_ERR_MSG _( L"Expected a command name, got token of type '%ls'") /** Error message when a non-string token is found when expecting a command name */ #define CMD_OR_ERR_MSG _( L"Expected a command name, got token of type '%ls'. Did you mean 'COMMAND; or COMMAND'? See the help section for the 'or' builtin command by typing 'help or'.") /** Error message when a non-string token is found when expecting a command name */ #define CMD_AND_ERR_MSG _( L"Expected a command name, got token of type '%ls'. Did you mean 'COMMAND; and COMMAND'? See the help section for the 'and' builtin command by typing 'help and'.") /** Error message when encountering an illegal command name */ #define ILLEGAL_CMD_ERR_MSG _( L"Illegal command name '%ls'") /** Error message for wildcards with no matches */ #define WILDCARD_ERR_MSG _( L"Warning: No match for wildcard '%ls'. The command will not be executed.") /** Error when using case builtin outside of switch block */ #define INVALID_CASE_ERR_MSG _( L"'case' builtin not inside of switch block") /** Error when using loop control builtins (break or continue) outside of loop */ #define INVALID_LOOP_ERR_MSG _( L"Loop control command while not inside of loop" ) /** Error when using else builtin outside of if block */ #define INVALID_ELSE_ERR_MSG _( L"'else' builtin not inside of if block" ) /** Error when using end builtin outside of block */ #define INVALID_END_ERR_MSG _( L"'end' command outside of block") /** Error message for Posix-style assignment */ #define COMMAND_ASSIGN_ERR_MSG _( L"Unknown command '%ls'. Did you mean 'set %ls %ls'? For information on assigning values to variables, see the help section on the set command by typing 'help set'.") /** Error for invalid redirection token */ #define REDIRECT_TOKEN_ERR_MSG _( L"Expected redirection specification, got token of type '%ls'") /** Error when encountering redirection without a command */ #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 */ #define INVALID_SCOPE_ERR_MSG _( L"Tried to evaluate commands using invalid block type '%ls'" ) /** Error for wrong token type */ #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 N_( L"'while' block" ) /** For block description */ #define FOR_BLOCK N_( L"'for' block" ) /** If block description */ #define IF_BLOCK N_( L"'if' conditional block" ) /** Function definition block description */ #define FUNCTION_DEF_BLOCK N_( L"function definition block" ) /** Function invocation block description */ #define FUNCTION_CALL_BLOCK N_( L"function invocation block" ) /** Switch block description */ #define SWITCH_BLOCK N_( L"'switch' block" ) /** Fake block description */ #define FAKE_BLOCK N_( L"unexecutable block" ) /** Top block description */ #define TOP_BLOCK N_( L"global root block" ) /** Command substitution block description */ #define SUBST_BLOCK N_( L"command substitution block" ) /** Begin block description */ #define BEGIN_BLOCK N_( L"'begin' unconditional block" ) /** Source block description */ #define SOURCE_BLOCK N_( L"Block created by the . builtin" ) /** Source block description */ #define EVENT_BLOCK N_( L"event handler block" ) /** Unknown block description */ #define UNKNOWN_BLOCK N_( L"unknown/invalid block" ) /** Size of the error string buffer */ #define ERR_STR_SZ 1024 /** Datastructure to describe a block type, like while blocks, command substitution blocks, etc. */ struct block_lookup_entry { /** The block type id. The legal values are defined in parser.h. */ int type; /** The name of the builtin that creates this type of block, if any. */ const wchar_t *name; /** A description of this block type */ const wchar_t *desc; } ; /** List of all legal block types */ const static struct block_lookup_entry block_lookup[]= { { WHILE, L"while", WHILE_BLOCK } , { FOR, L"for", FOR_BLOCK } , { IF, L"if", IF_BLOCK } , { FUNCTION_DEF, L"function", FUNCTION_DEF_BLOCK } , { FUNCTION_CALL, 0, FUNCTION_CALL_BLOCK } , { SWITCH, L"switch", SWITCH_BLOCK } , { FAKE, 0, FAKE_BLOCK } , { TOP, 0, TOP_BLOCK } , { SUBST, 0, SUBST_BLOCK } , { BEGIN, L"begin", BEGIN_BLOCK } , { SOURCE, L".", SOURCE_BLOCK } , { EVENT, 0, EVENT_BLOCK } , { 0,0,0 } } ; /** Last error code */ static int error_code; event_block_t *global_event_block=0; io_data_t *block_io; /** Position of last error */ static int err_pos; /** Description of last error */ static wchar_t err_str[ERR_STR_SZ]; /** Pointer to the current tokenizer */ static tokenizer *current_tokenizer; /** String for representing the current line */ static string_buffer_t *lineinfo=0; /** 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; /** List of all profiling data */ static array_list_t profile_data; /** 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. */ static int eval_level=-1; static int parse_job( process_t *p, job_t *j, tokenizer *tok ); /** Struct used to keep track of profiling data for a command */ typedef struct { /** Time spent executing the specified command, including parse time for nested blocks. */ int exec; /** Time spent parsing the specified command, including execution time for command substitutions. */ int parse; /** The block level of the specified command. nested blocks and command substitutions both increase the block level. */ int level; /** If the execution of this command was skipped. */ int skipped; /** The command string. */ wchar_t *cmd; } profile_element_t; /** Return the current number of block nestings */ static int block_count( block_t *b ) { if( b==0) return 0; return( block_count(b->outer)+1); } void parser_push_block( int type ) { block_t *new = halloc( 0, sizeof( block_t )); new->src_lineno = parser_get_lineno(); new->src_filename = parser_current_filename()?intern(parser_current_filename()):0; new->outer = current_block; new->type = (current_block && current_block->skip)?FAKE:type; /* New blocks should be skipped if the outer block is skipped, except TOP ans SUBST block, which open up new environments. Fake blocks should always be skipped. Rather complicated... :-( */ new->skip=current_block?current_block->skip:0; /* Type TOP and SUBST are never skipped */ if( type == TOP || type == SUBST ) { new->skip = 0; } /* Fake blocks and function definition blocks are never executed */ if( type == FAKE || type == FUNCTION_DEF ) { new->skip = 1; } new->job = 0; new->loop_status=LOOP_NORMAL; current_block = new; if( (new->type != FUNCTION_DEF) && (new->type != FAKE) && (new->type != TOP) ) { env_push( type == FUNCTION_CALL ); halloc_register_function_void( current_block, &env_pop ); } } void parser_pop_block() { block_t *old = current_block; current_block = current_block->outer; halloc_free( old ); } const wchar_t *parser_get_block_desc( int block ) { int i; for( i=0; block_lookup[i].desc; i++ ) { if( block_lookup[i].type == block ) { return _( block_lookup[i].desc ); } } return _(UNKNOWN_BLOCK); } /** Check if the specified bcommand is one of the builtins that cannot have arguments, any followin argument is interpreted as a new command */ static int parser_skip_arguments( const wchar_t *cmd ) { return contains_str( cmd, L"else", L"begin", (void *)0 ); } int parser_is_subcommand( const wchar_t *cmd ) { return parser_skip_arguments( cmd ) || contains_str( cmd, L"command", L"builtin", L"while", L"exec", L"if", L"and", L"or", L"not", (void *)0 ); } int parser_is_block( const wchar_t *word) { return contains_str( word, L"for", L"while", L"if", L"function", L"switch", L"begin", (void *)0 ); } int parser_is_reserved( const 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", (void *)0 ); } /** Returns 1 if the specified command is a builtin that may not be used in a pipeline */ static int parser_is_pipe_forbidden( wchar_t *word ) { return contains_str( word, L"exec", L"case", L"break", L"return", L"continue", (void *)0 ); } /** Search the text for the end of the current block */ 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( void *context, wchar_t *dir ) { wchar_t *res = 0; if( !dir ) return 0; if( dir[0] == L'/'|| (wcsncmp( dir, L"./", 2 )==0) ) { struct stat buf; if( wstat( dir, &buf ) == 0 ) { if( S_ISDIR(buf.st_mode) ) { res = halloc_wcsdup( context, dir ); } } } else { wchar_t *path; wchar_t *path_cpy; wchar_t *nxt_path; wchar_t *state; wchar_t *whole_path; path = env_get(L"CDPATH"); if( !path || !wcslen(path) ) { path = L"."; } nxt_path = path; path_cpy = wcsdup( path ); if( !path_cpy ) { DIE_MEM(); } 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; halloc_register( context, 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 ) ); } void error( int ec, int p, const wchar_t *str, ... ) { va_list va; error_code = ec; err_pos = p; va_start( va, str ); vswprintf( err_str, ERR_STR_SZ, str, va ); va_end( va ); } wchar_t *parser_get_filename( void *context, const wchar_t *cmd ) { wchar_t *path; CHECK( cmd, 0 ); debug( 3, L"parser_get_filename( '%ls' )", cmd ); if(wcschr( cmd, L'/' ) != 0 ) { if( waccess( cmd, X_OK )==0 ) { struct stat buff; wstat( cmd, &buff ); if( S_ISREG(buff.st_mode) ) return halloc_wcsdup( context, cmd ); else return 0; } } else { path = env_get(L"PATH"); if( path == 0 ) { if( contains_str( PREFIX L"/bin", L"/bin", L"/usr/bin", (void *)0 ) ) { path = L"/bin" ARRAY_SEP_STR L"/usr/bin"; } else { path = L"/bin" ARRAY_SEP_STR L"/usr/bin" ARRAY_SEP_STR PREFIX L"/bin"; } } /* Allocate string long enough to hold the whole command */ wchar_t *new_cmd = halloc( context, 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] != L'/' ) { new_cmd[path_len++]=L'/'; } 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, MISSING_COMMAND_ERR_MSG, new_cmd ); wperror( L"access" ); } } } } free( path_cpy ); } return 0; } void parser_init() { if( profile ) { al_init( &profile_data); } forbidden_function = al_new(); } /** Print profiling information to the specified stream */ static 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; iskipped ) { 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 ) { if( fwprintf( out, L"%d\t%d\t", my_time, me->parse+me->exec ) < 0 ) { wperror( L"fwprintf" ); return; } for( i=0; ilevel; i++ ) { if( fwprintf( out, L"-" ) < 0 ) { wperror( L"fwprintf" ); return; } } if( fwprintf( out, L"> %ls\n", me->cmd ) < 0 ) { wperror( L"fwprintf" ); return; } } } 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, _(L"Could not write profiling information to file '%s'"), profile ); } else { if( fwprintf( f, _(L"Time\tSum\tCommand\n"), al_get_count( &profile_data ) ) < 0 ) { wperror( L"fwprintf" ); } else { print_profile( &profile_data, 0, f ); } if( fclose( f ) ) { wperror( L"fclose" ); } } al_destroy( &profile_data ); } if( lineinfo ) { sb_destroy( lineinfo ); free(lineinfo ); lineinfo = 0; } al_destroy( forbidden_function ); free( forbidden_function ); } /** Print error message to string_buffer_t if an error has occured while parsing \param target the buffer to write to \param prefix: The string token to prefix the ech line with. Usually the name of the command trying to parse something. */ static void print_errors( string_buffer_t *target, const wchar_t *prefix ) { if( error_code ) { int tmp; sb_printf( target, L"%ls: %ls\n", prefix, err_str ); tmp = current_tokenizer_pos; current_tokenizer_pos = err_pos; sb_printf( target, L"%ls", parser_current_line() ); current_tokenizer_pos=tmp; } } /** Print error message to stderr if an error has occured while parsing */ static void print_errors_stderr() { if( error_code ) { debug( 0, L"%ls", err_str ); int tmp; 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; CHECK( line, 1 ); CHECK( args, 1 ); proc_push_interactive(0); current_tokenizer = &tok; current_tokenizer_pos = 0; tok_init( &tok, line, 0 ); 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: { wchar_t *tmp = wcsdup(tok_last( &tok )); if( !tmp ) { DIE_MEM(); } if( expand_string( 0, tmp, args, 0 ) == EXPAND_ERROR ) { err_pos=tok_get_pos( &tok ); do_loop=0; } break; } case TOK_END: { break; } case TOK_ERROR: { error( SYNTAX_ERROR, tok_get_pos( &tok ), TOK_ERR_MSG, tok_last(&tok) ); do_loop=0; break; } default: { error( SYNTAX_ERROR, tok_get_pos( &tok ), UNEXPECTED_TOKEN_ERR_MSG, tok_get_desc( tok_last_type(&tok)) ); do_loop=0; break; } } } print_errors_stderr(); tok_destroy( &tok ); current_tokenizer=previous_tokenizer; current_tokenizer_pos = previous_pos; proc_pop_interactive(); return 1; } void parser_stack_trace( block_t *b, string_buffer_t *buff) { /* Validate input */ CHECK( buff, ); /* Check if we should end the recursion */ if( !b ) return; if( b->type==EVENT ) { /* This is an event handler */ sb_printf( buff, _(L"in event handler: %ls\n"), event_get_desc( b->param1.event )); sb_printf( buff, L"\n" ); /* Stop recursing at event handler. No reason to belive that any other code is relevant. It might make sense in the future to continue printing the stack trace of the code that invoked the event, if this is a programmatic event, but we can't currently detect that. */ return; } if( b->type == FUNCTION_CALL || b->type==SOURCE || b->type==SUBST) { /* These types of blocks should be printed */ int i; switch( b->type) { case SOURCE: { sb_printf( buff, _(L"in . (source) call of file '%ls',\n"), b->param1.source_dest ); break; } case FUNCTION_CALL: { sb_printf( buff, _(L"in function '%ls',\n"), b->param1.function_name ); break; } case SUBST: { sb_printf( buff, _(L"in command substitution\n") ); break; } } const wchar_t *file = b->src_filename; if( file ) { sb_printf( buff, _(L"\tcalled on line %d of file '%ls',\n"), b->src_lineno, file ); } else { sb_printf( buff, _(L"\tcalled on standard input,\n") ); } if( b->type == FUNCTION_CALL ) { if( b->param2.function_call_process->argv[1] ) { string_buffer_t tmp; sb_init( &tmp ); for( i=1; b->param2.function_call_process->argv[i]; i++ ) { sb_append2( &tmp, i>1?L" ":L"", b->param2.function_call_process->argv[i], (void *)0 ); } sb_printf( buff, _(L"\twith parameter list '%ls'\n"), (wchar_t *)tmp.buff ); sb_destroy( &tmp ); } } sb_printf( buff, L"\n" ); } /* Recursively print the next block */ parser_stack_trace( b->outer, buff ); } /** Returns true if we are currently evaluating a function. This is tested by moving down the block-scope-stack, checking every block if it is of type FUNCTION_CALL. */ static const wchar_t *is_function() { block_t *b = current_block; while( 1 ) { if( !b ) { return 0; } if( b->type == FUNCTION_CALL ) { return b->param1.function_name; } b=b->outer; } } int parser_get_lineno() { const wchar_t *whole_str; const wchar_t *function_name; int lineno; /* static const wchar_t *prev_str = 0; static int i=0; static int lineno=1; */ if( !current_tokenizer ) return -1; whole_str = tok_string( current_tokenizer ); if( !whole_str ) return -1; lineno = parse_util_lineno( whole_str, current_tokenizer_pos ); if( (function_name = is_function()) ) { lineno += function_get_definition_offset( function_name ); } return lineno; } const wchar_t *parser_current_filename() { block_t *b = current_block; while( 1 ) { if( !b ) { return reader_current_filename(); } if( b->type == FUNCTION_CALL ) { return function_get_definition_file(b->param1.function_name ); } b=b->outer; } } /** Calculates the on-screen width of the specified substring of the specified string. This function takes into account the width and allignment of the tab character, but other wise behaves like repeatedly calling wcwidth. */ static int printed_width( const wchar_t *str, int len ) { int res=0; int i; for( i=0; str[i] && ibuff ); if( file ) sb_printf( lineinfo, _(L"%ls (line %d): "), file, lineno ); else sb_printf( lineinfo, L"%ls: ", _(L"Standard input"), lineno ); offset = my_wcswidth( (wchar_t *)lineinfo->buff ) - prev_width; } else { offset=0; } // debug( 1, L"Current pos %d, line pos %d, file_length %d, is_interactive %d, offset %d\n", current_tokenizer_pos, current_line_pos, wcslen(whole_str), is_interactive, offset); /* Skip printing character position if we are in interactive mode and the error was on the first character of the line. */ if( !is_interactive || is_function() || (current_line_width!=0) ) { // Workaround since it seems impossible to print 0 copies of a character using %*lc if( offset+current_line_width ) { sb_printf( lineinfo, L"%ls\n%*lc^\n", line, offset+current_line_width, L' ' ); } else { sb_printf( lineinfo, L"%ls\n^\n", line ); } } free( line ); parser_stack_trace( current_block, lineinfo ); return (wchar_t *)lineinfo->buff; } 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; int matched_wildcard = 0, unmatched_wildcard = 0; wchar_t *unmatched = 0; int unmatched_pos=0; /* Test if this is the 'count' command. We need to special case count in the shell, since it should display a help message on 'count -h', but not on 'set foo -h; count $foo'. This is an ugly workaround and a huge hack, but as near as I can tell, the alternatives are worse. */ 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 ) { switch( tok_last_type( tok ) ) { case TOK_PIPE: { if( (p->type == INTERNAL_EXEC) ) { error( SYNTAX_ERROR, tok_get_pos( tok ), EXEC_ERR_MSG ); return; } p->pipe_fd = wcstol( tok_last( tok ), 0, 10 ); if( !p->argv ) halloc_register( j, p->argv = list_to_char_arr( args ) ); p->next = halloc( j, sizeof( process_t ) ); if( p->next == 0 ) { DIE_MEM(); } tok_next( tok ); /* Don't do anything on failiure. parse_job will notice the error flag and report any errors for us */ parse_job( p->next, j, tok ); is_finished = 1; break; } case TOK_BACKGROUND: { j->fg = 0; } case TOK_END: { if( !p->argv ) halloc_register( j, 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 ) { /* If this command should be skipped, we do not expand the arguments */ skip=1; /* But if this is in fact a case statement, then it should be evaluated */ if( (current_block->type == SWITCH) && (wcscmp( al_get( args, 0), L"case" )==0) && p->type == INTERNAL_BUILTIN ) { 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; p->actual_cmd = L"count"; } switch( expand_string( j, wcsdup(tok_last( tok )), args, 0 ) ) { case EXPAND_ERROR: { err_pos=tok_get_pos( tok ); if( error_code == 0 ) { error( SYNTAX_ERROR, tok_get_pos( tok ), _(L"Could not expand string '%ls'"), tok_last(tok) ); } break; } case EXPAND_WILDCARD_NO_MATCH: { unmatched_wildcard = 1; if( !unmatched ) { unmatched = halloc_wcsdup( j, tok_last( tok )); unmatched_pos = tok_get_pos( tok ); } break; } case EXPAND_WILDCARD_MATCH: { matched_wildcard = 1; break; } case EXPAND_OK: { break; } } } 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 Otherwise, bogus errors may be the result. (Do check that token is string, though) */ if( current_block->skip ) { tok_next( tok ); 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)) ); } break; } new_io = halloc( j, 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 = (wchar_t *)expand_one( j, wcsdup( tok_last( tok ) ), 0); if( target == 0 && error_code == 0 ) { error( SYNTAX_ERROR, tok_get_pos( tok ), REDIRECT_TOKEN_ERR_MSG, tok_last( tok ) ); } break; } default: error( SYNTAX_ERROR, tok_get_pos( tok ), REDIRECT_TOKEN_ERR_MSG, tok_get_desc( tok_last_type(tok)) ); } if( target == 0 || wcslen( target )==0 ) { if( error_code == 0 ) error( SYNTAX_ERROR, tok_get_pos( tok ), _(L"Invalid IO redirection") ); tok_next(tok); } else { switch( type ) { case TOK_REDIRECT_APPEND: new_io->io_mode = IO_FILE; new_io->param2.flags = O_CREAT | O_APPEND | O_WRONLY; new_io->param1.filename = target; break; case TOK_REDIRECT_OUT: new_io->io_mode = IO_FILE; new_io->param2.flags = O_CREAT | O_WRONLY | O_TRUNC; new_io->param1.filename = target; break; case TOK_REDIRECT_IN: new_io->io_mode = IO_FILE; new_io->param2.flags = O_RDONLY; new_io->param1.filename = target; break; case TOK_REDIRECT_FD: { if( wcscmp( target, L"-" ) == 0 ) { new_io->io_mode = IO_CLOSE; } else { new_io->io_mode = IO_FD; new_io->param1.old_fd = wcstol( target, 0, 10 ); if( ( new_io->param1.old_fd < 0 ) || ( new_io->param1.old_fd > 10 ) ) { error( SYNTAX_ERROR, tok_get_pos( tok ), _(L"Requested redirection to something that is not a file descriptor %ls"), target ); tok_next(tok); } } break; } } } j->io = io_add( j->io, new_io ); } break; case TOK_ERROR: { error( SYNTAX_ERROR, tok_get_pos( tok ), TOK_ERR_MSG, tok_last(tok) ); return; } default: error( SYNTAX_ERROR, tok_get_pos( tok ), UNEXPECTED_TOKEN_ERR_MSG, tok_get_desc( tok_last_type(tok)) ); tok_next(tok); break; } if( (is_finished) || (error_code != 0) ) break; tok_next( tok ); } if( !error_code ) { if( unmatched_wildcard && !matched_wildcard ) { 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; } } } return; } /** Fully parse a single job. Does not call exec on it, but any command substitutions in the job will be executed. \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 = al_halloc( j ); // 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; int prev_tokenizer_pos = current_tokenizer_pos; current_tokenizer_pos = tok_get_pos( tok ); while( al_get_count( args ) == 0 ) { wchar_t *nxt=0; int consumed = 0; // Set to one if the command requires a second command, like e.g. while does int mark; // Use to save the position of the beginning of the token switch( tok_last_type( tok )) { case TOK_STRING: { nxt = expand_one( j, wcsdup(tok_last( tok )), EXPAND_SKIP_SUBSHELL | EXPAND_SKIP_VARIABLES); if( nxt == 0 ) { error( SYNTAX_ERROR, tok_get_pos( tok ), ILLEGAL_CMD_ERR_MSG, tok_last( tok ) ); current_tokenizer_pos = prev_tokenizer_pos; return 0; } break; } case TOK_ERROR: { error( SYNTAX_ERROR, tok_get_pos( tok ), TOK_ERR_MSG, tok_last(tok) ); current_tokenizer_pos = prev_tokenizer_pos; return 0; } 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) ) ); } current_tokenizer_pos = prev_tokenizer_pos; return 0; } default: { error( SYNTAX_ERROR, tok_get_pos( tok ), CMD_ERR_MSG, tok_get_desc( tok_last_type(tok) ) ); current_tokenizer_pos = prev_tokenizer_pos; return 0; } } mark = tok_get_pos( tok ); /* Test if this command is one of the many special builtins that work directly on the parser, like e.g. 'not', that simply flips the status inversion flag in the job. */ 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; consumed=1; } } 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; consumed=1; } } 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; consumed=1; } } else if( wcscmp( L"and", nxt )==0 ) { tok_next( tok ); if( tok_last(tok)[0] == L'-' ) { tok_set_pos( tok, mark); } else { j->skip = proc_get_last_status(); consumed=1; } } else if( wcscmp( L"or", nxt )==0 ) { tok_next( tok ); if( tok_last(tok)[0] == L'-' ) { tok_set_pos( tok, mark); } else { j->skip = !proc_get_last_status(); consumed=1; } } else if( wcscmp( L"exec", nxt )==0 ) { if( p != j->first_process ) { error( SYNTAX_ERROR, tok_get_pos( tok ), EXEC_ERR_MSG ); current_tokenizer_pos = prev_tokenizer_pos; 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; consumed=1; current_tokenizer_pos = prev_tokenizer_pos; } } else if( wcscmp( L"while", nxt ) ==0 ) { int new_block = 0; tok_next( tok ); if( ( current_block->type != WHILE ) ) { new_block = 1; } else if( current_block->param1.while_state == WHILE_TEST_AGAIN ) { current_block->param1.while_state = WHILE_TEST_FIRST; } else { new_block = 1; } if( new_block ) { parser_push_block( WHILE ); current_block->param1.while_state=WHILE_TEST_FIRST; current_block->tok_pos = mark; } consumed=1; is_new_block=1; } else if( wcscmp( L"if", nxt ) ==0 ) { tok_next( tok ); parser_push_block( IF ); current_block->param1.if_state=0; current_block->tok_pos = mark; is_new_block=1; consumed=1; } /* Test if we need another command */ if( consumed ) { /* Yes we do, around in the loop for another lap, then! */ continue; } if( use_function && !current_block->skip ) { 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, tok_get_pos( tok ), RECURSION_ERR_MSG ); } 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 = L""; } else { p->actual_cmd = parser_get_filename( j, (wchar_t *)al_get( args, 0 ) ); /* 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( j, (wchar_t *)al_get( args, 0 ) ); if( pp ) { wchar_t *tmp; tmp = (wchar_t *)al_get( args, 0 ); al_truncate( args, 0 ); al_push( args, halloc_wcsdup( j, 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 { int tmp; wchar_t *cmd = (wchar_t *)al_get( args, 0 ); /* We couldn't find the specified command. What we want to happen now is that the specified job won't get executed, and an error message is printed on-screen, but otherwise, the parsing/execution of the file continues. Because of this, we don't want to call error(), since that would stop execution of the file. Instead we let p->actual_command be 0 (null), which will cause the job to silently not execute. We also print an error message. */ if( wcschr( cmd, L'=' ) ) { wchar_t *cpy = halloc_wcsdup( j, cmd ); wchar_t *valpart = wcschr( cpy, L'=' ); *valpart++=0; debug( 0, COMMAND_ASSIGN_ERR_MSG, cmd, cpy, valpart); } else if(cmd[0]==L'$') { wchar_t *val = env_get( cmd+1 ); if( val ) { debug( 0, _(L"Variables may not be used as commands. Instead, define a function like 'function %ls; %ls $argv; end'. See the help section for the function command by typing 'help function'." ), cmd+1, val, cmd ); } else { debug( 0, _(L"Variables may not be used as commands. Instead, define a function. See the help section for the function command by typing 'help function'." ), cmd ); } } else if(wcschr( cmd, L'$' )) { debug( 0, _(L"Commands may not contain variables. Use the eval builtin instead, like 'eval %ls'. See the help section for the eval command by typing 'help eval'." ), cmd, cmd ); } else { debug( 0, _(L"Unknown command '%ls'"), cmd ); } tmp = current_tokenizer_pos; current_tokenizer_pos = tok_get_pos(tok); fwprintf( stderr, L"%ls", parser_current_line() ); current_tokenizer_pos=tmp; j->skip=1; } } } } } 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, tok_get_pos( tok ), BLOCK_END_ERR_MSG ); } 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: { error( SYNTAX_ERROR, current_tokenizer_pos, BLOCK_END_ERR_MSG ); } } tok_destroy( &subtok ); } if( make_sub_block ) { int end_pos = end-tok_string( tok ); wchar_t *sub_block= halloc_wcsndup( j, tok_string( tok ) + current_tokenizer_pos, end_pos - current_tokenizer_pos); p->type = INTERNAL_BLOCK; al_set( args, 0, sub_block ); tok_set_pos( tok, end_pos ); while( prev_block != current_block ) { parser_pop_block(); } } else tok_next( tok ); } else tok_next( tok ); if( !error_code ) { if( p->type == INTERNAL_BUILTIN && parser_skip_arguments( (wchar_t *)al_get(args, 0) ) ) { if( !p->argv ) halloc_register( j, p->argv = list_to_char_arr( args ) ); } else { parse_job_main_loop( p, j, tok, args ); } } if( error_code ) { /* Make sure the block stack is consistent */ while( prev_block != current_block ) { parser_pop_block(); } } current_tokenizer_pos = prev_tokenizer_pos; 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; for( p = j->first_process; p; p=p->next ) { if( p->type == INTERNAL_BUILTIN ) { if(( wcscmp( p->argv[0], L"for" )==0) || ( wcscmp( p->argv[0], L"switch" )==0) || ( wcscmp( p->argv[0], L"begin" )==0) || ( 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 ) && (current_block->param1.if_state != 0)) { 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 ); long long t1=0, t2=0, t3=0; profile_element_t *p=0; int skip = 0; int job_begin_pos, prev_tokenizer_pos; 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->fg=1; j->terminal = j->job_control && (!is_subshell && !is_event); j->skip_notification = is_subshell || is_block || is_event || (!is_interactive); current_block->job = j; if( is_interactive ) { if( tcgetattr (0, &j->tmodes) ) { tok_next( tok ); wperror( L"tcgetattr" ); job_free( j ); break; } } j->first_process = halloc( j, sizeof( process_t ) ); job_begin_pos = tok_get_pos( tok ); 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 = halloc_wcsndup( j, tok_string(tok)+start_pos, stop_pos-start_pos ); } else j->command = L""; if( profile ) { t2 = get_time(); p->cmd = wcsdup( j->command ); p->skipped=current_block->skip; } skip |= current_block->skip; skip |= j->wildcard_error; skip |= j->skip; if(!skip ) { int was_builtin = 0; // if( j->first_process->type==INTERNAL_BUILTIN && !j->first_process->next) // was_builtin = 1; prev_tokenizer_pos = current_tokenizer_pos; current_tokenizer_pos = job_begin_pos; exec( j ); current_tokenizer_pos = prev_tokenizer_pos; /* Only external commands require a new fishd barrier */ if( !was_builtin ) proc_had_barrier=0; } 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 ) { switch( current_block->param1.while_state ) { case WHILE_TEST_FIRST: { current_block->skip = proc_get_last_status()!= 0; current_block->param1.while_state=WHILE_TESTED; } break; } } if( current_block->type == IF ) { if( (!current_block->param1.if_state) && (!current_block->skip) ) { current_block->skip = proc_get_last_status()!= 0; current_block->param1.if_state++; } } } else { /* This job could not be properly parsed. We free it instead, and set the status to 1. This should be rare, since most errors should be detected by the ahead of time validator. */ job_free( j ); proc_set_last_status( 1 ); } current_block->job = 0; break; } case TOK_END: { if( tok_has_next( tok )) tok_next( tok ); break; } 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) ) ); } return; } case TOK_ERROR: { error( SYNTAX_ERROR, tok_get_pos( tok ), TOK_ERR_MSG, tok_last(tok) ); return; } default: { error( SYNTAX_ERROR, tok_get_pos( tok ), CMD_ERR_MSG, tok_get_desc( tok_last_type(tok)) ); return; } } job_reap( 0 ); } 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; array_list_t *prev_forbidden = forbidden_function; if( block_type == SUBST ) { forbidden_function = al_new(); } forbid_count = al_get_count( forbidden_function ); block_io = io; job_reap( 0 ); debug( 4, L"eval: %ls", cmd ); if( !cmd ) { debug( 1, EVAL_NULL_ERR_MSG ); debug( 1, BUGREPORT_MSG, PACKAGE_BUGREPORT ); return 1; } if( (block_type != TOP) && (block_type != SUBST)) { debug( 1, INVALID_SCOPE_ERR_MSG, parser_get_block_desc( block_type ) ); debug( 1, BUGREPORT_MSG, PACKAGE_BUGREPORT ); return 1; } eval_level++; parser_push_block( block_type ); current_tokenizer = malloc( sizeof(tokenizer)); tok_init( current_tokenizer, cmd, 0 ); error_code = 0; event_fire( 0 ); while( tok_has_next( current_tokenizer ) && !error_code && !sanity_check() && !exit_status() ) { eval_job( current_tokenizer ); event_fire( 0 ); } int prev_block_type = current_block->type; parser_pop_block(); while( start_current_block != current_block ) { if( current_block == 0 ) { debug( 0, _(L"End of block mismatch. Program terminating.") ); debug( 0, BUGREPORT_MSG, PACKAGE_BUGREPORT ); exit(1); break; } if( (!error_code) && (!exit_status()) && (!proc_get_last_status()) ) { char *h; //debug( 2, L"Status %d\n", proc_get_last_status() ); debug( 1, L"%ls", parser_get_block_desc( current_block->type ) ); debug( 1, BLOCK_END_ERR_MSG ); fwprintf( stderr, L"%ls", parser_current_line() ); h = builtin_help_get( L"end" ); if( h ) fwprintf( stderr, L"%s", h ); break; } prev_block_type = current_block->type; parser_pop_block(); } print_errors_stderr(); tok_destroy( current_tokenizer ); free( current_tokenizer ); while( al_get_count( forbidden_function ) > forbid_count ) parser_allow_function(); if( block_type == SUBST ) { al_destroy( forbidden_function ); free( forbidden_function ); } /* Restore previous eval state */ forbidden_function = prev_forbidden; current_tokenizer=previous_tokenizer; block_io = prev_io; eval_level--; code=error_code; error_code=0; job_reap( 0 ); return code; } /** \return the block type created by the specified builtin, or -1 on error. */ static int parser_get_block_type( const wchar_t *cmd ) { int i; for( i=0; block_lookup[i].desc; i++ ) { if( block_lookup[i].name && (wcscmp( block_lookup[i].name, cmd ) == 0) ) { return block_lookup[i].type; } } return -1; } /** \return the block type created by the specified builtin, or -1 on error. */ static const wchar_t *parser_get_block_command( int type ) { int i; for( i=0; block_lookup[i].desc; i++ ) { if( block_lookup[i].type == type ) { return block_lookup[i].name; } } return 0; } /** Test if this argument contains any errors. Detected errors include syntax errors in command substitutions, improperly escaped characters and improper use of the variable expansion operator. */ static int parser_test_argument( const wchar_t *arg, string_buffer_t *out, const wchar_t *prefix, int offset ) { wchar_t *unesc; wchar_t *pos; int err=0; wchar_t *paran_begin, *paran_end; wchar_t *arg_cpy; int do_loop = 1; CHECK( arg, 1 ); arg_cpy = wcsdup( arg ); while( do_loop ) { switch( parse_util_locate_cmdsubst(arg_cpy, ¶n_begin, ¶n_end, 0 ) ) { case -1: err=1; if( out ) { error( SYNTAX_ERROR, offset, L"Mismatched parans" ); print_errors( out, prefix); } free( arg_cpy ); return 1; case 0: do_loop = 0; break; case 1: { wchar_t *subst = wcsndup( paran_begin+1, paran_end-paran_begin-1 ); string_buffer_t tmp; sb_init( &tmp ); sb_append_substring( &tmp, arg_cpy, paran_begin - arg_cpy); sb_append_char( &tmp, INTERNAL_SEPARATOR); sb_append( &tmp, paran_end+1); // debug( 1, L"%ls -> %ls %ls", arg_cpy, subst, tmp.buff ); err |= parser_test( subst, out, prefix ); free( subst ); free( arg_cpy ); arg_cpy = (wchar_t *)tmp.buff; /* Do _not_ call sb_destroy on this stringbuffer - it's buffer is used as the new 'arg_cpy'. It is free'd at the end of the loop. */ break; } } } unesc = unescape( arg_cpy, 1 ); if( !unesc ) { if( out ) { error( SYNTAX_ERROR, offset, L"Invalid token '%ls'", arg_cpy ); print_errors( out, prefix); } return 1; } else { /* Check for invalid variable expansions */ for( pos = unesc; *pos; pos++ ) { switch( *pos ) { case VARIABLE_EXPAND: case VARIABLE_EXPAND_SINGLE: { wchar_t n = *(pos+1); if( n != VARIABLE_EXPAND && n != VARIABLE_EXPAND_SINGLE && !wcsvarchr(n) ) { err=1; if( out ) { expand_variable_error( unesc, pos-unesc, offset ); print_errors( out, prefix); } } break; } } } } free( arg_cpy ); free( unesc ); return err; } int parser_test_args(const wchar_t * buff, string_buffer_t *out, const wchar_t *prefix ) { tokenizer tok; tokenizer *previous_tokenizer = current_tokenizer; int previous_pos = current_tokenizer_pos; int do_loop = 1; int err = 0; CHECK( buff, 1 ); current_tokenizer = &tok; for( tok_init( &tok, buff, 0 ); do_loop && tok_has_next( &tok ); tok_next( &tok ) ) { current_tokenizer_pos = tok_get_pos( &tok ); switch( tok_last_type( &tok ) ) { case TOK_STRING: { err |= parser_test_argument( tok_last( &tok ), out, prefix, tok_get_pos( &tok ) ); break; } case TOK_END: { break; } case TOK_ERROR: { if( out ) { error( SYNTAX_ERROR, tok_get_pos( &tok ), TOK_ERR_MSG, tok_last(&tok) ); print_errors( out, prefix ); } err=1; do_loop=0; break; } default: { if( out ) { error( SYNTAX_ERROR, tok_get_pos( &tok ), UNEXPECTED_TOKEN_ERR_MSG, tok_get_desc( tok_last_type(&tok)) ); print_errors( out, prefix ); } err=1; do_loop=0; break; } } } tok_destroy( &tok ); current_tokenizer=previous_tokenizer; current_tokenizer_pos = previous_pos; error_code=0; return err; } int parser_test( const wchar_t * buff, string_buffer_t *out, const wchar_t *prefix ) { tokenizer tok; /* Set to one if a command name has been given for the currently parsed process specification */ 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]; /* Set to 1 if the current command is inside a pipeline */ int is_pipeline = 0; /* Set to one if the currently specified process can not be used inside a pipeline */ int forbid_pipeline = 0; /* Set to one if an additional process specification is needed */ int needs_cmd=0; /* halloc context used for calls to expand() and other memory allocations. Free'd at end of this function. */ void *context; /* Counter on the number of arguments this function has encountered so far. Is set to -1 when the count is unknown, i.e. after encountering an argument that contains substitutions that can expand to more/less arguemtns then 1. */ int arg_count=0; /* The currently validated command. */ wchar_t *cmd=0; CHECK( buff, 1 ); context = halloc( 0, 0 ); current_tokenizer = &tok; for( tok_init( &tok, buff, 0 ); tok_has_next( &tok ); 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; arg_count=0; if( !(cmd = expand_one( context, wcsdup( tok_last( &tok ) ), EXPAND_SKIP_SUBSHELL | EXPAND_SKIP_VARIABLES ) ) ) { err=1; if( out ) { error( SYNTAX_ERROR, tok_get_pos( &tok ), ILLEGAL_CMD_ERR_MSG, tok_last( &tok ) ); print_errors( out, prefix ); } break; } if( needs_cmd ) { /* end is not a valid command when a followup command is needed, such as after 'and' or 'while' */ if( contains_str( cmd, L"end", 0 ) ) { err=1; if( out ) { error( SYNTAX_ERROR, tok_get_pos( &tok ), COND_ERR_MSG ); print_errors( out, prefix ); } } needs_cmd=0; } /* Decrement block count on end command */ if( wcscmp(cmd, L"end")==0) { tok_next( &tok ); count--; tok_set_pos( &tok, mark ); } /* Handle block commands */ if( parser_is_block( cmd ) ) { if( count >= BLOCK_MAX_COUNT ) { error( SYNTAX_ERROR, tok_get_pos( &tok ), BLOCK_ERR_MSG ); print_errors( out, prefix ); } else { block_type[count] = parser_get_block_type( cmd ); block_pos[count] = current_tokenizer_pos; tok_next( &tok ); count++; tok_set_pos( &tok, mark ); } } /* 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( cmd ) && !parser_skip_arguments(cmd ) ) { needs_cmd = 1; had_cmd = 0; } if( contains_str( cmd, L"or", L"and", 0 ) ) { /* 'or' and 'and' can not be used inside pipelines */ if( is_pipeline ) { err=1; if( out ) { error( SYNTAX_ERROR, tok_get_pos( &tok ), EXEC_ERR_MSG ); print_errors( out, prefix ); } } } /* There are a lot of situations where pipelines are forbidden, including when using the exec builtin. */ if( parser_is_pipe_forbidden( cmd ) ) { if( is_pipeline ) { err=1; if( out ) { error( SYNTAX_ERROR, tok_get_pos( &tok ), EXEC_ERR_MSG ); print_errors( out, prefix ); } } forbid_pipeline = 1; } /* Test that the case builtin is only used directly in a switch block */ if( wcscmp( L"case", cmd )==0 ) { if( !count || block_type[count-1]!=SWITCH ) { err=1; if( out ) { char *h; error( SYNTAX_ERROR, tok_get_pos( &tok ), INVALID_CASE_ERR_MSG ); print_errors( out, prefix); h = builtin_help_get( L"case" ); if( h ) sb_printf( out, L"%s", h ); } } } /* Test that break and continue are only used within loop blocks */ if( contains_str( cmd, L"break", L"continue", (void *)0 ) ) { 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; if( out ) { error( SYNTAX_ERROR, tok_get_pos( &tok ), INVALID_LOOP_ERR_MSG ); print_errors( out, prefix ); } } } /* Test that else is only used directly in an if-block */ if( wcscmp( L"else", cmd )==0 ) { if( !count || block_type[count-1]!=IF ) { err=1; if( out ) { error( SYNTAX_ERROR, tok_get_pos( &tok ), INVALID_ELSE_ERR_MSG ); print_errors( out, prefix ); } } } /* Test that end is not used when not inside any block */ if( count < 0 ) { err = 1; if( out ) { char *h; error( SYNTAX_ERROR, tok_get_pos( &tok ), INVALID_END_ERR_MSG ); print_errors( out, prefix ); h = builtin_help_get( L"end" ); if( h ) sb_printf( out, L"%s", h ); } } } else { err |= parser_test_argument( tok_last( &tok ), out, prefix, tok_get_pos( &tok ) ); /* If possible, keep track of number of supplied arguments */ if( arg_count >= 0 && expand_is_clean( tok_last( &tok ) ) ) { arg_count++; } else { arg_count = -1; } /* Try to make sure the second argument to 'for' is 'in' */ if( wcscmp( cmd, L"for" ) == 0 ) { if( arg_count == 2 ) { if( wcscmp( tok_last( &tok ), L"in" ) != 0 ) { err = 1; if( out ) { error( SYNTAX_ERROR, tok_get_pos( &tok ), BUILTIN_FOR_ERR_IN, L"for" ); print_errors( out, prefix ); } } } } } break; } case TOK_REDIRECT_OUT: case TOK_REDIRECT_IN: case TOK_REDIRECT_APPEND: case TOK_REDIRECT_FD: { if( !had_cmd ) { err = 1; if( out ) { error( SYNTAX_ERROR, tok_get_pos( &tok ), INVALID_REDIRECTION_ERR_MSG ); print_errors( out, prefix ); } } break; } case TOK_END: { if( needs_cmd && !had_cmd ) { err = 1; if( out ) { error( SYNTAX_ERROR, tok_get_pos( &tok ), CMD_ERR_MSG, tok_get_desc( tok_last_type(&tok))); print_errors( out, prefix ); } } needs_cmd=0; had_cmd = 0; is_pipeline=0; forbid_pipeline=0; break; } case TOK_PIPE: { if( !had_cmd ) { err=1; if( out ) { if( tok_get_pos(&tok)>0 && buff[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))); } print_errors( out, prefix ); } } else if( forbid_pipeline ) { err=1; if( out ) { error( SYNTAX_ERROR, tok_get_pos( &tok ), EXEC_ERR_MSG ); print_errors( out, prefix ); } } else { needs_cmd=1; is_pipeline=1; had_cmd=0; } break; } case TOK_BACKGROUND: { if( !had_cmd ) { err = 1; if( out ) { if( tok_get_pos(&tok)>0 && buff[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))); } print_errors( out, prefix ); } } if( had_cmd ) { had_cmd = 0; } break; } case TOK_ERROR: default: err = 1; if( out ) { error( SYNTAX_ERROR, tok_get_pos( &tok ), TOK_ERR_MSG, tok_last(&tok) ); print_errors( out, prefix ); //debug( 2, tok_last( &tok) ); } break; } } if( needs_cmd ) { err=1; if( out ) { error( SYNTAX_ERROR, tok_get_pos( &tok ), COND_ERR_MSG ); print_errors( out, prefix ); } } if( out && count>0 ) { const char *h; const wchar_t *cmd; error( SYNTAX_ERROR, block_pos[count-1], BLOCK_END_ERR_MSG ); print_errors( out, prefix ); cmd = parser_get_block_command( block_type[count -1] ); if( cmd ) { h = builtin_help_get( cmd ); if( cmd ) { sb_printf( out, L"%s", h ); } } } tok_destroy( &tok ); current_tokenizer=previous_tokenizer; current_tokenizer_pos = previous_pos; error_code=0; halloc_free( context ); return err | ((count!=0)<<1); }