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

[1]  2  3  4  5  6  7  8  9  10  11  12  13  14  15  ...
NoWriterDateCnt.TitleFile(s)
13901정성태3/9/20251228Windows: 280. Hyper-V의 3가지 Thread Scheduler (Classic, Core, Root)
13900정성태3/8/20251308스크립트: 72. 파이썬 - SQLAlchemy + oracledb 연동
13899정성태3/7/20251266스크립트: 71. 파이썬 - asyncio의 ContextVar 전달
13898정성태3/5/20251266오류 유형: 948. Visual Studio - Proxy Authentication Required: dotnetfeed.blob.core.windows.net
13897정성태3/5/20251349닷넷: 2326. C# - PowerShell과 연동하는 방법 (두 번째 이야기)파일 다운로드1
13896정성태3/5/20251378Windows: 279. Hyper-V Manager - VM 목록의 CPU Usage 항목이 항상 0%로 나오는 문제
13895정성태3/4/20251431Linux: 117. eBPF / bpf2go - Map에 추가된 요소의 개수를 확인하는 방법
13894정성태2/28/20251508Linux: 116. eBPF / bpf2go - BTF Style Maps 정의 구문과 데이터 정렬 문제
13893정성태2/27/20251559Linux: 115. eBPF (bpf2go) - ARRAY / HASH map 기본 사용법
13892정성태2/24/20251689닷넷: 2325. C# - PowerShell과 연동하는 방법파일 다운로드1
13891정성태2/23/20251596닷넷: 2324. C# - 프로세스의 성능 카운터용 인스턴스 이름을 구하는 방법파일 다운로드1
13890정성태2/21/20251586닷넷: 2323. C# - 프로세스 메모리 중 Private Working Set 크기를 구하는 방법(Win32 API)파일 다운로드1
13889정성태2/20/20251773닷넷: 2322. C# - 프로세스 메모리 중 Private Working Set 크기를 구하는 방법(성능 카운터, WMI) [1]파일 다운로드1
13888정성태2/17/20251590닷넷: 2321. Blazor에서 발생할 수 있는 async void 메서드의 부작용
13887정성태2/17/20251568닷넷: 2320. Blazor의 razor 페이지에서 code-behind 파일로 코드를 분리하는 방법
13886정성태2/15/20251800VS.NET IDE: 196. Visual Studio - Code-behind처럼 cs 파일을 그룹핑하는 방법
13885정성태2/14/20251822닷넷: 2319. ASP.NET Core Web API / Razor 페이지에서 발생할 수 있는 async void 메서드의 부작용
13884정성태2/13/20252044닷넷: 2318. C# - (async Task가 아닌) async void 사용 시의 부작용파일 다운로드1
13883정성태2/12/20252008닷넷: 2317. C# - Memory Mapped I/O를 이용한 PCI Configuration Space 정보 열람파일 다운로드1
13882정성태2/10/20252101스크립트: 70. 파이썬 - oracledb 패키지 연동 시 Thin / Thick 모드
13881정성태2/7/20252204닷넷: 2316. C# - Port I/O를 이용한 PCI Configuration Space 정보 열람파일 다운로드1
13880정성태2/5/20252051오류 유형: 947. sshd - Failed to start OpenSSH server daemon.
13879정성태2/5/20252348오류 유형: 946. Ubuntu - N: Updating from such a repository can't be done securely, and is therefore disabled by default.
13878정성태2/3/20252152오류 유형: 945. Windows - 최대 절전 모드 시 DRIVER_POWER_STATE_FAILURE 발생 (pacer.sys)
13877정성태1/25/20252341닷넷: 2315. C# - PCI 장치 열거 (레지스트리, SetupAPI)파일 다운로드1
[1]  2  3  4  5  6  7  8  9  10  11  12  13  14  15  ...