Microsoft MVP성태의 닷넷 이야기
C++의 double pointer를 C#에서 구현하는 방법이 잘 안됩니다. [링크 복사], [링크+제목 복사],
조회: 5103
글쓴 사람
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나 비주얼 스튜디오의 버전과는 무관합니다.)

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

... 76  77  [78]  79  80  81  82  83  84  85  86  87  88  89  90  ...
NoWriterDateCnt.TitleFile(s)
519이진형12/8/20068082스마트 클라이언트에서 부모 IE 컨트롤 하기
520정성태12/11/20069067    답변글 [답변]: 스마트 클라이언트에서 부모 IE 컨트롤 하기 [3]
517주낙현12/5/20068828Property 재정의가 가능한가요?? [1]
516양덕진11/29/20068488SSL인증서 질문? [1]
515경혜원11/27/20069514화면 캡춰 관련해서 문의 드립니다. [1]
513양덕진11/26/200612018SSL인증문제?
514양덕진11/27/200613621    답변글 [답변]: SSL인증문제? [1]
512윤창일11/24/200611167[ActiveX vs 가장기법] [1]
511방문자11/24/20069889스마트 클라이언트 질문입니다. [3]
509주낙현11/22/200610960스마트 클라이언트에서 WebBrowser 컨트롤사용? [4]
508임장현11/20/200610599서로다른 프로젝트간의 세션공유에 대해 질문드립니다.
510정성태11/22/20069025    답변글 [답변]: 서로다른 프로젝트 간의 세션 공유에 대해 질문드립니다.
504탑.11/10/20069797COM+ 구성 좀 봐주세요...
505정성태11/14/20069611    답변글 [답변]: COM+ 구성 좀 봐주세요...
506탑.11/15/20068716        답변글 [답변]: [답변]: COM+ 구성 좀 봐주세요... [2]
503sagi...11/10/20068710bho 폼 관련 질문입니다/ [1]
501안연준11/7/20069314(SmartClient) 프레임워크 1.1 과 2.0 은 호환이 안된다? [3]
507안연준11/17/20069014    답변글 [답변]: (SmartClient) 프레임워크 1.1 과 2.0 은 호환이 안된다? [1]
498쿠리마10/30/20069408Smart Client를 VC60 MFC Dialog 에서 사용하기
500정성태10/30/200611178    답변글 [답변]: Smart Client를 VC60 MFC Dialog 에서 사용하기 [3]
497이방은10/30/200611293질문 있습니다..ㅡ.ㅠ;
499정성태10/30/200611569    답변글 [답변]: 질문 있습니다..ㅡ.ㅠ; [2]
495엔틱스10/25/20068944안녕하세요. 세션에 관련해서 질문을 올립니다. [2]
493안연준10/25/20069630스마트클라이언트 배포에서 Config 내용이 이해가 안되요
494안연준10/25/20067877    답변글 [답변]: 스마트클라이언트 배포에서 Config 내용이 이해가 안되요 [2]
489안연준10/23/20068848스마트 클라이언트 배포시 문제점
... 76  77  [78]  79  80  81  82  83  84  85  86  87  88  89  90  ...