CLR profiler: проблема в использовании DefineAssemblyRef - PullRequest
19 голосов
/ 28 июля 2011

Я хочу написать CLR-профилировщик, чтобы подключить нашу прикладную функцию к GetILFunctionBody/SetILFunctionBody.

Я хочу использовать DefineAssemblyRef для импорта нашей c # dll (для использования в коде IL) в этом коде. DefineAssemblyRef всегда возвращает True?Должна ли моя dll быть подписана?Нужно ли его устанавливать в глобальный кэш сборок (GAC)?

     HRESULT CProfilerCallback::JITCompilationStarted
        (
        UINT functionId,
        BOOL fIsSafeToBlock
        )
    {
        ClassID classID;
        ModuleID moduleID;
        mdToken token;
        wchar_t wszClass[512];
        wchar_t wszMethod[512];
        HRESULT result = S_OK;
        ClassID classId = 0;
        ModuleID moduleId = 0;
        mdToken tkMethod = 0;

        // Get the moduleID and tkMethod    
        m_pICorProfilerInfo->GetFunctionInfo(functionId, &classId, &moduleId, &tkMethod);

        if(!GetMethodNameFromFunctionId(functionId,wszClass,wszMethod))
        {return S_FALSE;}


        if(wcscmp(wszMethod,L"FunctionName") == 0)
        {
            // Get the metadata import
            IMetaDataImport* pMetaDataImport = NULL;
            DebugBreak();
            result = m_pICorProfilerInfo->GetModuleMetaData
                (
                moduleId,
                ofRead, 
                IID_IMetaDataImport,
                (IUnknown** )&pMetaDataImport
                );


            if (FAILED(result))
            { return S_FALSE;}  
        //
        // Metadata modification
        //
        IMetaDataEmit* pMetaDataEmit = NULL;    
        IMetaDataAssemblyEmit* pMetaDataAssemblyEmit = NULL;  
        mdAssemblyRef tkLoggerLib;  
        HRESULT res;
        res = m_pICorProfilerInfo->GetModuleMetaData
            (
            moduleId,         /// The ID of the module to which the interface instance will be mapped
            ofRead | ofWrite,
            IID_IMetaDataEmit,
            (IUnknown** )&pMetaDataEmit
            );

        if (FAILED(res)) {DebugBreak();  return S_FALSE;}  /// DebugBreak for debug 

        res = pMetaDataEmit->QueryInterface
            (
            IID_IMetaDataAssemblyEmit,
            (void**)&pMetaDataAssemblyEmit
            );

        if (FAILED(res)) { return S_FALSE;}

        // Get the token for the Logger class and its Log method
        mdTypeDef tkLogger = 0;
        mdMethodDef tkLog = 0;

        // Create a token for the Log.dll assembly
        ASSEMBLYMETADATA amd;
        ZeroMemory(&amd, sizeof(amd));
        amd.usMajorVersion = 0;
        amd.usMinorVersion = 0;
        amd.usBuildNumber = 0;
        amd.usRevisionNumber = 0;

        res= pMetaDataAssemblyEmit->DefineAssemblyRef
            (
            NULL, 0, // No public key token
            L"Dllname",    ///dll name
            &amd, NULL, 0, 0,
            &tkLoggerLib
            );

        if (FAILED(res))  {return S_FALSE;  }

                ......

1 Ответ

1 голос
/ 27 февраля 2012

Согласно этому блогу MSDN http://blogs.msdn.com/b/davbr/archive/2006/02/27/540280.aspx:

IMetaDataAssemblyEmit :: DefineAssemblyRef () дает вам mdAssemblyRef для вашей сборки. Небольшая работа необходима, чтобы понять это правильно. Надежный способ ссылки на вашу сборку - это подписать вашу сборку, добавить ее в GAC и использовать открытый ключ, который «gacutil / l» распечатывает для вас

Вы также можете найти полезным этот проект - динамическое внедрение хуков CLR http://www.dupuis.me/node/18, который демонстрирует то, что вы пытаетесь сделать.

...