Microsoft MVP성태의 닷넷 이야기
.NET Framework: 570. .NET 4.5부터 추가된 CLR Profiler의 실행 시 Rejit 기능 [링크 복사], [링크+제목 복사],
조회: 21468
글쓴 사람
정성태 (techsharer at outlook.com)
홈페이지
첨부 파일
 

.NET 4.5부터 추가된 CLR Profiler의 실행 시 Rejit 기능

닷넷 프로파일러를 개발하는 입장에서, 가장 기다리던 기능이 바로 실행시 Rejit할 수 있는 것이었는데 드디어 .NET 4.5부터 가능하게 되었습니다. ^^

ReJIT: A How-To Guide
; https://docs.microsoft.com/en-us/archive/blogs/davbr/rejit-a-how-to-guide

자세한 것은 위의 글에 다 나오니 생략하고 ^^ 저는 시행 착오 부분을 설명할까 합니다.

우선, Rejit이 가능하도록 하려면 프로파일러가 CLR에게 SetEventMask 메서드를 통해 COR_PRF_ENABLE_REJIT을 설정해야 합니다.

COR_PRF_MONITOR Enumeration - COR_PRF_ENABLE_REJIT
; https://docs.microsoft.com/en-us/dotnet/framework/unmanaged-api/profiling/cor-prf-monitor-enumeration

DWORD dwEventMask = ... | COR_PRF_ENABLE_REJIT;

HRESULT hr = corProfilerInfo2->SetEventMask(m_dwEventMask);

그런데, 저렇게만 하는 경우 hr == 0x8013137C 오류 코드가 반환되는 경우가 있습니다. 이를 검색해 보면 다음의 도움말을 찾을 수 있는데,

0x8013137C
; https://github.com/dotnet/coreclr/blob/master/src/inc/corerror.xml

<HRESULT NumericValue="0x8013137C"> 
<SymbolicName>CORPROF_E_REJIT_NOT_ENABLED</SymbolicName> 
<Comment> This call is not supported unless ReJIT is first enabled during initialization by setting COR_PRF_ENABLE_REJIT via SetEventMask. </Comment> 
</HRESULT> 

아쉽게도 오류 메시지에서는 별다른 해결의 힌트를 얻을 수 없었고, 대신 COR_PRF_ENABLE_REJIT의 도움말에서 찾을 수 있었습니다. 이를 자세히 보면 다음과 같은 설명이 나옵니다.

Enables calls to the RequestReJIT and RequestRevert methods. The profiler must set this flag on startup. If the profiler specifies this flag, it must also specify COR_PRF_DISABLE_ALL_NGEN_IMAGES.


그렇습니다. SetEventMask에 설정할 때 "COR_PRF_ENABLE_REJIT | COR_PRF_DISABLE_ALL_NGEN_IMAGES"와 같이 함께 값을 설정해 주어야 하는 것입니다.

재미있는 것은, 저 문제는 .NET 4.5에서 발생하고 (정확한 버전은 테스트를 해봐야 하지만 확실하게는) .NET 4.6.1에서는 COR_PRF_DISABLE_ALL_NGEN_IMAGES 옵션을 주지 않아도 잘 됩니다. 즉 개선이 된 것입니다. 따라서, COR_PRF_DISABLE_ALL_NGEN_IMAGES에 대한 부하를 4.6.1에서 없애고 싶다면 다음과 같은 식으로 SetEventMask를 호출해 주시면 됩니다.

// 일단 COR_PRF_ENABLE_REJIT 옵션만 설정해 보고,
m_dwEventMask |= COR_PRF_ENABLE_REJIT;
hr = m_pICorProfilerInfo2->SetEventMask(m_dwEventMask);

if (hr != S_OK)
{
    // 오류가 발생했을 때에 한해 COR_PRF_DISABLE_ALL_NGEN_IMAGES 옵션도 함께 설정
    {
        DWORD dwEventMask = m_dwEventMask | COR_PRF_DISABLE_ALL_NGEN_IMAGES;
        hr = m_pICorProfilerInfo2->SetEventMask(dwEventMask);
    }
}

