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.
223 lines
7.6 KiB
223 lines
7.6 KiB
// Copyright (c) 2010-2013 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 ICSharpCode.Decompiler.TypeSystem.Implementation; |
|
using ICSharpCode.Decompiler.Util; |
|
|
|
namespace ICSharpCode.Decompiler.TypeSystem |
|
{ |
|
/// <summary> |
|
/// Represents an array type. |
|
/// </summary> |
|
public sealed class ArrayType : TypeWithElementType, ICompilationProvider |
|
{ |
|
readonly int dimensions; |
|
readonly ICompilation compilation; |
|
readonly Nullability nullability; |
|
|
|
public ArrayType(ICompilation compilation, IType elementType, int dimensions = 1, Nullability nullability = Nullability.Oblivious) : base(elementType) |
|
{ |
|
if (compilation == null) |
|
throw new ArgumentNullException(nameof(compilation)); |
|
if (dimensions <= 0) |
|
throw new ArgumentOutOfRangeException(nameof(dimensions), dimensions, "dimensions must be positive"); |
|
this.compilation = compilation; |
|
this.dimensions = dimensions; |
|
this.nullability = nullability; |
|
|
|
ICompilationProvider p = elementType as ICompilationProvider; |
|
if (p != null && p.Compilation != compilation) |
|
throw new InvalidOperationException("Cannot create an array type using a different compilation from the element type."); |
|
} |
|
|
|
public override TypeKind Kind { |
|
get { return TypeKind.Array; } |
|
} |
|
|
|
public ICompilation Compilation { |
|
get { return compilation; } |
|
} |
|
|
|
public int Dimensions { |
|
get { return dimensions; } |
|
} |
|
|
|
public override Nullability Nullability => nullability; |
|
|
|
public override IType ChangeNullability(Nullability nullability) |
|
{ |
|
if (nullability == this.nullability) |
|
return this; |
|
else |
|
return new ArrayType(compilation, elementType, dimensions, nullability); |
|
} |
|
|
|
public override string NameSuffix { |
|
get { |
|
return "[" + new string(',', dimensions - 1) + "]"; |
|
} |
|
} |
|
|
|
public override bool? IsReferenceType { |
|
get { return true; } |
|
} |
|
|
|
public override int GetHashCode() |
|
{ |
|
return unchecked(elementType.GetHashCode() * 71681 + dimensions); |
|
} |
|
|
|
public override bool Equals(IType other) |
|
{ |
|
ArrayType a = other as ArrayType; |
|
return a != null && elementType.Equals(a.elementType) && a.dimensions == dimensions && a.nullability == nullability; |
|
} |
|
|
|
public override string ToString() |
|
{ |
|
switch (nullability) |
|
{ |
|
case Nullability.Nullable: |
|
return elementType.ToString() + NameSuffix + "?"; |
|
case Nullability.NotNullable: |
|
return elementType.ToString() + NameSuffix + "!"; |
|
default: |
|
return elementType.ToString() + NameSuffix; |
|
} |
|
} |
|
|
|
public override IEnumerable<IType> DirectBaseTypes { |
|
get { |
|
List<IType> baseTypes = new List<IType>(); |
|
IType t = compilation.FindType(KnownTypeCode.Array); |
|
if (t.Kind != TypeKind.Unknown) |
|
baseTypes.Add(t); |
|
if (dimensions == 1 && elementType.Kind != TypeKind.Pointer) |
|
{ |
|
// single-dimensional arrays implement IList<T> |
|
ITypeDefinition def = compilation.FindType(KnownTypeCode.IListOfT) as ITypeDefinition; |
|
if (def != null) |
|
baseTypes.Add(new ParameterizedType(def, new[] { elementType })); |
|
// And in .NET 4.5 they also implement IReadOnlyList<T> |
|
def = compilation.FindType(KnownTypeCode.IReadOnlyListOfT) as ITypeDefinition; |
|
if (def != null) |
|
baseTypes.Add(new ParameterizedType(def, new[] { elementType })); |
|
} |
|
return baseTypes; |
|
} |
|
} |
|
|
|
public override IEnumerable<IMethod> GetMethods(Predicate<IMethod> filter = null, GetMemberOptions options = GetMemberOptions.None) |
|
{ |
|
if ((options & GetMemberOptions.IgnoreInheritedMembers) == GetMemberOptions.IgnoreInheritedMembers) |
|
return EmptyList<IMethod>.Instance; |
|
else |
|
return compilation.FindType(KnownTypeCode.Array).GetMethods(filter, options); |
|
} |
|
|
|
public override IEnumerable<IMethod> GetMethods(IReadOnlyList<IType> typeArguments, Predicate<IMethod> filter = null, GetMemberOptions options = GetMemberOptions.None) |
|
{ |
|
if ((options & GetMemberOptions.IgnoreInheritedMembers) == GetMemberOptions.IgnoreInheritedMembers) |
|
return EmptyList<IMethod>.Instance; |
|
else |
|
return compilation.FindType(KnownTypeCode.Array).GetMethods(typeArguments, filter, options); |
|
} |
|
|
|
public override IEnumerable<IMethod> GetAccessors(Predicate<IMethod> filter = null, GetMemberOptions options = GetMemberOptions.None) |
|
{ |
|
if ((options & GetMemberOptions.IgnoreInheritedMembers) == GetMemberOptions.IgnoreInheritedMembers) |
|
return EmptyList<IMethod>.Instance; |
|
else |
|
return compilation.FindType(KnownTypeCode.Array).GetAccessors(filter, options); |
|
} |
|
|
|
public override IEnumerable<IProperty> GetProperties(Predicate<IProperty> filter = null, GetMemberOptions options = GetMemberOptions.None) |
|
{ |
|
if ((options & GetMemberOptions.IgnoreInheritedMembers) == GetMemberOptions.IgnoreInheritedMembers) |
|
return EmptyList<IProperty>.Instance; |
|
else |
|
return compilation.FindType(KnownTypeCode.Array).GetProperties(filter, options); |
|
} |
|
|
|
// NestedTypes, Events, Fields: System.Array doesn't have any; so we can use the AbstractType default implementation |
|
// that simply returns an empty list |
|
|
|
public override IType AcceptVisitor(TypeVisitor visitor) |
|
{ |
|
return visitor.VisitArrayType(this); |
|
} |
|
|
|
public override IType VisitChildren(TypeVisitor visitor) |
|
{ |
|
IType e = elementType.AcceptVisitor(visitor); |
|
if (e == elementType) |
|
return this; |
|
else |
|
return new ArrayType(compilation, e, dimensions, nullability); |
|
} |
|
} |
|
|
|
[Serializable] |
|
public sealed class ArrayTypeReference : ITypeReference, ISupportsInterning |
|
{ |
|
readonly ITypeReference elementType; |
|
readonly int dimensions; |
|
|
|
public ArrayTypeReference(ITypeReference elementType, int dimensions = 1) |
|
{ |
|
if (elementType == null) |
|
throw new ArgumentNullException(nameof(elementType)); |
|
if (dimensions <= 0) |
|
throw new ArgumentOutOfRangeException(nameof(dimensions), dimensions, "dimensions must be positive"); |
|
this.elementType = elementType; |
|
this.dimensions = dimensions; |
|
} |
|
|
|
public ITypeReference ElementType { |
|
get { return elementType; } |
|
} |
|
|
|
public int Dimensions { |
|
get { return dimensions; } |
|
} |
|
|
|
public IType Resolve(ITypeResolveContext context) |
|
{ |
|
return new ArrayType(context.Compilation, elementType.Resolve(context), dimensions); |
|
} |
|
|
|
public override string ToString() |
|
{ |
|
return elementType.ToString() + "[" + new string(',', dimensions - 1) + "]"; |
|
} |
|
|
|
int ISupportsInterning.GetHashCodeForInterning() |
|
{ |
|
return elementType.GetHashCode() ^ dimensions; |
|
} |
|
|
|
bool ISupportsInterning.EqualsForInterning(ISupportsInterning other) |
|
{ |
|
ArrayTypeReference o = other as ArrayTypeReference; |
|
return o != null && elementType == o.elementType && dimensions == o.dimensions; |
|
} |
|
} |
|
}
|
|
|