Files
wasmtime/lib/simplejit/tests/basic.rs
Dan Gohman d4f8eb7453 Introduce a TargetFrontendConfig type. (#570)
* Introduce a `TargetFrontendConfig` type.

`TargetFrontendConfig` is information specific to the target which is
provided to frontends to allow them to produce Cranelift IR for the
target. Currently this includes the pointer size and the default calling
convention.

The default calling convention is now inferred from the target, rather
than being a setting. cranelift-native is now just a provider of target
information, rather than also being a provider of settings, which gives
it a clearer role.

And instead of having cranelift-frontend routines require the whole
`TargetIsa`, just require the `TargetFrontendConfig`, and add a way to
get the `TargetFrontendConfig` from a `Module`.

Fixes #529.
Fixes #555.
2018-11-02 13:51:42 -07:00

153 lines
4.6 KiB
Rust

extern crate cranelift_codegen;
extern crate cranelift_entity;
extern crate cranelift_frontend;
extern crate cranelift_module;
extern crate cranelift_simplejit;
use cranelift_codegen::ir::*;
use cranelift_codegen::isa::CallConv;
use cranelift_codegen::Context;
use cranelift_entity::EntityRef;
use cranelift_frontend::*;
use cranelift_module::*;
use cranelift_simplejit::*;
#[test]
fn error_on_incompatible_sig_in_declare_function() {
let mut module: Module<SimpleJITBackend> = Module::new(SimpleJITBuilder::new());
let mut sig = Signature {
params: vec![AbiParam::new(types::I64)],
returns: vec![],
call_conv: CallConv::SystemV,
};
module
.declare_function("abc", Linkage::Local, &sig)
.unwrap();
sig.params[0] = AbiParam::new(types::I32);
module
.declare_function("abc", Linkage::Local, &sig)
.err()
.unwrap(); // Make sure this is an error
}
fn define_simple_function(module: &mut Module<SimpleJITBackend>) -> FuncId {
let sig = Signature {
params: vec![],
returns: vec![],
call_conv: CallConv::SystemV,
};
let func_id = module
.declare_function("abc", Linkage::Local, &sig)
.unwrap();
let mut ctx = Context::new();
ctx.func = Function::with_name_signature(ExternalName::user(0, func_id.index() as u32), sig);
let mut func_ctx = FunctionBuilderContext::new();
{
let mut bcx: FunctionBuilder = FunctionBuilder::new(&mut ctx.func, &mut func_ctx);
let ebb = bcx.create_ebb();
bcx.switch_to_block(ebb);
bcx.ins().return_(&[]);
}
module.define_function(func_id, &mut ctx).unwrap();
func_id
}
#[test]
fn double_finalize() {
let mut module: Module<SimpleJITBackend> = Module::new(SimpleJITBuilder::new());
define_simple_function(&mut module);
module.finalize_definitions();
// Calling `finalize_definitions` a second time without any new definitions
// should have no effect.
module.finalize_definitions();
}
#[test]
#[should_panic(expected = "Result::unwrap()` on an `Err` value: DuplicateDefinition(\"abc\")")]
fn panic_on_define_after_finalize() {
let mut module: Module<SimpleJITBackend> = Module::new(SimpleJITBuilder::new());
define_simple_function(&mut module);
module.finalize_definitions();
define_simple_function(&mut module);
}
#[test]
fn switch_error() {
use cranelift_codegen::settings;
let sig = Signature {
params: vec![AbiParam::new(types::I32)],
returns: vec![AbiParam::new(types::I32)],
call_conv: CallConv::SystemV,
};
let mut func = Function::with_name_signature(ExternalName::user(0, 0), sig);
let mut func_ctx = FunctionBuilderContext::new();
{
let mut bcx: FunctionBuilder = FunctionBuilder::new(&mut func, &mut func_ctx);
let start = bcx.create_ebb();
let bb0 = bcx.create_ebb();
let bb1 = bcx.create_ebb();
let bb2 = bcx.create_ebb();
let bb3 = bcx.create_ebb();
println!("{} {} {} {} {}", start, bb0, bb1, bb2, bb3);
bcx.declare_var(Variable::new(0), types::I32);
bcx.declare_var(Variable::new(1), types::I32);
let in_val = bcx.append_ebb_param(start, types::I32);
bcx.switch_to_block(start);
bcx.def_var(Variable::new(0), in_val);
bcx.ins().jump(bb0, &[]);
bcx.switch_to_block(bb0);
let discr = bcx.use_var(Variable::new(0));
let mut switch = cranelift_frontend::Switch::new();
for &(index, bb) in &[
(9, bb1),
(13, bb1),
(10, bb1),
(92, bb1),
(39, bb1),
(34, bb1),
] {
switch.set_entry(index, bb);
}
switch.emit(&mut bcx, discr, bb2);
bcx.switch_to_block(bb1);
let v = bcx.use_var(Variable::new(0));
bcx.def_var(Variable::new(1), v);
bcx.ins().jump(bb3, &[]);
bcx.switch_to_block(bb2);
let v = bcx.use_var(Variable::new(0));
bcx.def_var(Variable::new(1), v);
bcx.ins().jump(bb3, &[]);
bcx.switch_to_block(bb3);
let r = bcx.use_var(Variable::new(1));
bcx.ins().return_(&[r]);
bcx.seal_all_blocks();
bcx.finalize();
}
let flags = settings::Flags::new(settings::builder());
match cranelift_codegen::verify_function(&func, &flags) {
Ok(_) => {}
Err(err) => {
let pretty_error =
cranelift_codegen::print_errors::pretty_verifier_error(&func, None, None, err);
panic!("pretty_error:\n{}", pretty_error);
}
}
}