Microsoft MVP성태의 닷넷 이야기
글쓴 사람
정성태 (techsharer at outlook.com)
홈페이지
첨부 파일
(연관된 글이 1개 있습니다.)
(시리즈 글이 2개 있습니다.)
.NET Framework: 400. 눈으로 확인하는 LayoutKind 옵션 효과
; https://www.sysnet.pe.kr/2/0/1558

닷넷: 2250. PInvoke 호출 시 참조 타입(class)을 마샬링하는 [IN], [OUT] 특성
; https://www.sysnet.pe.kr/2/0/13609




PInvoke 호출 시 참조 타입(class)을 마샬링하는 [IN], [OUT] 특성

예를 하나 들어볼까요? ^^

우선, 비교를 위해 값 형식인 struct 타입을 P/Invoke로 호출해 보겠습니다. 이를 위해 다음과 같이 API 하나를 노출하는 DLL을 만들고,

// C++ 헤더

struct Store
{
public:
    __int64 value1;
};

extern "C"
{
    __declspec(dllexport) void GetStore(Store* pValue);
};

// C++ cpp

__declspec(dllexport) void GetStore(Store* pValue)
{
    printf("Called: %I64d\n", pValue->value1);
    pValue->value1 = 50;
}

C# 측에서 다음과 같이 호출하면,

using System.Runtime.InteropServices;

[StructLayout(LayoutKind.Sequential)]
public struct StoreStruct
{
    public long v1;
    public StoreStruct(long value) => v1 = value;
}

internal class Program
{
    [DllImport("Dll1.dll")]
    static extern void GetStore(out StoreStruct value);

    static void Main(string[] args)
    {
        StoreStruct value = new StoreStruct(10);

        GetStore(out value);
        Debug.Assert(value.v1 == 50);
    }
}

/* 실행 시 출력 결과:
Called: 10
*/

C/C++ 측의 GetStoreStruct 함수도 잘 호출이 되었고, 거기서 반환한 v1의 값도 C# 측에서 50으로 잘 받았습니다. 여기서 재미있는 것은, C#의 out 예약어를 사용했지만 그와 무관하게 C/C++ 측의 API에서는 마치 ref와 동일하게 작용했다는 점입니다. 사실 ref와 out은 C# 언어의 구분일 뿐 내부적으로는 2개 모두 ref와 동일하게 처리되기 때문입니다.




그런데, 저 예제를 class로 바꿔서, 즉 struct StoreStruct를 class StoreClass로 테스트하면 어떻게 될까요? 이를 위해 C# 측에서는 다음과 같은 코드를 추가한 다음,

[StructLayout(LayoutKind.Sequential)]
public class StoreClass
{
    public long v1;
    public StoreClass(long value) => v1 = value;
}

internal class Program
{
    [DllImport("Dll1.dll", EntryPoint = "GetStore")]
    static extern void GetStoreClass(ref StoreClass value);

    static void Main(string[] args)
    {
        StoreClass value = new StoreClass(20);

        GetStoreClass(ref value); // 예외 발생
    }
}

/* 실행 결과: C/C++ 측의 printf 출력과 함께,
Called: 1809131705472

C# 측으로 넘어오는 시점에 예외 발생
System.NullReferenceException
  HResult=0x80004003
  Message=Object reference not set to an instance of an object.
  Source=<Cannot evaluate the exception source>
  StackTrace:
<Cannot evaluate the exception stack trace>
*/

실행해 보면, 일단 C/C++ 함수까지는 코드가 실행된 것임을 printf 출력으로 알 수 있지만, 예외는 C# 측으로 제어를 반환하면서 발생합니다. 추적을 해보면, C/C++ 측에서 "Store *pValue"로 받은 값의 첫 번째 필드인 1809131705472 값을 포인터라고 가정하고 hex 주소로 따지만 0x1a538a71480가 됩니다. 위의 경우 디버거에서 메모리 창을 이용해 그 주소의 값을 보면 20이 들어있는 것을 확인할 수 있습니다.

따라서, 오히려 C/C++ 측에서 다음과 같이 값을 받은 거나 마찬가지입니다.

__declspec(dllexport) void GetStore(Store value)
{
    printf("Called: %I64d(%I64x)\n", value.value1);
}

이유가 뭘까요? ^^




사실, C#의 out/ref 인자는 마샬링에 어떠한 영향도 주지 않습니다. 단지 그것은 C#의 문법대로 변수의 값, 또는 참조 주소를 넘기는 역할만 하며 거기에 struct나 class의 차별은 없습니다.

하지만, P/Invoke에 의한 Managed <- -> Native 간의 데이터 이동은 struct 값 형식은 상관없지만 class 참조 형식의 경우에는 반드시 닷넷 런타임이 관여를 하게 됩니다. 따라서, 애당초 참조 형식이라면 C# 측에서 out이나 ref가 아닌, 대상 API에 맞춰 마샬링 방법을 지정해야 합니다.

아래의 문서가 바로 그런 내용을 설명합니다. ^^

Marshalling Classes, Structures, and Unions
; https://learn.microsoft.com/en-us/dotnet/framework/interop/marshalling-classes-structures-and-unions#systime-sample

