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
|
|
|
|
2019-02-21 05:41:35 +00:00
|
|
|
/// event_blockage_t represents a block on events.
|
2019-05-05 10:09:25 +00:00
|
|
|
struct event_blockage_t {};
|
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
|
|
|
|
2019-02-21 06:42:58 +00:00
|
|
|
inline bool event_block_list_blocks_type(const event_blockage_list_t &ebls) {
|
2019-02-21 05:41:35 +00:00
|
|
|
return !ebls.empty();
|
2012-02-08 05:04:51 +00:00
|
|
|
}
|
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
|
Support FOO=bar syntax for passing variables to individual commands
This adds initial support for statements with prefixed variable assignments.
Statments like this are supported:
a=1 b=$a echo $b # outputs 1
Just like in other shells, the left-hand side of each assignment must
be a valid variable identifier (no quoting/escaping). Array indexing
(PATH[1]=/bin ls $PATH) is *not* yet supported, but can be added fairly
easily.
The right hand side may be any valid string token, like a command
substitution, or a brace expansion.
Since `a=* foo` is equivalent to `begin set -lx a *; foo; end`,
the assignment, like `set`, uses nullglob behavior, e.g. below command
can safely be used to check if a directory is empty.
x=/nothing/{,.}* test (count $x) -eq 0
Generic file completion is done after the equal sign, so for example
pressing tab after something like `HOME=/` completes files in the
root directory
Subcommand completion works, so something like
`GIT_DIR=repo.git and command git ` correctly calls git completions
(but the git completion does not use the variable as of now).
The variable assignment is highlighted like an argument.
Closes #6048
2019-10-23 01:13:29 +00:00
|
|
|
VARIABLE_ASSIGNMENT, /// Variable assignment before a command
|
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.
|
2019-05-19 06:12:34 +00:00
|
|
|
enum class loop_status_t {
|
|
|
|
normals, /// current loop block executed as normal
|
|
|
|
breaks, /// current loop block should be removed
|
|
|
|
continues, /// 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.
|
2019-05-19 21:44:17 +00:00
|
|
|
class block_t {
|
|
|
|
/// Construct from a block type.
|
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
|
|
|
/// Type of block.
|
|
|
|
const block_type_t block_type;
|
|
|
|
|
|
|
|
public:
|
|
|
|
/// 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
|
|
|
|
2019-05-19 19:55:16 +00:00
|
|
|
// If this is a function block, the function name and arguments.
|
|
|
|
// Otherwise empty.
|
|
|
|
wcstring function_name{};
|
|
|
|
wcstring_list_t function_args{};
|
|
|
|
|
2019-05-19 20:01:59 +00:00
|
|
|
// If this is a source block, the source'd file, interned.
|
|
|
|
// Otherwise nothing.
|
|
|
|
const wchar_t *sourced_file{};
|
|
|
|
|
2019-05-19 20:07:06 +00:00
|
|
|
// If this is an event block, the event. Otherwise ignored.
|
|
|
|
maybe_t<event_t> event;
|
|
|
|
|
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;
|
|
|
|
|
2019-05-19 21:40:06 +00:00
|
|
|
/// Entry points for creating blocks.
|
|
|
|
static block_t if_block();
|
|
|
|
static block_t event_block(event_t evt);
|
|
|
|
static block_t function_block(wcstring name, wcstring_list_t args, bool shadows);
|
|
|
|
static block_t source_block(const wchar_t *src);
|
|
|
|
static block_t for_block();
|
|
|
|
static block_t while_block();
|
|
|
|
static block_t switch_block();
|
|
|
|
static block_t scope_block(block_type_t type);
|
|
|
|
static block_t breakpoint_block();
|
Support FOO=bar syntax for passing variables to individual commands
This adds initial support for statements with prefixed variable assignments.
Statments like this are supported:
a=1 b=$a echo $b # outputs 1
Just like in other shells, the left-hand side of each assignment must
be a valid variable identifier (no quoting/escaping). Array indexing
(PATH[1]=/bin ls $PATH) is *not* yet supported, but can be added fairly
easily.
The right hand side may be any valid string token, like a command
substitution, or a brace expansion.
Since `a=* foo` is equivalent to `begin set -lx a *; foo; end`,
the assignment, like `set`, uses nullglob behavior, e.g. below command
can safely be used to check if a directory is empty.
x=/nothing/{,.}* test (count $x) -eq 0
Generic file completion is done after the equal sign, so for example
pressing tab after something like `HOME=/` completes files in the
root directory
Subcommand completion works, so something like
`GIT_DIR=repo.git and command git ` correctly calls git completions
(but the git completion does not use the variable as of now).
The variable assignment is highlighted like an argument.
Closes #6048
2019-10-23 01:13:29 +00:00
|
|
|
static block_t variable_assignment_block();
|
2019-05-19 21:40:06 +00:00
|
|
|
|
2019-05-19 21:44:17 +00:00
|
|
|
~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;
|
2019-06-03 09:31:13 +00:00
|
|
|
struct event_t;
|
2012-01-23 04:47:13 +00:00
|
|
|
|
2019-10-22 00:21:40 +00:00
|
|
|
/// Miscellaneous data used to avoid recursion and others.
|
2019-04-28 23:11:49 +00:00
|
|
|
struct library_data_t {
|
|
|
|
/// A counter incremented every time a command executes.
|
|
|
|
uint64_t exec_count{0};
|
2019-04-28 23:41:58 +00:00
|
|
|
|
2019-04-29 01:13:55 +00:00
|
|
|
/// Last reader run count.
|
|
|
|
uint64_t last_exec_run_counter{UINT64_MAX};
|
|
|
|
|
2019-04-28 23:41:58 +00:00
|
|
|
/// Number of recursive calls to builtin_complete().
|
|
|
|
uint32_t builtin_complete_recursion_level{0};
|
2019-04-30 03:58:58 +00:00
|
|
|
|
2019-10-06 13:34:43 +00:00
|
|
|
/// Whether we called builtin_complete -C without parameter.
|
|
|
|
bool builtin_complete_current_commandline{false};
|
|
|
|
|
2019-04-30 03:58:58 +00:00
|
|
|
/// Whether we are currently cleaning processes.
|
|
|
|
bool is_cleaning_procs{false};
|
2019-05-12 21:42:18 +00:00
|
|
|
|
|
|
|
/// The job id of the job being populated.
|
|
|
|
/// This supports the '--on-job-exit caller' feature.
|
|
|
|
job_id_t caller_job_id{-1};
|
2019-05-13 01:02:57 +00:00
|
|
|
|
|
|
|
/// Whether we are running a subshell command.
|
|
|
|
bool is_subshell{false};
|
|
|
|
|
|
|
|
/// Whether we are running a block of commands.
|
|
|
|
bool is_block{false};
|
|
|
|
|
|
|
|
/// Whether we are running due to a `breakpoint` command.
|
|
|
|
bool is_breakpoint{false};
|
|
|
|
|
|
|
|
/// Whether we are running an event handler. This is not a bool because we keep count of the
|
|
|
|
/// event nesting level.
|
|
|
|
int is_event{0};
|
2019-05-19 06:12:34 +00:00
|
|
|
|
2019-05-27 21:52:48 +00:00
|
|
|
/// Whether we are currently interactive.
|
|
|
|
bool is_interactive{false};
|
|
|
|
|
2019-10-19 01:08:22 +00:00
|
|
|
/// Whether to suppress fish_trace output. This occurs in the prompt, event handlers, and key
|
|
|
|
/// bindings.
|
|
|
|
bool suppress_fish_trace{false};
|
|
|
|
|
2019-05-19 06:12:34 +00:00
|
|
|
/// Whether we should break or continue the current loop.
|
|
|
|
enum loop_status_t loop_status { loop_status_t::normals };
|
2019-05-20 02:29:25 +00:00
|
|
|
|
|
|
|
/// Whether we should return from the current function.
|
|
|
|
bool returning{false};
|
2019-05-22 20:34:03 +00:00
|
|
|
|
|
|
|
/// The current filename we are evaluating, either from builtin source or on the command line.
|
|
|
|
/// This is an intern'd string.
|
|
|
|
const wchar_t *current_filename{};
|
2019-06-03 09:31:13 +00:00
|
|
|
|
|
|
|
/// List of events that have been sent but have not yet been delivered because they are blocked.
|
2019-06-09 21:11:25 +00:00
|
|
|
std::vector<shared_ptr<event_t>> blocked_events{};
|
|
|
|
|
|
|
|
/// A stack of fake values to be returned by builtin_commandline. This is used by the completion
|
|
|
|
/// machinery when wrapping: e.g. if `tig` wraps `git` then git completions need to see git on
|
|
|
|
/// the command line.
|
|
|
|
wcstring_list_t transient_commandlines{};
|
2019-06-10 16:27:51 +00:00
|
|
|
|
|
|
|
/// A file descriptor holding the current working directory, for use in openat().
|
|
|
|
/// This is never null and never invalid.
|
|
|
|
std::shared_ptr<const autoclose_fd_t> cwd_fd{};
|
2019-04-28 23:11:49 +00:00
|
|
|
};
|
|
|
|
|
2019-02-24 20:12:24 +00:00
|
|
|
class parser_t : public std::enable_shared_from_this<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
|
|
|
/// The jobs associated with this parser.
|
2019-05-05 05:12:31 +00:00
|
|
|
job_list_t job_list;
|
2019-05-20 16:42:18 +00:00
|
|
|
/// The list of blocks. This is a deque because we give out raw pointers to callers, who hold
|
|
|
|
/// them across manipulating this stack.
|
|
|
|
std::deque<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.
|
2019-05-20 16:27:46 +00:00
|
|
|
const std::shared_ptr<env_stack_t> variables;
|
2019-04-28 23:11:49 +00:00
|
|
|
/// Miscellaneous library data.
|
|
|
|
library_data_t library_data{};
|
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
|
|
|
|
2018-09-22 04:52:47 +00:00
|
|
|
// Given a file path, return something nicer. Currently we just "unexpand" tildes.
|
|
|
|
wcstring user_presentable_path(const wcstring &path) const;
|
|
|
|
|
2016-05-03 00:22:44 +00:00
|
|
|
/// Helper for stack_trace().
|
2019-11-19 00:54:36 +00:00
|
|
|
void stack_trace_internal(size_t block_idx, wcstring *buff) const;
|
2015-09-21 18:24:49 +00:00
|
|
|
|
2018-09-09 08:36:21 +00:00
|
|
|
/// Create a parser.
|
|
|
|
parser_t();
|
2019-05-20 16:27:46 +00:00
|
|
|
parser_t(std::shared_ptr<env_stack_t> vars);
|
2018-09-09 08:36:21 +00:00
|
|
|
|
|
|
|
/// The main parser.
|
2019-02-24 20:12:24 +00:00
|
|
|
static std::shared_ptr<parser_t> principal;
|
2018-09-09 08:36:21 +00:00
|
|
|
|
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.
|
2019-04-02 03:19:28 +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.
|
2019-04-13 21:57:00 +00:00
|
|
|
/// \return 0 on success, 1 on a parse error.
|
|
|
|
int 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>
|
2019-12-08 21:03:42 +00:00
|
|
|
int eval_node(parsed_source_ref_t ps, tnode_t<T> node, block_type_t block_type,
|
|
|
|
job_lineage_t lineage);
|
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
|
2019-05-05 02:16:26 +00:00
|
|
|
/// cmdsubst execution on the tokens. Errors are ignored. If a parser is provided, it is used
|
|
|
|
/// for command substitution expansion.
|
|
|
|
static std::vector<completion_t> expand_argument_list(const wcstring &arg_list_src,
|
|
|
|
expand_flags_t flags,
|
|
|
|
const environment_t &vars,
|
|
|
|
const std::shared_ptr<parser_t> &parser);
|
2016-05-03 00:22:44 +00:00
|
|
|
|
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.
|
2019-05-05 05:12:31 +00:00
|
|
|
job_list_t &jobs() { return job_list; }
|
|
|
|
const job_list_t &jobs() const { return job_list; }
|
2014-03-31 17:01:39 +00:00
|
|
|
|
2018-09-10 08:17:57 +00:00
|
|
|
/// Get the variables.
|
2019-05-20 16:27:46 +00:00
|
|
|
env_stack_t &vars() { return *variables; }
|
|
|
|
const env_stack_t &vars() const { return *variables; }
|
2018-09-10 08:17:57 +00:00
|
|
|
|
2019-04-28 23:11:49 +00:00
|
|
|
/// Get the library data.
|
|
|
|
library_data_t &libdata() { return library_data; }
|
|
|
|
const library_data_t &libdata() const { return library_data; }
|
|
|
|
|
2019-05-12 21:00:44 +00:00
|
|
|
/// Get and set the last proc statuses.
|
|
|
|
int get_last_status() const { return vars().get_last_status(); }
|
|
|
|
statuses_t get_last_statuses() const { return vars().get_last_statuses(); }
|
|
|
|
void set_last_statuses(statuses_t s) { vars().set_last_statuses(std::move(s)); }
|
|
|
|
|
2019-05-19 21:40:06 +00:00
|
|
|
/// Pushes a new block. Returns a pointer to the block, stored in the parser. The pointer is
|
|
|
|
/// valid until the call to pop_block()
|
|
|
|
block_t *push_block(block_t &&b);
|
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.
|
2019-11-19 00:54:36 +00:00
|
|
|
void pop_block(const block_t *expected);
|
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
|
|
|
/// 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
|
|
|
|
|
|
|
/// 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
|
2019-11-25 11:03:25 +00:00
|
|
|
/// reader_current_filename, e.g. if we are evaluating a function defined in a different file
|
2016-05-03 00:22:44 +00:00
|
|
|
/// 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
|
|
|
|
2019-05-27 21:52:48 +00:00
|
|
|
/// Return if we are interactive, which means we are executing a command that the user typed in
|
|
|
|
/// (and not, say, a prompt).
|
|
|
|
bool is_interactive() const { return libdata().is_interactive; }
|
|
|
|
|
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
|
|
|
|
2019-11-10 20:36:46 +00:00
|
|
|
/// \return whether the number of functions in the stack exceeds our stack depth limit.
|
|
|
|
bool function_stack_is_overflowing() const;
|
|
|
|
|
2019-05-05 01:17:18 +00:00
|
|
|
/// \return a shared pointer reference to this parser.
|
|
|
|
std::shared_ptr<parser_t> shared();
|
|
|
|
|
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
|