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

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

1  2  3  4  5  6  7  [8]  9  10  11  12  13  14  15  ...
NoWriterDateCnt.TitleFile(s)
5756흰털너부리11/25/20223440asp.net core web api에서 json 특정 property 무시하는 방법 문의 드립니다. System.Text.Json 사용중입니다. [1]
5755문정환11/24/20223781싱글스레드 프로그램도 컨텍스트 스위칭이 생길 수 있나요? [4]
5754초급11/24/20223625c# 소켓통신 [1]
5753흰털너부리11/24/20223426List와 ObservableCollection을 비교 해서 다른 값 추출 FirstOrDefault 객체 비교 [4]파일 다운로드1
5752푸헐11/15/20223566app.config 에 connectionStrings를 aspnet_regiis로 enctyption [4]
5751차가워11/8/20224365vs2022 preview net7 AOT 콘솔 실행 성능 [7]
5749차가워11/4/20223779전처리 지시문 #if DEBUG 배포시 실행 여부 [1]
5748김기헌11/3/20224114안녕하세요 선생님 싱글톤 패턴을 꼭 이렇게 사용해야 하나요? [6]
5747김기헌11/2/20223504안녕하세요 선생님 네트워크 관련 용어 중 IP 주소가 왜 논리적 주소라고 표현되는 건가요? [2]
5746물냉면이...11/2/20223693서로 다른 클래스에 있는 동일 함수의 일괄 호출 방법에 대해 궁금합니다. [3]
5745흰털너부리11/1/20224042.net core web api 사용 제한에 관한 질문 입니다. [2]
5744차가워10/31/20224432윈폼 Console.WriteLine(); 연산 문의 [1]
5743흰털너부리10/27/20223746reflection, static, override 질문입니다. [1]
5742차가워10/27/20223660하나의 socket에 여러 스레드가 접근 하는 경우 [1]
5741조호상10/27/20223886OpenCVSharp4 구현 가능 문의 [1]
5740혜성10/26/20224723Visual Studio 2022 C# 콘솔 프로그램 기본 코드 변경된 이유는 무엇인가요? [2]
5739슬픈단잠10/25/20223687조언 주신 방법으로 해봤으나, 여전히 어디가 문제인지 파악을 못했습니다. [2]
5738슬픈단잠10/24/20224076C# 마샬링 관련 질문입니다. [2]
5737감사합니...10/20/20224734찾은 어셈블리의 매니페스트 정의와 어셈블리 참조가 일치하지 않습니다. [8]파일 다운로드1
5736초보 개...10/19/20223740c# winforms 프로그램을 setup 파일로 배포 시, 설정 문의 [2]
5735농상10/17/20224402싱글톤 공부중 질문이 있습니다. [8]
5734mins10/14/20224141델파이 dll을 c#에서 사용하기 관련해서 포인트 관련이라 질문을 올립니다. [2]
5733김경환10/12/20224154선생님 질문하나만드리겠습니다. [1]
5732kss10/8/20224441c# socket.poll 버그인가요? [2]파일 다운로드1
5731kss10/8/20223998c# socket.poll 버그인가요? [3]
5730김재환10/7/20224531WPF에서 디스플레이 배율이 100%가 아닌 경우, Window의 포지션 정보가 부정확해지는 문제 [2]
1  2  3  4  5  6  7  [8]  9  10  11  12  13  14  15  ...