1
Fork 0

cover statements

This commit is contained in:
Oğuz Ağcayazı 2023-11-08 15:25:48 +03:00 committed by ouz-a
parent ae179a04b6
commit 0f0e9baf19

View file

@ -2,37 +2,43 @@ use std::io;
use rustc_middle::ty::TyCtxt; use rustc_middle::ty::TyCtxt;
use stable_mir::{ use stable_mir::{
mir::{Mutability, Operand, Rvalue, StatementKind},
ty::{RigidTy, TyKind}, ty::{RigidTy, TyKind},
CrateItem, mir::Mutability, CrateItem,
}; };
use super::{run, RustcInternal}; use super::{run, RustcInternal};
pub fn write_smir_pretty<'tcx>(tcx: TyCtxt<'tcx>, w: &mut dyn io::Write) -> io::Result<()> { pub fn write_smir_pretty<'tcx>(tcx: TyCtxt<'tcx>, w: &mut dyn io::Write) -> io::Result<()> {
run(tcx, || { run(tcx, || {
let items = stable_mir::all_local_items(); let items = stable_mir::all_local_items();
items.iter().for_each(|item| { items.iter().for_each(|item| {
// Because we can't return a Result from a closure, we have to unwrap here. // Because we can't return a Result from a closure, we have to unwrap here.
writeln!(w, "{}", function_name(*item,tcx)).unwrap(); writeln!(w, "{}", function_name(*item, tcx)).unwrap();
writeln!(w, "{}", function_body(*item,tcx)).unwrap(); writeln!(w, "{}", function_body(*item, tcx)).unwrap();
writeln!(w, "------------------").unwrap();
item.body().blocks.iter().for_each(|block| {
block.statements.iter().for_each(|statement| {
writeln!(w, "{}", pretty_statement(&statement.kind, tcx)).unwrap();
});
})
}) })
}); });
Ok(()) Ok(())
} }
pub fn function_name(item: CrateItem,tcx: TyCtxt<'_>) -> String { pub fn function_name(item: CrateItem, tcx: TyCtxt<'_>) -> String {
let mut name = String::new(); let mut name = String::new();
let body = item.body(); let body = item.body();
name.push_str("fn "); name.push_str("fn ");
name.push_str(item.name().as_str()); name.push_str(item.name().as_str());
if body.arg_locals().is_empty() { if body.arg_locals().is_empty() {
name.push_str("()"); name.push_str("()");
}else{ } else {
name.push_str("("); name.push_str("(");
} }
body.arg_locals().iter().for_each(|local| { body.arg_locals().iter().for_each(|local| {
name.push_str(format!("_{}: ",local.local).as_str()); name.push_str(format!("_{}: ", local.local).as_str());
name.push_str(&pretty_ty(local.ty.kind(), tcx)); name.push_str(&pretty_ty(local.ty.kind(), tcx));
}); });
if !body.arg_locals().is_empty() { if !body.arg_locals().is_empty() {
@ -45,20 +51,18 @@ pub fn function_name(item: CrateItem,tcx: TyCtxt<'_>) -> String {
name name
} }
pub fn function_body(item: CrateItem,_tcx: TyCtxt<'_>) -> String { pub fn function_body(item: CrateItem, _tcx: TyCtxt<'_>) -> String {
let mut body_str = String::new(); let mut body_str = String::new();
let body = item.body(); let body = item.body();
body.inner_locals().iter().for_each(|local| { body.inner_locals().iter().for_each(|local| {
body_str.push_str(" "); body_str.push_str(" ");
body_str.push_str(format!("let {}",ret_mutability(&local.mutability)).as_str()); body_str.push_str(format!("let {}", ret_mutability(&local.mutability)).as_str());
body_str.push_str(format!("_{}: ",local.local).as_str()); body_str.push_str(format!("_{}: ", local.local).as_str());
body_str.push_str(format!("{}",pretty_ty(local.ty.kind(), _tcx)).as_str()); body_str.push_str(format!("{}", pretty_ty(local.ty.kind(), _tcx)).as_str());
body_str.push_str(";\n"); body_str.push_str(";\n");
}); });
body_str.push_str("}"); body_str.push_str("}");
body_str body_str
} }
pub fn ret_mutability(mutability: &Mutability) -> String { pub fn ret_mutability(mutability: &Mutability) -> String {
@ -68,7 +72,129 @@ pub fn ret_mutability(mutability: &Mutability) -> String {
} }
} }
pub fn pretty_ty<'tcx>(ty: TyKind,tcx: TyCtxt<'tcx>) -> String { pub fn pretty_statement(statement: &StatementKind, tcx: TyCtxt<'_>) -> String {
let mut pretty = String::new();
match statement {
StatementKind::Assign(place, rval) => {
pretty.push_str(format!("_{} = ", place.local).as_str());
pretty.push_str(&pretty_rvalue(rval, tcx))
}
StatementKind::FakeRead(_, _) => todo!(),
StatementKind::SetDiscriminant { .. } => todo!(),
StatementKind::Deinit(_) => todo!(),
StatementKind::StorageLive(_) => todo!(),
StatementKind::StorageDead(_) => todo!(),
StatementKind::Retag(_, _) => todo!(),
StatementKind::PlaceMention(_) => todo!(),
StatementKind::AscribeUserType { .. } => todo!(),
StatementKind::Coverage(_) => todo!(),
StatementKind::Intrinsic(_) => todo!(),
StatementKind::ConstEvalCounter => (),
StatementKind::Nop => (),
}
pretty
}
pub fn pretty_operand(operand: &Operand, _tcx: TyCtxt<'_>) -> String {
let mut pretty = String::new();
match operand {
Operand::Copy(copy) => {
pretty.push_str("");
pretty.push_str(format!("{}", copy.local).as_str());
}
Operand::Move(mv) => {
pretty.push_str("move");
pretty.push_str(format!("{}", mv.local).as_str());
}
Operand::Constant(cnst) => {
pretty.push_str("const ");
pretty.push_str(cnst.literal.internal_via_tls().to_string().as_str());
}
}
pretty
}
pub fn pretty_rvalue(rval: &Rvalue, tcx: TyCtxt<'_>) -> String {
let mut pretty = String::new();
match rval {
Rvalue::AddressOf(muta, addr) => {
pretty.push_str("address_of");
pretty.push_str(&ret_mutability(&muta));
pretty.push_str(format!("{}", addr.local).as_str());
}
Rvalue::Aggregate(aggregatekind, operands) => {
pretty.push_str(format!("{:#?}", aggregatekind).as_str());
pretty.push_str("(");
operands.iter().enumerate().for_each(|(i, op)| {
pretty.push_str(&pretty_operand(op, tcx));
if i != operands.len() - 1 {
pretty.push_str(", ");
}
});
pretty.push_str(")");
}
Rvalue::BinaryOp(bin, op, op2) => {
pretty.push_str(&pretty_operand(op, tcx));
pretty.push_str(" ");
pretty.push_str(format!("{:#?}", bin).as_str());
pretty.push_str(" ");
pretty.push_str(&pretty_operand(op2, tcx));
}
Rvalue::Cast(_, op, ty) => {
pretty.push_str(&pretty_operand(op, tcx));
pretty.push_str(" as ");
pretty.push_str(&pretty_ty(ty.kind(), tcx));
}
Rvalue::CheckedBinaryOp(bin, op1, op2) => {
pretty.push_str(&pretty_operand(op1, tcx));
pretty.push_str(" ");
pretty.push_str(format!("{:#?}", bin).as_str());
pretty.push_str(" ");
pretty.push_str(&pretty_operand(op2, tcx));
}
Rvalue::CopyForDeref(deref) => {
pretty.push_str("CopyForDeref");
pretty.push_str(format!("{}", deref.local).as_str());
}
Rvalue::Discriminant(place) => {
pretty.push_str("discriminant");
pretty.push_str(format!("{}", place.local).as_str());
}
Rvalue::Len(len) => {
pretty.push_str("len");
pretty.push_str(format!("{}", len.local).as_str());
}
Rvalue::Ref(_, borrowkind, place) => {
pretty.push_str("ref");
pretty.push_str(format!("{:#?}", borrowkind).as_str());
pretty.push_str(format!("{}", place.local).as_str());
}
Rvalue::Repeat(op, cnst) => {
pretty.push_str(&pretty_operand(op, tcx));
pretty.push_str(" ");
pretty.push_str(&pretty_ty(cnst.ty().kind(), tcx));
}
Rvalue::ShallowInitBox(_, _) => todo!(),
Rvalue::ThreadLocalRef(item) => {
pretty.push_str("thread_local_ref");
pretty.push_str(format!("{:#?}", item).as_str());
}
Rvalue::NullaryOp(nul, ty) => {
pretty.push_str(format!("{:#?}", nul).as_str());
pretty.push_str(&&pretty_ty(ty.kind(), tcx));
pretty.push_str(" ");
}
Rvalue::UnaryOp(un, op) => {
pretty.push_str(&pretty_operand(op, tcx));
pretty.push_str(" ");
pretty.push_str(format!("{:#?}", un).as_str());
}
Rvalue::Use(op) => pretty.push_str(&pretty_operand(op, tcx)),
}
pretty
}
pub fn pretty_ty(ty: TyKind, tcx: TyCtxt<'_>) -> String {
let mut pretty = String::new(); let mut pretty = String::new();
pretty.push_str(""); pretty.push_str("");
match ty { match ty {
@ -95,14 +221,17 @@ pub fn pretty_ty<'tcx>(ty: TyKind,tcx: TyCtxt<'tcx>) -> String {
stable_mir::ty::FloatTy::F32 => "f32".to_string(), stable_mir::ty::FloatTy::F32 => "f32".to_string(),
stable_mir::ty::FloatTy::F64 => "f64".to_string(), stable_mir::ty::FloatTy::F64 => "f64".to_string(),
}, },
RigidTy::Adt(def, _) => format!("{:#?}", tcx.type_of(def.0.internal_via_tls()).instantiate_identity()), RigidTy::Adt(def, _) => {
format!("{:#?}", tcx.type_of(def.0.internal_via_tls()).instantiate_identity())
}
RigidTy::Foreign(_) => format!("{:#?}", rigid_ty), RigidTy::Foreign(_) => format!("{:#?}", rigid_ty),
RigidTy::Str => "str".to_string(), RigidTy::Str => "str".to_string(),
RigidTy::Array(_ty, len) => { RigidTy::Array(_ty, len) => {
format!("[{};{:#?}]", 1,len.internal_via_tls())}, format!("[{};{:#?}]", 1, len.internal_via_tls())
RigidTy::Slice(ty) => pretty_ty(ty.kind(),tcx), }
RigidTy::Slice(ty) => pretty_ty(ty.kind(), tcx),
RigidTy::RawPtr(_, _) => format!("{:#?}", rigid_ty), RigidTy::RawPtr(_, _) => format!("{:#?}", rigid_ty),
RigidTy::Ref(_, ty, _) => pretty_ty(ty.kind(),tcx), RigidTy::Ref(_, ty, _) => pretty_ty(ty.kind(), tcx),
RigidTy::FnDef(_, _) => format!("{:#?}", rigid_ty), RigidTy::FnDef(_, _) => format!("{:#?}", rigid_ty),
RigidTy::FnPtr(_) => format!("{:#?}", rigid_ty), RigidTy::FnPtr(_) => format!("{:#?}", rigid_ty),
RigidTy::Closure(_, _) => format!("{:#?}", rigid_ty), RigidTy::Closure(_, _) => format!("{:#?}", rigid_ty),
@ -110,13 +239,13 @@ pub fn pretty_ty<'tcx>(ty: TyKind,tcx: TyCtxt<'tcx>) -> String {
RigidTy::Dynamic(_, _, _) => format!("{:#?}", rigid_ty), RigidTy::Dynamic(_, _, _) => format!("{:#?}", rigid_ty),
RigidTy::Never => "!".to_string(), RigidTy::Never => "!".to_string(),
RigidTy::Tuple(tuple) => { RigidTy::Tuple(tuple) => {
if tuple.is_empty(){ if tuple.is_empty() {
"()".to_string() "()".to_string()
}else { } else {
let mut tuple_str = String::new(); let mut tuple_str = String::new();
tuple_str.push_str("("); tuple_str.push_str("(");
tuple.iter().enumerate().for_each(|(i,ty)| { tuple.iter().enumerate().for_each(|(i, ty)| {
tuple_str.push_str(&pretty_ty(ty.kind(),tcx)); tuple_str.push_str(&pretty_ty(ty.kind(), tcx));
if i != tuple.len() - 1 { if i != tuple.len() - 1 {
tuple_str.push_str(", "); tuple_str.push_str(", ");
} }
@ -124,7 +253,7 @@ pub fn pretty_ty<'tcx>(ty: TyKind,tcx: TyCtxt<'tcx>) -> String {
tuple_str.push_str(")"); tuple_str.push_str(")");
tuple_str tuple_str
} }
}, }
}, },
TyKind::Alias(_, _) => format!("{:#?}", ty), TyKind::Alias(_, _) => format!("{:#?}", ty),
TyKind::Param(_) => format!("{:#?}", ty), TyKind::Param(_) => format!("{:#?}", ty),