Skip to content

Commit

Permalink
Implement a safer api for forward jumps (fixes #231)
Browse files Browse the repository at this point in the history
  • Loading branch information
codehag committed Feb 27, 2020
1 parent 594a09b commit fe46ecc
Show file tree
Hide file tree
Showing 4 changed files with 122 additions and 60 deletions.
98 changes: 39 additions & 59 deletions crates/emitter/src/ast_emitter.rs
Original file line number Diff line number Diff line change
Expand Up @@ -2,10 +2,12 @@
//!
//! Converts AST nodes to bytecode.
use super::emitter::{BytecodeOffset, EmitError, EmitOptions, EmitResult, InstructionWriter};
use super::emitter::{EmitError, EmitOptions, EmitResult, InstructionWriter};
use super::opcode::Opcode;
use ast::types::*;

use crate::forward_jump_emitter::{ForwardJumpEmitter, JumpKind};

/// Emit a program, converting the AST directly to bytecode.
pub fn emit_program(ast: &Program, options: &EmitOptions) -> Result<EmitResult, EmitError> {
let mut emitter = AstEmitter {
Expand All @@ -23,8 +25,8 @@ pub fn emit_program(ast: &Program, options: &EmitOptions) -> Result<EmitResult,
Ok(emitter.emit.into_emit_result())
}

struct AstEmitter<'alloc> {
emit: InstructionWriter,
pub struct AstEmitter<'alloc> {
pub emit: InstructionWriter,
options: &'alloc EmitOptions,
}

Expand Down Expand Up @@ -129,27 +131,31 @@ impl<'alloc> AstEmitter<'alloc> {
fn emit_if(&mut self, if_statement: &IfStatement) -> Result<(), EmitError> {
self.emit_expression(&if_statement.test)?;

let offset_alternate = self.emit.bytecode_offset();
self.emit.if_eq(0);
let alternate_jump = ForwardJumpEmitter {
jump: JumpKind::IfEq,
}
.emit(self);

// Then branch
self.emit.jump_target();
self.emit_statement(&if_statement.consequent)?;

if let Some(alternate) = &if_statement.alternate {
let offset_final = self.emit.bytecode_offset();
self.emit.goto(0);
let then_jump = ForwardJumpEmitter {
jump: JumpKind::IfEq,
}
.emit(self);
// ^^ part of then branch

// Else branch
self.emit_jump_target(vec![offset_alternate]);
alternate_jump.patch(self);
self.emit_statement(alternate)?;

// Merge point after else
self.emit_jump_target(vec![offset_final]);
then_jump.patch(self);
} else {
// Merge point without else
self.emit_jump_target(vec![offset_alternate])
alternate_jump.patch(self);
}

Ok(())
Expand Down Expand Up @@ -402,10 +408,16 @@ impl<'alloc> AstEmitter<'alloc> {
BinaryOperator::BitwiseXor { .. } => Opcode::BitXor,
BinaryOperator::BitwiseAnd { .. } => Opcode::BitAnd,

BinaryOperator::Coalesce { .. }
| BinaryOperator::LogicalOr { .. }
| BinaryOperator::LogicalAnd { .. } => {
self.emit_short_circuit(operator, left, right)?;
BinaryOperator::Coalesce { .. } => {
self.emit_short_circuit(JumpKind::Coalesce, left, right)?;
return Ok(());
}
BinaryOperator::LogicalOr { .. } => {
self.emit_short_circuit(JumpKind::LogicalOr, left, right)?;
return Ok(());
}
BinaryOperator::LogicalAnd { .. } => {
self.emit_short_circuit(JumpKind::LogicalAnd, left, right)?;
return Ok(());
}

Expand All @@ -425,54 +437,18 @@ impl<'alloc> AstEmitter<'alloc> {

fn emit_short_circuit(
&mut self,
operator: &BinaryOperator,
jump: JumpKind,
left: &Expression,
right: &Expression,
) -> Result<(), EmitError> {
self.emit_expression(left)?;
let mut jumplist: Vec<BytecodeOffset> = Vec::with_capacity(1);
self.emit_jump(operator, &mut jumplist)?;
self.emit.jump_target();
let jump = ForwardJumpEmitter { jump: jump }.emit(self);
self.emit.pop();
self.emit_expression(right)?;
self.emit_jump_target(jumplist);
jump.patch(self);
return Ok(());
}

fn emit_jump(
&mut self,
operator: &BinaryOperator,
jumplist: &mut Vec<BytecodeOffset>,
) -> Result<(), EmitError> {
let offset: BytecodeOffset = self.emit.bytecode_offset();
jumplist.push(offset);

// in the c++ bytecode emitter, the jumplist is emitted
// and four bytes are used in order to save memory. We are not using that
// here, so instead we are using a placeholder offset set to 0, which will
// be updated later in patch_jump_target.
let placeholder_offset: i32 = 0;
match operator {
BinaryOperator::Coalesce { .. } => {
self.emit.coalesce(placeholder_offset);
}
BinaryOperator::LogicalOr { .. } => {
self.emit.or(placeholder_offset);
}
BinaryOperator::LogicalAnd { .. } => {
self.emit.and(placeholder_offset);
}
_ => panic!("unrecognized operator used in jump"),
}

return Ok(());
}

fn emit_jump_target(&mut self, jumplist: Vec<BytecodeOffset>) {
self.emit.patch_jump_target(jumplist);
self.emit.jump_target();
}

fn emit_numeric_expression(&mut self, value: f64) {
if value.is_finite() && value.fract() == 0.0 {
if i8::min_value() as f64 <= value && value <= i8::max_value() as f64 {
Expand Down Expand Up @@ -565,22 +541,26 @@ impl<'alloc> AstEmitter<'alloc> {
) -> Result<(), EmitError> {
self.emit_expression(test)?;

let offset_else = self.emit.bytecode_offset();
self.emit.if_eq(0);
let else_jump = ForwardJumpEmitter {
jump: JumpKind::IfEq,
}
.emit(self);

// Then branch
self.emit.jump_target();
self.emit_expression(consequent)?;

let offset_final = self.emit.bytecode_offset();
self.emit.goto(0);
let finally_jump = ForwardJumpEmitter {
jump: JumpKind::Goto,
}
.emit(self);

// Else branch
self.emit_jump_target(vec![offset_else]);
else_jump.patch(self);
self.emit_expression(alternate)?;

// Merge point
self.emit_jump_target(vec![offset_final]);
finally_jump.patch(self);

Ok(())
}
Expand Down
3 changes: 2 additions & 1 deletion crates/emitter/src/emitter.rs
Original file line number Diff line number Diff line change
Expand Up @@ -26,7 +26,8 @@ pub enum AsyncFunctionResolveKind {
pub type u24 = u32;

/// For tracking bytecode offsets in jumps
#[derive(PartialEq, Debug)]
#[derive(Clone, Copy, PartialEq, Debug)]
#[must_use]
pub struct BytecodeOffset {
pub offset: usize,
}
Expand Down
80 changes: 80 additions & 0 deletions crates/emitter/src/forward_jump_emitter.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,80 @@
use super::emitter::{BytecodeOffset};
use crate::ast_emitter::AstEmitter;

#[derive(Debug)]
pub enum JumpKind {
Coalesce,
LogicalAnd,
LogicalOr,
IfEq,
Goto,
}

#[derive(Debug)]
#[must_use]
pub struct JumpPatchEmitter {
offset: BytecodeOffset,
}
impl JumpPatchEmitter {
pub fn patch(self, emitter: &mut AstEmitter) {
// TODO: if multiple jumps arrive at same point, only single JumpTarget
// should be emitted. See:
// https://searchfox.org/mozilla-central/rev/49ed791eec93335abfe6c2880f84c324e73e47e6/js/src/frontend/BytecodeEmitter.cpp#371-377
emitter.emit.patch_jump_target(vec![self.offset]);
emitter.emit.jump_target();
}
}

// Struct for emitting bytecode for forward jump.
#[derive(Debug)]
pub struct ForwardJumpEmitter {
pub jump: JumpKind,
}
impl ForwardJumpEmitter {
pub fn emit(&mut self, emitter: &mut AstEmitter) -> JumpPatchEmitter {
let offset: BytecodeOffset = emitter.emit.bytecode_offset();
self.emit_jump(emitter);

// The JITs rely on a jump target being emitted after the
// conditional jump
if self.should_fallthrough() {
emitter.emit.jump_target();
}
JumpPatchEmitter { offset: offset }
}

fn should_fallthrough(&mut self) -> bool {
// a fallthrough occurs if the jump is a conditional jump and if the
// condition doesn't met, the execution goes to the next opcode
// instead of the target of the jump.
match self.jump {
JumpKind::Goto { .. } => false,
_ => true,
}
}

fn emit_jump(&mut self, emitter: &mut AstEmitter) {
// in the c++ bytecode emitter, the jumplist is emitted
// and four bytes are used in order to save memory. We are not using that
// here, so instead we are using a placeholder offset set to 0, which will
// be updated later in patch_jump_target.
let placeholder_offset: i32 = 0;
match self.jump {
JumpKind::Coalesce { .. } => {
emitter.emit.coalesce(placeholder_offset);
}
JumpKind::LogicalOr { .. } => {
emitter.emit.or(placeholder_offset);
}
JumpKind::LogicalAnd { .. } => {
emitter.emit.and(placeholder_offset);
}
JumpKind::IfEq { .. } => {
emitter.emit.if_eq(placeholder_offset);
}
JumpKind::Goto { .. } => {
emitter.emit.goto(placeholder_offset);
}
}
}
}
1 change: 1 addition & 0 deletions crates/emitter/src/lib.rs
Original file line number Diff line number Diff line change
@@ -1,6 +1,7 @@
mod ast_emitter;
mod dis;
mod emitter;
mod forward_jump_emitter;
mod lower;
pub mod opcode;

Expand Down

0 comments on commit fe46ecc

Please sign in to comment.