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.
125 lines
5.1 KiB
125 lines
5.1 KiB
// Copyright (c) 2015 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; |
|
using ICSharpCode.NRefactory.TypeSystem; |
|
|
|
namespace ICSharpCode.Decompiler |
|
{ |
|
public static class DecompilerTypeSystemUtils |
|
{ |
|
static int GetNativeSize(IType type) |
|
{ |
|
const int NativeIntSize = 6; // between 4 (Int32) and 8 (Int64) |
|
if (type.Kind == TypeKind.Pointer) |
|
return NativeIntSize; |
|
|
|
var typeForConstant = (type.Kind == TypeKind.Enum) ? type.GetDefinition().EnumUnderlyingType : type; |
|
var typeDef = typeForConstant.GetDefinition(); |
|
if (typeDef == null) |
|
return 0; |
|
switch (typeDef.KnownTypeCode) { |
|
case KnownTypeCode.Boolean: |
|
case KnownTypeCode.SByte: |
|
case KnownTypeCode.Byte: |
|
return 1; |
|
case KnownTypeCode.Char: |
|
case KnownTypeCode.Int16: |
|
case KnownTypeCode.UInt16: |
|
return 2; |
|
case KnownTypeCode.Int32: |
|
case KnownTypeCode.UInt32: |
|
case KnownTypeCode.Single: |
|
return 4; |
|
case KnownTypeCode.IntPtr: |
|
case KnownTypeCode.UIntPtr: |
|
return NativeIntSize; |
|
case KnownTypeCode.Int64: |
|
case KnownTypeCode.UInt64: |
|
case KnownTypeCode.Double: |
|
return 8; |
|
} |
|
return 0; |
|
} |
|
|
|
public static IType GetLargerType(IType type1, IType type2) |
|
{ |
|
return GetNativeSize(type1) >= GetNativeSize(type2) ? type1 : type2; |
|
} |
|
|
|
public static bool IsSmallIntegerType(this IType type) |
|
{ |
|
return GetNativeSize(type) < 4; |
|
} |
|
|
|
/// <summary> |
|
/// Gets whether reading/writing an element of accessType from the pointer |
|
/// is equivalent to reading/writing an element of the pointer's element type. |
|
/// </summary> |
|
public static bool IsCompatibleTypeForMemoryAccess(IType pointerType, IType accessType) |
|
{ |
|
IType memoryType; |
|
if (pointerType is PointerType) |
|
memoryType = ((PointerType)pointerType).ElementType; |
|
else if (pointerType is ByReferenceType) |
|
memoryType = ((ByReferenceType)pointerType).ElementType; |
|
else |
|
return false; |
|
ITypeDefinition memoryTypeDef = memoryType.GetDefinition(); |
|
ITypeDefinition accessTypeDef = accessType.GetDefinition(); |
|
if (memoryType.Kind == TypeKind.Enum && memoryTypeDef != null) { |
|
memoryType = memoryTypeDef.EnumUnderlyingType; |
|
memoryTypeDef = memoryType.GetDefinition(); |
|
} |
|
if (accessType.Kind == TypeKind.Enum && accessTypeDef != null) { |
|
accessType = accessTypeDef.EnumUnderlyingType; |
|
accessTypeDef = accessType.GetDefinition(); |
|
} |
|
if (memoryType.Equals(accessType)) |
|
return true; |
|
// If the types are not equal, the access still might produce equal results: |
|
if (memoryType.IsReferenceType == true && accessType.IsReferenceType == true) |
|
return true; |
|
if (memoryTypeDef != null) { |
|
switch (memoryTypeDef.KnownTypeCode) { |
|
case KnownTypeCode.Byte: |
|
case KnownTypeCode.SByte: |
|
// Reading small integers of different signs is not equivalent due to sign extension, |
|
// but writes are equivalent (truncation works the same for signed and unsigned) |
|
return accessType.IsKnownType(KnownTypeCode.Byte) || accessType.IsKnownType(KnownTypeCode.SByte); |
|
case KnownTypeCode.Int16: |
|
case KnownTypeCode.UInt16: |
|
return accessType.IsKnownType(KnownTypeCode.Int16) || accessType.IsKnownType(KnownTypeCode.UInt16); |
|
case KnownTypeCode.Int32: |
|
case KnownTypeCode.UInt32: |
|
return accessType.IsKnownType(KnownTypeCode.Int32) || accessType.IsKnownType(KnownTypeCode.UInt32); |
|
case KnownTypeCode.IntPtr: |
|
case KnownTypeCode.UIntPtr: |
|
return accessType.IsKnownType(KnownTypeCode.IntPtr) || accessType.IsKnownType(KnownTypeCode.UIntPtr); |
|
case KnownTypeCode.Int64: |
|
case KnownTypeCode.UInt64: |
|
return accessType.IsKnownType(KnownTypeCode.Int64) || accessType.IsKnownType(KnownTypeCode.UInt64); |
|
case KnownTypeCode.Char: |
|
return accessType.IsKnownType(KnownTypeCode.Char) || accessType.IsKnownType(KnownTypeCode.UInt16) || accessType.IsKnownType(KnownTypeCode.Int16); |
|
case KnownTypeCode.Boolean: |
|
return accessType.IsKnownType(KnownTypeCode.Boolean) || accessType.IsKnownType(KnownTypeCode.Byte) || accessType.IsKnownType(KnownTypeCode.SByte); |
|
} |
|
} |
|
return false; |
|
} |
|
} |
|
}
|
|
|