Run verifier after legalizer and regalloc file tests.
Run the verify_contexti() function after invoking the legalize() and regalloc() context functions. This will help catch bad code produced by these passes.
This commit is contained in:
@@ -4,11 +4,11 @@
|
|||||||
//! the result to filecheck.
|
//! the result to filecheck.
|
||||||
|
|
||||||
use std::borrow::Cow;
|
use std::borrow::Cow;
|
||||||
use cretonne::{legalize_function, write_function};
|
use cretonne::{self, write_function};
|
||||||
use cretonne::flowgraph::ControlFlowGraph;
|
|
||||||
use cretonne::ir::Function;
|
use cretonne::ir::Function;
|
||||||
use cton_reader::TestCommand;
|
use cton_reader::TestCommand;
|
||||||
use filetest::subtest::{SubTest, Context, Result, run_filecheck};
|
use filetest::subtest::{SubTest, Context, Result, run_filecheck};
|
||||||
|
use utils::pretty_verifier_error;
|
||||||
|
|
||||||
struct TestLegalizer;
|
struct TestLegalizer;
|
||||||
|
|
||||||
@@ -35,13 +35,16 @@ impl SubTest for TestLegalizer {
|
|||||||
}
|
}
|
||||||
|
|
||||||
fn run(&self, func: Cow<Function>, context: &Context) -> Result<()> {
|
fn run(&self, func: Cow<Function>, context: &Context) -> Result<()> {
|
||||||
let mut func = func.into_owned();
|
let mut comp_ctx = cretonne::Context::new();
|
||||||
|
comp_ctx.func = func.into_owned();
|
||||||
let isa = context.isa.expect("legalizer needs an ISA");
|
let isa = context.isa.expect("legalizer needs an ISA");
|
||||||
let mut cfg = ControlFlowGraph::with_function(&func);
|
|
||||||
legalize_function(&mut func, &mut cfg, isa);
|
comp_ctx.flowgraph();
|
||||||
|
comp_ctx.legalize(isa);
|
||||||
|
comp_ctx.verify(isa).map_err(|e| pretty_verifier_error(&comp_ctx.func, e))?;
|
||||||
|
|
||||||
let mut text = String::new();
|
let mut text = String::new();
|
||||||
write_function(&mut text, &func, Some(isa)).map_err(|e| e.to_string())?;
|
write_function(&mut text, &comp_ctx.func, Some(isa)).map_err(|e| e.to_string())?;
|
||||||
run_filecheck(&text, context)
|
run_filecheck(&text, context)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -5,11 +5,12 @@
|
|||||||
//!
|
//!
|
||||||
//! The resulting function is sent to `filecheck`.
|
//! The resulting function is sent to `filecheck`.
|
||||||
|
|
||||||
use std::borrow::Cow;
|
|
||||||
use cretonne::{self, write_function};
|
|
||||||
use cretonne::ir::Function;
|
use cretonne::ir::Function;
|
||||||
|
use cretonne::{self, write_function};
|
||||||
use cton_reader::TestCommand;
|
use cton_reader::TestCommand;
|
||||||
use filetest::subtest::{SubTest, Context, Result, run_filecheck};
|
use filetest::subtest::{SubTest, Context, Result, run_filecheck};
|
||||||
|
use std::borrow::Cow;
|
||||||
|
use utils::pretty_verifier_error;
|
||||||
|
|
||||||
struct TestRegalloc;
|
struct TestRegalloc;
|
||||||
|
|
||||||
@@ -46,6 +47,7 @@ impl SubTest for TestRegalloc {
|
|||||||
// TODO: Should we have an option to skip legalization?
|
// TODO: Should we have an option to skip legalization?
|
||||||
comp_ctx.legalize(isa);
|
comp_ctx.legalize(isa);
|
||||||
comp_ctx.regalloc(isa);
|
comp_ctx.regalloc(isa);
|
||||||
|
comp_ctx.verify(isa).map_err(|e| pretty_verifier_error(&comp_ctx.func, e))?;
|
||||||
|
|
||||||
let mut text = String::new();
|
let mut text = String::new();
|
||||||
write_function(&mut text, &comp_ctx.func, Some(isa)).map_err(|e| e.to_string())?;
|
write_function(&mut text, &comp_ctx.func, Some(isa)).map_err(|e| e.to_string())?;
|
||||||
|
|||||||
@@ -1,8 +1,11 @@
|
|||||||
//! Utility functions.
|
//! Utility functions.
|
||||||
|
|
||||||
use std::path::Path;
|
use cretonne::ir::entities::AnyEntity;
|
||||||
|
use cretonne::{ir, verifier, write_function};
|
||||||
|
use std::fmt::Write;
|
||||||
use std::fs::File;
|
use std::fs::File;
|
||||||
use std::io::{Result, Read};
|
use std::io::{Result, Read};
|
||||||
|
use std::path::Path;
|
||||||
|
|
||||||
/// Read an entire file into a string.
|
/// Read an entire file into a string.
|
||||||
pub fn read_to_string<P: AsRef<Path>>(path: P) -> Result<String> {
|
pub fn read_to_string<P: AsRef<Path>>(path: P) -> Result<String> {
|
||||||
@@ -29,6 +32,19 @@ pub fn match_directive<'a>(comment: &'a str, directive: &str) -> Option<&'a str>
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Pretty-print a verifier error.
|
||||||
|
pub fn pretty_verifier_error(func: &ir::Function, err: verifier::Error) -> String {
|
||||||
|
let mut msg = err.to_string();
|
||||||
|
match err.location {
|
||||||
|
AnyEntity::Inst(inst) => {
|
||||||
|
write!(msg, "\n{}: {}\n\n", inst, func.dfg.display_inst(inst)).unwrap()
|
||||||
|
}
|
||||||
|
_ => {}
|
||||||
|
}
|
||||||
|
write_function(&mut msg, func, None).unwrap();
|
||||||
|
msg
|
||||||
|
}
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
fn test_match_directive() {
|
fn test_match_directive() {
|
||||||
assert_eq!(match_directive("; foo: bar ", "foo:"), Some("bar"));
|
assert_eq!(match_directive("; foo: bar ", "foo:"), Some("bar"));
|
||||||
|
|||||||
@@ -15,6 +15,7 @@ use ir::Function;
|
|||||||
use isa::TargetIsa;
|
use isa::TargetIsa;
|
||||||
use legalize_function;
|
use legalize_function;
|
||||||
use regalloc;
|
use regalloc;
|
||||||
|
use verifier;
|
||||||
|
|
||||||
/// Persistent data structures and compilation pipeline.
|
/// Persistent data structures and compilation pipeline.
|
||||||
pub struct Context {
|
pub struct Context {
|
||||||
@@ -45,6 +46,16 @@ impl Context {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Run the verifier on the function.
|
||||||
|
///
|
||||||
|
/// Also check that the dominator tree and control flow graph are consistent with the function.
|
||||||
|
///
|
||||||
|
/// The `TargetIsa` argument is currently unused, but the verifier will soon be able to also
|
||||||
|
/// check ISA-dependent constraints.
|
||||||
|
pub fn verify<'a, ISA: Into<Option<&'a TargetIsa>>>(&self, _isa: ISA) -> verifier::Result<()> {
|
||||||
|
verifier::verify_context(self)
|
||||||
|
}
|
||||||
|
|
||||||
/// Run the legalizer for `isa` on the function.
|
/// Run the legalizer for `isa` on the function.
|
||||||
pub fn legalize(&mut self, isa: &TargetIsa) {
|
pub fn legalize(&mut self, isa: &TargetIsa) {
|
||||||
legalize_function(&mut self.func, &mut self.cfg, isa);
|
legalize_function(&mut self.func, &mut self.cfg, isa);
|
||||||
|
|||||||
Reference in New Issue
Block a user