Microsoft MVP성태의 닷넷 이야기
글쓴 사람
정성태 (techsharer at outlook.com)
홈페이지
첨부 파일
 
(연관된 글이 1개 있습니다.)
(시리즈 글이 13개 있습니다.)
.NET Framework: 397. C# - OCX 컨트롤에 구현된 메서드에 배열을 in, out으로 전달하는 방법
; https://www.sysnet.pe.kr/2/0/1547

.NET Framework: 652. C# 개발자를 위한 C++ COM 객체의 기본 구현 방식 설명
; https://www.sysnet.pe.kr/2/0/11175

.NET Framework: 792. C# COM 서버가 제공하는 COM 이벤트를 C++에서 받는 방법
; https://www.sysnet.pe.kr/2/0/11679

.NET Framework: 907. C# DLL로부터 TLB 및 C/C++ 헤더 파일(TLH)을 생성하는 방법
; https://www.sysnet.pe.kr/2/0/12220

.NET Framework: 977. C# PInvoke - C++의 매개변수에 대한 마샬링을 tlbexp.exe를 이용해 확인하는 방법
; https://www.sysnet.pe.kr/2/0/12443

.NET Framework: 1008. 배열을 반환하는 C# COM 개체의 메서드를 C++에서 사용 시 메모리 누수 현상
; https://www.sysnet.pe.kr/2/0/12491

.NET Framework: 1064. C# COM 개체를 PIA(Primary Interop Assembly)로써 "Embed Interop Types" 참조하는 방법
; https://www.sysnet.pe.kr/2/0/12662

.NET Framework: 1069. C# - DLL Surrogate를 이용한 Out-of-process COM 개체 제작
; https://www.sysnet.pe.kr/2/0/12668

.NET Framework: 1095. C# COM 개체를 C++에서 사용하는 예제
; https://www.sysnet.pe.kr/2/0/12791

.NET Framework: 2003. C# - COM 개체의 이벤트 핸들러에서 발생하는 예외에 대한 CLR의 특별 대우
; https://www.sysnet.pe.kr/2/0/13050

닷넷: 2177. C# - (Interop DLL 없이) CoClass를 이용한 COM 개체 생성 방법
; https://www.sysnet.pe.kr/2/0/13469

닷넷: 2248. C# - 인터페이스 타입의 다중 포인터를 인자로 갖는 C/C++ 함수 연동
; https://www.sysnet.pe.kr/2/0/13607

닷넷: 2254. C# - COM 인터페이스의 상속 시 중복으로 메서드를 선언
; https://www.sysnet.pe.kr/2/0/13614




C# - 인터페이스 타입의 다중 포인터를 인자로 갖는 C/C++ 함수 연동

마침 그런 매개변수를 테스트를 할 수 있는 함수가 하나 있으니,

MFEnumDeviceSources function (mfidl.h)
; https://learn.microsoft.com/en-us/windows/win32/api/mfidl/nf-mfidl-mfenumdevicesources

HRESULT MFEnumDeviceSources(
  [in]  IMFAttributes *pAttributes,
  [out] IMFActivate   ***pppSourceActivate,
  [out] UINT32        *pcSourceActivate
);

과연 C#에서 저걸 Interop 하는 것이 가능할까요? ^^

우선, 이런 경우 아주 범용적으로 쓸 수 있는 (어차피 포인터를 받아오는 것이므로) IntPtr을 이용해 다음과 같이 마샬링할 수 있습니다.

[DllImport("Mf.dll")]
static extern HRESULT MFEnumDeviceSources(IMFAttributes pAttributes, out IntPtr pppSourceActivate, out uint pcSourceActivate);

이후 실행했을 때, pcSourceActivate에는 pppSourceActivate의 개수가 넘어올 것입니다. 그리고 디버거에서는 pppSourceActivate의 주소, 아래의 그림에서는 0x000002730066B990 주소인데, 그 부분을 메모리 창으로 보면 2개의 (유효해 보이는) 포인터 값이 있는 것을 확인할 수 있습니다.

marshal_interface_array_1.png

그러니까, 따지고 보면 결국 2개의 요소를 갖는 IntPtr 배열에 불과하므로 다음과 같이 호출해도 될 듯합니다.

