Microsoft MVP성태의 닷넷 이야기
글쓴 사람
정성태 (techsharer at outlook.com)
홈페이지
첨부 파일
(연관된 글이 1개 있습니다.)
(시리즈 글이 4개 있습니다.)
.NET Framework: 199. .NET 코드 - Named Pipe 닷넷 서버와 VC++ 클라이언트 제작
; https://www.sysnet.pe.kr/2/0/971

.NET Framework: 464. 프로세스 간 통신 시 소켓 필요 없이 간단하게 Pipe를 열어 통신하는 방법
; https://www.sysnet.pe.kr/2/0/1751

.NET Framework: 477. SeCreateGlobalPrivilege 특권과 WCF NamedPipe
; https://www.sysnet.pe.kr/2/0/1806

Linux: 20. C# - Linux에서의 Named Pipe를 이용한 통신
; https://www.sysnet.pe.kr/2/0/11964




.NET 코드 - Named Pipe 닷넷 서버와 VC++ 클라이언트 제작

개인적으로, 닷넷 응용프로그램만으로 구성된 서버/클라이언트 상황에서는 웬만한 성능 요건이 아니고서는 단일 머신의 IPC(Inter-Process Communication)에서도 소켓을 즐겨 사용합니다. 사실, 소켓이라기보다는 WCF라고 볼 수 있는데요. 개체의 직렬화/역직렬화 부분을 대단히 쉽게 해결해 주기 때문에 개발 효율성을 보면 그만한 IPC 수단이 없습니다.

그런데, 간혹 소켓을 사용하기 껄끄러운 경우가 있습니다. 바로 C/C++ 프로그램과의 통신인데요. 다행히 최근에는 RESTful API를 WCF로도 쉽게 구현이 가능해서 그나마 C/C++과의 통신이 쉬워지긴 했습니다. 그렇긴 해도, 대부분의 경우에 C/C++이 사용되었다는 것은 상당히 속도에 민감하거나 시스템 하부에서 동작하는 경우여서 소켓 사용이 바람직하지 않을 수 있습니다.

예를 들면, 지난번 설명한 SPI를 이용한 LSP 모듈이 이에 해당할 것입니다.

Winsock 2 Layered Service Provider - Visual Studio 2010용 프로젝트
; https://www.sysnet.pe.kr/2/0/969

소켓 모니터링을 하는 응용 프로그램에서 IPC 수단으로 소켓을 선택하는 것은 재귀호출에 가까운 효과를 냅니다. 이런 경우라면 소켓이 아닌 "Named Pipe" 등의 또 다른 IPC 수단들이 적절한 해답이 될 수 있습니다.




다양한 IPC 수단 중에서 여기서는 "Named Pipe"만을 예제로 다뤄볼 것입니다. 다행히 웹에 구현 자료가 워낙 공개가 잘 되어 있어서 그다지 어려움 없이 만들 수 있습니다.

첨부한 예제 프로젝트의 경우에도, 서버는 다음의 글에서 공개된 소스 코드를 가져다 썼고,

방법: 명명된 파이프를 사용하여 네트워크를 통한 프로세스 간 통신
; https://learn.microsoft.com/ko-kr/dotnet/standard/io/how-to-use-named-pipes-for-network-interprocess-communication

클라이언트는 다음의 코드를 가져다 썼습니다.

Named Pipe Client
; https://learn.microsoft.com/en-us/windows/win32/ipc/named-pipe-client

서버의 경우, 위의 예제 코드에서 살짝 변경한 것이 있다면 스레드를 이용한 동기적인 방식이 아닌, Begin/End를 이용한 비동기 통신을 구현해 보았습니다.

private void Form1_Load(object sender, EventArgs e)
{
    NamedPipeServerStream aPipe;
                
    aPipe = new NamedPipeServerStream("testpipe", PipeDirection.In, numServers, PipeTransmissionMode.Byte, PipeOptions.Asynchronous);

    aPipe.BeginWaitForConnection(
        delegate(IAsyncResult state)
        {
            ProcessConnection(state, aPipe);
        }, null);
}