The parameter must be declared with the InAttribute and OutAttribute attributes because classes, which are reference types, are passed as In parameters by default.


기본 동작이 class 참조 형식에 대해 "InAttribute" 역할로 전달된다고 하는데요, 이전 예제 코드에서 왜 "__declspec(dllexport) void GetStore(Store value)"처럼 값이 전달된 것인지 설명이 됩니다.

그러니까, class 참조 형식은 InAttribute, OutAttribute 특성을 지정해 닷넷 런타임 측에 마샬링을 제어할 수 있는 힌트를 줘야 하는 것입니다. 따라서, 위에서 오류가 발생했던 코드를 올바르게 동작하게 하려면 다음과 같이 호출해 주면 됩니다.

[DllImport("Dll1.dll", EntryPoint = "GetStore")]
static extern void GetStoreClass([In, Out] StoreClass value);

StoreClass value = new StoreClass(20);

GetStoreClass(value);
Debug.Assert(value.v1 == 50);




그런데, 사실 위의 구조체 같은 경우에는 굳이 [In, Out]을 지정하지 않아도 잘 동작합니다.

[DllImport("Dll1.dll", EntryPoint = "GetStore")]
static extern void GetStoreClass(StoreClass value);

StoreClass value = new StoreClass(20);

GetStoreClass(value);
Debug.Assert(value.v1 == 50);

왜 그런 걸까요? ^^ 이유는 간단합니다. 닷넷 런타임이 해당 타입은 별도의 전/후처리를 할 필요가 없다고 판단하므로 그냥 value 인스턴스 자체의 주소를 C/C++ 측에 넘기기 때문입니다. 실제로 포인터 값을 출력해 확인할 수 있습니다.

// C# 측 코드

{
    StoreClass value = new StoreClass(20);

    fixed (long* ptr = &value.v1)
    {
        Console.WriteLine($"ptr: 0x{new IntPtr(ptr):X}");
        GetStoreClass(value);
        Debug.Assert(value.v1 == 50);
    }
}

// C/C++ 측 코드

__declspec(dllexport) void GetStore(Store* pValue)
{
    printf("Called: %I64d - ptr == 0x%p\n", pValue->value1, pValue);
    pValue->value1 = 50;
}

실행해 보면 이런 출력이 나옵니다.

ptr: 0x1E402415640
Called: 20(14) - ptr == 0x000001E402415640

동일하죠? ^^ 위와 같은 이유 덕분에 [In, Out]을 지정하지 않아도 상관없지만, 언제나 그런 것은 아닙니다. 즉, 닷넷 런타임이 마샬링에 관여하게 되는 경우에 문제가 발생합니다. 이를 재현하기 위해선, 마샬링이 관여하도록 일부러 MarshalAs를 사용하는 타입을 하나 넣으면 됩니다.

[StructLayout(LayoutKind.Sequential)]
public class OrderTestClass
{
    public int i;
    [MarshalAs(UnmanagedType.LPWStr, SizeConst = 10)]
    public string s;
}

위의 타입을 C/C++에서는 이렇게 사용하도록 코드를 추가하고,

// C/C++ 헤더

struct OrderTest
{
public:
    int i;
    wchar_t* string;
};

extern "C"
{
    __declspec(dllexport) void GetOrderTest(OrderTest* pValue);
};

// C/C++ 측 코드

__declspec(dllexport) void GetOrderTest(OrderTest* pValue)
{
    printf("Called: %d, %ls - ptr == 0x%p\n", pValue->i, pValue->string, pValue);

    wcscpy_s(pValue->string, 10, L"GOOD");
    pValue->i = 70;
}

C#에서 호출하면,

[DllImport("Dll1.dll")]
static extern void GetOrderTest([In, Out] OrderTestClass value);

OrderTestClass value = new OrderTestClass();
value.i = 5;
value.s = "TEST";

fixed (int* ptr = &value.i)
{
    Console.WriteLine($"[GetOrderTest] ptr: 0x{new IntPtr(ptr):X}");
    GetOrderTest(value);
    Debug.Assert(value.i == 70);
    Debug.Assert(value.s == "GOOD");
}

이번에는 출력 결과에서 포인터 값이 다르게 나옵니다.

ptr: 0x15193015700
Called: 5, TEST - ptr == 0x0000003FF837E150

이유는 간단합니다. C#은 value 인스턴스를 넘겼지만, 닷넷 런타임은 int 4바이트 + sizeof(wchar_t) * 10개의 연속된 메모리 공간을 별도로 할당하고 거기에 ([In] 특성을 지정했으므로) 값을 채운 다음에 그 포인터를 C/C++ 측에 넘깁니다.

그리고 C/C++ 측에서는 해당 포인터에 값을 채운 다음 제어를 닷넷 런타임으로 반환하면, ([Out] 특성을 지정했으므로) 별도로 할당된 공간에 채워졌던 그 값을 다시 C#의 value 인스턴스에 설정해 주면서 P/Invoke API 호출을 마칩니다.

