Skip to content

Commit

Permalink
Document the AST
Browse files Browse the repository at this point in the history
  • Loading branch information
jedel1043 committed Oct 24, 2022
1 parent de231df commit 31902af
Show file tree
Hide file tree
Showing 64 changed files with 675 additions and 488 deletions.
24 changes: 10 additions & 14 deletions boa_engine/src/bytecompiler/mod.rs
Original file line number Diff line number Diff line change
Expand Up @@ -8,20 +8,20 @@ use crate::{
access::{PrivatePropertyAccess, PropertyAccess, PropertyAccessField},
literal::{self, TemplateElement},
operator::{
assign::{op::AssignOp, AssignTarget},
binary::op::{ArithmeticOp, BinaryOp, BitwiseOp, LogicalOp, RelationalOp},
unary::op::UnaryOp,
assign::{AssignOp, AssignTarget},
binary::{ArithmeticOp, BinaryOp, BitwiseOp, LogicalOp, RelationalOp},
unary::UnaryOp,
},
Call, Identifier, New,
},
function::{
ArrowFunction, AsyncFunction, AsyncGenerator, Class, ClassElement, FormalParameterList,
Function, Generator,
},
pattern::{Pattern, PatternArrayElement, PatternObjectElement},
pattern::{ArrayPatternElement, ObjectPatternElement, Pattern},
property::{MethodDefinition, PropertyDefinition, PropertyName},
statement::{
iteration::{for_loop::ForLoopInitializer, IterableLoopInitializer},
iteration::{ForLoopInitializer, IterableLoopInitializer},
Block, DoWhileLoop, ForInLoop, ForLoop, ForOfLoop, LabelledItem, WhileLoop,
},
Declaration, Expression, Statement, StatementList, StatementListItem,
Expand Down Expand Up @@ -1221,7 +1221,7 @@ impl<'b> ByteCompiler<'b> {
use_expr,
)?,
Expression::Conditional(op) => {
self.compile_expr(op.cond(), true)?;
self.compile_expr(op.condition(), true)?;
let jelse = self.jump_if_false();
self.compile_expr(op.if_true(), true)?;
let exit = self.jump();
Expand Down Expand Up @@ -2453,14 +2453,12 @@ impl<'b> ByteCompiler<'b> {
let rest_exits = pattern.has_rest();

for binding in pattern.bindings() {
use PatternObjectElement::{
AssignmentPropertyAccess, AssignmentRestPropertyAccess, Empty, Pattern,
use ObjectPatternElement::{
AssignmentPropertyAccess, AssignmentRestPropertyAccess, Pattern,
RestProperty, SingleName,
};

match binding {
// ObjectBindingPattern : { }
Empty => {}
// SingleNameBinding : BindingIdentifier Initializer[opt]
SingleName {
ident,
Expand Down Expand Up @@ -2605,14 +2603,12 @@ impl<'b> ByteCompiler<'b> {
self.emit_opcode(Opcode::InitIterator);

for binding in pattern.bindings().iter() {
use PatternArrayElement::{
Elision, Empty, Pattern, PatternRest, PropertyAccess, PropertyAccessRest,
use ArrayPatternElement::{
Elision, Pattern, PatternRest, PropertyAccess, PropertyAccessRest,
SingleName, SingleNameRest,
};

match binding {
// ArrayBindingPattern : [ ]
Empty => {}
// ArrayBindingPattern : [ Elision ]
Elision => {
self.emit_opcode(Opcode::IteratorNext);
Expand Down
4 changes: 2 additions & 2 deletions boa_engine/src/lib.rs
Original file line number Diff line number Diff line change
Expand Up @@ -40,7 +40,7 @@
unused_lifetimes,
unreachable_pub,
trivial_numeric_casts,
// rustdoc,
rustdoc::broken_intra_doc_links,
missing_debug_implementations,
missing_copy_implementations,
deprecated_in_future,
Expand All @@ -49,7 +49,7 @@
rust_2018_compatibility,
rust_2018_idioms,
future_incompatible,
nonstandard_style,
nonstandard_style
)]
#![allow(
clippy::missing_inline_in_public_items,
Expand Down
19 changes: 19 additions & 0 deletions boa_engine/src/syntax/ast/declaration/mod.rs
Original file line number Diff line number Diff line change
@@ -1,3 +1,19 @@
//! The [`Declaration`] Parse Node, as defined by the [spec].
//!
//! Javascript declarations include:
//! - [Lexical][lex] declarations (`let`, `const`).
//! - [Function][fun] declarations (`function`, `async function`).
//! - [Class][class] declarations.
//!
//! See [*Difference between statements and declarations*][diff] for an explanation on why `Declaration`s
//! and `Statement`s are distinct nodes.
//!
//! [spec]: https://tc39.es/ecma262/#prod-Declaration
//! [lex]: https://tc39.es/ecma262/#prod-LexicalDeclaration
//! [fun]: https://tc39.es/ecma262/#prod-HoistableDeclaration
//! [class]: https://tc39.es/ecma262/#prod-ClassDeclaration
//! [diff]: https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Statements#difference_between_statements_and_declarations
use super::{
expression::Identifier,
function::{AsyncFunction, AsyncGenerator, Class, Function, Generator},
Expand All @@ -10,6 +26,9 @@ mod variable;

pub use variable::*;

/// The `Declaration` Parse Node.
///
/// See the [module level documentation][self] for more information.
#[cfg_attr(feature = "deser", derive(serde::Serialize, serde::Deserialize))]
#[derive(Clone, Debug, PartialEq)]
pub enum Declaration {
Expand Down
25 changes: 22 additions & 3 deletions boa_engine/src/syntax/ast/declaration/variable.rs
Original file line number Diff line number Diff line change
Expand Up @@ -12,18 +12,34 @@ use boa_interner::{Interner, ToInternedString};

use super::Declaration;

/// A [`var`][var] declaration list, also called [`VariableDeclarationList`][vardecl]
/// in the spec.
/// A [`var`][var] statement, also called [`VariableStatement`][varstmt] in the spec.
///
/// The scope of a variable declared with `var` is its current execution context, which is either
/// the enclosing function or, for variables declared outside any function, global. If you
/// re-declare a JavaScript variable, it will not lose its value.
///
/// Although a bit confusing, `VarDeclaration`s are not considered [`Declaration`]s by the spec.
/// This is partly because it has very different semantics from `let` and `const` declarations, but
/// also because a `var` statement can be labelled just like any other [`Statement`]:
///
/// ```javascript
/// label: var a = 5;
/// a;
/// ```
///
/// returns `5` as the value of the statement list, while:
///
/// ```javascript
/// label: let a = 5;
/// a;
/// ```
/// throws a `SyntaxError`.
///
/// `var` declarations, wherever they occur, are processed before any code is executed. This is
/// called <code>[hoisting]</code>.
///
/// [var]: https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Statements/var
/// [vardecl]: https://tc39.es/ecma262/#prod-VariableStatement
/// [varstmt]: https://tc39.es/ecma262/#prod-VariableStatement
/// [hoisting]: https://developer.mozilla.org/en-US/docs/Glossary/Hoisting
#[cfg_attr(feature = "deser", derive(serde::Serialize, serde::Deserialize))]
#[derive(Clone, Debug, PartialEq)]
Expand Down Expand Up @@ -155,6 +171,7 @@ impl ToInternedString for VariableList {
}
}

/// The error returned by the [`VariableList::try_from`] function.
#[derive(Debug, Copy, Clone, PartialEq, Eq)]
pub struct TryFromVariableListError(());

Expand Down Expand Up @@ -280,7 +297,9 @@ impl Variable {
#[cfg_attr(feature = "deser", derive(serde::Serialize, serde::Deserialize))]
#[derive(Clone, Debug, PartialEq)]
pub enum Binding {
/// A single identifier binding.
Identifier(Identifier),
/// A pattern binding.
Pattern(Pattern),
}

Expand Down
59 changes: 28 additions & 31 deletions boa_engine/src/syntax/ast/expression/access.rs
Original file line number Diff line number Diff line change
@@ -1,10 +1,27 @@
//! Property access expressions, as defined by the [spec].
//!
//! [Property access expressions][access] provide two ways to access properties of an object: *dot notation*
//! and *bracket notation*.
//! - *Dot notation* is mostly used when the name of the property is static, and a valid Javascript
//! identifier e.g. `obj.prop`, `arr.$val`.
//! - *Bracket notation* is used when the name of the property is either variable, not a valid
//! identifier or a symbol e.g. `arr[var]`, `arr[5]`, `arr[Symbol.iterator]`.
//!
//! [spec]: https://tc39.es/ecma262/multipage/ecmascript-language-expressions.html#sec-property-accessors
//! [access]: https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Operators/Property_Accessors
use crate::syntax::ast::{expression::Expression, ContainsSymbol};
use boa_interner::{Interner, Sym, ToInternedString};

/// A property access field.
///
/// See the [module level documentation][self] for more information.
#[cfg_attr(feature = "deser", derive(serde::Serialize, serde::Deserialize))]
#[derive(Clone, Debug, PartialEq)]
pub enum PropertyAccessField {
/// A constant property field, such as `x.prop`.
Const(Sym),
/// An expression property field, such as `x["val"]`.
Expr(Box<Expression>),
}

Expand Down Expand Up @@ -38,28 +55,9 @@ impl From<Expression> for PropertyAccessField {
}
}

/// This property accessor provides access to an object's properties by using the
/// [bracket notation][mdn].
///
/// In the `object[property_name]` syntax, the `property_name` is just a string or
/// [Symbol][symbol]. So, it can be any string, including '1foo', '!bar!', or even ' ' (a
/// space).
///
/// One can think of an object as an associative array (a.k.a. map, dictionary, hash, lookup
/// table). The keys in this array are the names of the object's properties.
///
/// It's typical when speaking of an object's properties to make a distinction between
/// properties and methods. However, the property/method distinction is little more than a
/// convention. A method is simply a property that can be called (for example, if it has a
/// reference to a Function instance as its value).
///
/// More information:
/// - [ECMAScript reference][spec]
/// - [MDN documentation][mdn]
/// A property access expression.
///
/// [spec]: https://tc39.es/ecma262/#sec-property-accessors
/// [symbol]: https://developer.mozilla.org/en-US/docs/Glossary/Symbol
/// [mdn]: https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Operators/Property_accessors#Bracket_notation
/// See the [module level documentation][self] for more information.
#[cfg_attr(feature = "deser", derive(serde::Serialize, serde::Deserialize))]
#[derive(Clone, Debug, PartialEq)]
pub struct PropertyAccess {
Expand All @@ -68,11 +66,13 @@ pub struct PropertyAccess {
}

impl PropertyAccess {
/// Gets the target object of the property access.
#[inline]
pub fn target(&self) -> &Expression {
&self.target
}

/// Gets the accessed field of the target object.
#[inline]
pub fn field(&self) -> &PropertyAccessField {
&self.field
Expand Down Expand Up @@ -121,14 +121,12 @@ impl From<PropertyAccess> for Expression {
}
}

/// This property accessor provides access to an class object's private fields.
/// An access expression to a class object's [private fields][mdn].
///
/// This expression can be described as ` MemberExpression.PrivateIdentifier`
/// Example: `this.#a`
/// Private property accesses differ slightly from plain property accesses, since the accessed
/// property must be prefixed by `#`, and the bracket notation is not allowed e.g. `this.#a`.
///
/// More information:
/// - [ECMAScript reference][spec]
/// - [MDN documentation][mdn]
/// This expression corresponds to the [`MemberExpression.PrivateIdentifier`][spec] production.
///
/// [spec]: https://tc39.es/ecma262/#prod-MemberExpression
/// [mdn]: https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Classes/Private_class_fields
Expand Down Expand Up @@ -190,11 +188,10 @@ impl From<PrivatePropertyAccess> for Expression {
}
}

/// The `super` keyword is used to access fields on an object's parent.
/// A property access of an object's parent, as defined by the [spec].
///
/// More information:
/// - [ECMAScript reference][spec]
/// - [MDN documentation][mdn]
/// A `SuperPropertyAccess` is much like a regular [`PropertyAccess`], but where its `target` object
/// is not a regular object, but a reference to the parent object of the current object ([`super`][mdn]).
///
/// [spec]: https://tc39.es/ecma262/#prod-SuperProperty
/// [mdn]: https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Operators/super
Expand Down
7 changes: 5 additions & 2 deletions boa_engine/src/syntax/ast/expression/identifier.rs
Original file line number Diff line number Diff line change
Expand Up @@ -24,8 +24,11 @@ use super::Expression;
///
/// [spec]: https://tc39.es/ecma262/#prod-Identifier
/// [mdn]: https://developer.mozilla.org/en-US/docs/Glossary/Identifier
#[cfg_attr(feature = "deser", derive(serde::Serialize, serde::Deserialize))]
#[cfg_attr(feature = "deser", serde(transparent))]
#[cfg_attr(
feature = "deser",
derive(serde::Serialize, serde::Deserialize),
serde(transparent)
)]
#[derive(Debug, Clone, Copy, PartialEq, Eq, Hash)]
#[repr(transparent)]
pub struct Identifier {
Expand Down
9 changes: 9 additions & 0 deletions boa_engine/src/syntax/ast/expression/literal/template.rs
Original file line number Diff line number Diff line change
Expand Up @@ -30,14 +30,23 @@ impl From<TemplateLiteral> for Expression {
}
}

/// An element found within a [`TemplateLiteral`].
///
/// The [spec] doesn't define an element akin to `TemplateElement`. However, the AST defines this
/// node as the equivalent of the components found in a template literal.
///
/// [spec]: https://tc39.es/ecma262/#sec-template-literals
#[cfg_attr(feature = "deser", derive(serde::Serialize, serde::Deserialize))]
#[derive(Clone, Debug, PartialEq)]
pub enum TemplateElement {
/// A simple string.
String(Sym),
/// An expression that is evaluated and replaced by its string representation.
Expr(Expression),
}

impl TemplateLiteral {
/// Creates a new `TemplateLiteral` from a list of [`TemplateElement`]s.
#[inline]
pub fn new(elements: Box<[TemplateElement]>) -> Self {
Self { elements }
Expand Down
16 changes: 15 additions & 1 deletion boa_engine/src/syntax/ast/expression/mod.rs
Original file line number Diff line number Diff line change
@@ -1,9 +1,20 @@
//! The [`Expression`] Parse Node, as defined by the [spec].
//!
//! Javascript expressions include:
//! - [Primary][primary] expressions (`this`, function expressions, literals).
//! - [Left hand side][lhs] expressions (accessors, `new` operator, `super`).
//! - [operator] expressions.
//!
//! [spec]: https://tc39.es/ecma262/#prod-Expression
//! [primary]: https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Operators#primary_expressions
//! [lhs]: https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Operators#left-hand-side_expressions
use boa_interner::{Interner, Sym, ToIndentedString, ToInternedString};

use self::{
access::{PrivatePropertyAccess, PropertyAccess, SuperPropertyAccess},
literal::{ArrayLiteral, Literal, ObjectLiteral, TemplateLiteral},
operator::{conditional::Conditional, Assign, Binary, Unary},
operator::{Assign, Binary, Conditional, Unary},
};

use super::{
Expand Down Expand Up @@ -32,6 +43,9 @@ pub mod access;
pub mod literal;
pub mod operator;

/// The `Expression` Parse Node.
///
/// See the [module level documentation][self] for more information.
#[cfg_attr(feature = "deser", derive(serde::Serialize, serde::Deserialize))]
#[derive(Debug, Clone, PartialEq)]
pub enum Expression {
Expand Down
Loading

0 comments on commit 31902af

Please sign in to comment.