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-05-31 20:27:36 +03:00
|
|
|
use build::{Location, ScopeAuxiliaryVec, ScopeId};
|
2016-05-03 00:26:41 +03:00
|
|
|
use rustc::hir;
|
2016-08-08 18:42:06 -04:00
|
|
|
use rustc::hir::def_id::DefId;
|
2015-12-29 20:06:19 -06:00
|
|
|
use rustc::mir::repr::*;
|
2016-08-08 18:42:06 -04:00
|
|
|
use rustc::mir::mir_map::MirMap;
|
2016-05-03 00:26:41 +03:00
|
|
|
use rustc::mir::transform::MirSource;
|
2016-03-22 17:30:57 +02:00
|
|
|
use rustc::ty::{self, TyCtxt};
|
2016-03-22 10:08:44 -04:00
|
|
|
use rustc_data_structures::fnv::FnvHashMap;
|
2016-06-07 17:28:36 +03:00
|
|
|
use rustc_data_structures::indexed_vec::{Idx};
|
2016-03-22 16:05:28 -04:00
|
|
|
use std::fmt::Display;
|
|
|
|
use std::fs;
|
2015-12-29 20:06:19 -06:00
|
|
|
use std::io::{self, Write};
|
2016-07-07 16:40:01 -07:00
|
|
|
use std::path::{PathBuf, Path};
|
2015-12-29 20:06:19 -06:00
|
|
|
|
|
|
|
const INDENT: &'static str = " ";
|
2016-05-13 00:20:59 +02:00
|
|
|
/// Alignment for lining up comments following MIR statements
|
2016-05-13 23:40:06 +02:00
|
|
|
const ALIGN: usize = 40;
|
2015-12-29 20:06:19 -06:00
|
|
|
|
2016-03-22 16:05:28 -04:00
|
|
|
/// If the session is properly configured, dumps a human-readable
|
|
|
|
/// representation of the mir into:
|
|
|
|
///
|
2016-04-13 16:16:14 +05:30
|
|
|
/// ```text
|
2016-03-22 16:05:28 -04:00
|
|
|
/// rustc.node<node_id>.<pass_name>.<disambiguator>
|
|
|
|
/// ```
|
|
|
|
///
|
|
|
|
/// Output from this function is controlled by passing `-Z dump-mir=<filter>`,
|
|
|
|
/// where `<filter>` takes the following forms:
|
|
|
|
///
|
|
|
|
/// - `all` -- dump MIR for all fns, all passes, all everything
|
|
|
|
/// - `substring1&substring2,...` -- `&`-separated list of substrings
|
|
|
|
/// that can appear in the pass-name or the `item_path_str` for the given
|
|
|
|
/// node-id. If any one of the substrings match, the data is dumped out.
|
2016-05-03 05:23:22 +03:00
|
|
|
pub fn dump_mir<'a, 'tcx>(tcx: TyCtxt<'a, 'tcx, 'tcx>,
|
2016-03-22 16:05:28 -04:00
|
|
|
pass_name: &str,
|
|
|
|
disambiguator: &Display,
|
2016-05-03 00:26:41 +03:00
|
|
|
src: MirSource,
|
2016-03-22 16:05:28 -04:00
|
|
|
mir: &Mir<'tcx>,
|
2016-03-23 05:01:30 -04:00
|
|
|
auxiliary: Option<&ScopeAuxiliaryVec>) {
|
2016-03-22 16:05:28 -04:00
|
|
|
let filters = match tcx.sess.opts.debugging_opts.dump_mir {
|
|
|
|
None => return,
|
|
|
|
Some(ref filters) => filters,
|
|
|
|
};
|
2016-05-03 00:26:41 +03:00
|
|
|
let node_id = src.item_id();
|
2016-03-22 16:05:28 -04:00
|
|
|
let node_path = tcx.item_path_str(tcx.map.local_def_id(node_id));
|
|
|
|
let is_matched =
|
|
|
|
filters.split("&")
|
|
|
|
.any(|filter| {
|
|
|
|
filter == "all" ||
|
|
|
|
pass_name.contains(filter) ||
|
|
|
|
node_path.contains(filter)
|
|
|
|
});
|
|
|
|
if !is_matched {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2016-06-16 16:52:20 +03:00
|
|
|
let promotion_id = match src {
|
2016-06-16 17:30:09 +03:00
|
|
|
MirSource::Promoted(_, id) => format!("-{:?}", id),
|
2016-06-16 16:52:20 +03:00
|
|
|
_ => String::new()
|
|
|
|
};
|
|
|
|
|
2016-07-07 16:40:01 -07:00
|
|
|
let mut file_path = PathBuf::new();
|
|
|
|
if let Some(ref file_dir) = tcx.sess.opts.debugging_opts.dump_mir_dir {
|
|
|
|
let p = Path::new(file_dir);
|
|
|
|
file_path.push(p);
|
|
|
|
};
|
2016-06-16 16:52:20 +03:00
|
|
|
let file_name = format!("rustc.node{}{}.{}.{}.mir",
|
|
|
|
node_id, promotion_id, pass_name, disambiguator);
|
2016-07-07 16:40:01 -07:00
|
|
|
file_path.push(&file_name);
|
|
|
|
let _ = fs::File::create(&file_path).and_then(|mut file| {
|
2016-03-22 16:05:28 -04:00
|
|
|
try!(writeln!(file, "// MIR for `{}`", node_path));
|
|
|
|
try!(writeln!(file, "// node_id = {}", node_id));
|
|
|
|
try!(writeln!(file, "// pass_name = {}", pass_name));
|
|
|
|
try!(writeln!(file, "// disambiguator = {}", disambiguator));
|
|
|
|
try!(writeln!(file, ""));
|
2016-05-03 00:26:41 +03:00
|
|
|
try!(write_mir_fn(tcx, src, mir, &mut file, auxiliary));
|
2016-03-22 16:05:28 -04:00
|
|
|
Ok(())
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2015-12-29 20:06:19 -06:00
|
|
|
/// Write out a human-readable textual representation for the given MIR.
|
2016-05-03 05:23:22 +03:00
|
|
|
pub fn write_mir_pretty<'a, 'b, 'tcx, I>(tcx: TyCtxt<'b, 'tcx, 'tcx>,
|
2016-05-03 04:56:42 +03:00
|
|
|
iter: I,
|
2016-08-08 18:42:06 -04:00
|
|
|
mir_map: &MirMap<'tcx>,
|
2016-05-03 04:56:42 +03:00
|
|
|
w: &mut Write)
|
|
|
|
-> io::Result<()>
|
2016-08-08 18:42:06 -04:00
|
|
|
where I: Iterator<Item=DefId>, 'tcx: 'a
|
2016-03-19 07:13:54 -04:00
|
|
|
{
|
2016-05-13 00:20:59 +02:00
|
|
|
let mut first = true;
|
2016-08-08 18:42:06 -04:00
|
|
|
for def_id in iter {
|
|
|
|
let mir = &mir_map.map[&def_id];
|
|
|
|
|
2016-05-13 00:20:59 +02:00
|
|
|
if first {
|
|
|
|
first = false;
|
|
|
|
} else {
|
|
|
|
// Put empty lines between all items
|
|
|
|
writeln!(w, "")?;
|
|
|
|
}
|
|
|
|
|
2016-08-08 18:42:06 -04:00
|
|
|
let id = tcx.map.as_local_node_id(def_id).unwrap();
|
2016-05-03 00:26:41 +03:00
|
|
|
let src = MirSource::from_node(tcx, id);
|
|
|
|
write_mir_fn(tcx, src, mir, w, None)?;
|
|
|
|
|
2016-06-16 17:30:09 +03:00
|
|
|
for (i, mir) in mir.promoted.iter_enumerated() {
|
2016-05-13 00:20:59 +02:00
|
|
|
writeln!(w, "")?;
|
2016-05-03 00:26:41 +03:00
|
|
|
write_mir_fn(tcx, MirSource::Promoted(id, i), 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),
|
|
|
|
}
|
|
|
|
|
2016-04-28 16:06:18 +02:00
|
|
|
fn scope_entry_exit_annotations(auxiliary: Option<&ScopeAuxiliaryVec>)
|
|
|
|
-> FnvHashMap<Location, Vec<Annotation>>
|
|
|
|
{
|
2016-03-22 11:52:34 -04:00
|
|
|
// compute scope/entry exit annotations
|
|
|
|
let mut annotations = FnvHashMap();
|
|
|
|
if let Some(auxiliary) = auxiliary {
|
2016-06-07 17:28:36 +03:00
|
|
|
for (scope_id, auxiliary) in auxiliary.iter_enumerated() {
|
2016-03-22 11:52:34 -04:00
|
|
|
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-04-28 16:06:18 +02:00
|
|
|
return annotations;
|
|
|
|
}
|
2016-03-22 11:52:34 -04:00
|
|
|
|
2016-04-28 16:06:18 +02:00
|
|
|
pub fn write_mir_fn<'a, 'tcx>(tcx: TyCtxt<'a, 'tcx, 'tcx>,
|
|
|
|
src: MirSource,
|
|
|
|
mir: &Mir<'tcx>,
|
|
|
|
w: &mut Write,
|
|
|
|
auxiliary: Option<&ScopeAuxiliaryVec>)
|
|
|
|
-> io::Result<()> {
|
|
|
|
let annotations = scope_entry_exit_annotations(auxiliary);
|
2016-05-03 00:26:41 +03:00
|
|
|
write_mir_intro(tcx, src, mir, w)?;
|
2016-06-07 21:20:50 +03:00
|
|
|
for block in mir.basic_blocks().indices() {
|
2016-03-22 11:52:34 -04:00
|
|
|
write_basic_block(tcx, block, mir, w, &annotations)?;
|
2016-06-07 21:20:50 +03:00
|
|
|
if block.index() + 1 != mir.basic_blocks().len() {
|
2016-05-31 20:27:36 +03:00
|
|
|
writeln!(w, "")?;
|
|
|
|
}
|
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-05-03 04:56:42 +03:00
|
|
|
fn write_basic_block(tcx: TyCtxt,
|
2016-03-22 10:08:44 -04:00
|
|
|
block: BasicBlock,
|
|
|
|
mir: &Mir,
|
|
|
|
w: &mut Write,
|
|
|
|
annotations: &FnvHashMap<Location, Vec<Annotation>>)
|
|
|
|
-> io::Result<()> {
|
2016-06-07 21:20:50 +03:00
|
|
|
let data = &mir[block];
|
2015-12-29 20:06:19 -06:00
|
|
|
|
|
|
|
// Basic block label at the top.
|
2016-05-13 00:20:59 +02:00
|
|
|
writeln!(w, "{}{:?}: {{", 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())?,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-05-13 00:20:59 +02:00
|
|
|
let indented_mir = format!("{0}{0}{1:?};", INDENT, statement);
|
|
|
|
writeln!(w, "{0:1$} // {2}",
|
|
|
|
indented_mir,
|
|
|
|
ALIGN,
|
2016-06-07 19:21:56 +03:00
|
|
|
comment(tcx, statement.source_info))?;
|
2016-03-22 10:08:44 -04:00
|
|
|
|
|
|
|
current_location.statement_index += 1;
|
2015-12-29 20:06:19 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
// Terminator at the bottom.
|
2016-05-13 00:20:59 +02:00
|
|
|
let indented_terminator = format!("{0}{0}{1:?};", INDENT, data.terminator().kind);
|
|
|
|
writeln!(w, "{0:1$} // {2}",
|
|
|
|
indented_terminator,
|
|
|
|
ALIGN,
|
2016-06-07 19:21:56 +03:00
|
|
|
comment(tcx, data.terminator().source_info))?;
|
2015-12-29 20:06:19 -06:00
|
|
|
|
2016-07-14 05:29:50 +09:00
|
|
|
writeln!(w, "{}}}", INDENT)
|
2015-12-29 20:06:19 -06:00
|
|
|
}
|
|
|
|
|
2016-06-07 19:21:56 +03:00
|
|
|
fn comment(tcx: TyCtxt, SourceInfo { span, scope }: SourceInfo) -> String {
|
2016-05-13 00:20:59 +02:00
|
|
|
format!("scope {} at {}", scope.index(), tcx.sess.codemap().span_to_string(span))
|
2016-03-22 10:08:44 -04:00
|
|
|
}
|
|
|
|
|
2016-05-03 04:56:42 +03:00
|
|
|
fn write_scope_tree(tcx: TyCtxt,
|
2016-03-22 10:08:44 -04:00
|
|
|
mir: &Mir,
|
2016-05-31 20:27:36 +03:00
|
|
|
scope_tree: &FnvHashMap<VisibilityScope, Vec<VisibilityScope>>,
|
2016-03-22 10:08:44 -04:00
|
|
|
w: &mut Write,
|
2016-05-31 20:27:36 +03:00
|
|
|
parent: VisibilityScope,
|
|
|
|
depth: usize)
|
2016-03-22 10:08:44 -04:00
|
|
|
-> io::Result<()> {
|
2016-05-31 20:27:36 +03:00
|
|
|
let indent = depth * INDENT.len();
|
2016-05-13 23:36:50 +02:00
|
|
|
|
|
|
|
let children = match scope_tree.get(&parent) {
|
|
|
|
Some(childs) => childs,
|
|
|
|
None => return Ok(()),
|
|
|
|
};
|
|
|
|
|
2016-05-31 20:27:36 +03:00
|
|
|
for &child in children {
|
|
|
|
let data = &mir.visibility_scopes[child];
|
|
|
|
assert_eq!(data.parent_scope, Some(parent));
|
|
|
|
writeln!(w, "{0:1$}scope {2} {{", "", indent, child.index())?;
|
2016-03-22 11:52:34 -04:00
|
|
|
|
2016-05-31 20:27:36 +03:00
|
|
|
// User variable types (including the user's name in a comment).
|
2016-06-07 17:28:36 +03:00
|
|
|
for (id, var) in mir.var_decls.iter_enumerated() {
|
2016-05-31 20:27:36 +03:00
|
|
|
// Skip if not declared in this scope.
|
2016-06-07 19:21:56 +03:00
|
|
|
if var.source_info.scope != child {
|
2016-05-31 20:27:36 +03:00
|
|
|
continue;
|
|
|
|
}
|
2016-03-22 11:52:34 -04:00
|
|
|
|
2016-05-31 20:27:36 +03:00
|
|
|
let mut_str = if var.mutability == Mutability::Mut {
|
|
|
|
"mut "
|
|
|
|
} else {
|
|
|
|
""
|
|
|
|
};
|
|
|
|
|
|
|
|
let indent = indent + INDENT.len();
|
|
|
|
let indented_var = format!("{0:1$}let {2}{3:?}: {4};",
|
|
|
|
INDENT,
|
|
|
|
indent,
|
|
|
|
mut_str,
|
2016-06-07 17:28:36 +03:00
|
|
|
id,
|
2016-05-31 20:27:36 +03:00
|
|
|
var.ty);
|
|
|
|
writeln!(w, "{0:1$} // \"{2}\" in {3}",
|
|
|
|
indented_var,
|
|
|
|
ALIGN,
|
|
|
|
var.name,
|
2016-06-07 19:21:56 +03:00
|
|
|
comment(tcx, var.source_info))?;
|
2016-03-22 11:52:34 -04:00
|
|
|
}
|
|
|
|
|
2016-05-31 20:27:36 +03:00
|
|
|
write_scope_tree(tcx, mir, scope_tree, w, child, depth + 1)?;
|
2016-05-13 23:36:50 +02:00
|
|
|
|
2016-05-31 20:27:36 +03:00
|
|
|
writeln!(w, "{0:1$}}}", "", depth * INDENT.len())?;
|
2016-03-22 10:08:44 -04:00
|
|
|
}
|
2016-05-13 00:20:59 +02:00
|
|
|
|
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-05-03 05:23:22 +03:00
|
|
|
fn write_mir_intro<'a, 'tcx>(tcx: TyCtxt<'a, 'tcx, 'tcx>,
|
|
|
|
src: MirSource,
|
|
|
|
mir: &Mir,
|
|
|
|
w: &mut Write)
|
|
|
|
-> io::Result<()> {
|
2016-04-28 16:13:44 +02:00
|
|
|
write_mir_sig(tcx, src, mir, w)?;
|
|
|
|
writeln!(w, " {{")?;
|
2016-05-31 20:27:36 +03:00
|
|
|
|
|
|
|
// construct a scope tree and write it out
|
|
|
|
let mut scope_tree: FnvHashMap<VisibilityScope, Vec<VisibilityScope>> = FnvHashMap();
|
|
|
|
for (index, scope_data) in mir.visibility_scopes.iter().enumerate() {
|
|
|
|
if let Some(parent) = scope_data.parent_scope {
|
|
|
|
scope_tree.entry(parent)
|
|
|
|
.or_insert(vec![])
|
|
|
|
.push(VisibilityScope::new(index));
|
|
|
|
} else {
|
|
|
|
// Only the argument scope has no parent, because it's the root.
|
|
|
|
assert_eq!(index, ARGUMENT_VISIBILITY_SCOPE.index());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
write_scope_tree(tcx, mir, &scope_tree, w, ARGUMENT_VISIBILITY_SCOPE, 1)?;
|
|
|
|
|
|
|
|
write_mir_decls(mir, w)
|
2016-04-28 16:13:44 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
fn write_mir_sig(tcx: TyCtxt, src: MirSource, mir: &Mir, w: &mut Write)
|
|
|
|
-> io::Result<()>
|
|
|
|
{
|
2016-05-03 00:26:41 +03:00
|
|
|
match src {
|
|
|
|
MirSource::Fn(_) => write!(w, "fn")?,
|
|
|
|
MirSource::Const(_) => write!(w, "const")?,
|
|
|
|
MirSource::Static(_, hir::MutImmutable) => write!(w, "static")?,
|
|
|
|
MirSource::Static(_, hir::MutMutable) => write!(w, "static mut")?,
|
2016-06-16 17:30:09 +03:00
|
|
|
MirSource::Promoted(_, i) => write!(w, "{:?} in", i)?
|
2016-05-03 00:26:41 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
write!(w, " {}", tcx.node_path_str(src.item_id()))?;
|
2015-12-29 20:06:19 -06:00
|
|
|
|
2016-05-03 00:26:41 +03:00
|
|
|
if let MirSource::Fn(_) = src {
|
|
|
|
write!(w, "(")?;
|
|
|
|
|
|
|
|
// fn argument types.
|
2016-06-07 17:28:36 +03:00
|
|
|
for (i, arg) in mir.arg_decls.iter_enumerated() {
|
|
|
|
if i.index() != 0 {
|
2016-05-03 00:26:41 +03:00
|
|
|
write!(w, ", ")?;
|
|
|
|
}
|
2016-06-07 17:28:36 +03:00
|
|
|
write!(w, "{:?}: {}", Lvalue::Arg(i), arg.ty)?;
|
2015-12-29 20:06:19 -06:00
|
|
|
}
|
|
|
|
|
2016-05-03 00:26:41 +03:00
|
|
|
write!(w, ") -> ")?;
|
2015-12-29 20:06:19 -06:00
|
|
|
|
2016-05-03 00:26:41 +03:00
|
|
|
// fn return type.
|
|
|
|
match mir.return_ty {
|
2016-04-28 16:13:44 +02:00
|
|
|
ty::FnOutput::FnConverging(ty) => write!(w, "{}", ty),
|
|
|
|
ty::FnOutput::FnDiverging => write!(w, "!"),
|
2016-05-03 00:26:41 +03:00
|
|
|
}
|
|
|
|
} else {
|
|
|
|
assert!(mir.arg_decls.is_empty());
|
2016-04-28 16:13:44 +02:00
|
|
|
write!(w, ": {} =", mir.return_ty.unwrap())
|
2015-12-29 20:06:19 -06:00
|
|
|
}
|
2016-04-28 16:13:44 +02:00
|
|
|
}
|
2015-12-29 20:06:19 -06:00
|
|
|
|
2016-05-31 20:27:36 +03:00
|
|
|
fn write_mir_decls(mir: &Mir, w: &mut Write) -> io::Result<()> {
|
2015-12-29 20:06:19 -06:00
|
|
|
// Compiler-introduced temporary types.
|
2016-06-07 17:28:36 +03:00
|
|
|
for (id, temp) in mir.temp_decls.iter_enumerated() {
|
|
|
|
writeln!(w, "{}let mut {:?}: {};", INDENT, id, temp.ty)?;
|
2015-12-29 20:06:19 -06:00
|
|
|
}
|
|
|
|
|
2016-05-13 00:20:59 +02:00
|
|
|
// Wrote any declaration? Add an empty line before the first block is printed.
|
|
|
|
if !mir.var_decls.is_empty() || !mir.temp_decls.is_empty() {
|
|
|
|
writeln!(w, "")?;
|
|
|
|
}
|
|
|
|
|
2015-12-29 20:06:19 -06:00
|
|
|
Ok(())
|
|
|
|
}
|