Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Single-pass native code generation for x86-64 using dynasm. #276

Merged
merged 112 commits into from
Mar 21, 2019
Merged
Changes from 1 commit
Commits
Show all changes
112 commits
Select commit Hold shift + click to select a range
9649219
Initial work on WebAssembly parser for Dynasm backend.
losfair Feb 8, 2019
f735471
Merge remote-tracking branch 'private/master' into feature/dynasm-bac…
losfair Feb 11, 2019
af19f5c
Update dependencies.
losfair Feb 11, 2019
bbb27be
Code generation framework.
losfair Feb 11, 2019
ffc1bde
Cleanup & add imports.
losfair Feb 11, 2019
2fbb5e3
Codegen backend for x64.
losfair Feb 11, 2019
4ebb22f
Calling conventions, value stack, and runtime stack layout.
losfair Feb 12, 2019
a69c5b4
stack: Helper functions and unified return types
losfair Feb 13, 2019
43df3dd
Use System V calling convention and implement a few opcodes.
losfair Feb 13, 2019
7df7204
codegen: More opcodes
losfair Feb 13, 2019
8d8db4a
Starting integration.
losfair Feb 13, 2019
1526d35
Emit a Return opcode if the last one is not return.
losfair Feb 14, 2019
bb52a4e
Now we can run `add`!
losfair Feb 14, 2019
5583e96
Mitigate a bug that causes incorrect code generation.
losfair Feb 19, 2019
93d2713
Operators.
losfair Feb 19, 2019
61c8350
Control frames, jumps & stack unwinding.
losfair Feb 20, 2019
aaabbf1
Operator::Br and control stack fixes
losfair Feb 20, 2019
6f97ebd
Remove a hack in parser.
losfair Feb 21, 2019
63b3f41
Handle unreachable and fix return.
losfair Feb 21, 2019
7c43993
Block and conditional branch.
losfair Feb 21, 2019
08a2ec8
Unary operators, comparison opcodes, loops, etc.
losfair Feb 22, 2019
693c9fd
Single-pass backend tests.
losfair Feb 22, 2019
e9c0325
Update dependencies
losfair Feb 23, 2019
09cbd4a
Changed to using custom calling conventions; Implemented direct calls.
losfair Feb 23, 2019
b2f5f77
Add direct call test
losfair Feb 23, 2019
dbebdf9
Code generation for br_table.
losfair Feb 24, 2019
2432a6c
Fix function calls.
losfair Feb 25, 2019
da1a3fa
Add test for br_table.
losfair Feb 25, 2019
80812e3
Implement native call & fix stack alignment.
losfair Feb 25, 2019
78fd995
Fix argument passing at entry.
losfair Feb 25, 2019
b18595f
Pass command-line arguments to wasm as i32.
losfair Feb 25, 2019
5302949
Allow more registers to be used.
losfair Feb 25, 2019
9d8c5a5
Add a note on incorrect code generation.
losfair Feb 25, 2019
3c3c5db
Native trampolines.
losfair Feb 26, 2019
d50f1cc
If/Else.
losfair Feb 27, 2019
ec9a8f0
Add test for if/else.
losfair Feb 27, 2019
b7ca5e4
Add Select opcode.
losfair Feb 28, 2019
adb309f
Add select test.
losfair Feb 28, 2019
64142c4
Reserve R15.
losfair Feb 28, 2019
e026adf
Initial work on linear memory.
losfair Feb 28, 2019
aa75994
Add memory opcodes and test.
losfair Mar 1, 2019
2962fe2
Add unwinding test.
losfair Mar 1, 2019
fa61b66
Strongly type scratch registers and fixed an unwinding issue.
losfair Mar 4, 2019
bd7698e
64-bit operators.
losfair Mar 4, 2019
5a97a25
Add basic test for i64.
losfair Mar 4, 2019
27b2061
Implemented missing integer operators and fixed division.
losfair Mar 5, 2019
8d2e877
tee_local and div tests
losfair Mar 5, 2019
12c2137
Hack around calling imports. Not yet working.
losfair Mar 7, 2019
258dea6
Fix calling imports.
losfair Mar 8, 2019
4c4743e
Globals.
losfair Mar 8, 2019
3efccbe
Support imported globals.
losfair Mar 8, 2019
25034ec
Almost finished indirect calls.
losfair Mar 8, 2019
c6dfbcd
Add tests.
losfair Mar 8, 2019
c5ef0a9
Update vm::Ctx.
losfair Mar 8, 2019
683cb20
Fix call_indirect.
losfair Mar 12, 2019
c3b0bd7
Validate return values in indirect call test.
losfair Mar 12, 2019
1fc7b31
Add missing trait items to DynasmCompiler so that cargo build --all w…
bjfish Mar 13, 2019
557be77
Merge branch 'master' into feature/dynasm-backend
bjfish Mar 13, 2019
a5bab8c
Updates to compile dynasm after merge from master
bjfish Mar 13, 2019
68181ac
Add dynasm to spectests
bjfish Mar 13, 2019
d4ded2c
Fix bugs found when running spectests.
losfair Mar 13, 2019
179bbf9
Fix integer division.
losfair Mar 14, 2019
e5d67c9
Add dynasm feature to emscripten tests
bjfish Mar 14, 2019
d80ea47
FIx bugs found with spectests.
losfair Mar 14, 2019
c76887d
Merge remote-tracking branch 'private/feature/dynasm-backend' into fe…
losfair Mar 14, 2019
592c3fb
Fixes and some floating point operations.
losfair Mar 14, 2019
4d2b6a0
Add Copy + Clone for IfElseState
losfair Mar 14, 2019
1104073
Implemented more floating point operations.
losfair Mar 14, 2019
1b5ea9b
Passes all `assert_return(_*)` tests.
losfair Mar 14, 2019
53a8fca
Make wasmer executable work with dynasm
syrusakbary Mar 16, 2019
81af8cf
Fixed several bugs in the single-pass backend and implemented a runti…
losfair Mar 16, 2019
1f8c644
Merge remote-tracking branch 'private/feature/dynasm-backend' into fe…
losfair Mar 16, 2019
f8fe999
Implemented protected call and floating point traps; passing all spec…
losfair Mar 17, 2019
d8d39c3
Fix warnings.
losfair Mar 17, 2019
7394df2
FIx floating point trunc's.
losfair Mar 17, 2019
08f4526
Merge remote-tracking branch 'origin/master' into feature/dynasm-backend
losfair Mar 17, 2019
fcfde73
Use R15 as temporary register.
losfair Mar 17, 2019
337b2eb
Add dynasm tests to the Makefile
bjfish Mar 17, 2019
99faa79
Run cargo fmt.
losfair Mar 17, 2019
6c40ea1
Cargo fmt more files.
losfair Mar 17, 2019
e2a3887
Merge remote-tracking branch 'private/feature/dynasm-backend' into fe…
losfair Mar 17, 2019
4ca27b6
Manually fix lint errors.
losfair Mar 17, 2019
e48ff02
Remove commented out code.
losfair Mar 17, 2019
c5694ec
Fix lint errors.
losfair Mar 17, 2019
662a649
Only run dynasm tests on nightly.
losfair Mar 17, 2019
e1cb4fc
Removed submodule cranlift
bjfish Mar 17, 2019
caa239a
Make wasmer-dynasm-backend optional.
losfair Mar 18, 2019
75678b5
Merge remote-tracking branch 'origin/feature/dynasm-backend' into fea…
losfair Mar 18, 2019
b94c046
Remove println.
losfair Mar 18, 2019
af8f307
Fix dependencies.
losfair Mar 18, 2019
cd5c145
Fix default compiler.
losfair Mar 18, 2019
b06a49e
Move wasmer-dynasm-backend out of default-compiler.
losfair Mar 18, 2019
eb606a6
Disable clippy temporarily.
losfair Mar 18, 2019
af24cfc
Exclude dynasm backend in cargo test.
losfair Mar 18, 2019
2ab2205
Allow cross-module indirect calls.
losfair Mar 19, 2019
ebaf2dc
Make DynFunc::func_index private.
losfair Mar 19, 2019
61abe70
cargo fmt
losfair Mar 19, 2019
a006a36
Cleanup.
losfair Mar 19, 2019
14da8ab
Run-time memory bound checking.
losfair Mar 19, 2019
4256ccb
Cleanup & fix need_check condition.
losfair Mar 19, 2019
7ee364a
Cargo fmt.
losfair Mar 19, 2019
fd60631
Remove '+nightly' in Makefile.
losfair Mar 19, 2019
f0e8f22
Merge remote-tracking branch 'origin/master' into feature/dynasm-backend
losfair Mar 19, 2019
82b2034
Run clippy on nightly rust.
losfair Mar 19, 2019
8b85099
Unset global git redirection.
losfair Mar 19, 2019
295efbf
Fix clippy errors.
losfair Mar 19, 2019
01f18b2
Merge remote-tracking branch 'origin/master' into feature/dynasm-backend
losfair Mar 19, 2019
395161a
Merge remote-tracking branch 'origin/master' into feature/dynasm-backend
losfair Mar 20, 2019
cb3846f
Add `InternalCtx`.
losfair Mar 21, 2019
a4ee873
Merge remote-tracking branch 'origin/master' into feature/dynasm-backend
losfair Mar 21, 2019
26e4278
Make the `internal` field private from outside.
losfair Mar 21, 2019
08ba696
Fix vmctx offset tests.
losfair Mar 21, 2019
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Prev Previous commit
Next Next commit
Control frames, jumps & stack unwinding.
  • Loading branch information
