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.
		
		
		
		
		
			
		
			
				
					
					
						
							378 lines
						
					
					
						
							15 KiB
						
					
					
				
			
		
		
	
	
							378 lines
						
					
					
						
							15 KiB
						
					
					
				#nullable enable | 
						|
// Copyright (c) 2014 Daniel Grunwald | 
						|
//  | 
						|
// 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.Reflection; | 
						|
using System.Reflection.Metadata; | 
						|
using System.Reflection.Metadata.Ecma335; | 
						|
 | 
						|
using ICSharpCode.Decompiler.Disassembler; | 
						|
using ICSharpCode.Decompiler.Metadata; | 
						|
using ICSharpCode.Decompiler.TypeSystem; | 
						|
using ICSharpCode.Decompiler.Util; | 
						|
 | 
						|
namespace ICSharpCode.Decompiler.IL | 
						|
{ | 
						|
	public static partial class InstructionOutputExtensions | 
						|
	{ | 
						|
		public static void Write(this ITextOutput output, OpCode opCode) | 
						|
		{ | 
						|
			output.Write(originalOpCodeNames[(int)opCode]); | 
						|
		} | 
						|
 | 
						|
		public static void Write(this ITextOutput output, StackType stackType) | 
						|
		{ | 
						|
			output.Write(stackType.ToString().ToLowerInvariant()); | 
						|
		} | 
						|
 | 
						|
		public static void Write(this ITextOutput output, PrimitiveType primitiveType) | 
						|
		{ | 
						|
			output.Write(primitiveType.ToString().ToLowerInvariant()); | 
						|
		} | 
						|
 | 
						|
		public static void WriteTo(this IType type, ITextOutput output) | 
						|
		{ | 
						|
			output.WriteReference(type, type.ReflectionName); | 
						|
		} | 
						|
 | 
						|
		public static void WriteTo(this IMember member, ITextOutput output) | 
						|
		{ | 
						|
			if (member is IMethod method && method.IsConstructor) | 
						|
				output.WriteReference(member, method.DeclaringType?.Name + "." + method.Name); | 
						|
			else | 
						|
				output.WriteReference(member, member.Name); | 
						|
		} | 
						|
 | 
						|
		public static void WriteTo(this Interval interval, ITextOutput output, ILAstWritingOptions options) | 
						|
		{ | 
						|
			if (!options.ShowILRanges) | 
						|
				return; | 
						|
			if (interval.IsEmpty) | 
						|
				output.Write("[empty] "); | 
						|
			else | 
						|
				output.Write($"[{interval.Start:x4}..{interval.InclusiveEnd:x4}] "); | 
						|
		} | 
						|
 | 
						|
		public static void WriteTo(this EntityHandle entity, PEFile module, ITextOutput output, Metadata.MetadataGenericContext genericContext, ILNameSyntax syntax = ILNameSyntax.Signature) | 
						|
		{ | 
						|
			if (entity.IsNil) | 
						|
			{ | 
						|
				output.Write("<nil>"); | 
						|
				return; | 
						|
			} | 
						|
			if (module == null) | 
						|
				throw new ArgumentNullException(nameof(module)); | 
						|
			var metadata = module.Metadata; | 
						|
			Action<ILNameSyntax> signature; | 
						|
			MethodSignature<Action<ILNameSyntax>> methodSignature; | 
						|
			string memberName; | 
						|
			switch (entity.Kind) | 
						|
			{ | 
						|
				case HandleKind.TypeDefinition: | 
						|
				{ | 
						|
					var td = metadata.GetTypeDefinition((TypeDefinitionHandle)entity); | 
						|
					output.WriteReference(module, entity, td.GetFullTypeName(metadata).ToILNameString()); | 
						|
					break; | 
						|
				} | 
						|
				case HandleKind.TypeReference: | 
						|
				{ | 
						|
					var tr = metadata.GetTypeReference((TypeReferenceHandle)entity); | 
						|
					EntityHandle resolutionScope; | 
						|
					try | 
						|
					{ | 
						|
						resolutionScope = tr.ResolutionScope; | 
						|
					} | 
						|
					catch (BadImageFormatException) | 
						|
					{ | 
						|
						resolutionScope = default; | 
						|
					} | 
						|
					if (!resolutionScope.IsNil) | 
						|
					{ | 
						|
						output.Write("["); | 
						|
						var currentTypeRef = tr; | 
						|
						while (currentTypeRef.ResolutionScope.Kind == HandleKind.TypeReference) | 
						|
						{ | 
						|
							currentTypeRef = metadata.GetTypeReference((TypeReferenceHandle)currentTypeRef.ResolutionScope); | 
						|
						} | 
						|
						switch (currentTypeRef.ResolutionScope.Kind) | 
						|
						{ | 
						|
							case HandleKind.ModuleDefinition: | 
						|
								var modDef = metadata.GetModuleDefinition(); | 
						|
								output.Write(DisassemblerHelpers.Escape(metadata.GetString(modDef.Name))); | 
						|
								break; | 
						|
							case HandleKind.ModuleReference: | 
						|
								break; | 
						|
							case HandleKind.AssemblyReference: | 
						|
								var asmRef = metadata.GetAssemblyReference((AssemblyReferenceHandle)currentTypeRef.ResolutionScope); | 
						|
								output.Write(DisassemblerHelpers.Escape(metadata.GetString(asmRef.Name))); | 
						|
								break; | 
						|
						} | 
						|
						output.Write("]"); | 
						|
					} | 
						|
					output.WriteReference(module, entity, entity.GetFullTypeName(metadata).ToILNameString()); | 
						|
					break; | 
						|
				} | 
						|
				case HandleKind.TypeSpecification: | 
						|
				{ | 
						|
					var ts = metadata.GetTypeSpecification((TypeSpecificationHandle)entity); | 
						|
					signature = ts.DecodeSignature(new DisassemblerSignatureTypeProvider(module, output), genericContext); | 
						|
					signature(syntax); | 
						|
					break; | 
						|
				} | 
						|
				case HandleKind.FieldDefinition: | 
						|
				{ | 
						|
					var fd = metadata.GetFieldDefinition((FieldDefinitionHandle)entity); | 
						|
					signature = fd.DecodeSignature(new DisassemblerSignatureTypeProvider(module, output), new Metadata.MetadataGenericContext(fd.GetDeclaringType(), module)); | 
						|
					signature(ILNameSyntax.SignatureNoNamedTypeParameters); | 
						|
					output.Write(' '); | 
						|
					((EntityHandle)fd.GetDeclaringType()).WriteTo(module, output, default, ILNameSyntax.TypeName); | 
						|
					output.Write("::"); | 
						|
					output.WriteReference(module, entity, DisassemblerHelpers.Escape(metadata.GetString(fd.Name))); | 
						|
					break; | 
						|
				} | 
						|
				case HandleKind.MethodDefinition: | 
						|
				{ | 
						|
					var md = metadata.GetMethodDefinition((MethodDefinitionHandle)entity); | 
						|
					methodSignature = md.DecodeSignature(new DisassemblerSignatureTypeProvider(module, output), new Metadata.MetadataGenericContext((MethodDefinitionHandle)entity, module)); | 
						|
					methodSignature.Header.WriteTo(output); | 
						|
					methodSignature.ReturnType(ILNameSyntax.SignatureNoNamedTypeParameters); | 
						|
					output.Write(' '); | 
						|
					var declaringType = md.GetDeclaringType(); | 
						|
					if (!declaringType.IsNil) | 
						|
					{ | 
						|
						((EntityHandle)declaringType).WriteTo(module, output, genericContext, ILNameSyntax.TypeName); | 
						|
						output.Write("::"); | 
						|
					} | 
						|
					bool isCompilerControlled = (md.Attributes & MethodAttributes.MemberAccessMask) == MethodAttributes.PrivateScope; | 
						|
					if (isCompilerControlled) | 
						|
					{ | 
						|
						output.WriteReference(module, entity, DisassemblerHelpers.Escape(metadata.GetString(md.Name) + "$PST" + MetadataTokens.GetToken(entity).ToString("X8"))); | 
						|
					} | 
						|
					else | 
						|
					{ | 
						|
						output.WriteReference(module, entity, DisassemblerHelpers.Escape(metadata.GetString(md.Name))); | 
						|
					} | 
						|
					var genericParameters = md.GetGenericParameters(); | 
						|
					if (genericParameters.Count > 0) | 
						|
					{ | 
						|
						output.Write('<'); | 
						|
						for (int i = 0; i < genericParameters.Count; i++) | 
						|
						{ | 
						|
							if (i > 0) | 
						|
								output.Write(", "); | 
						|
							var gp = metadata.GetGenericParameter(genericParameters[i]); | 
						|
							if ((gp.Attributes & GenericParameterAttributes.ReferenceTypeConstraint) == GenericParameterAttributes.ReferenceTypeConstraint) | 
						|
							{ | 
						|
								output.Write("class "); | 
						|
							} | 
						|
							else if ((gp.Attributes & GenericParameterAttributes.NotNullableValueTypeConstraint) == GenericParameterAttributes.NotNullableValueTypeConstraint) | 
						|
							{ | 
						|
								output.Write("valuetype "); | 
						|
							} | 
						|
							if ((gp.Attributes & GenericParameterAttributes.DefaultConstructorConstraint) == GenericParameterAttributes.DefaultConstructorConstraint) | 
						|
							{ | 
						|
								output.Write(".ctor "); | 
						|
							} | 
						|
							var constraints = gp.GetConstraints(); | 
						|
							if (constraints.Count > 0) | 
						|
							{ | 
						|
								output.Write('('); | 
						|
								for (int j = 0; j < constraints.Count; j++) | 
						|
								{ | 
						|
									if (j > 0) | 
						|
										output.Write(", "); | 
						|
									var constraint = metadata.GetGenericParameterConstraint(constraints[j]); | 
						|
									constraint.Type.WriteTo(module, output, new Metadata.MetadataGenericContext((MethodDefinitionHandle)entity, module), ILNameSyntax.TypeName); | 
						|
								} | 
						|
								output.Write(") "); | 
						|
							} | 
						|
							if ((gp.Attributes & GenericParameterAttributes.Contravariant) == GenericParameterAttributes.Contravariant) | 
						|
							{ | 
						|
								output.Write('-'); | 
						|
							} | 
						|
							else if ((gp.Attributes & GenericParameterAttributes.Covariant) == GenericParameterAttributes.Covariant) | 
						|
							{ | 
						|
								output.Write('+'); | 
						|
							} | 
						|
							output.Write(DisassemblerHelpers.Escape(metadata.GetString(gp.Name))); | 
						|
						} | 
						|
						output.Write('>'); | 
						|
					} | 
						|
					WriteParameterList(output, methodSignature); | 
						|
					break; | 
						|
				} | 
						|
				case HandleKind.MemberReference: | 
						|
					var mr = metadata.GetMemberReference((MemberReferenceHandle)entity); | 
						|
					memberName = metadata.GetString(mr.Name); | 
						|
					switch (mr.GetKind()) | 
						|
					{ | 
						|
						case MemberReferenceKind.Method: | 
						|
							methodSignature = mr.DecodeMethodSignature(new DisassemblerSignatureTypeProvider(module, output), genericContext); | 
						|
							methodSignature.Header.WriteTo(output); | 
						|
							methodSignature.ReturnType(ILNameSyntax.SignatureNoNamedTypeParameters); | 
						|
							output.Write(' '); | 
						|
							WriteParent(output, module, metadata, mr.Parent, genericContext, syntax); | 
						|
							output.Write("::"); | 
						|
							output.WriteReference(module, entity, DisassemblerHelpers.Escape(memberName)); | 
						|
							WriteParameterList(output, methodSignature); | 
						|
							break; | 
						|
						case MemberReferenceKind.Field: | 
						|
							var fieldSignature = mr.DecodeFieldSignature(new DisassemblerSignatureTypeProvider(module, output), genericContext); | 
						|
							fieldSignature(ILNameSyntax.SignatureNoNamedTypeParameters); | 
						|
							output.Write(' '); | 
						|
							WriteParent(output, module, metadata, mr.Parent, genericContext, syntax); | 
						|
							output.Write("::"); | 
						|
							output.WriteReference(module, entity, DisassemblerHelpers.Escape(memberName)); | 
						|
							break; | 
						|
					} | 
						|
					break; | 
						|
				case HandleKind.MethodSpecification: | 
						|
					var ms = metadata.GetMethodSpecification((MethodSpecificationHandle)entity); | 
						|
					var substitution = ms.DecodeSignature(new DisassemblerSignatureTypeProvider(module, output), genericContext); | 
						|
					switch (ms.Method.Kind) | 
						|
					{ | 
						|
						case HandleKind.MethodDefinition: | 
						|
							var methodDefinition = metadata.GetMethodDefinition((MethodDefinitionHandle)ms.Method); | 
						|
							var methodName = metadata.GetString(methodDefinition.Name); | 
						|
							methodSignature = methodDefinition.DecodeSignature(new DisassemblerSignatureTypeProvider(module, output), genericContext); | 
						|
							methodSignature.Header.WriteTo(output); | 
						|
							methodSignature.ReturnType(ILNameSyntax.SignatureNoNamedTypeParameters); | 
						|
							output.Write(' '); | 
						|
							var declaringType = methodDefinition.GetDeclaringType(); | 
						|
							if (!declaringType.IsNil) | 
						|
							{ | 
						|
								((EntityHandle)declaringType).WriteTo(module, output, genericContext, ILNameSyntax.TypeName); | 
						|
								output.Write("::"); | 
						|
							} | 
						|
							bool isCompilerControlled = (methodDefinition.Attributes & MethodAttributes.MemberAccessMask) == MethodAttributes.PrivateScope; | 
						|
							if (isCompilerControlled) | 
						|
							{ | 
						|
								output.Write(DisassemblerHelpers.Escape(methodName + "$PST" + MetadataTokens.GetToken(ms.Method).ToString("X8"))); | 
						|
							} | 
						|
							else | 
						|
							{ | 
						|
								output.Write(DisassemblerHelpers.Escape(methodName)); | 
						|
							} | 
						|
							WriteTypeParameterList(output, syntax, substitution); | 
						|
							WriteParameterList(output, methodSignature); | 
						|
							break; | 
						|
						case HandleKind.MemberReference: | 
						|
							var memberReference = metadata.GetMemberReference((MemberReferenceHandle)ms.Method); | 
						|
							memberName = metadata.GetString(memberReference.Name); | 
						|
							methodSignature = memberReference.DecodeMethodSignature(new DisassemblerSignatureTypeProvider(module, output), genericContext); | 
						|
							methodSignature.Header.WriteTo(output); | 
						|
							methodSignature.ReturnType(ILNameSyntax.SignatureNoNamedTypeParameters); | 
						|
							output.Write(' '); | 
						|
							WriteParent(output, module, metadata, memberReference.Parent, genericContext, syntax); | 
						|
							output.Write("::"); | 
						|
							output.Write(DisassemblerHelpers.Escape(memberName)); | 
						|
							WriteTypeParameterList(output, syntax, substitution); | 
						|
							WriteParameterList(output, methodSignature); | 
						|
							break; | 
						|
					} | 
						|
					break; | 
						|
				case HandleKind.StandaloneSignature: | 
						|
					var standaloneSig = metadata.GetStandaloneSignature((StandaloneSignatureHandle)entity); | 
						|
					var header = metadata.GetBlobReader(standaloneSig.Signature).ReadSignatureHeader(); | 
						|
					switch (header.Kind) | 
						|
					{ | 
						|
						case SignatureKind.Method: | 
						|
							methodSignature = standaloneSig.DecodeMethodSignature(new DisassemblerSignatureTypeProvider(module, output), genericContext); | 
						|
							methodSignature.Header.WriteTo(output); | 
						|
							methodSignature.ReturnType(ILNameSyntax.SignatureNoNamedTypeParameters); | 
						|
							WriteParameterList(output, methodSignature); | 
						|
							break; | 
						|
						default: | 
						|
							output.Write($"@{MetadataTokens.GetToken(entity):X8} /* signature {header.Kind} */"); | 
						|
							break; | 
						|
					} | 
						|
					break; | 
						|
				default: | 
						|
					output.Write($"@{MetadataTokens.GetToken(entity):X8}"); | 
						|
					break; | 
						|
			} | 
						|
		} | 
						|
 | 
						|
		static void WriteTypeParameterList(ITextOutput output, ILNameSyntax syntax, System.Collections.Immutable.ImmutableArray<Action<ILNameSyntax>> substitution) | 
						|
		{ | 
						|
			output.Write('<'); | 
						|
			for (int i = 0; i < substitution.Length; i++) | 
						|
			{ | 
						|
				if (i > 0) | 
						|
					output.Write(", "); | 
						|
				substitution[i](syntax); | 
						|
			} | 
						|
			output.Write('>'); | 
						|
		} | 
						|
 | 
						|
		internal static void WriteParameterList(ITextOutput output, MethodSignature<Action<ILNameSyntax>> methodSignature) | 
						|
		{ | 
						|
			output.Write("("); | 
						|
			for (int i = 0; i < methodSignature.ParameterTypes.Length; ++i) | 
						|
			{ | 
						|
				if (i > 0) | 
						|
					output.Write(", "); | 
						|
				if (i == methodSignature.RequiredParameterCount) | 
						|
					output.Write("..., "); | 
						|
				methodSignature.ParameterTypes[i](ILNameSyntax.SignatureNoNamedTypeParameters); | 
						|
			} | 
						|
			output.Write(")"); | 
						|
		} | 
						|
 | 
						|
		internal static void WriteTo(this in SignatureHeader header, ITextOutput output) | 
						|
		{ | 
						|
			if (header.HasExplicitThis) | 
						|
			{ | 
						|
				output.Write("instance explicit "); | 
						|
			} | 
						|
			else if (header.IsInstance) | 
						|
			{ | 
						|
				output.Write("instance "); | 
						|
			} | 
						|
			if (header.CallingConvention != SignatureCallingConvention.Default) | 
						|
			{ | 
						|
				output.Write(header.CallingConvention.ToILSyntax()); | 
						|
				output.Write(' '); | 
						|
			} | 
						|
		} | 
						|
 | 
						|
		static void WriteParent(ITextOutput output, PEFile module, MetadataReader metadata, EntityHandle parentHandle, Metadata.MetadataGenericContext genericContext, ILNameSyntax syntax) | 
						|
		{ | 
						|
			switch (parentHandle.Kind) | 
						|
			{ | 
						|
				case HandleKind.MethodDefinition: | 
						|
					var methodDef = metadata.GetMethodDefinition((MethodDefinitionHandle)parentHandle); | 
						|
					((EntityHandle)methodDef.GetDeclaringType()).WriteTo(module, output, genericContext, syntax); | 
						|
					break; | 
						|
				case HandleKind.ModuleReference: | 
						|
					output.Write('['); | 
						|
					var moduleRef = metadata.GetModuleReference((ModuleReferenceHandle)parentHandle); | 
						|
					output.Write(metadata.GetString(moduleRef.Name)); | 
						|
					output.Write(']'); | 
						|
					break; | 
						|
				case HandleKind.TypeDefinition: | 
						|
				case HandleKind.TypeReference: | 
						|
				case HandleKind.TypeSpecification: | 
						|
					parentHandle.WriteTo(module, output, genericContext, syntax); | 
						|
					break; | 
						|
			} | 
						|
		} | 
						|
 | 
						|
	} | 
						|
}
 | 
						|
 |