void ProcessConnection(IAsyncResult asyncResult, NamedPipeServerStream data)
{
    NamedPipeServerStream pipe = data as NamedPipeServerStream;

    pipe.EndWaitForConnection(asyncResult);

    byte[] byteBuffer = new byte[...];
    int readBytes = pipe.Read(byteBuffer,,,);
    ...[생략]...

    pipe.Disconnect();

    pipe.BeginWaitForConnection(
            delegate(IAsyncResult state)
            {
                ProcessConnection(state, pipe);
            }, null);
}

반면, 클라이언트는 별다르게 특별한 점이 없습니다. 왜냐하면 일반적인 CreateFile/WriteFile/CloseHandle 절차를 따르기 때문입니다. 단지 아래의 예제와 같이 닷넷 측의 NamedPipeServerStream 타입에서 지정된 Pipe 이름을 CreateFile에서 맞춰주면 되는 정도입니다.

=== 닷넷 서버 측 ===
new NamedPipeServerStream("testpipe", ...);

=== VC++ 클라이언트 측 ===
HANDLE hNamedPipe = ::CreateFile(L"\\\\.\\pipe\\testpipe", ...);

그 외에는 상호 간에 데이터 패킷에 대한 구조를 정의해야 하는데 이는 모두 개발자의 몫이죠. (바로 이 부분 때문에 제가 WCF를 즐겨 씁니다.)

비록 예제이긴 하지만, 아래와 같이 기본적인 데이터 구조는 갖춰 놓았습니다.

typedef struct tagPipePacket
{
    int totalDataSize;
    EnumPacketType packetVersion;
    SOCKET socket;
    int opType;
    int bufLen;
    wchar_t Data[0];
} PipePacket;

마지막의 wchar_t [0]은 C/C++ 개발자들이 가변 배열을 정의할 필요가 있을 때 즐겨쓰는 표현이죠. 그 외, 통신 상에 고려해야 할 것이 있다면 바로 "SOCKET" 데이터 타입입니다. 이는 UINT_PTR로 정의되어 있는데 32비트/64비트 운영체제에 따라 각각 4byte/8byte로 바뀌기 때문에 주의해야 합니다. 아니면 그냥 ^^ 8byte로 정의해도 되겠고.

기타, 보다 자세한 구현 사항은 예제 코드를 참고하시면 되겠습니다. ^^



[이 글에 대해서 여러분들과 의견을 공유하고 싶습니다. 틀리거나 미흡한 부분 또는 의문 사항이 있으시면 언제든 댓글 남겨주십시오.]

[연관 글]






[최초 등록일: ]
[최종 수정일: 8/2/2024]

Creative Commons License
이 저작물은 크리에이티브 커먼즈 코리아 저작자표시-비영리-변경금지 2.0 대한민국 라이센스에 따라 이용하실 수 있습니다.
by SeongTae Jeong, mailto:techsharer at outlook.com

비밀번호

댓글 작성자
 



2011-11-15 04시07분
[Lyn] 음... SOCKET 형은 고민 할 필요 없이 C++ 처럼 UIntPtr 로 선언 하면 되지 않을까요?

어차피 연산이 필요한 값도 아니고, UIntPtr이라면 프로젝트 type 에 따라서 4<->8로 알아서 바뀌어 줄텐데
[guest]
2011-11-15 08시00분
예제를 안 본 것 같군요. ^^ 예제 보시고... 다시 질문해 주세요.
정성태