losfair committed Feb 20, 2019
commit 61c83507a4160c168f279c7ab026ea0674588dce
198 changes: 196 additions & 2 deletions lib/dynasm-backend/src/codegen_x64.rs
Original file line number Diff line number Diff line change
@@ -1,5 +1,5 @@
use super::codegen::*;
use super::stack::{ValueInfo, ValueLocation, ValueStack};
use super::stack::{ControlFrame, ControlStack, ValueInfo, ValueLocation, ValueStack};
use dynasmrt::{
x64::Assembler, AssemblyOffset, DynamicLabel, DynasmApi, DynasmLabelApi, ExecutableBuffer,
};
@@ -86,6 +86,7 @@ pub struct X64FunctionCode {
num_params: usize,
current_stack_offset: usize,
value_stack: ValueStack,
control_stack: Option<ControlStack>,
}

pub struct X64ExecutionContext {
@@ -182,6 +183,7 @@ impl ModuleCodeGenerator<X64FunctionCode, X64ExecutionContext> for X64ModuleCode
num_params: 0,
current_stack_offset: 0,
value_stack: ValueStack::new(4),
control_stack: None,
};
self.functions.push(code);
Ok(self.functions.last_mut().unwrap())
@@ -348,6 +350,159 @@ impl X64FunctionCode {
);
}
}

