summaryrefslogtreecommitdiff
path: root/ICSharpCode.Decompiler/Disassembler/DisassemblerHelpers.cs
diff options
context:
space:
mode:
Diffstat (limited to 'ICSharpCode.Decompiler/Disassembler/DisassemblerHelpers.cs')
-rw-r--r--ICSharpCode.Decompiler/Disassembler/DisassemblerHelpers.cs447
1 files changed, 447 insertions, 0 deletions
diff --git a/ICSharpCode.Decompiler/Disassembler/DisassemblerHelpers.cs b/ICSharpCode.Decompiler/Disassembler/DisassemblerHelpers.cs
new file mode 100644
index 00000000..ea6e54db
--- /dev/null
+++ b/ICSharpCode.Decompiler/Disassembler/DisassemblerHelpers.cs
@@ -0,0 +1,447 @@
+// 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;
+
+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 void WriteOffsetReference(ITextOutput writer, Instruction instruction)
+ {
+ writer.WriteReference(CecilExtensions.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());
+ 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(CecilExtensions.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 ");
+ }
+ 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 "'" + NRefactory.CSharp.TextWriterTokenWriter.ConvertString(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 {
+ 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) {
+ writer.Write("\"" + NRefactory.CSharp.TextWriterTokenWriter.ConvertString(s) + "\"");
+ } else if (operand is char) {
+ writer.Write(((int)(char)operand).ToString());
+ } else if (operand is float) {
+ float val = (float)operand;
+ 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));
+ }
+ } else if (operand is double) {
+ double val = (double)operand;
+ 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));
+ }
+ } else if (operand is bool) {
+ writer.Write((bool)operand ? "true" : "false");
+ } else {
+ s = ToInvariantCultureString(operand);
+ writer.Write(s);
+ }
+ }
+
+ 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;
+ }
+ }
+ }
+}