[DllImport("Mf.dll")]
static extern HRESULT MFEnumDeviceSources(IMFAttributes pAttributes, out nint[] pppSourceActivate, [MarshalUsing out uint pcSourceActivate);

하지만 실제로 해보면, 2개의 요소를 반환하는 상황에서도 nint[] 배열에는 1개의 값만 채워져 옵니다. 이런 경우 예전 글에서 설명한 것처럼 SizeParamIndex를 지정해야 합니다.

[DllImport("Mf.dll")]
static extern HRESULT MFEnumDeviceSources(IMFAttributes pAttributes,
    [MarshalAs(UnmanagedType.LPArray, SizeParamIndex = 2)] out nint[] pppSourceActivate, out uint pcSourceActivate);




그런데, MFEnumDeviceSources의 경우가 특별한 것이 있다면, 저 매개변수의 타입이 3중 포인터라는 점입니다. 즉, 우리가 받아온 저 배열의 값은 또 다른 값을 가리키는 포인터라는 건데요, 다행인 점은 저것이 COM Interface라는 점입니다.

따라서, 포인터의 포인터를 처리할 필요 없이 애당초 Interface임을 알리는 마샬링을 지정해 object 배열로 받는 것도 가능합니다.

[DllImport("Mf.dll")]
static extern HRESULT MFEnumDeviceSources(IMFAttributes pAttributes,
    [MarshalAs(UnmanagedType.LPArray, ArraySubType = UnmanagedType.IUnknown, SizeParamIndex = 2)] out object[] pppSourceActivate, out uint pcSourceActivate);

이렇게 object로 받았으면 다음과 같은 식으로 "as" 형변환, 즉 내부적으로 QueryInterface를 통해 원래의 인터페이스로 복원하는 것이 가능합니다.

HRESULT hr = MFEnumDeviceSources(pAttributes, out pppSourceActivate, out pcSourceActivate);

if (pcSourceActivate != 0)
{
    IMFActivate? item = pppSourceActivate[0] as IMFActivate;
    if (item != null)
    {
        hr = item.ActivateObject(ref IMFMediaSourceGuid, out IntPtr pMFMediaSource);
    }
}

참고로, 위의 경우 ArraySubType을 UnmanagedType.IUnknown으로 명시하지 않으면 이런 예외가 발생합니다.

System.Runtime.InteropServices.InvalidOleVariantTypeException
  HResult=0x80131531
  Message=Specified OLE variant is invalid.
  Source=<Cannot evaluate the exception source>
  StackTrace:
<Cannot evaluate the exception stack trace>




위의 처리 단계까지 이해했다면, 이제 부가 코드 없이 곧바로 MFEnumDeviceSources DllImport 단계에서 아예 인터페이스를 마샬링하도록 다음과 같이 정의할 수 있습니다.

[DllImport("Mf.dll")]
static extern HRESULT MFEnumDeviceSources(IMFAttributes pAttributes,
[MarshalAs(UnmanagedType.LPArray, SizeParamIndex = 2)] out IMFActivate[] pppSourceActivate, out uint pcSourceActivate);

무려 3중 포인터를 인자로 갖는 괴상한 Win32 API를 별다른 부가 코드 없이 DllImport 정의 수준에서 그대로 연동할 수 있다는 점이 바로! C# 언어만의 매력이 되겠습니다. ^^




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

[연관 글]






[최초 등록일: ]
[최종 수정일: 5/1/2024]

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

비밀번호

댓글 작성자
 




... [181]  182  183  184  185  186  187  188  189  190  191  192  193  194  195  ...
NoWriterDateCnt.TitleFile(s)
458정성태2/12/200721810.NET Framework: 81. LINQ 개발 환경 설정 [1]
457정성태2/8/200726275.NET Framework: 80. LINQ 관련 용어 정리 및 리소스 소개 [2]
456정성태2/6/200724537Windows: 22. 가상화에 대해서.
455정성태2/4/200719365오류 유형: 25. 원격 데스크톱 환경에서의 Virtual Server 관리 환경 제어
454정성태2/4/200716890오류 유형: 24. VPC에서의 Vista 네트워킹 문제
509손대성6/24/200718982    답변글 오류 유형: 24.1. [답변]: 오류 유형 : 23. VPC 에서의 Vista 네트워킹 문제
453정성태2/4/200725051개발 환경 구성: 21. 서버 측 SoapExtension을 클라이언트에 알리고 싶다
452정성태1/31/200724776VC++: 31. 비스타에서 VS.NET 2005로 COM 프로젝트 빌드시 오류 [2]
451정성태1/31/200723930Windows: 21. Preview Handler 소개
450정성태1/30/200732222VS.NET IDE: 43. .NET에서의 필수 무결성 제어 조절하는 방법 - Manifest 파일 이용파일 다운로드2
449정성태2/4/200729035Windows: 20. UAC 이모저모 [2]
448정성태1/28/200724717Windows: 19. 3가지 유형의 가젯 프로그램
447정성태1/27/200721879Windows: 18. 비스타 도구 - 사양 정보 및 도구(Performance Information and Tools)
446정성태1/27/200730842VC++: 30. 필수 무결성 제어를 조절하는 방법(2) - 직접 코딩파일 다운로드1
445정성태2/8/200729407VC++: 29. 필수 무결성 제어를 조절하는 방법(1) - Manifest 파일 이용파일 다운로드2
444정성태1/27/200722859VC++: 28. 비스타 응용 프로그램 개발을 위한 VS.NET 2005 환경 설정
443정성태1/26/200721296VC++: 27. COM 개체로 인해 IE 7 비스타 버전이 종료될 때 오류 화면이 뜬다면?파일 다운로드1
442정성태1/24/200724303.NET Framework: 79. 새로운 암호화 클래스 (ECDsaCng, ECDiffieHellmanCng) 소개 [1]
441정성태1/23/200729030Windows: 17. 보안 데스크톱에서 활성화되지 않은 UAC 창이 안전할까?
440정성태1/24/200723122.NET Framework: 78. C# 3.0 - Anonymous types [1]
439정성태1/25/200724147.NET Framework: 77. C# 3.0 - Lambda 표현식 [1]
438정성태1/24/200723615.NET Framework: 76. C# 3.0 - 확장 함수
437정성태1/23/200731094Windows: 16. 개발자를 위한 UAC 환경 설정 [3]
436정성태1/17/200720255VS.NET IDE: 42. Orcas 2007년 1월 CTP 버전 설치 [5]
435정성태1/14/200719904기타: 17. 베타 제품과 최종 제품은 다르다 [2]
434정성태2/4/200723497Windows: 15. MIC 환경 구성 - Windows XP와 유사한 보안 설정 [4]
... [181]  182  183  184  185  186  187  188  189  190  191  192  193  194  195  ...