fn emit_leave_frame(
assembler: &mut Assembler,
frame: &ControlFrame,
value_stack: &mut ValueStack,
) -> Result<(), CodegenError> {
let ret_ty = match frame.returns.len() {
1 => Some(frame.returns[0]),
0 => None,
_ => {
return Err(CodegenError {
message: "more than one block returns are not yet supported",
})
}
};

if ret_ty.is_some() && frame.loop_like {
return Err(CodegenError {
message: "return value is not supported for loops",
});
}

if value_stack.values.len() < frame.value_stack_depth_before + frame.returns.len() {
return Err(CodegenError {
message: "value stack underflow",
});
}

if let Some(ty) = ret_ty {
let ret = value_stack.pop()?;
match ret.location {
ValueLocation::Register(x) => {
dynasm!(
assembler
; mov rax, Rq(x)
);
}
ValueLocation::Stack => {
if is_dword(get_size_of_type(&ty)?) {
dynasm!(
assembler
; mov eax, [rsp]
; add rsp, 4
);
} else {
dynasm!(
assembler
; mov rax, [rsp]
; add rsp, 8
);
}
}
}
}

Ok(())
}

fn emit_block_end(
assembler: &mut Assembler,
control_stack: &mut ControlStack,
value_stack: &mut ValueStack,
) -> Result<(), CodegenError> {
let frame = match control_stack.frames.pop() {
Some(x) => x,
None => {
return Err(CodegenError {
message: "no frame",
})
}
};

Self::emit_leave_frame(assembler, &frame, value_stack)?;

if value_stack.values.len() != frame.value_stack_depth_before {
return Err(CodegenError {
message: "value_stack.values.len() != frame.value_stack_depth_before",
});
}

if !frame.loop_like {
dynasm!(
assembler
; => frame.label
);
}

if frame.returns.len() == 1 {
let loc = value_stack.push(frame.returns[0]);
match loc {
ValueLocation::Register(x) => {
let reg = Register::from_scratch_reg(x);
dynasm!(
assembler
; mov Rq(x as u8), rax
);
}
ValueLocation::Stack => {
if is_dword(get_size_of_type(&frame.returns[0])?) {
dynasm!(
assembler
; sub rsp, 4
; mov [rsp], eax
);
} else {
dynasm!(
assembler
; sub rsp, 8
; mov [rsp], rax
);
}
}
}
}

Ok(())
}

