mirror of
https://github.com/fish-shell/fish-shell
synced 2025-01-16 23:14:04 +00:00
5708 lines
183 KiB
Rust
5708 lines
183 KiB
Rust
/*!
|
||
* This defines the fish abstract syntax tree.
|
||
* The fish ast is a tree data structure. The nodes of the tree
|
||
* are divided into three categories:
|
||
*
|
||
* - leaf nodes refer to a range of source, and have no child nodes.
|
||
* - branch nodes have ONLY child nodes, and no other fields.
|
||
* - list nodes contain a list of some other node type (branch or leaf).
|
||
*
|
||
* Most clients will be interested in visiting the nodes of an ast.
|
||
*/
|
||
use crate::common::{unescape_string, UnescapeStringStyle};
|
||
use crate::flog::FLOG;
|
||
use crate::parse_constants::{
|
||
token_type_user_presentable_description, ParseError, ParseErrorCode, ParseErrorList,
|
||
ParseKeyword, ParseTokenType, ParseTreeFlags, SourceRange, StatementDecoration,
|
||
INVALID_PIPELINE_CMD_ERR_MSG, PARSE_FLAG_ACCEPT_INCOMPLETE_TOKENS,
|
||
PARSE_FLAG_CONTINUE_AFTER_ERROR, PARSE_FLAG_INCLUDE_COMMENTS, PARSE_FLAG_LEAVE_UNTERMINATED,
|
||
PARSE_FLAG_SHOW_EXTRA_SEMIS, SOURCE_OFFSET_INVALID,
|
||
};
|
||
use crate::parse_tree::ParseToken;
|
||
use crate::tokenizer::{
|
||
variable_assignment_equals_pos, TokFlags, TokenType, Tokenizer, TokenizerError,
|
||
TOK_ACCEPT_UNFINISHED, TOK_CONTINUE_AFTER_ERROR, TOK_SHOW_COMMENTS,
|
||
};
|
||
use crate::wchar::{wstr, WString, L};
|
||
use crate::wchar_ext::WExt;
|
||
use crate::wchar_ffi::{wcharz, wcharz_t, AsWstr, WCharToFFI};
|
||
use crate::wutil::printf::sprintf;
|
||
use crate::wutil::wgettext_fmt;
|
||
use cxx::{type_id, ExternType};
|
||
use cxx::{CxxWString, UniquePtr};
|
||
use std::ops::{ControlFlow, Index, IndexMut};
|
||
use widestring_suffix::widestrs;
|
||
|
||
/**
|
||
* A NodeVisitor is something which can visit an AST node.
|
||
*
|
||
* To visit a node's fields, use the node's accept() function:
|
||
* let mut v = MyNodeVisitor{};
|
||
* node.accept(&mut v);
|
||
*/
|
||
pub trait NodeVisitor<'a> {
|
||
fn visit(&mut self, node: &'a dyn Node);
|
||
}
|
||
|
||
pub trait Acceptor {
|
||
fn accept<'a>(&'a self, visitor: &mut dyn NodeVisitor<'a>, reversed: bool);
|
||
}
|
||
|
||
impl<T: Acceptor> Acceptor for Option<T> {
|
||
fn accept<'a>(&'a self, visitor: &mut dyn NodeVisitor<'a>, reversed: bool) {
|
||
match self {
|
||
Some(node) => node.accept(visitor, reversed),
|
||
None => (),
|
||
}
|
||
}
|
||
}
|
||
|
||
pub struct MissingEndError {
|
||
allowed_keywords: &'static [ParseKeyword],
|
||
token: ParseToken,
|
||
}
|
||
|
||
pub type VisitResult = ControlFlow<MissingEndError>;
|
||
|
||
trait NodeVisitorMut {
|
||
/// will_visit (did_visit) is called before (after) a node's fields are visited.
|
||
fn will_visit_fields_of(&mut self, node: &mut dyn NodeMut);
|
||
fn visit_mut(&mut self, node: &mut dyn NodeMut) -> VisitResult;
|
||
fn did_visit_fields_of<'a>(&'a mut self, node: &'a dyn NodeMut, flow: VisitResult);
|
||
|
||
fn visit_argument_or_redirection(
|
||
&mut self,
|
||
_node: &mut Box<ArgumentOrRedirectionVariant>,
|
||
) -> VisitResult;
|
||
fn visit_block_statement_header(
|
||
&mut self,
|
||
_node: &mut Box<BlockStatementHeaderVariant>,
|
||
) -> VisitResult;
|
||
fn visit_statement(&mut self, _node: &mut Box<StatementVariant>) -> VisitResult;
|
||
|
||
fn visit_decorated_statement_decorator(
|
||
&mut self,
|
||
_node: &mut Option<DecoratedStatementDecorator>,
|
||
);
|
||
fn visit_job_conjunction_decorator(&mut self, _node: &mut Option<JobConjunctionDecorator>);
|
||
fn visit_else_clause(&mut self, _node: &mut Option<ElseClause>);
|
||
fn visit_semi_nl(&mut self, _node: &mut Option<SemiNl>);
|
||
fn visit_time(&mut self, _node: &mut Option<KeywordTime>);
|
||
fn visit_token_background(&mut self, _node: &mut Option<TokenBackground>);
|
||
}
|
||
|
||
trait AcceptorMut {
|
||
fn accept_mut(&mut self, visitor: &mut dyn NodeVisitorMut, reversed: bool);
|
||
}
|
||
|
||
impl<T: AcceptorMut> AcceptorMut for Option<T> {
|
||
fn accept_mut(&mut self, visitor: &mut dyn NodeVisitorMut, reversed: bool) {
|
||
match self {
|
||
Some(node) => node.accept_mut(visitor, reversed),
|
||
None => (),
|
||
}
|
||
}
|
||
}
|
||
|
||
/// Node is the base trait of all AST nodes.
|
||
pub trait Node: Acceptor + ConcreteNode + std::fmt::Debug {
|
||
/// The parent node, or null if this is root.
|
||
fn parent(&self) -> Option<&dyn Node>;
|
||
fn parent_ffi(&self) -> &Option<*const dyn Node>;
|
||
|
||
/// The type of this node.
|
||
fn typ(&self) -> Type;
|
||
|
||
/// The category of this node.
|
||
fn category(&self) -> Category;
|
||
|
||
/// \return a helpful string description of this node.
|
||
#[widestrs]
|
||
fn describe(&self) -> WString {
|
||
let mut res = ast_type_to_string(self.typ()).to_owned();
|
||
if let Some(n) = self.as_token() {
|
||
let token_type: &'static wstr = n.token_type().into();
|
||
res += &sprintf!(" '%ls'"L, token_type)[..];
|
||
} else if let Some(n) = self.as_keyword() {
|
||
let keyword: &'static wstr = n.keyword().into();
|
||
res += &sprintf!(" '%ls'"L, keyword)[..];
|
||
}
|
||
res
|
||
}
|
||
|
||
/// \return the source range for this node, or none if unsourced.
|
||
/// This may return none if the parse was incomplete or had an error.
|
||
fn try_source_range(&self) -> Option<SourceRange>;
|
||
|
||
/// \return the source range for this node, or an empty range {0, 0} if unsourced.
|
||
fn source_range(&self) -> SourceRange {
|
||
self.try_source_range().unwrap_or(SourceRange::new(0, 0))
|
||
}
|
||
|
||
/// \return the source code for this node, or none if unsourced.
|
||
fn try_source<'s>(&self, orig: &'s wstr) -> Option<&'s wstr> {
|
||
self.try_source_range()
|
||
.map(|r| &orig[r.start as usize..r.end() as usize])
|
||
}
|
||
|
||
/// \return the source code for this node, or an empty string if unsourced.
|
||
fn source<'s>(&self, orig: &'s wstr) -> &'s wstr {
|
||
self.try_source(orig).unwrap_or_default()
|
||
}
|
||
|
||
// The address of the object, for comparison.
|
||
fn as_ptr(&self) -> *const ();
|
||
}
|
||
|
||
/// NodeMut is a mutable node.
|
||
trait NodeMut: Node + AcceptorMut + ConcreteNodeMut {
|
||
fn as_node(&self) -> &dyn Node;
|
||
}
|
||
|
||
pub trait ConcreteNode {
|
||
// Cast to any sub-trait.
|
||
fn as_leaf(&self) -> Option<&dyn Leaf> {
|
||
None
|
||
}
|
||
fn as_keyword(&self) -> Option<&dyn Keyword> {
|
||
None
|
||
}
|
||
fn as_token(&self) -> Option<&dyn Token> {
|
||
None
|
||
}
|
||
|
||
// Cast to any node type.
|
||
fn as_redirection(&self) -> Option<&Redirection> {
|
||
None
|
||
}
|
||
fn as_variable_assignment(&self) -> Option<&VariableAssignment> {
|
||
None
|
||
}
|
||
fn as_variable_assignment_list(&self) -> Option<&VariableAssignmentList> {
|
||
None
|
||
}
|
||
fn as_argument_or_redirection(&self) -> Option<&ArgumentOrRedirection> {
|
||
None
|
||
}
|
||
fn as_argument_or_redirection_list(&self) -> Option<&ArgumentOrRedirectionList> {
|
||
None
|
||
}
|
||
fn as_statement(&self) -> Option<&Statement> {
|
||
None
|
||
}
|
||
fn as_job_pipeline(&self) -> Option<&JobPipeline> {
|
||
None
|
||
}
|
||
fn as_job_conjunction(&self) -> Option<&JobConjunction> {
|
||
None
|
||
}
|
||
fn as_for_header(&self) -> Option<&ForHeader> {
|
||
None
|
||
}
|
||
fn as_while_header(&self) -> Option<&WhileHeader> {
|
||
None
|
||
}
|
||
fn as_function_header(&self) -> Option<&FunctionHeader> {
|
||
None
|
||
}
|
||
fn as_begin_header(&self) -> Option<&BeginHeader> {
|
||
None
|
||
}
|
||
fn as_block_statement(&self) -> Option<&BlockStatement> {
|
||
None
|
||
}
|
||
fn as_if_clause(&self) -> Option<&IfClause> {
|
||
None
|
||
}
|
||
fn as_elseif_clause(&self) -> Option<&ElseifClause> {
|
||
None
|
||
}
|
||
fn as_elseif_clause_list(&self) -> Option<&ElseifClauseList> {
|
||
None
|
||
}
|
||
fn as_else_clause(&self) -> Option<&ElseClause> {
|
||
None
|
||
}
|
||
fn as_if_statement(&self) -> Option<&IfStatement> {
|
||
None
|
||
}
|
||
fn as_case_item(&self) -> Option<&CaseItem> {
|
||
None
|
||
}
|
||
fn as_switch_statement(&self) -> Option<&SwitchStatement> {
|
||
None
|
||
}
|
||
fn as_decorated_statement(&self) -> Option<&DecoratedStatement> {
|
||
None
|
||
}
|
||
fn as_not_statement(&self) -> Option<&NotStatement> {
|
||
None
|
||
}
|
||
fn as_job_continuation(&self) -> Option<&JobContinuation> {
|
||
None
|
||
}
|
||
fn as_job_continuation_list(&self) -> Option<&JobContinuationList> {
|
||
None
|
||
}
|
||
fn as_job_conjunction_continuation(&self) -> Option<&JobConjunctionContinuation> {
|
||
None
|
||
}
|
||
fn as_andor_job(&self) -> Option<&AndorJob> {
|
||
None
|
||
}
|
||
fn as_andor_job_list(&self) -> Option<&AndorJobList> {
|
||
None
|
||
}
|
||
fn as_freestanding_argument_list(&self) -> Option<&FreestandingArgumentList> {
|
||
None
|
||
}
|
||
fn as_job_conjunction_continuation_list(&self) -> Option<&JobConjunctionContinuationList> {
|
||
None
|
||
}
|
||
fn as_maybe_newlines(&self) -> Option<&MaybeNewlines> {
|
||
None
|
||
}
|
||
fn as_case_item_list(&self) -> Option<&CaseItemList> {
|
||
None
|
||
}
|
||
fn as_argument(&self) -> Option<&Argument> {
|
||
None
|
||
}
|
||
fn as_argument_list(&self) -> Option<&ArgumentList> {
|
||
None
|
||
}
|
||
fn as_job_list(&self) -> Option<&JobList> {
|
||
None
|
||
}
|
||
}
|
||
|
||
trait ConcreteNodeMut {
|
||
// Cast to any sub-trait.
|
||
fn as_mut_leaf(&mut self) -> Option<&mut dyn Leaf> {
|
||
None
|
||
}
|
||
fn as_mut_keyword(&mut self) -> Option<&mut dyn Keyword> {
|
||
None
|
||
}
|
||
fn as_mut_token(&mut self) -> Option<&mut dyn Token> {
|
||
None
|
||
}
|
||
|
||
// Cast to any node type.
|
||
fn as_mut_redirection(&mut self) -> Option<&mut Redirection> {
|
||
None
|
||
}
|
||
fn as_mut_variable_assignment(&mut self) -> Option<&mut VariableAssignment> {
|
||
None
|
||
}
|
||
fn as_mut_variable_assignment_list(&mut self) -> Option<&mut VariableAssignmentList> {
|
||
None
|
||
}
|
||
fn as_mut_argument_or_redirection(&mut self) -> Option<&mut ArgumentOrRedirection> {
|
||
None
|
||
}
|
||
fn as_mut_argument_or_redirection_list(&mut self) -> Option<&mut ArgumentOrRedirectionList> {
|
||
None
|
||
}
|
||
fn as_mut_statement(&mut self) -> Option<&mut Statement> {
|
||
None
|
||
}
|
||
fn as_mut_job_pipeline(&mut self) -> Option<&mut JobPipeline> {
|
||
None
|
||
}
|
||
fn as_mut_job_conjunction(&mut self) -> Option<&mut JobConjunction> {
|
||
None
|
||
}
|
||
fn as_mut_for_header(&mut self) -> Option<&mut ForHeader> {
|
||
None
|
||
}
|
||
fn as_mut_while_header(&mut self) -> Option<&mut WhileHeader> {
|
||
None
|
||
}
|
||
fn as_mut_function_header(&mut self) -> Option<&mut FunctionHeader> {
|
||
None
|
||
}
|
||
fn as_mut_begin_header(&mut self) -> Option<&mut BeginHeader> {
|
||
None
|
||
}
|
||
fn as_mut_block_statement(&mut self) -> Option<&mut BlockStatement> {
|
||
None
|
||
}
|
||
fn as_mut_if_clause(&mut self) -> Option<&mut IfClause> {
|
||
None
|
||
}
|
||
fn as_mut_elseif_clause(&mut self) -> Option<&mut ElseifClause> {
|
||
None
|
||
}
|
||
fn as_mut_elseif_clause_list(&mut self) -> Option<&mut ElseifClauseList> {
|
||
None
|
||
}
|
||
fn as_mut_else_clause(&mut self) -> Option<&mut ElseClause> {
|
||
None
|
||
}
|
||
fn as_mut_if_statement(&mut self) -> Option<&mut IfStatement> {
|
||
None
|
||
}
|
||
fn as_mut_case_item(&mut self) -> Option<&mut CaseItem> {
|
||
None
|
||
}
|
||
fn as_mut_switch_statement(&mut self) -> Option<&mut SwitchStatement> {
|
||
None
|
||
}
|
||
fn as_mut_decorated_statement(&mut self) -> Option<&mut DecoratedStatement> {
|
||
None
|
||
}
|
||
fn as_mut_not_statement(&mut self) -> Option<&mut NotStatement> {
|
||
None
|
||
}
|
||
fn as_mut_job_continuation(&mut self) -> Option<&mut JobContinuation> {
|
||
None
|
||
}
|
||
fn as_mut_job_continuation_list(&mut self) -> Option<&mut JobContinuationList> {
|
||
None
|
||
}
|
||
fn as_mut_job_conjunction_continuation(&mut self) -> Option<&mut JobConjunctionContinuation> {
|
||
None
|
||
}
|
||
fn as_mut_andor_job(&mut self) -> Option<&mut AndorJob> {
|
||
None
|
||
}
|
||
fn as_mut_andor_job_list(&mut self) -> Option<&mut AndorJobList> {
|
||
None
|
||
}
|
||
fn as_mut_freestanding_argument_list(&mut self) -> Option<&mut FreestandingArgumentList> {
|
||
None
|
||
}
|
||
fn as_mut_job_conjunction_continuation_list(
|
||
&mut self,
|
||
) -> Option<&mut JobConjunctionContinuationList> {
|
||
None
|
||
}
|
||
fn as_mut_maybe_newlines(&mut self) -> Option<&mut MaybeNewlines> {
|
||
None
|
||
}
|
||
fn as_mut_case_item_list(&mut self) -> Option<&mut CaseItemList> {
|
||
None
|
||
}
|
||
fn as_mut_argument(&mut self) -> Option<&mut Argument> {
|
||
None
|
||
}
|
||
fn as_mut_argument_list(&mut self) -> Option<&mut ArgumentList> {
|
||
None
|
||
}
|
||
fn as_mut_job_list(&mut self) -> Option<&mut JobList> {
|
||
None
|
||
}
|
||
}
|
||
|
||
/// Trait for all "leaf" nodes: nodes with no ast children.
|
||
pub trait Leaf: Node {
|
||
/// Returns none if this node is "unsourced." This happens if for whatever reason we are
|
||
/// unable to parse the node, either because we had a parse error and recovered, or because
|
||
/// we accepted incomplete and the token stream was exhausted.
|
||
fn range(&self) -> Option<SourceRange>;
|
||
fn range_mut(&mut self) -> &mut Option<SourceRange>;
|
||
fn leaf_as_node_ffi(&self) -> &dyn Node;
|
||
}
|
||
|
||
// A token node is a node which contains a token, which must be one of a fixed set.
|
||
pub trait Token: Leaf {
|
||
/// The token type which was parsed.
|
||
fn token_type(&self) -> ParseTokenType;
|
||
fn token_type_mut(&mut self) -> &mut ParseTokenType;
|
||
fn allowed_tokens(&self) -> &'static [ParseTokenType];
|
||
/// \return whether a token type is allowed in this token_t, i.e. is a member of our Toks list.
|
||
fn allows_token(&self, token_type: ParseTokenType) -> bool {
|
||
self.allowed_tokens().contains(&token_type)
|
||
}
|
||
}
|
||
|
||
/// A keyword node is a node which contains a keyword, which must be one of a fixed set.
|
||
pub trait Keyword: Leaf {
|
||
fn keyword(&self) -> ParseKeyword;
|
||
fn keyword_mut(&mut self) -> &mut ParseKeyword;
|
||
fn allowed_keywords(&self) -> &'static [ParseKeyword];
|
||
fn allows_keyword(&self, kw: ParseKeyword) -> bool {
|
||
self.allowed_keywords().contains(&kw)
|
||
}
|
||
}
|
||
|
||
// A simple variable-sized array, possibly empty.
|
||
pub trait List: Node {
|
||
type ContentsNode: Node + Default;
|
||
fn contents(&self) -> &[Box<Self::ContentsNode>];
|
||
fn contents_mut(&mut self) -> &mut Vec<Box<Self::ContentsNode>>;
|
||
/// \return our count.
|
||
fn count(&self) -> usize {
|
||
self.contents().len()
|
||
}
|
||
/// \return whether we are empty.
|
||
fn is_empty(&self) -> bool {
|
||
self.contents().is_empty()
|
||
}
|
||
}
|
||
|
||
/// Implement the node trait.
|
||
macro_rules! implement_node {
|
||
(
|
||
$name:ident,
|
||
$category:ident,
|
||
$type:ident $(,)?
|
||
) => {
|
||
impl Node for $name {
|
||
fn typ(&self) -> Type {
|
||
Type::$type
|
||
}
|
||
fn parent(&self) -> Option<&dyn Node> {
|
||
self.parent.map(|p| unsafe { &*p })
|
||
}
|
||
fn parent_ffi(&self) -> &Option<*const dyn Node> {
|
||
&self.parent
|
||
}
|
||
fn category(&self) -> Category {
|
||
Category::$category
|
||
}
|
||
fn try_source_range(&self) -> Option<SourceRange> {
|
||
let mut visitor = SourceRangeVisitor {
|
||
total: SourceRange::new(0, 0),
|
||
any_unsourced: false,
|
||
};
|
||
visitor.visit(self);
|
||
if visitor.any_unsourced {
|
||
None
|
||
} else {
|
||
Some(visitor.total)
|
||
}
|
||
}
|
||
fn as_ptr(&self) -> *const () {
|
||
(self as *const $name).cast()
|
||
}
|
||
}
|
||
impl NodeMut for $name {
|
||
fn as_node(&self) -> &dyn Node {
|
||
self
|
||
}
|
||
}
|
||
};
|
||
}
|
||
|
||
/// Implement the leaf trait.
|
||
macro_rules! implement_leaf {
|
||
( $name:ident ) => {
|
||
impl Leaf for $name {
|
||
fn range(&self) -> Option<SourceRange> {
|
||
self.range
|
||
}
|
||
fn range_mut(&mut self) -> &mut Option<SourceRange> {
|
||
&mut self.range
|
||
}
|
||
fn leaf_as_node_ffi(&self) -> &dyn Node {
|
||
self
|
||
}
|
||
}
|
||
impl Acceptor for $name {
|
||
#[allow(unused_variables)]
|
||
fn accept<'a>(&'a self, visitor: &mut dyn NodeVisitor<'a>, reversed: bool) {}
|
||
}
|
||
impl AcceptorMut for $name {
|
||
#[allow(unused_variables)]
|
||
fn accept_mut(&mut self, visitor: &mut dyn NodeVisitorMut, reversed: bool) {
|
||
visitor.will_visit_fields_of(self);
|
||
visitor.did_visit_fields_of(self, VisitResult::Continue(()));
|
||
}
|
||
}
|
||
impl $name {
|
||
/// Set the parent fields of all nodes in the tree rooted at \p self.
|
||
fn set_parents(&mut self) {}
|
||
}
|
||
};
|
||
}
|
||
|
||
/// Define a node that implements the keyword trait.
|
||
macro_rules! define_keyword_node {
|
||
( $name:ident, $($allowed:expr),* $(,)? ) => {
|
||
#[derive(Default, Debug)]
|
||
pub struct $name {
|
||
parent: Option<*const dyn Node>,
|
||
range: Option<SourceRange>,
|
||
keyword: ParseKeyword,
|
||
}
|
||
implement_node!($name, leaf, keyword_base);
|
||
implement_leaf!($name);
|
||
impl ConcreteNode for $name {
|
||
fn as_leaf(&self) -> Option<&dyn Leaf> {
|
||
Some(self)
|
||
}
|
||
fn as_keyword(&self) -> Option<&dyn Keyword> {
|
||
Some(self)
|
||
}
|
||
}
|
||
impl ConcreteNodeMut for $name {
|
||
fn as_mut_leaf(&mut self) -> Option<&mut dyn Leaf> {
|
||
Some(self)
|
||
}
|
||
fn as_mut_keyword(&mut self) -> Option<&mut dyn Keyword> {
|
||
Some(self)
|
||
}
|
||
}
|
||
impl Keyword for $name {
|
||
fn keyword(&self) -> ParseKeyword {
|
||
self.keyword
|
||
}
|
||
fn keyword_mut(&mut self) -> &mut ParseKeyword {
|
||
&mut self.keyword
|
||
}
|
||
fn allowed_keywords(&self) -> &'static [ParseKeyword] {
|
||
&[$($allowed),*]
|
||
}
|
||
}
|
||
}
|
||
}
|
||
|
||
/// Define a node that implements the token trait.
|
||
macro_rules! define_token_node {
|
||
( $name:ident, $($allowed:expr),* $(,)? ) => {
|
||
#[derive(Default, Debug)]
|
||
pub struct $name {
|
||
parent: Option<*const dyn Node>,
|
||
range: Option<SourceRange>,
|
||
parse_token_type: ParseTokenType,
|
||
}
|
||
implement_node!($name, leaf, token_base);
|
||
implement_leaf!($name);
|
||
impl ConcreteNode for $name {
|
||
fn as_leaf(&self) -> Option<&dyn Leaf> {
|
||
Some(self)
|
||
}
|
||
fn as_token(&self) -> Option<&dyn Token> {
|
||
Some(self)
|
||
}
|
||
}
|
||
impl ConcreteNodeMut for $name {
|
||
fn as_mut_leaf(&mut self) -> Option<&mut dyn Leaf> {
|
||
Some(self)
|
||
}
|
||
fn as_mut_token(&mut self) -> Option<&mut dyn Token> {
|
||
Some(self)
|
||
}
|
||
}
|
||
impl Token for $name {
|
||
fn token_type(&self) -> ParseTokenType {
|
||
self.parse_token_type
|
||
}
|
||
fn token_type_mut(&mut self) -> &mut ParseTokenType {
|
||
&mut self.parse_token_type
|
||
}
|
||
fn allowed_tokens(&self) -> &'static [ParseTokenType] {
|
||
&[$($allowed),*]
|
||
}
|
||
}
|
||
}
|
||
}
|
||
|
||
/// Define a node that implements the list trait.
|
||
macro_rules! define_list_node {
|
||
(
|
||
$name:ident,
|
||
$type:tt,
|
||
$contents:ident
|
||
) => {
|
||
#[derive(Default, Debug)]
|
||
pub struct $name {
|
||
parent: Option<*const dyn Node>,
|
||
list_contents: Vec<Box<$contents>>,
|
||
}
|
||
implement_node!($name, list, $type);
|
||
impl List for $name {
|
||
type ContentsNode = $contents;
|
||
fn contents(&self) -> &[Box<Self::ContentsNode>] {
|
||
&self.list_contents
|
||
}
|
||
fn contents_mut(&mut self) -> &mut Vec<Box<Self::ContentsNode>> {
|
||
&mut self.list_contents
|
||
}
|
||
}
|
||
impl $name {
|
||
/// Iteration support.
|
||
fn iter<F>(&self) -> impl Iterator<Item = &<$name as List>::ContentsNode> {
|
||
self.contents().iter().map(|b| &**b)
|
||
}
|
||
}
|
||
impl Index<usize> for $name {
|
||
type Output = <$name as List>::ContentsNode;
|
||
fn index(&self, index: usize) -> &Self::Output {
|
||
&*self.contents()[index]
|
||
}
|
||
}
|
||
impl IndexMut<usize> for $name {
|
||
fn index_mut(&mut self, index: usize) -> &mut Self::Output {
|
||
&mut *self.contents_mut()[index]
|
||
}
|
||
}
|
||
impl Acceptor for $name {
|
||
#[allow(unused_variables)]
|
||
fn accept<'a>(&'a self, visitor: &mut dyn NodeVisitor<'a>, reversed: bool) {
|
||
accept_list_visitor!(Self, accept, visit, self, visitor, reversed, $contents);
|
||
}
|
||
}
|
||
impl AcceptorMut for $name {
|
||
#[allow(unused_variables)]
|
||
fn accept_mut(&mut self, visitor: &mut dyn NodeVisitorMut, reversed: bool) {
|
||
visitor.will_visit_fields_of(self);
|
||
let flow = accept_list_visitor!(
|
||
Self, accept_mut, visit_mut, self, visitor, reversed, $contents
|
||
);
|
||
visitor.did_visit_fields_of(self, flow);
|
||
}
|
||
}
|
||
impl $name {
|
||
/// Set the parent fields of all nodes in the tree rooted at \p self.
|
||
fn set_parents(&mut self) {
|
||
for i in 0..self.count() {
|
||
self[i].parent = Some(self);
|
||
self[i].set_parents();
|
||
}
|
||
}
|
||
}
|
||
};
|
||
}
|
||
|
||
macro_rules! accept_list_visitor {
|
||
(
|
||
$Self:ident,
|
||
$accept:ident,
|
||
$visit:ident,
|
||
$self:ident,
|
||
$visitor:ident,
|
||
$reversed:ident,
|
||
$list_element:ident
|
||
) => {
|
||
loop {
|
||
let mut result = VisitResult::Continue(());
|
||
// list types pretend their child nodes are direct embeddings.
|
||
// This isn't used during AST construction because we need to construct the list.
|
||
if $reversed {
|
||
for i in (0..$self.count()).rev() {
|
||
result = accept_list_visitor_impl!($self, $visitor, $visit, $self[i]);
|
||
if result.is_break() {
|
||
break;
|
||
}
|
||
}
|
||
} else {
|
||
for i in 0..$self.count() {
|
||
result = accept_list_visitor_impl!($self, $visitor, $visit, $self[i]);
|
||
if result.is_break() {
|
||
break;
|
||
}
|
||
}
|
||
}
|
||
break result;
|
||
}
|
||
};
|
||
}
|
||
|
||
macro_rules! accept_list_visitor_impl {
|
||
(
|
||
$self:ident,
|
||
$visitor:ident,
|
||
visit,
|
||
$child:expr) => {{
|
||
$visitor.visit(&$child);
|
||
VisitResult::Continue(())
|
||
}};
|
||
(
|
||
$self:ident,
|
||
$visitor:ident,
|
||
visit_mut,
|
||
$child:expr) => {
|
||
$visitor.visit_mut(&mut $child)
|
||
};
|
||
}
|
||
|
||
/// Implement the acceptor trait for the given branch node.
|
||
macro_rules! implement_acceptor_for_branch {
|
||
(
|
||
$name:ident
|
||
$(, ($field_name:ident: $field_type:tt) )*
|
||
$(,)?
|
||
) => {
|
||
impl Acceptor for $name {
|
||
#[allow(unused_variables)]
|
||
fn accept<'a>(&'a self, visitor: &mut dyn NodeVisitor<'a>, reversed: bool){
|
||
visitor_accept_field!(
|
||
Self,
|
||
accept,
|
||
visit,
|
||
self,
|
||
visitor,
|
||
reversed,
|
||
( $( $field_name: $field_type, )* ) );
|
||
}
|
||
}
|
||
impl AcceptorMut for $name {
|
||
#[allow(unused_variables)]
|
||
fn accept_mut(&mut self, visitor: &mut dyn NodeVisitorMut, reversed: bool) {
|
||
visitor.will_visit_fields_of(self);
|
||
let flow = visitor_accept_field!(
|
||
Self,
|
||
accept_mut,
|
||
visit_mut,
|
||
self,
|
||
visitor,
|
||
reversed,
|
||
( $( $field_name: $field_type, )* ));
|
||
visitor.did_visit_fields_of(self, flow);
|
||
}
|
||
}
|
||
impl $name {
|
||
/// Set the parent fields of all nodes in the tree rooted at \p self.
|
||
fn set_parents(&mut self) {
|
||
$(
|
||
set_parent_of_field!(self, $field_name, $field_type);
|
||
)*
|
||
}
|
||
}
|
||
}
|
||
}
|
||
|
||
/// Visit the given fields in order, returning whether the visitation succeded.
|
||
macro_rules! visitor_accept_field {
|
||
(
|
||
$Self:ident,
|
||
$accept:ident,
|
||
$visit:ident,
|
||
$self:ident,
|
||
$visitor:ident,
|
||
$reversed:ident,
|
||
$fields:tt
|
||
) => {
|
||
loop {
|
||
visitor_accept_field_impl!($visit, $self, $visitor, $reversed, $fields);
|
||
break VisitResult::Continue(());
|
||
}
|
||
};
|
||
}
|
||
|
||
/// Visit the given fields in order, breaking if a visitation fails.
|
||
macro_rules! visitor_accept_field_impl {
|
||
// Base case: no fields left to visit.
|
||
(
|
||
$visit:ident,
|
||
$self:ident,
|
||
$visitor:ident,
|
||
$reversed:ident,
|
||
()
|
||
) => {};
|
||
// Visit the first or last field and then the rest.
|
||
(
|
||
$visit:ident,
|
||
$self:ident,
|
||
$visitor:ident,
|
||
$reversed:ident,
|
||
(
|
||
$field_name:ident: $field_type:tt,
|
||
$( $field_names:ident: $field_types:tt, )*
|
||
)
|
||
) => {
|
||
if !$reversed {
|
||
visit_1_field!($visit, ($self.$field_name), $field_type, $visitor);
|
||
}
|
||
visitor_accept_field_impl!(
|
||
$visit, $self, $visitor, $reversed,
|
||
( $( $field_names: $field_types, )* ));
|
||
if $reversed {
|
||
visit_1_field!($visit, ($self.$field_name), $field_type, $visitor);
|
||
}
|
||
}
|
||
}
|
||
|
||
/// Visit the given field, breaking on failure.
|
||
macro_rules! visit_1_field {
|
||
(
|
||
visit,
|
||
$field:expr,
|
||
$field_type:tt,
|
||
$visitor:ident
|
||
) => {
|
||
visit_1_field_impl!(visit, $field, $field_type, $visitor);
|
||
};
|
||
(
|
||
visit_mut,
|
||
$field:expr,
|
||
$field_type:tt,
|
||
$visitor:ident
|
||
) => {
|
||
let result = visit_1_field_impl!(visit_mut, $field, $field_type, $visitor);
|
||
if result.is_break() {
|
||
break result;
|
||
}
|
||
};
|
||
}
|
||
|
||
/// Visit the given field.
|
||
macro_rules! visit_1_field_impl {
|
||
(
|
||
$visit:ident,
|
||
$field:expr,
|
||
(Box<$field_type:ident>),
|
||
$visitor:ident
|
||
) => {
|
||
visit_union_field!($visit, $field_type, $field, $visitor)
|
||
};
|
||
(
|
||
$visit:ident,
|
||
$field:expr,
|
||
(Option<$field_type:ident>),
|
||
$visitor:ident
|
||
) => {
|
||
visit_optional_field!($visit, $field_type, $field, $visitor)
|
||
};
|
||
(
|
||
$visit:ident,
|
||
$field:expr,
|
||
$field_type:tt,
|
||
$visitor:ident
|
||
) => {
|
||
$visitor.$visit(apply_borrow!($visit, $field))
|
||
};
|
||
}
|
||
|
||
macro_rules! apply_borrow {
|
||
( visit, $expr:expr ) => {
|
||
&$expr
|
||
};
|
||
( visit_mut, $expr:expr ) => {
|
||
&mut $expr
|
||
};
|
||
}
|
||
|
||
macro_rules! visit_union_field {
|
||
(
|
||
visit,
|
||
$field_type:ident,
|
||
$field:expr,
|
||
$visitor:ident
|
||
) => {
|
||
$visitor.visit($field.embedded_node().as_node())
|
||
};
|
||
(
|
||
visit_mut,
|
||
$field_type:ident,
|
||
$field:expr,
|
||
$visitor:ident
|
||
) => {
|
||
visit_union_field_mut!($field_type, $visitor, $field)
|
||
};
|
||
}
|
||
|
||
macro_rules! visit_union_field_mut {
|
||
(ArgumentOrRedirectionVariant, $visitor:ident, $field:expr) => {
|
||
$visitor.visit_argument_or_redirection(&mut $field)
|
||
};
|
||
(BlockStatementHeaderVariant, $visitor:ident, $field:expr) => {
|
||
$visitor.visit_block_statement_header(&mut $field)
|
||
};
|
||
(StatementVariant, $visitor:ident, $field:expr) => {
|
||
$visitor.visit_statement(&mut $field)
|
||
};
|
||
}
|
||
|
||
macro_rules! visit_optional_field {
|
||
(
|
||
visit,
|
||
$field_type:ident,
|
||
$field:expr,
|
||
$visitor:ident
|
||
) => {
|
||
match &$field {
|
||
Some(value) => $visitor.visit(&*value),
|
||
None => visit_result!(visit),
|
||
}
|
||
};
|
||
(
|
||
visit_mut,
|
||
$field_type:ident,
|
||
$field:expr,
|
||
$visitor:ident
|
||
) => {{
|
||
visit_optional_field_mut!($field_type, $field, $visitor);
|
||
VisitResult::Continue(())
|
||
}};
|
||
}
|
||
|
||
macro_rules! visit_optional_field_mut {
|
||
(DecoratedStatementDecorator, $field:expr, $visitor:ident) => {
|
||
$visitor.visit_decorated_statement_decorator(&mut $field);
|
||
};
|
||
(JobConjunctionDecorator, $field:expr, $visitor:ident) => {
|
||
$visitor.visit_job_conjunction_decorator(&mut $field);
|
||
};
|
||
(ElseClause, $field:expr, $visitor:ident) => {
|
||
$visitor.visit_else_clause(&mut $field);
|
||
};
|
||
(SemiNl, $field:expr, $visitor:ident) => {
|
||
$visitor.visit_semi_nl(&mut $field);
|
||
};
|
||
(KeywordTime, $field:expr, $visitor:ident) => {
|
||
$visitor.visit_time(&mut $field);
|
||
};
|
||
(TokenBackground, $field:expr, $visitor:ident) => {
|
||
$visitor.visit_token_background(&mut $field);
|
||
};
|
||
}
|
||
|
||
macro_rules! visit_result {
|
||
( visit) => {
|
||
()
|
||
};
|
||
( visit_mut ) => {
|
||
VisitResult::Continue(())
|
||
};
|
||
}
|
||
|
||
macro_rules! set_parent_of_field {
|
||
(
|
||
$self:ident,
|
||
$field_name:ident,
|
||
(Box<$field_type:ident>)
|
||
) => {
|
||
set_parent_of_union_field!($self, $field_name, $field_type);
|
||
};
|
||
(
|
||
$self:ident,
|
||
$field_name:ident,
|
||
(Option<$field_type:ident>)
|
||
) => {
|
||
if $self.$field_name.is_some() {
|
||
$self.$field_name.as_mut().unwrap().parent = Some($self);
|
||
$self.$field_name.as_mut().unwrap().set_parents();
|
||
}
|
||
};
|
||
(
|
||
$self:ident,
|
||
$field_name:ident,
|
||
$field_type:tt
|
||
) => {
|
||
$self.$field_name.parent = Some($self);
|
||
$self.$field_name.set_parents();
|
||
};
|
||
}
|
||
|
||
macro_rules! set_parent_of_union_field {
|
||
(
|
||
$self:ident,
|
||
$field_name:ident,
|
||
ArgumentOrRedirectionVariant
|
||
) => {
|
||
if matches!(
|
||
*$self.$field_name,
|
||
ArgumentOrRedirectionVariant::Argument(_)
|
||
) {
|
||
$self.$field_name.as_mut_argument().parent = Some($self);
|
||
$self.$field_name.as_mut_argument().set_parents();
|
||
} else {
|
||
$self.$field_name.as_mut_redirection().parent = Some($self);
|
||
$self.$field_name.as_mut_redirection().set_parents();
|
||
}
|
||
};
|
||
(
|
||
$self:ident,
|
||
$field_name:ident,
|
||
StatementVariant
|
||
) => {
|
||
if matches!(*$self.$field_name, StatementVariant::NotStatement(_)) {
|
||
$self.$field_name.as_mut_not_statement().parent = Some($self);
|
||
$self.$field_name.as_mut_not_statement().set_parents();
|
||
} else if matches!(*$self.$field_name, StatementVariant::BlockStatement(_)) {
|
||
$self.$field_name.as_mut_block_statement().parent = Some($self);
|
||
$self.$field_name.as_mut_block_statement().set_parents();
|
||
} else if matches!(*$self.$field_name, StatementVariant::IfStatement(_)) {
|
||
$self.$field_name.as_mut_if_statement().parent = Some($self);
|
||
$self.$field_name.as_mut_if_statement().set_parents();
|
||
} else if matches!(*$self.$field_name, StatementVariant::SwitchStatement(_)) {
|
||
$self.$field_name.as_mut_switch_statement().parent = Some($self);
|
||
$self.$field_name.as_mut_switch_statement().set_parents();
|
||
} else if matches!(*$self.$field_name, StatementVariant::DecoratedStatement(_)) {
|
||
$self.$field_name.as_mut_decorated_statement().parent = Some($self);
|
||
$self.$field_name.as_mut_decorated_statement().set_parents();
|
||
}
|
||
};
|
||
(
|
||
$self:ident,
|
||
$field_name:ident,
|
||
BlockStatementHeaderVariant
|
||
) => {
|
||
if matches!(
|
||
*$self.$field_name,
|
||
BlockStatementHeaderVariant::ForHeader(_)
|
||
) {
|
||
$self.$field_name.as_mut_for_header().parent = Some($self);
|
||
$self.$field_name.as_mut_for_header().set_parents();
|
||
} else if matches!(
|
||
*$self.$field_name,
|
||
BlockStatementHeaderVariant::WhileHeader(_)
|
||
) {
|
||
$self.$field_name.as_mut_while_header().parent = Some($self);
|
||
$self.$field_name.as_mut_while_header().set_parents();
|
||
} else if matches!(
|
||
*$self.$field_name,
|
||
BlockStatementHeaderVariant::FunctionHeader(_)
|
||
) {
|
||
$self.$field_name.as_mut_function_header().parent = Some($self);
|
||
$self.$field_name.as_mut_function_header().set_parents();
|
||
} else if matches!(
|
||
*$self.$field_name,
|
||
BlockStatementHeaderVariant::BeginHeader(_)
|
||
) {
|
||
$self.$field_name.as_mut_begin_header().parent = Some($self);
|
||
$self.$field_name.as_mut_begin_header().set_parents();
|
||
}
|
||
};
|
||
}
|
||
|
||
/// A redirection has an operator like > or 2>, and a target like /dev/null or &1.
|
||
/// Note that pipes are not redirections.
|
||
#[derive(Default, Debug)]
|
||
pub struct Redirection {
|
||
parent: Option<*const dyn Node>,
|
||
pub oper: TokenRedirection,
|
||
pub target: String_,
|
||
}
|
||
implement_node!(Redirection, branch, redirection);
|
||
implement_acceptor_for_branch!(Redirection, (oper: TokenRedirection), (target: String_));
|
||
impl ConcreteNode for Redirection {
|
||
fn as_redirection(&self) -> Option<&Redirection> {
|
||
Some(self)
|
||
}
|
||
}
|
||
impl ConcreteNodeMut for Redirection {
|
||
fn as_mut_redirection(&mut self) -> Option<&mut Redirection> {
|
||
Some(self)
|
||
}
|
||
}
|
||
|
||
define_list_node!(
|
||
VariableAssignmentList,
|
||
variable_assignment_list,
|
||
VariableAssignment
|
||
);
|
||
impl ConcreteNode for VariableAssignmentList {
|
||
fn as_variable_assignment_list(&self) -> Option<&VariableAssignmentList> {
|
||
Some(self)
|
||
}
|
||
}
|
||
impl ConcreteNodeMut for VariableAssignmentList {
|
||
fn as_mut_variable_assignment_list(&mut self) -> Option<&mut VariableAssignmentList> {
|
||
Some(self)
|
||
}
|
||
}
|
||
|
||
/// An argument or redirection holds either an argument or redirection.
|
||
#[derive(Default, Debug)]
|
||
pub struct ArgumentOrRedirection {
|
||
parent: Option<*const dyn Node>,
|
||
pub contents: Box<ArgumentOrRedirectionVariant>,
|
||
}
|
||
implement_node!(ArgumentOrRedirection, branch, argument_or_redirection);
|
||
implement_acceptor_for_branch!(
|
||
ArgumentOrRedirection,
|
||
(contents: (Box<ArgumentOrRedirectionVariant>))
|
||
);
|
||
impl ConcreteNode for ArgumentOrRedirection {
|
||
fn as_argument_or_redirection(&self) -> Option<&ArgumentOrRedirection> {
|
||
Some(self)
|
||
}
|
||
}
|
||
impl ConcreteNodeMut for ArgumentOrRedirection {
|
||
fn as_mut_argument_or_redirection(&mut self) -> Option<&mut ArgumentOrRedirection> {
|
||
Some(self)
|
||
}
|
||
}
|
||
|
||
define_list_node!(
|
||
ArgumentOrRedirectionList,
|
||
argument_or_redirection_list,
|
||
ArgumentOrRedirection
|
||
);
|
||
impl ConcreteNode for ArgumentOrRedirectionList {
|
||
fn as_argument_or_redirection_list(&self) -> Option<&ArgumentOrRedirectionList> {
|
||
Some(self)
|
||
}
|
||
}
|
||
impl ConcreteNodeMut for ArgumentOrRedirectionList {
|
||
fn as_mut_argument_or_redirection_list(&mut self) -> Option<&mut ArgumentOrRedirectionList> {
|
||
Some(self)
|
||
}
|
||
}
|
||
|
||
/// A statement is a normal command, or an if / while / etc
|
||
#[derive(Default, Debug)]
|
||
pub struct Statement {
|
||
parent: Option<*const dyn Node>,
|
||
pub contents: Box<StatementVariant>,
|
||
}
|
||
implement_node!(Statement, branch, statement);
|
||
implement_acceptor_for_branch!(Statement, (contents: (Box<StatementVariant>)));
|
||
impl ConcreteNode for Statement {
|
||
fn as_statement(&self) -> Option<&Statement> {
|
||
Some(self)
|
||
}
|
||
}
|
||
impl ConcreteNodeMut for Statement {
|
||
fn as_mut_statement(&mut self) -> Option<&mut Statement> {
|
||
Some(self)
|
||
}
|
||
}
|
||
|
||
/// A job is a non-empty list of statements, separated by pipes. (Non-empty is useful for cases
|
||
/// like if statements, where we require a command).
|
||
#[derive(Default, Debug)]
|
||
pub struct JobPipeline {
|
||
parent: Option<*const dyn Node>,
|
||
/// Maybe the time keyword.
|
||
pub time: Option<KeywordTime>,
|
||
/// A (possibly empty) list of variable assignments.
|
||
pub variables: VariableAssignmentList,
|
||
/// The statement.
|
||
pub statement: Statement,
|
||
/// Piped remainder.
|
||
pub continuation: JobContinuationList,
|
||
/// Maybe backgrounded.
|
||
pub bg: Option<TokenBackground>,
|
||
}
|
||
implement_node!(JobPipeline, branch, job_pipeline);
|
||
implement_acceptor_for_branch!(
|
||
JobPipeline,
|
||
(time: (Option<KeywordTime>)),
|
||
(variables: (VariableAssignmentList)),
|
||
(statement: (Statement)),
|
||
(continuation: (JobContinuationList)),
|
||
(bg: (Option<TokenBackground>)),
|
||
);
|
||
impl ConcreteNode for JobPipeline {
|
||
fn as_job_pipeline(&self) -> Option<&JobPipeline> {
|
||
Some(self)
|
||
}
|
||
}
|
||
impl ConcreteNodeMut for JobPipeline {
|
||
fn as_mut_job_pipeline(&mut self) -> Option<&mut JobPipeline> {
|
||
Some(self)
|
||
}
|
||
}
|
||
|
||
/// A job_conjunction is a job followed by a && or || continuations.
|
||
#[derive(Default, Debug)]
|
||
pub struct JobConjunction {
|
||
parent: Option<*const dyn Node>,
|
||
/// The job conjunction decorator.
|
||
pub decorator: Option<JobConjunctionDecorator>,
|
||
/// The job itself.
|
||
pub job: JobPipeline,
|
||
/// The rest of the job conjunction, with && or ||s.
|
||
pub continuations: JobConjunctionContinuationList,
|
||
/// A terminating semicolon or newline. This is marked optional because it may not be
|
||
/// present, for example the command `echo foo` may not have a terminating newline. It will
|
||
/// only fail to be present if we ran out of tokens.
|
||
pub semi_nl: Option<SemiNl>,
|
||
}
|
||
implement_node!(JobConjunction, branch, job_conjunction);
|
||
implement_acceptor_for_branch!(
|
||
JobConjunction,
|
||
(decorator: (Option<JobConjunctionDecorator>)),
|
||
(job: (JobPipeline)),
|
||
(continuations: (JobConjunctionContinuationList)),
|
||
(semi_nl: (Option<SemiNl>)),
|
||
);
|
||
impl ConcreteNode for JobConjunction {
|
||
fn as_job_conjunction(&self) -> Option<&JobConjunction> {
|
||
Some(self)
|
||
}
|
||
}
|
||
impl ConcreteNodeMut for JobConjunction {
|
||
fn as_mut_job_conjunction(&mut self) -> Option<&mut JobConjunction> {
|
||
Some(self)
|
||
}
|
||
}
|
||
|
||
#[derive(Default, Debug)]
|
||
pub struct ForHeader {
|
||
parent: Option<*const dyn Node>,
|
||
/// 'for'
|
||
pub kw_for: KeywordFor,
|
||
/// var_name
|
||
pub var_name: String_,
|
||
/// 'in'
|
||
pub kw_in: KeywordIn,
|
||
/// list of arguments
|
||
pub args: ArgumentList,
|
||
/// newline or semicolon
|
||
pub semi_nl: SemiNl,
|
||
}
|
||
implement_node!(ForHeader, branch, for_header);
|
||
implement_acceptor_for_branch!(
|
||
ForHeader,
|
||
(kw_for: (KeywordFor)),
|
||
(var_name: (String_)),
|
||
(kw_in: (KeywordIn)),
|
||
(args: (ArgumentList)),
|
||
(semi_nl: (SemiNl)),
|
||
);
|
||
impl ConcreteNode for ForHeader {
|
||
fn as_for_header(&self) -> Option<&ForHeader> {
|
||
Some(self)
|
||
}
|
||
}
|
||
impl ConcreteNodeMut for ForHeader {
|
||
fn as_mut_for_header(&mut self) -> Option<&mut ForHeader> {
|
||
Some(self)
|
||
}
|
||
}
|
||
|
||
#[derive(Default, Debug)]
|
||
pub struct WhileHeader {
|
||
parent: Option<*const dyn Node>,
|
||
/// 'while'
|
||
pub kw_while: KeywordWhile,
|
||
pub condition: JobConjunction,
|
||
pub andor_tail: AndorJobList,
|
||
}
|
||
implement_node!(WhileHeader, branch, while_header);
|
||
implement_acceptor_for_branch!(
|
||
WhileHeader,
|
||
(kw_while: (KeywordWhile)),
|
||
(condition: (JobConjunction)),
|
||
(andor_tail: (AndorJobList)),
|
||
);
|
||
impl ConcreteNode for WhileHeader {
|
||
fn as_while_header(&self) -> Option<&WhileHeader> {
|
||
Some(self)
|
||
}
|
||
}
|
||
impl ConcreteNodeMut for WhileHeader {
|
||
fn as_mut_while_header(&mut self) -> Option<&mut WhileHeader> {
|
||
Some(self)
|
||
}
|
||
}
|
||
|
||
#[derive(Default, Debug)]
|
||
pub struct FunctionHeader {
|
||
parent: Option<*const dyn Node>,
|
||
pub kw_function: KeywordFunction,
|
||
/// functions require at least one argument.
|
||
pub first_arg: Argument,
|
||
pub args: ArgumentList,
|
||
pub semi_nl: SemiNl,
|
||
}
|
||
implement_node!(FunctionHeader, branch, function_header);
|
||
implement_acceptor_for_branch!(
|
||
FunctionHeader,
|
||
(kw_function: (KeywordFunction)),
|
||
(first_arg: (Argument)),
|
||
(args: (ArgumentList)),
|
||
(semi_nl: (SemiNl)),
|
||
);
|
||
impl ConcreteNode for FunctionHeader {
|
||
fn as_function_header(&self) -> Option<&FunctionHeader> {
|
||
Some(self)
|
||
}
|
||
}
|
||
impl ConcreteNodeMut for FunctionHeader {
|
||
fn as_mut_function_header(&mut self) -> Option<&mut FunctionHeader> {
|
||
Some(self)
|
||
}
|
||
}
|
||
|
||
#[derive(Default, Debug)]
|
||
pub struct BeginHeader {
|
||
parent: Option<*const dyn Node>,
|
||
pub kw_begin: KeywordBegin,
|
||
/// Note that 'begin' does NOT require a semi or nl afterwards.
|
||
/// This is valid: begin echo hi; end
|
||
pub semi_nl: Option<SemiNl>,
|
||
}
|
||
implement_node!(BeginHeader, branch, begin_header);
|
||
implement_acceptor_for_branch!(
|
||
BeginHeader,
|
||
(kw_begin: (KeywordBegin)),
|
||
(semi_nl: (Option<SemiNl>))
|
||
);
|
||
impl ConcreteNode for BeginHeader {
|
||
fn as_begin_header(&self) -> Option<&BeginHeader> {
|
||
Some(self)
|
||
}
|
||
}
|
||
impl ConcreteNodeMut for BeginHeader {
|
||
fn as_mut_begin_header(&mut self) -> Option<&mut BeginHeader> {
|
||
Some(self)
|
||
}
|
||
}
|
||
|
||
#[derive(Default, Debug)]
|
||
pub struct BlockStatement {
|
||
parent: Option<*const dyn Node>,
|
||
/// A header like for, while, etc.
|
||
pub header: Box<BlockStatementHeaderVariant>,
|
||
/// List of jobs in this block.
|
||
pub jobs: JobList,
|
||
/// The 'end' node.
|
||
pub end: KeywordEnd,
|
||
/// Arguments and redirections associated with the block.
|
||
pub args_or_redirs: ArgumentOrRedirectionList,
|
||
}
|
||
implement_node!(BlockStatement, branch, block_statement);
|
||
implement_acceptor_for_branch!(
|
||
BlockStatement,
|
||
(header: (Box<BlockStatementHeaderVariant>)),
|
||
(jobs: (JobList)),
|
||
(end: (KeywordEnd)),
|
||
(args_or_redirs: (ArgumentOrRedirectionList)),
|
||
);
|
||
impl ConcreteNode for BlockStatement {
|
||
fn as_block_statement(&self) -> Option<&BlockStatement> {
|
||
Some(self)
|
||
}
|
||
}
|
||
impl ConcreteNodeMut for BlockStatement {
|
||
fn as_mut_block_statement(&mut self) -> Option<&mut BlockStatement> {
|
||
Some(self)
|
||
}
|
||
}
|
||
|
||
#[derive(Default, Debug)]
|
||
pub struct IfClause {
|
||
parent: Option<*const dyn Node>,
|
||
/// The 'if' keyword.
|
||
pub kw_if: KeywordIf,
|
||
/// The 'if' condition.
|
||
pub condition: JobConjunction,
|
||
/// 'and/or' tail.
|
||
pub andor_tail: AndorJobList,
|
||
/// The body to execute if the condition is true.
|
||
pub body: JobList,
|
||
}
|
||
implement_node!(IfClause, branch, if_clause);
|
||
implement_acceptor_for_branch!(
|
||
IfClause,
|
||
(kw_if: (KeywordIf)),
|
||
(condition: (JobConjunction)),
|
||
(andor_tail: (AndorJobList)),
|
||
(body: (JobList)),
|
||
);
|
||
impl ConcreteNode for IfClause {
|
||
fn as_if_clause(&self) -> Option<&IfClause> {
|
||
Some(self)
|
||
}
|
||
}
|
||
impl ConcreteNodeMut for IfClause {
|
||
fn as_mut_if_clause(&mut self) -> Option<&mut IfClause> {
|
||
Some(self)
|
||
}
|
||
}
|
||
|
||
#[derive(Default, Debug)]
|
||
pub struct ElseifClause {
|
||
parent: Option<*const dyn Node>,
|
||
/// The 'else' keyword.
|
||
pub kw_else: KeywordElse,
|
||
/// The 'if' clause following it.
|
||
pub if_clause: IfClause,
|
||
}
|
||
implement_node!(ElseifClause, branch, elseif_clause);
|
||
implement_acceptor_for_branch!(
|
||
ElseifClause,
|
||
(kw_else: (KeywordElse)),
|
||
(if_clause: (IfClause)),
|
||
);
|
||
impl ConcreteNode for ElseifClause {
|
||
fn as_elseif_clause(&self) -> Option<&ElseifClause> {
|
||
Some(self)
|
||
}
|
||
}
|
||
impl ConcreteNodeMut for ElseifClause {
|
||
fn as_mut_elseif_clause(&mut self) -> Option<&mut ElseifClause> {
|
||
Some(self)
|
||
}
|
||
}
|
||
|
||
define_list_node!(ElseifClauseList, elseif_clause_list, ElseifClause);
|
||
impl ConcreteNode for ElseifClauseList {
|
||
fn as_elseif_clause_list(&self) -> Option<&ElseifClauseList> {
|
||
Some(self)
|
||
}
|
||
}
|
||
impl ConcreteNodeMut for ElseifClauseList {
|
||
fn as_mut_elseif_clause_list(&mut self) -> Option<&mut ElseifClauseList> {
|
||
Some(self)
|
||
}
|
||
}
|
||
|
||
#[derive(Default, Debug)]
|
||
pub struct ElseClause {
|
||
parent: Option<*const dyn Node>,
|
||
/// else ; body
|
||
pub kw_else: KeywordElse,
|
||
pub semi_nl: SemiNl,
|
||
pub body: JobList,
|
||
}
|
||
implement_node!(ElseClause, branch, else_clause);
|
||
implement_acceptor_for_branch!(
|
||
ElseClause,
|
||
(kw_else: (KeywordElse)),
|
||
(semi_nl: (SemiNl)),
|
||
(body: (JobList)),
|
||
);
|
||
impl ConcreteNode for ElseClause {
|
||
fn as_else_clause(&self) -> Option<&ElseClause> {
|
||
Some(self)
|
||
}
|
||
}
|
||
impl ConcreteNodeMut for ElseClause {
|
||
fn as_mut_else_clause(&mut self) -> Option<&mut ElseClause> {
|
||
Some(self)
|
||
}
|
||
}
|
||
|
||
#[derive(Default, Debug)]
|
||
pub struct IfStatement {
|
||
parent: Option<*const dyn Node>,
|
||
/// if part
|
||
pub if_clause: IfClause,
|
||
/// else if list
|
||
pub elseif_clauses: ElseifClauseList,
|
||
/// else part
|
||
pub else_clause: Option<ElseClause>,
|
||
/// literal end
|
||
pub end: KeywordEnd,
|
||
/// block args / redirs
|
||
pub args_or_redirs: ArgumentOrRedirectionList,
|
||
}
|
||
implement_node!(IfStatement, branch, if_statement);
|
||
implement_acceptor_for_branch!(
|
||
IfStatement,
|
||
(if_clause: (IfClause)),
|
||
(elseif_clauses: (ElseifClauseList)),
|
||
(else_clause: (Option<ElseClause>)),
|
||
(end: (KeywordEnd)),
|
||
(args_or_redirs: (ArgumentOrRedirectionList)),
|
||
);
|
||
impl ConcreteNode for IfStatement {
|
||
fn as_if_statement(&self) -> Option<&IfStatement> {
|
||
Some(self)
|
||
}
|
||
}
|
||
impl ConcreteNodeMut for IfStatement {
|
||
fn as_mut_if_statement(&mut self) -> Option<&mut IfStatement> {
|
||
Some(self)
|
||
}
|
||
}
|
||
|
||
#[derive(Default, Debug)]
|
||
pub struct CaseItem {
|
||
parent: Option<*const dyn Node>,
|
||
/// case <arguments> ; body
|
||
pub kw_case: KeywordCase,
|
||
pub arguments: ArgumentList,
|
||
pub semi_nl: SemiNl,
|
||
pub body: JobList,
|
||
}
|
||
implement_node!(CaseItem, branch, case_item);
|
||
implement_acceptor_for_branch!(
|
||
CaseItem,
|
||
(kw_case: (KeywordCase)),
|
||
(arguments: (ArgumentList)),
|
||
(semi_nl: (SemiNl)),
|
||
(body: (JobList)),
|
||
);
|
||
impl ConcreteNode for CaseItem {
|
||
fn as_case_item(&self) -> Option<&CaseItem> {
|
||
Some(self)
|
||
}
|
||
}
|
||
impl ConcreteNodeMut for CaseItem {
|
||
fn as_mut_case_item(&mut self) -> Option<&mut CaseItem> {
|
||
Some(self)
|
||
}
|
||
}
|
||
|
||
#[derive(Default, Debug)]
|
||
pub struct SwitchStatement {
|
||
parent: Option<*const dyn Node>,
|
||
/// switch <argument> ; body ; end args_redirs
|
||
pub kw_switch: KeywordSwitch,
|
||
pub argument: Argument,
|
||
pub semi_nl: SemiNl,
|
||
pub cases: CaseItemList,
|
||
pub end: KeywordEnd,
|
||
pub args_or_redirs: ArgumentOrRedirectionList,
|
||
}
|
||
implement_node!(SwitchStatement, branch, switch_statement);
|
||
implement_acceptor_for_branch!(
|
||
SwitchStatement,
|
||
(kw_switch: (KeywordSwitch)),
|
||
(argument: (Argument)),
|
||
(semi_nl: (SemiNl)),
|
||
(cases: (CaseItemList)),
|
||
(end: (KeywordEnd)),
|
||
(args_or_redirs: (ArgumentOrRedirectionList)),
|
||
);
|
||
impl ConcreteNode for SwitchStatement {
|
||
fn as_switch_statement(&self) -> Option<&SwitchStatement> {
|
||
Some(self)
|
||
}
|
||
}
|
||
impl ConcreteNodeMut for SwitchStatement {
|
||
fn as_mut_switch_statement(&mut self) -> Option<&mut SwitchStatement> {
|
||
Some(self)
|
||
}
|
||
}
|
||
|
||
/// A decorated_statement is a command with a list of arguments_or_redirections, possibly with
|
||
/// "builtin" or "command" or "exec"
|
||
#[derive(Default, Debug)]
|
||
pub struct DecoratedStatement {
|
||
parent: Option<*const dyn Node>,
|
||
/// An optional decoration (command, builtin, exec, etc).
|
||
pub opt_decoration: Option<DecoratedStatementDecorator>,
|
||
/// Command to run.
|
||
pub command: String_,
|
||
/// Args and redirs
|
||
pub args_or_redirs: ArgumentOrRedirectionList,
|
||
}
|
||
implement_node!(DecoratedStatement, branch, decorated_statement);
|
||
implement_acceptor_for_branch!(
|
||
DecoratedStatement,
|
||
(opt_decoration: (Option<DecoratedStatementDecorator>)),
|
||
(command: (String_)),
|
||
(args_or_redirs: (ArgumentOrRedirectionList)),
|
||
);
|
||
impl ConcreteNode for DecoratedStatement {
|
||
fn as_decorated_statement(&self) -> Option<&DecoratedStatement> {
|
||
Some(self)
|
||
}
|
||
}
|
||
impl ConcreteNodeMut for DecoratedStatement {
|
||
fn as_mut_decorated_statement(&mut self) -> Option<&mut DecoratedStatement> {
|
||
Some(self)
|
||
}
|
||
}
|
||
|
||
/// A not statement like `not true` or `! true`
|
||
#[derive(Default, Debug)]
|
||
pub struct NotStatement {
|
||
parent: Option<*const dyn Node>,
|
||
/// Keyword, either not or exclam.
|
||
pub kw: KeywordNot,
|
||
pub variables: VariableAssignmentList,
|
||
pub time: Option<KeywordTime>,
|
||
pub contents: Statement,
|
||
}
|
||
implement_node!(NotStatement, branch, not_statement);
|
||
implement_acceptor_for_branch!(
|
||
NotStatement,
|
||
(kw: (KeywordNot)),
|
||
(variables: (VariableAssignmentList)),
|
||
(time: (Option<KeywordTime>)),
|
||
(contents: (Statement)),
|
||
);
|
||
impl ConcreteNode for NotStatement {
|
||
fn as_not_statement(&self) -> Option<&NotStatement> {
|
||
Some(self)
|
||
}
|
||
}
|
||
impl ConcreteNodeMut for NotStatement {
|
||
fn as_mut_not_statement(&mut self) -> Option<&mut NotStatement> {
|
||
Some(self)
|
||
}
|
||
}
|
||
|
||
#[derive(Default, Debug)]
|
||
pub struct JobContinuation {
|
||
parent: Option<*const dyn Node>,
|
||
pub pipe: TokenPipe,
|
||
pub newlines: MaybeNewlines,
|
||
pub variables: VariableAssignmentList,
|
||
pub statement: Statement,
|
||
}
|
||
implement_node!(JobContinuation, branch, job_continuation);
|
||
implement_acceptor_for_branch!(
|
||
JobContinuation,
|
||
(pipe: (TokenPipe)),
|
||
(newlines: (MaybeNewlines)),
|
||
(variables: (VariableAssignmentList)),
|
||
(statement: (Statement)),
|
||
);
|
||
impl ConcreteNode for JobContinuation {
|
||
fn as_job_continuation(&self) -> Option<&JobContinuation> {
|
||
Some(self)
|
||
}
|
||
}
|
||
impl ConcreteNodeMut for JobContinuation {
|
||
fn as_mut_job_continuation(&mut self) -> Option<&mut JobContinuation> {
|
||
Some(self)
|
||
}
|
||
}
|
||
|
||
define_list_node!(JobContinuationList, job_continuation_list, JobContinuation);
|
||
impl ConcreteNode for JobContinuationList {
|
||
fn as_job_continuation_list(&self) -> Option<&JobContinuationList> {
|
||
Some(self)
|
||
}
|
||
}
|
||
impl ConcreteNodeMut for JobContinuationList {
|
||
fn as_mut_job_continuation_list(&mut self) -> Option<&mut JobContinuationList> {
|
||
Some(self)
|
||
}
|
||
}
|
||
|
||
#[derive(Default, Debug)]
|
||
pub struct JobConjunctionContinuation {
|
||
parent: Option<*const dyn Node>,
|
||
/// The && or || token.
|
||
pub conjunction: TokenConjunction,
|
||
pub newlines: MaybeNewlines,
|
||
/// The job itself.
|
||
pub job: JobPipeline,
|
||
}
|
||
implement_node!(
|
||
JobConjunctionContinuation,
|
||
branch,
|
||
job_conjunction_continuation
|
||
);
|
||
implement_acceptor_for_branch!(
|
||
JobConjunctionContinuation,
|
||
(conjunction: (TokenConjunction)),
|
||
(newlines: (MaybeNewlines)),
|
||
(job: (JobPipeline)),
|
||
);
|
||
impl ConcreteNode for JobConjunctionContinuation {
|
||
fn as_job_conjunction_continuation(&self) -> Option<&JobConjunctionContinuation> {
|
||
Some(self)
|
||
}
|
||
}
|
||
impl ConcreteNodeMut for JobConjunctionContinuation {
|
||
fn as_mut_job_conjunction_continuation(&mut self) -> Option<&mut JobConjunctionContinuation> {
|
||
Some(self)
|
||
}
|
||
}
|
||
|
||
/// An andor_job just wraps a job, but requires that the job have an 'and' or 'or' job_decorator.
|
||
/// Note this is only used for andor_job_list; jobs that are not part of an andor_job_list are not
|
||
/// instances of this.
|
||
#[derive(Default, Debug)]
|
||
pub struct AndorJob {
|
||
parent: Option<*const dyn Node>,
|
||
pub job: JobConjunction,
|
||
}
|
||
implement_node!(AndorJob, branch, andor_job);
|
||
implement_acceptor_for_branch!(AndorJob, (job: (JobConjunction)));
|
||
impl ConcreteNode for AndorJob {
|
||
fn as_andor_job(&self) -> Option<&AndorJob> {
|
||
Some(self)
|
||
}
|
||
}
|
||
impl ConcreteNodeMut for AndorJob {
|
||
fn as_mut_andor_job(&mut self) -> Option<&mut AndorJob> {
|
||
Some(self)
|
||
}
|
||
}
|
||
|
||
define_list_node!(AndorJobList, andor_job_list, AndorJob);
|
||
impl ConcreteNode for AndorJobList {
|
||
fn as_andor_job_list(&self) -> Option<&AndorJobList> {
|
||
Some(self)
|
||
}
|
||
}
|
||
impl ConcreteNodeMut for AndorJobList {
|
||
fn as_mut_andor_job_list(&mut self) -> Option<&mut AndorJobList> {
|
||
Some(self)
|
||
}
|
||
}
|
||
|
||
/// A freestanding_argument_list is equivalent to a normal argument list, except it may contain
|
||
/// TOK_END (newlines, and even semicolons, for historical reasons).
|
||
/// In practice the tok_ends are ignored by fish code so we do not bother to store them.
|
||
#[derive(Default, Debug)]
|
||
pub struct FreestandingArgumentList {
|
||
parent: Option<*const dyn Node>,
|
||
pub arguments: ArgumentList,
|
||
}
|
||
implement_node!(FreestandingArgumentList, branch, freestanding_argument_list);
|
||
implement_acceptor_for_branch!(FreestandingArgumentList, (arguments: (ArgumentList)));
|
||
impl ConcreteNode for FreestandingArgumentList {
|
||
fn as_freestanding_argument_list(&self) -> Option<&FreestandingArgumentList> {
|
||
Some(self)
|
||
}
|
||
}
|
||
impl ConcreteNodeMut for FreestandingArgumentList {
|
||
fn as_mut_freestanding_argument_list(&mut self) -> Option<&mut FreestandingArgumentList> {
|
||
Some(self)
|
||
}
|
||
}
|
||
|
||
define_list_node!(
|
||
JobConjunctionContinuationList,
|
||
job_conjunction_continuation_list,
|
||
JobConjunctionContinuation
|
||
);
|
||
impl ConcreteNode for JobConjunctionContinuationList {
|
||
fn as_job_conjunction_continuation_list(&self) -> Option<&JobConjunctionContinuationList> {
|
||
Some(self)
|
||
}
|
||
}
|
||
impl ConcreteNodeMut for JobConjunctionContinuationList {
|
||
fn as_mut_job_conjunction_continuation_list(
|
||
&mut self,
|
||
) -> Option<&mut JobConjunctionContinuationList> {
|
||
Some(self)
|
||
}
|
||
}
|
||
|
||
define_list_node!(ArgumentList, argument_list, Argument);
|
||
impl ConcreteNode for ArgumentList {
|
||
fn as_argument_list(&self) -> Option<&ArgumentList> {
|
||
Some(self)
|
||
}
|
||
}
|
||
impl ConcreteNodeMut for ArgumentList {
|
||
fn as_mut_argument_list(&mut self) -> Option<&mut ArgumentList> {
|
||
Some(self)
|
||
}
|
||
}
|
||
|
||
// For historical reasons, a job list is a list of job *conjunctions*. This should be fixed.
|
||
define_list_node!(JobList, job_list, JobConjunction);
|
||
impl ConcreteNode for JobList {
|
||
fn as_job_list(&self) -> Option<&JobList> {
|
||
Some(self)
|
||
}
|
||
}
|
||
impl ConcreteNodeMut for JobList {
|
||
fn as_mut_job_list(&mut self) -> Option<&mut JobList> {
|
||
Some(self)
|
||
}
|
||
}
|
||
|
||
define_list_node!(CaseItemList, case_item_list, CaseItem);
|
||
impl ConcreteNode for CaseItemList {
|
||
fn as_case_item_list(&self) -> Option<&CaseItemList> {
|
||
Some(self)
|
||
}
|
||
}
|
||
impl ConcreteNodeMut for CaseItemList {
|
||
fn as_mut_case_item_list(&mut self) -> Option<&mut CaseItemList> {
|
||
Some(self)
|
||
}
|
||
}
|
||
|
||
/// A variable_assignment contains a source range like FOO=bar.
|
||
#[derive(Default, Debug)]
|
||
pub struct VariableAssignment {
|
||
parent: Option<*const dyn Node>,
|
||
range: Option<SourceRange>,
|
||
}
|
||
implement_node!(VariableAssignment, leaf, variable_assignment);
|
||
implement_leaf!(VariableAssignment);
|
||
impl ConcreteNode for VariableAssignment {
|
||
fn as_leaf(&self) -> Option<&dyn Leaf> {
|
||
Some(self)
|
||
}
|
||
fn as_variable_assignment(&self) -> Option<&VariableAssignment> {
|
||
Some(self)
|
||
}
|
||
}
|
||
impl ConcreteNodeMut for VariableAssignment {
|
||
fn as_mut_variable_assignment(&mut self) -> Option<&mut VariableAssignment> {
|
||
Some(self)
|
||
}
|
||
}
|
||
|
||
/// Zero or more newlines.
|
||
#[derive(Default, Debug)]
|
||
pub struct MaybeNewlines {
|
||
parent: Option<*const dyn Node>,
|
||
range: Option<SourceRange>,
|
||
}
|
||
implement_node!(MaybeNewlines, leaf, maybe_newlines);
|
||
implement_leaf!(MaybeNewlines);
|
||
impl ConcreteNode for MaybeNewlines {
|
||
fn as_leaf(&self) -> Option<&dyn Leaf> {
|
||
Some(self)
|
||
}
|
||
fn as_maybe_newlines(&self) -> Option<&MaybeNewlines> {
|
||
Some(self)
|
||
}
|
||
}
|
||
impl ConcreteNodeMut for MaybeNewlines {
|
||
fn as_mut_leaf(&mut self) -> Option<&mut dyn Leaf> {
|
||
Some(self)
|
||
}
|
||
fn as_mut_maybe_newlines(&mut self) -> Option<&mut MaybeNewlines> {
|
||
Some(self)
|
||
}
|
||
}
|
||
|
||
/// An argument is just a node whose source range determines its contents.
|
||
/// This is a separate type because it is sometimes useful to find all arguments.
|
||
#[derive(Default, Debug)]
|
||
pub struct Argument {
|
||
parent: Option<*const dyn Node>,
|
||
range: Option<SourceRange>,
|
||
}
|
||
implement_node!(Argument, leaf, argument);
|
||
implement_leaf!(Argument);
|
||
impl ConcreteNode for Argument {
|
||
fn as_leaf(&self) -> Option<&dyn Leaf> {
|
||
Some(self)
|
||
}
|
||
fn as_argument(&self) -> Option<&Argument> {
|
||
Some(self)
|
||
}
|
||
}
|
||
impl ConcreteNodeMut for Argument {
|
||
fn as_mut_leaf(&mut self) -> Option<&mut dyn Leaf> {
|
||
Some(self)
|
||
}
|
||
fn as_mut_argument(&mut self) -> Option<&mut Argument> {
|
||
Some(self)
|
||
}
|
||
}
|
||
|
||
define_token_node!(SemiNl, ParseTokenType::end);
|
||
define_token_node!(String_, ParseTokenType::string);
|
||
define_token_node!(TokenBackground, ParseTokenType::background);
|
||
#[rustfmt::skip]
|
||
define_token_node!(TokenConjunction, ParseTokenType::andand, ParseTokenType::oror);
|
||
define_token_node!(TokenPipe, ParseTokenType::pipe);
|
||
define_token_node!(TokenRedirection, ParseTokenType::redirection);
|
||
|
||
#[rustfmt::skip]
|
||
define_keyword_node!(DecoratedStatementDecorator, ParseKeyword::kw_command, ParseKeyword::kw_builtin, ParseKeyword::kw_exec);
|
||
#[rustfmt::skip]
|
||
define_keyword_node!(JobConjunctionDecorator, ParseKeyword::kw_and, ParseKeyword::kw_or);
|
||
#[rustfmt::skip]
|
||
define_keyword_node!(KeywordBegin, ParseKeyword::kw_begin);
|
||
define_keyword_node!(KeywordCase, ParseKeyword::kw_case);
|
||
define_keyword_node!(KeywordElse, ParseKeyword::kw_else);
|
||
define_keyword_node!(KeywordEnd, ParseKeyword::kw_end);
|
||
define_keyword_node!(KeywordFor, ParseKeyword::kw_for);
|
||
define_keyword_node!(KeywordFunction, ParseKeyword::kw_function);
|
||
define_keyword_node!(KeywordIf, ParseKeyword::kw_if);
|
||
define_keyword_node!(KeywordIn, ParseKeyword::kw_in);
|
||
#[rustfmt::skip]
|
||
define_keyword_node!(KeywordNot, ParseKeyword::kw_not, ParseKeyword::kw_builtin, ParseKeyword::kw_exclam);
|
||
define_keyword_node!(KeywordSwitch, ParseKeyword::kw_switch);
|
||
define_keyword_node!(KeywordTime, ParseKeyword::kw_time);
|
||
define_keyword_node!(KeywordWhile, ParseKeyword::kw_while);
|
||
|
||
impl DecoratedStatement {
|
||
/// \return the decoration for this statement.
|
||
fn decoration(&self) -> StatementDecoration {
|
||
let Some(decorator) = &self.opt_decoration else {
|
||
return StatementDecoration::none;
|
||
};
|
||
let decorator: &dyn Keyword = decorator;
|
||
match decorator.keyword() {
|
||
ParseKeyword::kw_command => StatementDecoration::command,
|
||
ParseKeyword::kw_builtin => StatementDecoration::builtin,
|
||
ParseKeyword::kw_exec => StatementDecoration::exec,
|
||
_ => panic!("Unexpected keyword in statement decoration"),
|
||
}
|
||
}
|
||
}
|
||
|
||
#[derive(Debug)]
|
||
pub enum ArgumentOrRedirectionVariant {
|
||
Argument(Argument),
|
||
Redirection(Redirection),
|
||
}
|
||
|
||
impl Default for ArgumentOrRedirectionVariant {
|
||
fn default() -> Self {
|
||
ArgumentOrRedirectionVariant::Argument(Argument::default())
|
||
}
|
||
}
|
||
|
||
impl Acceptor for ArgumentOrRedirectionVariant {
|
||
fn accept<'a>(&'a self, visitor: &mut dyn NodeVisitor<'a>, reversed: bool) {
|
||
match self {
|
||
ArgumentOrRedirectionVariant::Argument(child) => child.accept(visitor, reversed),
|
||
ArgumentOrRedirectionVariant::Redirection(child) => child.accept(visitor, reversed),
|
||
}
|
||
}
|
||
}
|
||
impl AcceptorMut for ArgumentOrRedirectionVariant {
|
||
fn accept_mut(&mut self, visitor: &mut dyn NodeVisitorMut, reversed: bool) {
|
||
match self {
|
||
ArgumentOrRedirectionVariant::Argument(child) => child.accept_mut(visitor, reversed),
|
||
ArgumentOrRedirectionVariant::Redirection(child) => child.accept_mut(visitor, reversed),
|
||
}
|
||
}
|
||
}
|
||
|
||
impl ArgumentOrRedirectionVariant {
|
||
fn embedded_node(&self) -> &dyn NodeMut {
|
||
match self {
|
||
ArgumentOrRedirectionVariant::Argument(node) => node,
|
||
ArgumentOrRedirectionVariant::Redirection(node) => node,
|
||
}
|
||
}
|
||
fn as_mut_argument(&mut self) -> &mut Argument {
|
||
match self {
|
||
ArgumentOrRedirectionVariant::Argument(node) => node,
|
||
_ => panic!(),
|
||
}
|
||
}
|
||
fn as_mut_redirection(&mut self) -> &mut Redirection {
|
||
match self {
|
||
ArgumentOrRedirectionVariant::Redirection(redirection) => redirection,
|
||
_ => panic!(),
|
||
}
|
||
}
|
||
}
|
||
|
||
impl ArgumentOrRedirection {
|
||
/// \return whether this represents an argument.
|
||
pub fn is_argument(&self) -> bool {
|
||
matches!(*self.contents, ArgumentOrRedirectionVariant::Argument(_))
|
||
}
|
||
|
||
/// \return whether this represents a redirection
|
||
pub fn is_redirection(&self) -> bool {
|
||
matches!(*self.contents, ArgumentOrRedirectionVariant::Redirection(_))
|
||
}
|
||
|
||
/// \return this as an argument, assuming it wraps one.
|
||
pub fn argument(&self) -> &Argument {
|
||
match *self.contents {
|
||
ArgumentOrRedirectionVariant::Argument(ref arg) => arg,
|
||
_ => panic!("Is not an argument"),
|
||
}
|
||
}
|
||
|
||
/// \return this as an argument, assuming it wraps one.
|
||
pub fn redirection(&self) -> &Redirection {
|
||
match *self.contents {
|
||
ArgumentOrRedirectionVariant::Redirection(ref arg) => arg,
|
||
_ => panic!("Is not a redirection"),
|
||
}
|
||
}
|
||
}
|
||
|
||
#[derive(Debug)]
|
||
pub enum StatementVariant {
|
||
None,
|
||
NotStatement(NotStatement),
|
||
BlockStatement(BlockStatement),
|
||
IfStatement(IfStatement),
|
||
SwitchStatement(SwitchStatement),
|
||
DecoratedStatement(DecoratedStatement),
|
||
}
|
||
|
||
impl Default for StatementVariant {
|
||
fn default() -> Self {
|
||
StatementVariant::None
|
||
}
|
||
}
|
||
|
||
impl Acceptor for StatementVariant {
|
||
fn accept<'a>(&'a self, visitor: &mut dyn NodeVisitor<'a>, reversed: bool) {
|
||
match self {
|
||
StatementVariant::None => panic!("cannot visit null statement"),
|
||
StatementVariant::NotStatement(node) => node.accept(visitor, reversed),
|
||
StatementVariant::BlockStatement(node) => node.accept(visitor, reversed),
|
||
StatementVariant::IfStatement(node) => node.accept(visitor, reversed),
|
||
StatementVariant::SwitchStatement(node) => node.accept(visitor, reversed),
|
||
StatementVariant::DecoratedStatement(node) => node.accept(visitor, reversed),
|
||
}
|
||
}
|
||
}
|
||
impl AcceptorMut for StatementVariant {
|
||
fn accept_mut(&mut self, visitor: &mut dyn NodeVisitorMut, reversed: bool) {
|
||
match self {
|
||
StatementVariant::None => panic!("cannot visit null statement"),
|
||
StatementVariant::NotStatement(node) => node.accept_mut(visitor, reversed),
|
||
StatementVariant::BlockStatement(node) => node.accept_mut(visitor, reversed),
|
||
StatementVariant::IfStatement(node) => node.accept_mut(visitor, reversed),
|
||
StatementVariant::SwitchStatement(node) => node.accept_mut(visitor, reversed),
|
||
StatementVariant::DecoratedStatement(node) => node.accept_mut(visitor, reversed),
|
||
}
|
||
}
|
||
}
|
||
|
||
impl StatementVariant {
|
||
fn embedded_node(&self) -> &dyn NodeMut {
|
||
match self {
|
||
StatementVariant::None => panic!("cannot visit null statement"),
|
||
StatementVariant::NotStatement(node) => node,
|
||
StatementVariant::BlockStatement(node) => node,
|
||
StatementVariant::IfStatement(node) => node,
|
||
StatementVariant::SwitchStatement(node) => node,
|
||
StatementVariant::DecoratedStatement(node) => node,
|
||
}
|
||
}
|
||
fn as_mut_not_statement(&mut self) -> &mut NotStatement {
|
||
match self {
|
||
StatementVariant::NotStatement(node) => node,
|
||
_ => panic!(),
|
||
}
|
||
}
|
||
fn as_mut_block_statement(&mut self) -> &mut BlockStatement {
|
||
match self {
|
||
StatementVariant::BlockStatement(node) => node,
|
||
_ => panic!(),
|
||
}
|
||
}
|
||
fn as_mut_if_statement(&mut self) -> &mut IfStatement {
|
||
match self {
|
||
StatementVariant::IfStatement(node) => node,
|
||
_ => panic!(),
|
||
}
|
||
}
|
||
fn as_mut_switch_statement(&mut self) -> &mut SwitchStatement {
|
||
match self {
|
||
StatementVariant::SwitchStatement(node) => node,
|
||
_ => panic!(),
|
||
}
|
||
}
|
||
fn as_mut_decorated_statement(&mut self) -> &mut DecoratedStatement {
|
||
match self {
|
||
StatementVariant::DecoratedStatement(node) => node,
|
||
_ => panic!(),
|
||
}
|
||
}
|
||
}
|
||
|
||
#[derive(Debug)]
|
||
pub enum BlockStatementHeaderVariant {
|
||
None,
|
||
ForHeader(ForHeader),
|
||
WhileHeader(WhileHeader),
|
||
FunctionHeader(FunctionHeader),
|
||
BeginHeader(BeginHeader),
|
||
}
|
||
|
||
impl Default for BlockStatementHeaderVariant {
|
||
fn default() -> Self {
|
||
BlockStatementHeaderVariant::None
|
||
}
|
||
}
|
||
|
||
impl Acceptor for BlockStatementHeaderVariant {
|
||
fn accept<'a>(&'a self, visitor: &mut dyn NodeVisitor<'a>, reversed: bool) {
|
||
match self {
|
||
BlockStatementHeaderVariant::None => panic!("cannot visit null block header"),
|
||
BlockStatementHeaderVariant::ForHeader(node) => node.accept(visitor, reversed),
|
||
BlockStatementHeaderVariant::WhileHeader(node) => node.accept(visitor, reversed),
|
||
BlockStatementHeaderVariant::FunctionHeader(node) => node.accept(visitor, reversed),
|
||
BlockStatementHeaderVariant::BeginHeader(node) => node.accept(visitor, reversed),
|
||
}
|
||
}
|
||
}
|
||
impl AcceptorMut for BlockStatementHeaderVariant {
|
||
fn accept_mut(&mut self, visitor: &mut dyn NodeVisitorMut, reversed: bool) {
|
||
match self {
|
||
BlockStatementHeaderVariant::None => panic!("cannot visit null block header"),
|
||
BlockStatementHeaderVariant::ForHeader(node) => node.accept_mut(visitor, reversed),
|
||
BlockStatementHeaderVariant::WhileHeader(node) => node.accept_mut(visitor, reversed),
|
||
BlockStatementHeaderVariant::FunctionHeader(node) => node.accept_mut(visitor, reversed),
|
||
BlockStatementHeaderVariant::BeginHeader(node) => node.accept_mut(visitor, reversed),
|
||
}
|
||
}
|
||
}
|
||
|
||
impl BlockStatementHeaderVariant {
|
||
fn embedded_node(&self) -> &dyn NodeMut {
|
||
match self {
|
||
BlockStatementHeaderVariant::None => panic!("cannot visit null block header"),
|
||
BlockStatementHeaderVariant::ForHeader(node) => node,
|
||
BlockStatementHeaderVariant::WhileHeader(node) => node,
|
||
BlockStatementHeaderVariant::FunctionHeader(node) => node,
|
||
BlockStatementHeaderVariant::BeginHeader(node) => node,
|
||
}
|
||
}
|
||
fn as_mut_for_header(&mut self) -> &mut ForHeader {
|
||
match self {
|
||
BlockStatementHeaderVariant::ForHeader(node) => node,
|
||
_ => panic!(),
|
||
}
|
||
}
|
||
fn as_mut_while_header(&mut self) -> &mut WhileHeader {
|
||
match self {
|
||
BlockStatementHeaderVariant::WhileHeader(node) => node,
|
||
_ => panic!(),
|
||
}
|
||
}
|
||
fn as_mut_function_header(&mut self) -> &mut FunctionHeader {
|
||
match self {
|
||
BlockStatementHeaderVariant::FunctionHeader(node) => node,
|
||
_ => panic!(),
|
||
}
|
||
}
|
||
fn as_mut_begin_header(&mut self) -> &mut BeginHeader {
|
||
match self {
|
||
BlockStatementHeaderVariant::BeginHeader(node) => node,
|
||
_ => panic!(),
|
||
}
|
||
}
|
||
}
|
||
|
||
/// \return a string literal name for an ast type.
|
||
#[widestrs]
|
||
pub fn ast_type_to_string(t: Type) -> &'static wstr {
|
||
match t {
|
||
Type::token_base => "token_base"L,
|
||
Type::keyword_base => "keyword_base"L,
|
||
Type::redirection => "redirection"L,
|
||
Type::variable_assignment => "variable_assignment"L,
|
||
Type::variable_assignment_list => "variable_assignment_list"L,
|
||
Type::argument_or_redirection => "argument_or_redirection"L,
|
||
Type::argument_or_redirection_list => "argument_or_redirection_list"L,
|
||
Type::statement => "statement"L,
|
||
Type::job_pipeline => "job_pipeline"L,
|
||
Type::job_conjunction => "job_conjunction"L,
|
||
Type::for_header => "for_header"L,
|
||
Type::while_header => "while_header"L,
|
||
Type::function_header => "function_header"L,
|
||
Type::begin_header => "begin_header"L,
|
||
Type::block_statement => "block_statement"L,
|
||
Type::if_clause => "if_clause"L,
|
||
Type::elseif_clause => "elseif_clause"L,
|
||
Type::elseif_clause_list => "elseif_clause_list"L,
|
||
Type::else_clause => "else_clause"L,
|
||
Type::if_statement => "if_statement"L,
|
||
Type::case_item => "case_item"L,
|
||
Type::switch_statement => "switch_statement"L,
|
||
Type::decorated_statement => "decorated_statement"L,
|
||
Type::not_statement => "not_statement"L,
|
||
Type::job_continuation => "job_continuation"L,
|
||
Type::job_continuation_list => "job_continuation_list"L,
|
||
Type::job_conjunction_continuation => "job_conjunction_continuation"L,
|
||
Type::andor_job => "andor_job"L,
|
||
Type::andor_job_list => "andor_job_list"L,
|
||
Type::freestanding_argument_list => "freestanding_argument_list"L,
|
||
Type::token_conjunction => "token_conjunction"L,
|
||
Type::job_conjunction_continuation_list => "job_conjunction_continuation_list"L,
|
||
Type::maybe_newlines => "maybe_newlines"L,
|
||
Type::token_pipe => "token_pipe"L,
|
||
Type::case_item_list => "case_item_list"L,
|
||
Type::argument => "argument"L,
|
||
Type::argument_list => "argument_list"L,
|
||
Type::job_list => "job_list"L,
|
||
_ => panic!("unknown AST type"),
|
||
}
|
||
}
|
||
|
||
// A way to visit nodes iteratively.
|
||
// This is pre-order. Each node is visited before its children.
|
||
// Example:
|
||
// let tv = Traversal::new(start);
|
||
// while let Some(node) = tv.next() {...}
|
||
pub struct Traversal<'a> {
|
||
stack: Vec<&'a dyn Node>,
|
||
}
|
||
|
||
impl<'a> Traversal<'a> {
|
||
// Construct starting with a node
|
||
pub fn new(n: &'a dyn Node) -> Self {
|
||
Self { stack: vec![n] }
|
||
}
|
||
}
|
||
|
||
impl<'a> Iterator for Traversal<'a> {
|
||
type Item = &'a dyn Node;
|
||
fn next(&mut self) -> Option<&'a dyn Node> {
|
||
let Some(node) = self.stack.pop() else {
|
||
return None;
|
||
};
|
||
// We want to visit in reverse order so the first child ends up on top of the stack.
|
||
node.accept(self, true /* reverse */);
|
||
Some(node)
|
||
}
|
||
}
|
||
|
||
impl<'a, 'v: 'a> NodeVisitor<'v> for Traversal<'a> {
|
||
fn visit(&mut self, node: &'a dyn Node) {
|
||
self.stack.push(node)
|
||
}
|
||
}
|
||
|
||
fn ast_type_to_string_ffi(typ: Type) -> wcharz_t {
|
||
wcharz!(ast_type_to_string(typ))
|
||
}
|
||
|
||
pub type SourceRangeList = Vec<SourceRange>;
|
||
|
||
/// Extra source ranges.
|
||
/// These are only generated if the corresponding flags are set.
|
||
#[derive(Default)]
|
||
pub struct Extras {
|
||
/// Set of comments, sorted by offset.
|
||
pub comments: SourceRangeList,
|
||
|
||
/// Set of semicolons, sorted by offset.
|
||
pub semis: SourceRangeList,
|
||
|
||
/// Set of error ranges, sorted by offset.
|
||
pub errors: SourceRangeList,
|
||
}
|
||
|
||
/// The ast type itself.
|
||
pub struct Ast {
|
||
// The top node.
|
||
// Its type depends on what was requested to parse.
|
||
top: Box<dyn NodeMut>,
|
||
/// Whether any errors were encountered during parsing.
|
||
pub any_error: bool,
|
||
/// Extra fields.
|
||
pub extras: Extras,
|
||
}
|
||
|
||
#[allow(clippy::derivable_impls)] // false positive
|
||
impl Default for Ast {
|
||
fn default() -> Ast {
|
||
Self {
|
||
top: Box::<String_>::default(),
|
||
any_error: false,
|
||
extras: Extras::default(),
|
||
}
|
||
}
|
||
}
|
||
|
||
impl Ast {
|
||
/// Construct an ast by parsing \p src as a job list.
|
||
/// The ast attempts to produce \p type as the result.
|
||
/// \p type may only be JobList or FreestandingArgumentList.
|
||
pub fn parse(
|
||
src: &wstr,
|
||
flags: ParseTreeFlags,
|
||
out_errors: &mut Option<ParseErrorList>,
|
||
) -> Self {
|
||
parse_from_top(src, flags, out_errors, Type::job_list)
|
||
}
|
||
/// Like parse(), but constructs a freestanding_argument_list.
|
||
pub fn parse_argument_list(
|
||
src: &wstr,
|
||
flags: ParseTreeFlags,
|
||
out_errors: &mut Option<ParseErrorList>,
|
||
) -> Self {
|
||
parse_from_top(src, flags, out_errors, Type::freestanding_argument_list)
|
||
}
|
||
/// \return a traversal, allowing iteration over the nodes.
|
||
pub fn walk(&'_ self) -> Traversal<'_> {
|
||
Traversal::new(self.top.as_node())
|
||
}
|
||
/// \return the top node. This has the type requested in the 'parse' method.
|
||
pub fn top(&self) -> &dyn Node {
|
||
self.top.as_node()
|
||
}
|
||
fn top_mut(&mut self) -> &mut dyn NodeMut {
|
||
&mut *self.top
|
||
}
|
||
/// \return whether any errors were encountered during parsing.
|
||
pub fn errored(&self) -> bool {
|
||
self.any_error
|
||
}
|
||
/// \return a textual representation of the tree.
|
||
/// Pass the original source as \p orig.
|
||
#[widestrs]
|
||
fn dump(&self, orig: &wstr) -> WString {
|
||
let mut result = WString::new();
|
||
|
||
let mut tv = self.walk();
|
||
while let Some(node) = tv.next() {
|
||
let depth = get_depth(node);
|
||
// dot-| padding
|
||
result += &wstr::repeat("! "L, depth)[..];
|
||
|
||
if let Some(n) = node.as_argument() {
|
||
result += "argument"L;
|
||
if let Some(argsrc) = n.try_source(orig) {
|
||
result += &sprintf!(": '%ls'"L, argsrc)[..];
|
||
}
|
||
} else if let Some(n) = node.as_keyword() {
|
||
result += &sprintf!("keyword: %ls"L, Into::<&'static wstr>::into(n.keyword()))[..];
|
||
} else if let Some(n) = node.as_token() {
|
||
let desc = match n.token_type() {
|
||
ParseTokenType::string => {
|
||
let mut desc = "string"L.to_owned();
|
||
if let Some(strsource) = n.try_source(orig) {
|
||
desc += &sprintf!(": '%ls'"L, strsource)[..];
|
||
}
|
||
desc
|
||
}
|
||
ParseTokenType::redirection => {
|
||
let mut desc = "redirection"L.to_owned();
|
||
if let Some(strsource) = n.try_source(orig) {
|
||
desc += &sprintf!(": '%ls'"L, strsource)[..];
|
||
}
|
||
desc
|
||
}
|
||
ParseTokenType::end => "<;>"L.to_owned(),
|
||
ParseTokenType::invalid => {
|
||
// This may occur with errors, e.g. we expected to see a string but saw a
|
||
// redirection.
|
||
"<error>"L.to_owned()
|
||
}
|
||
_ => {
|
||
token_type_user_presentable_description(n.token_type(), ParseKeyword::none)
|
||
}
|
||
};
|
||
result += &desc[..];
|
||
} else {
|
||
result += &node.describe()[..];
|
||
}
|
||
result += "\n"L;
|
||
}
|
||
result
|
||
}
|
||
}
|
||
|
||
// \return the depth of a node, i.e. number of parent links.
|
||
fn get_depth(node: &dyn Node) -> usize {
|
||
let mut result = 0;
|
||
let mut cursor = node;
|
||
loop {
|
||
cursor = match cursor.parent() {
|
||
Some(parent) => parent,
|
||
None => return result,
|
||
};
|
||
result += 1;
|
||
}
|
||
}
|
||
|
||
struct SourceRangeVisitor {
|
||
/// Total range we have encountered.
|
||
total: SourceRange,
|
||
/// Whether any node was found to be unsourced.
|
||
any_unsourced: bool,
|
||
}
|
||
|
||
impl<'a> NodeVisitor<'a> for SourceRangeVisitor {
|
||
fn visit(&mut self, node: &'a dyn Node) {
|
||
match node.category() {
|
||
Category::leaf => match node.as_leaf().unwrap().range() {
|
||
None => self.any_unsourced = true,
|
||
// Union with our range.
|
||
Some(range) if range.length > 0 => {
|
||
if self.total.length == 0 {
|
||
self.total = range;
|
||
} else {
|
||
let end =
|
||
(self.total.start + self.total.length).max(range.start + range.length);
|
||
self.total.start = self.total.start.min(range.start);
|
||
self.total.length = end - self.total.start;
|
||
}
|
||
}
|
||
_ => (),
|
||
},
|
||
_ => {
|
||
// Other node types recurse.
|
||
node.accept(self, false);
|
||
}
|
||
}
|
||
}
|
||
}
|
||
|
||
/// A token stream generates a sequence of parser tokens, permitting arbitrary lookahead.
|
||
struct TokenStream<'a> {
|
||
// We implement a queue with a simple circular buffer.
|
||
// Note that peek() returns an address, so we must not move elements which are peek'd.
|
||
// This prevents using vector (which may reallocate).
|
||
// Deque would work but is too heavyweight for just 2 items.
|
||
lookahead: [ParseToken; TokenStream::MAX_LOOKAHEAD],
|
||
|
||
// Starting index in our lookahead.
|
||
// The "first" token is at this index.
|
||
start: usize,
|
||
|
||
// Number of items in our lookahead.
|
||
count: usize,
|
||
|
||
// A reference to the original source.
|
||
src: &'a wstr,
|
||
|
||
// The tokenizer to generate new tokens.
|
||
tok: Tokenizer,
|
||
|
||
/// Any comment nodes are collected here.
|
||
/// These are only collected if parse_flag_include_comments is set.
|
||
comment_ranges: SourceRangeList,
|
||
}
|
||
|
||
impl<'a> TokenStream<'a> {
|
||
// The maximum number of lookahead supported.
|
||
const MAX_LOOKAHEAD: usize = 2;
|
||
|
||
fn new(src: &'a wstr, flags: ParseTreeFlags) -> Self {
|
||
Self {
|
||
lookahead: [ParseToken::new(ParseTokenType::invalid); Self::MAX_LOOKAHEAD],
|
||
start: 0,
|
||
count: 0,
|
||
src,
|
||
tok: Tokenizer::new(src, TokFlags::from(flags)),
|
||
comment_ranges: vec![],
|
||
}
|
||
}
|
||
|
||
/// \return the token at the given index, without popping it. If the token stream is exhausted,
|
||
/// it will have parse_token_type_t::terminate. idx = 0 means the next token, idx = 1 means the
|
||
/// next-next token, and so forth.
|
||
/// We must have that idx < kMaxLookahead.
|
||
fn peek(&mut self, idx: usize) -> &ParseToken {
|
||
assert!(idx < Self::MAX_LOOKAHEAD, "Trying to look too far ahead");
|
||
while idx >= self.count {
|
||
self.lookahead[Self::mask(self.start + self.count)] = self.next_from_tok();
|
||
self.count += 1
|
||
}
|
||
&self.lookahead[Self::mask(self.start + idx)]
|
||
}
|
||
|
||
/// Pop the next token.
|
||
fn pop(&mut self) -> ParseToken {
|
||
if self.count == 0 {
|
||
return self.next_from_tok();
|
||
}
|
||
let result = self.lookahead[self.start];
|
||
self.start = Self::mask(self.start + 1);
|
||
self.count -= 1;
|
||
result
|
||
}
|
||
|
||
// Helper to mask our circular buffer.
|
||
fn mask(idx: usize) -> usize {
|
||
idx % Self::MAX_LOOKAHEAD
|
||
}
|
||
|
||
/// \return the next parse token from the tokenizer.
|
||
/// This consumes and stores comments.
|
||
fn next_from_tok(&mut self) -> ParseToken {
|
||
loop {
|
||
let res = self.advance_1();
|
||
if res.typ == ParseTokenType::comment {
|
||
self.comment_ranges.push(res.range());
|
||
continue;
|
||
}
|
||
return res;
|
||
}
|
||
}
|
||
|
||
/// \return a new parse token, advancing the tokenizer.
|
||
/// This returns comments.
|
||
fn advance_1(&mut self) -> ParseToken {
|
||
let Some(token) = self.tok.next() else {
|
||
return ParseToken::new(ParseTokenType::terminate);
|
||
};
|
||
// Set the type, keyword, and whether there's a dash prefix. Note that this is quite
|
||
// sketchy, because it ignores quotes. This is the historical behavior. For example,
|
||
// `builtin --names` lists builtins, but `builtin "--names"` attempts to run --names as a
|
||
// command. Amazingly as of this writing (10/12/13) nobody seems to have noticed this.
|
||
// Squint at it really hard and it even starts to look like a feature.
|
||
let mut result = ParseToken::new(ParseTokenType::from(token.type_));
|
||
let text = self.tok.text_of(&token);
|
||
result.keyword = keyword_for_token(token.type_, text);
|
||
result.has_dash_prefix = text.starts_with('-');
|
||
result.is_help_argument = [L!("-h"), L!("--help")].contains(&text);
|
||
result.is_newline = result.typ == ParseTokenType::end && text == L!("\n");
|
||
result.may_be_variable_assignment = variable_assignment_equals_pos(text).is_some();
|
||
result.tok_error = token.error;
|
||
|
||
assert!(token.offset < SOURCE_OFFSET_INVALID);
|
||
result.source_start = token.offset;
|
||
result.source_length = token.length;
|
||
|
||
if token.error != TokenizerError::none {
|
||
let subtoken_offset = token.error_offset_within_token;
|
||
// Skip invalid tokens that have a zero length, especially if they are at EOF.
|
||
if subtoken_offset < result.source_length {
|
||
result.source_start += subtoken_offset;
|
||
result.source_length = token.error_length;
|
||
}
|
||
}
|
||
|
||
result
|
||
}
|
||
}
|
||
|
||
/// This indicates a bug in fish code.
|
||
macro_rules! internal_error {
|
||
(
|
||
$self:ident,
|
||
$func:ident,
|
||
$fmt:expr
|
||
$(, $args:expr)*
|
||
$(,)?
|
||
) => {
|
||
FLOG!(
|
||
debug,
|
||
concat!(
|
||
"Internal parse error from {$func} - this indicates a bug in fish.",
|
||
$fmt,
|
||
)
|
||
$(, $args)*
|
||
);
|
||
FLOG!(debug, "Encountered while parsing:<<<<\n{}\n>>>", $self.tokens.src);
|
||
panic!();
|
||
};
|
||
}
|
||
|
||
/// Report an error based on \p fmt for the tokens' range
|
||
macro_rules! parse_error {
|
||
(
|
||
$self:ident,
|
||
$token:expr,
|
||
$code:expr,
|
||
$fmt:expr
|
||
$(, $args:expr)*
|
||
$(,)?
|
||
) => {
|
||
let range = $token.range();
|
||
parse_error_range!($self, range, $code, $fmt $(, $args)*);
|
||
}
|
||
}
|
||
|
||
/// Report an error based on \p fmt for the source range \p range.
|
||
macro_rules! parse_error_range {
|
||
(
|
||
$self:ident,
|
||
$range:expr,
|
||
$code:expr,
|
||
$fmt:expr
|
||
$(, $args:expr)*
|
||
$(,)?
|
||
) => {
|
||
let text = if $self.out_errors.is_some() && !$self.unwinding {
|
||
Some(wgettext_fmt!($fmt $(, $args)*))
|
||
} else {
|
||
None
|
||
};
|
||
$self.any_error = true;
|
||
|
||
// Ignore additional parse errors while unwinding.
|
||
// These may come about e.g. from `true | and`.
|
||
if !$self.unwinding {
|
||
$self.unwinding = true;
|
||
|
||
FLOG!(ast_construction, "%*sparse error - begin unwinding", $self.spaces(), "");
|
||
// TODO: can store this conditionally dependent on flags.
|
||
if $range.start != SOURCE_OFFSET_INVALID {
|
||
$self.errors.push($range);
|
||
}
|
||
|
||
if let Some(errors) = &mut $self.out_errors {
|
||
let mut err = ParseError::default();
|
||
err.text = text.unwrap();
|
||
err.code = $code;
|
||
err.source_start = $range.start as usize;
|
||
err.source_length = $range.length as usize;
|
||
errors.0.push(err);
|
||
}
|
||
}
|
||
}
|
||
}
|
||
|
||
struct Populator<'a> {
|
||
/// Flags controlling parsing.
|
||
flags: ParseTreeFlags,
|
||
|
||
/// Set of semicolons, sorted by offset.
|
||
semis: SourceRangeList,
|
||
|
||
/// Set of error ranges, sorted by offset.
|
||
errors: SourceRangeList,
|
||
|
||
/// Stream of tokens which we consume.
|
||
tokens: TokenStream<'a>,
|
||
|
||
/** The type which we are attempting to parse, typically job_list but may be
|
||
freestanding_argument_list. */
|
||
top_type: Type,
|
||
|
||
/// If set, we are unwinding due to error recovery.
|
||
unwinding: bool,
|
||
|
||
/// If set, we have encountered an error.
|
||
any_error: bool,
|
||
|
||
/// The number of parent links of the node we are visiting
|
||
depth: usize,
|
||
|
||
// If non-null, populate with errors.
|
||
out_errors: &'a mut Option<ParseErrorList>,
|
||
}
|
||
|
||
impl<'s> NodeVisitorMut for Populator<'s> {
|
||
fn visit_mut(&mut self, node: &mut dyn NodeMut) -> VisitResult {
|
||
match node.typ() {
|
||
Type::argument => {
|
||
self.visit_argument(node.as_mut_argument().unwrap());
|
||
return VisitResult::Continue(());
|
||
}
|
||
Type::variable_assignment => {
|
||
self.visit_variable_assignment(node.as_mut_variable_assignment().unwrap());
|
||
return VisitResult::Continue(());
|
||
}
|
||
Type::job_continuation => {
|
||
self.visit_job_continuation(node.as_mut_job_continuation().unwrap());
|
||
return VisitResult::Continue(());
|
||
}
|
||
Type::token_base => {
|
||
self.visit_token(node.as_mut_token().unwrap());
|
||
return VisitResult::Continue(());
|
||
}
|
||
Type::keyword_base => {
|
||
return self.visit_keyword(node.as_mut_keyword().unwrap());
|
||
}
|
||
Type::maybe_newlines => {
|
||
self.visit_maybe_newlines(node.as_mut_maybe_newlines().unwrap());
|
||
return VisitResult::Continue(());
|
||
}
|
||
|
||
_ => (),
|
||
}
|
||
|
||
match node.category() {
|
||
Category::leaf => {}
|
||
// Visit branch nodes by just calling accept() to visit their fields.
|
||
Category::branch => {
|
||
// This field is a direct embedding of an AST value.
|
||
node.accept_mut(self, false);
|
||
return VisitResult::Continue(());
|
||
}
|
||
Category::list => {
|
||
// This field is an embedding of an array of (pointers to) ContentsNode.
|
||
// Parse as many as we can.
|
||
match node.typ() {
|
||
Type::andor_job_list => self.populate_list::<AndorJobList>(
|
||
node.as_mut_andor_job_list().unwrap(),
|
||
false,
|
||
),
|
||
Type::argument_list => self
|
||
.populate_list::<ArgumentList>(node.as_mut_argument_list().unwrap(), false),
|
||
Type::argument_or_redirection_list => self
|
||
.populate_list::<ArgumentOrRedirectionList>(
|
||
node.as_mut_argument_or_redirection_list().unwrap(),
|
||
false,
|
||
),
|
||
Type::case_item_list => self.populate_list::<CaseItemList>(
|
||
node.as_mut_case_item_list().unwrap(),
|
||
false,
|
||
),
|
||
Type::elseif_clause_list => self.populate_list::<ElseifClauseList>(
|
||
node.as_mut_elseif_clause_list().unwrap(),
|
||
false,
|
||
),
|
||
Type::job_conjunction_continuation_list => self
|
||
.populate_list::<JobConjunctionContinuationList>(
|
||
node.as_mut_job_conjunction_continuation_list().unwrap(),
|
||
false,
|
||
),
|
||
Type::job_continuation_list => self.populate_list::<JobContinuationList>(
|
||
node.as_mut_job_continuation_list().unwrap(),
|
||
false,
|
||
),
|
||
Type::job_list => {
|
||
self.populate_list::<JobList>(node.as_mut_job_list().unwrap(), false)
|
||
}
|
||
Type::variable_assignment_list => self.populate_list::<VariableAssignmentList>(
|
||
node.as_mut_variable_assignment_list().unwrap(),
|
||
false,
|
||
),
|
||
_ => (),
|
||
}
|
||
}
|
||
_ => panic!(),
|
||
}
|
||
VisitResult::Continue(())
|
||
}
|
||
|
||
fn will_visit_fields_of(&mut self, node: &mut dyn NodeMut) {
|
||
FLOG!(
|
||
ast_construction,
|
||
"%*swill_visit %ls %p",
|
||
self.spaces(),
|
||
"",
|
||
node.describe()
|
||
);
|
||
self.depth += 1
|
||
}
|
||
|
||
#[widestrs]
|
||
fn did_visit_fields_of<'a>(&'a mut self, node: &'a dyn NodeMut, flow: VisitResult) {
|
||
self.depth -= 1;
|
||
|
||
if self.unwinding {
|
||
return;
|
||
}
|
||
let VisitResult::Break(error) = flow else { return; };
|
||
|
||
/// We believe the node is some sort of block statement. Attempt to find a source range
|
||
/// for the block's keyword (for, if, etc) and a user-presentable description. This
|
||
/// is used to provide better error messages. Note at this point the parse tree is
|
||
/// incomplete; in particular parent nodes are not set.
|
||
let mut cursor = node;
|
||
let header = loop {
|
||
match cursor.typ() {
|
||
Type::block_statement => {
|
||
cursor = cursor.as_block_statement().unwrap().header.embedded_node();
|
||
}
|
||
Type::for_header => {
|
||
let n = cursor.as_for_header().unwrap();
|
||
break Some((n.kw_for.range.unwrap(), "for loop"L));
|
||
}
|
||
Type::while_header => {
|
||
let n = cursor.as_while_header().unwrap();
|
||
break Some((n.kw_while.range.unwrap(), "while loop"L));
|
||
}
|
||
Type::function_header => {
|
||
let n = cursor.as_function_header().unwrap();
|
||
break Some((n.kw_function.range.unwrap(), "function definition"L));
|
||
}
|
||
Type::begin_header => {
|
||
let n = cursor.as_begin_header().unwrap();
|
||
break Some((n.kw_begin.range.unwrap(), "begin"L));
|
||
}
|
||
Type::if_statement => {
|
||
let n = cursor.as_if_statement().unwrap();
|
||
break Some((n.if_clause.kw_if.range.unwrap(), "if statement"L));
|
||
}
|
||
Type::switch_statement => {
|
||
let n = cursor.as_switch_statement().unwrap();
|
||
break Some((n.kw_switch.range.unwrap(), "switch statement"L));
|
||
}
|
||
_ => break None,
|
||
}
|
||
};
|
||
|
||
if let Some((header_kw_range, enclosing_stmt)) = header {
|
||
parse_error_range!(
|
||
self,
|
||
header_kw_range,
|
||
ParseErrorCode::generic,
|
||
"Missing end to balance this %ls",
|
||
enclosing_stmt
|
||
);
|
||
} else {
|
||
parse_error!(
|
||
self,
|
||
error.token,
|
||
ParseErrorCode::generic,
|
||
"Expected %ls, but found %ls",
|
||
keywords_user_presentable_description(error.allowed_keywords),
|
||
error.token.user_presentable_description(),
|
||
);
|
||
}
|
||
}
|
||
|
||
// We currently only have a handful of union pointer types.
|
||
// Handle them directly.
|
||
fn visit_argument_or_redirection(
|
||
&mut self,
|
||
node: &mut Box<ArgumentOrRedirectionVariant>,
|
||
) -> VisitResult {
|
||
if let Some(arg) = self.try_parse::<Argument>() {
|
||
**node = ArgumentOrRedirectionVariant::Argument(*arg);
|
||
} else if let Some(redir) = self.try_parse::<Redirection>() {
|
||
**node = ArgumentOrRedirectionVariant::Redirection(*redir);
|
||
} else {
|
||
internal_error!(
|
||
self,
|
||
visit_argument_or_redirection,
|
||
"Unable to parse argument or redirection"
|
||
);
|
||
}
|
||
VisitResult::Continue(())
|
||
}
|
||
fn visit_block_statement_header(
|
||
&mut self,
|
||
node: &mut Box<BlockStatementHeaderVariant>,
|
||
) -> VisitResult {
|
||
*node = self.allocate_populate_block_header();
|
||
VisitResult::Continue(())
|
||
}
|
||
fn visit_statement(&mut self, node: &mut Box<StatementVariant>) -> VisitResult {
|
||
*node = self.allocate_populate_statement_contents();
|
||
VisitResult::Continue(())
|
||
}
|
||
|
||
fn visit_decorated_statement_decorator(
|
||
&mut self,
|
||
node: &mut Option<DecoratedStatementDecorator>,
|
||
) {
|
||
*node = self.try_parse::<DecoratedStatementDecorator>().map(|b| *b);
|
||
}
|
||
fn visit_job_conjunction_decorator(&mut self, node: &mut Option<JobConjunctionDecorator>) {
|
||
*node = self.try_parse::<JobConjunctionDecorator>().map(|b| *b);
|
||
}
|
||
fn visit_else_clause(&mut self, node: &mut Option<ElseClause>) {
|
||
*node = self.try_parse::<ElseClause>().map(|b| *b);
|
||
}
|
||
fn visit_semi_nl(&mut self, node: &mut Option<SemiNl>) {
|
||
*node = self.try_parse::<SemiNl>().map(|b| *b);
|
||
}
|
||
fn visit_time(&mut self, node: &mut Option<KeywordTime>) {
|
||
*node = self.try_parse::<KeywordTime>().map(|b| *b);
|
||
}
|
||
fn visit_token_background(&mut self, node: &mut Option<TokenBackground>) {
|
||
*node = self.try_parse::<TokenBackground>().map(|b| *b);
|
||
}
|
||
}
|
||
|
||
/// Helper to describe a list of keywords.
|
||
/// TODO: these need to be localized properly.
|
||
#[widestrs]
|
||
fn keywords_user_presentable_description(kws: &'static [ParseKeyword]) -> WString {
|
||
assert!(!kws.is_empty(), "Should not be empty list");
|
||
if kws.len() == 1 {
|
||
return sprintf!("keyword '%ls'"L, kws[0]);
|
||
}
|
||
let mut res = "keywords "L.to_owned();
|
||
for (i, kw) in kws.iter().enumerate() {
|
||
if i != 0 {
|
||
res += " or "L;
|
||
}
|
||
res += &sprintf!("'%ls'"L, *kw)[..];
|
||
}
|
||
res
|
||
}
|
||
|
||
/// Helper to describe a list of token types.
|
||
/// TODO: these need to be localized properly.
|
||
#[widestrs]
|
||
fn token_types_user_presentable_description(types: &'static [ParseTokenType]) -> WString {
|
||
assert!(!types.is_empty(), "Should not be empty list");
|
||
let mut res = WString::new();
|
||
for typ in types {
|
||
if !res.is_empty() {
|
||
res += " or "L;
|
||
}
|
||
res += &token_type_user_presentable_description(*typ, ParseKeyword::none)[..];
|
||
}
|
||
res
|
||
}
|
||
|
||
impl<'s> Populator<'s> {
|
||
/// Construct from a source, flags, top type, and out_errors, which may be null.
|
||
fn new(
|
||
src: &'s wstr,
|
||
flags: ParseTreeFlags,
|
||
top_type: Type,
|
||
out_errors: &'s mut Option<ParseErrorList>,
|
||
) -> Self {
|
||
Self {
|
||
flags,
|
||
semis: vec![],
|
||
errors: vec![],
|
||
tokens: TokenStream::new(src, flags),
|
||
top_type,
|
||
unwinding: false,
|
||
any_error: false,
|
||
depth: 0,
|
||
out_errors,
|
||
}
|
||
}
|
||
|
||
/// Helper for FLOGF. This returns a number of spaces appropriate for a '%*c' format.
|
||
fn spaces(&self) -> usize {
|
||
self.depth * 2
|
||
}
|
||
|
||
/// \return the parser's status.
|
||
fn status(&mut self) -> ParserStatus {
|
||
if self.unwinding {
|
||
ParserStatus::unwinding
|
||
} else if self.flags & PARSE_FLAG_LEAVE_UNTERMINATED
|
||
&& self.peek_type(0) == ParseTokenType::terminate
|
||
{
|
||
ParserStatus::unsourcing
|
||
} else {
|
||
ParserStatus::ok
|
||
}
|
||
}
|
||
|
||
/// \return whether any leaf nodes we visit should be marked as unsourced.
|
||
fn unsource_leaves(&mut self) -> bool {
|
||
matches!(
|
||
self.status(),
|
||
ParserStatus::unsourcing | ParserStatus::unwinding
|
||
)
|
||
}
|
||
|
||
/// \return whether we permit an incomplete parse tree.
|
||
fn allow_incomplete(&self) -> bool {
|
||
self.flags & PARSE_FLAG_LEAVE_UNTERMINATED
|
||
}
|
||
|
||
/// \return whether a list type \p type allows arbitrary newlines in it.
|
||
fn list_type_chomps_newlines(&self, typ: Type) -> bool {
|
||
match typ {
|
||
Type::argument_list => {
|
||
// Hackish. If we are producing a freestanding argument list, then it allows
|
||
// semicolons, for hysterical raisins.
|
||
self.top_type == Type::freestanding_argument_list
|
||
}
|
||
Type::argument_or_redirection_list => {
|
||
// No newlines inside arguments.
|
||
false
|
||
}
|
||
Type::variable_assignment_list => {
|
||
// No newlines inside variable assignment lists.
|
||
false
|
||
}
|
||
Type::job_list => {
|
||
// Like echo a \n \n echo b
|
||
true
|
||
}
|
||
Type::case_item_list => {
|
||
// Like switch foo \n \n \n case a \n end
|
||
true
|
||
}
|
||
Type::andor_job_list => {
|
||
// Like while true ; \n \n and true ; end
|
||
true
|
||
}
|
||
Type::elseif_clause_list => {
|
||
// Like if true ; \n \n else if false; end
|
||
true
|
||
}
|
||
Type::job_conjunction_continuation_list => {
|
||
// This would be like echo a && echo b \n && echo c
|
||
// We could conceivably support this but do not now.
|
||
false
|
||
}
|
||
Type::job_continuation_list => {
|
||
// This would be like echo a \n | echo b
|
||
// We could conceivably support this but do not now.
|
||
false
|
||
}
|
||
_ => {
|
||
internal_error!(
|
||
self,
|
||
list_type_chomps_newlines,
|
||
"Type %ls not handled",
|
||
ast_type_to_string(typ)
|
||
);
|
||
}
|
||
}
|
||
}
|
||
|
||
/// \return whether a list type \p type allows arbitrary semicolons in it.
|
||
fn list_type_chomps_semis(&self, typ: Type) -> bool {
|
||
match typ {
|
||
Type::argument_list => {
|
||
// Hackish. If we are producing a freestanding argument list, then it allows
|
||
// semicolons, for hysterical raisins.
|
||
// That is, this is OK: complete -c foo -a 'x ; y ; z'
|
||
// But this is not: foo x ; y ; z
|
||
self.top_type == Type::freestanding_argument_list
|
||
}
|
||
|
||
Type::argument_or_redirection_list | Type::variable_assignment_list => false,
|
||
Type::job_list => {
|
||
// Like echo a ; ; echo b
|
||
true
|
||
}
|
||
Type::case_item_list => {
|
||
// Like switch foo ; ; ; case a \n end
|
||
// This is historically allowed.
|
||
true
|
||
}
|
||
Type::andor_job_list => {
|
||
// Like while true ; ; ; and true ; end
|
||
true
|
||
}
|
||
Type::elseif_clause_list => {
|
||
// Like if true ; ; ; else if false; end
|
||
false
|
||
}
|
||
Type::job_conjunction_continuation_list => {
|
||
// Like echo a ; ; && echo b. Not supported.
|
||
false
|
||
}
|
||
Type::job_continuation_list => {
|
||
// This would be like echo a ; | echo b
|
||
// Not supported.
|
||
// We could conceivably support this but do not now.
|
||
false
|
||
}
|
||
_ => {
|
||
internal_error!(
|
||
self,
|
||
list_type_chomps_semis,
|
||
"Type %ls not handled",
|
||
ast_type_to_string(typ)
|
||
);
|
||
}
|
||
}
|
||
}
|
||
|
||
/// Chomp extra comments, semicolons, etc. for a given list type.
|
||
fn chomp_extras(&mut self, typ: Type) {
|
||
let chomp_semis = self.list_type_chomps_semis(typ);
|
||
let chomp_newlines = self.list_type_chomps_newlines(typ);
|
||
loop {
|
||
let peek = self.tokens.peek(0);
|
||
if chomp_newlines && peek.typ == ParseTokenType::end && peek.is_newline {
|
||
// Just skip this newline, no need to save it.
|
||
self.tokens.pop();
|
||
} else if chomp_semis && peek.typ == ParseTokenType::end && !peek.is_newline {
|
||
let tok = self.tokens.pop();
|
||
// Perhaps save this extra semi.
|
||
if self.flags & PARSE_FLAG_SHOW_EXTRA_SEMIS {
|
||
self.semis.push(tok.range());
|
||
}
|
||
} else {
|
||
break;
|
||
}
|
||
}
|
||
}
|
||
|
||
/// \return whether a list type should recover from errors.s
|
||
/// That is, whether we should stop unwinding when we encounter this type.
|
||
fn list_type_stops_unwind(&self, typ: Type) -> bool {
|
||
typ == Type::job_list && self.flags & PARSE_FLAG_CONTINUE_AFTER_ERROR
|
||
}
|
||
|
||
/// \return a reference to a non-comment token at index \p idx.
|
||
fn peek_token(&mut self, idx: usize) -> &ParseToken {
|
||
self.tokens.peek(idx)
|
||
}
|
||
|
||
/// \return the type of a non-comment token.
|
||
fn peek_type(&mut self, idx: usize) -> ParseTokenType {
|
||
self.peek_token(idx).typ
|
||
}
|
||
|
||
/// Consume the next token, chomping any comments.
|
||
/// It is an error to call this unless we know there is a non-terminate token available.
|
||
/// \return the token.
|
||
fn consume_any_token(&mut self) -> ParseToken {
|
||
let tok = self.tokens.pop();
|
||
assert!(
|
||
tok.typ != ParseTokenType::comment,
|
||
"Should not be a comment"
|
||
);
|
||
assert!(
|
||
tok.typ != ParseTokenType::terminate,
|
||
"Cannot consume terminate token, caller should check status first"
|
||
);
|
||
tok
|
||
}
|
||
|
||
/// Consume the next token which is expected to be of the given type.
|
||
fn consume_token_type(&mut self, typ: ParseTokenType) -> SourceRange {
|
||
assert!(
|
||
typ != ParseTokenType::terminate,
|
||
"Should not attempt to consume terminate token"
|
||
);
|
||
let tok = self.consume_any_token();
|
||
if tok.typ != typ {
|
||
parse_error!(
|
||
self,
|
||
tok,
|
||
ParseErrorCode::generic,
|
||
"Expected %ls, but found %ls",
|
||
token_type_user_presentable_description(typ, ParseKeyword::none),
|
||
tok.user_presentable_description()
|
||
);
|
||
return SourceRange::new(0, 0);
|
||
}
|
||
tok.range()
|
||
}
|
||
|
||
/// The next token could not be parsed at the top level.
|
||
/// For example a trailing end like `begin ; end ; end`
|
||
/// Or an unexpected redirection like `>`
|
||
/// Consume it and add an error.
|
||
fn consume_excess_token_generating_error(&mut self) {
|
||
let tok = self.consume_any_token();
|
||
|
||
// In the rare case that we are parsing a freestanding argument list and not a job list,
|
||
// generate a generic error.
|
||
// TODO: this is a crummy message if we get a tokenizer error, for example:
|
||
// complete -c foo -a "'abc"
|
||
if self.top_type == Type::freestanding_argument_list {
|
||
parse_error!(
|
||
self,
|
||
tok,
|
||
ParseErrorCode::generic,
|
||
"Expected %ls, but found %ls",
|
||
token_type_user_presentable_description(ParseTokenType::string, ParseKeyword::none),
|
||
tok.user_presentable_description()
|
||
);
|
||
return;
|
||
}
|
||
|
||
assert!(self.top_type == Type::job_list);
|
||
match tok.typ {
|
||
ParseTokenType::string => {
|
||
// There are three keywords which end a job list.
|
||
match tok.keyword {
|
||
ParseKeyword::kw_end => {
|
||
parse_error!(
|
||
self,
|
||
tok,
|
||
ParseErrorCode::unbalancing_end,
|
||
"'end' outside of a block"
|
||
);
|
||
}
|
||
ParseKeyword::kw_else => {
|
||
parse_error!(
|
||
self,
|
||
tok,
|
||
ParseErrorCode::unbalancing_else,
|
||
"'else' builtin not inside of if block"
|
||
);
|
||
}
|
||
ParseKeyword::kw_case => {
|
||
parse_error!(
|
||
self,
|
||
tok,
|
||
ParseErrorCode::unbalancing_case,
|
||
"'case' builtin not inside of switch block"
|
||
);
|
||
}
|
||
_ => {
|
||
internal_error!(
|
||
self,
|
||
consume_excess_token_generating_error,
|
||
"Token %ls should not have prevented parsing a job list",
|
||
tok.user_presentable_description()
|
||
);
|
||
}
|
||
}
|
||
}
|
||
ParseTokenType::pipe
|
||
| ParseTokenType::redirection
|
||
| ParseTokenType::background
|
||
| ParseTokenType::andand
|
||
| ParseTokenType::oror => {
|
||
parse_error!(
|
||
self,
|
||
tok,
|
||
ParseErrorCode::generic,
|
||
"Expected a string, but found %ls",
|
||
tok.user_presentable_description()
|
||
);
|
||
}
|
||
ParseTokenType::tokenizer_error => {
|
||
parse_error!(
|
||
self,
|
||
tok,
|
||
ParseErrorCode::from(tok.tok_error),
|
||
"%ls",
|
||
tok.tok_error
|
||
);
|
||
}
|
||
ParseTokenType::end => {
|
||
internal_error!(
|
||
self,
|
||
consume_excess_token_generating_error,
|
||
"End token should never be excess"
|
||
);
|
||
}
|
||
ParseTokenType::terminate => {
|
||
internal_error!(
|
||
self,
|
||
consume_excess_token_generating_error,
|
||
"Terminate token should never be excess"
|
||
);
|
||
}
|
||
_ => {
|
||
internal_error!(
|
||
self,
|
||
consume_excess_token_generating_error,
|
||
"Unexpected excess token type: %ls",
|
||
tok.user_presentable_description()
|
||
);
|
||
}
|
||
}
|
||
}
|
||
|
||
/// This is for optional values and for lists.
|
||
/// A true return means we should descend into the production, false means stop.
|
||
/// Note that the argument is always nullptr and should be ignored. It is provided strictly
|
||
/// for overloading purposes.
|
||
fn can_parse(&mut self, node: &dyn Node) -> bool {
|
||
match node.typ() {
|
||
Type::job_conjunction => {
|
||
let token = self.peek_token(0);
|
||
if token.typ != ParseTokenType::string {
|
||
return false;
|
||
}
|
||
!matches!(
|
||
token.keyword,
|
||
// These end a job list.
|
||
ParseKeyword::kw_end | ParseKeyword::kw_else | ParseKeyword::kw_case
|
||
)
|
||
}
|
||
Type::argument => self.peek_type(0) == ParseTokenType::string,
|
||
Type::redirection => self.peek_type(0) == ParseTokenType::redirection,
|
||
Type::argument_or_redirection => {
|
||
[ParseTokenType::string, ParseTokenType::redirection].contains(&self.peek_type(0))
|
||
}
|
||
Type::variable_assignment => {
|
||
// Do we have a variable assignment at all?
|
||
if !self.peek_token(0).may_be_variable_assignment {
|
||
return false;
|
||
}
|
||
// What is the token after it?
|
||
match self.peek_type(1) {
|
||
ParseTokenType::string => {
|
||
// We have `a= cmd` and should treat it as a variable assignment.
|
||
true
|
||
}
|
||
ParseTokenType::terminate => {
|
||
// We have `a=` which is OK if we are allowing incomplete, an error
|
||
// otherwise.
|
||
self.allow_incomplete()
|
||
}
|
||
_ => {
|
||
// We have e.g. `a= >` which is an error.
|
||
// Note that we do not produce an error here. Instead we return false
|
||
// so this the token will be seen by allocate_populate_statement_contents.
|
||
false
|
||
}
|
||
}
|
||
}
|
||
Type::token_base => node
|
||
.as_token()
|
||
.unwrap()
|
||
.allows_token(self.peek_token(0).typ),
|
||
|
||
// Note we have specific overloads for our keyword nodes, as they need custom logic.
|
||
Type::keyword_base => {
|
||
let keyword = node.as_keyword().unwrap();
|
||
match keyword.allowed_keywords() {
|
||
// job conjunction decorator
|
||
[ParseKeyword::kw_and, ParseKeyword::kw_or] => {
|
||
// This is for a job conjunction like `and stuff`
|
||
// But if it's `and --help` then we treat it as an ordinary command.
|
||
keyword.allows_keyword(self.peek_token(0).keyword)
|
||
&& !self.peek_token(1).is_help_argument
|
||
}
|
||
// decorated statement decorator
|
||
[ParseKeyword::kw_command, ParseKeyword::kw_builtin, ParseKeyword::kw_exec] => {
|
||
// Here the keyword is 'command' or 'builtin' or 'exec'.
|
||
// `command stuff` executes a command called stuff.
|
||
// `command -n` passes the -n argument to the 'command' builtin.
|
||
// `command` by itself is a command.
|
||
if !keyword.allows_keyword(self.peek_token(0).keyword) {
|
||
return false;
|
||
}
|
||
let tok1 = self.peek_token(1);
|
||
tok1.typ == ParseTokenType::string && !tok1.is_dash_prefix_string()
|
||
}
|
||
[ParseKeyword::kw_time] => {
|
||
// Time keyword is only the time builtin if the next argument doesn't
|
||
// have a dash.
|
||
keyword.allows_keyword(self.peek_token(0).keyword)
|
||
&& !self.peek_token(1).is_dash_prefix_string()
|
||
}
|
||
_ => panic!("Unexpected keyword in can_parse()"),
|
||
}
|
||
}
|
||
Type::job_continuation => self.peek_type(0) == ParseTokenType::pipe,
|
||
Type::job_conjunction_continuation => {
|
||
[ParseTokenType::andand, ParseTokenType::oror].contains(&self.peek_type(0))
|
||
}
|
||
Type::andor_job => {
|
||
match self.peek_token(0).keyword {
|
||
ParseKeyword::kw_and | ParseKeyword::kw_or => {
|
||
// Check that the argument to and/or is a string that's not help. Otherwise
|
||
// it's either 'and --help' or a naked 'and', and not part of this list.
|
||
let nexttok = self.peek_token(1);
|
||
nexttok.typ == ParseTokenType::string && !nexttok.is_help_argument
|
||
}
|
||
_ => false,
|
||
}
|
||
}
|
||
Type::elseif_clause => {
|
||
self.peek_token(0).keyword == ParseKeyword::kw_else
|
||
&& self.peek_token(1).keyword == ParseKeyword::kw_if
|
||
}
|
||
Type::else_clause => self.peek_token(0).keyword == ParseKeyword::kw_else,
|
||
Type::case_item => self.peek_token(0).keyword == ParseKeyword::kw_case,
|
||
_ => panic!("Unexpected token type in can_parse()"),
|
||
}
|
||
}
|
||
|
||
/// Given that we are a list of type ListNodeType, whose contents type is ContentsNode,
|
||
/// populate as many elements as we can.
|
||
/// If exhaust_stream is set, then keep going until we get parse_token_type_t::terminate.
|
||
fn populate_list<ListType: List>(&mut self, list: &mut ListType, exhaust_stream: bool)
|
||
where
|
||
<ListType as List>::ContentsNode: NodeMut,
|
||
{
|
||
assert!(list.contents().is_empty(), "List is not initially empty");
|
||
|
||
// Do not attempt to parse a list if we are unwinding.
|
||
if self.unwinding {
|
||
assert!(
|
||
!exhaust_stream,
|
||
"exhaust_stream should only be set at top level, and so we should not be unwinding"
|
||
);
|
||
// Mark in the list that it was unwound.
|
||
FLOG!(
|
||
ast_construction,
|
||
"%*sunwinding %ls",
|
||
self.spaces(),
|
||
"",
|
||
ast_type_to_string(list.typ())
|
||
);
|
||
assert!(list.contents().is_empty(), "Should be an empty list");
|
||
return;
|
||
}
|
||
|
||
// We're going to populate a vector with our nodes.
|
||
// Later on we will copy this to the heap with a single allocation.
|
||
let mut contents = vec![];
|
||
|
||
loop {
|
||
// If we are unwinding, then either we recover or we break the loop, dependent on the
|
||
// loop type.
|
||
if self.unwinding {
|
||
if !self.list_type_stops_unwind(list.typ()) {
|
||
break;
|
||
}
|
||
// We are going to stop unwinding.
|
||
// Rather hackish. Just chomp until we get to a string or end node.
|
||
loop {
|
||
let typ = self.peek_type(0);
|
||
if [
|
||
ParseTokenType::string,
|
||
ParseTokenType::terminate,
|
||
ParseTokenType::end,
|
||
]
|
||
.contains(&typ)
|
||
{
|
||
break;
|
||
}
|
||
let tok = self.tokens.pop();
|
||
self.errors.push(tok.range());
|
||
FLOG!(
|
||
ast_construction,
|
||
"%*schomping range %u-%u",
|
||
self.spaces(),
|
||
"",
|
||
tok.source_start,
|
||
tok.source_length
|
||
);
|
||
}
|
||
FLOG!(ast_construction, "%*sdone unwinding", self.spaces(), "");
|
||
self.unwinding = false;
|
||
}
|
||
|
||
// Chomp semis and newlines.
|
||
self.chomp_extras(list.typ());
|
||
|
||
// Now try parsing a node.
|
||
if let Some(node) = self.try_parse::<ListType::ContentsNode>() {
|
||
// #7201: Minimize reallocations of contents vector
|
||
if contents.is_empty() {
|
||
contents.reserve(64);
|
||
}
|
||
contents.push(node);
|
||
} else if exhaust_stream && self.peek_type(0) != ParseTokenType::terminate {
|
||
// We aren't allowed to stop. Produce an error and keep going.
|
||
self.consume_excess_token_generating_error()
|
||
} else {
|
||
// We either stop once we can't parse any more of this contents node, or we
|
||
// exhausted the stream as requested.
|
||
break;
|
||
}
|
||
}
|
||
|
||
// Populate our list from our contents.
|
||
if !contents.is_empty() {
|
||
assert!(
|
||
contents.len() <= u32::MAX.try_into().unwrap(),
|
||
"Contents size out of bounds"
|
||
);
|
||
assert!(list.contents().is_empty(), "List should still be empty");
|
||
*list.contents_mut() = contents;
|
||
}
|
||
|
||
FLOG!(
|
||
ast_construction,
|
||
"%*s%ls size: %lu",
|
||
self.spaces(),
|
||
"",
|
||
ast_type_to_string(list.typ()),
|
||
list.count()
|
||
);
|
||
}
|
||
|
||
/// Allocate and populate a statement contents pointer.
|
||
/// This must never return null.
|
||
fn allocate_populate_statement_contents(&mut self) -> Box<StatementVariant> {
|
||
// In case we get a parse error, we still need to return something non-null. Use a
|
||
// decorated statement; all of its leaf nodes will end up unsourced.
|
||
fn got_error(slf: &mut Populator<'_>) -> Box<StatementVariant> {
|
||
assert!(slf.unwinding, "Should have produced an error");
|
||
new_decorated_statement(slf)
|
||
}
|
||
|
||
fn new_decorated_statement(slf: &mut Populator<'_>) -> Box<StatementVariant> {
|
||
let embedded = slf.allocate_visit::<DecoratedStatement>();
|
||
Box::new(StatementVariant::DecoratedStatement(*embedded))
|
||
}
|
||
|
||
if self.peek_token(0).typ == ParseTokenType::terminate && self.allow_incomplete() {
|
||
// This may happen if we just have a 'time' prefix.
|
||
// Construct a decorated statement, which will be unsourced.
|
||
self.allocate_visit::<DecoratedStatement>();
|
||
} else if self.peek_token(0).typ != ParseTokenType::string {
|
||
// We may be unwinding already; do not produce another error.
|
||
// For example in `true | and`.
|
||
parse_error!(
|
||
self,
|
||
self.peek_token(0),
|
||
ParseErrorCode::generic,
|
||
"Expected a command, but found %ls",
|
||
self.peek_token(0).user_presentable_description()
|
||
);
|
||
return got_error(self);
|
||
} else if self.peek_token(0).may_be_variable_assignment {
|
||
// Here we have a variable assignment which we chose to not parse as a variable
|
||
// assignment because there was no string after it.
|
||
// Ensure we consume the token, so we don't get back here again at the same place.
|
||
parse_error!(
|
||
self,
|
||
self.consume_any_token(),
|
||
ParseErrorCode::bare_variable_assignment,
|
||
""
|
||
);
|
||
return got_error(self);
|
||
}
|
||
|
||
// The only block-like builtin that takes any parameters is 'function'. So go to decorated
|
||
// statements if the subsequent token looks like '--'. The logic here is subtle:
|
||
//
|
||
// If we are 'begin', then we expect to be invoked with no arguments.
|
||
// If we are 'function', then we are a non-block if we are invoked with -h or --help
|
||
// If we are anything else, we require an argument, so do the same thing if the subsequent
|
||
// token is a statement terminator.
|
||
if self.peek_token(0).typ == ParseTokenType::string {
|
||
// If we are a function, then look for help arguments. Otherwise, if the next token
|
||
// looks like an option (starts with a dash), then parse it as a decorated statement.
|
||
if (self.peek_token(0).keyword == ParseKeyword::kw_function
|
||
&& self.peek_token(1).is_help_argument)
|
||
|| (self.peek_token(0).keyword != ParseKeyword::kw_function
|
||
&& self.peek_token(1).has_dash_prefix)
|
||
{
|
||
return new_decorated_statement(self);
|
||
}
|
||
|
||
// Likewise if the next token doesn't look like an argument at all. This corresponds to
|
||
// e.g. a "naked if".
|
||
let naked_invocation_invokes_help = ![ParseKeyword::kw_begin, ParseKeyword::kw_end]
|
||
.contains(&self.peek_token(0).keyword);
|
||
if naked_invocation_invokes_help
|
||
&& [ParseTokenType::end, ParseTokenType::terminate]
|
||
.contains(&self.peek_token(1).typ)
|
||
{
|
||
return new_decorated_statement(self);
|
||
}
|
||
}
|
||
|
||
match self.peek_token(0).keyword {
|
||
ParseKeyword::kw_not | ParseKeyword::kw_exclam => {
|
||
let embedded = self.allocate_visit::<NotStatement>();
|
||
Box::new(StatementVariant::NotStatement(*embedded))
|
||
}
|
||
ParseKeyword::kw_for
|
||
| ParseKeyword::kw_while
|
||
| ParseKeyword::kw_function
|
||
| ParseKeyword::kw_begin => {
|
||
let embedded = self.allocate_visit::<BlockStatement>();
|
||
Box::new(StatementVariant::BlockStatement(*embedded))
|
||
}
|
||
ParseKeyword::kw_if => {
|
||
let embedded = self.allocate_visit::<IfStatement>();
|
||
Box::new(StatementVariant::IfStatement(*embedded))
|
||
}
|
||
ParseKeyword::kw_switch => {
|
||
let embedded = self.allocate_visit::<SwitchStatement>();
|
||
Box::new(StatementVariant::SwitchStatement(*embedded))
|
||
}
|
||
ParseKeyword::kw_end => {
|
||
// 'end' is forbidden as a command.
|
||
// For example, `if end` or `while end` will produce this error.
|
||
// We still have to descend into the decorated statement because
|
||
// we can't leave our pointer as null.
|
||
parse_error!(
|
||
self,
|
||
self.peek_token(0),
|
||
ParseErrorCode::generic,
|
||
"Expected a command, but found %ls",
|
||
self.peek_token(0).user_presentable_description()
|
||
);
|
||
return got_error(self);
|
||
}
|
||
_ => new_decorated_statement(self),
|
||
}
|
||
}
|
||
|
||
/// Allocate and populate a block statement header.
|
||
/// This must never return null.
|
||
fn allocate_populate_block_header(&mut self) -> Box<BlockStatementHeaderVariant> {
|
||
Box::new(match self.peek_token(0).keyword {
|
||
ParseKeyword::kw_for => {
|
||
let embedded = self.allocate_visit::<ForHeader>();
|
||
BlockStatementHeaderVariant::ForHeader(*embedded)
|
||
}
|
||
ParseKeyword::kw_while => {
|
||
let embedded = self.allocate_visit::<WhileHeader>();
|
||
BlockStatementHeaderVariant::WhileHeader(*embedded)
|
||
}
|
||
ParseKeyword::kw_function => {
|
||
let embedded = self.allocate_visit::<FunctionHeader>();
|
||
BlockStatementHeaderVariant::FunctionHeader(*embedded)
|
||
}
|
||
ParseKeyword::kw_begin => {
|
||
let embedded = self.allocate_visit::<BeginHeader>();
|
||
BlockStatementHeaderVariant::BeginHeader(*embedded)
|
||
}
|
||
_ => {
|
||
internal_error!(
|
||
self,
|
||
allocate_populate_block_header,
|
||
"should not have descended into block_header"
|
||
);
|
||
}
|
||
})
|
||
}
|
||
|
||
fn try_parse<T: NodeMut + Default>(&mut self) -> Option<Box<T>> {
|
||
// TODO Optimize this.
|
||
let prototype = T::default();
|
||
if !self.can_parse(&prototype) {
|
||
return None;
|
||
}
|
||
Some(self.allocate_visit())
|
||
}
|
||
|
||
/// Given a node type, allocate it and invoke its default constructor.
|
||
/// \return the resulting Node
|
||
fn allocate<T: NodeMut + Default>(&self) -> Box<T> {
|
||
let result = Box::<T>::default();
|
||
FLOG!(
|
||
ast_construction,
|
||
"%*smake %ls %p",
|
||
self.spaces(),
|
||
"",
|
||
ast_type_to_string(result.typ()),
|
||
format!("{result:p}")
|
||
);
|
||
result
|
||
}
|
||
|
||
// Given a node type, allocate it, invoke its default constructor,
|
||
// and then visit it as a field.
|
||
// \return the resulting Node pointer. It is never null.
|
||
fn allocate_visit<T: NodeMut + Default>(&mut self) -> Box<T> {
|
||
let mut result = Box::<T>::default();
|
||
self.visit_mut(&mut *result);
|
||
result
|
||
}
|
||
|
||
fn visit_argument(&mut self, arg: &mut Argument) {
|
||
if self.unsource_leaves() {
|
||
arg.range = None;
|
||
return;
|
||
}
|
||
arg.range = Some(self.consume_token_type(ParseTokenType::string));
|
||
}
|
||
|
||
fn visit_variable_assignment(&mut self, varas: &mut VariableAssignment) {
|
||
if self.unsource_leaves() {
|
||
varas.range = None;
|
||
return;
|
||
}
|
||
if !self.peek_token(0).may_be_variable_assignment {
|
||
internal_error!(
|
||
self,
|
||
visit_variable_assignment,
|
||
"Should not have created variable_assignment_t from this token"
|
||
);
|
||
}
|
||
varas.range = Some(self.consume_token_type(ParseTokenType::string));
|
||
}
|
||
|
||
fn visit_job_continuation(&mut self, node: &mut JobContinuation) {
|
||
// Special error handling to catch 'and' and 'or' in pipelines, like `true | and false`.
|
||
if [ParseKeyword::kw_and, ParseKeyword::kw_or].contains(&self.peek_token(1).keyword) {
|
||
parse_error!(
|
||
self,
|
||
self.peek_token(1),
|
||
ParseErrorCode::andor_in_pipeline,
|
||
INVALID_PIPELINE_CMD_ERR_MSG,
|
||
self.peek_token(1).keyword
|
||
);
|
||
}
|
||
node.accept_mut(self, false);
|
||
}
|
||
|
||
// Overload for token fields.
|
||
fn visit_token(&mut self, token: &mut dyn Token) {
|
||
if self.unsource_leaves() {
|
||
*token.range_mut() = None;
|
||
return;
|
||
}
|
||
|
||
if !token.allows_token(self.peek_token(0).typ) {
|
||
if self.flags & PARSE_FLAG_LEAVE_UNTERMINATED
|
||
&& [
|
||
TokenizerError::unterminated_quote,
|
||
TokenizerError::unterminated_subshell,
|
||
]
|
||
.contains(&self.peek_token(0).tok_error)
|
||
{
|
||
return;
|
||
}
|
||
|
||
parse_error!(
|
||
self,
|
||
self.peek_token(0),
|
||
ParseErrorCode::generic,
|
||
"Expected %ls, but found %ls",
|
||
token_types_user_presentable_description(token.allowed_tokens()),
|
||
self.peek_token(0).user_presentable_description()
|
||
);
|
||
*token.range_mut() = None;
|
||
return;
|
||
}
|
||
let tok = self.consume_any_token();
|
||
*token.token_type_mut() = tok.typ;
|
||
*token.range_mut() = Some(tok.range());
|
||
}
|
||
|
||
// Overload for keyword fields.
|
||
fn visit_keyword(&mut self, keyword: &mut dyn Keyword) -> VisitResult {
|
||
if self.unsource_leaves() {
|
||
*keyword.range_mut() = None;
|
||
return VisitResult::Continue(());
|
||
}
|
||
|
||
if !keyword.allows_keyword(self.peek_token(0).keyword) {
|
||
*keyword.range_mut() = None;
|
||
|
||
if self.flags & PARSE_FLAG_LEAVE_UNTERMINATED
|
||
&& [
|
||
TokenizerError::unterminated_quote,
|
||
TokenizerError::unterminated_subshell,
|
||
]
|
||
.contains(&self.peek_token(0).tok_error)
|
||
{
|
||
return VisitResult::Continue(());
|
||
}
|
||
|
||
// Special error reporting for keyword_t<kw_end>.
|
||
let allowed_keywords = keyword.allowed_keywords();
|
||
if keyword.allowed_keywords() == [ParseKeyword::kw_end] {
|
||
return VisitResult::Break(MissingEndError {
|
||
allowed_keywords,
|
||
token: *self.peek_token(0),
|
||
});
|
||
} else {
|
||
parse_error!(
|
||
self,
|
||
self.peek_token(0),
|
||
ParseErrorCode::generic,
|
||
"Expected %ls, but found %ls",
|
||
keywords_user_presentable_description(allowed_keywords),
|
||
self.peek_token(0).user_presentable_description(),
|
||
);
|
||
return VisitResult::Continue(());
|
||
}
|
||
}
|
||
let tok = self.consume_any_token();
|
||
*keyword.keyword_mut() = tok.keyword;
|
||
*keyword.range_mut() = Some(tok.range());
|
||
VisitResult::Continue(())
|
||
}
|
||
|
||
fn visit_maybe_newlines(&mut self, nls: &mut MaybeNewlines) {
|
||
if self.unsource_leaves() {
|
||
nls.range = None;
|
||
return;
|
||
}
|
||
let mut range = SourceRange::new(0, 0);
|
||
// TODO: it would be nice to have the start offset be the current position in the token
|
||
// stream, even if there are no newlines.
|
||
while self.peek_token(0).is_newline {
|
||
let r = self.consume_token_type(ParseTokenType::end);
|
||
if range.length == 0 {
|
||
range = r;
|
||
} else {
|
||
range.length = r.start + r.length - range.start
|
||
}
|
||
}
|
||
nls.range = Some(range);
|
||
}
|
||
}
|
||
|
||
/// The status of our parser.
|
||
enum ParserStatus {
|
||
/// Parsing is going just fine, thanks for asking.
|
||
ok,
|
||
|
||
/// We have exhausted the token stream, but the caller was OK with an incomplete parse tree.
|
||
/// All further leaf nodes should have the unsourced flag set.
|
||
unsourcing,
|
||
|
||
/// We encountered an parse error and are "unwinding."
|
||
/// Do not consume any tokens until we get back to a list type which stops unwinding.
|
||
unwinding,
|
||
}
|
||
|
||
fn parse_from_top(
|
||
src: &wstr,
|
||
flags: ParseTreeFlags,
|
||
out_errors: &mut Option<ParseErrorList>,
|
||
top_type: Type,
|
||
) -> Ast {
|
||
assert!(
|
||
[Type::job_list, Type::freestanding_argument_list].contains(&top_type),
|
||
"Invalid top type"
|
||
);
|
||
|
||
let mut ast = Ast::default();
|
||
|
||
let mut pops = Populator::new(src, flags, top_type, out_errors);
|
||
if top_type == Type::job_list {
|
||
let mut list = pops.allocate::<JobList>();
|
||
pops.populate_list(&mut *list, true /* exhaust_stream */);
|
||
ast.top = list;
|
||
} else {
|
||
let mut list = pops.allocate::<FreestandingArgumentList>();
|
||
pops.populate_list(&mut list.arguments, true /* exhaust_stream */);
|
||
ast.top = list;
|
||
}
|
||
|
||
// Chomp trailing extras, etc.
|
||
pops.chomp_extras(Type::job_list);
|
||
|
||
ast.any_error = pops.any_error;
|
||
ast.extras = Extras {
|
||
comments: pops.tokens.comment_ranges,
|
||
semis: pops.semis,
|
||
errors: pops.errors,
|
||
};
|
||
|
||
if top_type == Type::job_list {
|
||
// Set all parent nodes.
|
||
// It turns out to be more convenient to do this after the parse phase.
|
||
ast.top_mut()
|
||
.as_mut_job_list()
|
||
.as_mut()
|
||
.unwrap()
|
||
.set_parents();
|
||
} else {
|
||
ast.top_mut()
|
||
.as_mut_freestanding_argument_list()
|
||
.as_mut()
|
||
.unwrap()
|
||
.set_parents();
|
||
}
|
||
|
||
ast
|
||
}
|
||
|
||
/// \return tokenizer flags corresponding to parse tree flags.
|
||
impl From<ParseTreeFlags> for TokFlags {
|
||
fn from(flags: ParseTreeFlags) -> Self {
|
||
let mut tok_flags = TokFlags(0);
|
||
// Note we do not need to respect parse_flag_show_blank_lines, no clients are interested
|
||
// in them.
|
||
if flags & PARSE_FLAG_INCLUDE_COMMENTS {
|
||
tok_flags |= TOK_SHOW_COMMENTS;
|
||
}
|
||
if flags & PARSE_FLAG_ACCEPT_INCOMPLETE_TOKENS {
|
||
tok_flags |= TOK_ACCEPT_UNFINISHED;
|
||
}
|
||
if flags & PARSE_FLAG_CONTINUE_AFTER_ERROR {
|
||
tok_flags |= TOK_CONTINUE_AFTER_ERROR
|
||
}
|
||
tok_flags
|
||
}
|
||
}
|
||
|
||
/// Convert from Tokenizer's token type to a parse_token_t type.
|
||
impl From<TokenType> for ParseTokenType {
|
||
fn from(token_type: TokenType) -> Self {
|
||
match token_type {
|
||
TokenType::string => ParseTokenType::string,
|
||
TokenType::pipe => ParseTokenType::pipe,
|
||
TokenType::andand => ParseTokenType::andand,
|
||
TokenType::oror => ParseTokenType::oror,
|
||
TokenType::end => ParseTokenType::end,
|
||
TokenType::background => ParseTokenType::background,
|
||
TokenType::redirect => ParseTokenType::redirection,
|
||
TokenType::error => ParseTokenType::tokenizer_error,
|
||
TokenType::comment => ParseTokenType::comment,
|
||
_ => panic!("bad token type"),
|
||
}
|
||
}
|
||
}
|
||
|
||
fn is_keyword_char(c: char) -> bool {
|
||
('a'..='z').contains(&c)
|
||
|| ('A'..='Z').contains(&c)
|
||
|| ('0'..='9').contains(&c)
|
||
|| c == '\''
|
||
|| c == '"'
|
||
|| c == '\\'
|
||
|| c == '\n'
|
||
|| c == '!'
|
||
}
|
||
|
||
/// Given a token, returns the keyword it matches, or ParseKeyword::none.
|
||
fn keyword_for_token(tok: TokenType, token: &wstr) -> ParseKeyword {
|
||
/* Only strings can be keywords */
|
||
if tok != TokenType::string {
|
||
return ParseKeyword::none;
|
||
}
|
||
|
||
// If token is clean (which most are), we can compare it directly. Otherwise we have to expand
|
||
// it. We only expand quotes, and we don't want to do expensive expansions like tilde
|
||
// expansions. So we do our own "cleanliness" check; if we find a character not in our allowed
|
||
// set we know it's not a keyword, and if we never find a quote we don't have to expand! Note
|
||
// that this lowercase set could be shrunk to be just the characters that are in keywords.
|
||
let mut result = ParseKeyword::none;
|
||
let mut needs_expand = false;
|
||
let mut all_chars_valid = true;
|
||
for c in token.chars() {
|
||
if !is_keyword_char(c) {
|
||
all_chars_valid = false;
|
||
break;
|
||
}
|
||
// If we encounter a quote, we need expansion.
|
||
needs_expand = needs_expand || c == '"' || c == '\'' || c == '\\'
|
||
}
|
||
|
||
if all_chars_valid {
|
||
// Expand if necessary.
|
||
if !needs_expand {
|
||
result = ParseKeyword::from(token);
|
||
} else if let Some(unescaped) = unescape_string(token, UnescapeStringStyle::default()) {
|
||
result = ParseKeyword::from(&unescaped[..]);
|
||
}
|
||
}
|
||
result
|
||
}
|
||
|
||
use crate::ffi_tests::add_test;
|
||
add_test!("test_ast_parse", || {
|
||
use crate::parse_constants::PARSE_FLAG_NONE;
|
||
let src = L!("echo");
|
||
let ast = Ast::parse(src, PARSE_FLAG_NONE, &mut None);
|
||
assert!(!ast.any_error);
|
||
});
|
||
|
||
pub use ast_ffi::{Category, Type};
|
||
|
||
#[cxx::bridge]
|
||
#[allow(clippy::needless_lifetimes)] // false positive
|
||
pub mod ast_ffi {
|
||
extern "C++" {
|
||
include!("wutil.h");
|
||
include!("tokenizer.h");
|
||
include!("parse_constants.h");
|
||
type wcharz_t = crate::wchar_ffi::wcharz_t;
|
||
type ParseTokenType = crate::parse_constants::ParseTokenType;
|
||
type ParseKeyword = crate::parse_constants::ParseKeyword;
|
||
type SourceRange = crate::parse_constants::SourceRange;
|
||
type ParseErrorList = crate::parse_constants::ParseErrorList;
|
||
type StatementDecoration = crate::parse_constants::StatementDecoration;
|
||
}
|
||
|
||
#[derive(Debug)]
|
||
pub enum Category {
|
||
branch,
|
||
leaf,
|
||
list,
|
||
}
|
||
|
||
#[derive(Debug)]
|
||
pub enum Type {
|
||
token_base,
|
||
keyword_base,
|
||
redirection,
|
||
variable_assignment,
|
||
variable_assignment_list,
|
||
argument_or_redirection,
|
||
argument_or_redirection_list,
|
||
statement,
|
||
job_pipeline,
|
||
job_conjunction,
|
||
for_header,
|
||
while_header,
|
||
function_header,
|
||
begin_header,
|
||
block_statement,
|
||
if_clause,
|
||
elseif_clause,
|
||
elseif_clause_list,
|
||
else_clause,
|
||
if_statement,
|
||
case_item,
|
||
switch_statement,
|
||
decorated_statement,
|
||
not_statement,
|
||
job_continuation,
|
||
job_continuation_list,
|
||
job_conjunction_continuation,
|
||
andor_job,
|
||
andor_job_list,
|
||
freestanding_argument_list,
|
||
token_conjunction,
|
||
job_conjunction_continuation_list,
|
||
maybe_newlines,
|
||
token_pipe,
|
||
case_item_list,
|
||
argument,
|
||
argument_list,
|
||
job_list,
|
||
}
|
||
extern "Rust" {
|
||
type Ast;
|
||
type NodeFfi<'a>;
|
||
type ExtrasFFI<'a>;
|
||
unsafe fn ast_parse_ffi(
|
||
src: &CxxWString,
|
||
flags: u8,
|
||
errors: *mut ParseErrorList,
|
||
) -> Box<Ast>;
|
||
unsafe fn ast_parse_argument_list_ffi(
|
||
src: &CxxWString,
|
||
flags: u8,
|
||
errors: *mut ParseErrorList,
|
||
) -> Box<Ast>;
|
||
unsafe fn errored(self: &Ast) -> bool;
|
||
#[cxx_name = "top"]
|
||
unsafe fn top_ffi(self: &Ast) -> Box<NodeFfi<'_>>;
|
||
|
||
#[cxx_name = "parent"]
|
||
unsafe fn parent_ffi<'a>(self: &'a NodeFfi<'a>) -> Box<NodeFfi<'a>>;
|
||
|
||
#[cxx_name = "dump"]
|
||
unsafe fn dump_ffi(self: &Ast, orig: &CxxWString) -> UniquePtr<CxxWString>;
|
||
#[cxx_name = "extras"]
|
||
fn extras_ffi(self: &Ast) -> Box<ExtrasFFI<'_>>;
|
||
unsafe fn comments<'a>(self: &'a ExtrasFFI<'a>) -> &'a [SourceRange];
|
||
unsafe fn semis<'a>(self: &'a ExtrasFFI<'a>) -> &'a [SourceRange];
|
||
unsafe fn errors<'a>(self: &'a ExtrasFFI<'a>) -> &'a [SourceRange];
|
||
#[cxx_name = "ast_type_to_string"]
|
||
fn ast_type_to_string_ffi(typ: Type) -> wcharz_t;
|
||
type Traversal<'a>;
|
||
unsafe fn new_ast_traversal<'a>(root: &'a NodeFfi<'a>) -> Box<Traversal<'a>>;
|
||
#[cxx_name = "next"]
|
||
unsafe fn next_ffi<'a>(self: &'a mut Traversal) -> Box<NodeFfi<'a>>;
|
||
}
|
||
|
||
#[rustfmt::skip]
|
||
extern "Rust" {
|
||
type BlockStatementHeaderVariant;
|
||
type StatementVariant;
|
||
|
||
unsafe fn ptr<'a>(self: &'a NodeFfi<'a>) -> Box<NodeFfi<'a>>;
|
||
unsafe fn describe(self: &NodeFfi<'_>) -> UniquePtr<CxxWString>;
|
||
unsafe fn typ(self: &NodeFfi<'_>) -> Type;
|
||
unsafe fn category(self: &NodeFfi<'_>) -> Category;
|
||
unsafe fn pointer_eq(self: &NodeFfi<'_>, rhs: &NodeFfi) -> bool;
|
||
unsafe fn has_value(self: &NodeFfi<'_>) -> bool;
|
||
|
||
unsafe fn kw(self: &NodeFfi<'_>) -> ParseKeyword;
|
||
unsafe fn token_type(self: &NodeFfi<'_>) -> ParseTokenType;
|
||
unsafe fn has_source(self: &NodeFfi<'_>) -> bool;
|
||
|
||
fn token_type(self: &TokenConjunction) -> ParseTokenType;
|
||
|
||
type AndorJobList;
|
||
type AndorJob;
|
||
type ArgumentList;
|
||
type Argument;
|
||
type ArgumentOrRedirectionList;
|
||
type ArgumentOrRedirection;
|
||
type BeginHeader;
|
||
type BlockStatement;
|
||
type CaseItemList;
|
||
type CaseItem;
|
||
type DecoratedStatementDecorator;
|
||
type DecoratedStatement;
|
||
type ElseClause;
|
||
type ElseifClauseList;
|
||
type ElseifClause;
|
||
type ForHeader;
|
||
type FreestandingArgumentList;
|
||
type FunctionHeader;
|
||
type IfClause;
|
||
type IfStatement;
|
||
type JobConjunctionContinuationList;
|
||
type JobConjunctionContinuation;
|
||
type JobConjunctionDecorator;
|
||
type JobConjunction;
|
||
type JobContinuationList;
|
||
type JobContinuation;
|
||
type JobList;
|
||
type JobPipeline;
|
||
type KeywordBegin;
|
||
type KeywordCase;
|
||
type KeywordElse;
|
||
type KeywordEnd;
|
||
type KeywordFor;
|
||
type KeywordFunction;
|
||
type KeywordIf;
|
||
type KeywordIn;
|
||
type KeywordNot;
|
||
type KeywordTime;
|
||
type KeywordWhile;
|
||
type MaybeNewlines;
|
||
type NotStatement;
|
||
type Redirection;
|
||
type SemiNl;
|
||
type Statement;
|
||
type String_;
|
||
type SwitchStatement;
|
||
type TokenBackground;
|
||
type TokenConjunction;
|
||
type TokenPipe;
|
||
type TokenRedirection;
|
||
type VariableAssignmentList;
|
||
type VariableAssignment;
|
||
type WhileHeader;
|
||
|
||
fn count(self: &ArgumentList) -> usize;
|
||
fn empty(self: &ArgumentList) -> bool;
|
||
unsafe fn at(self: & ArgumentList, i: usize) -> *const Argument;
|
||
|
||
fn count(self: &ArgumentOrRedirectionList) -> usize;
|
||
fn empty(self: &ArgumentOrRedirectionList) -> bool;
|
||
unsafe fn at(self: & ArgumentOrRedirectionList, i: usize) -> *const ArgumentOrRedirection;
|
||
|
||
fn count(self: &JobList) -> usize;
|
||
fn empty(self: &JobList) -> bool;
|
||
unsafe fn at(self: & JobList, i: usize) -> *const JobConjunction;
|
||
|
||
fn count(self: &JobContinuationList) -> usize;
|
||
fn empty(self: &JobContinuationList) -> bool;
|
||
unsafe fn at(self: & JobContinuationList, i: usize) -> *const JobContinuation;
|
||
|
||
fn count(self: &ElseifClauseList) -> usize;
|
||
fn empty(self: &ElseifClauseList) -> bool;
|
||
unsafe fn at(self: & ElseifClauseList, i: usize) -> *const ElseifClause;
|
||
|
||
fn count(self: &CaseItemList) -> usize;
|
||
fn empty(self: &CaseItemList) -> bool;
|
||
unsafe fn at(self: & CaseItemList, i: usize) -> *const CaseItem;
|
||
|
||
fn count(self: &VariableAssignmentList) -> usize;
|
||
fn empty(self: &VariableAssignmentList) -> bool;
|
||
unsafe fn at(self: & VariableAssignmentList, i: usize) -> *const VariableAssignment;
|
||
|
||
fn count(self: &JobConjunctionContinuationList) -> usize;
|
||
fn empty(self: &JobConjunctionContinuationList) -> bool;
|
||
unsafe fn at(self: & JobConjunctionContinuationList, i: usize) -> *const JobConjunctionContinuation;
|
||
|
||
fn count(self: &AndorJobList) -> usize;
|
||
fn empty(self: &AndorJobList) -> bool;
|
||
unsafe fn at(self: & AndorJobList, i: usize) -> *const AndorJob;
|
||
|
||
fn describe(self: &Statement) -> UniquePtr<CxxWString>;
|
||
|
||
fn kw(self: &JobConjunctionDecorator) -> ParseKeyword;
|
||
fn decoration(self: &DecoratedStatement) -> StatementDecoration;
|
||
|
||
fn is_argument(self: &ArgumentOrRedirection) -> bool;
|
||
unsafe fn argument(self: & ArgumentOrRedirection) -> & Argument;
|
||
fn is_redirection(self: &ArgumentOrRedirection) -> bool;
|
||
unsafe fn redirection(self: & ArgumentOrRedirection) -> & Redirection;
|
||
|
||
unsafe fn contents(self: &Statement) -> &StatementVariant;
|
||
unsafe fn oper(self: &Redirection) -> &TokenRedirection;
|
||
unsafe fn target(self: &Redirection) -> &String_;
|
||
unsafe fn argument_ffi(self: &ArgumentOrRedirection) -> &Argument;
|
||
unsafe fn redirection_ffi(self: &ArgumentOrRedirection) -> &Redirection;
|
||
fn has_time(self: &JobPipeline) -> bool;
|
||
unsafe fn time(self: &JobPipeline) -> &KeywordTime;
|
||
unsafe fn variables(self: &JobPipeline) -> &VariableAssignmentList;
|
||
unsafe fn statement(self: &JobPipeline) -> &Statement;
|
||
unsafe fn continuation(self: &JobPipeline) -> &JobContinuationList;
|
||
fn has_bg(self: &JobPipeline) -> bool;
|
||
unsafe fn bg(self: &JobPipeline) -> &TokenBackground;
|
||
fn has_decorator(self: &JobConjunction) -> bool;
|
||
unsafe fn decorator(self: &JobConjunction) -> &JobConjunctionDecorator;
|
||
unsafe fn job(self: &JobConjunction) -> &JobPipeline;
|
||
unsafe fn continuations(self: &JobConjunction) -> &JobConjunctionContinuationList;
|
||
fn has_semi_nl(self: &JobConjunction) -> bool;
|
||
unsafe fn semi_nl(self: &JobConjunction) -> &SemiNl;
|
||
unsafe fn var_name(self: &ForHeader) -> &String_;
|
||
unsafe fn args(self: &ForHeader) -> &ArgumentList;
|
||
unsafe fn semi_nl(self: &ForHeader) -> &SemiNl;
|
||
unsafe fn condition(self: &WhileHeader) -> &JobConjunction;
|
||
unsafe fn andor_tail(self: &WhileHeader) -> &AndorJobList;
|
||
unsafe fn first_arg(self: &FunctionHeader) -> &Argument;
|
||
unsafe fn args(self: &FunctionHeader) -> &ArgumentList;
|
||
unsafe fn semi_nl(self: &FunctionHeader) -> &SemiNl;
|
||
fn has_semi_nl(self: &BeginHeader) -> bool;
|
||
unsafe fn semi_nl(self: &BeginHeader) -> &SemiNl;
|
||
unsafe fn header(self: &BlockStatement) -> &BlockStatementHeaderVariant;
|
||
unsafe fn jobs(self: &BlockStatement) -> &JobList;
|
||
unsafe fn args_or_redirs(self: &BlockStatement) -> &ArgumentOrRedirectionList;
|
||
unsafe fn condition(self: &IfClause) -> &JobConjunction;
|
||
unsafe fn andor_tail(self: &IfClause) -> &AndorJobList;
|
||
unsafe fn body(self: &IfClause) -> &JobList;
|
||
unsafe fn if_clause(self: &ElseifClause) -> &IfClause;
|
||
unsafe fn semi_nl(self: &ElseClause) -> &SemiNl;
|
||
unsafe fn body(self: &ElseClause) -> &JobList;
|
||
unsafe fn if_clause(self: &IfStatement) -> &IfClause;
|
||
unsafe fn elseif_clauses(self: &IfStatement) -> &ElseifClauseList;
|
||
fn has_else_clause(self: &IfStatement) -> bool;
|
||
unsafe fn else_clause(self: &IfStatement) -> &ElseClause;
|
||
unsafe fn end(self: &IfStatement) -> &KeywordEnd;
|
||
unsafe fn args_or_redirs(self: &IfStatement) -> &ArgumentOrRedirectionList;
|
||
unsafe fn arguments(self: &CaseItem) -> &ArgumentList;
|
||
unsafe fn semi_nl(self: &CaseItem) -> &SemiNl;
|
||
unsafe fn body(self: &CaseItem) -> &JobList;
|
||
unsafe fn argument(self: &SwitchStatement) -> &Argument;
|
||
unsafe fn semi_nl(self: &SwitchStatement) -> &SemiNl;
|
||
unsafe fn cases(self: &SwitchStatement) -> &CaseItemList;
|
||
unsafe fn end(self: &SwitchStatement) -> &KeywordEnd;
|
||
unsafe fn args_or_redirs(self: &SwitchStatement) -> &ArgumentOrRedirectionList;
|
||
fn has_opt_decoration(self: &DecoratedStatement) -> bool;
|
||
unsafe fn opt_decoration(self: &DecoratedStatement) -> &DecoratedStatementDecorator;
|
||
unsafe fn command(self: &DecoratedStatement) -> &String_;
|
||
unsafe fn args_or_redirs(self: &DecoratedStatement) -> &ArgumentOrRedirectionList;
|
||
unsafe fn variables(self: &NotStatement) -> &VariableAssignmentList;
|
||
fn has_time(self: &NotStatement) -> bool;
|
||
unsafe fn time(self: &NotStatement) -> &KeywordTime;
|
||
unsafe fn contents(self: &NotStatement) -> &Statement;
|
||
unsafe fn pipe(self: &JobContinuation) -> &TokenPipe;
|
||
unsafe fn newlines(self: &JobContinuation) -> &MaybeNewlines;
|
||
unsafe fn variables(self: &JobContinuation) -> &VariableAssignmentList;
|
||
unsafe fn statement(self: &JobContinuation) -> &Statement;
|
||
unsafe fn conjunction(self: &JobConjunctionContinuation) -> &TokenConjunction;
|
||
unsafe fn newlines(self: &JobConjunctionContinuation) -> &MaybeNewlines;
|
||
unsafe fn job(self: &JobConjunctionContinuation) -> &JobPipeline;
|
||
unsafe fn job(self: &AndorJob) -> &JobConjunction;
|
||
unsafe fn arguments(self: &FreestandingArgumentList) -> &ArgumentList;
|
||
unsafe fn kw_begin(self: &BeginHeader) -> &KeywordBegin;
|
||
unsafe fn end(self: &BlockStatement) -> &KeywordEnd;
|
||
}
|
||
|
||
#[rustfmt::skip]
|
||
extern "Rust" {
|
||
#[cxx_name="source"] fn source_ffi(self: &NodeFfi<'_>, orig: &CxxWString) -> UniquePtr<CxxWString>;
|
||
#[cxx_name="source"] fn source_ffi(self: &Argument, orig: &CxxWString) -> UniquePtr<CxxWString>;
|
||
#[cxx_name="source"] fn source_ffi(self: &VariableAssignment, orig: &CxxWString) -> UniquePtr<CxxWString>;
|
||
#[cxx_name="source"] fn source_ffi(self: &String_, orig: &CxxWString) -> UniquePtr<CxxWString>;
|
||
#[cxx_name="source"] fn source_ffi(self: &TokenRedirection, orig: &CxxWString) -> UniquePtr<CxxWString>;
|
||
|
||
#[cxx_name = "try_source_range"] unsafe fn try_source_range_ffi(self: &NodeFfi) -> bool;
|
||
#[cxx_name = "try_source_range"] fn try_source_range_ffi(self: &Argument) -> bool;
|
||
#[cxx_name = "try_source_range"] fn try_source_range_ffi(self: &JobPipeline) -> bool;
|
||
#[cxx_name = "try_source_range"] fn try_source_range_ffi(self: &String_) -> bool;
|
||
#[cxx_name = "try_source_range"] fn try_source_range_ffi(self: &BlockStatement) -> bool;
|
||
#[cxx_name = "try_source_range"] fn try_source_range_ffi(self: &KeywordEnd) -> bool;
|
||
#[cxx_name = "try_source_range"] fn try_source_range_ffi(self: &VariableAssignment) -> bool;
|
||
#[cxx_name = "try_source_range"] fn try_source_range_ffi(self: &SemiNl) -> bool;
|
||
|
||
#[cxx_name = "source_range"] unsafe fn source_range_ffi(self: &NodeFfi) -> SourceRange;
|
||
#[cxx_name = "source_range"] fn source_range_ffi(self: &JobConjunctionDecorator) -> SourceRange;
|
||
#[cxx_name = "source_range"] fn source_range_ffi(self: &DecoratedStatement) -> SourceRange;
|
||
#[cxx_name = "source_range"] fn source_range_ffi(self: &Argument) -> SourceRange;
|
||
#[cxx_name = "source_range"] fn source_range_ffi(self: &JobPipeline) -> SourceRange;
|
||
#[cxx_name = "source_range"] fn source_range_ffi(self: &String_) -> SourceRange;
|
||
#[cxx_name = "source_range"] fn source_range_ffi(self: &BlockStatement) -> SourceRange;
|
||
#[cxx_name = "source_range"] fn source_range_ffi(self: &KeywordEnd) -> SourceRange;
|
||
#[cxx_name = "source_range"] fn source_range_ffi(self: &VariableAssignment) -> SourceRange;
|
||
#[cxx_name = "source_range"] fn source_range_ffi(self: &SemiNl) -> SourceRange;
|
||
}
|
||
|
||
#[rustfmt::skip]
|
||
extern "Rust" {
|
||
unsafe fn try_as_block_statement(self: & StatementVariant) -> *const BlockStatement;
|
||
unsafe fn try_as_if_statement(self: & StatementVariant) -> *const IfStatement;
|
||
unsafe fn try_as_switch_statement(self: & StatementVariant) -> *const SwitchStatement;
|
||
unsafe fn try_as_decorated_statement(self: & StatementVariant) -> *const DecoratedStatement;
|
||
}
|
||
|
||
#[rustfmt::skip]
|
||
extern "Rust" {
|
||
unsafe fn try_as_argument(self: &NodeFfi) -> *const Argument;
|
||
unsafe fn try_as_begin_header(self: &NodeFfi) -> *const BeginHeader;
|
||
unsafe fn try_as_block_statement(self: &NodeFfi) -> *const BlockStatement;
|
||
unsafe fn try_as_decorated_statement(self: &NodeFfi) -> *const DecoratedStatement;
|
||
unsafe fn try_as_for_header(self: &NodeFfi) -> *const ForHeader;
|
||
unsafe fn try_as_function_header(self: &NodeFfi) -> *const FunctionHeader;
|
||
unsafe fn try_as_if_clause(self: &NodeFfi) -> *const IfClause;
|
||
unsafe fn try_as_if_statement(self: &NodeFfi) -> *const IfStatement;
|
||
unsafe fn try_as_job_conjunction(self: &NodeFfi) -> *const JobConjunction;
|
||
unsafe fn try_as_job_conjunction_continuation(self: &NodeFfi) -> *const JobConjunctionContinuation;
|
||
unsafe fn try_as_job_continuation(self: &NodeFfi) -> *const JobContinuation;
|
||
unsafe fn try_as_job_list(self: &NodeFfi) -> *const JobList;
|
||
unsafe fn try_as_job_pipeline(self: &NodeFfi) -> *const JobPipeline;
|
||
unsafe fn try_as_not_statement(self: &NodeFfi) -> *const NotStatement;
|
||
unsafe fn try_as_switch_statement(self: &NodeFfi) -> *const SwitchStatement;
|
||
unsafe fn try_as_while_header(self: &NodeFfi) -> *const WhileHeader;
|
||
}
|
||
|
||
#[rustfmt::skip]
|
||
extern "Rust" {
|
||
unsafe fn as_if_clause<'a>(self: &'a NodeFfi<'a>) -> &'a IfClause;
|
||
unsafe fn as_job_conjunction<'a>(self: &'a NodeFfi) -> &'a JobConjunction;
|
||
unsafe fn as_job_pipeline<'a>(self: &'a NodeFfi<'a>) -> &'a JobPipeline;
|
||
unsafe fn as_argument<'a>(self: &'a NodeFfi<'a>) -> &'a Argument;
|
||
unsafe fn as_begin_header<'a>(self: &'a NodeFfi<'a>) -> &'a BeginHeader;
|
||
unsafe fn as_block_statement<'a>(self: &'a NodeFfi<'a>) -> &'a BlockStatement;
|
||
unsafe fn as_decorated_statement<'a>(self: &'a NodeFfi<'a>) -> &'a DecoratedStatement;
|
||
unsafe fn as_for_header<'a>(self: &'a NodeFfi<'a>) -> &'a ForHeader;
|
||
unsafe fn as_freestanding_argument_list<'a>(self: &'a NodeFfi<'a>) -> &'a FreestandingArgumentList;
|
||
unsafe fn as_function_header<'a>(self: &'a NodeFfi<'a>) -> &'a FunctionHeader;
|
||
unsafe fn as_if_statement<'a>(self: &'a NodeFfi<'a>) -> &'a IfStatement;
|
||
unsafe fn as_job_conjunction_continuation<'a>(self: &'a NodeFfi<'a>) -> &'a JobConjunctionContinuation;
|
||
unsafe fn as_job_continuation<'a>(self: &'a NodeFfi<'a>) -> &'a JobContinuation;
|
||
unsafe fn as_job_list<'a>(self: &'a NodeFfi<'a>) -> &'a JobList;
|
||
unsafe fn as_not_statement<'a>(self: &'a NodeFfi<'a>) -> &'a NotStatement;
|
||
unsafe fn as_redirection<'a>(self: &'a NodeFfi<'a>) -> &'a Redirection;
|
||
unsafe fn as_statement<'a>(self: &'a NodeFfi<'a>) -> &'a Statement;
|
||
unsafe fn as_switch_statement<'a>(self: &'a NodeFfi<'a>) -> &'a SwitchStatement;
|
||
unsafe fn as_while_header<'a>(self: &'a NodeFfi<'a>) -> &'a WhileHeader;
|
||
}
|
||
|
||
#[rustfmt::skip]
|
||
extern "Rust" {
|
||
unsafe fn ptr(self: &StatementVariant) -> Box<NodeFfi<'_>>;
|
||
unsafe fn ptr(self: &BlockStatementHeaderVariant) -> Box<NodeFfi<'_>>;
|
||
unsafe fn ptr(self: &AndorJobList) -> Box<NodeFfi<'_>>;
|
||
unsafe fn ptr(self: &AndorJob) -> Box<NodeFfi<'_>>;
|
||
unsafe fn ptr(self: &ArgumentList) -> Box<NodeFfi<'_>>;
|
||
unsafe fn ptr(self: &Argument) -> Box<NodeFfi<'_>>;
|
||
unsafe fn ptr(self: &ArgumentOrRedirectionList) -> Box<NodeFfi<'_>>;
|
||
unsafe fn ptr(self: &ArgumentOrRedirection) -> Box<NodeFfi<'_>>;
|
||
unsafe fn ptr(self: &BeginHeader) -> Box<NodeFfi<'_>>;
|
||
unsafe fn ptr(self: &BlockStatement) -> Box<NodeFfi<'_>>;
|
||
unsafe fn ptr(self: &CaseItemList) -> Box<NodeFfi<'_>>;
|
||
unsafe fn ptr(self: &CaseItem) -> Box<NodeFfi<'_>>;
|
||
unsafe fn ptr(self: &DecoratedStatementDecorator) -> Box<NodeFfi<'_>>;
|
||
unsafe fn ptr(self: &DecoratedStatement) -> Box<NodeFfi<'_>>;
|
||
unsafe fn ptr(self: &ElseClause) -> Box<NodeFfi<'_>>;
|
||
unsafe fn ptr(self: &ElseifClauseList) -> Box<NodeFfi<'_>>;
|
||
unsafe fn ptr(self: &ElseifClause) -> Box<NodeFfi<'_>>;
|
||
unsafe fn ptr(self: &ForHeader) -> Box<NodeFfi<'_>>;
|
||
unsafe fn ptr(self: &FreestandingArgumentList) -> Box<NodeFfi<'_>>;
|
||
unsafe fn ptr(self: &FunctionHeader) -> Box<NodeFfi<'_>>;
|
||
unsafe fn ptr(self: &IfClause) -> Box<NodeFfi<'_>>;
|
||
unsafe fn ptr(self: &IfStatement) -> Box<NodeFfi<'_>>;
|
||
unsafe fn ptr(self: &JobConjunctionContinuationList) -> Box<NodeFfi<'_>>;
|
||
unsafe fn ptr(self: &JobConjunctionContinuation) -> Box<NodeFfi<'_>>;
|
||
unsafe fn ptr(self: &JobConjunctionDecorator) -> Box<NodeFfi<'_>>;
|
||
unsafe fn ptr(self: &JobConjunction) -> Box<NodeFfi<'_>>;
|
||
unsafe fn ptr(self: &JobContinuationList) -> Box<NodeFfi<'_>>;
|
||
unsafe fn ptr(self: &JobContinuation) -> Box<NodeFfi<'_>>;
|
||
unsafe fn ptr(self: &JobList) -> Box<NodeFfi<'_>>;
|
||
unsafe fn ptr(self: &JobPipeline) -> Box<NodeFfi<'_>>;
|
||
unsafe fn ptr(self: &KeywordBegin) -> Box<NodeFfi<'_>>;
|
||
unsafe fn ptr(self: &KeywordCase) -> Box<NodeFfi<'_>>;
|
||
unsafe fn ptr(self: &KeywordElse) -> Box<NodeFfi<'_>>;
|
||
unsafe fn ptr(self: &KeywordEnd) -> Box<NodeFfi<'_>>;
|
||
unsafe fn ptr(self: &KeywordFor) -> Box<NodeFfi<'_>>;
|
||
unsafe fn ptr(self: &KeywordFunction) -> Box<NodeFfi<'_>>;
|
||
unsafe fn ptr(self: &KeywordIf) -> Box<NodeFfi<'_>>;
|
||
unsafe fn ptr(self: &KeywordIn) -> Box<NodeFfi<'_>>;
|
||
unsafe fn ptr(self: &KeywordNot) -> Box<NodeFfi<'_>>;
|
||
unsafe fn ptr(self: &KeywordTime) -> Box<NodeFfi<'_>>;
|
||
unsafe fn ptr(self: &KeywordWhile) -> Box<NodeFfi<'_>>;
|
||
unsafe fn ptr(self: &MaybeNewlines) -> Box<NodeFfi<'_>>;
|
||
unsafe fn ptr(self: &NotStatement) -> Box<NodeFfi<'_>>;
|
||
unsafe fn ptr(self: &Redirection) -> Box<NodeFfi<'_>>;
|
||
unsafe fn ptr(self: &SemiNl) -> Box<NodeFfi<'_>>;
|
||
unsafe fn ptr(self: &Statement) -> Box<NodeFfi<'_>>;
|
||
unsafe fn ptr(self: &String_) -> Box<NodeFfi<'_>>;
|
||
unsafe fn ptr(self: &SwitchStatement) -> Box<NodeFfi<'_>>;
|
||
unsafe fn ptr(self: &TokenBackground) -> Box<NodeFfi<'_>>;
|
||
unsafe fn ptr(self: &TokenConjunction) -> Box<NodeFfi<'_>>;
|
||
unsafe fn ptr(self: &TokenPipe) -> Box<NodeFfi<'_>>;
|
||
unsafe fn ptr(self: &TokenRedirection) -> Box<NodeFfi<'_>>;
|
||
unsafe fn ptr(self: &VariableAssignmentList) -> Box<NodeFfi<'_>>;
|
||
unsafe fn ptr(self: &VariableAssignment) -> Box<NodeFfi<'_>>;
|
||
unsafe fn ptr(self: &WhileHeader) -> Box<NodeFfi<'_>>;
|
||
}
|
||
|
||
#[rustfmt::skip]
|
||
extern "Rust" {
|
||
unsafe fn range(self: &VariableAssignment) -> SourceRange;
|
||
unsafe fn range(self: &TokenConjunction) -> SourceRange;
|
||
unsafe fn range(self: &MaybeNewlines) -> SourceRange;
|
||
unsafe fn range(self: &TokenPipe) -> SourceRange;
|
||
unsafe fn range(self: &KeywordNot) -> SourceRange;
|
||
unsafe fn range(self: &DecoratedStatementDecorator) -> SourceRange;
|
||
unsafe fn range(self: &KeywordEnd) -> SourceRange;
|
||
unsafe fn range(self: &KeywordCase) -> SourceRange;
|
||
unsafe fn range(self: &KeywordElse) -> SourceRange;
|
||
unsafe fn range(self: &KeywordIf) -> SourceRange;
|
||
unsafe fn range(self: &KeywordBegin) -> SourceRange;
|
||
unsafe fn range(self: &KeywordFunction) -> SourceRange;
|
||
unsafe fn range(self: &KeywordWhile) -> SourceRange;
|
||
unsafe fn range(self: &KeywordFor) -> SourceRange;
|
||
unsafe fn range(self: &KeywordIn) -> SourceRange;
|
||
unsafe fn range(self: &SemiNl) -> SourceRange;
|
||
unsafe fn range(self: &JobConjunctionDecorator) -> SourceRange;
|
||
unsafe fn range(self: &TokenBackground) -> SourceRange;
|
||
unsafe fn range(self: &KeywordTime) -> SourceRange;
|
||
unsafe fn range(self: &TokenRedirection) -> SourceRange;
|
||
unsafe fn range(self: &String_) -> SourceRange;
|
||
unsafe fn range(self: &Argument) -> SourceRange;
|
||
}
|
||
}
|
||
|
||
impl Ast {
|
||
fn extras_ffi(self: &Ast) -> Box<ExtrasFFI<'_>> {
|
||
Box::new(ExtrasFFI(&self.extras))
|
||
}
|
||
}
|
||
|
||
struct ExtrasFFI<'a>(&'a Extras);
|
||
|
||
impl<'a> ExtrasFFI<'a> {
|
||
fn comments(&self) -> &'a [SourceRange] {
|
||
&self.0.comments
|
||
}
|
||
fn semis(&self) -> &'a [SourceRange] {
|
||
&self.0.semis
|
||
}
|
||
fn errors(&self) -> &'a [SourceRange] {
|
||
&self.0.errors
|
||
}
|
||
}
|
||
|
||
unsafe impl ExternType for Ast {
|
||
type Id = type_id!("Ast");
|
||
type Kind = cxx::kind::Opaque;
|
||
}
|
||
|
||
impl Ast {
|
||
fn top_ffi(&self) -> Box<NodeFfi> {
|
||
Box::new(NodeFfi::new(self.top.as_node()))
|
||
}
|
||
fn dump_ffi(&self, orig: &CxxWString) -> UniquePtr<CxxWString> {
|
||
self.dump(orig.as_wstr()).to_ffi()
|
||
}
|
||
}
|
||
|
||
fn ast_parse_ffi(src: &CxxWString, flags: u8, errors: *mut ParseErrorList) -> Box<Ast> {
|
||
let mut out_errors: Option<ParseErrorList> = if errors.is_null() {
|
||
None
|
||
} else {
|
||
Some(unsafe { &*errors }.clone())
|
||
};
|
||
let ast = Box::new(Ast::parse(
|
||
src.as_wstr(),
|
||
ParseTreeFlags(flags),
|
||
&mut out_errors,
|
||
));
|
||
if let Some(out_errors) = out_errors {
|
||
unsafe { *errors = out_errors };
|
||
}
|
||
ast
|
||
}
|
||
|
||
fn ast_parse_argument_list_ffi(
|
||
src: &CxxWString,
|
||
flags: u8,
|
||
errors: *mut ParseErrorList,
|
||
) -> Box<Ast> {
|
||
let mut out_errors: Option<ParseErrorList> = if errors.is_null() {
|
||
None
|
||
} else {
|
||
Some(unsafe { &*errors }.clone())
|
||
};
|
||
let ast = Box::new(Ast::parse_argument_list(
|
||
src.as_wstr(),
|
||
ParseTreeFlags(flags),
|
||
&mut out_errors,
|
||
));
|
||
if let Some(out_errors) = out_errors {
|
||
unsafe { *errors = out_errors };
|
||
}
|
||
ast
|
||
}
|
||
|
||
fn new_ast_traversal<'a>(root: &'a NodeFfi<'a>) -> Box<Traversal<'a>> {
|
||
Box::new(Traversal::new(root.as_node()))
|
||
}
|
||
|
||
impl<'a> Traversal<'a> {
|
||
fn next_ffi(&mut self) -> Box<NodeFfi<'a>> {
|
||
Box::new(match self.next() {
|
||
Some(node) => NodeFfi::Reference(node),
|
||
None => NodeFfi::None,
|
||
})
|
||
}
|
||
}
|
||
|
||
impl TokenConjunction {
|
||
fn token_type(&self) -> ParseTokenType {
|
||
(self as &dyn Token).token_type()
|
||
}
|
||
}
|
||
|
||
impl Statement {
|
||
fn contents(&self) -> &StatementVariant {
|
||
&self.contents
|
||
}
|
||
}
|
||
|
||
#[derive(Clone)]
|
||
pub enum NodeFfi<'a> {
|
||
None,
|
||
Reference(&'a dyn Node),
|
||
Pointer(*const dyn Node),
|
||
}
|
||
|
||
unsafe impl ExternType for NodeFfi<'_> {
|
||
type Id = type_id!("NodeFfi");
|
||
type Kind = cxx::kind::Opaque;
|
||
}
|
||
|
||
impl<'a> NodeFfi<'a> {
|
||
pub fn new(node: &'a dyn Node) -> Self {
|
||
NodeFfi::Reference(node)
|
||
}
|
||
fn has_value(&self) -> bool {
|
||
!matches!(self, NodeFfi::None)
|
||
}
|
||
pub fn as_node(&self) -> &dyn Node {
|
||
match *self {
|
||
NodeFfi::None => panic!("tried to dereference null node"),
|
||
NodeFfi::Reference(node) => node,
|
||
NodeFfi::Pointer(node) => unsafe { &*node },
|
||
}
|
||
}
|
||
fn parent_ffi(&self) -> Box<NodeFfi<'a>> {
|
||
Box::new(match *self.as_node().parent_ffi() {
|
||
Some(node) => NodeFfi::Pointer(node),
|
||
None => NodeFfi::None,
|
||
})
|
||
}
|
||
fn category(&self) -> Category {
|
||
self.as_node().category()
|
||
}
|
||
fn typ(&self) -> Type {
|
||
self.as_node().typ()
|
||
}
|
||
fn describe(&self) -> UniquePtr<CxxWString> {
|
||
self.as_node().describe().to_ffi()
|
||
}
|
||
// Pointer comparison
|
||
fn pointer_eq(&self, rhs: &NodeFfi) -> bool {
|
||
std::ptr::eq(self.as_node().as_ptr(), rhs.as_node().as_ptr())
|
||
}
|
||
fn kw(&self) -> ParseKeyword {
|
||
self.as_node().as_keyword().unwrap().keyword()
|
||
}
|
||
fn token_type(&self) -> ParseTokenType {
|
||
self.as_node().as_token().unwrap().token_type()
|
||
}
|
||
fn has_source(&self) -> bool {
|
||
self.as_node().as_leaf().unwrap().range().is_some()
|
||
}
|
||
fn ptr(&self) -> Box<NodeFfi<'_>> {
|
||
Box::new(self.clone())
|
||
}
|
||
fn try_source_range_ffi(&self) -> bool {
|
||
self.as_node().try_source_range().is_some()
|
||
}
|
||
fn source_range_ffi(&self) -> SourceRange {
|
||
self.as_node().source_range()
|
||
}
|
||
fn source_ffi(&self, orig: &CxxWString) -> UniquePtr<CxxWString> {
|
||
self.as_node().source(orig.as_wstr()).to_ffi()
|
||
}
|
||
}
|
||
|
||
impl Argument {
|
||
fn source_ffi(&self, orig: &CxxWString) -> UniquePtr<CxxWString> {
|
||
self.source(orig.as_wstr()).to_ffi()
|
||
}
|
||
}
|
||
impl VariableAssignment {
|
||
fn source_ffi(&self, orig: &CxxWString) -> UniquePtr<CxxWString> {
|
||
self.source(orig.as_wstr()).to_ffi()
|
||
}
|
||
}
|
||
impl String_ {
|
||
fn source_ffi(&self, orig: &CxxWString) -> UniquePtr<CxxWString> {
|
||
self.source(orig.as_wstr()).to_ffi()
|
||
}
|
||
}
|
||
impl TokenRedirection {
|
||
fn source_ffi(&self, orig: &CxxWString) -> UniquePtr<CxxWString> {
|
||
self.source(orig.as_wstr()).to_ffi()
|
||
}
|
||
}
|
||
|
||
impl ArgumentList {
|
||
fn count(&self) -> usize {
|
||
<dyn List<ContentsNode = Argument>>::count(self)
|
||
}
|
||
fn empty(&self) -> bool {
|
||
self.is_empty()
|
||
}
|
||
fn at(&self, i: usize) -> *const Argument {
|
||
&self[i]
|
||
}
|
||
}
|
||
|
||
impl ArgumentOrRedirectionList {
|
||
fn count(&self) -> usize {
|
||
<dyn List<ContentsNode = ArgumentOrRedirection>>::count(self)
|
||
}
|
||
fn empty(&self) -> bool {
|
||
self.is_empty()
|
||
}
|
||
fn at(&self, i: usize) -> *const ArgumentOrRedirection {
|
||
if i >= self.count() {
|
||
std::ptr::null()
|
||
} else {
|
||
&self[i]
|
||
}
|
||
}
|
||
}
|
||
|
||
impl JobList {
|
||
fn count(&self) -> usize {
|
||
<dyn List<ContentsNode = JobConjunction>>::count(self)
|
||
}
|
||
fn empty(&self) -> bool {
|
||
self.is_empty()
|
||
}
|
||
fn at(&self, i: usize) -> *const JobConjunction {
|
||
if i >= self.count() {
|
||
std::ptr::null()
|
||
} else {
|
||
&self[i]
|
||
}
|
||
}
|
||
}
|
||
|
||
impl JobContinuationList {
|
||
fn count(&self) -> usize {
|
||
<dyn List<ContentsNode = JobContinuation>>::count(self)
|
||
}
|
||
fn empty(&self) -> bool {
|
||
self.is_empty()
|
||
}
|
||
fn at(&self, i: usize) -> *const JobContinuation {
|
||
if i >= self.count() {
|
||
std::ptr::null()
|
||
} else {
|
||
&self[i]
|
||
}
|
||
}
|
||
}
|
||
|
||
impl ElseifClauseList {
|
||
fn count(&self) -> usize {
|
||
<dyn List<ContentsNode = ElseifClause>>::count(self)
|
||
}
|
||
fn empty(&self) -> bool {
|
||
self.is_empty()
|
||
}
|
||
fn at(&self, i: usize) -> *const ElseifClause {
|
||
if i >= self.count() {
|
||
std::ptr::null()
|
||
} else {
|
||
&self[i]
|
||
}
|
||
}
|
||
}
|
||
|
||
impl CaseItemList {
|
||
fn count(&self) -> usize {
|
||
<dyn List<ContentsNode = CaseItem>>::count(self)
|
||
}
|
||
fn empty(&self) -> bool {
|
||
self.is_empty()
|
||
}
|
||
fn at(&self, i: usize) -> *const CaseItem {
|
||
if i >= self.count() {
|
||
std::ptr::null()
|
||
} else {
|
||
&self[i]
|
||
}
|
||
}
|
||
}
|
||
|
||
impl VariableAssignmentList {
|
||
fn count(&self) -> usize {
|
||
<dyn List<ContentsNode = VariableAssignment>>::count(self)
|
||
}
|
||
fn empty(&self) -> bool {
|
||
self.is_empty()
|
||
}
|
||
fn at(&self, i: usize) -> *const VariableAssignment {
|
||
if i >= self.count() {
|
||
std::ptr::null()
|
||
} else {
|
||
&self[i]
|
||
}
|
||
}
|
||
}
|
||
|
||
impl JobConjunctionContinuationList {
|
||
fn count(&self) -> usize {
|
||
<dyn List<ContentsNode = JobConjunctionContinuation>>::count(self)
|
||
}
|
||
fn empty(&self) -> bool {
|
||
self.is_empty()
|
||
}
|
||
fn at(&self, i: usize) -> *const JobConjunctionContinuation {
|
||
if i >= self.count() {
|
||
std::ptr::null()
|
||
} else {
|
||
&self[i]
|
||
}
|
||
}
|
||
}
|
||
|
||
impl AndorJobList {
|
||
fn count(&self) -> usize {
|
||
<dyn List<ContentsNode = AndorJob>>::count(self)
|
||
}
|
||
fn empty(&self) -> bool {
|
||
self.is_empty()
|
||
}
|
||
fn at(&self, i: usize) -> *const AndorJob {
|
||
if i >= self.count() {
|
||
std::ptr::null()
|
||
} else {
|
||
&self[i]
|
||
}
|
||
}
|
||
}
|
||
|
||
impl Statement {
|
||
fn describe(&self) -> UniquePtr<CxxWString> {
|
||
(self as &dyn Node).describe().to_ffi()
|
||
}
|
||
}
|
||
|
||
impl JobConjunctionDecorator {
|
||
fn kw(&self) -> ParseKeyword {
|
||
self.keyword()
|
||
}
|
||
fn source_range_ffi(&self) -> SourceRange {
|
||
self.source_range()
|
||
}
|
||
}
|
||
impl DecoratedStatement {
|
||
fn source_range_ffi(&self) -> SourceRange {
|
||
self.source_range()
|
||
}
|
||
}
|
||
|
||
impl Argument {
|
||
fn try_source_range_ffi(&self) -> bool {
|
||
self.try_source_range().is_some()
|
||
}
|
||
fn source_range_ffi(&self) -> SourceRange {
|
||
self.source_range()
|
||
}
|
||
}
|
||
|
||
impl JobPipeline {
|
||
fn try_source_range_ffi(&self) -> bool {
|
||
self.try_source_range().is_some()
|
||
}
|
||
fn source_range_ffi(&self) -> SourceRange {
|
||
self.source_range()
|
||
}
|
||
}
|
||
|
||
impl String_ {
|
||
fn try_source_range_ffi(&self) -> bool {
|
||
self.try_source_range().is_some()
|
||
}
|
||
fn source_range_ffi(&self) -> SourceRange {
|
||
self.source_range()
|
||
}
|
||
}
|
||
|
||
impl BlockStatement {
|
||
fn try_source_range_ffi(&self) -> bool {
|
||
self.try_source_range().is_some()
|
||
}
|
||
fn source_range_ffi(&self) -> SourceRange {
|
||
self.source_range()
|
||
}
|
||
}
|
||
|
||
impl KeywordEnd {
|
||
fn try_source_range_ffi(&self) -> bool {
|
||
self.try_source_range().is_some()
|
||
}
|
||
fn source_range_ffi(&self) -> SourceRange {
|
||
self.source_range()
|
||
}
|
||
}
|
||
|
||
impl VariableAssignment {
|
||
fn try_source_range_ffi(&self) -> bool {
|
||
self.try_source_range().is_some()
|
||
}
|
||
fn source_range_ffi(&self) -> SourceRange {
|
||
self.source_range()
|
||
}
|
||
}
|
||
|
||
impl SemiNl {
|
||
fn try_source_range_ffi(&self) -> bool {
|
||
self.try_source_range().is_some()
|
||
}
|
||
fn source_range_ffi(&self) -> SourceRange {
|
||
self.source_range()
|
||
}
|
||
}
|
||
|
||
impl Redirection {
|
||
fn oper(&self) -> &TokenRedirection {
|
||
&self.oper
|
||
}
|
||
}
|
||
impl Redirection {
|
||
fn target(&self) -> &String_ {
|
||
&self.target
|
||
}
|
||
}
|
||
impl ArgumentOrRedirection {
|
||
fn argument_ffi(&self) -> &Argument {
|
||
self.argument()
|
||
}
|
||
}
|
||
impl ArgumentOrRedirection {
|
||
fn redirection_ffi(&self) -> &Redirection {
|
||
self.redirection()
|
||
}
|
||
}
|
||
impl JobPipeline {
|
||
fn has_time(&self) -> bool {
|
||
self.time.is_some()
|
||
}
|
||
}
|
||
impl JobPipeline {
|
||
fn time(&self) -> &KeywordTime {
|
||
self.time.as_ref().unwrap()
|
||
}
|
||
}
|
||
impl JobPipeline {
|
||
fn variables(&self) -> &VariableAssignmentList {
|
||
&self.variables
|
||
}
|
||
}
|
||
impl JobPipeline {
|
||
fn statement(&self) -> &Statement {
|
||
&self.statement
|
||
}
|
||
}
|
||
impl JobPipeline {
|
||
fn continuation(&self) -> &JobContinuationList {
|
||
&self.continuation
|
||
}
|
||
}
|
||
impl JobPipeline {
|
||
fn has_bg(&self) -> bool {
|
||
self.bg.is_some()
|
||
}
|
||
}
|
||
impl JobPipeline {
|
||
fn bg(&self) -> &TokenBackground {
|
||
self.bg.as_ref().unwrap()
|
||
}
|
||
}
|
||
impl JobConjunction {
|
||
fn has_decorator(&self) -> bool {
|
||
self.decorator.is_some()
|
||
}
|
||
}
|
||
impl JobConjunction {
|
||
fn decorator(&self) -> &JobConjunctionDecorator {
|
||
self.decorator.as_ref().unwrap()
|
||
}
|
||
}
|
||
impl JobConjunction {
|
||
fn job(&self) -> &JobPipeline {
|
||
&self.job
|
||
}
|
||
}
|
||
impl JobConjunction {
|
||
fn continuations(&self) -> &JobConjunctionContinuationList {
|
||
&self.continuations
|
||
}
|
||
}
|
||
impl JobConjunction {
|
||
fn has_semi_nl(&self) -> bool {
|
||
self.semi_nl.is_some()
|
||
}
|
||
}
|
||
impl JobConjunction {
|
||
fn semi_nl(&self) -> &SemiNl {
|
||
self.semi_nl.as_ref().unwrap()
|
||
}
|
||
}
|
||
impl ForHeader {
|
||
fn var_name(&self) -> &String_ {
|
||
&self.var_name
|
||
}
|
||
}
|
||
impl ForHeader {
|
||
fn args(&self) -> &ArgumentList {
|
||
&self.args
|
||
}
|
||
}
|
||
impl ForHeader {
|
||
fn semi_nl(&self) -> &SemiNl {
|
||
&self.semi_nl
|
||
}
|
||
}
|
||
impl WhileHeader {
|
||
fn condition(&self) -> &JobConjunction {
|
||
&self.condition
|
||
}
|
||
}
|
||
impl WhileHeader {
|
||
fn andor_tail(&self) -> &AndorJobList {
|
||
&self.andor_tail
|
||
}
|
||
}
|
||
impl FunctionHeader {
|
||
fn first_arg(&self) -> &Argument {
|
||
&self.first_arg
|
||
}
|
||
}
|
||
impl FunctionHeader {
|
||
fn args(&self) -> &ArgumentList {
|
||
&self.args
|
||
}
|
||
}
|
||
impl FunctionHeader {
|
||
fn semi_nl(&self) -> &SemiNl {
|
||
&self.semi_nl
|
||
}
|
||
}
|
||
impl BeginHeader {
|
||
fn has_semi_nl(&self) -> bool {
|
||
self.semi_nl.is_some()
|
||
}
|
||
}
|
||
impl BeginHeader {
|
||
fn semi_nl(&self) -> &SemiNl {
|
||
self.semi_nl.as_ref().unwrap()
|
||
}
|
||
}
|
||
impl BlockStatement {
|
||
fn header(&self) -> &BlockStatementHeaderVariant {
|
||
&self.header
|
||
}
|
||
}
|
||
impl BlockStatement {
|
||
fn jobs(&self) -> &JobList {
|
||
&self.jobs
|
||
}
|
||
}
|
||
impl BlockStatement {
|
||
fn args_or_redirs(&self) -> &ArgumentOrRedirectionList {
|
||
&self.args_or_redirs
|
||
}
|
||
}
|
||
impl IfClause {
|
||
fn condition(&self) -> &JobConjunction {
|
||
&self.condition
|
||
}
|
||
}
|
||
impl IfClause {
|
||
fn andor_tail(&self) -> &AndorJobList {
|
||
&self.andor_tail
|
||
}
|
||
}
|
||
impl IfClause {
|
||
fn body(&self) -> &JobList {
|
||
&self.body
|
||
}
|
||
}
|
||
impl ElseifClause {
|
||
fn if_clause(&self) -> &IfClause {
|
||
&self.if_clause
|
||
}
|
||
}
|
||
impl ElseClause {
|
||
fn semi_nl(&self) -> &SemiNl {
|
||
&self.semi_nl
|
||
}
|
||
}
|
||
impl ElseClause {
|
||
fn body(&self) -> &JobList {
|
||
&self.body
|
||
}
|
||
}
|
||
impl IfStatement {
|
||
fn if_clause(&self) -> &IfClause {
|
||
&self.if_clause
|
||
}
|
||
}
|
||
impl IfStatement {
|
||
fn elseif_clauses(&self) -> &ElseifClauseList {
|
||
&self.elseif_clauses
|
||
}
|
||
}
|
||
impl IfStatement {
|
||
fn has_else_clause(&self) -> bool {
|
||
self.else_clause.is_some()
|
||
}
|
||
}
|
||
impl IfStatement {
|
||
fn else_clause(&self) -> &ElseClause {
|
||
self.else_clause.as_ref().unwrap()
|
||
}
|
||
}
|
||
impl IfStatement {
|
||
fn end(&self) -> &KeywordEnd {
|
||
&self.end
|
||
}
|
||
}
|
||
impl IfStatement {
|
||
fn args_or_redirs(&self) -> &ArgumentOrRedirectionList {
|
||
&self.args_or_redirs
|
||
}
|
||
}
|
||
impl CaseItem {
|
||
fn arguments(&self) -> &ArgumentList {
|
||
&self.arguments
|
||
}
|
||
}
|
||
impl CaseItem {
|
||
fn semi_nl(&self) -> &SemiNl {
|
||
&self.semi_nl
|
||
}
|
||
}
|
||
impl CaseItem {
|
||
fn body(&self) -> &JobList {
|
||
&self.body
|
||
}
|
||
}
|
||
impl SwitchStatement {
|
||
fn argument(&self) -> &Argument {
|
||
&self.argument
|
||
}
|
||
}
|
||
impl SwitchStatement {
|
||
fn semi_nl(&self) -> &SemiNl {
|
||
&self.semi_nl
|
||
}
|
||
}
|
||
impl SwitchStatement {
|
||
fn cases(&self) -> &CaseItemList {
|
||
&self.cases
|
||
}
|
||
}
|
||
impl SwitchStatement {
|
||
fn end(&self) -> &KeywordEnd {
|
||
&self.end
|
||
}
|
||
}
|
||
impl SwitchStatement {
|
||
fn args_or_redirs(&self) -> &ArgumentOrRedirectionList {
|
||
&self.args_or_redirs
|
||
}
|
||
}
|
||
impl DecoratedStatement {
|
||
fn has_opt_decoration(&self) -> bool {
|
||
self.opt_decoration.is_some()
|
||
}
|
||
}
|
||
impl DecoratedStatement {
|
||
fn opt_decoration(&self) -> &DecoratedStatementDecorator {
|
||
self.opt_decoration.as_ref().unwrap()
|
||
}
|
||
}
|
||
impl DecoratedStatement {
|
||
fn command(&self) -> &String_ {
|
||
&self.command
|
||
}
|
||
}
|
||
impl DecoratedStatement {
|
||
fn args_or_redirs(&self) -> &ArgumentOrRedirectionList {
|
||
&self.args_or_redirs
|
||
}
|
||
}
|
||
impl NotStatement {
|
||
fn variables(&self) -> &VariableAssignmentList {
|
||
&self.variables
|
||
}
|
||
}
|
||
impl NotStatement {
|
||
fn has_time(&self) -> bool {
|
||
self.time.is_some()
|
||
}
|
||
}
|
||
impl NotStatement {
|
||
fn time(&self) -> &KeywordTime {
|
||
self.time.as_ref().unwrap()
|
||
}
|
||
}
|
||
impl NotStatement {
|
||
fn contents(&self) -> &Statement {
|
||
&self.contents
|
||
}
|
||
}
|
||
impl JobContinuation {
|
||
fn pipe(&self) -> &TokenPipe {
|
||
&self.pipe
|
||
}
|
||
}
|
||
impl JobContinuation {
|
||
fn newlines(&self) -> &MaybeNewlines {
|
||
&self.newlines
|
||
}
|
||
}
|
||
impl JobContinuation {
|
||
fn variables(&self) -> &VariableAssignmentList {
|
||
&self.variables
|
||
}
|
||
}
|
||
impl JobContinuation {
|
||
fn statement(&self) -> &Statement {
|
||
&self.statement
|
||
}
|
||
}
|
||
impl JobConjunctionContinuation {
|
||
fn conjunction(&self) -> &TokenConjunction {
|
||
&self.conjunction
|
||
}
|
||
}
|
||
impl JobConjunctionContinuation {
|
||
fn newlines(&self) -> &MaybeNewlines {
|
||
&self.newlines
|
||
}
|
||
}
|
||
impl JobConjunctionContinuation {
|
||
fn job(&self) -> &JobPipeline {
|
||
&self.job
|
||
}
|
||
}
|
||
impl AndorJob {
|
||
fn job(&self) -> &JobConjunction {
|
||
&self.job
|
||
}
|
||
}
|
||
impl FreestandingArgumentList {
|
||
fn arguments(&self) -> &ArgumentList {
|
||
&self.arguments
|
||
}
|
||
}
|
||
impl BeginHeader {
|
||
fn kw_begin(&self) -> &KeywordBegin {
|
||
&self.kw_begin
|
||
}
|
||
}
|
||
impl BlockStatement {
|
||
fn end(&self) -> &KeywordEnd {
|
||
&self.end
|
||
}
|
||
}
|
||
|
||
impl StatementVariant {
|
||
fn try_as_not_statement(&self) -> *const NotStatement {
|
||
match self {
|
||
StatementVariant::NotStatement(node) => node,
|
||
_ => std::ptr::null(),
|
||
}
|
||
}
|
||
fn try_as_block_statement(&self) -> *const BlockStatement {
|
||
match self {
|
||
StatementVariant::BlockStatement(node) => node,
|
||
_ => std::ptr::null(),
|
||
}
|
||
}
|
||
fn try_as_if_statement(&self) -> *const IfStatement {
|
||
match self {
|
||
StatementVariant::IfStatement(node) => node,
|
||
_ => std::ptr::null(),
|
||
}
|
||
}
|
||
fn try_as_switch_statement(&self) -> *const SwitchStatement {
|
||
match self {
|
||
StatementVariant::SwitchStatement(node) => node,
|
||
_ => std::ptr::null(),
|
||
}
|
||
}
|
||
fn try_as_decorated_statement(&self) -> *const DecoratedStatement {
|
||
match self {
|
||
StatementVariant::DecoratedStatement(node) => node,
|
||
_ => std::ptr::null(),
|
||
}
|
||
}
|
||
}
|
||
|
||
#[rustfmt::skip]
|
||
impl NodeFfi<'_> {
|
||
fn try_as_argument(&self) -> *const Argument {
|
||
match self.as_node().as_argument() {
|
||
Some(node) => node,
|
||
None => std::ptr::null(),
|
||
}
|
||
}
|
||
fn try_as_begin_header(&self) -> *const BeginHeader {
|
||
match self.as_node().as_begin_header() {
|
||
Some(node) => node,
|
||
None => std::ptr::null(),
|
||
}
|
||
}
|
||
fn try_as_block_statement(&self) -> *const BlockStatement {
|
||
match self.as_node().as_block_statement() {
|
||
Some(node) => node,
|
||
None => std::ptr::null(),
|
||
}
|
||
}
|
||
fn try_as_decorated_statement(&self) -> *const DecoratedStatement {
|
||
match self.as_node().as_decorated_statement() {
|
||
Some(node) => node,
|
||
None => std::ptr::null(),
|
||
}
|
||
}
|
||
fn try_as_for_header(&self) -> *const ForHeader {
|
||
match self.as_node().as_for_header() {
|
||
Some(node) => node,
|
||
None => std::ptr::null(),
|
||
}
|
||
}
|
||
fn try_as_function_header(&self) -> *const FunctionHeader {
|
||
match self.as_node().as_function_header() {
|
||
Some(node) => node,
|
||
None => std::ptr::null(),
|
||
}
|
||
}
|
||
fn try_as_if_clause(&self) -> *const IfClause {
|
||
match self.as_node().as_if_clause() {
|
||
Some(node) => node,
|
||
None => std::ptr::null(),
|
||
}
|
||
}
|
||
fn try_as_if_statement(&self) -> *const IfStatement {
|
||
match self.as_node().as_if_statement() {
|
||
Some(node) => node,
|
||
None => std::ptr::null(),
|
||
}
|
||
}
|
||
fn try_as_job_conjunction(&self) -> *const JobConjunction {
|
||
match self.as_node().as_job_conjunction() {
|
||
Some(node) => node,
|
||
None => std::ptr::null(),
|
||
}
|
||
}
|
||
fn try_as_job_conjunction_continuation(&self) -> *const JobConjunctionContinuation {
|
||
match self.as_node().as_job_conjunction_continuation() {
|
||
Some(node) => node,
|
||
None => std::ptr::null(),
|
||
}
|
||
}
|
||
fn try_as_job_continuation(&self) -> *const JobContinuation {
|
||
match self.as_node().as_job_continuation() {
|
||
Some(node) => node,
|
||
None => std::ptr::null(),
|
||
}
|
||
}
|
||
fn try_as_job_list(&self) -> *const JobList {
|
||
match self.as_node().as_job_list() {
|
||
Some(node) => node,
|
||
None => std::ptr::null(),
|
||
}
|
||
}
|
||
fn try_as_job_pipeline(&self) -> *const JobPipeline {
|
||
match self.as_node().as_job_pipeline() {
|
||
Some(node) => node,
|
||
None => std::ptr::null(),
|
||
}
|
||
}
|
||
fn try_as_not_statement(&self) -> *const NotStatement {
|
||
match self.as_node().as_not_statement() {
|
||
Some(node) => node,
|
||
None => std::ptr::null(),
|
||
}
|
||
}
|
||
fn try_as_switch_statement(&self) -> *const SwitchStatement {
|
||
match self.as_node().as_switch_statement() {
|
||
Some(node) => node,
|
||
None => std::ptr::null(),
|
||
}
|
||
}
|
||
fn try_as_while_header(&self) -> *const WhileHeader {
|
||
match self.as_node().as_while_header() {
|
||
Some(node) => node,
|
||
None => std::ptr::null(),
|
||
}
|
||
}
|
||
}
|
||
|
||
#[rustfmt::skip]
|
||
impl NodeFfi<'_> {
|
||
fn as_if_clause(&self) -> &IfClause {
|
||
self.as_node().as_if_clause().unwrap()
|
||
}
|
||
fn as_job_conjunction(&self) -> &JobConjunction {
|
||
self.as_node().as_job_conjunction().unwrap()
|
||
}
|
||
fn as_job_pipeline(&self) -> &JobPipeline {
|
||
self.as_node().as_job_pipeline().unwrap()
|
||
}
|
||
fn as_argument(&self) -> &Argument {
|
||
self.as_node().as_argument().unwrap()
|
||
}
|
||
fn as_begin_header(&self) -> &BeginHeader {
|
||
self.as_node().as_begin_header().unwrap()
|
||
}
|
||
fn as_block_statement(&self) -> &BlockStatement {
|
||
self.as_node().as_block_statement().unwrap()
|
||
}
|
||
fn as_decorated_statement(&self) -> &DecoratedStatement {
|
||
self.as_node().as_decorated_statement().unwrap()
|
||
}
|
||
fn as_for_header(&self) -> &ForHeader {
|
||
self.as_node().as_for_header().unwrap()
|
||
}
|
||
fn as_freestanding_argument_list(&self) -> &FreestandingArgumentList {
|
||
self.as_node().as_freestanding_argument_list().unwrap()
|
||
}
|
||
fn as_function_header(&self) -> &FunctionHeader {
|
||
self.as_node().as_function_header().unwrap()
|
||
}
|
||
fn as_if_statement(&self) -> &IfStatement {
|
||
self.as_node().as_if_statement().unwrap()
|
||
}
|
||
fn as_job_conjunction_continuation(&self) -> &JobConjunctionContinuation {
|
||
self.as_node().as_job_conjunction_continuation().unwrap()
|
||
}
|
||
fn as_job_continuation(&self) -> &JobContinuation {
|
||
self.as_node().as_job_continuation().unwrap()
|
||
}
|
||
fn as_job_list(&self) -> &JobList {
|
||
self.as_node().as_job_list().unwrap()
|
||
}
|
||
fn as_not_statement(&self) -> &NotStatement {
|
||
self.as_node().as_not_statement().unwrap()
|
||
}
|
||
fn as_redirection(&self) -> &Redirection {
|
||
self.as_node().as_redirection().unwrap()
|
||
}
|
||
fn as_statement(&self) -> &Statement {
|
||
self.as_node().as_statement().unwrap()
|
||
}
|
||
fn as_switch_statement(&self) -> &SwitchStatement {
|
||
self.as_node().as_switch_statement().unwrap()
|
||
}
|
||
fn as_while_header(&self) -> &WhileHeader {
|
||
self.as_node().as_while_header().unwrap()
|
||
}
|
||
}
|
||
|
||
impl StatementVariant {
|
||
fn ptr(&self) -> Box<NodeFfi<'_>> {
|
||
match self {
|
||
StatementVariant::None => panic!(),
|
||
StatementVariant::NotStatement(node) => node.ptr(),
|
||
StatementVariant::BlockStatement(node) => node.ptr(),
|
||
StatementVariant::IfStatement(node) => node.ptr(),
|
||
StatementVariant::SwitchStatement(node) => node.ptr(),
|
||
StatementVariant::DecoratedStatement(node) => node.ptr(),
|
||
}
|
||
}
|
||
}
|
||
impl BlockStatementHeaderVariant {
|
||
fn ptr(&self) -> Box<NodeFfi<'_>> {
|
||
match self {
|
||
BlockStatementHeaderVariant::None => panic!(),
|
||
BlockStatementHeaderVariant::ForHeader(node) => node.ptr(),
|
||
BlockStatementHeaderVariant::WhileHeader(node) => node.ptr(),
|
||
BlockStatementHeaderVariant::FunctionHeader(node) => node.ptr(),
|
||
BlockStatementHeaderVariant::BeginHeader(node) => node.ptr(),
|
||
}
|
||
}
|
||
}
|
||
|
||
impl AndorJobList {
|
||
fn ptr(&self) -> Box<NodeFfi<'_>> {
|
||
Box::new(NodeFfi::new(self))
|
||
}
|
||
}
|
||
impl AndorJob {
|
||
fn ptr(&self) -> Box<NodeFfi<'_>> {
|
||
Box::new(NodeFfi::new(self))
|
||
}
|
||
}
|
||
impl ArgumentList {
|
||
fn ptr(&self) -> Box<NodeFfi<'_>> {
|
||
Box::new(NodeFfi::new(self))
|
||
}
|
||
}
|
||
impl Argument {
|
||
fn ptr(&self) -> Box<NodeFfi<'_>> {
|
||
Box::new(NodeFfi::new(self))
|
||
}
|
||
}
|
||
impl ArgumentOrRedirectionList {
|
||
fn ptr(&self) -> Box<NodeFfi<'_>> {
|
||
Box::new(NodeFfi::new(self))
|
||
}
|
||
}
|
||
impl ArgumentOrRedirection {
|
||
fn ptr(&self) -> Box<NodeFfi<'_>> {
|
||
Box::new(NodeFfi::new(self))
|
||
}
|
||
}
|
||
impl BeginHeader {
|
||
fn ptr(&self) -> Box<NodeFfi<'_>> {
|
||
Box::new(NodeFfi::new(self))
|
||
}
|
||
}
|
||
impl BlockStatement {
|
||
fn ptr(&self) -> Box<NodeFfi<'_>> {
|
||
Box::new(NodeFfi::new(self))
|
||
}
|
||
}
|
||
impl CaseItemList {
|
||
fn ptr(&self) -> Box<NodeFfi<'_>> {
|
||
Box::new(NodeFfi::new(self))
|
||
}
|
||
}
|
||
impl CaseItem {
|
||
fn ptr(&self) -> Box<NodeFfi<'_>> {
|
||
Box::new(NodeFfi::new(self))
|
||
}
|
||
}
|
||
impl DecoratedStatementDecorator {
|
||
fn ptr(&self) -> Box<NodeFfi<'_>> {
|
||
Box::new(NodeFfi::new(self))
|
||
}
|
||
}
|
||
impl DecoratedStatement {
|
||
fn ptr(&self) -> Box<NodeFfi<'_>> {
|
||
Box::new(NodeFfi::new(self))
|
||
}
|
||
}
|
||
impl ElseClause {
|
||
fn ptr(&self) -> Box<NodeFfi<'_>> {
|
||
Box::new(NodeFfi::new(self))
|
||
}
|
||
}
|
||
impl ElseifClauseList {
|
||
fn ptr(&self) -> Box<NodeFfi<'_>> {
|
||
Box::new(NodeFfi::new(self))
|
||
}
|
||
}
|
||
impl ElseifClause {
|
||
fn ptr(&self) -> Box<NodeFfi<'_>> {
|
||
Box::new(NodeFfi::new(self))
|
||
}
|
||
}
|
||
impl ForHeader {
|
||
fn ptr(&self) -> Box<NodeFfi<'_>> {
|
||
Box::new(NodeFfi::new(self))
|
||
}
|
||
}
|
||
impl FreestandingArgumentList {
|
||
fn ptr(&self) -> Box<NodeFfi<'_>> {
|
||
Box::new(NodeFfi::new(self))
|
||
}
|
||
}
|
||
impl FunctionHeader {
|
||
fn ptr(&self) -> Box<NodeFfi<'_>> {
|
||
Box::new(NodeFfi::new(self))
|
||
}
|
||
}
|
||
impl IfClause {
|
||
fn ptr(&self) -> Box<NodeFfi<'_>> {
|
||
Box::new(NodeFfi::new(self))
|
||
}
|
||
}
|
||
impl IfStatement {
|
||
fn ptr(&self) -> Box<NodeFfi<'_>> {
|
||
Box::new(NodeFfi::new(self))
|
||
}
|
||
}
|
||
impl JobConjunctionContinuationList {
|
||
fn ptr(&self) -> Box<NodeFfi<'_>> {
|
||
Box::new(NodeFfi::new(self))
|
||
}
|
||
}
|
||
impl JobConjunctionContinuation {
|
||
fn ptr(&self) -> Box<NodeFfi<'_>> {
|
||
Box::new(NodeFfi::new(self))
|
||
}
|
||
}
|
||
impl JobConjunctionDecorator {
|
||
fn ptr(&self) -> Box<NodeFfi<'_>> {
|
||
Box::new(NodeFfi::new(self))
|
||
}
|
||
}
|
||
impl JobConjunction {
|
||
fn ptr(&self) -> Box<NodeFfi<'_>> {
|
||
Box::new(NodeFfi::new(self))
|
||
}
|
||
}
|
||
impl JobContinuationList {
|
||
fn ptr(&self) -> Box<NodeFfi<'_>> {
|
||
Box::new(NodeFfi::new(self))
|
||
}
|
||
}
|
||
impl JobContinuation {
|
||
fn ptr(&self) -> Box<NodeFfi<'_>> {
|
||
Box::new(NodeFfi::new(self))
|
||
}
|
||
}
|
||
impl JobList {
|
||
fn ptr(&self) -> Box<NodeFfi<'_>> {
|
||
Box::new(NodeFfi::new(self))
|
||
}
|
||
}
|
||
impl JobPipeline {
|
||
fn ptr(&self) -> Box<NodeFfi<'_>> {
|
||
Box::new(NodeFfi::new(self))
|
||
}
|
||
}
|
||
impl KeywordBegin {
|
||
fn ptr(&self) -> Box<NodeFfi<'_>> {
|
||
Box::new(NodeFfi::new(self))
|
||
}
|
||
}
|
||
impl KeywordCase {
|
||
fn ptr(&self) -> Box<NodeFfi<'_>> {
|
||
Box::new(NodeFfi::new(self))
|
||
}
|
||
}
|
||
impl KeywordElse {
|
||
fn ptr(&self) -> Box<NodeFfi<'_>> {
|
||
Box::new(NodeFfi::new(self))
|
||
}
|
||
}
|
||
impl KeywordEnd {
|
||
fn ptr(&self) -> Box<NodeFfi<'_>> {
|
||
Box::new(NodeFfi::new(self))
|
||
}
|
||
}
|
||
impl KeywordFor {
|
||
fn ptr(&self) -> Box<NodeFfi<'_>> {
|
||
Box::new(NodeFfi::new(self))
|
||
}
|
||
}
|
||
impl KeywordFunction {
|
||
fn ptr(&self) -> Box<NodeFfi<'_>> {
|
||
Box::new(NodeFfi::new(self))
|
||
}
|
||
}
|
||
impl KeywordIf {
|
||
fn ptr(&self) -> Box<NodeFfi<'_>> {
|
||
Box::new(NodeFfi::new(self))
|
||
}
|
||
}
|
||
impl KeywordIn {
|
||
fn ptr(&self) -> Box<NodeFfi<'_>> {
|
||
Box::new(NodeFfi::new(self))
|
||
}
|
||
}
|
||
impl KeywordNot {
|
||
fn ptr(&self) -> Box<NodeFfi<'_>> {
|
||
Box::new(NodeFfi::new(self))
|
||
}
|
||
}
|
||
impl KeywordTime {
|
||
fn ptr(&self) -> Box<NodeFfi<'_>> {
|
||
Box::new(NodeFfi::new(self))
|
||
}
|
||
}
|
||
impl KeywordWhile {
|
||
fn ptr(&self) -> Box<NodeFfi<'_>> {
|
||
Box::new(NodeFfi::new(self))
|
||
}
|
||
}
|
||
impl MaybeNewlines {
|
||
fn ptr(&self) -> Box<NodeFfi<'_>> {
|
||
Box::new(NodeFfi::new(self))
|
||
}
|
||
}
|
||
impl NotStatement {
|
||
fn ptr(&self) -> Box<NodeFfi<'_>> {
|
||
Box::new(NodeFfi::new(self))
|
||
}
|
||
}
|
||
impl Redirection {
|
||
fn ptr(&self) -> Box<NodeFfi<'_>> {
|
||
Box::new(NodeFfi::new(self))
|
||
}
|
||
}
|
||
impl SemiNl {
|
||
fn ptr(&self) -> Box<NodeFfi<'_>> {
|
||
Box::new(NodeFfi::new(self))
|
||
}
|
||
}
|
||
impl Statement {
|
||
fn ptr(&self) -> Box<NodeFfi<'_>> {
|
||
Box::new(NodeFfi::new(self))
|
||
}
|
||
}
|
||
impl String_ {
|
||
fn ptr(&self) -> Box<NodeFfi<'_>> {
|
||
Box::new(NodeFfi::new(self))
|
||
}
|
||
}
|
||
impl SwitchStatement {
|
||
fn ptr(&self) -> Box<NodeFfi<'_>> {
|
||
Box::new(NodeFfi::new(self))
|
||
}
|
||
}
|
||
impl TokenBackground {
|
||
fn ptr(&self) -> Box<NodeFfi<'_>> {
|
||
Box::new(NodeFfi::new(self))
|
||
}
|
||
}
|
||
impl TokenConjunction {
|
||
fn ptr(&self) -> Box<NodeFfi<'_>> {
|
||
Box::new(NodeFfi::new(self))
|
||
}
|
||
}
|
||
impl TokenPipe {
|
||
fn ptr(&self) -> Box<NodeFfi<'_>> {
|
||
Box::new(NodeFfi::new(self))
|
||
}
|
||
}
|
||
impl TokenRedirection {
|
||
fn ptr(&self) -> Box<NodeFfi<'_>> {
|
||
Box::new(NodeFfi::new(self))
|
||
}
|
||
}
|
||
impl VariableAssignmentList {
|
||
fn ptr(&self) -> Box<NodeFfi<'_>> {
|
||
Box::new(NodeFfi::new(self))
|
||
}
|
||
}
|
||
impl VariableAssignment {
|
||
fn ptr(&self) -> Box<NodeFfi<'_>> {
|
||
Box::new(NodeFfi::new(self))
|
||
}
|
||
}
|
||
impl WhileHeader {
|
||
fn ptr(&self) -> Box<NodeFfi<'_>> {
|
||
Box::new(NodeFfi::new(self))
|
||
}
|
||
}
|
||
|
||
impl VariableAssignment {
|
||
fn range(&self) -> SourceRange {
|
||
self.range.unwrap()
|
||
}
|
||
}
|
||
impl TokenConjunction {
|
||
fn range(&self) -> SourceRange {
|
||
self.range.unwrap()
|
||
}
|
||
}
|
||
impl MaybeNewlines {
|
||
fn range(&self) -> SourceRange {
|
||
self.range.unwrap()
|
||
}
|
||
}
|
||
impl TokenPipe {
|
||
fn range(&self) -> SourceRange {
|
||
self.range.unwrap()
|
||
}
|
||
}
|
||
impl KeywordNot {
|
||
fn range(&self) -> SourceRange {
|
||
self.range.unwrap()
|
||
}
|
||
}
|
||
impl DecoratedStatementDecorator {
|
||
fn range(&self) -> SourceRange {
|
||
self.range.unwrap()
|
||
}
|
||
}
|
||
impl KeywordEnd {
|
||
fn range(&self) -> SourceRange {
|
||
self.range.unwrap()
|
||
}
|
||
}
|
||
impl KeywordCase {
|
||
fn range(&self) -> SourceRange {
|
||
self.range.unwrap()
|
||
}
|
||
}
|
||
impl KeywordElse {
|
||
fn range(&self) -> SourceRange {
|
||
self.range.unwrap()
|
||
}
|
||
}
|
||
impl KeywordIf {
|
||
fn range(&self) -> SourceRange {
|
||
self.range.unwrap()
|
||
}
|
||
}
|
||
impl KeywordBegin {
|
||
fn range(&self) -> SourceRange {
|
||
self.range.unwrap()
|
||
}
|
||
}
|
||
impl KeywordFunction {
|
||
fn range(&self) -> SourceRange {
|
||
self.range.unwrap()
|
||
}
|
||
}
|
||
impl KeywordWhile {
|
||
fn range(&self) -> SourceRange {
|
||
self.range.unwrap()
|
||
}
|
||
}
|
||
impl KeywordFor {
|
||
fn range(&self) -> SourceRange {
|
||
self.range.unwrap()
|
||
}
|
||
}
|
||
impl KeywordIn {
|
||
fn range(&self) -> SourceRange {
|
||
self.range.unwrap()
|
||
}
|
||
}
|
||
impl SemiNl {
|
||
fn range(&self) -> SourceRange {
|
||
self.range.unwrap()
|
||
}
|
||
}
|
||
impl JobConjunctionDecorator {
|
||
fn range(&self) -> SourceRange {
|
||
self.range.unwrap()
|
||
}
|
||
}
|
||
impl TokenBackground {
|
||
fn range(&self) -> SourceRange {
|
||
self.range.unwrap()
|
||
}
|
||
}
|
||
impl KeywordTime {
|
||
fn range(&self) -> SourceRange {
|
||
self.range.unwrap()
|
||
}
|
||
}
|
||
impl TokenRedirection {
|
||
fn range(&self) -> SourceRange {
|
||
self.range.unwrap()
|
||
}
|
||
}
|
||
impl String_ {
|
||
fn range(&self) -> SourceRange {
|
||
self.range.unwrap()
|
||
}
|
||
}
|
||
impl Argument {
|
||
fn range(&self) -> SourceRange {
|
||
self.range.unwrap()
|
||
}
|
||
}
|