-
Notifications
You must be signed in to change notification settings - Fork 20
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Implement a safer api for forward jumps (fixes #231)
- Loading branch information
Showing
4 changed files
with
122 additions
and
60 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
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); | ||
} | ||
} | ||
} | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
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; | ||
|
||
|