Microsoft MVP성태의 닷넷 이야기
글쓴 사람
정성태 (techsharer at outlook.com)
홈페이지
첨부 파일
 

요즘, ... VC++ 프로그램에서 .NET CLR을 호스팅하는 프로그램을 개발하고 있습니다.

그러다 보니, 여러 가지 상황을 만나게 되는데요. 그 중에 한 예를 소개하고자 합니다. ^^

일단, 정상적으로 Default Domain까지 받아오고 이후에 어셈블리를 로드시킨 후 그 안에 정의된 타입을 정상적으로 생성을 시켰습니다.
그다음은 당연히 그 타입의 인터페이스 포인터로 메서드를 호출하게 되겠지요.

그런데, 이게 웬일입니까?
분명히 메서드 호출은 되었는데, 예외도 발생하지 않았고, ... 아무런 동작 없이 메서드에서 리턴을 하는 것이었습니다.

==== unmanaged C++
MessageBox( "LoadTarget before" );
m_pHostLib->LoadTarget();
MessageBox( "LoadTarget after" );

MessageBox( "TestMethod before" );
m_pHostLib->TestMethod();
MessageBox( "TestMethod after" );

==== managed C#
public void LoadTarget()
{
   MessageBox.Show( "LoadTarget Called" );
   // 여러 가지 작업
}

public void TestMethod()
{
   MessageBox.Show( "TestMethod Called" );
}

코드는 대략 위와 같았는데요. 메시지 박스가 뜨는 것을 보면,

LoadTarget before
LoadTarget after
TestMethod before
TestMethod Called
TestMethod after

와 같은 순으로 팝업이 됩니다. 즉, TestMethod는 정상적으로 실행이 되었고, LoadTarget은 아무 한 일 없이 바로 리턴을 한 것입니다. 그렇다고 C++ 코드에서 예외를 발생한 것도 아니고요. 예외조차도 그냥 먹어버린 거죠. 도대체... 뭐가 문제인지 알 수 없는 상황이었습니다.

자, 이제 문제 해결을 위해 접근을 달리해야겠다는 생각을 했습니다.
C#의 메서드를 IDispatch 인터페이스를 이용해서 접근해 보자는 것이었죠.
그래서, 다음과 같이 호출을 변경시켰습니다.

DISPID dispID;
OLECHAR FAR* szMember = L"LoadTarget";

HRESULT hr = obj->GetIDsOfNames( IID_NULL, &szMember, 1, LOCALE_USER_DEFAULT, &dispID );
hr = obj->Invoke( dispID, IID_NULL, LOCALE_USER_DEFAULT,
    DISPATCH_METHOD, NULL, NULL, NULL, NULL );
if ( hr == S_OK )
{
    ::MessageBox( NULL, "LoadTarget", NULL, MB_OK );
} else {
      /// Win32 API FormatMessage를 이용하여 hr 값에 따른 오류 텍스트를 출력.
      // 물론, VS.NET 2003 도구 메뉴에서 제공되는 "오류 조회" 기능을 이용해도 된다.
}

물론, 운이 좋게도 hr != S_OK였고, 그래서 해당 hr 값에 따른 오류 텍스트를 반환 받았는데. ^^;
"예외가 발생하였습니다"
라는 문자열이 반환되더군요.

^^; 절망적이었습니다. 아니... 누군 지금 예외가 발생한지 몰라서 그러고 있었나!

음... 잠시 마음을 가다듬고... EXCEPINFO 인자를 넣어보았습니다.

EXCEPINFO excepInfo;
를 추가하고, 메서드 호출을 다음과 같이 변경했습니다.
HRESULT hr = obj->GetIDsOfNames( IID_NULL, &szMember, 1, LOCALE_USER_DEFAULT, &dispID );
hr = obj->Invoke( dispID, IID_NULL, LOCALE_USER_DEFAULT,
    DISPATCH_METHOD, NULL, NULL, &exepInfo, NULL );
if ( hr == S_OK )
{
    ::MessageBox( NULL, "LoadTarget", NULL, MB_OK );
} else {
    MessageBox( NULL, excepInfo.bstrDescription, NULL, MB_OK );
}

휴... 이제서야 제대로 된 오류 메시지를 얻을 수 있었습니다.
무슨무슨 어셈블리를 발견할 수 없다고 하는 것이었지요. 실제로 LoadTarget 메서드에서는 어떤 어셈블리에서 제공되는 기능을 사용을 했는 데, 그 부분이 문제가 되었던 것입니다. 그래서, 해당 어셈블리를 복사하고 다시 호출을 했더니 정상적으로 호출이 되었습니다.

