SemanticAST parses a SyntaxNode
as provided by JuliaSyntax into its internal ASTNode
structures for use by downstream analyses. This process is performed by one of two methods: expand_forms
and expand_toplevel
.
SemanticAST.expand_forms
— Functionexpand_forms(ast::JuliaSyntax.SyntaxNode, ctx::ExpandCtx)
Expands an expression (such as a method body) from a SyntaxNode to an Expression with respect to a ExpandCtx. See expand_toplevel for more discussion.
SemanticAST.expand_toplevel
— Functionexpand_toplevel(ast::JuliaSyntax.SyntaxNode)::ToplevelStmts
expand_toplevel(ast::JuliaSyntax.SyntaxNode, ctx::ExpandCtx)::ToplevelStmts
Takes a SyntaxNode representing an entire file and lowers it to a ToplevelStmts.
Toplevel vs. expression can be thought of generally as "where would it make sense for a module
or using
statement to exist?". As an example, the contents of a file are usually at the top level as is the arugment to an eval
call. In constrast, a method body will not be at the top level as neither module nor using statements are valid inside of it.
For most use cases this is the reccomended entry point. Use expand_forms
if you need to expand a specific expression.
SemanticAST.ExpandCtx
— TypeExpandCtx(is_toplevel::Bool=false, is_loop::Bool= false; macro_context=DefaultMacroContext(), error_context=ExceptionErrorReporting())
The context in which to perform expansion from SyntaxNodes to ASTNodes.
is_toplevel
: is the analyzed expression at the top level or not?is_loop
: is the expression in a loop or not (used for determining ifbreak
orcontinue
are valid)?macro_context
: The context with which to expand macros. See the discussion in readme.error_context
: The error reporting context. By default, two are provided (ExceptionErrorReporting and SilentErrorReporting), with exceptions being the default.
SemanticAST.ErrorReporting
— TypeThe error reporting mode to use
SemanticAST.ExceptionErrorReporting
— TypeCreate an ASTException when an issue is encountered and continue with the fallback.
SemanticAST.SilentErrorReporting
— TypeDo nothing when an issue is encountered and continue with the fallback.
SemanticAST.MacroContext
— TypeThe abstract type for macro expansion contexts.
There are two macro extension points in the analyzer: resolve_toplevel_macro(ast::SyntaxNode, ::MacroContext, ::Val{Symbol}, args::Vector{SyntaxNode}, ctx::ExpandCtx)::ToplevelStmts
and resolve_macro(ast, ::MacroContext, ::Val{Symbol}, args, ctx)::Expression
. Both share the same signature:
ast
: The SyntaxNode root of the macro invocation.MacroContext
: The context with which to resolve macros in. Implement a newMacroContext
by implementingresolve_toplevel_macro
andresolve_macro
that accept it; the provided one isDefaultMacroContext
.Val{Symbol}
: The macro's name (inVal
form).args
: The SyntaxNodes that represent the arguments to the macro.ctx
: The expansion context inside which the analysis is being done.
They differ in that resolve_toplevel_macro
returns a TopLevelStmts
(a statement that can only exist at the top level), while resolve_macro
returns an Expression
. An example of how to write a macro analyzer can be seen in the default implementation for @inline
and @noinline
:
resolve_macro(ast, ::DefaultMacroContext, ::Union{Val{Symbol("@inline")}, Val{Symbol("@noinline")}, Val{Symbol("@inbounds")}}, args, ctx) = expand_forms(args[1], ctx)
From the perspective of the default analyzer @inline
and @noinline
are no-ops, so analysis continues by simply calling back into expand_forms
on the first argument.
SemanticAST.DefaultMacroContext
— TypeThe default macro analysis context.
Currently implements top-level support for:
* @doc
* @inline, @noinline
* @assume_effects, @constprop
Expression-level support is provided for:
* @doc
* @inline, @noinline
* @eval
* @generated
* @assume_effects