Microsoft MVP성태의 닷넷 이야기
P/Inovke 관련 질문입니다. [링크 복사], [링크+제목 복사]
조회: 12581
글쓴 사람
이성환 (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]

1  2  3  4  5  [6]  7  8  9  10  11  12  13  14  15  ...
NoWriterDateCnt.TitleFile(s)
5802모바일앱1/22/20232925XAMARINE vs Android Studio [7]
5801블루투스1/19/20233673WPF 은행지폐계수기 개조 후 결과값 서버 전송 [2]파일 다운로드1
5800김민아1/19/20233982안녕하세요 가비지 컬렉터 동작 원리 중 궁금한점이 있습니다 [5]
5799guest1/19/20233149C# 공유폴더 내 Acess 디비 공유 [2]
5798kss1/19/20232939책 오탈인가요? [1]
5797이거비버1/19/20233028C# 공부 이후 MS 프레임워크.. 어느것을 공부해야할까요? 너무 많아서 정신이 없네요 [5]
5796guest1/19/20232751해킹 test [3]
5795동기1/18/20232593동기 스레드와 메서드와 While [2]
5794박규동1/18/20232565.net publish 할때마다 runtimeconfig 값이 바뀌는 현상 [1]
5793후후훗1/18/20232727.NET Core 에서 사용중인 함수 후킹 방법 [2]
5792Will...1/18/20232671소스코드 Log Write 기능을 리스트업 질문 [9]
5791PLC1/16/20233363C# - 실시간 5개 룸 모니터링 [15]
5790집으로 ...1/13/20233017[UI 멈춤 현상]deadlock 관련 글을 보고 혹시나 하고 문의 드립니다. [4]
5789guest1/10/20232910스레드와 Async Task [2]파일 다운로드1
5788kr11/10/20233356C# 에서 제공하는 컬렉션들의 차이점이 궁금합니다. [3]
5787stack1/10/20233144STACKOVERFLOW [1]
5786Dev ...1/9/20234014익명 클래스 말고 익명 구조체는 불가능한걸까요? [4]
5785음성인식1/8/20233616음성인식 System.Speech - 문법에 사용된 언어가 음성 인식기의 언어와 일치하지 않습니다. [2]
5784MS워드1/8/20233209MS워드에서 ctrl Z는 클립보드를 이용하나요? 아니면 참조자 이용하나요? [7]
5783구직자1/7/20233271C#개발자 구인광고와 초급개발자 [3]파일 다운로드1
5782Sqli...1/5/20233216윈도우11 노트북에서 exe(Sqlite)만들어 윈도우 7 PC에 설치 시 [5]
5781List맨1/5/20233744List.Add("newobj") 속도는 빠른 편인지요? [11]
5780임세1/3/20234352C# 프로그래밍 10 책 구매한 사람입니다. 3부 자료는 어디서 다운 받을 수 있나요? [4]
5779이건우1/2/20233217안녕하십니까 루프안에서 메세지처리에 관하여 질문드립니다! [2]
5778이건우12/30/20224474안녕하세요 c#에서 dll참조 관련 질문드립니다. [4]
5777감사합니...12/29/20223489UI Thread에 Invoke 처리관련 궁금합니다. [3]
1  2  3  4  5  [6]  7  8  9  10  11  12  13  14  15  ...