use alloc::{boxed::Box, rc::Rc};
use crate::{
BlockRef, Builder, Context, Ident, OpBuilder, Type,
diagnostics::Report,
dialects::builtin::{
BuiltinOpBuilder, Function, FunctionBuilder, FunctionRef, ModuleRef,
attributes::{Signature, Visibility},
},
interner,
pass::{OperationPass, Pass, PassManager},
};
#[cfg(feature = "logging")]
pub fn enable_compiler_instrumentation() {
let _ = midenc_log::Builder::from_env("MIDENC_TRACE")
.format_timestamp(None)
.is_test(true)
.try_init();
}
#[cfg(not(feature = "logging"))]
pub fn enable_compiler_instrumentation() {}
pub struct Test {
context: Rc<Context>,
name: &'static str,
builder: OpBuilder,
module: Option<ModuleRef>,
function: Option<FunctionRef>,
}
impl Default for Test {
fn default() -> Self {
enable_compiler_instrumentation();
let context = Rc::new(Context::default());
let builder = OpBuilder::new(context.clone());
Self {
context,
name: "test",
builder,
module: None,
function: None,
}
}
}
impl Test {
pub fn new(name: &'static str, params: &[Type], results: &[Type]) -> Self {
let mut test = Self::named(name);
test.with_function(name, params, results);
test
}
pub fn named(name: &'static str) -> Self {
Self {
name,
..Default::default()
}
}
pub fn in_module(mut self, name: impl Into<interner::Symbol>) -> Self {
let name = name.into();
let module = self.builder.create_module(Ident::with_empty_span(name)).unwrap();
let module_body = module.borrow().body().as_region_ref();
self.builder.create_block(module_body, None, &[]);
self.module = Some(module);
self
}
pub fn with_function(&mut self, name: &'static str, params: &[Type], results: &[Type]) {
self.name = name;
let function = self
.builder
.create_function(
Ident::with_empty_span(name.into()),
Visibility::Public,
Signature::new(&self.context, params.iter().cloned(), results.iter().cloned()),
)
.expect("failed to create function");
let _ = FunctionBuilder::new(function, &mut self.builder);
self.function = Some(function);
}
pub fn define_function(
&mut self,
name: impl Into<interner::Symbol>,
params: &[Type],
results: &[Type],
) -> FunctionRef {
let module = self.module.expect("cannot define non-test functions without a module");
let module_body = { module.borrow().body().entry_block_ref().unwrap() };
self.builder.set_insertion_point_to_end(module_body);
self.builder
.create_function(
Ident::with_empty_span(name.into()),
Visibility::Public,
Signature::new(&self.context, params.iter().cloned(), results.iter().cloned()),
)
.expect("failed to define function")
}
pub const fn name(&self) -> &'static str {
self.name
}
pub fn context(&self) -> &Context {
&self.context
}
pub fn context_rc(&self) -> Rc<Context> {
self.context.clone()
}
pub fn function(&self) -> FunctionRef {
self.function.unwrap()
}
pub fn module(&self) -> ModuleRef {
self.module.unwrap()
}
pub fn function_builder(&mut self) -> FunctionBuilder<'_, OpBuilder> {
FunctionBuilder::new(self.function(), &mut self.builder)
}
pub fn builder_mut(&mut self) -> &mut OpBuilder {
&mut self.builder
}
pub fn entry_block(&self) -> BlockRef {
self.function.unwrap().borrow().entry_block()
}
pub fn apply_pass<P: Pass + Default>(&self, verify: bool) -> Result<(), Report> {
let mut pm = PassManager::on::<Function>(self.context_rc(), crate::pass::Nesting::Implicit);
pm.add_pass(Box::new(P::default()));
pm.enable_verifier(verify);
pm.run(self.function().as_operation_ref())
}
pub fn apply_boxed_pass(
&self,
pass: Box<dyn OperationPass>,
verify: bool,
) -> Result<(), Report> {
let mut pm = PassManager::on::<Function>(self.context_rc(), crate::pass::Nesting::Implicit);
pm.add_pass(pass);
pm.enable_verifier(verify);
pm.run(self.function().as_operation_ref())
}
pub fn apply_passes(
&mut self,
passes: impl IntoIterator<Item = Box<dyn OperationPass>>,
verify: bool,
) -> Result<(), Report> {
let mut pm = PassManager::on::<Function>(self.context_rc(), crate::pass::Nesting::Implicit);
for pass in passes {
pm.add_pass(pass);
}
pm.enable_verifier(verify);
pm.run(self.function().as_operation_ref())
}
}