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

비밀번호

댓글 작성자
 




... 46  47  [48]  49  50  51  52  53  54  55  56  57  58  59  60  ...
NoWriterDateCnt.TitleFile(s)
12737정성태7/28/202114982오류 유형: 744. Azure Active Directory - The resource principal named api://...[client_id]... was not found in the tenant
12736정성태7/28/202115253오류 유형: 743. Active Azure Directory에서 "API permissions"의 권한 설정이 "Not granted for ..."로 나오는 문제
12735정성태7/27/202115160.NET Framework: 1081. C# - Azure AD 인증을 지원하는 데스크톱 애플리케이션 예제(Windows Forms) [2]파일 다운로드1
12734정성태7/26/202131966스크립트: 20. 특정 단어로 시작하거나/끝나는 문자열을 포함/제외하는 정규 표현식 - Look-around
12733정성태7/23/202119844.NET Framework: 1081. Self-Contained/SingleFile 유형의 .NET Core/5+ 실행 파일을 임베딩한다면? [1]파일 다운로드2
12732정성태7/23/202113297오류 유형: 742. SharePoint - The super user account utilized by the cache is not configured.
12731정성태7/23/202115320개발 환경 구성: 584. Add Internal URLs 화면에서 "Save" 버튼이 비활성화 된 경우
12730정성태7/23/202116765개발 환경 구성: 583. Visual Studio Code - Go 코드에서 입력을 받는 경우
12729정성태7/22/202115252.NET Framework: 1080. xUnit 단위 테스트에 메서드/클래스 수준의 문맥 제공 - Fixture
12728정성태7/22/202115275.NET Framework: 1079. MSTestv2 단위 테스트에 메서드/클래스/어셈블리 수준의 문맥 제공
12727정성태7/21/202116459.NET Framework: 1078. C# 단위 테스트 - MSTestv2/NUnit의 Assert.Inconclusive 사용법(?) [1]
12726정성태7/21/202116233VS.NET IDE: 169. 비주얼 스튜디오 - 단위 테스트 선택 시 MSTestv2 외의 xUnit, NUnit 사용법 [1]
12725정성태7/21/202114694오류 유형: 741. Failed to find the "go" binary in either GOROOT() or PATH
12724정성태7/21/202117588개발 환경 구성: 582. 윈도우 환경에서 Visual Studio Code + Go (Zip) 개발 환경 [1]
12723정성태7/21/202114512오류 유형: 740. SharePoint - Alternate access mappings have not been configured 경고
12722정성태7/20/202114258오류 유형: 739. MSVCR110.dll이 없어 exe 실행이 안 되는 경우
12721정성태7/20/202115412오류 유형: 738. The trust relationship between this workstation and the primary domain failed. - 세 번째 이야기
12720정성태7/19/202114677Linux: 43. .NET Core/5+ 응용 프로그램의 Ubuntu (Debian) 패키지 준비
12719정성태7/19/202113870오류 유형: 737. SharePoint 설치 시 "0x800710D8 The object identifier does not represent a valid object." 오류 발생
12718정성태7/19/202113899개발 환경 구성: 581. Windows에서 WSL로 파일 복사 시 root 소유권으로 적용되는 문제파일 다운로드1
12717정성태7/18/202114095Windows: 195. robocopy에서 파일의 ADS(Alternate Data Stream) 정보 복사를 제외하는 방법
12716정성태7/17/202115231개발 환경 구성: 580. msbuild의 Exec Task에 robocopy를 사용하는 방법파일 다운로드1
12715정성태7/17/202122430오류 유형: 736. Windows - MySQL zip 파일 버전의 "mysqld --skip-grant-tables" 실행 시 비정상 종료 [1]
12714정성태7/16/202115795오류 유형: 735. VCRUNTIME140.dll, MSVCP140.dll, VCRUNTIME140.dll, VCRUNTIME140_1.dll이 없어 exe 실행이 안 되는 경우
12713정성태7/16/202117183.NET Framework: 1077. C# - 동기 방식이면서 비동기 규약을 따르게 만드는 Task.FromResult파일 다운로드1
12712정성태7/15/202116106개발 환경 구성: 579. Azure - 리눅스 호스팅의 Site Extension 제작 방법
... 46  47  [48]  49  50  51  52  53  54  55  56  57  58  59  60  ...