여기서, 한 가지 알 수 있는 것이... CLR의 JIT 컴파일러가 메서드 단위로 컴파일 한다는 것을 확인할 수 있겠지요. LoadTarget은 문제가 발생하고, TestMethod는 정상적으로 호출된 것으로 볼 수 있습니다. 실제 메서드 호출에 앞서 TestMethod를 JIT 컴파일하는 중에는 오류가 발생하지 않았기 때문에 실행이 되었으나, LoadTarget의 경우에는 사용해야 하는 외부 어셈블리를 찾을 수 없어서 JIT 컴파일 단계에서 오류가 발생한 것이었습니다.

문제는,,, 그 오류를 CLR이 먹었다는 것인데요. 차라리 그 부분에서 CLR 오류 메시지 박스를 C# 프로그램에서처럼 띄워 주었더라면 좀 더 쉽게 해결이 가능했었는데 말이죠.

암튼... ^^; 험난한 디버깅이었습니다.








[최초 등록일: ]
[최종 수정일: 6/25/2021]

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)
1813정성태11/18/201423138오류 유형: 262. Build Events에 robocopy 작업이 있는 경우 "VCEnd exited with code 3" 오류 발생
1812정성태11/17/201420846.NET Framework: 482. ETW 자동 생성 코드의 VerificationException 예외 발생
1811정성태11/16/201422131.NET Framework: 481. Mono 내부의 문자열 처리 방식은 UTF-8
1810정성태11/15/201459153.NET Framework: 480. C# - 배치 파일 실행하고 출력 결과를 얻는 방법 [4]
1809정성태11/11/201424064.NET Framework: 479. Mono Profiler를 Unity의 Plugin으로 사용하는 방법
1808정성태11/11/201424333.NET Framework: 478. C# - 폴더 경로 문자열에서 "..", "." 표기를 고려한 최종 문자열을 얻는 방법 [2]
1807정성태11/10/201430742개발 환경 구성: 250. 서버용 Socket에서 사용하는 포트가 충돌한다면?파일 다운로드1
1806정성태11/10/201426530.NET Framework: 477. SeCreateGlobalPrivilege 특권과 WCF NamedPipe
1805정성태11/5/201423373.NET Framework: 476. Visual Studio에서 Mono용 Profiler 개발 [3]파일 다운로드1
1804정성태11/5/201429640.NET Framework: 475. ETW(Event Tracing for Windows)를 C#에서 사용하는 방법 [9]파일 다운로드1
1803정성태11/4/201421055오류 유형: 261. Windows Server Backup 오류 - Error in backup of E:\$Extend\$RmMetadata\$TxfLog
1802정성태11/4/201423591오류 유형: 260. 이벤트 로그 - Windows Error Reporting / AEAPPINVW8
1801정성태11/4/201428901오류 유형: 259. 이벤트 로그 - Windows Error Reporting / IPX Assertion / KorIME.exe [1]
1800정성태11/4/201419504오류 유형: 258. 이벤트 로그 - Starting a SMART disk polling operation in Automatic mode.
1799정성태11/4/201424211오류 유형: 257. 이벤트 로그 - The WMI Performance Adapter service entered the stopped state.
1798정성태11/4/201432842오류 유형: 256. 이벤트 로그 - The WinHTTP Web Proxy Auto-Discovery Service service entered the stopped state. [1]
1797정성태11/4/201418870오류 유형: 255. 이벤트 로그 - The Adobe Flash Player Update Service service entered the stopped state.
1796정성태10/30/201425922개발 환경 구성: 249. Visual Studio 2013에서 Mono 컴파일하는 방법
1795정성태10/29/201428084개발 환경 구성: 248. Lync 2013 서버 설치 방법
1794정성태10/29/201423546개발 환경 구성: 247. "Microsoft Office 365 Enterprise E3" 서비스에 대한 간략 소개
1793정성태10/27/201424315.NET Framework: 474. C# - chromiumembedded 사용 - 두 번째 이야기 [2]파일 다운로드1
1792정성태10/27/201424437.NET Framework: 473. WebClient 객체에 쿠키(Cookie)를 사용하는 방법
1791정성태10/22/201424035VC++: 83. G++ - 템플릿 클래스의 iterator 코드 사용에서 발생하는 컴파일 오류 [5]
1790정성태10/22/201419336오류 유형: 254. NETLOGON Service is paused on [... AD Server...]
1789정성태10/22/201422374오류 유형: 253. 이벤트 로그 - The client-side extension could not remove user policy settings for '...'
1788정성태10/22/201424203VC++: 82. COM 프로그래밍에서 HRESULT 타입의 S_FALSE는 실패일까요? 성공일까요? [2]
... 121  122  123  124  125  126  127  128  129  130  [131]  132  133  134  135  ...