Microsoft MVP성태의 닷넷 이야기
C++의 double pointer를 C#에서 구현하는 방법이 잘 안됩니다. [링크 복사], [링크+제목 복사],
조회: 4841
글쓴 사람
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)
5704따봉이8/4/20224666EventHandler 관련 [1]
5703조민준8/3/20224104안녕하세요 정적 멤버 초기화 관련 간단한 질문 [2]
5702석우8/3/20223998C# WPF Ribbon IsMinimized 프로퍼티 변경 감지 방법문의 [2]파일 다운로드1
5701김기헌8/1/20224235foreach 문으로 컬렉션을 열거할 때 궁금한 점 [2]
5699갓경섭7/29/20224643C# 프로그램 느려짐 현상 [2]
5698kss7/29/20225401ValueTask 질문입니다. [5]
5697스팸7/28/20225467윈도우서버 Port 고갈문제 (Event ID 4227, 4231) [3]파일 다운로드1
5696유석7/27/20225688닷넷 4.X 와 닷넷 6.0 버전 차이 [6]
5695초보코더7/26/20225031[WCF] EndpointNotFoundException 기본 연결이 닫혔습니다. 원격 서버에 연결할 수 없습니다. [1]
5693차가워7/22/20224447socket 종료 시 reveive수신부에서 에러 나는거 처리 문의 [5]
5692도비7/21/20224026Ubuntu LVM 확장 관련 [1]파일 다운로드1
5691따봉이7/21/20224731Task 비동기 UI 크로스 쓰레딩 Err 관련 [1]
5690훈이7/20/20223977wpf질문입니다 [2]
5689질문7/20/20223751비동기 함수를 사용하는 방법에 대한 질문입니다 [1]
5688김정현7/11/20223852소스 디버깅(F5)시 프로그램 켜지지 않고 디버깅 가능한 상태 [1]
5687규철7/6/20223979개발업무 문서 관리 방법 문의 [3]
5686권용완7/5/20223953Linq Entites 에서 string[]에 담긴 데이터를 제외하는 방법 문의 [3]
5685차가워7/4/20224120메모리 선택 문의 [1]
5683김민혁7/1/20224665HttpWebRequest POST 전송 관련해서 질문 드립니다. [4]파일 다운로드1
5682김기헌6/20/20224108안녕하세요 서로 다른 스레드에서 동일 인스턴스의 메소드를 호출할때 질문드립니다 [2]
5681미래는투...6/19/20224010VC++ 빌드 관련 문의 (h파일이 바뀌었는데 컴파일을 꼭 하지 않아도 되나요?) [1]
5680Syong6/12/20225002EXE파일에 속성값 추가하는 방법이 있나요? [2]파일 다운로드1
5679차가워6/8/20224181tcpip socket 여러 랜카드 사용 시 연결 문의 [1]
5678차가워6/7/20224087원자적 실행 문의 [3]
5677장성욱6/7/20224114블루투스 포트와 시리얼포트 연결시 에러 [2]
5675차가워5/29/20224796윈도우설정 프로세스사용계획 옵션과 c# 스레드 [1]
1  2  3  4  5  6  7  8  9  [10]  11  12  13  14  15  ...