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.
435 lines
15 KiB
435 lines
15 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 System.Collections.ObjectModel; |
|
using System.Diagnostics; |
|
using System.Linq; |
|
using System.Text; |
|
using ICSharpCode.Decompiler.TypeSystem.Implementation; |
|
|
|
namespace ICSharpCode.Decompiler.TypeSystem |
|
{ |
|
/// <summary> |
|
/// ParameterizedType represents an instance of a generic type. |
|
/// Example: List<string> |
|
/// </summary> |
|
/// <remarks> |
|
/// When getting the members, this type modifies the lists so that |
|
/// type parameters in the signatures of the members are replaced with |
|
/// the type arguments. |
|
/// </remarks> |
|
[Serializable] |
|
public sealed class ParameterizedType : IType, ICompilationProvider |
|
{ |
|
readonly ITypeDefinition genericType; |
|
readonly IType[] typeArguments; |
|
|
|
public ParameterizedType(ITypeDefinition genericType, IEnumerable<IType> typeArguments) |
|
{ |
|
if (genericType == null) |
|
throw new ArgumentNullException("genericType"); |
|
if (typeArguments == null) |
|
throw new ArgumentNullException("typeArguments"); |
|
this.genericType = genericType; |
|
this.typeArguments = typeArguments.ToArray(); // copy input array to ensure it isn't modified |
|
if (this.typeArguments.Length == 0) |
|
throw new ArgumentException("Cannot use ParameterizedType with 0 type arguments."); |
|
if (genericType.TypeParameterCount != this.typeArguments.Length) |
|
throw new ArgumentException("Number of type arguments must match the type definition's number of type parameters"); |
|
for (int i = 0; i < this.typeArguments.Length; i++) { |
|
if (this.typeArguments[i] == null) |
|
throw new ArgumentNullException("typeArguments[" + i + "]"); |
|
ICompilationProvider p = this.typeArguments[i] as ICompilationProvider; |
|
if (p != null && p.Compilation != genericType.Compilation) |
|
throw new InvalidOperationException("Cannot parameterize a type with type arguments from a different compilation."); |
|
} |
|
} |
|
|
|
/// <summary> |
|
/// Fast internal version of the constructor. (no safety checks) |
|
/// Keeps the array that was passed and assumes it won't be modified. |
|
/// </summary> |
|
internal ParameterizedType(ITypeDefinition genericType, IType[] typeArguments) |
|
{ |
|
Debug.Assert(genericType.TypeParameterCount == typeArguments.Length); |
|
this.genericType = genericType; |
|
this.typeArguments = typeArguments; |
|
} |
|
|
|
public TypeKind Kind { |
|
get { return genericType.Kind; } |
|
} |
|
|
|
public ICompilation Compilation { |
|
get { return genericType.Compilation; } |
|
} |
|
|
|
public bool? IsReferenceType { |
|
get { return genericType.IsReferenceType; } |
|
} |
|
|
|
public IType DeclaringType { |
|
get { |
|
ITypeDefinition declaringTypeDef = genericType.DeclaringTypeDefinition; |
|
if (declaringTypeDef != null && declaringTypeDef.TypeParameterCount > 0 |
|
&& declaringTypeDef.TypeParameterCount <= genericType.TypeParameterCount) |
|
{ |
|
IType[] newTypeArgs = new IType[declaringTypeDef.TypeParameterCount]; |
|
Array.Copy(this.typeArguments, 0, newTypeArgs, 0, newTypeArgs.Length); |
|
return new ParameterizedType(declaringTypeDef, newTypeArgs); |
|
} |
|
return declaringTypeDef; |
|
} |
|
} |
|
|
|
public int TypeParameterCount { |
|
get { return typeArguments.Length; } |
|
} |
|
|
|
public string FullName { |
|
get { return genericType.FullName; } |
|
} |
|
|
|
public string Name { |
|
get { return genericType.Name; } |
|
} |
|
|
|
public string Namespace { |
|
get { return genericType.Namespace; } |
|
} |
|
|
|
public string ReflectionName { |
|
get { |
|
StringBuilder b = new StringBuilder(genericType.ReflectionName); |
|
b.Append('['); |
|
for (int i = 0; i < typeArguments.Length; i++) { |
|
if (i > 0) |
|
b.Append(','); |
|
b.Append('['); |
|
b.Append(typeArguments[i].ReflectionName); |
|
b.Append(']'); |
|
} |
|
b.Append(']'); |
|
return b.ToString(); |
|
} |
|
} |
|
|
|
public override string ToString() |
|
{ |
|
return ReflectionName; |
|
} |
|
|
|
public IReadOnlyList<IType> TypeArguments { |
|
get { |
|
return typeArguments; |
|
} |
|
} |
|
|
|
public bool IsParameterized { |
|
get { |
|
return true; |
|
} |
|
} |
|
|
|
/// <summary> |
|
/// Same as 'parameterizedType.TypeArguments[index]', but is a bit more efficient (doesn't require the read-only wrapper). |
|
/// </summary> |
|
public IType GetTypeArgument(int index) |
|
{ |
|
return typeArguments[index]; |
|
} |
|
|
|
/// <summary> |
|
/// Gets the definition of the generic type. |
|
/// For <c>ParameterizedType</c>, this method never returns null. |
|
/// </summary> |
|
public ITypeDefinition GetDefinition() |
|
{ |
|
return genericType; |
|
} |
|
|
|
public ITypeReference ToTypeReference() |
|
{ |
|
return new ParameterizedTypeReference(genericType.ToTypeReference(), typeArguments.Select(t => t.ToTypeReference())); |
|
} |
|
|
|
/// <summary> |
|
/// Gets a type visitor that performs the substitution of class type parameters with the type arguments |
|
/// of this parameterized type. |
|
/// </summary> |
|
public TypeParameterSubstitution GetSubstitution() |
|
{ |
|
return new TypeParameterSubstitution(typeArguments, null); |
|
} |
|
|
|
/// <summary> |
|
/// Gets a type visitor that performs the substitution of class type parameters with the type arguments |
|
/// of this parameterized type, |
|
/// and also substitutes method type parameters with the specified method type arguments. |
|
/// </summary> |
|
public TypeParameterSubstitution GetSubstitution(IReadOnlyList<IType> methodTypeArguments) |
|
{ |
|
return new TypeParameterSubstitution(typeArguments, methodTypeArguments); |
|
} |
|
|
|
public IEnumerable<IType> DirectBaseTypes { |
|
get { |
|
var substitution = GetSubstitution(); |
|
return genericType.DirectBaseTypes.Select(t => t.AcceptVisitor(substitution)); |
|
} |
|
} |
|
|
|
public IEnumerable<IType> GetNestedTypes(Predicate<ITypeDefinition> filter = null, GetMemberOptions options = GetMemberOptions.None) |
|
{ |
|
if ((options & GetMemberOptions.ReturnMemberDefinitions) == GetMemberOptions.ReturnMemberDefinitions) |
|
return genericType.GetNestedTypes(filter, options); |
|
else |
|
return GetMembersHelper.GetNestedTypes(this, filter, options); |
|
} |
|
|
|
public IEnumerable<IType> GetNestedTypes(IReadOnlyList<IType> typeArguments, Predicate<ITypeDefinition> filter = null, GetMemberOptions options = GetMemberOptions.None) |
|
{ |
|
if ((options & GetMemberOptions.ReturnMemberDefinitions) == GetMemberOptions.ReturnMemberDefinitions) |
|
return genericType.GetNestedTypes(typeArguments, filter, options); |
|
else |
|
return GetMembersHelper.GetNestedTypes(this, typeArguments, filter, options); |
|
} |
|
|
|
public IEnumerable<IMethod> GetConstructors(Predicate<IUnresolvedMethod> filter = null, GetMemberOptions options = GetMemberOptions.IgnoreInheritedMembers) |
|
{ |
|
if ((options & GetMemberOptions.ReturnMemberDefinitions) == GetMemberOptions.ReturnMemberDefinitions) |
|
return genericType.GetConstructors(filter, options); |
|
else |
|
return GetMembersHelper.GetConstructors(this, filter, options); |
|
} |
|
|
|
public IEnumerable<IMethod> GetMethods(Predicate<IUnresolvedMethod> filter = null, GetMemberOptions options = GetMemberOptions.None) |
|
{ |
|
if ((options & GetMemberOptions.ReturnMemberDefinitions) == GetMemberOptions.ReturnMemberDefinitions) |
|
return genericType.GetMethods(filter, options); |
|
else |
|
return GetMembersHelper.GetMethods(this, filter, options); |
|
} |
|
|
|
public IEnumerable<IMethod> GetMethods(IReadOnlyList<IType> typeArguments, Predicate<IUnresolvedMethod> filter = null, GetMemberOptions options = GetMemberOptions.None) |
|
{ |
|
if ((options & GetMemberOptions.ReturnMemberDefinitions) == GetMemberOptions.ReturnMemberDefinitions) |
|
return genericType.GetMethods(typeArguments, filter, options); |
|
else |
|
return GetMembersHelper.GetMethods(this, typeArguments, filter, options); |
|
} |
|
|
|
public IEnumerable<IProperty> GetProperties(Predicate<IUnresolvedProperty> filter = null, GetMemberOptions options = GetMemberOptions.None) |
|
{ |
|
if ((options & GetMemberOptions.ReturnMemberDefinitions) == GetMemberOptions.ReturnMemberDefinitions) |
|
return genericType.GetProperties(filter, options); |
|
else |
|
return GetMembersHelper.GetProperties(this, filter, options); |
|
} |
|
|
|
public IEnumerable<IField> GetFields(Predicate<IUnresolvedField> filter = null, GetMemberOptions options = GetMemberOptions.None) |
|
{ |
|
if ((options & GetMemberOptions.ReturnMemberDefinitions) == GetMemberOptions.ReturnMemberDefinitions) |
|
return genericType.GetFields(filter, options); |
|
else |
|
return GetMembersHelper.GetFields(this, filter, options); |
|
} |
|
|
|
public IEnumerable<IEvent> GetEvents(Predicate<IUnresolvedEvent> filter = null, GetMemberOptions options = GetMemberOptions.None) |
|
{ |
|
if ((options & GetMemberOptions.ReturnMemberDefinitions) == GetMemberOptions.ReturnMemberDefinitions) |
|
return genericType.GetEvents(filter, options); |
|
else |
|
return GetMembersHelper.GetEvents(this, filter, options); |
|
} |
|
|
|
public IEnumerable<IMember> GetMembers(Predicate<IUnresolvedMember> filter = null, GetMemberOptions options = GetMemberOptions.None) |
|
{ |
|
if ((options & GetMemberOptions.ReturnMemberDefinitions) == GetMemberOptions.ReturnMemberDefinitions) |
|
return genericType.GetMembers(filter, options); |
|
else |
|
return GetMembersHelper.GetMembers(this, filter, options); |
|
} |
|
|
|
public IEnumerable<IMethod> GetAccessors(Predicate<IUnresolvedMethod> filter = null, GetMemberOptions options = GetMemberOptions.None) |
|
{ |
|
if ((options & GetMemberOptions.ReturnMemberDefinitions) == GetMemberOptions.ReturnMemberDefinitions) |
|
return genericType.GetAccessors(filter, options); |
|
else |
|
return GetMembersHelper.GetAccessors(this, filter, options); |
|
} |
|
|
|
public override bool Equals(object obj) |
|
{ |
|
return Equals(obj as IType); |
|
} |
|
|
|
public bool Equals(IType other) |
|
{ |
|
ParameterizedType c = other as ParameterizedType; |
|
if (c == null || !genericType.Equals(c.genericType) || typeArguments.Length != c.typeArguments.Length) |
|
return false; |
|
for (int i = 0; i < typeArguments.Length; i++) { |
|
if (!typeArguments[i].Equals(c.typeArguments[i])) |
|
return false; |
|
} |
|
return true; |
|
} |
|
|
|
public override int GetHashCode() |
|
{ |
|
int hashCode = genericType.GetHashCode(); |
|
unchecked { |
|
foreach (var ta in typeArguments) { |
|
hashCode *= 1000000007; |
|
hashCode += 1000000009 * ta.GetHashCode(); |
|
} |
|
} |
|
return hashCode; |
|
} |
|
|
|
public IType AcceptVisitor(TypeVisitor visitor) |
|
{ |
|
return visitor.VisitParameterizedType(this); |
|
} |
|
|
|
public IType VisitChildren(TypeVisitor visitor) |
|
{ |
|
IType g = genericType.AcceptVisitor(visitor); |
|
ITypeDefinition def = g as ITypeDefinition; |
|
if (def == null) |
|
return g; |
|
// Keep ta == null as long as no elements changed, allocate the array only if necessary. |
|
IType[] ta = (g != genericType) ? new IType[typeArguments.Length] : null; |
|
for (int i = 0; i < typeArguments.Length; i++) { |
|
IType r = typeArguments[i].AcceptVisitor(visitor); |
|
if (r == null) |
|
throw new NullReferenceException("TypeVisitor.Visit-method returned null"); |
|
if (ta == null && r != typeArguments[i]) { |
|
// we found a difference, so we need to allocate the array |
|
ta = new IType[typeArguments.Length]; |
|
for (int j = 0; j < i; j++) { |
|
ta[j] = typeArguments[j]; |
|
} |
|
} |
|
if (ta != null) |
|
ta[i] = r; |
|
} |
|
if (def == genericType && ta == null) |
|
return this; |
|
else |
|
return new ParameterizedType(def, ta ?? typeArguments); |
|
} |
|
} |
|
|
|
/// <summary> |
|
/// ParameterizedTypeReference is a reference to generic class that specifies the type parameters. |
|
/// Example: List<string> |
|
/// </summary> |
|
[Serializable] |
|
public sealed class ParameterizedTypeReference : ITypeReference, ISupportsInterning |
|
{ |
|
readonly ITypeReference genericType; |
|
readonly ITypeReference[] typeArguments; |
|
|
|
public ParameterizedTypeReference(ITypeReference genericType, IEnumerable<ITypeReference> typeArguments) |
|
{ |
|
if (genericType == null) |
|
throw new ArgumentNullException("genericType"); |
|
if (typeArguments == null) |
|
throw new ArgumentNullException("typeArguments"); |
|
this.genericType = genericType; |
|
this.typeArguments = typeArguments.ToArray(); |
|
for (int i = 0; i < this.typeArguments.Length; i++) { |
|
if (this.typeArguments[i] == null) |
|
throw new ArgumentNullException("typeArguments[" + i + "]"); |
|
} |
|
} |
|
|
|
public ITypeReference GenericType { |
|
get { return genericType; } |
|
} |
|
|
|
public ReadOnlyCollection<ITypeReference> TypeArguments { |
|
get { |
|
return Array.AsReadOnly(typeArguments); |
|
} |
|
} |
|
|
|
public IType Resolve(ITypeResolveContext context) |
|
{ |
|
IType baseType = genericType.Resolve(context); |
|
ITypeDefinition baseTypeDef = baseType.GetDefinition(); |
|
if (baseTypeDef == null) |
|
return baseType; |
|
int tpc = baseTypeDef.TypeParameterCount; |
|
if (tpc == 0) |
|
return baseTypeDef; |
|
IType[] resolvedTypes = new IType[tpc]; |
|
for (int i = 0; i < resolvedTypes.Length; i++) { |
|
if (i < typeArguments.Length) |
|
resolvedTypes[i] = typeArguments[i].Resolve(context); |
|
else |
|
resolvedTypes[i] = SpecialType.UnknownType; |
|
} |
|
return new ParameterizedType(baseTypeDef, resolvedTypes); |
|
} |
|
|
|
public override string ToString() |
|
{ |
|
StringBuilder b = new StringBuilder(genericType.ToString()); |
|
b.Append('['); |
|
for (int i = 0; i < typeArguments.Length; i++) { |
|
if (i > 0) |
|
b.Append(','); |
|
b.Append('['); |
|
b.Append(typeArguments[i].ToString()); |
|
b.Append(']'); |
|
} |
|
b.Append(']'); |
|
return b.ToString(); |
|
} |
|
|
|
int ISupportsInterning.GetHashCodeForInterning() |
|
{ |
|
int hashCode = genericType.GetHashCode(); |
|
unchecked { |
|
foreach (ITypeReference t in typeArguments) { |
|
hashCode *= 27; |
|
hashCode += t.GetHashCode(); |
|
} |
|
} |
|
return hashCode; |
|
} |
|
|
|
bool ISupportsInterning.EqualsForInterning(ISupportsInterning other) |
|
{ |
|
ParameterizedTypeReference o = other as ParameterizedTypeReference; |
|
if (o != null && genericType == o.genericType && typeArguments.Length == o.typeArguments.Length) { |
|
for (int i = 0; i < typeArguments.Length; i++) { |
|
if (typeArguments[i] != o.typeArguments[i]) |
|
return false; |
|
} |
|
return true; |
|
} |
|
return false; |
|
} |
|
} |
|
}
|
|
|