... 46  47  48  49  50  51  52  [53]  54  55  56  57  58  59  60  ...
NoWriterDateCnt.TitleFile(s)
12614정성태4/24/202118388개발 환경 구성: 570. C# - Azure AD 인증을 지원하는 ASP.NET Core/5+ 웹 애플리케이션 예제 구성 [4]파일 다운로드1
12613정성태4/23/202116718.NET Framework: 1048. C# - ETW 이벤트의 Keywords에 속한 EventId 구하는 방법 (2) 관리 코드파일 다운로드1
12612정성태4/23/202116584.NET Framework: 1047. C# - ETW 이벤트의 Keywords에 속한 EventId 구하는 방법 (1) PInvoke파일 다운로드1
12611정성태4/22/202115461오류 유형: 711. 닷넷 EXE 실행 오류 - Mixed mode assembly is build against version 'v2.0.50727' of the runtime
12610정성태4/22/202115381.NET Framework: 1046. C# - 컴파일 시점에 참조할 수 없는 타입을 포함한 이벤트 핸들러를 Reflection을 이용해 구독하는 방법파일 다운로드1
12609정성태4/22/202117892.NET Framework: 1045. C# - 런타임 시점에 이벤트 핸들러를 만들어 Reflection을 이용해 구독하는 방법파일 다운로드1
12608정성태4/21/202118444.NET Framework: 1044. C# - Generic Host를 이용해 .NET 5로 리눅스 daemon 프로그램 만드는 방법 [9]파일 다운로드1
12607정성태4/21/202115662.NET Framework: 1043. C# - 실행 시점에 동적으로 Delegate 타입을 만드는 방법파일 다운로드1
12606정성태4/21/202121559.NET Framework: 1042. C# - enum 값을 int로 암시적(implicit) 형변환하는 방법? [2]파일 다운로드1
12605정성태4/18/202116899.NET Framework: 1041. C# - AssemblyID, ModuleID를 관리 코드에서 구하는 방법파일 다운로드1
12604정성태4/18/202114791VS.NET IDE: 163. 비주얼 스튜디오 속성 창의 "Build(빌드)" / "Configuration(구성)"에서의 "활성" 의미
12603정성태4/16/202116460VS.NET IDE: 162. 비주얼 스튜디오 - 상속받은 컨트롤이 디자인 창에서 지원되지 않는 문제
12602정성태4/16/202117521VS.NET IDE: 161. x64 DLL 프로젝트의 컨트롤이 Visual Studio의 Designer에서 보이지 않는 문제 [1]
12601정성태4/15/202116548.NET Framework: 1040. C# - REST API 대신 github 클라이언트 라이브러리를 통해 프로그래밍으로 접근
12600정성태4/15/202116794.NET Framework: 1039. C# - Kubeconfig의 token 설정 및 인증서 구성을 자동화하는 프로그램
12599정성태4/14/202117562.NET Framework: 1038. C# - 인증서 및 키 파일로부터 pfx/p12 파일을 생성하는 방법파일 다운로드1
12598정성태4/14/202118141.NET Framework: 1037. openssl의 PEM 개인키 파일을 .NET RSACryptoServiceProvider에서 사용하는 방법 (2)파일 다운로드1
12597정성태4/13/202117728개발 환경 구성: 569. csproj의 내용을 공통 설정할 수 있는 Directory.Build.targets / Directory.Build.props 파일
12596정성태4/12/202117062개발 환경 구성: 568. Windows의 80 포트 점유를 해제하는 방법
12595정성태4/12/202116808.NET Framework: 1036. SQL 서버 - varbinary 타입에 대한 문자열의 CAST, CONVERT 변환을 C# 코드로 구현
12594정성태4/11/202116265.NET Framework: 1035. C# - kubectl 명령어 또는 REST API 대신 Kubernetes 클라이언트 라이브러리를 통해 프로그래밍으로 접근 [1]파일 다운로드1
12593정성태4/10/202117306개발 환경 구성: 567. Docker Desktop for Windows - kubectl proxy 없이 k8s 대시보드 접근 방법
12592정성태4/10/202116810개발 환경 구성: 566. Docker Desktop for Windows - k8s dashboard의 Kubeconfig 로그인 및 Skip 방법
12591정성태4/9/202120725.NET Framework: 1034. C# - byte 배열을 Hex(16진수) 문자열로 고속 변환하는 방법 [2]파일 다운로드1
12590정성태4/9/202116862.NET Framework: 1033. C# - .NET 4.0 이하에서 Console.IsInputRedirected 구현 [1]
12589정성태4/8/202118092.NET Framework: 1032. C# - Environment.OSVersion의 문제점 및 윈도우 운영체제의 버전을 구하는 다양한 방법 [1]
... 46  47  48  49  50  51  52  [53]  54  55  56  57  58  59  60  ...