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.
174 lines
5.1 KiB
174 lines
5.1 KiB
// 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 ICSharpCode.Decompiler.TypeSystem; |
|
using System.Reflection.Metadata; |
|
|
|
namespace ICSharpCode.Decompiler.IL |
|
{ |
|
static class ILTypeExtensions |
|
{ |
|
public static StackType GetStackType(this PrimitiveType primitiveType) |
|
{ |
|
switch (primitiveType) { |
|
case PrimitiveType.I1: |
|
case PrimitiveType.U1: |
|
case PrimitiveType.I2: |
|
case PrimitiveType.U2: |
|
case PrimitiveType.I4: |
|
case PrimitiveType.U4: |
|
return StackType.I4; |
|
case PrimitiveType.I8: |
|
case PrimitiveType.U8: |
|
return StackType.I8; |
|
case PrimitiveType.I: |
|
case PrimitiveType.U: |
|
case (PrimitiveType)0x0f: // Ptr |
|
case (PrimitiveType)0x1b: // FnPtr |
|
return StackType.I; |
|
case PrimitiveType.R4: |
|
return StackType.F4; |
|
case PrimitiveType.R8: |
|
return StackType.F8; |
|
case (PrimitiveType)0x10: // ByRef |
|
return StackType.Ref; |
|
case (PrimitiveType)0x01: // Void |
|
return StackType.Void; |
|
case PrimitiveType.Unknown: |
|
return StackType.Unknown; |
|
default: |
|
return StackType.O; |
|
} |
|
} |
|
|
|
public static Sign GetSign(this PrimitiveType primitiveType) |
|
{ |
|
switch (primitiveType) { |
|
case PrimitiveType.I1: |
|
case PrimitiveType.I2: |
|
case PrimitiveType.I4: |
|
case PrimitiveType.I8: |
|
case PrimitiveType.R4: |
|
case PrimitiveType.R8: |
|
case PrimitiveType.I: |
|
return Sign.Signed; |
|
case PrimitiveType.U1: |
|
case PrimitiveType.U2: |
|
case PrimitiveType.U4: |
|
case PrimitiveType.U8: |
|
case PrimitiveType.U: |
|
return Sign.Unsigned; |
|
default: |
|
return Sign.None; |
|
} |
|
} |
|
|
|
/// <summary> |
|
/// Gets the size in bytes of the primitive type. |
|
/// |
|
/// Returns 0 for non-primitive types. |
|
/// Returns <c>NativeIntSize</c> for native int/references. |
|
/// </summary> |
|
public static int GetSize(this PrimitiveType type) |
|
{ |
|
switch (type) { |
|
case PrimitiveType.I1: |
|
case PrimitiveType.U1: |
|
return 1; |
|
case PrimitiveType.I2: |
|
case PrimitiveType.U2: |
|
return 2; |
|
case PrimitiveType.I4: |
|
case PrimitiveType.U4: |
|
case PrimitiveType.R4: |
|
return 4; |
|
case PrimitiveType.I8: |
|
case PrimitiveType.R8: |
|
case PrimitiveType.U8: |
|
return 8; |
|
case PrimitiveType.I: |
|
case PrimitiveType.U: |
|
case PrimitiveType.Ref: |
|
return TypeUtils.NativeIntSize; |
|
default: |
|
return 0; |
|
} |
|
} |
|
|
|
/// <summary> |
|
/// Gets whether the type is a small integer type. |
|
/// Small integer types are: |
|
/// * bool, sbyte, byte, char, short, ushort |
|
/// * any enums that have a small integer type as underlying type |
|
/// </summary> |
|
public static bool IsSmallIntegerType(this PrimitiveType type) |
|
{ |
|
return GetSize(type) < 4; |
|
} |
|
|
|
public static bool IsIntegerType(this PrimitiveType primitiveType) |
|
{ |
|
return primitiveType.GetStackType().IsIntegerType(); |
|
} |
|
|
|
/// <summary> |
|
/// Infers the C# type for an IL instruction. |
|
/// |
|
/// Returns SpecialType.UnknownType for unsupported instructions. |
|
/// </summary> |
|
public static IType InferType(this ILInstruction inst) |
|
{ |
|
switch (inst) { |
|
case NewObj newObj: |
|
return newObj.Method.DeclaringType; |
|
case Call call: |
|
return call.Method.ReturnType; |
|
case CallVirt callVirt: |
|
return callVirt.Method.ReturnType; |
|
case CallIndirect calli: |
|
return calli.ReturnType; |
|
case UserDefinedLogicOperator logicOp: |
|
return logicOp.Method.ReturnType; |
|
case LdObj ldobj: |
|
return ldobj.Type; |
|
case StObj stobj: |
|
return stobj.Type; |
|
case LdLoc ldloc: |
|
return ldloc.Variable.Type; |
|
case StLoc stloc: |
|
return stloc.Variable.Type; |
|
case LdLoca ldloca: |
|
return new ByReferenceType(ldloca.Variable.Type); |
|
case LdFlda ldflda: |
|
return new ByReferenceType(ldflda.Field.Type); |
|
case LdsFlda ldsflda: |
|
return new ByReferenceType(ldsflda.Field.Type); |
|
case LdElema ldelema: |
|
if (ldelema.Array.InferType() is ArrayType arrayType) { |
|
var refType = new ByReferenceType(arrayType.ElementType); |
|
if (TypeUtils.IsCompatibleTypeForMemoryAccess(refType, ldelema.Type)) { |
|
return refType; |
|
} |
|
} |
|
return new ByReferenceType(ldelema.Type); |
|
default: |
|
return SpecialType.UnknownType; |
|
} |
|
} |
|
} |
|
}
|
|
|