/** \file parser.h The fish parser. */ #ifndef FISH_PARSER_H #define FISH_PARSER_H #include #include "proc.h" #include "util.h" #include "event.h" #include "function.h" #include "parse_tree.h" #include /** event_blockage_t represents a block on events of the specified type */ struct event_blockage_t { /** 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< event_blockage_list_t; inline bool event_block_list_blocks_type(const event_blockage_list_t &ebls, int type) { for (event_blockage_list_t::const_iterator iter = ebls.begin(); iter != ebls.end(); ++iter) { if (iter->typemask & (1<typemask & (1<made_fake ? FAKE : this->block_type; } /** Mark a block as fake; this is used by the return statement. */ void mark_as_fake() { this->made_fake = true; } bool skip; /**< Whether execution of the commands in this block should be skipped */ bool had_command; /**< Set to non-zero once a command has been executed in this block */ int tok_pos; /**< The start index of the block */ node_offset_t node_offset; /* Offset of the node */ /** Status for the current loop block. Can be any of the values from the loop_status enum. */ int loop_status; /** The job that is currently evaluated in the specified block. */ job_t *job; /** Name of file that created this block */ const wchar_t *src_filename; /** Line number where this block was created */ int src_lineno; /** Whether we should pop the environment variable stack when we're popped off of the block stack */ bool wants_pop_env; /** List of event blocks. */ event_blockage_list_t event_blocks; /** Destructor */ virtual ~block_t(); }; struct if_block_t : public block_t { bool if_expr_evaluated; // whether we've evaluated the if expression bool is_elseif_entry; // whether we're at the beginning of an ELSEIF branch bool any_branch_taken; // whether the clause of the if statement or any elseif has been found to be true bool else_evaluated; // whether we've encountered a terminal else block if_block_t(); }; struct event_block_t : public block_t { event_t const event; event_block_t(const event_t &evt); }; struct function_block_t : public block_t { const process_t *process; wcstring name; function_block_t(const process_t *p, const wcstring &n, bool shadows); }; struct source_block_t : public block_t { const wchar_t * const source_file; source_block_t(const wchar_t *src); }; struct for_block_t : public block_t { wcstring variable; // the variable that will be assigned each value in the sequence wcstring_list_t sequence; // the sequence of values for_block_t(const wcstring &var); }; struct while_block_t : public block_t { int status; while_block_t(); }; struct switch_block_t : public block_t { bool switch_taken; const wcstring switch_value; switch_block_t(const wcstring &sv); }; struct fake_block_t : public block_t { fake_block_t(); }; struct function_def_block_t : public block_t { function_data_t function_data; function_def_block_t(); }; struct scope_block_t : public block_t { scope_block_t(block_type_t type); //must be BEGIN, TOP or SUBST }; struct breakpoint_block_t : public block_t { breakpoint_block_t(); }; /** Possible states for a loop */ enum loop_status { LOOP_NORMAL, /**< Current loop block executed as normal */ LOOP_BREAK, /**< Current loop block should be removed */ LOOP_CONTINUE, /**< Current loop block should be skipped */ }; /** Possible states for a while block */ enum while_status { WHILE_TEST_FIRST, /**< This is the first command of the first lap of a while loop */ WHILE_TEST_AGAIN, /**< This is not the first lap of the while loop, but it is the first command of the loop */ WHILE_TESTED, /**< This is not the first command in the loop */ } ; /** Errors that can be generated by the parser */ enum parser_error { /** No error */ NO_ERR=0, /** An error in the syntax */ SYNTAX_ERROR, /** Error occured while evaluating commands */ EVAL_ERROR, /** Error while evaluating cmdsubst */ CMDSUBST_ERROR, }; enum parser_type_t { PARSER_TYPE_NONE, PARSER_TYPE_GENERAL, PARSER_TYPE_FUNCTIONS_ONLY, PARSER_TYPE_COMPLETIONS_ONLY, PARSER_TYPE_ERRORS_ONLY }; struct profile_item_t { /** 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. */ size_t level; /** If the execution of this command was skipped. */ bool skipped; /** The command string. */ wcstring cmd; }; struct tokenizer_t; class parse_execution_context_t; class parser_t { friend class parse_execution_context_t; private: enum parser_type_t parser_type; /** Whether or not we output errors */ const bool show_errors; /** Last error code */ int error_code; /** Position of last error */ int err_pos; /** Indication that we should skip all blocks */ bool cancellation_requested; /** Indicates that we are within the process of initializing fish */ bool is_within_fish_initialization; /** Stack of execution contexts. We own these pointers and must delete them */ std::vector execution_contexts; /** Description of last error */ wcstring err_buff; /** Pointer to the current tokenizer */ tokenizer_t *current_tokenizer; /** String for representing the current line */ wcstring 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; /** The jobs associated with this parser */ job_list_t my_job_list; /** The list of blocks, allocated with new. It's our responsibility to delete these */ std::vector block_stack; /** 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&); void parse_job_argument_list(process_t *p, job_t *j, tokenizer_t *tok, std::vector&, bool); int parse_job(process_t *p, job_t *j, tokenizer_t *tok); void skipped_exec(job_t * j); void eval_job(tokenizer_t *tok); int parser_test_argument(const wchar_t *arg, wcstring *out, const wchar_t *prefix, int offset); void print_errors(wcstring &target, const wchar_t *prefix); void print_errors_stderr(); /** Create a job */ job_t *job_create(const io_chain_t &io); /** Adds a job to the beginning of the job list. */ void job_add(job_t *job); public: std::vector profile_items; /** 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; /** Get the "principal" parser, whatever that is */ static parser_t &principal_parser(); /** Indicates that execution of all blocks in the principal parser should stop. This is called from signal handlers! */ static void skip_all_blocks(); /** Create a parser of the given type */ parser_t(enum parser_type_t type, bool show_errors); /** Global event blocks */ event_blockage_list_t global_event_blocks; /** Current block level io redirections */ io_chain_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_str, const io_chain_t &io, enum block_type_t block_type); int eval_new_parser(const wcstring &cmd, const io_chain_t &io, enum block_type_t block_type); /** Evaluates a block node at the given node offset in the topmost execution context */ int eval_block_node(node_offset_t node_idx, const io_chain_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. \param line Line to evaluate \param output List to insert output to */ /** \param line Line to evaluate \param output List to insert output to */ void 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, size_t 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 line number for the character at the given index */ int line_number_of_character_at_offset(size_t idx) 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); /** Returns the block at the given index. 0 corresponds to the innermost block. Returns NULL when idx is at or equal to the number of blocks. */ const block_t *block_at_index(size_t idx) const; block_t *block_at_index(size_t idx); /** Returns the current (innermost) block */ const block_t *current_block() const; block_t *current_block(); /** Count of blocks */ size_t block_count() const { return block_stack.size(); } /** Get the string currently parsed */ const wchar_t *get_buffer() const; /** Get the list of jobs */ job_list_t &job_list() { return my_job_list; } /* Hackish. In order to correctly report the origin of code with no associated file, we need to know whether it's run during initialization or not. */ void set_is_within_fish_initialization(bool flag); /** Pushes the block. pop_block will call delete on it. */ void push_block(block_t *newv); /** Remove the outermost block namespace */ void pop_block(); /** Remove the outermost block, asserting it's the given one */ void pop_block(const block_t *b); /** Return a description of the given blocktype */ const wchar_t *get_block_desc(int block) const; /** Removes a job */ bool job_remove(job_t *job); /** Promotes a job to the front of the list */ void job_promote(job_t *job); /** Return the job with the specified job id. If id is 0 or less, return the last job used. */ job_t *job_get(int job_id); /** Returns the job with the given pid */ job_t *job_get_from_pid(int pid); /* Returns a new profile item if profiling is active. The caller should fill it in. The parser_t will clean it up. */ profile_item_t *create_profile_item(); /** 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 */ void get_backtrace(const wcstring &src, const parse_error_list_t &errors, wcstring *output) const; /** 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, wcstring *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(); /** Output profiling data to the given filename */ void emit_profiling(const char *path) const; /** 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. */ static int is_help(const wchar_t *s, int min_match); /** 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 wcstring */ void stack_trace(size_t block_idx, wcstring &buff) const; int get_block_type(const wchar_t *cmd) const; const wchar_t *get_block_command(int type) const; }; /* Temporary */ bool parser_use_ast(void); #endif