2015-12-29 20:06:19 -06:00
|
|
|
// Copyright 2014 The Rust Project Developers. See the COPYRIGHT
|
|
|
|
// file at the top-level directory of this distribution and at
|
|
|
|
// http://rust-lang.org/COPYRIGHT.
|
|
|
|
//
|
|
|
|
// Licensed under the Apache License, Version 2.0 <LICENSE-APACHE or
|
|
|
|
// http://www.apache.org/licenses/LICENSE-2.0> or the MIT license
|
|
|
|
// <LICENSE-MIT or http://opensource.org/licenses/MIT>, at your
|
|
|
|
// option. This file may not be copied, modified, or distributed
|
|
|
|
// except according to those terms.
|
|
|
|
|
2016-03-22 11:52:34 -04:00
|
|
|
use build::{Location, ScopeAuxiliary};
|
2015-12-29 20:06:19 -06:00
|
|
|
use rustc::mir::repr::*;
|
2016-03-22 10:08:44 -04:00
|
|
|
use rustc::middle::ty::{self, TyCtxt};
|
|
|
|
use rustc_data_structures::fnv::FnvHashMap;
|
2015-12-29 20:06:19 -06:00
|
|
|
use std::io::{self, Write};
|
2016-03-12 19:07:00 +02:00
|
|
|
use syntax::ast::NodeId;
|
2016-03-22 10:08:44 -04:00
|
|
|
use syntax::codemap::Span;
|
2015-12-29 20:06:19 -06:00
|
|
|
|
|
|
|
const INDENT: &'static str = " ";
|
|
|
|
|
|
|
|
/// Write out a human-readable textual representation for the given MIR.
|
2016-03-22 10:08:44 -04:00
|
|
|
pub fn write_mir_pretty<'a, 'tcx, I>(tcx: &TyCtxt<'tcx>,
|
2016-03-19 07:13:54 -04:00
|
|
|
iter: I,
|
|
|
|
w: &mut Write)
|
|
|
|
-> io::Result<()>
|
|
|
|
where I: Iterator<Item=(&'a NodeId, &'a Mir<'tcx>)>, 'tcx: 'a
|
|
|
|
{
|
|
|
|
for (&node_id, mir) in iter {
|
2016-03-22 11:52:34 -04:00
|
|
|
write_mir_fn(tcx, node_id, mir, w, None)?;
|
2015-12-29 20:06:19 -06:00
|
|
|
}
|
2016-03-12 19:07:00 +02:00
|
|
|
Ok(())
|
2015-12-29 20:06:19 -06:00
|
|
|
}
|
|
|
|
|
2016-03-22 11:52:34 -04:00
|
|
|
enum Annotation {
|
2016-03-22 10:08:44 -04:00
|
|
|
EnterScope(ScopeId),
|
|
|
|
ExitScope(ScopeId),
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn write_mir_fn<'tcx>(tcx: &TyCtxt<'tcx>,
|
2016-03-19 07:13:54 -04:00
|
|
|
node_id: NodeId,
|
|
|
|
mir: &Mir<'tcx>,
|
2016-03-22 10:08:44 -04:00
|
|
|
w: &mut Write,
|
2016-03-22 11:52:34 -04:00
|
|
|
auxiliary: Option<&Vec<ScopeAuxiliary>>)
|
2016-03-19 07:13:54 -04:00
|
|
|
-> io::Result<()> {
|
2016-03-22 11:52:34 -04:00
|
|
|
// compute scope/entry exit annotations
|
|
|
|
let mut annotations = FnvHashMap();
|
|
|
|
if let Some(auxiliary) = auxiliary {
|
|
|
|
for (index, auxiliary) in auxiliary.iter().enumerate() {
|
|
|
|
let scope_id = ScopeId::new(index);
|
|
|
|
|
|
|
|
annotations.entry(auxiliary.dom)
|
|
|
|
.or_insert(vec![])
|
|
|
|
.push(Annotation::EnterScope(scope_id));
|
|
|
|
|
|
|
|
for &loc in &auxiliary.postdoms {
|
|
|
|
annotations.entry(loc)
|
|
|
|
.or_insert(vec![])
|
|
|
|
.push(Annotation::ExitScope(scope_id));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-03-19 07:13:54 -04:00
|
|
|
write_mir_intro(tcx, node_id, mir, w)?;
|
|
|
|
for block in mir.all_basic_blocks() {
|
2016-03-22 11:52:34 -04:00
|
|
|
write_basic_block(tcx, block, mir, w, &annotations)?;
|
2016-03-22 10:08:44 -04:00
|
|
|
}
|
|
|
|
|
2016-03-22 11:52:34 -04:00
|
|
|
// construct a scope tree and write it out
|
2016-03-22 10:08:44 -04:00
|
|
|
let mut scope_tree: FnvHashMap<Option<ScopeId>, Vec<ScopeId>> = FnvHashMap();
|
|
|
|
for (index, scope_data) in mir.scopes.vec.iter().enumerate() {
|
|
|
|
scope_tree.entry(scope_data.parent_scope)
|
|
|
|
.or_insert(vec![])
|
|
|
|
.push(ScopeId::new(index));
|
2016-03-19 07:13:54 -04:00
|
|
|
}
|
2016-03-22 11:52:34 -04:00
|
|
|
write_scope_tree(tcx, mir, auxiliary, &scope_tree, w, None, 1)?;
|
2016-03-22 10:08:44 -04:00
|
|
|
|
2016-03-19 07:13:54 -04:00
|
|
|
writeln!(w, "}}")?;
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
2015-12-29 20:06:19 -06:00
|
|
|
/// Write out a human-readable textual representation for the given basic block.
|
2016-03-22 10:08:44 -04:00
|
|
|
fn write_basic_block(tcx: &TyCtxt,
|
|
|
|
block: BasicBlock,
|
|
|
|
mir: &Mir,
|
|
|
|
w: &mut Write,
|
|
|
|
annotations: &FnvHashMap<Location, Vec<Annotation>>)
|
|
|
|
-> io::Result<()> {
|
2015-12-29 20:06:19 -06:00
|
|
|
let data = mir.basic_block_data(block);
|
|
|
|
|
|
|
|
// Basic block label at the top.
|
2016-03-22 22:01:37 -05:00
|
|
|
writeln!(w, "\n{}{:?}: {{", INDENT, block)?;
|
2015-12-29 20:06:19 -06:00
|
|
|
|
|
|
|
// List of statements in the middle.
|
2016-03-22 10:08:44 -04:00
|
|
|
let mut current_location = Location { block: block, statement_index: 0 };
|
2015-12-29 20:06:19 -06:00
|
|
|
for statement in &data.statements {
|
2016-03-22 10:08:44 -04:00
|
|
|
if let Some(ref annotations) = annotations.get(¤t_location) {
|
|
|
|
for annotation in annotations.iter() {
|
|
|
|
match *annotation {
|
|
|
|
Annotation::EnterScope(id) =>
|
|
|
|
writeln!(w, "{0}{0}// Enter Scope({1})",
|
|
|
|
INDENT, id.index())?,
|
|
|
|
Annotation::ExitScope(id) =>
|
|
|
|
writeln!(w, "{0}{0}// Exit Scope({1})",
|
|
|
|
INDENT, id.index())?,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
writeln!(w, "{0}{0}{1:?}; // {2}",
|
|
|
|
INDENT,
|
|
|
|
statement,
|
|
|
|
comment(tcx, statement.scope, statement.span))?;
|
|
|
|
|
|
|
|
current_location.statement_index += 1;
|
2015-12-29 20:06:19 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
// Terminator at the bottom.
|
2016-03-22 10:08:44 -04:00
|
|
|
writeln!(w, "{0}{0}{1:?}; // {2}",
|
|
|
|
INDENT,
|
|
|
|
data.terminator(),
|
|
|
|
comment(tcx, data.terminator().scope, data.terminator().span))?;
|
2015-12-29 20:06:19 -06:00
|
|
|
|
|
|
|
writeln!(w, "{}}}", INDENT)
|
|
|
|
}
|
|
|
|
|
2016-03-22 10:08:44 -04:00
|
|
|
fn comment(tcx: &TyCtxt,
|
|
|
|
scope: ScopeId,
|
|
|
|
span: Span)
|
|
|
|
-> String {
|
|
|
|
format!("Scope({}) at {}", scope.index(), tcx.sess.codemap().span_to_string(span))
|
|
|
|
}
|
|
|
|
|
|
|
|
fn write_scope_tree(tcx: &TyCtxt,
|
|
|
|
mir: &Mir,
|
2016-03-22 11:52:34 -04:00
|
|
|
auxiliary: Option<&Vec<ScopeAuxiliary>>,
|
2016-03-22 10:08:44 -04:00
|
|
|
scope_tree: &FnvHashMap<Option<ScopeId>, Vec<ScopeId>>,
|
|
|
|
w: &mut Write,
|
|
|
|
parent: Option<ScopeId>,
|
|
|
|
depth: usize)
|
|
|
|
-> io::Result<()> {
|
|
|
|
for &child in scope_tree.get(&parent).unwrap_or(&vec![]) {
|
|
|
|
let indent = depth * INDENT.len();
|
|
|
|
let data = &mir.scopes[child];
|
|
|
|
assert_eq!(data.parent_scope, parent);
|
|
|
|
writeln!(w, "{0:1$}Scope({2}) {{", "", indent, child.index())?;
|
2016-03-22 11:52:34 -04:00
|
|
|
|
2016-03-22 10:08:44 -04:00
|
|
|
let indent = indent + INDENT.len();
|
|
|
|
if let Some(parent) = parent {
|
|
|
|
writeln!(w, "{0:1$}Parent: Scope({2})", "", indent, parent.index())?;
|
|
|
|
}
|
2016-03-22 11:52:34 -04:00
|
|
|
|
|
|
|
if let Some(auxiliary) = auxiliary {
|
|
|
|
let extent = auxiliary[child.index()].extent;
|
|
|
|
let data = tcx.region_maps.code_extent_data(extent);
|
|
|
|
writeln!(w, "{0:1$}Extent: {2:?}", "", indent, data)?;
|
|
|
|
}
|
|
|
|
|
|
|
|
write_scope_tree(tcx, mir, auxiliary, scope_tree, w,
|
|
|
|
Some(child), depth + 1)?;
|
2016-03-22 10:08:44 -04:00
|
|
|
}
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
2015-12-29 20:06:19 -06:00
|
|
|
/// Write out a human-readable textual representation of the MIR's `fn` type and the types of its
|
|
|
|
/// local variables (both user-defined bindings and compiler temporaries).
|
2016-03-22 10:08:44 -04:00
|
|
|
fn write_mir_intro(tcx: &TyCtxt, nid: NodeId, mir: &Mir, w: &mut Write)
|
2016-03-19 07:13:54 -04:00
|
|
|
-> io::Result<()> {
|
2016-03-22 22:01:37 -05:00
|
|
|
write!(w, "fn {}(", tcx.map.path_to_string(nid))?;
|
2015-12-29 20:06:19 -06:00
|
|
|
|
|
|
|
// fn argument types.
|
|
|
|
for (i, arg) in mir.arg_decls.iter().enumerate() {
|
|
|
|
if i > 0 {
|
2016-03-22 22:01:37 -05:00
|
|
|
write!(w, ", ")?;
|
2015-12-29 20:06:19 -06:00
|
|
|
}
|
2016-03-22 22:01:37 -05:00
|
|
|
write!(w, "{:?}: {}", Lvalue::Arg(i as u32), arg.ty)?;
|
2015-12-29 20:06:19 -06:00
|
|
|
}
|
|
|
|
|
2016-03-22 22:01:37 -05:00
|
|
|
write!(w, ") -> ")?;
|
2015-12-29 20:06:19 -06:00
|
|
|
|
|
|
|
// fn return type.
|
|
|
|
match mir.return_ty {
|
2016-03-22 22:01:37 -05:00
|
|
|
ty::FnOutput::FnConverging(ty) => write!(w, "{}", ty)?,
|
|
|
|
ty::FnOutput::FnDiverging => write!(w, "!")?,
|
2015-12-29 20:06:19 -06:00
|
|
|
}
|
|
|
|
|
2016-03-22 22:01:37 -05:00
|
|
|
writeln!(w, " {{")?;
|
2015-12-29 20:06:19 -06:00
|
|
|
|
|
|
|
// User variable types (including the user's name in a comment).
|
|
|
|
for (i, var) in mir.var_decls.iter().enumerate() {
|
2016-03-22 22:01:37 -05:00
|
|
|
write!(w, "{}let ", INDENT)?;
|
2015-12-29 20:06:19 -06:00
|
|
|
if var.mutability == Mutability::Mut {
|
2016-03-22 22:01:37 -05:00
|
|
|
write!(w, "mut ")?;
|
2015-12-29 20:06:19 -06:00
|
|
|
}
|
2016-03-22 22:01:37 -05:00
|
|
|
writeln!(w, "{:?}: {}; // {}", Lvalue::Var(i as u32), var.ty, var.name)?;
|
2015-12-29 20:06:19 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
// Compiler-introduced temporary types.
|
|
|
|
for (i, temp) in mir.temp_decls.iter().enumerate() {
|
2016-03-22 22:01:37 -05:00
|
|
|
writeln!(w, "{}let mut {:?}: {};", INDENT, Lvalue::Temp(i as u32), temp.ty)?;
|
2015-12-29 20:06:19 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
Ok(())
|
|
|
|
}
|