Microsoft MVP성태의 닷넷 이야기
.NET Framework: 133. CallbackOnCollectedDelegate was detected [링크 복사], [링크+제목 복사],
조회: 28633
글쓴 사람
정성태 (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분
이런 자문자답 덧글 너무 좋습니다. ^^
정성태

... 16  17  18  19  20  21  22  [23]  24  25  26  27  28  29  30  ...
NoWriterDateCnt.TitleFile(s)
13363정성태6/7/202312033스크립트: 49. 파이썬 - "Transformers (신경망 언어모델 라이브러리) 강좌" - 1장 2절 코드 실행 결과
13362정성태6/1/202311318.NET Framework: 2126. C# - 서버 측의 요청 제어 (Microsoft.AspNetCore.RateLimiting)파일 다운로드1
13361정성태5/31/202311894오류 유형: 862. Facebook - ASP.NET/WebClient 사용 시 graph.facebook.com/me 호출에 대해 403 Forbidden 오류
13360정성태5/31/202310880오류 유형: 861. WSL/docker - failed to start shim: start failed: io.containerd.runc.v2: create new shim socket
13359정성태5/19/202311430오류 유형: 860. Docker Desktop - k8s 초기화 무한 반복한다면?
13358정성태5/17/202311832.NET Framework: 2125. C# - Semantic Kernel의 Semantic Memory 사용 예제 [1]파일 다운로드1
13357정성태5/16/202311251.NET Framework: 2124. C# - Semantic Kernel의 Planner 사용 예제파일 다운로드1
13356정성태5/15/202312458DDK: 10. Device Driver 테스트 설치 관련 오류 (Code 37, Code 31) 및 인증서 관련 정리
13355정성태5/12/202311624.NET Framework: 2123. C# - Semantic Kernel의 ChatGPT 대화 구현 [1]파일 다운로드1
13354정성태5/12/202312776.NET Framework: 2122. C# - "Use Unicode UTF-8 for worldwide language support" 설정을 한 경우, 한글 입력이 '\0' 문자로 처리
13352정성태5/12/202312066.NET Framework: 2121. C# - Semantic Kernel의 대화 문맥 유지파일 다운로드1
13351정성태5/11/202312589VS.NET IDE: 185. Visual Studio - 원격 Docker container 내에 실행 중인 응용 프로그램에 대한 디버깅 [1]
13350정성태5/11/202311680오류 유형: 859. Windows Date and Time - Unable to continue. You do not have permission to perform this task
13349정성태5/11/202312104.NET Framework: 2120. C# - Semantic Kernel의 Skill과 Function 사용 예제 [1]파일 다운로드1
13348정성태5/10/202312814.NET Framework: 2119. C# - Semantic Kernel의 "Basic Loading of the Kernel" 예제
13347정성태5/10/202313241.NET Framework: 2118. C# - Semantic Kernel의 Prompt chaining 예제파일 다운로드1
13346정성태5/10/202312571오류 유형: 858. RDP 원격 환경과 로컬 PC 간의 Ctrl+C, Ctrl+V 복사가 안 되는 문제
13345정성태5/9/202314789.NET Framework: 2117. C# - (OpenAI 기반의) Microsoft Semantic Kernel을 이용한 자연어 처리 [1]파일 다운로드1
13344정성태5/9/202315497.NET Framework: 2116. C# - OpenAI API 사용 - 지원 모델 목록 [1]파일 다운로드1
13343정성태5/9/202313190디버깅 기술: 192. Windbg - Hyper-V VM으로 이더넷 원격 디버깅 연결하는 방법
13342정성태5/8/202311932.NET Framework: 2115. System.Text.Json의 역직렬화 시 필드/속성 주의
13341정성태5/8/202311753닷넷: 2114. C# 12 - 모든 형식의 별칭(Using aliases for any type)
13340정성태5/8/202312069오류 유형: 857. Microsoft.Data.SqlClient.SqlException - 0x80131904
13339정성태5/6/202313276닷넷: 2113. C# 12 - 기본 생성자(Primary Constructors)
13338정성태5/6/202311823닷넷: 2112. C# 12 - 기본 람다 매개 변수파일 다운로드1
13337정성태5/5/202312858Linux: 59. dockerfile - docker exec로 container에 접속 시 자동으로 실행되는 코드 적용
... 16  17  18  19  20  21  22  [23]  24  25  26  27  28  29  30  ...