/** \file parser.h The fish parser. */ #ifndef FISH_PARSER_H #define FISH_PARSER_H #include #include "proc.h" #include "util.h" #include "parser.h" #include "event.h" #include #define PARSER_TEST_ERROR 1 #define PARSER_TEST_INCOMPLETE 2 /** event_block_t represents a block on events of the specified type */ typedef struct event_block { /** The types of events to block. This is interpreted as a bitset whete the value is 1 for every bit corresponding to a blocked event type. For example, if EVENT_VARIABLE type events should be blocked, (type & 1< blocks; /** Last error code */ int error_code; /** Position of last error */ int err_pos; /** Description of last error */ string_buffer_t *err_buff; /** Pointer to the current tokenizer */ tokenizer *current_tokenizer; /** String for representing the current line */ string_buffer_t *lineinfo; /** This is the position of the beginning of the currently parsed command */ int current_tokenizer_pos; /** List of called functions, used to help prevent infinite recursion */ wcstring_list_t forbidden_function; /** String index where the current job started. */ int job_start_pos; /** 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. */ int eval_level; /* No copying allowed */ parser_t(const parser_t&); parser_t& operator=(const parser_t&); /** Returns the name of the currently evaluated function if we are currently evaluating a function, null otherwise. This is tested by moving down the block-scope-stack, checking every block if it is of type FUNCTION_CALL. */ const wchar_t *is_function() const; void parse_job_argument_list( process_t *p, job_t *j, tokenizer *tok, std::vector& ); int parse_job( process_t *p, job_t *j, tokenizer *tok ); void skipped_exec( job_t * j ); void eval_job( tokenizer *tok ); int parser_test_argument( const wchar_t *arg, string_buffer_t *out, const wchar_t *prefix, int offset ); void print_errors( string_buffer_t *target, const wchar_t *prefix ); void print_errors_stderr(); public: std::vector profile_items; /** Get the "principal" parser, whatever that is */ static parser_t &principal_parser(); /** Create a parser of the given type */ parser_t(enum parser_type_t type); /** The current innermost block */ block_t *current_block; /** Global event blocks */ event_block_t *global_event_block; /** Current block level io redirections */ io_data_t *block_io; /** Evaluate the expressions contained in cmd. \param cmd the string to evaluate \param io io redirections to perform on all started jobs \param block_type The type of block to push on the block stack \return 0 on success, 1 otherwise */ int eval( const wcstring &cmd, io_data_t *io, enum block_type_t block_type ); /** Evaluate line as a list of parameters, i.e. tokenize it and perform parameter expansion and cmdsubst execution on the tokens. The output is inserted into output, and should be freed by the caller. \param line Line to evaluate \param output List to insert output to */ /** \param line Line to evaluate \param output List to insert output to */ int eval_args( const wchar_t *line, std::vector &output ); /** Sets the current evaluation error. This function should only be used by libraries that are called by \param ec The new error code \param p The character offset at which the error occured \param str The printf-style error message filter */ void error( int ec, int p, const wchar_t *str, ... ); /** Returns a string describing the current parser pisition in the format 'FILENAME (line LINE_NUMBER): LINE'. Example: init.fish (line 127): ls|grep pancake */ const wchar_t *current_line(); /** Returns the current line number */ int get_lineno() const; /** Returns the current position in the latest string of the tokenizer. */ int get_pos() const; /** Returns the position where the current job started in the latest string of the tokenizer. */ int get_job_pos() const; /** Set the current position in the latest string of the tokenizer. */ void set_pos( int p); /** Get the string currently parsed */ const wchar_t *get_buffer() const; /** Create block of specified type */ void push_block( int type); /** Remove the outermost block namespace */ void pop_block(); /** Return a description of the given blocktype */ const wchar_t *get_block_desc( int block ) const; /** Test if the specified string can be parsed, or if more bytes need to be read first. The result will have the PARSER_TEST_ERROR bit set if there is a syntax error in the code, and the PARSER_TEST_INCOMPLETE bit set if the code contains unclosed blocks. \param buff the text buffer to test \param block_level if non-null, the block nesting level will be filled out into this array \param out if non-null, any errors in the command will be filled out into this buffer \param prefix the prefix string to prepend to each error message written to the \c out buffer */ int test( const wchar_t * buff, int *block_level, string_buffer_t *out, const wchar_t *prefix ); /** Test if the specified string can be parsed as an argument list, e.g. sent to eval_args. The result has the first bit set if the string contains errors, and the second bit is set if the string contains an unclosed block. */ int test_args( const wchar_t * buff, string_buffer_t *out, const wchar_t *prefix ); /** Tell the parser that the specified function may not be run if not inside of a conditional block. This is to remove some possibilities of infinite recursion. */ void forbid_function( const wcstring &function ); /** Undo last call to parser_forbid_function(). */ void allow_function(); /** Initialize static parser data */ void init(); /** Destroy static parser data */ void destroy(); /** This function checks if the specified string is a help option. \param s the string to test \param min_match is the minimum number of characters that must match in a long style option, i.e. the longest common prefix between --help and any other option. If less than 3, 3 will be assumed. */ int is_help( const wchar_t *s, int min_match ) const; /** Returns the file currently evaluated by the parser. This can be different than reader_current_filename, e.g. if we are evaulating a function defined in a different file than the one curently read. */ const wchar_t *current_filename() const; /** Write a stack trace starting at the specified block to the specified string_buffer_t */ void stack_trace( block_t *b, string_buffer_t *buff); int get_block_type( const wchar_t *cmd ) const; const wchar_t *get_block_command( int type ) const; }; #endif