물론, 프로파일러 스스로 .NET 4.5와 4.6.1 버전을 인식할 수 있다면 그에 맞게 처리해주셔도 됩니다.




한 가지 더 설명할 것이 있는데, 바로 Rejit시킨 메서드를 다시 원본 메서드로 복원하고 싶을 때 호출하는 RequestRevert에 대해서입니다.

ICorProfilerInfo4::RequestRevert 메서드
; https://docs.microsoft.com/en-us/dotnet/framework/unmanaged-api/profiling/icorprofilerinfo4-requestrevert-method

이를 호출한 경우, 0x8013137D 오류가 발생하는 경우가 있습니다.

ICorProfilerInfo4 ptr = ...;
HRESULT hr = ptr->RequestRevert(count, modIDs, methodDefIDs, hrArray);
// hr == 0x8013137D

다행히 이번에는 검색해 보면 답이 나옵니다. ^^

<HRESULT NumericValue="0x8013137D"> 
<SymbolicName>CORPROF_E_ACTIVE_REJIT_REQUEST_NOT_FOUND</SymbolicName> 
<Comment> Could not revert the specified method, because a corresponding active ReJIT request was not found. Either the method was never requested to be reJITted OR the method was already reverted. </Comment> 
</HRESULT> 

보시는 바와 같이, Revert를 시키려면 반드시 그 메서드가 이전에 한번 RequestReJIT으로 요청된 적이 있었어야 합니다.

ICorProfilerInfo4::RequestReJIT 메서드
; https://docs.microsoft.com/en-us/dotnet/framework/unmanaged-api/profiling/icorprofilerinfo4-requestrejit-method

이게 일면 당연하게 생각될 수도 있는데요. 하지만 .NET Profiler가 IL 코드를 변경하는 시점을 이전의 버전에서 JITCompilationStarted로 제공했기 때문에,

ICorProfilerCallback::JITCompilationStarted 메서드
; https://docs.microsoft.com/en-us/dotnet/framework/unmanaged-api/profiling/icorprofilercallback-jitcompilationstarted-method

이 때 변경한 메서드에 대해서는 RequestRevert가 0x8013137D 반환만 하게 되는 것입니다. 따라서, 기존 프로파일러 제작자라면 Rejit 및 Revert를 모두 구현하고 싶은 경우 JITCompilationStarted 시점의 IL 코드 변경을 재고해 봐야 합니다.




참고로, 제니퍼 4 닷넷 에이전트는 CLR Profiler 기능을 이용해 만든 것인데요. 새로운 메서드를 프로파일링하려고 하면 아쉽지만 설정 후 w3wp.exe를 재시작해야 했습니다. 하지만, 올해 신규 버전으로 새로 출시된 제니퍼 5 닷넷 에이전트부터는 Rejit과 Revert 기능이 추가되었기 때문에 실행 시에 새로운 메서드에 대한 프로파일링 추가/삭제가 가능해졌습니다. ^^




[이 글에 대해서 여러분들과 의견을 공유하고 싶습니다. 틀리거나 미흡한 부분 또는 의문 사항이 있으시면 언제든 댓글 남겨주십시오.]







[최초 등록일: ]
[최종 수정일: 10/16/2020]

Creative Commons License
이 저작물은 크리에이티브 커먼즈 코리아 저작자표시-비영리-변경금지 2.0 대한민국 라이센스에 따라 이용하실 수 있습니다.
by SeongTae Jeong, mailto:techsharer at outlook.com

비밀번호

댓글 작성자
 




