Browse Source

Fix lexical environments in for loops (#2917)

* Fix lexical environments in for loops

* Fix typo
pull/2926/head
raskad 2 years ago committed by GitHub
parent
commit
f57872cf8c
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
  1. 6
      boa_ast/src/declaration/variable.rs
  2. 61
      boa_engine/src/bytecompiler/statement/loop.rs
  3. 4
      boa_engine/src/vm/call_frame/env_stack.rs

6
boa_ast/src/declaration/variable.rs

@ -115,6 +115,12 @@ impl LexicalDeclaration {
Self::Const(list) | Self::Let(list) => list, Self::Const(list) | Self::Let(list) => list,
} }
} }
/// Returns `true` if the declaration is a `const` declaration.
#[must_use]
pub const fn is_const(&self) -> bool {
matches!(self, Self::Const(_))
}
} }
impl From<LexicalDeclaration> for Declaration { impl From<LexicalDeclaration> for Declaration {

61
boa_engine/src/bytecompiler/statement/loop.rs

@ -1,5 +1,5 @@
use boa_ast::{ use boa_ast::{
declaration::{Binding, LexicalDeclaration}, declaration::Binding,
operations::bound_names, operations::bound_names,
statement::{ statement::{
iteration::{ForLoopInitializer, IterableLoopInitializer}, iteration::{ForLoopInitializer, IterableLoopInitializer},
@ -20,9 +20,9 @@ impl ByteCompiler<'_, '_> {
label: Option<Sym>, label: Option<Sym>,
use_expr: bool, use_expr: bool,
) { ) {
self.push_compile_environment(false); let mut let_binding_indices = None;
let push_env = self.emit_opcode_with_two_operands(Opcode::PushDeclarativeEnvironment); let mut env_labels = None;
self.push_empty_loop_jump_control(); let mut iteration_env_labels = None;
if let Some(init) = for_loop.init() { if let Some(init) = for_loop.init() {
match init { match init {
@ -31,23 +31,32 @@ impl ByteCompiler<'_, '_> {
self.compile_var_decl(decl); self.compile_var_decl(decl);
} }
ForLoopInitializer::Lexical(decl) => { ForLoopInitializer::Lexical(decl) => {
match decl { self.push_compile_environment(false);
LexicalDeclaration::Const(decl) => { env_labels = Some(
for name in bound_names(decl) { self.emit_opcode_with_two_operands(Opcode::PushDeclarativeEnvironment),
self.create_immutable_binding(name, true); );
}
} let names = bound_names(decl);
LexicalDeclaration::Let(decl) => { if decl.is_const() {
for name in bound_names(decl) { for name in &names {
self.create_mutable_binding(name, false); self.create_immutable_binding(*name, true);
} }
} else {
let mut indices = Vec::new();
for name in &names {
self.create_mutable_binding(*name, false);
let binding = self.initialize_mutable_binding(*name, false);
let index = self.get_or_insert_binding(binding);
indices.push(index);
} }
let_binding_indices = Some(indices);
} }
self.compile_lexical_decl(decl); self.compile_lexical_decl(decl);
} }
} }
} }
self.push_empty_loop_jump_control();
let (loop_start, loop_exit) = self.emit_opcode_with_two_operands(Opcode::LoopStart); let (loop_start, loop_exit) = self.emit_opcode_with_two_operands(Opcode::LoopStart);
let initial_jump = self.jump(); let initial_jump = self.jump();
let start_address = self.next_opcode_location(); let start_address = self.next_opcode_location();
@ -59,6 +68,18 @@ impl ByteCompiler<'_, '_> {
.expect("jump_control must exist as it was just pushed") .expect("jump_control must exist as it was just pushed")
.set_start_address(start_address); .set_start_address(start_address);
if let Some(let_binding_indices) = let_binding_indices {
for index in &let_binding_indices {
self.emit(Opcode::GetName, &[*index]);
}
self.emit_opcode(Opcode::PopEnvironment);
iteration_env_labels =
Some(self.emit_opcode_with_two_operands(Opcode::PushDeclarativeEnvironment));
for index in let_binding_indices.iter().rev() {
self.emit(Opcode::PutLexicalValue, &[*index]);
}
}
self.emit_opcode(Opcode::LoopContinue); self.emit_opcode(Opcode::LoopContinue);
self.patch_jump_with_target(loop_start, start_address); self.patch_jump_with_target(loop_start, start_address);
@ -83,10 +104,6 @@ impl ByteCompiler<'_, '_> {
self.emit(Opcode::Jump, &[start_address]); self.emit(Opcode::Jump, &[start_address]);
let env_info = self.pop_compile_environment();
self.patch_jump_with_target(push_env.0, env_info.num_bindings as u32);
self.patch_jump_with_target(push_env.1, env_info.index as u32);
self.patch_jump(exit); self.patch_jump(exit);
self.patch_jump(loop_exit); self.patch_jump(loop_exit);
self.pop_loop_control_info(); self.pop_loop_control_info();
@ -95,8 +112,18 @@ impl ByteCompiler<'_, '_> {
if !use_expr { if !use_expr {
self.emit_opcode(Opcode::Pop); self.emit_opcode(Opcode::Pop);
} }
if let Some(env_labels) = env_labels {
let env_info = self.pop_compile_environment();
self.patch_jump_with_target(env_labels.0, env_info.num_bindings as u32);
self.patch_jump_with_target(env_labels.1, env_info.index as u32);
if let Some(iteration_env_labels) = iteration_env_labels {
self.patch_jump_with_target(iteration_env_labels.0, env_info.num_bindings as u32);
self.patch_jump_with_target(iteration_env_labels.1, env_info.index as u32);
}
self.emit_opcode(Opcode::PopEnvironment); self.emit_opcode(Opcode::PopEnvironment);
} }
}
pub(crate) fn compile_for_in_loop( pub(crate) fn compile_for_in_loop(
&mut self, &mut self,

4
boa_engine/src/vm/call_frame/env_stack.rs

@ -184,12 +184,12 @@ impl EnvStackEntry {
/// Increments the `env_num` field for current `EnvEntryStack`. /// Increments the `env_num` field for current `EnvEntryStack`.
pub(crate) fn inc_env_num(&mut self) { pub(crate) fn inc_env_num(&mut self) {
self.env_num += 1; (self.env_num, _) = self.env_num.overflowing_add(1);
} }
/// Decrements the `env_num` field for current `EnvEntryStack`. /// Decrements the `env_num` field for current `EnvEntryStack`.
pub(crate) fn dec_env_num(&mut self) { pub(crate) fn dec_env_num(&mut self) {
self.env_num -= 1; (self.env_num, _) = self.env_num.overflowing_sub(1);
} }
/// Set the loop return value for the current `EnvStackEntry`. /// Set the loop return value for the current `EnvStackEntry`.

Loading…
Cancel
Save