Предоставление разрешения на отражение динамически создаваемой сборке - PullRequest
10 голосов
/ 20 марта 2011

Я пишу простое настольное клиент / серверное приложение на C #.В целях самообучения я создал свою собственную систему сериализации для сообщений (определенных как классы), отправляемых туда и обратно между двумя приложениями через соединение через сокет tcp / ip.Система использует отражение во время инициализации для создания методов сериализации / десериализации для каждого типа сообщения путем передачи IL.

Первая версия этой системы использовала DynamicMethod, передавая true в конструктор, чтобы разрешить сгенерированный IL (этоработать с произвольными полями в типе сообщения), чтобы игнорировать права доступа.Это работало, и люди радовались, но я был недоволен тем, насколько болезненно непрозрачна отладка полученных функций.Поэтому я решил отказаться от DynamicMethod и использовать классы * Builder для создания динамической сборки, которую я мог бы дополнительно сохранить на диск и проверить с помощью такого инструмента, как .NET Reflector.

Я произвел рефакторинг системы, а затем немного ударилкирпичная стена.Каждый раз, когда одна из новых функций сериализации пытается получить доступ к закрытому полю или методу в одном из моих типов сообщений, я получаю исключение FieldAccessException или MethodAccessException.Я думаю, что после долгих поисков и скрежета зубов я сузил проблему до одного из разрешений;в частности, я думаю, что в моей динамически создаваемой сборке отсутствует разрешение ReflectionPermissionFlag.MemberAccess относительно вызывающей / конструирующей сборки (где находятся все отраженные типы).

К сожалению, я не могу понять, как изменить процесс создания динамической сборки, чтобы сборка имела разрешение на отражение в создаваемой сборке.Параметры разрешений для DefineDynamicAssembly, по-видимому, связаны с ограничением разрешения, а не с предоставлением его, что оставляет нас с параметром Evidence.Кажется, что доказательства волшебным образом переводятся в набор разрешений, но я не могу найти никаких полезных примеров или объяснений того, как это происходит.

Итак, мои вопросы:

(1) Я прав?в моем предположении, что моей проблемой является отсутствие разрешения на мою динамически создаваемую сборку?

(2) Если да, то как мне, как вызывающей сборке, предоставить необходимое разрешение моей динамической сборке?

Текущий код создания динамической сборки:

        AssemblyName assembly_name = new AssemblyName( "LCSerialization" );
        assembly_name.Version = new Version( 1, 0, 0, 0 );

        m_SerializationAssembly = current_domain.DefineDynamicAssembly( assembly_name, AssemblyBuilderAccess.RunAndSave );  // Fix me
        m_SerializationModule = m_SerializationAssembly.DefineDynamicModule( "MainModule", "LCSerialization.dll" );
        m_SerializationWrapperClass = m_SerializationModule.DefineType( "CSerializationWrapper", TypeAttributes.Public );

Обратите внимание, что мой проект ориентирован на .NET 3.5;В документации утверждается, что .NET 4.0 использует другое понятие безопасности и не поддерживает методы на основе Evidence / PemissionSet в DefineDynamicAssembly.

Чтобы привести конкретный пример, предположим, что у меня был такой класс:

[NetworkMessage]
public class CTestMessage
{
    public CTestMessage( int cheeseburgers )
    {
        m_CheeseBurgers = cheeseburgers
    }

    private int m_CheeseBurgers = 0;
}

, тогда моя система сериализации, столкнувшись с этим во время размышления инициализации, примерно в конечном итоге делает (вырезать и вставить здесь невозможно) с type = typeof (CTestMessage):

MethodBuilder serialization_builder = m_SerializationWrapperClass.DefineMethod( "Serialize_" + type.Name, 
                                                                                MethodAttributes.Public | MethodAttributes.Static, 
                                                                                null,
                                                                                new [] { type, typeof( BinaryWriter ) } );

ILGenerator s_il_gen = serialization_builder.GetILGenerator();
BindingFlags binding_flags_local_non_static = BindingFlags.Public | BindingFlags.NonPublic | BindingFlags.Instance | BindingFlags.DeclaredOnly;

