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

... 46  [47]  48  49  50  51  52  53  54  55  56  57  58  59  60  ...
NoWriterDateCnt.TitleFile(s)
12762정성태8/8/202119396Java: 28. IntelliJ - Unable to open debugger port 오류
12761정성태8/8/202116069Java: 27. IntelliJ - java: package javax.inject does not exist [2]
12760정성태8/8/202112848개발 환경 구성: 594. 전용 "Command Prompt for ..." 단축 아이콘 만들기
12759정성태8/8/202117463Java: 26. IntelliJ + Spring Framework + 새로운 Controller 추가 [2]파일 다운로드1
12758정성태8/7/202116865오류 유형: 751. Error assembling WAR: webxml attribute is required (or pre-existing WEB-INF/web.xml if executing in update mode)
12757정성태8/7/202117474Java: 25. IntelliJ + Spring Framework 프로젝트 생성
12756정성태8/6/202115742.NET Framework: 1084. C# - .NET Core Web API 단위 테스트 방법 [1]파일 다운로드1
12755정성태8/5/202115776개발 환경 구성: 593. MSTest - 단위 테스트에 static/instance 유형의 private 멤버 접근 방법파일 다운로드1
12754정성태8/5/202116174오류 유형: 750. manage.py - Your project may not work properly until you apply the migrations for app(s): admin, auth, contenttypes, sessions.
12753정성태8/5/202117103오류 유형: 749. PyCharm - Error: Django is not importable in this environment
12752정성태8/4/202113963개발 환경 구성: 592. JetBrains의 IDE(예를 들어, PyCharm)에서 Visual Studio 키보드 매핑 적용
12751정성태8/4/202116886개발 환경 구성: 591. Windows 10 WSL2 환경에서 docker-compose 빌드하는 방법
12750정성태8/3/202113916디버깅 기술: 181. windbg - 콜 스택의 "Call Site" 오프셋 값이 가리키는 위치
12749정성태8/2/202113347개발 환경 구성: 590. Visual Studio 2017부터 단위 테스트에 DataRow 특성 지원
12748정성태8/2/202114371개발 환경 구성: 589. Azure Active Directory - tenant의 관리자(admin) 계정 로그인 방법
12747정성태8/1/202114642오류 유형: 748. 오류 기록 - MICROSOFT GRAPH – HOW TO IMPLEMENT IAUTHENTICATIONPROVIDER파일 다운로드1
12746정성태7/31/202119098개발 환경 구성: 588. 네트워크 장비 환경을 시뮬레이션하는 Packet Tracer 프로그램 소개
12745정성태7/31/202114905개발 환경 구성: 587. Azure Active Directory - tenant의 관리자 계정 로그인 방법
12744정성태7/30/202115293개발 환경 구성: 586. Azure Active Directory에 연결된 App 목록을 확인하는 방법?
12743정성태7/30/202116527.NET Framework: 1083. Azure Active Directory - 외부 Token Cache 저장소를 사용하는 방법파일 다운로드1
12742정성태7/30/202114554개발 환경 구성: 585. Azure AD 인증을 위한 사용자 인증 유형
12741정성태7/29/202116046.NET Framework: 1082. Azure Active Directory - Microsoft Graph API 호출 방법파일 다운로드1
12740정성태7/29/202114549오류 유형: 747. SharePoint - InvalidOperationException 0x80131509
12739정성태7/28/202114973오류 유형: 746. Azure Active Directory - IDW10106: The 'ClientId' option must be provided.
12738정성태7/28/202115923오류 유형: 745. Azure Active Directory - Client credential flows must have a scope value with /.default suffixed to the resource identifier (application ID URI).
12737정성태7/28/202115011오류 유형: 744. Azure Active Directory - The resource principal named api://...[client_id]... was not found in the tenant
... 46  [47]  48  49  50  51  52  53  54  55  56  57  58  59  60  ...