Files
2026-03-30 20:11:57 +07:00

294 lines
9.9 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
{
private ushort attributes;
private Collection<CustomAttribute> custom_attributes;
private int offset = Mixin.NotResolvedMarker;
internal int rva = Mixin.NotResolvedMarker;
private byte[] initial_value;
private object constant = Mixin.NotResolved;
private MarshalInfo marshal_info;
private 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; }
}
private 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;
}
}
internal static partial class Mixin
{
public const int NotResolvedMarker = -2;
public const int NoDataMarker = -1;
}
}