따라서, 위와 같은 경우에는 [In], [Out] 특성을 모두 지정해야만 C/C++ 측의 GetOrderTest 함수가 정상적으로 동작합니다. 여기서 만약,

  1. [In]을 뺀다면, 닷넷 런타임은 새로 할당한 공간에 C# 측의 value 인스턴스 값을 채우지 않고, 즉 0으로 초기화된 공간에 대한 포인터를 C/C++에 전달하므로 "OrderTest *pValue"의 i, string 값은 모두 0, null로 채워집니다.
  2. [Out]을 뺀다면, 닷넷 런타임은 [In]에 따라 값은 채워서 보내지만, C/C++ API 호출이 완료된 후 C# 측으로 값 복사를 해주지 않아 value 인스턴스의 값은 변하지 않게 됩니다.

이 정도면... 대충 이해가 되셨죠? ^^

(첨부 파일은 이 글의 예제 코드를 포함합니다.)




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

[연관 글]






[최초 등록일: ]
[최종 수정일: 4/27/2024]

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

비밀번호

댓글 작성자
 




... 121  122  123  124  125  126  127  128  129  130  131  132  133  [134]  135  ...
NoWriterDateCnt.TitleFile(s)
1705정성태7/2/201435068VC++: 78. 보이어-무어(Boyer-Moore) 알고리즘이 정말 빠를까? [6]파일 다운로드1
1704정성태7/2/201421663.NET Framework: 447. w3wp.exe AppPool 재생(recycle)하는 방법 정리
1703정성태7/2/201422506.NET Framework: 446. Assembly.Load를 이용해 GAC에 등록된 어셈블리를 로드하는 방법 [1]파일 다운로드1
1702정성태6/23/201422287Phone: 11. Xamarin.Forms - 2. XAML을 이용한 페이지 개발파일 다운로드1
1701정성태6/23/201434454개발 환경 구성: 229. .NET Reflector + Reflexil 도구를 이용해 DLL 코드 변경 [4]
1700정성태6/23/201421301VS.NET IDE: 89. Visual Studio에서 기본 제공되는 성능 프로파일 [2]
1699정성태6/22/201424097Phone: 10. Xamarin.Forms - 1. Forms 시작하기 [2]파일 다운로드1
1698정성태6/22/201426067.NET Framework: 445. [부연 설명] 쉬운 C# 코드를 어럽게 이해하기 [2]
1697정성태6/22/201421365VS.NET IDE: 88. Visual Studio에서 직접 컴파일하는 IL 언어 확장 도구 - IL Support
1696정성태6/22/201421164.NET Framework: 444. clojure와 C#을 통해 이해하는 Sequence와 Vector 형식의 차이점 [1]
1695정성태6/21/201420194개발 환경 구성: 228. PowerShell ISE에서 (입력 기능이 있는) 콘솔 응용 프로그램을 시작하는 방법
1694정성태6/21/201421341개발 환경 구성: 227. 닷넷 용 ClojureCLR 개발환경 설정
1693정성태6/20/201421660개발 환경 구성: 226. Clojure 언어의 윈도우 개발환경 설정
1692정성태6/19/201432248오류 유형: 231. Visual Studio 2013 한글 버전 설치 오류 - The form specified for the subject is not one supported or known by the specified trust provider
1691정성태6/18/201427461개발 환경 구성: 225. 유닉스 계열의 tail 명령어가 제공되는 PowerShell [1]
1690정성태6/18/201430253개발 환경 구성: 224. DirectShow 예제 구하는 방법 [3]
1689정성태6/18/201427098오류 유형: 230. C++ 가변 인자 사용시 va_start 파라미터 전달 방법 [2]
1688정성태6/15/201420650오류 유형: 229. 갤럭시 노트 3 환경에서 Xamarin 앱 배포 충돌
1687정성태6/15/201426655개발 환경 구성: 223. PowerShell로 Visual Studio 빌드 스크립트 작성파일 다운로드1
1686정성태6/12/201424362Windows: 96. 윈도우 8 - 그림 암호를 이용해 로그인 시 지연 현상을 해결하는 방법 [1]
1685정성태6/10/201431119.NET Framework: 443. 자바 8과 C#의 람다(Lambda) 지원에 대한 비교 [12]
1684정성태6/9/201441289.NET Framework: 442. C# - 시스템의 CPU 사용량 및 프로세스(EXE)의 CPU 사용량 알아내는 방법 [5]파일 다운로드1
1683정성태6/2/201420855오류 유형: 228. CLR4 보안 - yield 구문 내에서 SecurityCritical 메서드 사용 불가 [2]파일 다운로드1
1682정성태6/1/201426068.NET Framework: 441. .NET CLR4 보안 모델 - 3. CLR4 보안 모델에서의 APTCA 역할파일 다운로드2
1681정성태6/1/201421935.NET Framework: 440. .NET CLR4 보안 모델 - 2. 샌드박스(Sandbox)을 이용한 보안 [2]파일 다운로드1
1680정성태6/1/201421415.NET Framework: 439. .NET CLR4 보안 모델 - 1. "Security Level 2"란?파일 다운로드1
... 121  122  123  124  125  126  127  128  129  130  131  132  133  [134]  135  ...