Microsoft MVP성태의 닷넷 이야기
P/Inovke 관련 질문입니다. [링크 복사], [링크+제목 복사]
조회: 12568
글쓴 사람
이성환 (vactorman at naver.com)
홈페이지
첨부 파일

이렇게 연타석 질문을 날리게 됐습니다....(__)

다른 회사의 SDK를 받아다가 dll로 만든 후에 닷넷 솔루션에서 사용해야하는 상황입니다.
SDK를 받아서 만들어야하는 dll 역시 C#으로 만들어야하고 64비트 빌드되어야 하는 상황입니다.

일단 SDK를 받아보니
종류별로 dll과 lib 파일이 들어 있고 문서도 함께 들어있는데요. C/C++로 만들어져 있습니다.

이걸 가지고 P/Invoke를 하려고 하는데요.
P/Invoke로 가능한 것인지 의문 나는 게 있어서 질문 드립니다.

일단 사용하고자 하는 dll의 헤더 파일을 보니
제가 호출하고자 하는 함수가 특정 Struct 안에 함수 포인터로 작성되어 있더군요.
그리고 그 Struct는 Init(params)이라는 함수를 호출해서 포인터로 넘겨받는 거구요.

간단히 일부만 예로 들면

헤더 파일에

#ifdef WIN32
    #ifdef _USRDLL
        #define NVSNETLIB_API EXTERN_C __declspec(dllexport)
    #elif _LIB
        #define NVSNETLIB_API EXTERN_C
    #else
        #define NVSNETLIB_API EXTERN_C __declspec(dllimport)
    #endif
#else
    #define NVSNETLIB_API
#endif

typedef struct CNetSession {
    unsigned int dataSize;
    char *recvData;
    
    int (*sendAudio)(void *_this, char *_buf, unsigned int _size);
    int (*sendControl)(void *_this, unsigned int _type, char *_buf, unsigned int _size);
} CNetSession, *pCNetSession;

NVSNETLIB_API void * NETSESSION_Init(
    char *_address,
    unsigned short _port,
    char *_url,
    char *_id,
    char *_passwd,
    unsigned char _connectType
    );

뭐 이렇게 작성되어 있구요.

예제 소스에서는

int main(int argc, char **argv) {
pCNetSession netSession;
char connectType = 0x07;

netSession = (pCNetSession)NETSESSION_Init(argv[1], atoi(argv[2]), NULL, argv[3], argv[4], connectType );

if( netSession->sendAudio(netSession, audioBuf, dwByteRead) == 0 )
{
    printf(".");
}
//

}

이런 식으로 사용하더군요.

지금 제가 호출해서 사용해야하는 함수가 sendControl이라는 함수인데
(이 함수에 대한 사용예제는 문서에 없더군요.)

이걸 C#에서 P/Invoke로 호출하려고 합니다.

일단 NETSESSION_Init()은

[DllImport("References/DLL.dll")]
public static extern IntPtr NETSESSION_Init(String _address, UInt16 _port, String _url, String _id, String _passwd, Byte _connectType);

이렇게 작성했습니다.

그리고 CNetSession Struct를 사용하기 위해

[StructLayout(LayoutKind.Sequential)]
public class CNetSession
{
    public UInt32 dataSize;
    public String recvData;
    
    public delegate int sendAudio(IntPtr _this, String _buf, UInt32 _size);
    public delegate int sendControl(IntPtr _this, UInt32 _type, String _buf, UInt32 _size);
}

이렇게 만들었습니다.

근데 여기서부터 이해가 잘 안 되는데요. 요약하자면

1. dll 헤더파일에 정의 되어있는 CNetSession Struct 안에 함수포인터로 작성되어 있는 것들을 C#의 클래스로 치환할 때 델리게이트로 만드는게 맞는건지?
2. 그렇다면 이 델리게이트들의 콜백은 어떻게 연결하고 호출하는지?
3. NETSESSION_Init() 메서드를 호출해서 반환 받은 IntPtr를 제가 만든 CNetSession 클래스로 변환 하는 게 가능한건지?