fn emit_jmp(
assembler: &mut Assembler,
control_stack: &ControlStack,
value_stack: &mut ValueStack,
relative_frame_offset: usize,
) -> Result<(), CodegenError> {
let frame = if relative_frame_offset >= control_stack.frames.len() {
return Err(CodegenError {
message: "jmp offset out of bounds",
});
} else {
&control_stack.frames[control_stack.frames.len() - 1 - relative_frame_offset]
};

Self::emit_leave_frame(assembler, frame, value_stack)?;

let mut sp_diff: usize = 0;

for i in 0..value_stack.values.len() - frame.value_stack_depth_before {
let vi = value_stack.values[value_stack.values.len() - 1 - i];
if vi.location == ValueLocation::Stack {
sp_diff += get_size_of_type(&vi.ty)?;
} else {
break;
}
}

dynasm!(
assembler
; add rsp, sp_diff as i32
; jmp =>frame.label
);

Ok(())
}
}

impl FunctionCodeGenerator for X64FunctionCode {
@@ -425,6 +580,10 @@ impl FunctionCodeGenerator for X64FunctionCode {
Ok(())
}
fn begin_body(&mut self) -> Result<(), CodegenError> {
self.control_stack = Some(ControlStack::new(
self.assembler.as_mut().unwrap().new_dynamic_label(),
self.returns.clone(),
));
Ok(())
}
fn feed_opcode(&mut self, op: Operator) -> Result<(), CodegenError> {
@@ -574,6 +733,7 @@ impl FunctionCodeGenerator for X64FunctionCode {
},
)?;
}
Operator::Block { ty } => {}
Operator::Drop => {
let info = self.value_stack.pop()?;
Self::gen_rt_pop(assembler, &info)?;
@@ -605,22 +765,56 @@ impl FunctionCodeGenerator for X64FunctionCode {
}
},
Operator::End => {
// todo
Self::emit_block_end(
assembler,
self.control_stack.as_mut().unwrap(),
&mut self.value_stack,
)?;
}
_ => unimplemented!(),
}
Ok(())
}

fn finalize(&mut self) -> Result<(), CodegenError> {
let assembler = self.assembler.as_mut().unwrap();

dynasm!(
assembler
; ud2
; => self.cleanup_label
);

if self.returns.len() == 1 {
if self.value_stack.values.len() != 1 {
return Err(CodegenError {
message: "returns.len() != value_stack.values.len()",
});
}
let value_info = self.value_stack.pop()?;
if value_info.ty != self.returns[0] {
return Err(CodegenError {
message: "return type mismatch",
});
}
if let ValueLocation::Register(x) = value_info.location {
let reg = Register::from_scratch_reg(x);
dynasm!(
assembler
; mov rax, Rq(reg as u8)
);
} else {
unreachable!();
}
}

dynasm!(
assembler
; mov rsp, rbp
; pop rbp
; ret
);

Ok(())
}
}
17 changes: 15 additions & 2 deletions lib/dynasm-backend/src/stack.rs
Original file line number Diff line number Diff line change
@@ -46,7 +46,7 @@ pub struct ValueInfo {
pub location: ValueLocation,
}

#[derive(Copy, Clone, Debug)]
#[derive(Copy, Clone, Debug, Eq, PartialEq)]
pub enum ValueLocation {
Register(u8),
Stack,
@@ -66,7 +66,7 @@ impl ValueLocation {
Ok(id)
} else {
Err(CodegenError {
message: "not a register location"
message: "not a register location",
})
}
}
@@ -140,3 +140,16 @@ impl ValueStack {
self.values.truncate(target_depth);
}
}

impl ControlStack {
pub fn new(label: DynamicLabel, returns: Vec<WpType>) -> ControlStack {
ControlStack {
frames: vec![ControlFrame {
label: label,
loop_like: false,
returns: returns,
value_stack_depth_before: 0,
}],
}
}
}