// 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 or the MIT license // , at your // option. This file may not be copied, modified, or distributed // except according to those terms. use dot; use rustc::mir::repr::*; use rustc::middle::ty; use std::fmt::Debug; use std::io::{self, Write}; /// Write a graphviz DOT graph for the given MIR. pub fn write_mir_graphviz(mir: &Mir, w: &mut W) -> io::Result<()> { try!(writeln!(w, "digraph Mir {{")); // Global graph properties try!(writeln!(w, r#" graph [fontname="monospace"];"#)); try!(writeln!(w, r#" node [fontname="monospace"];"#)); try!(writeln!(w, r#" edge [fontname="monospace"];"#)); // Graph label try!(write_graph_label(mir, w)); // Nodes for block in mir.all_basic_blocks() { try!(write_node(block, mir, w)); } // Edges for source in mir.all_basic_blocks() { try!(write_edges(source, mir, w)); } writeln!(w, "}}") } /// Write a graphviz DOT node for the given basic block. fn write_node(block: BasicBlock, mir: &Mir, w: &mut W) -> io::Result<()> { let data = mir.basic_block_data(block); // Start a new node with the label to follow, in one of DOT's pseudo-HTML tables. try!(write!(w, r#" {} [shape="none", label=<"#, node(block))); try!(write!(w, r#""#)); // Basic block number at the top. try!(write!(w, r#""#, block.index())); // List of statements in the middle. if !data.statements.is_empty() { try!(write!(w, r#"")); } // Terminator head at the bottom, not including the list of successor blocks. Those will be // displayed as labels on the edges between blocks. let mut terminator_head = String::new(); data.terminator.fmt_head(&mut terminator_head).unwrap(); try!(write!(w, r#""#, dot::escape_html(&terminator_head))); // Close the table, node label, and the node itself. writeln!(w, "
{}
"#)); for statement in &data.statements { try!(write!(w, "{}
", escape(statement))); } try!(write!(w, "
{}
>];") } /// Write graphviz DOT edges with labels between the given basic block and all of its successors. fn write_edges(source: BasicBlock, mir: &Mir, w: &mut W) -> io::Result<()> { let terminator = &mir.basic_block_data(source).terminator; let labels = terminator.fmt_successor_labels(); for (&target, label) in terminator.successors().iter().zip(labels) { try!(writeln!(w, r#" {} -> {} [label="{}"];"#, node(source), node(target), label)); } Ok(()) } /// Write the graphviz DOT label for the overall graph. This is essentially a block of text that /// will appear below the graph, showing the type of the `fn` this MIR represents and the types of /// all the variables and temporaries. fn write_graph_label(mir: &Mir, w: &mut W) -> io::Result<()> { try!(write!(w, " label= 0 { try!(write!(w, ", ")); } try!(write!(w, "a{}: {}", i, escape(&arg.ty))); } try!(write!(w, ") -> ")); // fn return type. match mir.return_ty { ty::FnOutput::FnConverging(ty) => try!(write!(w, "{}", escape(ty))), ty::FnOutput::FnDiverging => try!(write!(w, "!")), } try!(write!(w, r#"
"#)); // User variable types (including the user's name in a comment). for (i, var) in mir.var_decls.iter().enumerate() { try!(write!(w, "let ")); if var.mutability == Mutability::Mut { try!(write!(w, "mut ")); } try!(write!(w, r#"v{}: {}; // {}
"#, i, escape(&var.ty), var.name)); } // Compiler-introduced temporary types. for (i, temp) in mir.temp_decls.iter().enumerate() { try!(write!(w, r#"let t{}: {};
"#, i, escape(&temp.ty))); } writeln!(w, ">;") } fn node(block: BasicBlock) -> String { format!("bb{}", block.index()) } fn escape(t: &T) -> String { dot::escape_html(&format!("{:?}", t)) }