Microsoft MVP성태의 닷넷 이야기
.NET Framework: 133. CallbackOnCollectedDelegate was detected [링크 복사], [링크+제목 복사],
조회: 30234
글쓴 사람
정성태 (techsharer at outlook.com)
홈페이지
첨부 파일
(연관된 글이 6개 있습니다.)

CallbackOnCollectedDelegate was detected


고객사에서 재미있는 리포트가 전달되어왔습니다. 얼마 전, 해당 고객사에서는 .NET 응용 프로그램에서 Win32 DLL을 호출하는 코드를 만들어야했는데, 이 과정에서 Win32 DLL에 .NET에서 만들어진 메서드를 콜백으로 전달해야 하는 것을 문의해왔었습니다.

당연히, delegate를 이용해서 전달하라고 알려줬지요. 고객사에서는 제가 준 샘플 코드를 동일하게 만들지 않고 나름대로의 생략과정을 거쳐서 아래와 같은 식으로 코드를 만들었습니다.

public delegate void 
    ByteArrayFunctionHandler([MarshalAs(UnmanagedType.LPArray, SizeConst = 6)] byte[] byteBuffer);

[DllImport("TestNativeAPI.dll")]
public static extern bool fnTestNativeAPI(ByteArrayFunctionHandler handler);

public Form1()
{
    InitializeComponent();

    fnTestNativeAPI(testFunc); // C/C++에 .NET 함수 포인터를 전달
}

void testFunc(byte[] byteBuffer) // Win32 DLL에서 콜백으로 testFunc을 호출
{
    foreach (byte aByte in byteBuffer)
    {
        Debug.WriteLine(aByte);
    }
}

위의 코드는 고객사가 전달해 준 코드를 다른 식으로 해석한 것이고 fnTestNativeAPI를 호출한 이후 꽤 많은 코드가 더 있는 상황이었습니다.

그런데, 여기서 문제가 발생한 것입니다. 콜백을 호출하게 되는 Win32 DLL의 함수를 호출하면 여지없이 다음과 같은 오류가 발생하는 것이었습니다.

[그림 1: CallbackOnCollectedDelegate 오류]
cpp_function_pointer_interop_1.png

CallbackOnCollectedDelegate was detected

Message: A callback was made on a garbage collected delegate of type 'WindowsFormsApplication1!WindowsFormsApplication1.Form1+ByteArrayFunctionHandler::Invoke'. This may cause application crashes, corruption and data loss. When passing delegates to unmanaged code, they must be kept alive by the managed application until it is guaranteed that they will never be called.



보자마자 느낌이 팍 오시는 분이 계시겠지요? ^^

그렇습니다. Managed 환경의 delegate 인스턴스를 Native에 전달했으니 Garbage Collector가 구동된 이후 delegate 인스턴스가 정리되어버린 것입니다. 그러니, 이후에 native에서 삭제된 인스턴스의 delegate 값으로 호출하니 "CallbackOnCollectedDelegate"라는 오류 메시지가 출력된 것입니다.

그렇다면 어떻게 고쳐야 할까요?
GC에 의해서 인스턴스가 정리되지 않도록 참조 포인터를 하나라도 유지하고 있으면 되는 것입니다. 이를 위해 다음과 같이 타입 멤버로 들고 있는 것도 좋은 방법이 될 수 있습니다.

ByteArrayFunctionHandler handler; // 참조 카운트 유지

public Form1()
{
    InitializeComponent();

    this.handler = new ByteArrayFunctionHandler(testFunc);
    fnTestNativeAPI(this.handler);
}

이렇게 되면 this.handler 인스턴스가 타입 멤버로 참조 카운트를 유지하고 있기 때문에 오류가 발생하지 않습니다. 물론, 아래와 같이 테스트를 해보면 다시 오류가 발생합니다.

public Form1()
{
    InitializeComponent();

    this.handler = new ByteArrayFunctionHandler(testFunc);
    fnTestNativeAPI(this.handler);
    this.handler = null; // 참조 카운트 제거
}

첨부된 솔루션 파일은 위의 코드를 테스트해볼 수 있도록 Win32 DLL 프로젝트와 WinForm 닷넷 프로젝트를 포함하고 있습니다.

이것과 연결되는 것이 MDA(Managed Debugging Assistants) 기능인데, 이 부분은 나중에 ^^ 설명드리도록 하겠습니다.

[다운로드: 예제 솔루션]




(2025-02-14 업데이트) 본문의 "fnTestNativeAPI(testFunc);" 코드를 좀 더 설명해 볼까요? 얼핏 보면 이것은 testFunc 함수가 놓인 코드 영역의 주소를 fnTestNativeAPI에 직접 전달하는 것처럼 여겨지는데, 그런 탓에 왜 이것이 잘못되었는가...라는 의문마저 들게 됩니다.

사실 저건 C# 컴파일러에 의해 상당히 압축된 문법이라서 그런 건데요, 원래 저 코드는 다음과 같이 풀어져서 컴파일됩니다.

// C# 2.0부터 지원하는 약식 문법
// fnTestNativeAPI(testFunc); 

// 위의 호출은 아래와 같이 풀어져서 컴파일
ByteArrayFunctionHandler func = new ByteArrayFunctionHandler(testFunc);
fnTestNativeAPI(func);

즉, testFunc 함수를 감싸는 ByteArrayFunctionHandler 객체를 생성하고, 그 객체를 fnTestNativeAPI에 전달하는 것이기 때문에 func 인스턴스 자체는 메서드 내부의 범위에서 로컬 변수로 정의되므로 호출이 완료된 후에는 언제든 GC에 의해 회수 가능한 대상이 되는 것입니다.



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

[연관 글]






