Skip to content

Commit

Permalink
Correctly initialize functions inside modules (#2993)
Browse files Browse the repository at this point in the history
  • Loading branch information
jedel1043 authored and Razican committed Jun 26, 2023
1 parent 70e865d commit 49d8526
Show file tree
Hide file tree
Showing 3 changed files with 45 additions and 38 deletions.
64 changes: 31 additions & 33 deletions boa_ast/src/operations.rs
Original file line number Diff line number Diff line change
Expand Up @@ -1358,37 +1358,37 @@ where
}

/// The type of a lexically scoped declaration.
#[derive(Clone, Debug)]
pub enum LexicallyScopedDeclaration {
#[derive(Copy, Clone, Debug)]
pub enum LexicallyScopedDeclaration<'a> {
/// See [`LexicalDeclaration`]
LexicalDeclaration(LexicalDeclaration),
LexicalDeclaration(&'a LexicalDeclaration),

/// See [`Function`]
Function(Function),
Function(&'a Function),

/// See [`Generator`]
Generator(Generator),
Generator(&'a Generator),

/// See [`AsyncFunction`]
AsyncFunction(AsyncFunction),
AsyncFunction(&'a AsyncFunction),

/// See [`AsyncGenerator`]
AsyncGenerator(AsyncGenerator),
AsyncGenerator(&'a AsyncGenerator),

/// See [`Class`]
Class(Class),
Class(&'a Class),

/// A default assignment expression as an export declaration.
///
/// Only valid inside module exports.
AssignmentExpression(Expression),
AssignmentExpression(&'a Expression),
}

impl LexicallyScopedDeclaration {
impl LexicallyScopedDeclaration<'_> {
/// Return the bound names of the declaration.
#[must_use]
pub fn bound_names(&self) -> Vec<Identifier> {
match self {
match *self {
Self::LexicalDeclaration(v) => bound_names(v),
Self::Function(f) => bound_names(f),
Self::Generator(g) => bound_names(g),
Expand All @@ -1400,8 +1400,8 @@ impl LexicallyScopedDeclaration {
}
}

impl From<Declaration> for LexicallyScopedDeclaration {
fn from(value: Declaration) -> Self {
impl<'ast> From<&'ast Declaration> for LexicallyScopedDeclaration<'ast> {
fn from(value: &'ast Declaration) -> LexicallyScopedDeclaration<'ast> {
match value {
Declaration::Function(f) => Self::Function(f),
Declaration::Generator(g) => Self::Generator(g),
Expand All @@ -1419,7 +1419,7 @@ impl From<Declaration> for LexicallyScopedDeclaration {
///
/// [spec]: https://tc39.es/ecma262/#sec-static-semantics-lexicallyscopeddeclarations
#[must_use]
pub fn lexically_scoped_declarations<'a, N>(node: &'a N) -> Vec<LexicallyScopedDeclaration>
pub fn lexically_scoped_declarations<'a, N>(node: &'a N) -> Vec<LexicallyScopedDeclaration<'a>>
where
&'a N: Into<NodeRef<'a>>,
{
Expand All @@ -1430,9 +1430,9 @@ where

/// The [`Visitor`] used to obtain the lexically scoped declarations of a node.
#[derive(Debug)]
struct LexicallyScopedDeclarationsVisitor<'a>(&'a mut Vec<LexicallyScopedDeclaration>);
struct LexicallyScopedDeclarationsVisitor<'a, 'ast>(&'a mut Vec<LexicallyScopedDeclaration<'ast>>);

impl<'ast> Visitor<'ast> for LexicallyScopedDeclarationsVisitor<'_> {
impl<'ast> Visitor<'ast> for LexicallyScopedDeclarationsVisitor<'_, 'ast> {
type BreakTy = Infallible;

// ScriptBody : StatementList
Expand Down Expand Up @@ -1460,34 +1460,30 @@ impl<'ast> Visitor<'ast> for LexicallyScopedDeclarationsVisitor<'_> {
// ExportDeclaration : export Declaration
ExportDeclaration::Declaration(decl) => {
// 1. Return a List whose sole element is DeclarationPart of Declaration.
decl.clone().into()
decl.into()
}

// ExportDeclaration : export default HoistableDeclaration
// 1. Return a List whose sole element is DeclarationPart of HoistableDeclaration.
ExportDeclaration::DefaultFunction(f) => {
LexicallyScopedDeclaration::Function(f.clone())
}
ExportDeclaration::DefaultGenerator(g) => {
LexicallyScopedDeclaration::Generator(g.clone())
}
ExportDeclaration::DefaultFunction(f) => LexicallyScopedDeclaration::Function(f),
ExportDeclaration::DefaultGenerator(g) => LexicallyScopedDeclaration::Generator(g),
ExportDeclaration::DefaultAsyncFunction(af) => {
LexicallyScopedDeclaration::AsyncFunction(af.clone())
LexicallyScopedDeclaration::AsyncFunction(af)
}
ExportDeclaration::DefaultAsyncGenerator(ag) => {
LexicallyScopedDeclaration::AsyncGenerator(ag.clone())
LexicallyScopedDeclaration::AsyncGenerator(ag)
}

// ExportDeclaration : export default ClassDeclaration
ExportDeclaration::DefaultClassDeclaration(c) => {
// 1. Return a List whose sole element is ClassDeclaration.
LexicallyScopedDeclaration::Class(c.clone())
LexicallyScopedDeclaration::Class(c)
}

// ExportDeclaration : export default AssignmentExpression ;
ExportDeclaration::DefaultAssignmentExpression(expr) => {
// 1. Return a List whose sole element is this ExportDeclaration.
LexicallyScopedDeclaration::AssignmentExpression(expr.clone())
LexicallyScopedDeclaration::AssignmentExpression(expr)
}
};

Expand All @@ -1514,7 +1510,7 @@ impl<'ast> Visitor<'ast> for LexicallyScopedDeclarationsVisitor<'_> {
// StatementListItem : Declaration
StatementListItem::Declaration(declaration) => {
// 1. Return a List whose sole element is DeclarationPart of Declaration.
self.0.push(declaration.clone().into());
self.0.push(declaration.into());
ControlFlow::Continue(())
}
}
Expand All @@ -1525,7 +1521,7 @@ impl<'ast> Visitor<'ast> for LexicallyScopedDeclarationsVisitor<'_> {
// LabelledItem : FunctionDeclaration
LabelledItem::Function(f) => {
// 1. Return « FunctionDeclaration ».
self.0.push(LexicallyScopedDeclaration::Function(f.clone()));
self.0.push(LexicallyScopedDeclaration::Function(f));
}

// LabelledItem : Statement
Expand Down Expand Up @@ -1555,9 +1551,11 @@ impl<'ast> Visitor<'ast> for LexicallyScopedDeclarationsVisitor<'_> {
///
/// [spec]: https://tc39.es/ecma262/#sec-static-semantics-toplevellexicallyscopeddeclarations
#[derive(Debug)]
struct TopLevelLexicallyScopedDeclarationsVisitor<'a>(&'a mut Vec<LexicallyScopedDeclaration>);
struct TopLevelLexicallyScopedDeclarationsVisitor<'a, 'ast>(
&'a mut Vec<LexicallyScopedDeclaration<'ast>>,
);

impl<'ast> Visitor<'ast> for TopLevelLexicallyScopedDeclarationsVisitor<'_> {
impl<'ast> Visitor<'ast> for TopLevelLexicallyScopedDeclarationsVisitor<'_, 'ast> {
type BreakTy = Infallible;

fn visit_statement_list_item(
Expand All @@ -1577,11 +1575,11 @@ impl<'ast> Visitor<'ast> for TopLevelLexicallyScopedDeclarationsVisitor<'_> {

// 2. Return « Declaration ».
Declaration::Class(cl) => {
self.0.push(LexicallyScopedDeclaration::Class(cl.clone()));
self.0.push(LexicallyScopedDeclaration::Class(cl));
}
Declaration::Lexical(lex) => {
self.0
.push(LexicallyScopedDeclaration::LexicalDeclaration(lex.clone()));
.push(LexicallyScopedDeclaration::LexicalDeclaration(lex));
}
},

Expand Down
2 changes: 1 addition & 1 deletion boa_engine/src/bytecompiler/declarations.rs
Original file line number Diff line number Diff line change
Expand Up @@ -351,7 +351,7 @@ impl ByteCompiler<'_, '_> {
// ii. Let fo be InstantiateFunctionObject of d with arguments env and privateEnv.
// iii. Perform ! env.InitializeBinding(fn, fo). NOTE: This step is replaced in section B.3.2.6.
// TODO: Support B.3.2.6.
for d in &declarations {
for d in declarations {
match d {
LexicallyScopedDeclaration::Function(function) => {
self.function_with_binding(function.into(), NodeKind::Declaration, false);
Expand Down
17 changes: 13 additions & 4 deletions boa_engine/src/module/source.rs
Original file line number Diff line number Diff line change
Expand Up @@ -1505,7 +1505,7 @@ impl SourceTextModule {
let lex_declarations = lexically_scoped_declarations(&self.inner.code.source);
let mut functions = Vec::new();
// 24. For each element d of lexDeclarations, do
for declaration in &lex_declarations {
for declaration in lex_declarations {
// ii. Else,
// a. For each element dn of the BoundNames of d, do
// 1. Perform ! env.CreateMutableBinding(dn, false).
Expand Down Expand Up @@ -1576,11 +1576,20 @@ impl SourceTextModule {
}
};

let kind = spec.kind;

functions.push((compiler.function(spec), locator, kind));
functions.push((spec, locator));
}

// Should compile after initializing bindings first to ensure inner calls
// are correctly resolved to the outer functions instead of as global bindings.
let functions = functions
.into_iter()
.map(|(spec, locator)| {
let kind = spec.kind;

(compiler.function(spec), locator, kind)
})
.collect::<Vec<_>>();

compiler.compile_module_item_list(self.inner.code.source.items());

(Gc::new(compiler.finish()), functions)
Expand Down

0 comments on commit 49d8526

Please sign in to comment.