mirror of https://github.com/icsharpcode/ILSpy.git
				
				
			
			You can not select more than 25 topics
			Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
		
		
		
		
		
			
		
			
				
					
					
						
							162 lines
						
					
					
						
							5.0 KiB
						
					
					
				
			
		
		
	
	
							162 lines
						
					
					
						
							5.0 KiB
						
					
					
				// Copyright (c) 2011 AlphaSierraPapa for the SharpDevelop Team | 
						|
//  | 
						|
// Permission is hereby granted, free of charge, to any person obtaining a copy of this | 
						|
// software and associated documentation files (the "Software"), to deal in the Software | 
						|
// without restriction, including without limitation the rights to use, copy, modify, merge, | 
						|
// publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons | 
						|
// to whom the Software is furnished to do so, subject to the following conditions: | 
						|
//  | 
						|
// The above copyright notice and this permission notice shall be included in all copies or | 
						|
// substantial portions of the Software. | 
						|
//  | 
						|
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, | 
						|
// INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR | 
						|
// PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE | 
						|
// FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR | 
						|
// OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER | 
						|
// DEALINGS IN THE SOFTWARE. | 
						|
 | 
						|
using System; | 
						|
using System.Collections.Generic; | 
						|
using System.Diagnostics; | 
						|
 | 
						|
using ICSharpCode.Decompiler.Util; | 
						|
 | 
						|
namespace ICSharpCode.Decompiler.FlowAnalysis | 
						|
{ | 
						|
	/// <summary> | 
						|
	/// Represents a block in the control flow graph. | 
						|
	/// </summary> | 
						|
	[DebuggerDisplay("CFG UserIndex={UserIndex}, UserData={UserData}")] | 
						|
	public class ControlFlowNode | 
						|
	{ | 
						|
		/// <summary> | 
						|
		/// User index, can be used to look up additional information in an array. | 
						|
		/// </summary> | 
						|
		public int UserIndex; | 
						|
 | 
						|
		/// <summary> | 
						|
		/// User data. | 
						|
		/// </summary> | 
						|
		public object UserData; | 
						|
 | 
						|
		/// <summary> | 
						|
		/// Visited flag, used in various algorithms. | 
						|
		/// </summary> | 
						|
		public bool Visited; | 
						|
 | 
						|
		/// <summary> | 
						|
		/// Gets the node index in a post-order traversal of the control flow graph, starting at the | 
						|
		/// entry point. This field gets computed by dominance analysis. | 
						|
		/// </summary> | 
						|
		public int PostOrderNumber; | 
						|
 | 
						|
		/// <summary> | 
						|
		/// Gets whether this node is reachable. Requires that dominance is computed! | 
						|
		/// </summary> | 
						|
		public bool IsReachable { | 
						|
			get { return DominatorTreeChildren != null; } | 
						|
		} | 
						|
 | 
						|
		/// <summary> | 
						|
		/// Gets the immediate dominator (the parent in the dominator tree). | 
						|
		/// Null if dominance has not been calculated; or if the node is unreachable. | 
						|
		/// </summary> | 
						|
		public ControlFlowNode ImmediateDominator { get; internal set; } | 
						|
 | 
						|
		/// <summary> | 
						|
		/// List of children in the dominator tree. | 
						|
		/// Null if dominance has not been calculated; or if the node is unreachable. | 
						|
		/// </summary> | 
						|
		public List<ControlFlowNode> DominatorTreeChildren { get; internal set; } | 
						|
 | 
						|
		/// <summary> | 
						|
		/// List of incoming control flow edges. | 
						|
		/// </summary> | 
						|
		public readonly List<ControlFlowNode> Predecessors = new List<ControlFlowNode>(); | 
						|
 | 
						|
		/// <summary> | 
						|
		/// List of outgoing control flow edges. | 
						|
		/// </summary> | 
						|
		public readonly List<ControlFlowNode> Successors = new List<ControlFlowNode>(); | 
						|
 | 
						|
		public void AddEdgeTo(ControlFlowNode target) | 
						|
		{ | 
						|
			this.Successors.Add(target); | 
						|
			target.Predecessors.Add(this); | 
						|
		} | 
						|
 | 
						|
		public void TraversePreOrder(Func<ControlFlowNode, IEnumerable<ControlFlowNode>> children, Action<ControlFlowNode> visitAction) | 
						|
		{ | 
						|
			if (Visited) | 
						|
				return; | 
						|
			Visited = true; | 
						|
			visitAction(this); | 
						|
			foreach (ControlFlowNode t in children(this)) | 
						|
				t.TraversePreOrder(children, visitAction); | 
						|
		} | 
						|
 | 
						|
		public void TraversePostOrder(Func<ControlFlowNode, IEnumerable<ControlFlowNode>> children, Action<ControlFlowNode> visitAction) | 
						|
		{ | 
						|
			if (Visited) | 
						|
				return; | 
						|
			Visited = true; | 
						|
			foreach (ControlFlowNode t in children(this)) | 
						|
				t.TraversePostOrder(children, visitAction); | 
						|
			visitAction(this); | 
						|
		} | 
						|
 | 
						|
		/// <summary> | 
						|
		/// Gets whether <c>this</c> dominates <paramref name="node"/>. | 
						|
		/// </summary> | 
						|
		public bool Dominates(ControlFlowNode node) | 
						|
		{ | 
						|
			// TODO: this can be made O(1) by numbering the dominator tree | 
						|
			ControlFlowNode tmp = node; | 
						|
			while (tmp != null) | 
						|
			{ | 
						|
				if (tmp == this) | 
						|
					return true; | 
						|
				tmp = tmp.ImmediateDominator; | 
						|
			} | 
						|
			return false; | 
						|
		} | 
						|
 | 
						|
#if DEBUG | 
						|
		internal static GraphVizGraph ExportGraph(IReadOnlyList<ControlFlowNode> nodes, Func<ControlFlowNode, string> labelFunc = null) | 
						|
		{ | 
						|
			if (labelFunc == null) | 
						|
			{ | 
						|
				labelFunc = node => { | 
						|
					var block = node.UserData as IL.Block; | 
						|
					return block != null ? block.Label : node.UserData?.ToString(); | 
						|
				}; | 
						|
			} | 
						|
			GraphVizGraph g = new GraphVizGraph(); | 
						|
			GraphVizNode[] n = new GraphVizNode[nodes.Count]; | 
						|
			for (int i = 0; i < n.Length; i++) | 
						|
			{ | 
						|
				n[i] = new GraphVizNode(nodes[i].UserIndex); | 
						|
				n[i].shape = "box"; | 
						|
				n[i].label = labelFunc(nodes[i]); | 
						|
				g.AddNode(n[i]); | 
						|
			} | 
						|
			foreach (var source in nodes) | 
						|
			{ | 
						|
				foreach (var target in source.Successors) | 
						|
				{ | 
						|
					g.AddEdge(new GraphVizEdge(source.UserIndex, target.UserIndex)); | 
						|
				} | 
						|
				if (source.ImmediateDominator != null) | 
						|
				{ | 
						|
					g.AddEdge( | 
						|
						new GraphVizEdge(source.ImmediateDominator.UserIndex, source.UserIndex) { | 
						|
							color = "green" | 
						|
						}); | 
						|
				} | 
						|
			} | 
						|
			return g; | 
						|
		} | 
						|
#endif | 
						|
	} | 
						|
}
 | 
						|
 |