[최초 등록일: ]
[최종 수정일: 2/14/2025]

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

비밀번호

댓글 작성자
 



2023-01-31 02시47분
[activedesk] 박수를 보냅니다....
[guest]
2025-02-14 12시11분
참조 카운트를 유지한다 하더라도, 만일 testFunc() 메서드가 Form1의 내부 field에 접근한다면, 별도로 this나 해당 필드를 Pinning 해줘야 할까요?
delegate가 캡쳐하는 this나 this.field가 native에서 콜백되는 도중 GC에 의해서 주소가 이동되어 버리면 문제가 되지 않는지 궁금합니다.
copyrat90
2025-02-14 12시46분
https://www.sysnet.pe.kr/2/0/13600 <- 이 글을 읽고 답을 얻었습니다.

말한 경우에는 Native to Managed Transition이 일어나서, 실제 콜백을 수행하는 주체는 관리 스레드이기 때문에,
GC가 작동되어 주소가 바뀔 때는 관리 스레드도 정지하기 때문에 문제가 되지 않겠군요.
copyrat90
2025-02-14 03시38분
이런 자문자답 덧글 너무 좋습니다. ^^
정성태

... 106  107  [108]  109  110  111  112  113  114  115  116  117  118  119  120  ...
NoWriterDateCnt.TitleFile(s)
11258정성태8/1/201720316.NET Framework: 668. 지연 서명된 DLL과 서명된 DLL의 차이점파일 다운로드1
11257정성태7/31/201719901.NET Framework: 667. bypassTrustedAppStrongNames 옵션 설명파일 다운로드1
11256정성태7/25/201721834디버깅 기술: 90. windbg의 lm 명령으로 보이지 않는 .NET 4.0 ClassLibrary를 명시적으로 로드하는 방법 [1]
11255정성태7/18/201724344디버깅 기술: 89. Win32 Debug CRT Heap Internals의 0xBAADF00D 표시 재현 [1]파일 다운로드3
11254정성태7/17/201720769개발 환경 구성: 322. "Visual Studio Emulator for Android" 에뮬레이터를 "Android Studio"와 함께 쓰는 방법
11253정성태7/17/201721408Math: 21. "Coding the Matrix" 문제 2.5.1 풀이 [1]파일 다운로드1
11252정성태7/13/201719139오류 유형: 411. RTVS 또는 PTVS 실행 시 Could not load type 'Microsoft.VisualStudio.InteractiveWindow.Shell.IVsInteractiveWindowFactory2'
11251정성태7/13/201718628디버깅 기술: 88. windbg 분석 - webengine4.dll의 MgdExplicitFlush에서 발생한 System.AccessViolationException의 crash 문제 (2)
11250정성태7/13/201722217디버깅 기술: 87. windbg 분석 - webengine4.dll의 MgdExplicitFlush에서 발생한 System.AccessViolationException의 crash 문제 [1]
11249정성태7/12/201719972오류 유형: 410. LoadLibrary("[...].dll") failed - The specified procedure could not be found.
11248정성태7/12/201726520오류 유형: 409. pip install pefile - 'cp949' codec can't decode byte 0xe2 in position 208687: illegal multibyte sequence
11247정성태7/12/201720854오류 유형: 408. SqlConnection 객체 생성 시 무한 대기 문제파일 다운로드1
11246정성태7/11/201718886VS.NET IDE: 118. Visual Studio - 다중 폴더에 포함된 파일들에 대한 "Copy to Output Directory"를 한 번에 설정하는 방법
11245정성태7/10/201724629개발 환경 구성: 321. Visual Studio Emulator for Android 소개 [2]
11244정성태7/10/201724814오류 유형: 407. Visual Studio에서 ASP.NET Core 실행할 때 dotnet.exe 프로세스의 -532462766 오류 발생 [1]
11243정성태7/10/201721593.NET Framework: 666. dotnet.exe - 윈도우 운영체제에서의 .NET Core 버전 찾기 규칙
11242정성태7/8/201721132제니퍼 .NET: 27. 제니퍼 닷넷 적용 사례 (7) - 노후된 스토리지 장비로 인한 웹 서비스 Hang (멈춤) 현상
11241정성태7/8/201719777오류 유형: 406. Xamarin 빌드 에러 XA5209, APT0000
11240정성태7/7/201723597.NET Framework: 665. ClickOnce를 웹 브라우저를 이용하지 않고 쿼리 문자열을 전달하면서 실행하는 방법 [3]파일 다운로드1
11239정성태7/6/201724252.NET Framework: 664. Protocol Handler - 웹 브라우저에서 데스크톱 응용 프로그램을 실행하는 방법 [5]파일 다운로드1
11238정성태7/6/201721780오류 유형: 405. NT 서비스 시작 시 "Error 1067: The process terminated unexpectedly." 오류 발생 [2]
11237정성태7/5/201723435.NET Framework: 663. C# - PDB 파일 경로를 PE 파일로부터 얻는 방법파일 다운로드1
11236정성태7/4/201727167.NET Framework: 662. C# - VHD/VHDX 가상 디스크를 마운트하지 않고 파일을 복사하는 방법파일 다운로드1
11235정성태6/29/201721404Math: 20. Matlab/Octave로 Gram-Schmidt 정규 직교 집합 구하는 방법
11234정성태6/29/201718889오류 유형: 404. SharePoint 2013 설치 과정에서 "The username is invalid The account must be a valid domain account" 오류 발생
11233정성태6/28/201718782오류 유형: 403. SharePoint Server 2013을 Windows Server 2016에 설치할 때 .NET 4.5 설치 오류 발생
... 106  107  [108]  109  110  111  112  113  114  115  116  117  118  119  120  ...