입니다.

SDK 문서의 사용 예제처럼, NETSESSION_Init() 함수를 호출해 반환 받은 포인터에서
직접 sendAudio() 함수를 호출하는 걸로봐서 (netSession->sendAudio() 이런 식...)
이미 dll 내부에서 sendAudio() 의 콜백을 연결 해 놓은 거 같은데 (아닌가요?)

이걸 C#에서 바로 사용할 수 있을지가 의문입니다.
(이미 콜백이 연결된 녀석인지 아니면 따로 연결을 해줘야하는 건지 모르겠습니다. 따로 연결해야한다면 어떻게 해야하나요?)

단순하게 Win32 API정도 P/Invoke 해서 쓰다가 이런 경우를 처음 마주치니까
하나도 이해가 안 되네요.

C/C++ 지식이 전무하다보니 사실 SDK에서 제공되는 예제를 봐도 이해가 잘 안 가는데,
필요한 부분만 가지고와서 P/Invoke로 쓸려니 더 이해가 안 가네요.
(예제 솔루션도 못 만들겠습니다...;ㅅ;)




혹시 질문 내용이 난해하시다면 SDK를 보내드릴게요.

답변 부탁드립니다...;ㅅ;








[최초 등록일: ]
[최종 수정일: 7/28/2011]


비밀번호

댓글 작성자
 



2011-07-28 10시07분
[이성환] 간략하게 테스트 솔루션하나 추가 해봤습니다.

이렇게 쓰는 게 맞는 지 모르겠네요.

일단 실행하면

Attempted to read or write protected memory. This is often an indication that other memory is corrupt.

이 에러가 떨어지는 데요.

아무래도 시그니처를 제대로 못 맞춰서 그런 거 같은데 어디서 잘못 됐는지 확인이 안 되네요.

바쁘신데 다른 건 신경 안 쓰셔도 괜찮은데

제가 만든 테스트 소스 처럼 하는 게 맞는 지만 알려주시면 될 거 같습니다.

그럼 좀 고민을 해볼 게요.

...C 메모리 구조(packed 한다고 들었는데 맞는지 모르겠네요)와 닷넷 메모리 구조를 맞추는 게 너무 어려워요.
[guest]
2011-07-29 03시20분
[이성환] 아.. 이틀 동안 P/Invoke로 어떻게든 해보려고 했는데

결국 C++/CLI로 dll을 만들어 작업은 마무리 했네요.

작업은 작업이지만 그냥 제가 생각한대로 사용하는 게 맞는지 궁금합니다.

플랫폼간 호출은 P/Invoke으로 뭐든 쉽게 할 줄 알았는데 그게 아니었네요...;ㅅ;
[guest]
2011-08-02 12시51분
(아... 답변이 너무 늦었군요. ^^ 제가 근래들어 퇴근 후 컴퓨터를 쓸 일이 거의 없어서... ^^;)

예, 소스 코드를 보았는데... 잘 하셨습니다. ^^ 그런데, 몇몇 부분에서 약간의 실수가 보이는데요. 대표적으로 dataSize, recvData 필드 쪽을 살펴보면,

unsigned int dataSize;
 char *recvData;

이런 경우, 대개 recvData 포인터에 할당된 크기를 dataSize가 지정하는 경우라고 볼 수 있는데 C# 의 PtrToStructure 함수는 이에 대해 마샬링을 자동으로 하지 못합니다. 그런 부분 때문에 "Attemped..." 오류가 발생한 것일테고.

제가 이성환님이 올려준 소스 코드를 약간 변경해서 sendAudio 메서드가 '오류 없이' 호출되는 것 까지 변경해 보았으니 참고하세요.

http://www.sysnet.pe.kr/temp/CellinxPInvokeTest.zip

