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

비밀번호

댓글 작성자
 




... 76  77  78  79  80  81  82  83  84  85  86  [87]  88  89  90  ...
NoWriterDateCnt.TitleFile(s)
11761정성태10/25/201821435Graphics: 26. 임의 축을 기반으로 3D 벡터 회전파일 다운로드1
11760정성태10/24/201816844개발 환경 구성: 418. Azure - Runbook 내에서 또 다른 Runbook 스크립트를 실행
11759정성태10/24/201818444개발 환경 구성: 417. Azure - Runbook에서 사용할 수 있는 다양한 메서드를 위한 부가 Module 추가
11758정성태10/23/201820705.NET Framework: 800. C# - Azure REST API 사용을 위한 인증 획득 [3]파일 다운로드1
11757정성태10/19/201817193개발 환경 구성: 416. Visual Studio 2017을 이용한 아두이노 프로그램 개발(및 디버깅)
11756정성태10/19/201820444오류 유형: 500. Visual Studio Code의 아두이노 프로그램 개발 시 인텔리센스가 안 된다면?
11755정성태10/19/201821538오류 유형: 499. Visual Studio Code extension for Arduino - #include errors detected. [1]
11754정성태10/19/201818369개발 환경 구성: 415. Visual Studio Code를 이용한 아두이노 프로그램 개발 - 새 프로젝트
11753정성태10/19/201825768개발 환경 구성: 414. Visual Studio Code를 이용한 아두이노 프로그램 개발
11752정성태10/18/201818391오류 유형: 498. SQL 서버 - Database source is not a supported version of SQL Server
11751정성태10/18/201818615오류 유형: 497. Visual Studio 실행 시 그래픽이 투명해진다거나, 깨진다면?
11750정성태10/18/201816905오류 유형: 496. 비주얼 스튜디오 - One or more projects in the solution were not loaded correctly.
11749정성태10/18/201819087개발 환경 구성: 413. 비주얼 스튜디오에서 작성한 프로그램을 빌드하는 가장 쉬운 방법
11748정성태10/18/201819298개발 환경 구성: 412. Arduino IDE를 Store App으로 설치한 경우 컴파일만 되고 배포가 안 되는 문제
11747정성태10/17/201820025.NET Framework: 799. C# - DLL에도 EXE처럼 Main 메서드를 넣어 실행할 수 있도록 만드는 방법파일 다운로드1
11746정성태10/15/201819604개발 환경 구성: 411. Bitvise SSH Client의 인증서 모드에서 자동 로그인 방법파일 다운로드1
11745정성태10/15/201817420오류 유형: 495. TFS 파일/폴더 삭제 - The item [...] could not be found in your workspace, or you do not have permission to access it.
11744정성태10/15/201818277개발 환경 구성: 410. msbuild로 .pubxml 설정에 따른 배포 파일을 만드는 방법
11743정성태10/15/201819440웹: 37. Bootstrap의 dl/dt/dd 조합에서 문자열이 잘리지 않도록 CSS 설정
11742정성태10/15/201825246스크립트: 13. 윈도우 배치(Batch) 스크립트에서 날짜/시간 문자열을 구하는 방법
11741정성태10/15/201819046Phone: 13. Android - LinearLayout 간략 설명
11740정성태10/15/201821002사물인터넷: 51. Synology NAS(DS216+II)를 이용한 원격 컴퓨터의 전원 스위치 제어
11739정성태10/15/201822677Windows: 151. 윈도우 10의 전원 관리가 "균형 조정(Balanced)"으로 바뀌는 문제
11738정성태10/15/201820971오류 유형: 494. docker - 윈도우에서 실행 시 "unknown shorthand flag" 오류 [1]
11737정성태10/13/201816847오류 유형: 493. Azure Kudu - There are ... items in this directory, but maxViewItems is set to 299
11736정성태10/12/201818504오류 유형: 492. Visual Studio 로딩 시 오류 - The 'Scc Display Information' package did not load correctly.
... 76  77  78  79  80  81  82  83  84  85  86  [87]  88  89  90  ...