Microsoft MVP성태의 닷넷 이야기
C++의 double pointer를 C#에서 구현하는 방법이 잘 안됩니다. [링크 복사], [링크+제목 복사],
조회: 14034
글쓴 사람
Heegyoo Lee
홈페이지
첨부 파일
 
(연관된 글이 1개 있습니다.)

안녕하세요.
좀 오래된 이미지 그래버 보드를 C#에서 사용하려고 합니다.
보드는 DLL 형태로 드라이버를 제공하는데
Init함수를 호출하면 보드에서 사용할 버퍼의 메모리 주소를 리턴하고
Scan을 시작하면 위 주소부터 이미지 데이터를 쓰게 됩니다.
C++에서는 다음과 같이 사용하면 동작합니다.

extern "C" __declspec(dllimport) bool OpenPCI(DWORD **RAM);
    pOpen = (DLL_Open)GetProcAddress(hdll, "OpenPCI");
    HMODULE hdll = LoadLibrary(L"aa.dll");
DWORD* RAMPCI;

    pOpen(&RAMPCI);
 ...
    DWORD data[i] = RAMPCI[i];
...

이를 C#으로 바꾸고자 해서 다음과 같이

        [DllImport("aa.dll", CallingConvention = CallingConvention.Cdecl)]
        public static extern bool OpenPCI(ref IntPtr pVRAM);

        IntPtr pVRAM = IntPtr.Zero;
        bool res = OpenPCI(ref pVRAM); //
...
        Marshal.Copy(pVRAM, linebuff, 0, Length); // 1 line read

코드를 작성했는데 컴파일 에러는 없이 동작은 하나 주소가 맞는지 알 수 없습니다.
실행해 보면 Marshal.Copy 수행 중에 보호된 메모리를 참조한다는 오류가 발생합니다.
디버거에서는 Open시 시그니처 오류로 스택이 불안정 해진다는 메세지가 나옵니다.

질문 : C++에서 RAMPCI 포인터의 주소를 인자로 보내면 버퍼 메모리 주소를 포인터의 값에 넣어 주는데
  C#에서 IntPtr의 주소를 갖는 포인터를 또 만들 수 있는지, 위의 코드처럼 ref pVRAM 으로 직접 받아 오면
  어떤 값이 리턴되어 오는지 궁금합니다.


[연관 글]






[최초 등록일: ]
[최종 수정일: 10/27/2023]


비밀번호

댓글 작성자
 



2023-10-27 07시34분
일단, 올려주신 조각 코드로는 문제가 없어 보입니다.

혹시 Marshal.Copy에서 크기에 맞지 않게 지정한 것이 아닐까요? 혹은, 확인을 위해 DWORD **pptr 인자를 받는 C++ DLL을 예제 코드로 작성해 interop 해보시면 좀 더 문제가 명확해 질 것입니다. 그렇게 했는데도 여전히 동일한 문제가 발생하면 만드신 예제 코드를 첨부해 주시면 코드를 확인해 보겠습니다.

다음의 글도 참고하세요.

Win32 Interop - C/C++ DLL로부터 이중 포인터 버퍼를 C#으로 받는 예제
; https://www.sysnet.pe.kr/2/0/13429
정성태
2023-11-03 07시38분
코드 자체는 동작하는 것처럼 보이는데요. (어떤 때는 시그니쳐가 맞지않아 스택이 불안정해진다.라는 메세지가 나옵니다.)
보드를 Open하면 버퍼를 0x100000 바이트 크기로 데이터를 써넣는데 중간에 '보호된 메모리를 액세스 하려고 한다.'는 에러가 납니다.
디버거에서 메모리를 보면 중간에 일부 구역이 ??로 표시되고 있습니다.
이것은 메모리에 대한 접근 권한이 없어서 나오는 에러 같은데 OS나 Visual Studio 버전을 낮춰봐도 같은 에러가 발생합니다.
원인을 찾아 보려면 어떻게 해야 할지 조언 구합니다.
Heegyoo Lee
2023-11-06 08시44분
일단, PInvokeStackImbalance 오류가 발생했다는 것은 아마도 호출 규약이 맞지 않았다는 것이고, 그렇다면 64비트 응용 프로그램이 아닌, 32비트 응용 프로그램인 것 같은데, 이런 경우 정확한 호출 규약을 맞춰줘야 합니다. 다음의 글을 보시고 이에 대해 Cdecl, Stdcall 등의 유형을 DllImport에 지정하시면 됩니다.

