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.
		
		
		
		
		
			
		
			
				
					
					
						
							98 lines
						
					
					
						
							3.7 KiB
						
					
					
				
			
		
		
	
	
							98 lines
						
					
					
						
							3.7 KiB
						
					
					
				// Copyright (c) 2018 Siegfried Pammer | 
						|
//  | 
						|
// 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.Collections.Generic; | 
						|
using System.Reflection.Metadata; | 
						|
 | 
						|
namespace ICSharpCode.Decompiler.Metadata | 
						|
{ | 
						|
	/// <summary> | 
						|
	/// Describes which parts of the (compiler-generated) code belong to which user code. | 
						|
	/// A part could be: | 
						|
	/// - the body (method) of a lambda. | 
						|
	/// - the MoveNext method of async/yield state machines. | 
						|
	/// </summary> | 
						|
	public class CodeMappingInfo | 
						|
	{ | 
						|
		/// <summary> | 
						|
		/// The module containing the code. | 
						|
		/// </summary> | 
						|
		public PEFile Module { get; } | 
						|
 | 
						|
		/// <summary> | 
						|
		/// The (parent) TypeDef containing the code. | 
						|
		/// </summary> | 
						|
		public TypeDefinitionHandle TypeDefinition { get; } | 
						|
 | 
						|
		readonly Dictionary<MethodDefinitionHandle, List<MethodDefinitionHandle>> parts; | 
						|
		readonly Dictionary<MethodDefinitionHandle, MethodDefinitionHandle> parents; | 
						|
 | 
						|
		/// <summary> | 
						|
		/// Creates a <see cref="CodeMappingInfo"/> instance using the given <paramref name="module"/> and <paramref name="type"/>. | 
						|
		/// </summary> | 
						|
		public CodeMappingInfo(PEFile module, TypeDefinitionHandle type) | 
						|
		{ | 
						|
			this.Module = module; | 
						|
			this.TypeDefinition = type; | 
						|
			this.parts = new Dictionary<MethodDefinitionHandle, List<MethodDefinitionHandle>>(); | 
						|
			this.parents = new Dictionary<MethodDefinitionHandle, MethodDefinitionHandle>(); | 
						|
		} | 
						|
 | 
						|
		/// <summary> | 
						|
		/// Returns all parts of a method. | 
						|
		/// A method has at least one part, that is, the method itself. | 
						|
		/// If no parts are found, only the method itself is returned. | 
						|
		/// </summary> | 
						|
		public IEnumerable<MethodDefinitionHandle> GetMethodParts(MethodDefinitionHandle method) | 
						|
		{ | 
						|
			if (parts.TryGetValue(method, out var p)) | 
						|
				return p; | 
						|
			return new[] { method }; | 
						|
		} | 
						|
 | 
						|
		/// <summary> | 
						|
		/// Returns the parent of a part. | 
						|
		/// The parent is usually the "calling method" of lambdas, async and yield state machines. | 
						|
		/// The "calling method" has itself as parent. | 
						|
		/// If no parent is found, the method itself is returned. | 
						|
		/// </summary> | 
						|
		public MethodDefinitionHandle GetParentMethod(MethodDefinitionHandle method) | 
						|
		{ | 
						|
			if (parents.TryGetValue(method, out var p)) | 
						|
				return p; | 
						|
			return method; | 
						|
		} | 
						|
 | 
						|
		/// <summary> | 
						|
		/// Adds a bidirectional mapping between <paramref name="parent"/> and <paramref name="part"/>. | 
						|
		/// </summary> | 
						|
		public void AddMapping(MethodDefinitionHandle parent, MethodDefinitionHandle part) | 
						|
		{ | 
						|
			//Debug.Print("Parent: " + MetadataTokens.GetRowNumber(parent) + " Part: " + MetadataTokens.GetRowNumber(part)); | 
						|
			if (parents.ContainsKey(part)) | 
						|
				return; | 
						|
			parents.Add(part, parent); | 
						|
			if (!parts.TryGetValue(parent, out var list)) | 
						|
			{ | 
						|
				list = new List<MethodDefinitionHandle>(); | 
						|
				parts.Add(parent, list); | 
						|
			} | 
						|
			list.Add(part); | 
						|
		} | 
						|
	} | 
						|
}
 | 
						|
 |