//! Test command for testing the register allocator. //! //! The `regalloc` test command runs each function through the register allocator after ensuring //! that all instructions are legal for the target. //! //! The resulting function is sent to `filecheck`. use cretonne::ir::Function; use cretonne; use cretonne::print_errors::pretty_error; use cton_reader::TestCommand; use subtest::{SubTest, Context, Result, run_filecheck}; use std::borrow::Cow; use std::fmt::Write; struct TestRegalloc; pub fn subtest(parsed: &TestCommand) -> Result> { assert_eq!(parsed.command, "regalloc"); if !parsed.options.is_empty() { Err(format!("No options allowed on {}", parsed)) } else { Ok(Box::new(TestRegalloc)) } } impl SubTest for TestRegalloc { fn name(&self) -> Cow { Cow::from("regalloc") } fn is_mutating(&self) -> bool { true } fn needs_isa(&self) -> bool { true } fn run(&self, func: Cow, context: &Context) -> Result<()> { let isa = context.isa.expect("register allocator needs an ISA"); // Create a compilation context, and drop in the function. let mut comp_ctx = cretonne::Context::new(); comp_ctx.func = func.into_owned(); comp_ctx.compute_cfg(); // TODO: Should we have an option to skip legalization? comp_ctx.legalize(isa).map_err(|e| { pretty_error(&comp_ctx.func, context.isa, e) })?; comp_ctx.compute_domtree(); comp_ctx.regalloc(isa).map_err(|e| { pretty_error(&comp_ctx.func, context.isa, e) })?; let mut text = String::new(); write!(&mut text, "{}", &comp_ctx.func.display(Some(isa))) .map_err(|e| e.to_string())?; run_filecheck(&text, context) } }