239 lines
7.0 KiB
C#
Raw Normal View History

//
// GenericParameter.cs
//
// Author:
// Jb Evain (jbevain@gmail.com)
//
// (C) 2005 Jb Evain
//
// 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.
//
namespace Mono.Cecil {
using System;
internal sealed class GenericParameter : TypeReference {
int m_position;
string m_name;
GenericParameterAttributes m_attributes;
IGenericParameterProvider m_owner;
ConstraintCollection m_constraints;
public int Position {
get { return m_position; }
set { m_position = value; }
}
public GenericParameterAttributes Attributes {
get { return m_attributes; }
set { m_attributes = value; }
}
public IGenericParameterProvider Owner {
get { return m_owner; }
}
public bool HasConstraints {
get { return (m_constraints == null) ? false : (m_constraints.Count > 0); }
}
public ConstraintCollection Constraints {
get {
if (m_constraints == null)
m_constraints = new ConstraintCollection (this);
return m_constraints;
}
}
public override IMetadataScope Scope {
get {
if (m_owner is TypeReference)
return ((TypeReference) m_owner).Scope;
if (m_owner is MethodReference)
return ((MethodReference) m_owner).DeclaringType.Scope;
throw new InvalidOperationException ();
}
}
public override ModuleDefinition Module {
get {
if (m_owner is TypeReference)
return ((TypeReference) m_owner).Module;
if (m_owner is MethodReference)
return ((MethodReference) m_owner).DeclaringType.Module;
throw new InvalidOperationException ();
}
}
public override string Name {
get {
if (m_name != null)
return m_name;
if (m_owner is TypeReference)
return string.Concat ("!", m_position.ToString ());
else if (m_owner is MethodReference)
return string.Concat ("!!", m_position.ToString ());
else
throw new InvalidOperationException ();
}
set { m_name = value; }
}
public override string Namespace {
get { return string.Empty; }
set { throw new InvalidOperationException (); }
}
public override string FullName {
get { return Name; }
}
#region GenericParameterAttributes
public bool IsNonVariant {
get { return (m_attributes & GenericParameterAttributes.VarianceMask) == GenericParameterAttributes.NonVariant; }
set {
if (value) {
m_attributes &= ~GenericParameterAttributes.VarianceMask;
m_attributes |= GenericParameterAttributes.NonVariant;
} else
m_attributes &= ~(GenericParameterAttributes.VarianceMask & GenericParameterAttributes.NonVariant);
}
}
public bool IsCovariant {
get { return (m_attributes & GenericParameterAttributes.VarianceMask) == GenericParameterAttributes.Covariant; }
set {
if (value) {
m_attributes &= ~GenericParameterAttributes.VarianceMask;
m_attributes |= GenericParameterAttributes.Covariant;
} else
m_attributes &= ~(GenericParameterAttributes.VarianceMask & GenericParameterAttributes.Covariant);
}
}
public bool IsContravariant {
get { return (m_attributes & GenericParameterAttributes.VarianceMask) == GenericParameterAttributes.Contravariant; }
set {
if (value) {
m_attributes &= ~GenericParameterAttributes.VarianceMask;
m_attributes |= GenericParameterAttributes.Contravariant;
} else
m_attributes &= ~(GenericParameterAttributes.VarianceMask & GenericParameterAttributes.Contravariant);
}
}
public bool HasReferenceTypeConstraint {
get { return (m_attributes & GenericParameterAttributes.ReferenceTypeConstraint) != 0; }
set {
if (value) {
m_attributes |= GenericParameterAttributes.ReferenceTypeConstraint;
} else
m_attributes &= ~GenericParameterAttributes.ReferenceTypeConstraint;
}
}
public bool HasNotNullableValueTypeConstraint {
get { return (m_attributes & GenericParameterAttributes.NotNullableValueTypeConstraint) != 0; }
set {
if (value) {
m_attributes |= GenericParameterAttributes.NotNullableValueTypeConstraint;
} else
m_attributes &= ~GenericParameterAttributes.NotNullableValueTypeConstraint;
}
}
public bool HasDefaultConstructorConstraint {
get { return (m_attributes & GenericParameterAttributes.DefaultConstructorConstraint) != 0; }
set {
if (value) {
m_attributes |= GenericParameterAttributes.DefaultConstructorConstraint;
} else
m_attributes &= ~GenericParameterAttributes.DefaultConstructorConstraint;
}
}
#endregion
internal GenericParameter (int pos, IGenericParameterProvider owner) :
base (string.Empty, string.Empty)
{
m_position = pos;
m_owner = owner;
}
public GenericParameter (string name, IGenericParameterProvider owner) :
base (string.Empty, string.Empty)
{
m_name = name;
m_owner = owner;
}
public override TypeDefinition Resolve ()
{
return null;
}
internal static void CloneInto (IGenericParameterProvider old, IGenericParameterProvider np, ImportContext context)
{
foreach (GenericParameter gp in old.GenericParameters) {
GenericParameter ngp = Clone (gp, context);
np.GenericParameters.Add (ngp);
CloneConstraints (gp, ngp, context);
}
}
internal static GenericParameter Clone (GenericParameter gp, ImportContext context)
{
GenericParameter ngp;
if (gp.Owner is TypeReference)
ngp = new GenericParameter (gp.m_name, context.GenericContext.Type);
else if (gp.Owner is MethodReference)
ngp = new GenericParameter (gp.m_name, context.GenericContext.Method);
else
throw new NotSupportedException ();
ngp.Position = gp.Owner.GenericParameters.IndexOf (gp);
ngp.Attributes = gp.Attributes;
if (gp.HasCustomAttributes) {
foreach (CustomAttribute ca in gp.CustomAttributes)
ngp.CustomAttributes.Add (CustomAttribute.Clone (ca, context));
}
return ngp;
}
static void CloneConstraints (GenericParameter gp, GenericParameter ngp, ImportContext context)
{
if (gp.HasConstraints) {
foreach (TypeReference constraint in gp.Constraints)
ngp.Constraints.Add (context.Import (constraint));
}
}
}
}