성태의 닷넷 이야기
홈 주인
모아 놓은 자료
프로그래밍
질문/답변
사용자 관리
사용자
메뉴
아티클
외부 아티클
유용한 코드
온라인 기능
MathJax 입력기
최근 덧글
[정성태] VT sequences to "CONOUT$" vs. STD_O...
[정성태] NetCoreDbg is a managed code debugg...
[정성태] Evaluating tail call elimination in...
[정성태] What’s new in System.Text.Json in ....
[정성태] What's new in .NET 9: Cryptography ...
[정성태] 아... 제시해 주신 "https://akrzemi1.wordp...
[정성태] 다시 질문을 정리할 필요가 있을 것 같습니다. 제가 본문에...
[이승준] 완전히 잘못 짚었습니다. 댓글 지우고 싶네요. 검색을 해보...
[정성태] 우선 답글 감사합니다. ^^ 그런데, 사실 저 예제는 (g...
[이승준] 수정이 안되어서... byteArray는 BYTE* 타입입니다...
글쓰기
제목
이름
암호
전자우편
HTML
홈페이지
유형
제니퍼 .NET
닷넷
COM 개체 관련
스크립트
VC++
VS.NET IDE
Windows
Team Foundation Server
디버깅 기술
오류 유형
개발 환경 구성
웹
기타
Linux
Java
DDK
Math
Phone
Graphics
사물인터넷
부모글 보이기/감추기
내용
<div style='display: inline'> <h1 style='font-family: Malgun Gothic, Consolas; font-size: 20pt; color: #006699; text-align: center; font-weight: bold'>CLR Profiler - 별도 정의한 .NET 코드를 호출하도록 IL 코드 변경</h1> <p> 지난번에 기본적인 CLR Profiler를 소개했고,<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > 기본적인 CLR Profiler 소스 코드 설명 ; <a target='tab' href='http://www.sysnet.pe.kr/2/0/10950'>http://www.sysnet.pe.kr/2/0/10950</a> </pre> <br /> 그중에서 ModuleLoadFinished 콜백 동작을 테스트해봤었습니다<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > CLR Profiler로 살펴보는 SharedDomain의 모듈 로드 동작 ; <a target='tab' href='http://www.sysnet.pe.kr/2/0/10951'>http://www.sysnet.pe.kr/2/0/10951</a> </pre> <br /> 이번엔 ModuleLoadFinished와 함께 JITCompilationStarted를 조합해서 원하는 닷넷 메서드에 대해 우리가 정의한 C# DLL의 메서드를 호출하는 코드를 끼워넣을 것입니다. 이렇게 하려면 다음의 2가지 절차를 따라야 합니다.<br /> <br /> <ol> <li>ModuleLoadFinished 콜백: JIT 컴파일 단계에서 끼워넣을 외부 코드의 메타데이터 추가</li> <li>JITCompilationStarted 콜백: 기존 JIT 코드에 외부 DLL에 정의된 메서드 호출</li> </ol> <br /> 순서에 따라, C# DLL을 다음과 같은 코드로 간단하게 정의해서 GAC에 등록합니다.<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > using System; using System.Diagnostics; [assembly: System.Security.SecurityCritical] [assembly: System.Security.AllowPartiallyTrustedCallers] namespace <span style='color: blue; font-weight: bold'>Intercept.Helper</span> { public class <span style='color: blue; font-weight: bold'>ManagedLayer</span> { [System.Security.SecuritySafeCritical] <span style='color: blue; font-weight: bold'>public static void Enter()</span> { StackFrame sf = new StackFrame(1); Console.WriteLine("[Profiler] " + sf.GetMethod().Name + " called"); } } } /* C:\WINDOWS\system32><span style='color: blue; font-weight: bold'>gacutil /i C:\call_managed_profiler\Intercept.Helper\bin\Debug\Intercept.Helper.dll</span> Microsoft (R) .NET Global Assembly Cache Utility. Version 4.0.30319.0 Copyright (c) Microsoft Corporation. All rights reserved. Assembly successfully added to the cache */ </pre> <br /> 그다음 공개키 토큰 값을 알아내서,<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > C:\WINDOWS\system32><span style='color: blue; font-weight: bold'>sn -T C:\call_managed_profiler\Intercept.Helper\bin\Debug\Intercept.Helper.dll</span> Microsoft (R) .NET Framework Strong Name Utility Version 4.0.30319.0 Copyright (c) Microsoft Corporation. All rights reserved. Public key token is <span style='color: blue; font-weight: bold'>20a5976064ab527b</span> </pre> <br /> 이 값을 C++ Profiler 코드에 바이트 배열로 정의해 둡니다.<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > BYTE g_rgbPublicKeyToken[] = { 0x20, 0xa5, 0x97, 0x60, 0x64, 0xab, 0x52, 0x7b }; </pre> <a name='mlf_event'></a> <br /> 이제 다음과 같이 ModuleLoadFinished 콜백에서 우리가 만들어 두었던 C# 라이브러리인 Intercept.Helper.dll에 대한 참조를 추가합니다.<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > BYTE g_rgbPublicKeyToken[] = { 0x20, 0xa5, 0x97, 0x60, 0x64, 0xab, 0x52, 0x7b }; HRESULT CBasicClrProfiler::ModuleLoadFinished(ModuleID moduleId, HRESULT hrStatus) { IMetaDataAssemblyEmit *pAssemblyEmit = nullptr; do { hr = m_pICorProfilerInfo2->GetModuleMetaData(moduleId, ofWrite, IID_IMetaDataAssemblyEmit, (LPUNKNOWN *)&pAssemblyEmit); if (hr != S_OK) { break; } WCHAR wszLocale[] = { L"neutral" }; ASSEMBLYMETADATA assemblyMetaData; ZeroMemory(&assemblyMetaData, sizeof(assemblyMetaData)); assemblyMetaData.usMajorVersion = 1; assemblyMetaData.usMinorVersion = 0; assemblyMetaData.usBuildNumber = 0; assemblyMetaData.usRevisionNumber = 0; assemblyMetaData.szLocale = wszLocale; assemblyMetaData.cbLocale = _countof(wszLocale); mdAssemblyRef assemblyRef = NULL; hr = <span style='color: blue; font-weight: bold'>pAssemblyEmit->DefineAssemblyRef( (void *)g_rgbPublicKeyToken, sizeof(g_rgbPublicKeyToken), L"Intercept.Helper", &assemblyMetaData, NULL, NULL, 0, &assemblyRef);</span> if (hr != S_OK) { break; } } while (false); if (pAssemblyEmit != nullptr) { pAssemblyEmit->Release(); } } </pre> <br /> 그런데, DLL 참조만 추가해서는 안됩니다. 해당 DLL에 정의된 ManagedLayer 타입을 사용할 것이기 때문에 그에 대한 참조를 추가해줘야 합니다.<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > HRESULT CBasicClrProfiler::ModuleLoadFinished(ModuleID moduleId, HRESULT hrStatus) { IMetaDataAssemblyEmit *pAssemblyEmit = nullptr; IMetaDataEmit *pEmit = nullptr; do { // ...[생략]... hr = m_pICorProfilerInfo2->GetModuleMetaData(moduleId, ofWrite, IID_IMetaDataEmit, (LPUNKNOWN *)&pEmit); if (hr != S_OK) { break; } // ...[생략]... mdTypeRef typeRef = mdTokenNil; hr = <span style='color: blue; font-weight: bold'>pEmit->DefineTypeRefByName(assemblyRef, L"Intercept.Helper.ManagedLayer", &typeRef);</span> if (hr != S_OK) { break; } } while (false); if (pAssemblyEmit != nullptr) { pAssemblyEmit->Release(); } if (pAssemblyEmit != nullptr) { pAssemblyEmit->Release(); } } </pre> <br /> 그래도 한 가지 남았군요. ^^ ManagedLayer 타입의 Enter 메서드를 호출할 것이므로 이에 대한 참조도 추가해야 합니다.<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > HRESULT CBasicClrProfiler::ModuleLoadFinished(ModuleID moduleId, HRESULT hrStatus) { IMetaDataAssemblyEmit *pAssemblyEmit = nullptr; IMetaDataEmit *pEmit = nullptr; do { // ...[생략]... COR_SIGNATURE sigFunctionProbe[] = { IMAGE_CEE_CS_CALLCONV_DEFAULT, // default calling convention 0x0, // number of arguments == 0 ELEMENT_TYPE_VOID, // return type == void }; mdToken mdEnterProbeRef; hr = <span style='color: blue; font-weight: bold'>pEmit->DefineMemberRef(typeRef, L"Enter", sigFunctionProbe, sizeof(sigFunctionProbe), &mdEnterProbeRef);</span> if (hr != S_OK) { break; } } while (false); // ...[생략]... } </pre> <br /> 이것으로 참조 작업은 모두 끝났는데, 아직 여분의 작업이 2가지 남았습니다. 먼저, 위의 작업의 마지막 결과물인 mdEnterProbeRef 값을 JITCompilationStarted 단계에서 쓸 수 있도록 보관해 놓는 작업입니다. 사실 JITCompilationStarted 단계에서도 참조된 값을 구할 수는 있지만 그 작업으로 인한 코드가 번거롭기 때문에 미리 값을 보존해 두는 것이 더 좋은 선택입니다. 따라서, 다음과 같은 부가적인 코드가 필요합니다.<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > // BasicClrProfiler.h #pragma once #include "resource.h" // main symbols #include "SampleProfiler_i.h" #include "ICorProfilerCallback3Impl.h" struct ModuleInfo { mdToken m_mdEnterProbeRef; }; // ..[생략]... class ATL_NO_VTABLE CBasicClrProfiler : // ..[생략]... public ICorProfilerCallback3Impl<CBasicClrProfiler> { public: CBasicClrProfiler() { } DECLARE_REGISTRY_RESOURCEID(IDR_BASICCLRPROFILER) BEGIN_COM_MAP(CBasicClrProfiler) COM_INTERFACE_ENTRY(IBasicClrProfiler) COM_INTERFACE_ENTRY(IDispatch) COM_INTERFACE_ENTRY(ICorProfilerCallback) COM_INTERFACE_ENTRY(ICorProfilerCallback2) COM_INTERFACE_ENTRY(ICorProfilerCallback3) END_COM_MAP() // ..[생략]... <span style='color: blue; font-weight: bold'>IDToInfoMap<ModuleID, ModuleInfo> m_moduleIDToInfoMap;</span> // 구체적인 코드는 첨부파일 참조 }; OBJECT_ENTRY_AUTO(__uuidof(BasicClrProfiler), CBasicClrProfiler) </pre> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > // BasicClrProfiler.cpp HRESULT CBasicClrProfiler::ModuleLoadFinished(ModuleID moduleId, HRESULT hrStatus) { ModuleInfo moduleInfo = { 0 }; // ...[생략]... do { // ...[생략]... mdToken mdEnterProbeRef; hr = <span style='color: blue; font-weight: bold'>pEmit->DefineMemberRef(typeRef, L"Enter", sigFunctionProbe, sizeof(sigFunctionProbe), &mdEnterProbeRef);</span> if (hr != S_OK) { break; } <span style='color: blue; font-weight: bold'>moduleInfo.m_mdEnterProbeRef = mdEnterProbeRef;</span> } while (false); // ...[생략]... <span style='color: blue; font-weight: bold'>m_moduleIDToInfoMap.Update(moduleId, moduleInfo);</span> } </pre> <br /> 물론, Module이 Unload되었을 때 보관해 두었던 값을 제거해 주는 것도 잊으면 안되겠지요.<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > HRESULT CBasicClrProfiler::ModuleUnloadStarted(ModuleID moduleId) { <span style='color: blue; font-weight: bold'>m_moduleIDToInfoMap.EraseIfExists(moduleId);</span> return S_OK; } </pre> <br /> 두 번째로 필요한 작업은 꼭 필요한 제약으로 인한 것인데, 바로 mscorlib.dll로 인한 ModuleLoadFinished가 발생했을 때는 우리가 만든 코드를 참조 추가해서는 안된다는 점입니다. 왜냐하면 Intercept.Helper.dll은 mscorlib.dll을 참조하고 있는데, mscorlib.dll에 다시 Intercept.Helper.dll을 참조하면 순환 참조가 되기 때문입니다. 실제로 마이크로소프트 측은 mscorlib.dll에 참조를 걸지 말라고 주의를 주고 있습니다. 따라서 다음과 같이 mscorlib.dll을 구분하는 코드를 넣어야 합니다.<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > BOOL CBasicClrProfiler::ContainsAtEnd(LPCWSTR wszContainer, LPCWSTR wszProspectiveEnding) { size_t cchContainer = wcslen(wszContainer); size_t cchEnding = wcslen(wszProspectiveEnding); if (cchContainer < cchEnding) return FALSE; if (cchEnding == 0) return FALSE; if (_wcsicmp(wszProspectiveEnding, &(wszContainer[cchContainer - cchEnding])) != 0) { return FALSE; } return TRUE; } HRESULT CBasicClrProfiler::ModuleLoadFinished(ModuleID moduleId, HRESULT hrStatus) { ULONG cchModule = _MAX_PATH; ULONG rCchModule = 0; AssemblyID assemblyId = 0; LPCBYTE pModuleBaseLoadAddress; wchar_t szModule[_MAX_PATH]; HRESULT hr = m_pICorProfilerInfo2->GetModuleInfo(moduleId, &pModuleBaseLoadAddress, cchModule, &rCchModule, szModule, &assemblyId); if (hr != S_OK) { return S_OK; } <span style='color: blue; font-weight: bold'>if (ContainsAtEnd(szModule, L"mscorlib.dll") || ContainsAtEnd(szModule, L"intercept.helper.dll")) { return S_OK; }</span> IMetaDataAssemblyEmit *pAssemblyEmit = nullptr; IMetaDataEmit *pEmit = nullptr; do { // ...[생략]... } while (false); // ...[생략]... } </pre> <br /> <hr style='width: 50%' /><br /> <br /> 여기까지 했으면 이제 나머지 작업은 여러분들이 원하는 대상 메서드가 JIT 컴파일될 때 Intercept.Helper.ManagedLayer.Enter 메서드를 호출하도록 IL 코드를 변경하는 것입니다. 아쉽게도, 이 작업은 굉장히 복잡하기 때문에 제가 일일이 설명드릴 수 없습니다. 마이크로소프트 역시 이를 잘 알고 있어서, 다음과 같이 코드를 공개해 놓고 있는데 이를 참조하면 그래도 상당히 쉽게 만들 수 있습니다. ^^<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > <span style='text-decoration: line-through'>New sample code for rewriting IL: ILRewrite Profiler ; <a target='tab' href='http://i1.blogs.msdn.com/b/davbr/archive/2012/11/19/new-sample-code-for-rewriting-il-ilrewrite-profiler.aspx'>http://i1.blogs.msdn.com/b/davbr/archive/2012/11/19/new-sample-code-for-rewriting-il-ilrewrite-profiler.aspx</a></span> Welcome to the CLR Profiler CodePlex site ; <a target='tab' href='https://github.com/MicrosoftArchive/clrprofiler'>https://github.com/MicrosoftArchive/clrprofiler</a> </pre> <br /> <a name='JITCompilationStarted'></a> 이렇게 공개된 코드를 사용하면 다음과 같이 간단하게 JITCompilationStarted 코드를 완성할 수 있습니다.<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > HRESULT CBasicClrProfiler::Initialize(IUnknown * pICorProfilerInfoUnk) { if (pICorProfilerInfoUnk == NULL) { return S_OK; } m_pICorProfilerInfo2 = pICorProfilerInfoUnk; DWORD dwEventMask = COR_PRF_MONITOR_MODULE_LOADS | <span style='color: blue; font-weight: bold'>COR_PRF_MONITOR_JIT_COMPILATION | COR_PRF_DISABLE_INLINING</span>; m_pICorProfilerInfo2->SetEventMask(dwEventMask); return S_OK;} } HRESULT CBasicClrProfiler::JITCompilationStarted(FunctionID functionId, BOOL fIsSafeToBlock) { mdToken methodToken = 0; ModuleID moduleId = 0; ClassID classId; HRESULT hr = m_pICorProfilerInfo2->GetFunctionInfo(functionId, &classId, &moduleId, &methodToken); if (hr != S_OK || methodToken == 0) { return S_OK; } ModuleInfo moduleInfo; if (m_moduleIDToInfoMap.LookupIfExists(moduleId, &moduleInfo) == FALSE) { return S_OK; } if (IsNilToken(moduleInfo.m_mdEnterProbeRef) == TRUE) { return S_OK; } <span style='color: blue; font-weight: bold'>RewriteIL(m_pICorProfilerInfo2, moduleId, methodToken, moduleInfo.m_mdEnterProbeRef);</span> return S_OK; } </pre> <br /> 여기까지 만들고 다음의 예제 코드에 대해 프로파일링을 시작하면,<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > using System; namespace ConsoleApplication1 { class Program { static void Main(string[] args) { MyMethod(); Console.WriteLine("TEST"); Console.ReadLine(); } static void MyMethod() { } } } </pre> <br /> 출력 결과는 다음과 같습니다.<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > <span style='color: blue; font-weight: bold'>[Profiler] Main called [Profiler] MyMethod called</span> TEST </pre> <br /> "[Profiler]...called" 문자열은 ManagedLayer.Enter 메서드로 구현했던 바로 그 출력 내용이 호출된 것입니다.<br /> <br /> <hr style='width: 50%' /><br /> <br /> 이 정도면, 대충 IL 코드를 변경하는 프로파일러가 돌아가는 구조가 눈에 보이실 것입니다.<br /> <br /> (<a target='tab' href='http://www.sysnet.pe.kr/bbs/DownloadAttachment.aspx?fid=1023&boardid=331301885'>첨부한 파일은 이 글의 예제 코드와 마이크로소프트의 ILRewrite Profiler 예제의 압축 파일(ILRewrite10Source.zip)을 포함</a>합니다.)<br /> </p><br /> <br /><hr /><span style='color: Maroon'>[이 글에 대해서 여러분들과 의견을 공유하고 싶습니다. 틀리거나 미흡한 부분 또는 의문 사항이 있으시면 언제든 댓글 남겨주십시오.]</span> </div>
첨부파일
스팸 방지용 인증 번호
1880
(왼쪽의 숫자를 입력해야 합니다.)