C# 개발자를 위한 Win32 DLL export 함수의 호출 규약 (1) - x86 환경에서의 __cdecl, __stdcall에 대한 Name mangling
; https://www.sysnet.pe.kr/2/0/11132

올려주신 조각 코드로 봐서는 아마도 "[DllImport("aa.dll", CallingConvention = CallingConvention.Cdecl)]"라고 쓴 것이 맞을 텐데... PInvokeStackImbalance 오류가 난다는 것이 좀 의아한 상황입니다.

'??'로 표시된 경우는, 접근 권한이 없다기보다는 잘못된 위치를 나타내고 있을 가능성이 더 큽니다. (따라서 OS나 비주얼 스튜디오의 버전과는 무관합니다.)

더 이상은 (원칙은 이미 모두 설명했기 때문에) 딱히 조언해 드릴 것이 없습니다. (재현 가능한 예제 코드가 필요합니다.)
정성태

... 31  32  33  34  35  36  37  38  39  40  [41]  42  43  44  45  ...
NoWriterDateCnt.TitleFile(s)
4865heyh...7/31/201719651클릭원스로 배포 한 프로젝트가 끝났는지 알 수 있는 방법 [8]
4864초보자7/28/201718803DllIImport질문 드립니다. [1]
4863다연아빠7/23/201719558전역 예외처리에 대해 질문있습니다. [3]
4861라르크7/17/201726406window form 예제 따라하는 중인데 12.3 서비스 응용 프로그램에서 진행이 안됩니다. [3]파일 다운로드1
4859heyh...7/10/201719761다른 환경에서 실행하기 [1]
4858heyh...7/10/201718924Clickonce update에 관한질문입니다. [1]
4857heyh...7/7/201719653제가 여태까지 작성한 보고서입니다. [2]파일 다운로드1
4856heyh...7/6/201717508성태님 다른질문입니다. [4]
4855JP7/6/201717554Dispose 패턴 구현시 Finalize 재정의에 대한 질문드립니다. [2]
4854heyh...7/6/201717810
4853heyh...7/5/201717122성태님이 작성한대로 해봤습니다. [1]파일 다운로드1
4852김레오7/4/201720412서드파티 dll 디버깅에 대해 질문드립니다. [2]
4851김현준7/3/201720906Datagridview VirtualMode 시 GC가 계속 호출되는 현상이 이해가 안갑니다. [2]
4850heyh...7/3/201719282성태님 밑에 질문드렸던 오류입니다. [1]파일 다운로드1
4849포플러7/2/201741291C#으로 만든 프로그램이 어느 순간 속도가 느려지거나 멈춤현상이 있습니다. [4]
4848윤진영7/2/201721578Microsoft Visual C++ 6.0 무설치 관련 [1]
4847heyh...6/30/201719048아무리 고민해도 답이 안나와서.. 질문 드립니다 [1]
4846heyh...6/28/201718592Clickonce에서 Clickonce로 변수 전달 [1]
4845heyh...6/26/201718037vb.net 에서 manifest 제거 방법 [1]파일 다운로드1
4844san6/21/201717695part3 pdf파일로 보는데 눈아퍼요.... 활자로 보고싶어요 [2]
4843윤현수6/20/201716563socket통신에 관한 질문입니다. [1]
4842오세운6/7/201717531로그인폼 다시 질문드려요. [1]
4841popo6/7/201718330궁금한사항이 있어 질문 드립니다. [1]
4840윤현수6/5/201718156datagridview에 바인딩시킨 dataset.table에 관련된 질문입니다. [2]파일 다운로드1
4839오세운6/2/201719175로그인폼 만들기 [1]
4838후배5/29/201719835PDF Reader 라이브러리.. [3]
... 31  32  33  34  35  36  37  38  39  40  [41]  42  43  44  45  ...