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.
534 lines
18 KiB
534 lines
18 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 Mono.Cecil; |
|
using Mono.Cecil.Cil; |
|
using System.Text; |
|
|
|
namespace ICSharpCode.Decompiler.Disassembler |
|
{ |
|
public enum ILNameSyntax |
|
{ |
|
/// <summary> |
|
/// class/valuetype + TypeName (built-in types use keyword syntax) |
|
/// </summary> |
|
Signature, |
|
/// <summary> |
|
/// Like signature, but always refers to type parameters using their position |
|
/// </summary> |
|
SignatureNoNamedTypeParameters, |
|
/// <summary> |
|
/// [assembly]Full.Type.Name (even for built-in types) |
|
/// </summary> |
|
TypeName, |
|
/// <summary> |
|
/// Name (but built-in types use keyword syntax) |
|
/// </summary> |
|
ShortTypeName |
|
} |
|
|
|
public static class DisassemblerHelpers |
|
{ |
|
public static string OffsetToString(int offset) |
|
{ |
|
return string.Format("IL_{0:x4}", offset); |
|
} |
|
|
|
public static string OffsetToString(long offset) |
|
{ |
|
return string.Format("IL_{0:x4}", offset); |
|
} |
|
|
|
public static void WriteOffsetReference(ITextOutput writer, Instruction instruction) |
|
{ |
|
if (instruction == null) |
|
writer.Write("null"); |
|
else |
|
writer.WriteReference(OffsetToString(instruction.Offset), instruction); |
|
} |
|
|
|
public static void WriteTo(this ExceptionHandler exceptionHandler, ITextOutput writer) |
|
{ |
|
writer.Write(".try "); |
|
WriteOffsetReference(writer, exceptionHandler.TryStart); |
|
writer.Write('-'); |
|
WriteOffsetReference(writer, exceptionHandler.TryEnd); |
|
writer.Write(' '); |
|
writer.Write(exceptionHandler.HandlerType.ToString().ToLowerInvariant()); |
|
if (exceptionHandler.FilterStart != null) { |
|
writer.Write(' '); |
|
WriteOffsetReference(writer, exceptionHandler.FilterStart); |
|
writer.Write(" handler "); |
|
} |
|
if (exceptionHandler.CatchType != null) { |
|
writer.Write(' '); |
|
exceptionHandler.CatchType.WriteTo(writer); |
|
} |
|
writer.Write(' '); |
|
WriteOffsetReference(writer, exceptionHandler.HandlerStart); |
|
writer.Write('-'); |
|
WriteOffsetReference(writer, exceptionHandler.HandlerEnd); |
|
} |
|
|
|
public static void WriteTo(this Instruction instruction, ITextOutput writer) |
|
{ |
|
writer.WriteDefinition(OffsetToString(instruction.Offset), instruction); |
|
writer.Write(": "); |
|
writer.WriteReference(instruction.OpCode.Name, instruction.OpCode); |
|
if (instruction.Operand != null) { |
|
writer.Write(' '); |
|
if (instruction.OpCode == OpCodes.Ldtoken) { |
|
if (instruction.Operand is MethodReference) |
|
writer.Write("method "); |
|
else if (instruction.Operand is FieldReference) |
|
writer.Write("field "); |
|
} |
|
WriteOperand(writer, instruction.Operand); |
|
} |
|
} |
|
|
|
static void WriteLabelList(ITextOutput writer, Instruction[] instructions) |
|
{ |
|
writer.Write("("); |
|
for (int i = 0; i < instructions.Length; i++) { |
|
if (i != 0) writer.Write(", "); |
|
WriteOffsetReference(writer, instructions[i]); |
|
} |
|
writer.Write(")"); |
|
} |
|
|
|
static string ToInvariantCultureString(object value) |
|
{ |
|
IConvertible convertible = value as IConvertible; |
|
return (null != convertible) |
|
? convertible.ToString(System.Globalization.CultureInfo.InvariantCulture) |
|
: value.ToString(); |
|
} |
|
|
|
public static void WriteTo(this MethodReference method, ITextOutput writer) |
|
{ |
|
if (method.ExplicitThis) { |
|
writer.Write("instance explicit "); |
|
} else if (method.HasThis) { |
|
writer.Write("instance "); |
|
} |
|
if (method.CallingConvention == MethodCallingConvention.VarArg) { |
|
writer.Write("vararg "); |
|
} |
|
method.ReturnType.WriteTo(writer, ILNameSyntax.SignatureNoNamedTypeParameters); |
|
writer.Write(' '); |
|
if (method.DeclaringType != null) { |
|
method.DeclaringType.WriteTo(writer, ILNameSyntax.TypeName); |
|
writer.Write("::"); |
|
} |
|
MethodDefinition md = method as MethodDefinition; |
|
if (md != null && md.IsCompilerControlled) { |
|
writer.WriteReference(Escape(method.Name + "$PST" + method.MetadataToken.ToInt32().ToString("X8")), method); |
|
} else { |
|
writer.WriteReference(Escape(method.Name), method); |
|
} |
|
GenericInstanceMethod gim = method as GenericInstanceMethod; |
|
if (gim != null) { |
|
writer.Write('<'); |
|
for (int i = 0; i < gim.GenericArguments.Count; i++) { |
|
if (i > 0) |
|
writer.Write(", "); |
|
gim.GenericArguments[i].WriteTo(writer); |
|
} |
|
writer.Write('>'); |
|
} |
|
writer.Write("("); |
|
var parameters = method.Parameters; |
|
for (int i = 0; i < parameters.Count; ++i) { |
|
if (i > 0) |
|
writer.Write(", "); |
|
parameters[i].ParameterType.WriteTo(writer, ILNameSyntax.SignatureNoNamedTypeParameters); |
|
} |
|
writer.Write(")"); |
|
} |
|
|
|
static void WriteTo(this FieldReference field, ITextOutput writer) |
|
{ |
|
field.FieldType.WriteTo(writer, ILNameSyntax.SignatureNoNamedTypeParameters); |
|
writer.Write(' '); |
|
field.DeclaringType.WriteTo(writer, ILNameSyntax.TypeName); |
|
writer.Write("::"); |
|
writer.WriteReference(Escape(field.Name), field); |
|
} |
|
|
|
static bool IsValidIdentifierCharacter(char c) |
|
{ |
|
return c == '_' || c == '$' || c == '@' || c == '?' || c == '`'; |
|
} |
|
|
|
static bool IsValidIdentifier(string identifier) |
|
{ |
|
if (string.IsNullOrEmpty(identifier)) |
|
return false; |
|
if (!(char.IsLetter(identifier[0]) || IsValidIdentifierCharacter(identifier[0]))) { |
|
// As a special case, .ctor and .cctor are valid despite starting with a dot |
|
return identifier == ".ctor" || identifier == ".cctor"; |
|
} |
|
for (int i = 1; i < identifier.Length; i++) { |
|
if (!(char.IsLetterOrDigit(identifier[i]) || IsValidIdentifierCharacter(identifier[i]) || identifier[i] == '.')) |
|
return false; |
|
} |
|
return true; |
|
} |
|
|
|
static readonly HashSet<string> ilKeywords = BuildKeywordList( |
|
"abstract", "algorithm", "alignment", "ansi", "any", "arglist", |
|
"array", "as", "assembly", "assert", "at", "auto", "autochar", "beforefieldinit", |
|
"blob", "blob_object", "bool", "brnull", "brnull.s", "brzero", "brzero.s", "bstr", |
|
"bytearray", "byvalstr", "callmostderived", "carray", "catch", "cdecl", "cf", |
|
"char", "cil", "class", "clsid", "const", "currency", "custom", "date", "decimal", |
|
"default", "demand", "deny", "endmac", "enum", "error", "explicit", "extends", "extern", |
|
"false", "famandassem", "family", "famorassem", "fastcall", "fault", "field", "filetime", |
|
"filter", "final", "finally", "fixed", "float", "float32", "float64", "forwardref", |
|
"fromunmanaged", "handler", "hidebysig", "hresult", "idispatch", "il", "illegal", |
|
"implements", "implicitcom", "implicitres", "import", "in", "inheritcheck", "init", |
|
"initonly", "instance", "int", "int16", "int32", "int64", "int8", "interface", "internalcall", |
|
"iunknown", "lasterr", "lcid", "linkcheck", "literal", "localloc", "lpstr", "lpstruct", "lptstr", |
|
"lpvoid", "lpwstr", "managed", "marshal", "method", "modopt", "modreq", "native", "nested", |
|
"newslot", "noappdomain", "noinlining", "nomachine", "nomangle", "nometadata", "noncasdemand", |
|
"noncasinheritance", "noncaslinkdemand", "noprocess", "not", "not_in_gc_heap", "notremotable", |
|
"notserialized", "null", "nullref", "object", "objectref", "opt", "optil", "out", |
|
"permitonly", "pinned", "pinvokeimpl", "prefix1", "prefix2", "prefix3", "prefix4", "prefix5", "prefix6", |
|
"prefix7", "prefixref", "prejitdeny", "prejitgrant", "preservesig", "private", "privatescope", "protected", |
|
"public", "record", "refany", "reqmin", "reqopt", "reqrefuse", "reqsecobj", "request", "retval", |
|
"rtspecialname", "runtime", "safearray", "sealed", "sequential", "serializable", "special", "specialname", |
|
"static", "stdcall", "storage", "stored_object", "stream", "streamed_object", "string", "struct", |
|
"synchronized", "syschar", "sysstring", "tbstr", "thiscall", "tls", "to", "true", "typedref", |
|
"unicode", "unmanaged", "unmanagedexp", "unsigned", "unused", "userdefined", "value", "valuetype", |
|
"vararg", "variant", "vector", "virtual", "void", "wchar", "winapi", "with", "wrapper", |
|
|
|
// These are not listed as keywords in spec, but ILAsm treats them as such |
|
"property", "type", "flags", "callconv", "strict" |
|
); |
|
|
|
static HashSet<string> BuildKeywordList(params string[] keywords) |
|
{ |
|
HashSet<string> s = new HashSet<string>(keywords); |
|
foreach (var field in typeof(OpCodes).GetFields()) { |
|
s.Add(((OpCode)field.GetValue(null)).Name); |
|
} |
|
return s; |
|
} |
|
|
|
public static string Escape(string identifier) |
|
{ |
|
if (IsValidIdentifier(identifier) && !ilKeywords.Contains(identifier)) { |
|
return identifier; |
|
} else { |
|
// The ECMA specification says that ' inside SQString should be ecaped using an octal escape sequence, |
|
// but we follow Microsoft's ILDasm and use \'. |
|
return "'" + EscapeString(identifier).Replace("'", "\\'") + "'"; |
|
} |
|
} |
|
|
|
public static void WriteTo(this TypeReference type, ITextOutput writer, ILNameSyntax syntax = ILNameSyntax.Signature) |
|
{ |
|
ILNameSyntax syntaxForElementTypes = syntax == ILNameSyntax.SignatureNoNamedTypeParameters ? syntax : ILNameSyntax.Signature; |
|
if (type is PinnedType) { |
|
((PinnedType)type).ElementType.WriteTo(writer, syntaxForElementTypes); |
|
writer.Write(" pinned"); |
|
} else if (type is ArrayType) { |
|
ArrayType at = (ArrayType)type; |
|
at.ElementType.WriteTo(writer, syntaxForElementTypes); |
|
writer.Write('['); |
|
writer.Write(string.Join(", ", at.Dimensions)); |
|
writer.Write(']'); |
|
} else if (type is GenericParameter) { |
|
writer.Write('!'); |
|
if (((GenericParameter)type).Owner.GenericParameterType == GenericParameterType.Method) |
|
writer.Write('!'); |
|
if (string.IsNullOrEmpty(type.Name) || type.Name[0] == '!' || syntax == ILNameSyntax.SignatureNoNamedTypeParameters) |
|
writer.Write(((GenericParameter)type).Position.ToString()); |
|
else |
|
writer.Write(Escape(type.Name)); |
|
} else if (type is ByReferenceType) { |
|
((ByReferenceType)type).ElementType.WriteTo(writer, syntaxForElementTypes); |
|
writer.Write('&'); |
|
} else if (type is PointerType) { |
|
((PointerType)type).ElementType.WriteTo(writer, syntaxForElementTypes); |
|
writer.Write('*'); |
|
} else if (type is GenericInstanceType) { |
|
type.GetElementType().WriteTo(writer, syntaxForElementTypes); |
|
writer.Write('<'); |
|
var arguments = ((GenericInstanceType)type).GenericArguments; |
|
for (int i = 0; i < arguments.Count; i++) { |
|
if (i > 0) |
|
writer.Write(", "); |
|
arguments[i].WriteTo(writer, syntaxForElementTypes); |
|
} |
|
writer.Write('>'); |
|
} else if (type is OptionalModifierType) { |
|
((OptionalModifierType)type).ElementType.WriteTo(writer, syntax); |
|
writer.Write(" modopt("); |
|
((OptionalModifierType)type).ModifierType.WriteTo(writer, ILNameSyntax.TypeName); |
|
writer.Write(") "); |
|
} else if (type is RequiredModifierType) { |
|
((RequiredModifierType)type).ElementType.WriteTo(writer, syntax); |
|
writer.Write(" modreq("); |
|
((RequiredModifierType)type).ModifierType.WriteTo(writer, ILNameSyntax.TypeName); |
|
writer.Write(") "); |
|
} else if (type is SentinelType) { |
|
writer.Write("..., "); |
|
((SentinelType)type).ElementType.WriteTo(writer, syntax); |
|
} else { |
|
string name = PrimitiveTypeName(type.FullName); |
|
if (syntax == ILNameSyntax.ShortTypeName) { |
|
if (name != null) |
|
writer.Write(name); |
|
else |
|
writer.WriteReference(Escape(type.Name), type); |
|
} else if ((syntax == ILNameSyntax.Signature || syntax == ILNameSyntax.SignatureNoNamedTypeParameters) && name != null) { |
|
writer.Write(name); |
|
} else { |
|
if (syntax == ILNameSyntax.Signature || syntax == ILNameSyntax.SignatureNoNamedTypeParameters) |
|
writer.Write(type.IsValueType ? "valuetype " : "class "); |
|
|
|
if (type.DeclaringType != null) { |
|
type.DeclaringType.WriteTo(writer, ILNameSyntax.TypeName); |
|
writer.Write('/'); |
|
writer.WriteReference(Escape(type.Name), type); |
|
} else { |
|
if (!type.IsDefinition && type.Scope != null && !(type is TypeSpecification)) |
|
writer.Write("[{0}]", Escape(type.Scope.Name)); |
|
writer.WriteReference(Escape(type.FullName), type); |
|
} |
|
} |
|
} |
|
} |
|
|
|
public static void WriteOperand(ITextOutput writer, object operand) |
|
{ |
|
if (operand == null) |
|
throw new ArgumentNullException("operand"); |
|
|
|
Instruction targetInstruction = operand as Instruction; |
|
if (targetInstruction != null) { |
|
WriteOffsetReference(writer, targetInstruction); |
|
return; |
|
} |
|
|
|
Instruction[] targetInstructions = operand as Instruction[]; |
|
if (targetInstructions != null) { |
|
WriteLabelList(writer, targetInstructions); |
|
return; |
|
} |
|
|
|
VariableReference variableRef = operand as VariableReference; |
|
if (variableRef != null) { |
|
if (string.IsNullOrEmpty(variableRef.Name)) |
|
writer.WriteReference(variableRef.Index.ToString(), variableRef); |
|
else |
|
writer.WriteReference(Escape(variableRef.Name), variableRef); |
|
return; |
|
} |
|
|
|
ParameterReference paramRef = operand as ParameterReference; |
|
if (paramRef != null) { |
|
if (string.IsNullOrEmpty(paramRef.Name)) |
|
writer.WriteReference(paramRef.Index.ToString(), paramRef); |
|
else |
|
writer.WriteReference(Escape(paramRef.Name), paramRef); |
|
return; |
|
} |
|
|
|
MethodReference methodRef = operand as MethodReference; |
|
if (methodRef != null) { |
|
methodRef.WriteTo(writer); |
|
return; |
|
} |
|
|
|
TypeReference typeRef = operand as TypeReference; |
|
if (typeRef != null) { |
|
typeRef.WriteTo(writer, ILNameSyntax.TypeName); |
|
return; |
|
} |
|
|
|
FieldReference fieldRef = operand as FieldReference; |
|
if (fieldRef != null) { |
|
fieldRef.WriteTo(writer); |
|
return; |
|
} |
|
|
|
string s = operand as string; |
|
if (s != null) { |
|
WriteOperand(writer, s); |
|
} else if (operand is char) { |
|
writer.Write(((int)(char)operand).ToString()); |
|
} else if (operand is float) { |
|
WriteOperand(writer, (float)operand); |
|
} else if (operand is double) { |
|
WriteOperand(writer, (double)operand); |
|
} else if (operand is bool) { |
|
writer.Write((bool)operand ? "true" : "false"); |
|
} else { |
|
s = ToInvariantCultureString(operand); |
|
writer.Write(s); |
|
} |
|
} |
|
|
|
public static void WriteOperand(ITextOutput writer, long val) |
|
{ |
|
writer.Write(ToInvariantCultureString(val)); |
|
} |
|
|
|
public static void WriteOperand(ITextOutput writer, float val) |
|
{ |
|
if (val == 0) { |
|
if (1 / val == float.NegativeInfinity) { |
|
// negative zero is a special case |
|
writer.Write('-'); |
|
} |
|
writer.Write("0.0"); |
|
} else if (float.IsInfinity(val) || float.IsNaN(val)) { |
|
byte[] data = BitConverter.GetBytes(val); |
|
writer.Write('('); |
|
for (int i = 0; i < data.Length; i++) { |
|
if (i > 0) |
|
writer.Write(' '); |
|
writer.Write(data[i].ToString("X2")); |
|
} |
|
writer.Write(')'); |
|
} else { |
|
writer.Write(val.ToString("R", System.Globalization.CultureInfo.InvariantCulture)); |
|
} |
|
} |
|
|
|
public static void WriteOperand(ITextOutput writer, double val) |
|
{ |
|
if (val == 0) { |
|
if (1 / val == double.NegativeInfinity) { |
|
// negative zero is a special case |
|
writer.Write('-'); |
|
} |
|
writer.Write("0.0"); |
|
} else if (double.IsInfinity(val) || double.IsNaN(val)) { |
|
byte[] data = BitConverter.GetBytes(val); |
|
writer.Write('('); |
|
for (int i = 0; i < data.Length; i++) { |
|
if (i > 0) |
|
writer.Write(' '); |
|
writer.Write(data[i].ToString("X2")); |
|
} |
|
writer.Write(')'); |
|
} else { |
|
writer.Write(val.ToString("R", System.Globalization.CultureInfo.InvariantCulture)); |
|
} |
|
} |
|
|
|
public static void WriteOperand(ITextOutput writer, string operand) |
|
{ |
|
writer.Write('"'); |
|
writer.Write(EscapeString(operand)); |
|
writer.Write('"'); |
|
} |
|
|
|
public static string EscapeString(string str) |
|
{ |
|
StringBuilder sb = new StringBuilder(); |
|
foreach (char ch in str) { |
|
switch (ch) { |
|
case '"': |
|
sb.Append("\\\""); |
|
break; |
|
case '\\': |
|
sb.Append("\\\\"); |
|
break; |
|
case '\0': |
|
sb.Append("\\0"); |
|
break; |
|
case '\a': |
|
sb.Append("\\a"); |
|
break; |
|
case '\b': |
|
sb.Append("\\b"); |
|
break; |
|
case '\f': |
|
sb.Append("\\f"); |
|
break; |
|
case '\n': |
|
sb.Append("\\n"); |
|
break; |
|
case '\r': |
|
sb.Append("\\r"); |
|
break; |
|
case '\t': |
|
sb.Append("\\t"); |
|
break; |
|
case '\v': |
|
sb.Append("\\v"); |
|
break; |
|
default: |
|
// print control characters and uncommon white spaces as numbers |
|
if (char.IsControl(ch) || char.IsSurrogate(ch) || (char.IsWhiteSpace(ch) && ch != ' ')) { |
|
sb.Append("\\u" + ((int)ch).ToString("x4")); |
|
} else { |
|
sb.Append(ch); |
|
} |
|
break; |
|
} |
|
} |
|
return sb.ToString(); |
|
} |
|
public static string PrimitiveTypeName(string fullName) |
|
{ |
|
switch (fullName) { |
|
case "System.SByte": |
|
return "int8"; |
|
case "System.Int16": |
|
return "int16"; |
|
case "System.Int32": |
|
return "int32"; |
|
case "System.Int64": |
|
return "int64"; |
|
case "System.Byte": |
|
return "uint8"; |
|
case "System.UInt16": |
|
return "uint16"; |
|
case "System.UInt32": |
|
return "uint32"; |
|
case "System.UInt64": |
|
return "uint64"; |
|
case "System.Single": |
|
return "float32"; |
|
case "System.Double": |
|
return "float64"; |
|
case "System.Void": |
|
return "void"; |
|
case "System.Boolean": |
|
return "bool"; |
|
case "System.String": |
|
return "string"; |
|
case "System.Char": |
|
return "char"; |
|
case "System.Object": |
|
return "object"; |
|
case "System.IntPtr": |
|
return "native int"; |
|
default: |
|
return null; |
|
} |
|
} |
|
} |
|
}
|
|
|