... 121  122  123  124  125  126  127  128  129  130  [131]  132  133  134  135  ...
NoWriterDateCnt.TitleFile(s)
1780정성태10/15/201424147오류 유형: 249. The application-specific permission settings do not grant Local Activation permission for the COM Server application with CLSID
1779정성태10/15/201419655오류 유형: 248. Active Directory에서 OU가 지워지지 않는 경우
1778정성태10/10/201418095오류 유형: 247. The Netlogon service could not create server share C:\Windows\SYSVOL\sysvol\[도메인명]\SCRIPTS.
1777정성태10/10/201421207오류 유형: 246. The processing of Group Policy failed. Windows attempted to read the file \\[도메인]\sysvol\[도메인]\Policies\{...GUID...}\gpt.ini
1776정성태10/10/201418217오류 유형: 245. 이벤트 로그 - Name resolution for the name _ldap._tcp.dc._msdcs.[도메인명]. timed out after none of the configured DNS servers responded.
1775정성태10/9/201419336오류 유형: 244. Visual Studio 디버깅 (2) - Unable to break execution. This process is not currently executing the type of code that you selected to debug.
1774정성태10/9/201426564개발 환경 구성: 246. IIS 작업자 프로세스의 20분 자동 재생(Recycle)을 끄는 방법
1773정성태10/8/201429747.NET Framework: 471. 웹 브라우저로 다운로드가 되는 파일을 왜 C# 코드로 하면 안되는 걸까요? [1]
1772정성태10/3/201418505.NET Framework: 470. C# 3.0의 기본 인자(default parameter)가 .NET 1.1/2.0에서도 실행될까? [3]
1771정성태10/2/201428027개발 환경 구성: 245. 실행된 프로세스(EXE)의 명령행 인자를 확인하고 싶다면 - Sysmon [4]
1770정성태10/2/201421663개발 환경 구성: 244. 매크로 정의를 이용해 파일 하나로 C++과 C#에서 공유하는 방법 [1]파일 다운로드1
1769정성태10/1/201424057개발 환경 구성: 243. Scala 개발 환경 구성(JVM, 닷넷) [1]
1768정성태10/1/201419471개발 환경 구성: 242. 배치 파일에서 Thread.Sleep 효과를 주는 방법 [5]
1767정성태10/1/201424602VS.NET IDE: 94. Visual Studio 2012/2013에서의 매크로 구현 - Visual Commander [2]
1766정성태10/1/201422418개발 환경 구성: 241. 책 "프로그래밍 클로저: Lisp"을 읽고 나서. [1]
1765정성태9/30/201425985.NET Framework: 469. Unity3d에서 transform을 변수에 할당해 사용하는 특별한 이유가 있을까요?
1764정성태9/30/201422243오류 유형: 243. 파일 삭제가 안 되는 경우 - The action can't be comleted because the file is open in System
1763정성태9/30/201423842.NET Framework: 468. PDB 파일을 연동해 소스 코드 라인 정보를 알아내는 방법파일 다운로드1
1762정성태9/30/201424533.NET Framework: 467. 닷넷에서 EIP/RIP 레지스터 값을 구하는 방법 [1]파일 다운로드1
1761정성태9/29/201421538.NET Framework: 466. 윈도우 운영체제의 보안 그룹 이름 및 설명 문자열을 바꾸는 방법파일 다운로드1
1760정성태9/28/201419797.NET Framework: 465. ICorProfilerInfo::GetILToNativeMapping 메서드가 0x80131358을 반환하는 경우
1759정성태9/27/201430962개발 환경 구성: 240. Visual C++ / x64 환경에서 inline-assembly를 매크로 어셈블리로 대체하는 방법파일 다운로드1
1758정성태9/23/201437799개발 환경 구성: 239. 원격 데스크톱 접속(RDP)을 기존의 콘솔 모드처럼 사용하는 방법 [1]
1757정성태9/23/201418376오류 유형: 242. Lync로 모임 참여 시 소리만 들리지 않는 경우 - 두 번째 이야기
1756정성태9/23/201427366기타: 48. NVidia 제품의 과다한 디스크 사용 [2]
1755정성태9/22/201434169오류 유형: 241. Unity Web Player를 설치해도 여전히 설치하라는 화면이 나오는 경우 [4]
... 121  122  123  124  125  126  127  128  129  130  [131]  132  133  134  135  ...