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.
Your error must be somewhere in the code you didn't show. Take a look at this example:
Do not forget to mark a new method as
override
:This code will create a method that looks like