System Field Acess Exception from Reflection.Emit derived class

236 Views Asked by At

I have a dynamically generated class that needs to access a field from the base class. I keep getting this exception:

Exception

System.FieldAccessException: 'Attempt by method 'AutoGenRelay.SetAsync(System.String, redius.Cell, System.Nullable`1, redius.SetPolicy)' to access field 'RediusTests.DaemonBase.IsTransaction' failed.'

Base Class:

public abstract class DaemonBase : Abs.Api.IAll,IDisposable {

        #region IAll implementation
         .......
        public abstract Task<Pair> BLPopAsync(IEnumerable<string> keys, TimeSpan? timeout);
          ......
        #endregion

        public bool IsTransaction;

        public DaemonBase() {

        }
        public void Dispose() {
            throw new NotImplementedException();
        }
    }

Autogenerated class

class RelayGen : DaemonBase {
        public override Task<Pair> BLPopAsync(IEnumerable<string> keys, TimeSpan? timeout) {
            if (this.IsTransaction) {
               return this.relay.normal.BLPopAsync(keys, timeout);
            } else
                return this.relay.tran.BLPopAsync(keys, timeout);
        }

How I access the FieldInfo of IsTransaction:

private static FieldInfo isTran = typeof(DaemonBase).GetFields(BindingFlags.Public | BindingFlags.NonPublic | BindingFlags.Instance )
                .First(x => x.Name == "IsTransaction");


   private void DefineMethod(MethodInfo abstractMethod) {

    Type essentialReturnType = abstractMethod.ReturnType.GetGenericArguments()[0];
    ParameterInfo[] parameters = abstractMethod.GetParameters();
    MethodBuilder newMethod = this.typeBuilder.DefineMethod(
        abstractMethod.Name,
        attributes: MethodAttributes.Public | MethodAttributes.Virtual,
        returnType: abstractMethod.ReturnType,
        parameterTypes: parameters.Select(par => par.ParameterType).ToArray()
    );
    this.typeBuilder.DefineMethodOverride(newMethod, abstractMethod);

    MethodInfo rediusMethod = typeof(OpsAbs)
        .GetMethods(BindingFlags.Public | BindingFlags.Instance)
        .Where(x => x.Name == newMethod.Name).First();

    ILGenerator ilgen = newMethod.GetILGenerator();
    Label falseLabel = ilgen.DefineLabel();

    ilgen.Emit(OpCodes.Ldarg_0);
    ilgen.Emit(OpCodes.Ldfld, isTran);
    ilgen.Emit(OpCodes.Brfalse, falseLabel);  //branching

    ilgen.Emit(OpCodes.Ldarg_0);
    ilgen.Emit(OpCodes.Ldfld, relay);
    ilgen.Emit(OpCodes.Ldfld, normalField);

    for (int argIndex = 1; argIndex <= parameters.Length; argIndex++) {
        ilgen.Emit(OpCodes.Ldarg, argIndex);
    }
    ilgen.Emit(OpCodes.Callvirt, rediusMethod);
    ilgen.Emit(OpCodes.Ret);


    ilgen.MarkLabel(falseLabel);  //false branch
    ilgen.Emit(OpCodes.Ldarg_0);
    ilgen.Emit(OpCodes.Ldfld, relay);
    ilgen.Emit(OpCodes.Ldfld, tranField);

    for (int argIndex = 1; argIndex <= parameters.Length; argIndex++) {
        ilgen.Emit(OpCodes.Ldarg, argIndex);
    }
    ilgen.Emit(OpCodes.Callvirt, rediusMethod);
    ilgen.Emit(OpCodes.Ret);
}

P.S I know I did not show a lot of code but I am trying to understand why would a derived class (Reflection.Emit generated) can not access a base class internal field?

IsTransaction is internal to DaemonBase. Why can't I access it from the derived class?

PS 2 I updated the code with the method.I did not provide the implementation of OpsAbs since it is used after i use the IsTransaction field implodes.

1

There are 1 best solutions below

1
On

Your error must be somewhere in the code you didn't show. Take a look at this example:

// create a TypeBuilder
var relayGenBuilder = module.DefineType("RelayGen", TypeAttributes.Public, typeof(DaemonBase));

var methodAttributes = MethodAttributes.Public
    | MethodAttributes.Virtual
    | MethodAttributes.HideBySig;
var returnType = typeof(Task<>).MakeGenericType(typeof(Pair));
var parameters = new []
{
    typeof(IEnumerable<>).MakeGenericType(typeof(string)),
    typeof(Nullable<>).MakeGenericType(typeof(TimeSpan))
};

// cteate a new method
var popAsync = relayGenBuilder.DefineMethod("BLPopAsync", 
    methodAttributes, 
    CallingConventions.HasThis,
    returnType,
    parameters); 

var ilGen = popAsync.GetILGenerator();

var falseLabel = ilGen.DefineLabel();
var isTran = typeof(DaemonBase).GetField("IsTransaction");
var writeLine = typeof(Console).GetMethod("WriteLine", new[] {typeof(string)});

ilGen.Emit(OpCodes.Ldarg_0);
ilGen.Emit(OpCodes.Ldfld, isTran);
ilGen.Emit(OpCodes.Brfalse_S, falseLabel);
// if (IsTransaction)
    ilGen.Emit(OpCodes.Ldstr, "IsTransaction equals true");
    ilGen.EmitCall(OpCodes.Call, writeLine, null);
    ilGen.Emit(OpCodes.Ldnull);
    ilGen.Emit(OpCodes.Ret);
// else
    ilGen.MarkLabel(falseLabel);
    ilGen.Emit(OpCodes.Ldnull);
    ilGen.Emit(OpCodes.Ret);

Do not forget to mark a new method as override:

relayGenBuilder.DefineMethodOverride(popAsync, typeof(DaemonBase).GetMethod("BLPopAsync"));

This code will create a method that looks like

public override Task<Pair> BLPopAsync(IEnumerable<string> keys, TimeSpan? timeout)
{
    if (IsTransaction)
        Console.WriteLine("IsTransaction equals true");

    return null;
}