2016-05-03 00:22:44 +00:00
|
|
|
// The fish parser.
|
2005-10-04 15:11:39 +00:00
|
|
|
#ifndef FISH_PARSER_H
|
|
|
|
#define FISH_PARSER_H
|
|
|
|
|
2016-04-21 06:00:54 +00:00
|
|
|
#include <stddef.h>
|
2017-04-30 04:33:50 +00:00
|
|
|
#include <unistd.h>
|
2017-02-11 02:47:02 +00:00
|
|
|
|
|
|
|
#include <csignal>
|
2016-04-21 06:00:54 +00:00
|
|
|
#include <list>
|
2017-02-11 02:47:02 +00:00
|
|
|
#include <memory>
|
|
|
|
#include <type_traits>
|
2016-04-21 06:00:54 +00:00
|
|
|
#include <vector>
|
2005-10-04 15:11:39 +00:00
|
|
|
|
2015-07-25 15:14:25 +00:00
|
|
|
#include "common.h"
|
2006-02-01 15:49:11 +00:00
|
|
|
#include "event.h"
|
2016-02-28 02:25:58 +00:00
|
|
|
#include "expand.h"
|
2016-05-03 00:22:44 +00:00
|
|
|
#include "parse_constants.h"
|
|
|
|
#include "parse_tree.h"
|
|
|
|
#include "proc.h"
|
2015-07-25 15:14:25 +00:00
|
|
|
|
2016-04-21 06:00:54 +00:00
|
|
|
class io_chain_t;
|
2005-10-04 15:11:39 +00:00
|
|
|
|
2016-05-03 00:22:44 +00:00
|
|
|
/// 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_BLOCKED) should be set.
|
|
|
|
///
|
|
|
|
/// Note that EVENT_ANY can be used to specify any event.
|
2012-11-19 00:30:30 +00:00
|
|
|
unsigned int typemask;
|
2012-02-08 05:04:51 +00:00
|
|
|
};
|
|
|
|
|
2012-08-27 05:42:29 +00:00
|
|
|
typedef std::list<event_blockage_t> event_blockage_list_t;
|
2012-02-08 05:04:51 +00:00
|
|
|
|
2016-05-03 00:22:44 +00:00
|
|
|
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 << EVENT_ANY)) return true;
|
|
|
|
if (iter->typemask & (1 << type)) return true;
|
2012-02-08 05:04:51 +00:00
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
2005-12-11 22:21:01 +00:00
|
|
|
|
2016-05-03 00:22:44 +00:00
|
|
|
/// Types of blocks.
|
|
|
|
enum block_type_t {
|
|
|
|
WHILE, /// While loop block
|
|
|
|
FOR, /// For loop block
|
|
|
|
IF, /// If block
|
|
|
|
FUNCTION_CALL, /// Function invocation block
|
|
|
|
FUNCTION_CALL_NO_SHADOW, /// Function invocation block with no variable shadowing
|
|
|
|
SWITCH, /// Switch block
|
|
|
|
SUBST, /// Command substitution scope
|
|
|
|
TOP, /// Outermost block
|
|
|
|
BEGIN, /// Unconditional block
|
|
|
|
SOURCE, /// Block created by the . (source) builtin
|
|
|
|
EVENT, /// Block created on event notifier invocation
|
|
|
|
BREAKPOINT, /// Breakpoint block
|
2014-03-02 21:46:30 +00:00
|
|
|
};
|
2012-08-27 05:42:29 +00:00
|
|
|
|
2016-05-03 00:22:44 +00:00
|
|
|
/// Possible states for a loop.
|
|
|
|
enum loop_status_t {
|
|
|
|
LOOP_NORMAL, /// current loop block executed as normal
|
|
|
|
LOOP_BREAK, /// current loop block should be removed
|
|
|
|
LOOP_CONTINUE, /// current loop block should be skipped
|
2014-10-31 08:15:50 +00:00
|
|
|
};
|
|
|
|
|
2016-05-03 00:22:44 +00:00
|
|
|
/// block_t represents a block of commands.
|
|
|
|
struct block_t {
|
|
|
|
protected:
|
|
|
|
/// Protected constructor. Use one of the subclasses below.
|
2016-02-28 03:38:15 +00:00
|
|
|
explicit block_t(block_type_t t);
|
2012-08-27 06:16:20 +00:00
|
|
|
|
2016-05-03 00:22:44 +00:00
|
|
|
private:
|
|
|
|
/// Type of block.
|
|
|
|
const block_type_t block_type;
|
|
|
|
|
|
|
|
public:
|
|
|
|
/// Whether execution of the commands in this block should be skipped.
|
2018-02-12 04:10:57 +00:00
|
|
|
bool skip{false};
|
2016-05-03 00:22:44 +00:00
|
|
|
/// Status for the current loop block. Can be any of the values from the loop_status enum.
|
2018-02-12 04:10:57 +00:00
|
|
|
enum loop_status_t loop_status { LOOP_NORMAL };
|
2016-05-03 00:22:44 +00:00
|
|
|
/// The job that is currently evaluated in the specified block.
|
2018-02-12 04:10:57 +00:00
|
|
|
shared_ptr<job_t> job{};
|
2016-05-03 00:22:44 +00:00
|
|
|
/// Name of file that created this block. This string is intern'd.
|
2018-02-12 04:10:57 +00:00
|
|
|
const wchar_t *src_filename{nullptr};
|
2016-05-03 00:22:44 +00:00
|
|
|
/// Line number where this block was created.
|
2018-02-12 04:10:57 +00:00
|
|
|
int src_lineno{0};
|
2016-05-03 00:22:44 +00:00
|
|
|
/// Whether we should pop the environment variable stack when we're popped off of the block
|
|
|
|
/// stack.
|
2018-02-12 04:10:57 +00:00
|
|
|
bool wants_pop_env{false};
|
|
|
|
/// List of event blocks.
|
|
|
|
event_blockage_list_t event_blocks{};
|
2012-11-18 10:23:22 +00:00
|
|
|
|
2016-05-03 00:22:44 +00:00
|
|
|
block_type_t type() const { return this->block_type; }
|
|
|
|
|
|
|
|
/// Description of the block, for debugging.
|
|
|
|
wcstring description() const;
|
|
|
|
|
|
|
|
/// Destructor
|
2012-08-27 05:42:29 +00:00
|
|
|
virtual ~block_t();
|
|
|
|
};
|
|
|
|
|
2016-05-03 00:22:44 +00:00
|
|
|
struct if_block_t : public block_t {
|
2012-08-27 06:16:20 +00:00
|
|
|
if_block_t();
|
2012-08-27 05:42:29 +00:00
|
|
|
};
|
2005-09-20 13:26:39 +00:00
|
|
|
|
2016-05-03 00:22:44 +00:00
|
|
|
struct event_block_t : public block_t {
|
2012-12-22 20:40:34 +00:00
|
|
|
event_t const event;
|
2016-02-28 03:38:15 +00:00
|
|
|
explicit event_block_t(const event_t &evt);
|
2012-08-27 05:42:29 +00:00
|
|
|
};
|
|
|
|
|
2016-05-03 00:22:44 +00:00
|
|
|
struct function_block_t : public block_t {
|
2013-01-30 10:22:38 +00:00
|
|
|
const process_t *process;
|
2012-08-27 05:42:29 +00:00
|
|
|
wcstring name;
|
2018-02-19 02:39:03 +00:00
|
|
|
function_block_t(const process_t *p, wcstring n, bool shadows);
|
2012-08-27 05:42:29 +00:00
|
|
|
};
|
|
|
|
|
2016-05-03 00:22:44 +00:00
|
|
|
struct source_block_t : public block_t {
|
|
|
|
const wchar_t *const source_file;
|
2016-02-28 03:38:15 +00:00
|
|
|
explicit source_block_t(const wchar_t *src);
|
2012-08-27 05:42:29 +00:00
|
|
|
};
|
|
|
|
|
2016-05-03 00:22:44 +00:00
|
|
|
struct for_block_t : public block_t {
|
2014-03-02 21:46:30 +00:00
|
|
|
for_block_t();
|
2012-08-27 05:42:29 +00:00
|
|
|
};
|
|
|
|
|
2016-05-03 00:22:44 +00:00
|
|
|
struct while_block_t : public block_t {
|
2012-08-27 06:16:20 +00:00
|
|
|
while_block_t();
|
2012-08-27 05:42:29 +00:00
|
|
|
};
|
|
|
|
|
2016-05-03 00:22:44 +00:00
|
|
|
struct switch_block_t : public block_t {
|
2014-03-02 21:46:30 +00:00
|
|
|
switch_block_t();
|
2012-08-27 06:16:20 +00:00
|
|
|
};
|
|
|
|
|
2016-05-03 00:22:44 +00:00
|
|
|
struct scope_block_t : public block_t {
|
|
|
|
explicit scope_block_t(block_type_t type); // must be BEGIN, TOP or SUBST
|
2012-08-27 06:16:20 +00:00
|
|
|
};
|
|
|
|
|
2016-05-03 00:22:44 +00:00
|
|
|
struct breakpoint_block_t : public block_t {
|
2012-08-27 06:16:20 +00:00
|
|
|
breakpoint_block_t();
|
2012-08-27 05:42:29 +00:00
|
|
|
};
|
2005-09-20 13:26:39 +00:00
|
|
|
|
2016-05-03 00:22:44 +00:00
|
|
|
struct profile_item_t {
|
|
|
|
/// Time spent executing the specified command, including parse time for nested blocks.
|
2012-11-19 00:30:30 +00:00
|
|
|
int exec;
|
2016-05-03 00:22:44 +00:00
|
|
|
/// Time spent parsing the specified command, including execution time for command
|
|
|
|
/// substitutions.
|
2012-11-19 00:30:30 +00:00
|
|
|
int parse;
|
2016-05-03 00:22:44 +00:00
|
|
|
/// The block level of the specified command. nested blocks and command substitutions both
|
|
|
|
/// increase the block level.
|
2012-11-19 00:30:30 +00:00
|
|
|
size_t level;
|
2016-05-03 00:22:44 +00:00
|
|
|
/// If the execution of this command was skipped.
|
2014-02-09 22:04:43 +00:00
|
|
|
bool skipped;
|
2016-05-03 00:22:44 +00:00
|
|
|
/// The command string.
|
2012-11-19 00:30:30 +00:00
|
|
|
wcstring cmd;
|
2012-01-20 19:24:43 +00:00
|
|
|
};
|
2012-01-19 18:28:44 +00:00
|
|
|
|
2013-12-26 21:24:10 +00:00
|
|
|
class parse_execution_context_t;
|
2015-07-25 15:14:25 +00:00
|
|
|
class completion_t;
|
2012-01-23 04:47:13 +00:00
|
|
|
|
2016-05-03 00:22:44 +00:00
|
|
|
class parser_t {
|
2013-12-24 21:17:24 +00:00
|
|
|
friend class parse_execution_context_t;
|
2014-03-31 17:01:39 +00:00
|
|
|
|
2016-05-03 00:22:44 +00:00
|
|
|
private:
|
|
|
|
/// Indication that we should skip all blocks.
|
2018-09-09 08:36:21 +00:00
|
|
|
volatile sig_atomic_t cancellation_requested = false;
|
2018-02-12 06:00:17 +00:00
|
|
|
/// The current execution context.
|
|
|
|
std::unique_ptr<parse_execution_context_t> execution_context;
|
2016-05-03 00:22:44 +00:00
|
|
|
/// List of called functions, used to help prevent infinite recursion.
|
2012-01-30 06:06:58 +00:00
|
|
|
wcstring_list_t forbidden_function;
|
2016-05-03 00:22:44 +00:00
|
|
|
/// The jobs associated with this parser.
|
2012-02-28 02:43:24 +00:00
|
|
|
job_list_t my_job_list;
|
2017-01-21 23:35:35 +00:00
|
|
|
/// The list of blocks
|
|
|
|
std::vector<std::unique_ptr<block_t>> block_stack;
|
2018-02-12 05:42:23 +00:00
|
|
|
/// The 'depth' of the fish call stack.
|
|
|
|
int eval_level = -1;
|
2018-09-10 08:17:57 +00:00
|
|
|
/// Set of variables for the parser.
|
|
|
|
env_stack_t &variables;
|
2016-05-29 05:28:26 +00:00
|
|
|
#if 0
|
|
|
|
// TODO: Lint says this isn't used (which is true). Should this be removed?
|
2016-05-03 00:22:44 +00:00
|
|
|
/// Gets a description of the block stack, for debugging.
|
2014-03-16 23:45:00 +00:00
|
|
|
wcstring block_stack_description() const;
|
2016-05-29 05:28:26 +00:00
|
|
|
#endif
|
2014-03-16 23:45:00 +00:00
|
|
|
|
2017-01-21 22:33:17 +00:00
|
|
|
/// List of profile items
|
|
|
|
/// These are pointers because we return pointers to them to callers,
|
|
|
|
/// who may hold them across blocks (which would cause reallocations internal
|
|
|
|
/// to profile_items)
|
|
|
|
std::vector<std::unique_ptr<profile_item_t>> profile_items;
|
2014-03-16 23:45:00 +00:00
|
|
|
|
2016-05-03 00:22:44 +00:00
|
|
|
// No copying allowed.
|
|
|
|
parser_t(const parser_t &);
|
|
|
|
parser_t &operator=(const parser_t &);
|
2014-01-15 09:40:40 +00:00
|
|
|
|
2016-05-03 00:22:44 +00:00
|
|
|
/// Adds a job to the beginning of the job list.
|
2017-01-26 22:47:32 +00:00
|
|
|
void job_add(shared_ptr<job_t> job);
|
Big fat refactoring of how redirections work. In fish 1.x and 2.0.0, the redirections for a process were flattened into a big list associated with the job, so there was no way to tell which redirections applied to each process. Each process therefore got all the redirections associated with the job. See https://github.com/fish-shell/fish-shell/issues/877 for how this could manifest.
With this change, jobs only track their block-level redirections. Process level redirections are correctly associated with the process, and at exec time we stitch them together (block, pipe, and process redirects).
This fixes the weird issues where redirects bleed across pipelines (like #877), and also allows us to play with the order in which redirections are applied, since the final list is constructed right before it's needed. This lets us put pipes after block level redirections but before process level redirections, so that a 2>&1-type redirection gets picked up after the pipe, i.e. it should fix https://github.com/fish-shell/fish-shell/issues/110
This is a significant change. The tests all pass. Cross your fingers.
2013-08-19 23:16:41 +00:00
|
|
|
|
2016-05-03 00:22:44 +00:00
|
|
|
/// 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.
|
2017-06-24 06:19:09 +00:00
|
|
|
const wchar_t *is_function(size_t idx = 0) const;
|
2012-11-18 10:23:22 +00:00
|
|
|
|
2016-05-03 00:22:44 +00:00
|
|
|
/// Helper for stack_trace().
|
2015-09-21 18:24:49 +00:00
|
|
|
void stack_trace_internal(size_t block_idx, wcstring *out) const;
|
|
|
|
|
2017-01-21 23:35:35 +00:00
|
|
|
/// Helper for push_block()
|
|
|
|
void push_block_int(block_t *b);
|
|
|
|
|
2018-09-09 08:36:21 +00:00
|
|
|
/// Create a parser.
|
|
|
|
parser_t();
|
|
|
|
|
|
|
|
/// The main parser.
|
|
|
|
static parser_t principal;
|
|
|
|
|
2016-05-03 00:22:44 +00:00
|
|
|
public:
|
|
|
|
/// Get the "principal" parser, whatever that is.
|
2012-01-23 05:40:08 +00:00
|
|
|
static parser_t &principal_parser();
|
2012-11-18 10:23:22 +00:00
|
|
|
|
2016-05-03 00:22:44 +00:00
|
|
|
/// Indicates that execution of all blocks in the principal parser should stop. This is called
|
|
|
|
/// from signal handlers!
|
2012-06-04 21:20:01 +00:00
|
|
|
static void skip_all_blocks();
|
2012-11-18 10:23:22 +00:00
|
|
|
|
2016-05-03 00:22:44 +00:00
|
|
|
/// Global event blocks.
|
2012-08-27 05:42:29 +00:00
|
|
|
event_blockage_list_t global_event_blocks;
|
2012-11-18 10:23:22 +00:00
|
|
|
|
2016-05-03 00:22:44 +00:00
|
|
|
/// 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
|
|
|
|
///
|
2018-02-12 07:13:06 +00:00
|
|
|
/// \return 0 on success, 1 on a parse error.
|
2017-12-22 22:40:15 +00:00
|
|
|
int eval(wcstring cmd, const io_chain_t &io, enum block_type_t block_type);
|
2012-01-16 19:09:19 +00:00
|
|
|
|
2017-12-22 22:40:15 +00:00
|
|
|
/// Evaluate the parsed source ps.
|
2018-02-12 07:13:06 +00:00
|
|
|
void eval(parsed_source_ref_t ps, const io_chain_t &io, enum block_type_t block_type);
|
2005-09-20 13:26:39 +00:00
|
|
|
|
2018-02-12 03:34:12 +00:00
|
|
|
/// Evaluates a node.
|
2018-02-11 03:16:35 +00:00
|
|
|
/// The node type must be grammar::statement or grammar::job_list.
|
|
|
|
template <typename T>
|
2018-02-12 03:34:12 +00:00
|
|
|
int eval_node(parsed_source_ref_t ps, tnode_t<T> node, const io_chain_t &io,
|
2018-11-04 07:58:44 +00:00
|
|
|
block_type_t block_type, std::shared_ptr<job_t> parent_job);
|
2005-09-20 13:26:39 +00:00
|
|
|
|
2016-05-03 00:22:44 +00:00
|
|
|
/// 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. Errors are ignored.
|
|
|
|
///
|
|
|
|
/// \param arg_src String to evaluate as an argument list
|
|
|
|
/// \param flags Some expand flags to use
|
|
|
|
/// \param output List to insert output into
|
|
|
|
static void expand_argument_list(const wcstring &arg_src, expand_flags_t flags,
|
|
|
|
std::vector<completion_t> *output);
|
|
|
|
|
2017-06-24 05:14:21 +00:00
|
|
|
/// Returns a string describing the current parser position in the format 'FILENAME (line
|
2016-05-03 00:22:44 +00:00
|
|
|
/// LINE_NUMBER): LINE'. Example:
|
|
|
|
///
|
|
|
|
/// init.fish (line 127): ls|grep pancake
|
2014-03-16 21:49:51 +00:00
|
|
|
wcstring current_line();
|
2005-09-20 13:26:39 +00:00
|
|
|
|
2016-05-03 00:22:44 +00:00
|
|
|
/// Returns the current line number.
|
2012-01-16 19:16:12 +00:00
|
|
|
int get_lineno() const;
|
2012-11-18 10:23:22 +00:00
|
|
|
|
2016-05-03 00:22:44 +00:00
|
|
|
/// 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.
|
2013-12-21 01:41:21 +00:00
|
|
|
const block_t *block_at_index(size_t idx) const;
|
|
|
|
block_t *block_at_index(size_t idx);
|
2014-01-15 09:40:40 +00:00
|
|
|
|
2016-05-03 00:22:44 +00:00
|
|
|
/// Returns the current (innermost) block.
|
2016-07-30 18:48:39 +00:00
|
|
|
block_t *current_block();
|
2014-01-15 09:40:40 +00:00
|
|
|
|
2016-05-03 00:22:44 +00:00
|
|
|
/// Count of blocks.
|
|
|
|
size_t block_count() const { return block_stack.size(); }
|
2005-09-20 13:26:39 +00:00
|
|
|
|
2016-05-03 00:22:44 +00:00
|
|
|
/// Get the list of jobs.
|
|
|
|
job_list_t &job_list() { return my_job_list; }
|
2014-03-31 17:01:39 +00:00
|
|
|
|
2018-09-10 08:17:57 +00:00
|
|
|
/// Get the variables.
|
|
|
|
env_stack_t &vars() { return variables; }
|
|
|
|
const env_stack_t &vars() const { return variables; }
|
|
|
|
|
2017-01-21 23:35:35 +00:00
|
|
|
/// Pushes a new block created with the given arguments
|
|
|
|
/// Returns a pointer to the block. The pointer is valid
|
|
|
|
/// until the call to pop_block()
|
2017-01-27 04:00:43 +00:00
|
|
|
template <typename BLOCKTYPE, typename... Args>
|
|
|
|
BLOCKTYPE *push_block(Args &&... args) {
|
2017-01-21 23:35:35 +00:00
|
|
|
BLOCKTYPE *ret = new BLOCKTYPE(std::forward<Args>(args)...);
|
|
|
|
this->push_block_int(ret);
|
|
|
|
return ret;
|
|
|
|
}
|
2014-01-15 09:40:40 +00:00
|
|
|
|
2016-05-03 00:22:44 +00:00
|
|
|
/// Remove the outermost block, asserting it's the given one.
|
2013-12-26 20:24:00 +00:00
|
|
|
void pop_block(const block_t *b);
|
2005-09-20 13:26:39 +00:00
|
|
|
|
2016-05-03 00:22:44 +00:00
|
|
|
/// Return a description of the given blocktype.
|
2012-11-19 00:30:30 +00:00
|
|
|
const wchar_t *get_block_desc(int block) const;
|
2005-09-20 13:26:39 +00:00
|
|
|
|
2017-06-24 05:14:21 +00:00
|
|
|
/// Return the function name for the specified stack frame. Default is one (current frame).
|
2017-06-24 06:19:09 +00:00
|
|
|
const wchar_t *get_function_name(int level = 1);
|
2017-04-12 22:34:25 +00:00
|
|
|
|
2016-05-03 00:22:44 +00:00
|
|
|
/// Removes a job.
|
2012-02-28 02:43:24 +00:00
|
|
|
bool job_remove(job_t *job);
|
2012-11-18 10:23:22 +00:00
|
|
|
|
2016-05-03 00:22:44 +00:00
|
|
|
/// Promotes a job to the front of the list.
|
2012-02-28 02:43:24 +00:00
|
|
|
void job_promote(job_t *job);
|
2012-11-18 10:23:22 +00:00
|
|
|
|
2016-05-03 00:22:44 +00:00
|
|
|
/// Return the job with the specified job id. If id is 0 or less, return the last job used.
|
2017-01-26 22:47:32 +00:00
|
|
|
job_t *job_get(job_id_t job_id);
|
2012-11-18 10:23:22 +00:00
|
|
|
|
2016-05-03 00:22:44 +00:00
|
|
|
/// Returns the job with the given pid.
|
2018-08-09 23:46:11 +00:00
|
|
|
job_t *job_get_from_pid(pid_t pid) const;
|
2014-03-31 17:01:39 +00:00
|
|
|
|
2016-05-03 00:22:44 +00:00
|
|
|
/// Returns a new profile item if profiling is active. The caller should fill it in. The
|
|
|
|
/// parser_t will clean it up.
|
2014-02-09 22:04:43 +00:00
|
|
|
profile_item_t *create_profile_item();
|
2005-09-20 13:26:39 +00:00
|
|
|
|
2016-05-03 00:22:44 +00:00
|
|
|
void get_backtrace(const wcstring &src, const parse_error_list_t &errors,
|
2017-06-18 05:36:56 +00:00
|
|
|
wcstring &output) const;
|
2016-05-03 00:22:44 +00:00
|
|
|
|
|
|
|
/// Detect errors in the specified string when parsed as an argument list. Returns true if an
|
|
|
|
/// error occurred.
|
|
|
|
bool detect_errors_in_argument_list(const wcstring &arg_list_src, wcstring *out_err,
|
2018-08-09 23:46:11 +00:00
|
|
|
const wchar_t *prefix) const;
|
2016-05-03 00:22:44 +00:00
|
|
|
|
|
|
|
/// 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.
|
2012-11-19 00:30:30 +00:00
|
|
|
void forbid_function(const wcstring &function);
|
2014-03-18 21:42:38 +00:00
|
|
|
|
2016-05-03 00:22:44 +00:00
|
|
|
/// Undo last call to parser_forbid_function().
|
2012-01-16 19:09:19 +00:00
|
|
|
void allow_function();
|
2005-09-20 13:26:39 +00:00
|
|
|
|
2016-05-03 00:22:44 +00:00
|
|
|
/// Output profiling data to the given filename.
|
2014-02-09 22:04:43 +00:00
|
|
|
void emit_profiling(const char *path) const;
|
2005-09-20 13:26:39 +00:00
|
|
|
|
2016-05-03 00:22:44 +00:00
|
|
|
/// 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.
|
2012-01-16 19:16:12 +00:00
|
|
|
const wchar_t *current_filename() const;
|
2006-01-26 15:47:22 +00:00
|
|
|
|
2016-05-03 00:22:44 +00:00
|
|
|
/// Return a string representing the current stack trace.
|
2015-09-21 18:24:49 +00:00
|
|
|
wcstring stack_trace() const;
|
2017-01-21 22:53:10 +00:00
|
|
|
|
|
|
|
~parser_t();
|
2012-01-16 19:16:12 +00:00
|
|
|
};
|
2006-09-05 20:43:47 +00:00
|
|
|
|
2005-10-04 15:11:39 +00:00
|
|
|
#endif
|