^^ 어쨌든 C++/CLI로 해결하셨으니, 의미가 좀 떨어지겠지만. ^^

(참고로, C++ 의 struct 인스턴스를 마샬링 할 때 포인터 부분은 IntPtr 로 일단 설정해 두면서 맞춰주는 것이 편합니다.)
정성태
2011-08-03 09시12분
[이성환] 답변 감사합니다.

역시 확인해 다른 소스에서도 마샬링할 때 포인터 부분을 IntPtr로 바꾸니까 잘 되는 군요.

마샬링에 대응하는 타입만 보고 그냥 치환해서 타입들을 집어넣었는데 역시나 포인터가 문제였군요.

C++/CLI로 작업된 DLL들 외에 다른 DLL들은 이렇게 수정해서 잘 쓰고 있습니다.

도움 감사합니다 (__)
[guest]

... 46  47  48  49  50  51  52  53  54  55  56  57  [58]  59  60  ...
NoWriterDateCnt.TitleFile(s)
1176김태훈10/25/201310274AxWebBrowser에 대해 질문드립니다. [1]
1175서경희10/20/201315282netscape 지원이 되지 않는다는 문구.. [2]파일 다운로드1
1174임동찬10/16/201314091프리징 현상에 대한 고민 [5]
1173김재영10/8/20139900인터페이스에 대해 기초적 질문이 있습니다. [2]파일 다운로드1
1172박진영10/2/201312792웹사이트 연결시 AJAX 어셈블리 오류 문의드립니다. [5]파일 다운로드1
1171링거8/30/201320733ClickOnce 업데이트 문제. [4]
1170임동찬8/28/201312071비동기적 이벤트 핸들링 방법 [2]
1167나종식8/20/201311188win7 에서 LSP 가 DNS Client 에 인젝션 안되는 문제 [6]
1165임동찬8/19/201310489오류 발생시 로깅 문제.. [3]
1164박철8/19/201311140모바일 게임서버를 작성 하려면 무엇부터 시작하여야 하나요? [2]
1163안연준8/2/201310613음... 안녕하세요 ^^ 윈도우즈 폼에 대해서 잠시 물어볼께요 [3]
1162박진영7/23/201390821개의 PC에서 동일사이트 접속제한을 어떻게 하죠? [1]
1161Ji Y...7/12/201311800안녕하세요? 음성인식 관련해서 질문있습니다, [2]
1160이상식7/12/201312189웹서비스 커넥션 풀 오류입니다. [6]
1159김덕성7/4/201320373C# 마샬링 관련 문의입니다 [3]
1157이상식6/26/201314034빌드시 오류인데 원인을 모르겠네요. [2]파일 다운로드1
1156정환나라6/20/201311217In-Process Side-By-Side 동작 원리는 어떻게 되는걸까요? [3]
1155MD워시퍼6/20/201310245안녕하세요. 한가지 궁금한 것이 있어서 문의를 드립니다. [1]
1154it 대...6/13/201313282무선 패킷 스니핑 프로그램 [1]
1153이상헌6/8/201310785죄송합니다;; 또다시 의문점이 생겨 질문드립니다. [1]
1152이상헌6/8/201310449넷두이노 플러스 펌웨어 다운로드가 안되네요;;; [1]
1151황기동6/5/201311255무선 패킷 전송관련 질문 [1]
1147이진권6/2/201311283c# 윈폼 facebook과 연동2 [1]
1144황은영5/29/201315774원격지 서버에서 COM+ 호출이 되지 않습니다. [7]
1143김선희5/28/201316700윈도우 8, 익스플로러 10, 64bit->32bit [1]
1142백지훈5/15/201311120음성인식 TTS 관련해서 질문드립니다 ㅜㅜ [2]파일 다운로드1
... 46  47  48  49  50  51  52  53  54  55  56  57  [58]  59  60  ...