282 lines
7.8 KiB
C#
282 lines
7.8 KiB
C#
|
//
|
||
|
// Author:
|
||
|
// Jb Evain (jbevain@gmail.com)
|
||
|
//
|
||
|
// Copyright (c) 2008 - 2015 Jb Evain
|
||
|
// Copyright (c) 2008 - 2011 Novell, Inc.
|
||
|
//
|
||
|
// Licensed under the MIT/X11 license.
|
||
|
//
|
||
|
|
||
|
using MonoFN.Collections.Generic;
|
||
|
using System;
|
||
|
|
||
|
namespace MonoFN.Cecil {
|
||
|
|
||
|
public sealed class FieldDefinition : FieldReference, IMemberDefinition, IConstantProvider, IMarshalInfoProvider {
|
||
|
|
||
|
ushort attributes;
|
||
|
Collection<CustomAttribute> custom_attributes;
|
||
|
|
||
|
int offset = Mixin.NotResolvedMarker;
|
||
|
|
||
|
internal int rva = Mixin.NotResolvedMarker;
|
||
|
byte [] initial_value;
|
||
|
|
||
|
object constant = Mixin.NotResolved;
|
||
|
|
||
|
MarshalInfo marshal_info;
|
||
|
|
||
|
void ResolveLayout ()
|
||
|
{
|
||
|
if (offset != Mixin.NotResolvedMarker)
|
||
|
return;
|
||
|
|
||
|
if (!HasImage) {
|
||
|
offset = Mixin.NoDataMarker;
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
lock (Module.SyncRoot) {
|
||
|
if (offset != Mixin.NotResolvedMarker)
|
||
|
return;
|
||
|
offset = Module.Read (this, (field, reader) => reader.ReadFieldLayout (field));
|
||
|
}
|
||
|
}
|
||
|
|
||
|
public bool HasLayoutInfo {
|
||
|
get {
|
||
|
if (offset >= 0)
|
||
|
return true;
|
||
|
|
||
|
ResolveLayout ();
|
||
|
|
||
|
return offset >= 0;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
public int Offset {
|
||
|
get {
|
||
|
if (offset >= 0)
|
||
|
return offset;
|
||
|
|
||
|
ResolveLayout ();
|
||
|
|
||
|
return offset >= 0 ? offset : -1;
|
||
|
}
|
||
|
set { offset = value; }
|
||
|
}
|
||
|
|
||
|
internal FieldDefinitionProjection WindowsRuntimeProjection {
|
||
|
get { return (FieldDefinitionProjection)projection; }
|
||
|
set { projection = value; }
|
||
|
}
|
||
|
|
||
|
void ResolveRVA ()
|
||
|
{
|
||
|
if (rva != Mixin.NotResolvedMarker)
|
||
|
return;
|
||
|
|
||
|
if (!HasImage)
|
||
|
return;
|
||
|
|
||
|
lock (Module.SyncRoot) {
|
||
|
if (rva != Mixin.NotResolvedMarker)
|
||
|
return;
|
||
|
rva = Module.Read (this, (field, reader) => reader.ReadFieldRVA (field));
|
||
|
}
|
||
|
}
|
||
|
|
||
|
public int RVA {
|
||
|
get {
|
||
|
if (rva > 0)
|
||
|
return rva;
|
||
|
|
||
|
ResolveRVA ();
|
||
|
|
||
|
return rva > 0 ? rva : 0;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
public byte [] InitialValue {
|
||
|
get {
|
||
|
if (initial_value != null)
|
||
|
return initial_value;
|
||
|
|
||
|
ResolveRVA ();
|
||
|
|
||
|
if (initial_value == null)
|
||
|
initial_value = Empty<byte>.Array;
|
||
|
|
||
|
return initial_value;
|
||
|
}
|
||
|
set {
|
||
|
initial_value = value;
|
||
|
HasFieldRVA = !initial_value.IsNullOrEmpty ();
|
||
|
rva = 0;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
public FieldAttributes Attributes {
|
||
|
get { return (FieldAttributes)attributes; }
|
||
|
set {
|
||
|
if (IsWindowsRuntimeProjection && (ushort)value != attributes)
|
||
|
throw new InvalidOperationException ();
|
||
|
|
||
|
attributes = (ushort)value;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
public bool HasConstant {
|
||
|
get {
|
||
|
this.ResolveConstant (ref constant, Module);
|
||
|
|
||
|
return constant != Mixin.NoValue;
|
||
|
}
|
||
|
set { if (!value) constant = Mixin.NoValue; }
|
||
|
}
|
||
|
|
||
|
public object Constant {
|
||
|
get { return HasConstant ? constant : null; }
|
||
|
set { constant = value; }
|
||
|
}
|
||
|
|
||
|
public bool HasCustomAttributes {
|
||
|
get {
|
||
|
if (custom_attributes != null)
|
||
|
return custom_attributes.Count > 0;
|
||
|
|
||
|
return this.GetHasCustomAttributes (Module);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
public Collection<CustomAttribute> CustomAttributes {
|
||
|
get { return custom_attributes ?? (this.GetCustomAttributes (ref custom_attributes, Module)); }
|
||
|
}
|
||
|
|
||
|
public bool HasMarshalInfo {
|
||
|
get {
|
||
|
if (marshal_info != null)
|
||
|
return true;
|
||
|
|
||
|
return this.GetHasMarshalInfo (Module);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
public MarshalInfo MarshalInfo {
|
||
|
get { return marshal_info ?? (this.GetMarshalInfo (ref marshal_info, Module)); }
|
||
|
set { marshal_info = value; }
|
||
|
}
|
||
|
|
||
|
#region FieldAttributes
|
||
|
|
||
|
public bool IsCompilerControlled {
|
||
|
get { return attributes.GetMaskedAttributes ((ushort)FieldAttributes.FieldAccessMask, (ushort)FieldAttributes.CompilerControlled); }
|
||
|
set { attributes = attributes.SetMaskedAttributes ((ushort)FieldAttributes.FieldAccessMask, (ushort)FieldAttributes.CompilerControlled, value); }
|
||
|
}
|
||
|
|
||
|
public bool IsPrivate {
|
||
|
get { return attributes.GetMaskedAttributes ((ushort)FieldAttributes.FieldAccessMask, (ushort)FieldAttributes.Private); }
|
||
|
set { attributes = attributes.SetMaskedAttributes ((ushort)FieldAttributes.FieldAccessMask, (ushort)FieldAttributes.Private, value); }
|
||
|
}
|
||
|
|
||
|
public bool IsFamilyAndAssembly {
|
||
|
get { return attributes.GetMaskedAttributes ((ushort)FieldAttributes.FieldAccessMask, (ushort)FieldAttributes.FamANDAssem); }
|
||
|
set { attributes = attributes.SetMaskedAttributes ((ushort)FieldAttributes.FieldAccessMask, (ushort)FieldAttributes.FamANDAssem, value); }
|
||
|
}
|
||
|
|
||
|
public bool IsAssembly {
|
||
|
get { return attributes.GetMaskedAttributes ((ushort)FieldAttributes.FieldAccessMask, (ushort)FieldAttributes.Assembly); }
|
||
|
set { attributes = attributes.SetMaskedAttributes ((ushort)FieldAttributes.FieldAccessMask, (ushort)FieldAttributes.Assembly, value); }
|
||
|
}
|
||
|
|
||
|
public bool IsFamily {
|
||
|
get { return attributes.GetMaskedAttributes ((ushort)FieldAttributes.FieldAccessMask, (ushort)FieldAttributes.Family); }
|
||
|
set { attributes = attributes.SetMaskedAttributes ((ushort)FieldAttributes.FieldAccessMask, (ushort)FieldAttributes.Family, value); }
|
||
|
}
|
||
|
|
||
|
public bool IsFamilyOrAssembly {
|
||
|
get { return attributes.GetMaskedAttributes ((ushort)FieldAttributes.FieldAccessMask, (ushort)FieldAttributes.FamORAssem); }
|
||
|
set { attributes = attributes.SetMaskedAttributes ((ushort)FieldAttributes.FieldAccessMask, (ushort)FieldAttributes.FamORAssem, value); }
|
||
|
}
|
||
|
|
||
|
public bool IsPublic {
|
||
|
get { return attributes.GetMaskedAttributes ((ushort)FieldAttributes.FieldAccessMask, (ushort)FieldAttributes.Public); }
|
||
|
set { attributes = attributes.SetMaskedAttributes ((ushort)FieldAttributes.FieldAccessMask, (ushort)FieldAttributes.Public, value); }
|
||
|
}
|
||
|
|
||
|
public bool IsStatic {
|
||
|
get { return attributes.GetAttributes ((ushort)FieldAttributes.Static); }
|
||
|
set { attributes = attributes.SetAttributes ((ushort)FieldAttributes.Static, value); }
|
||
|
}
|
||
|
|
||
|
public bool IsInitOnly {
|
||
|
get { return attributes.GetAttributes ((ushort)FieldAttributes.InitOnly); }
|
||
|
set { attributes = attributes.SetAttributes ((ushort)FieldAttributes.InitOnly, value); }
|
||
|
}
|
||
|
|
||
|
public bool IsLiteral {
|
||
|
get { return attributes.GetAttributes ((ushort)FieldAttributes.Literal); }
|
||
|
set { attributes = attributes.SetAttributes ((ushort)FieldAttributes.Literal, value); }
|
||
|
}
|
||
|
|
||
|
public bool IsNotSerialized {
|
||
|
get { return attributes.GetAttributes ((ushort)FieldAttributes.NotSerialized); }
|
||
|
set { attributes = attributes.SetAttributes ((ushort)FieldAttributes.NotSerialized, value); }
|
||
|
}
|
||
|
|
||
|
public bool IsSpecialName {
|
||
|
get { return attributes.GetAttributes ((ushort)FieldAttributes.SpecialName); }
|
||
|
set { attributes = attributes.SetAttributes ((ushort)FieldAttributes.SpecialName, value); }
|
||
|
}
|
||
|
|
||
|
public bool IsPInvokeImpl {
|
||
|
get { return attributes.GetAttributes ((ushort)FieldAttributes.PInvokeImpl); }
|
||
|
set { attributes = attributes.SetAttributes ((ushort)FieldAttributes.PInvokeImpl, value); }
|
||
|
}
|
||
|
|
||
|
public bool IsRuntimeSpecialName {
|
||
|
get { return attributes.GetAttributes ((ushort)FieldAttributes.RTSpecialName); }
|
||
|
set { attributes = attributes.SetAttributes ((ushort)FieldAttributes.RTSpecialName, value); }
|
||
|
}
|
||
|
|
||
|
public bool HasDefault {
|
||
|
get { return attributes.GetAttributes ((ushort)FieldAttributes.HasDefault); }
|
||
|
set { attributes = attributes.SetAttributes ((ushort)FieldAttributes.HasDefault, value); }
|
||
|
}
|
||
|
|
||
|
public bool HasFieldRVA {
|
||
|
get { return attributes.GetAttributes ((ushort)FieldAttributes.HasFieldRVA); }
|
||
|
set { attributes = attributes.SetAttributes ((ushort)FieldAttributes.HasFieldRVA, value); }
|
||
|
}
|
||
|
|
||
|
#endregion
|
||
|
|
||
|
public override bool IsDefinition {
|
||
|
get { return true; }
|
||
|
}
|
||
|
|
||
|
public new TypeDefinition DeclaringType {
|
||
|
get { return (TypeDefinition)base.DeclaringType; }
|
||
|
set { base.DeclaringType = value; }
|
||
|
}
|
||
|
|
||
|
public FieldDefinition (string name, FieldAttributes attributes, TypeReference fieldType)
|
||
|
: base (name, fieldType)
|
||
|
{
|
||
|
this.attributes = (ushort)attributes;
|
||
|
}
|
||
|
|
||
|
public override FieldDefinition Resolve ()
|
||
|
{
|
||
|
return this;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
static partial class Mixin {
|
||
|
|
||
|
public const int NotResolvedMarker = -2;
|
||
|
public const int NoDataMarker = -1;
|
||
|
}
|
||
|
}
|