s_il_gen.Emit( OpCodes.Ldarg_1 ); // Eval Stack: BinaryWriter
s_il_gen.Emit( OpCodes.Ldarg_0 ); // Eval Stack: BinaryWriter, testmessage
s_il_gen.Emit( OpCodes.Ldfld, type.GetField( "m_CheeseBurgers", binding_flags_local_non_static ) ); // Eval Stack: BinaryWriter, int
s_il_gen.Emit( OpCodes.Callvirt, typeof( BinaryWriter ).GetMethod( "Write", new Type[] { typeof( Int32 ) } ) ); // Eval Stack:
s_il_gen.Emit( OpCodes.Ret );

При последующем выполнении метода на Ldfld выдается исключениеинструкция.

Редактировать: Более подробно, демонстрируя, что то, что я прошу, должно быть возможным.Возьмите приведенный выше фрагмент кода, но замените MethodBuilder на DynamicMethod:

DynamicMethod serialization_builder = new DynamicMethod( "Serialize_" + type.Name, null, new [] { type, typeof( BinaryWriter ) }, true );

Теперь создайте делегат из DynamicMethod:

delegate void TestDelegate( CTestMessage, BinaryWriter );

TestDelegate test_delegate = serialization_builder.CreateDelegate( typeof( TestDelegate ) );

Этот делегат получает JITed и выполняется правильно без ошибок:

CTestMessage test_message = new CTestMessage( 5 );
BinaryWriter writer = new BinaryWriter( some_stream );
test_delegate( test_message, writer );

Ответы [ 2 ]

3 голосов
/ 01 апреля 2011

Проблема в том, что поле является приватным.Если вы сделаете это публичным, внешний метод будет работать нормально.DynamicMethod работает, несмотря на то, что он является частным, потому что CLR, по-видимому, разрешает внутримодульный доступ к частным полям - из SSCLI, clsload.cpp@2659:

// pCurrentClass can be NULL in the case of a global function
// pCurrentClass it the point from which we're trying to access something
// pTargetClass is the class containing the member we are trying to access
// dwMemberAccess is the member access within pTargetClass of the member we are trying to access
BOOL ClassLoader::CheckAccess(EEClass *pCurrentClass,
                              Assembly *pCurrentAssembly,
                              EEClass *pTargetClass,
                              Assembly *pTargetAssembly,
                              DWORD dwMemberAccess)
{
    // we're trying to access a member that is contained in the class pTargetClass, so need to 
    // check if have access to pTargetClass itself from the current point before worry about 
    // having access to the member within the class
    if (! CanAccessClass(pCurrentClass,
                         pCurrentAssembly, 
                         pTargetClass, 
                         pTargetAssembly))
        return FALSE;

    if (IsMdPublic(dwMemberAccess))
        return TRUE;

    // This is module-scope checking, to support C++ file & function statics.
    if (IsMdPrivateScope(dwMemberAccess)) {
        if (pCurrentClass == NULL)
            return FALSE;

        _ASSERTE(pTargetClass);

        return (pCurrentClass->GetModule() == pTargetClass->GetModule());
    }

Для внешнего доступа к частным полям вы, вероятно, должны использовать отражение, котороев значительной степени побеждает цель.

Редактировать Просто чтобы уточнить, что вы опубликовали, использует отражение для создания сборки, но сгенерированный вами IL не использует отражение для доступа к полю - этопрямой старый прямой доступ к полю, который взрывается, потому что целевое поле является внешним и частным.Вам бы пришлось испустить IL, который сам использует Type.GetField (). GetValue (), что довольно бессмысленно.

0 голосов
/ 27 июля 2013

Да, динамические сборки не разрешают такой доступ, будь то в .NET 3.5 или 4+.У меня такая же проблема.Мой обходной путь - разбить фактический код, генерирующий IL, на функцию, принимающую ILGenerator, и дважды вызывать его с другими аргументами, один раз (опционально) с ILGenerator из метода в динамической сборке, который я сохраню на диск, чтобыpeverify / ILDASM / и т.д..и один раз с ILGenerator от DynamicMethod.Таким образом, идентичный IL выделяется в обоих методах.

...