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

비밀번호

댓글 작성자
 




... 106  107  108  109  110  111  112  113  114  115  [116]  117  118  119  120  ...
NoWriterDateCnt.TitleFile(s)
11025정성태8/12/201622351개발 환경 구성: 294. .NET Core 프로젝트에서 "Copy to Output Directory" 처리 [1]
11024정성태8/12/201621658오류 유형: 350. "nProtect GameMon" 실행 중에는 Visual Studio 디버깅이 안됩니다! [1]
11023정성태8/10/201623177개발 환경 구성: 293. Azure 구독 후 PaaS 서비스 만들어 보기
11022정성태8/10/201623835개발 환경 구성: 292. Azure Cloud Service 배포시 사용자 정의 작업을 추가하는 방법
11021정성태8/10/201620882오류 유형: 349. System.Runtime.Remoting.RemotingException - Type '..., ..., Version=..., Culture=neutral, PublicKeyToken=null' is not registered for activation [2]
11020정성태8/10/201623613VC++: 98. 원본과 대상 버퍼가 같은 경우 memcpy, wmemcpy 주의점
11019정성태8/10/201640288기타: 60. 도서: 시작하세요! C# 6.0 프로그래밍: 기본 문법부터 실전 예제까지 (2쇄 정오표)
11018정성태8/9/201624751.NET Framework: 600. 단일 메서드 내에서의 할당으로 알아보는 자바와 닷넷의 GC 차이점 [1]
11017정성태8/9/201626807웹: 33. HTTP 쿠키에 한글 값을 설정하는 방법
11016정성태8/7/201624016개발 환경 구성: 291. Windows Server Containers 소개
11015정성태8/7/201622271오류 유형: 348. Windows Server 2016 TP5에서 Windows Containers의 docker run 실행 시 encountered an error during Start failed in Win32
11014정성태8/6/201623058오류 유형: 347. Hyper-V Virtual Machine Management service Account does not have permission to open attachment
11013정성태8/6/201633839개발 환경 구성: 290. Windows 10에서 경험해 보는 Windows Containers와 docker [4]
11012정성태8/6/201623896오류 유형: 346. Windows 10에서 Windows Containers의 docker run 실행 시 encountered an error during CreateContainer failed in Win32 발생
11011정성태8/6/201625518기타: 59. outlook.live.com 메일 서비스의 아웃룩 POP3 설정하는 방법
11010정성태8/6/201622883기타: 58. Outlook에 설정한 SMTP/POP3(예:천리안 메일) 계정 암호를 잊어버린 경우
11009정성태8/3/201628076개발 환경 구성: 289. 2016-08-02부터 시작된 윈도우 10 1주년 업데이트에서 Bash Shell 사용 [8]
11008정성태8/1/201621896오류 유형: 345. 2의 30승 이상의 원소를 갖는 경우 버그가 발생하는 이진 검색(Binary Search) 코드
11007정성태8/1/201623602오류 유형: 344. RDP ActiveX 컨트롤로 특정 PC에 연결할 수 없을 때, 오류 상황을 해결하기 위한 팁파일 다운로드1
11006정성태7/22/201626590개발 환경 구성: 288. SSL 인증서를 Azure Cloud Service에 적용하는 방법
11005정성태7/22/201625232개발 환경 구성: 287. Let's Encrypt 인증서 업데이트 주기: 90일
11004정성태7/22/201620084오류 유형: 343. Invalid service definition or service configuration. Please see the Error List for more details.
11003정성태7/20/201627365VS.NET IDE: 110. Visual Studio 2015에서 .NET Core 응용 프로그램 개발 [1]
11002정성태7/20/201620839개발 환경 구성: 286. Microsoft Azure 서비스의 구독은 반드시 IE로!
11001정성태7/19/201631908.NET Framework: 599. .NET Core/SDK 설치 및 기본 사용법 [6]
11000정성태7/16/201620616오류 유형: 342. Microsoft Visual Studio 2010 Tools for Office Runtime (x86 and x64) 설치 시 오류
... 106  107  108  109  110  111  112  113  114  115  [116]  117  118  119  120  ...