VirtualBox

source: vbox/trunk/src/VBox/Runtime/r3/win/localipc-win.cpp@ 49038

Last change on this file since 49038 was 47976, checked in by vboxsync, 11 years ago

Spaces.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 42.9 KB
Line 
1/* $Id: localipc-win.cpp 47976 2013-08-21 16:16:10Z vboxsync $ */
2/** @file
3 * IPRT - Local IPC, Windows Implementation Using Named Pipes.
4 */
5
6/*
7 * Copyright (C) 2008-2013 Oracle Corporation
8 *
9 * This file is part of VirtualBox Open Source Edition (OSE), as
10 * available from http://www.virtualbox.org. This file is free software;
11 * you can redistribute it and/or modify it under the terms of the GNU
12 * General Public License (GPL) as published by the Free Software
13 * Foundation, in version 2 as it comes in the "COPYING" file of the
14 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
15 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
16 *
17 * The contents of this file may alternatively be used under the terms
18 * of the Common Development and Distribution License Version 1.0
19 * (CDDL) only, as it comes in the "COPYING.CDDL" file of the
20 * VirtualBox OSE distribution, in which case the provisions of the
21 * CDDL are applicable instead of those of the GPL.
22 *
23 * You may elect to license modified versions of this file under the
24 * terms and conditions of either the GPL or the CDDL or both.
25 */
26
27/*******************************************************************************
28* Header Files *
29*******************************************************************************/
30/*
31 * We have to force NT 5.0 here because of
32 * ConvertStringSecurityDescriptorToSecurityDescriptor. Note that because of
33 * FILE_FLAG_FIRST_PIPE_INSTANCE this code actually requires W2K SP2+.
34 */
35#ifndef _WIN32_WINNT
36# define _WIN32_WINNT 0x0500 /* for ConvertStringSecurityDescriptorToSecurityDescriptor */
37#elif _WIN32_WINNT < 0x0500
38# undef _WIN32_WINNT
39# define _WIN32_WINNT 0x0500
40#endif
41#include <Windows.h>
42#include <sddl.h>
43
44#include <iprt/alloc.h>
45#include <iprt/asm.h>
46#include <iprt/assert.h>
47#include <iprt/critsect.h>
48#include <iprt/err.h>
49#include <iprt/ldr.h>
50#include <iprt/localipc.h>
51#include <iprt/param.h>
52#include <iprt/string.h>
53#include <iprt/thread.h>
54#include <iprt/time.h>
55
56#include "internal/magics.h"
57
58
59/*******************************************************************************
60* Defined Constants And Macros *
61*******************************************************************************/
62/** Pipe prefix string. */
63#define RTLOCALIPC_WIN_PREFIX "\\\\.\\pipe\\IPRT-"
64
65/** DACL for block all network access and local users other than the creator/owner.
66 *
67 * ACE format: (ace_type;ace_flags;rights;object_guid;inherit_object_guid;account_sid)
68 *
69 * Note! FILE_GENERIC_WRITE (SDDL_FILE_WRITE) is evil here because it includes
70 * the FILE_CREATE_PIPE_INSTANCE(=FILE_APPEND_DATA) flag. Thus the hardcoded
71 * value 0x0012019b in the client ACE. The server-side still needs
72 * setting FILE_CREATE_PIPE_INSTANCE although.
73 * It expands to:
74 * 0x00000001 - FILE_READ_DATA
75 * 0x00000008 - FILE_READ_EA
76 * 0x00000080 - FILE_READ_ATTRIBUTES
77 * 0x00020000 - READ_CONTROL
78 * 0x00100000 - SYNCHRONIZE
79 * 0x00000002 - FILE_WRITE_DATA
80 * 0x00000010 - FILE_WRITE_EA
81 * 0x00000100 - FILE_WRITE_ATTRIBUTES
82 * = 0x0012019b (client)
83 * + (only for server):
84 * 0x00000004 - FILE_CREATE_PIPE_INSTANCE
85 * = 0x0012019f
86 *
87 * @todo Triple check this!
88 * @todo EVERYONE -> AUTHENTICATED USERS or something more appropriate?
89 * @todo Have trouble allowing the owner FILE_CREATE_PIPE_INSTANCE access, so for now I'm hacking
90 * it just to get progress - the service runs as local system.
91 * The CREATOR OWNER and PERSONAL SELF works (the former is only involved in inheriting
92 * it seems, which is why it won't work. The latter I've no idea about. Perhaps the solution
93 * is to go the annoying route of OpenProcessToken, QueryTokenInformation,
94 * ConvertSidToStringSid and then use the result... Suggestions are very welcome
95 */
96#define RTLOCALIPC_WIN_SDDL_BASE \
97 SDDL_DACL SDDL_DELIMINATOR \
98 SDDL_ACE_BEGIN SDDL_ACCESS_DENIED ";;" SDDL_GENERIC_ALL ";;;" SDDL_NETWORK SDDL_ACE_END \
99 SDDL_ACE_BEGIN SDDL_ACCESS_ALLOWED ";;" SDDL_FILE_ALL ";;;" SDDL_LOCAL_SYSTEM SDDL_ACE_END
100
101#define RTLOCALIPC_WIN_SDDL_SERVER \
102 RTLOCALIPC_WIN_SDDL_BASE \
103 SDDL_ACE_BEGIN SDDL_ACCESS_ALLOWED ";;" "0x0012019f" ";;;" SDDL_EVERYONE SDDL_ACE_END
104
105#define RTLOCALIPC_WIN_SDDL_CLIENT \
106 RTLOCALIPC_WIN_SDDL_BASE \
107 SDDL_ACE_BEGIN SDDL_ACCESS_ALLOWED ";;" "0x0012019b" ";;;" SDDL_EVERYONE SDDL_ACE_END
108
109// SDDL_ACE_BEGIN SDDL_ACCESS_ALLOWED ";;" SDDL_GENERIC_ALL ";;;" SDDL_PERSONAL_SELF SDDL_ACE_END \
110// SDDL_ACE_BEGIN SDDL_ACCESS_ALLOWED ";CIOI;" SDDL_GENERIC_ALL ";;;" SDDL_CREATOR_OWNER SDDL_ACE_END
111// SDDL_ACE_BEGIN SDDL_ACCESS_ALLOWED ";;" "0x0012019b" ";;;" SDDL_EVERYONE SDDL_ACE_END
112// SDDL_ACE_BEGIN SDDL_ACCESS_ALLOWED ";;" SDDL_FILE_ALL ";;;" SDDL_LOCAL_SYSTEM SDDL_ACE_END
113
114
115/*******************************************************************************
116* Structures and Typedefs *
117*******************************************************************************/
118/**
119 * Local IPC service instance, Windows.
120 */
121typedef struct RTLOCALIPCSERVERINT
122{
123 /** The magic (RTLOCALIPCSERVER_MAGIC). */
124 uint32_t u32Magic;
125 /** The creation flags. */
126 uint32_t fFlags;
127 /** Critical section protecting the structure. */
128 RTCRITSECT CritSect;
129 /** The number of references to the instance.
130 * @remarks The reference counting isn't race proof. */
131 uint32_t volatile cRefs;
132 /** Indicates that there is a pending cancel request. */
133 bool volatile fCancelled;
134 /** The name pipe handle. */
135 HANDLE hNmPipe;
136 /** The handle to the event object we're using for overlapped I/O. */
137 HANDLE hEvent;
138 /** The overlapped I/O structure. */
139 OVERLAPPED OverlappedIO;
140 /** The pipe name. */
141 char szName[1];
142} RTLOCALIPCSERVERINT;
143/** Pointer to a local IPC server instance (Windows). */
144typedef RTLOCALIPCSERVERINT *PRTLOCALIPCSERVERINT;
145
146
147/**
148 * Local IPC session instance, Windows.
149 */
150typedef struct RTLOCALIPCSESSIONINT
151{
152 /** The magic (RTLOCALIPCSESSION_MAGIC). */
153 uint32_t u32Magic;
154 /** Critical section protecting the structure. */
155 RTCRITSECT CritSect;
156 /** The number of references to the instance.
157 * @remarks The reference counting isn't race proof. */
158 uint32_t volatile cRefs;
159 /** Set if there is already pending I/O. */
160 bool fIOPending;
161 /** Set if the zero byte read that the poll code using is pending. */
162 bool fZeroByteRead;
163 /** Indicates that there is a pending cancel request. */
164 bool volatile fCancelled;
165 /** The name pipe handle. */
166 HANDLE hNmPipe;
167 /** The handle to the event object we're using for overlapped I/O. */
168 HANDLE hEvent;
169 /** The overlapped I/O structure. */
170 OVERLAPPED OverlappedIO;
171 /** Bounce buffer for writes. */
172 uint8_t *pbBounceBuf;
173 /** Amount of used buffer space. */
174 size_t cbBounceBufUsed;
175 /** Amount of allocated buffer space. */
176 size_t cbBounceBufAlloc;
177 /** Buffer for the zero byte read.
178 * Used in RTLocalIpcSessionWaitForData(). */
179 uint8_t abBuf[8];
180} RTLOCALIPCSESSIONINT;
181/** Pointer to a local IPC session instance (Windows). */
182typedef RTLOCALIPCSESSIONINT *PRTLOCALIPCSESSIONINT;
183
184typedef BOOL WINAPI FNCONVERTSTRINGSECURITYDESCRIPTORTOSECURITYDESCRIPTOR(LPCTSTR, DWORD, PSECURITY_DESCRIPTOR, PULONG);
185typedef FNCONVERTSTRINGSECURITYDESCRIPTORTOSECURITYDESCRIPTOR
186 *PFNCONVERTSTRINGSECURITYDESCRIPTORTOSECURITYDESCRIPTOR; /* No, nobody fell on the keyboard, really! */
187
188
189/*******************************************************************************
190* Internal Functions *
191*******************************************************************************/
192static int rtLocalIpcWinCreateSession(PRTLOCALIPCSESSION phClientSession, HANDLE hNmPipeSession);
193
194
195/**
196 * Builds and allocates the security descriptor required for securing the local pipe.
197 *
198 * @return IPRT status code.
199 * @param ppDesc Where to store the allocated security descriptor on success.
200 * Must be free'd using LocalFree().
201 */
202static int rtLocalIpcServerWinAllocSecurityDescriptior(PSECURITY_DESCRIPTOR *ppDesc, bool fServer)
203{
204 /** @todo Stuff this into RTInitOnce? Later. */
205 PFNCONVERTSTRINGSECURITYDESCRIPTORTOSECURITYDESCRIPTOR
206 pfnConvertStringSecurityDescriptorToSecurityDescriptor = NULL;
207
208 RTLDRMOD hAdvApi32 = NIL_RTLDRMOD;
209 int rc = RTLdrLoadSystem("Advapi32.dll", true /*fNoUnload*/, &hAdvApi32);
210 if (RT_SUCCESS(rc))
211 rc = RTLdrGetSymbol(hAdvApi32, "ConvertStringSecurityDescriptorToSecurityDescriptorW",
212 (void**)&pfnConvertStringSecurityDescriptorToSecurityDescriptor);
213
214 PSECURITY_DESCRIPTOR pSecDesc = NULL;
215 if (RT_SUCCESS(rc))
216 {
217 AssertPtr(pfnConvertStringSecurityDescriptorToSecurityDescriptor);
218
219 /*
220 * We'll create a security descriptor from a SDDL that denies
221 * access to network clients (this is local IPC after all), it
222 * makes some further restrictions to prevent non-authenticated
223 * users from screwing around.
224 */
225 PRTUTF16 pwszSDDL;
226 rc = RTStrToUtf16(fServer
227 ? RTLOCALIPC_WIN_SDDL_SERVER : RTLOCALIPC_WIN_SDDL_CLIENT, &pwszSDDL);
228 if (RT_SUCCESS(rc))
229 {
230 if (!pfnConvertStringSecurityDescriptorToSecurityDescriptor((LPCTSTR)pwszSDDL,
231 SDDL_REVISION_1,
232 &pSecDesc,
233 NULL))
234 {
235 rc = RTErrConvertFromWin32(GetLastError());
236 }
237
238 RTUtf16Free(pwszSDDL);
239 }
240 }
241 else
242 {
243 /* Windows OSes < W2K SP2 not supported for now, bail out. */
244 /** @todo Implement me! */
245 rc = VERR_NOT_SUPPORTED;
246 }
247
248 if (hAdvApi32 != NIL_RTLDRMOD)
249 RTLdrClose(hAdvApi32);
250
251 if (RT_SUCCESS(rc))
252 {
253 AssertPtr(pSecDesc);
254 *ppDesc = pSecDesc;
255 }
256
257 return rc;
258}
259
260/**
261 * Creates a named pipe instance.
262 *
263 * This is used by both RTLocalIpcServerCreate and RTLocalIpcServerListen.
264 *
265 * @return IPRT status code.
266 * @param phNmPipe Where to store the named pipe handle on success. This
267 * will be set to INVALID_HANDLE_VALUE on failure.
268 * @param pszFullPipeName The full named pipe name.
269 * @param fFirst Set on the first call (from RTLocalIpcServerCreate), otherwise clear.
270 * Governs the FILE_FLAG_FIRST_PIPE_INSTANCE flag.
271 */
272static int rtLocalIpcServerWinCreatePipeInstance(PHANDLE phNmPipe, const char *pszFullPipeName, bool fFirst)
273{
274 *phNmPipe = INVALID_HANDLE_VALUE;
275
276 PSECURITY_DESCRIPTOR pSecDesc;
277 int rc = rtLocalIpcServerWinAllocSecurityDescriptior(&pSecDesc, fFirst /* Server? */);
278 if (RT_SUCCESS(rc))
279 {
280 SECURITY_ATTRIBUTES SecAttrs;
281 SecAttrs.nLength = sizeof(SECURITY_ATTRIBUTES);
282 SecAttrs.lpSecurityDescriptor = pSecDesc;
283 SecAttrs.bInheritHandle = FALSE;
284
285 DWORD fOpenMode = PIPE_ACCESS_DUPLEX
286 | PIPE_WAIT
287 | FILE_FLAG_OVERLAPPED;
288
289 bool fSupportsFirstInstance = false;
290
291 OSVERSIONINFOEX OSInfoEx;
292 RT_ZERO(OSInfoEx);
293 OSInfoEx.dwOSVersionInfoSize = sizeof(OSVERSIONINFOEX);
294 if ( GetVersionEx((LPOSVERSIONINFO) &OSInfoEx)
295 && OSInfoEx.dwPlatformId == VER_PLATFORM_WIN32_NT)
296 {
297 if ( /* Vista+. */
298 OSInfoEx.dwMajorVersion >= 6
299 /* Windows XP+. */
300 || ( OSInfoEx.dwMajorVersion == 5
301 && OSInfoEx.dwMinorVersion > 0)
302 /* Windows 2000. */
303 || ( OSInfoEx.dwMajorVersion == 5
304 && OSInfoEx.dwMinorVersion == 0
305 && OSInfoEx.wServicePackMajor >= 2))
306 {
307 /* Requires at least W2K (5.0) SP2+. This is non-fatal. */
308 fSupportsFirstInstance = true;
309 }
310 }
311
312 if (fFirst && fSupportsFirstInstance)
313 fOpenMode |= FILE_FLAG_FIRST_PIPE_INSTANCE;
314
315 HANDLE hNmPipe = CreateNamedPipe(pszFullPipeName, /* lpName */
316 fOpenMode, /* dwOpenMode */
317 PIPE_TYPE_BYTE, /* dwPipeMode */
318 PIPE_UNLIMITED_INSTANCES, /* nMaxInstances */
319 PAGE_SIZE, /* nOutBufferSize (advisory) */
320 PAGE_SIZE, /* nInBufferSize (ditto) */
321 30*1000, /* nDefaultTimeOut = 30 sec */
322 &SecAttrs); /* lpSecurityAttributes */
323 LocalFree(pSecDesc);
324 if (hNmPipe != INVALID_HANDLE_VALUE)
325 {
326 *phNmPipe = hNmPipe;
327 }
328 else
329 rc = RTErrConvertFromWin32(GetLastError());
330 }
331
332 return rc;
333}
334
335
336RTDECL(int) RTLocalIpcServerCreate(PRTLOCALIPCSERVER phServer, const char *pszName, uint32_t fFlags)
337{
338 /*
339 * Basic parameter validation.
340 */
341 AssertPtrReturn(phServer, VERR_INVALID_POINTER);
342 AssertPtrReturn(pszName, VERR_INVALID_POINTER);
343 AssertReturn(*pszName, VERR_INVALID_PARAMETER);
344 AssertReturn(!(fFlags & ~(RTLOCALIPC_FLAGS_VALID_MASK)), VERR_INVALID_PARAMETER);
345 AssertReturn((fFlags & RTLOCALIPC_FLAGS_MULTI_SESSION), VERR_INVALID_PARAMETER); /** @todo Implement !RTLOCALIPC_FLAGS_MULTI_SESSION */
346
347 /*
348 * Allocate and initialize the instance data.
349 */
350 size_t cchName = strlen(pszName);
351 size_t cch = RT_OFFSETOF(RTLOCALIPCSERVERINT, szName[cchName + sizeof(RTLOCALIPC_WIN_PREFIX)]);
352 PRTLOCALIPCSERVERINT pThis = (PRTLOCALIPCSERVERINT)RTMemAlloc(cch);
353 if (!pThis)
354 return VERR_NO_MEMORY;
355 pThis->u32Magic = RTLOCALIPCSERVER_MAGIC;
356 pThis->cRefs = 1; /* the one we return */
357 pThis->fCancelled = false;
358 memcpy(pThis->szName, RTLOCALIPC_WIN_PREFIX, sizeof(RTLOCALIPC_WIN_PREFIX) - 1);
359 memcpy(&pThis->szName[sizeof(RTLOCALIPC_WIN_PREFIX) - 1], pszName, cchName + 1);
360 int rc = RTCritSectInit(&pThis->CritSect);
361 if (RT_SUCCESS(rc))
362 {
363 pThis->hEvent = CreateEvent(NULL /*lpEventAttributes*/, TRUE /*bManualReset*/,
364 FALSE /*bInitialState*/, NULL /*lpName*/);
365 if (pThis->hEvent != NULL)
366 {
367 RT_ZERO(pThis->OverlappedIO);
368 pThis->OverlappedIO.Internal = STATUS_PENDING;
369 pThis->OverlappedIO.hEvent = pThis->hEvent;
370
371 rc = rtLocalIpcServerWinCreatePipeInstance(&pThis->hNmPipe,
372 pThis->szName, true /* fFirst */);
373 if (RT_SUCCESS(rc))
374 {
375 *phServer = pThis;
376 return VINF_SUCCESS;
377 }
378
379 BOOL fRc = CloseHandle(pThis->hEvent);
380 AssertMsg(fRc, ("%d\n", GetLastError())); NOREF(fRc);
381 }
382 else
383 rc = RTErrConvertFromWin32(GetLastError());
384
385 int rc2 = RTCritSectDelete(&pThis->CritSect);
386 AssertRC(rc2);
387 }
388 RTMemFree(pThis);
389 return rc;
390}
391
392
393/**
394 * Call when the reference count reaches 0.
395 * Caller owns the critsect.
396 * @param pThis The instance to destroy.
397 */
398static void rtLocalIpcServerWinDestroy(PRTLOCALIPCSERVERINT pThis)
399{
400 BOOL fRc = CloseHandle(pThis->hNmPipe);
401 AssertMsg(fRc, ("%d\n", GetLastError())); NOREF(fRc);
402 pThis->hNmPipe = INVALID_HANDLE_VALUE;
403
404 fRc = CloseHandle(pThis->hEvent);
405 AssertMsg(fRc, ("%d\n", GetLastError())); NOREF(fRc);
406 pThis->hEvent = NULL;
407
408 RTCritSectLeave(&pThis->CritSect);
409 RTCritSectDelete(&pThis->CritSect);
410
411 RTMemFree(pThis);
412}
413
414
415RTDECL(int) RTLocalIpcServerDestroy(RTLOCALIPCSERVER hServer)
416{
417 /*
418 * Validate input.
419 */
420 if (hServer == NIL_RTLOCALIPCSERVER)
421 return VINF_SUCCESS;
422 PRTLOCALIPCSERVERINT pThis = (PRTLOCALIPCSERVERINT)hServer;
423 AssertPtrReturn(pThis, VERR_INVALID_HANDLE);
424 AssertReturn(pThis->u32Magic == RTLOCALIPCSERVER_MAGIC, VERR_INVALID_MAGIC);
425
426 /*
427 * Cancel any thread currently busy using the server,
428 * leaving the cleanup to it.
429 */
430 RTCritSectEnter(&pThis->CritSect);
431 ASMAtomicUoWriteU32(&pThis->u32Magic, ~RTLOCALIPCSERVER_MAGIC);
432 ASMAtomicUoWriteBool(&pThis->fCancelled, true);
433 Assert(pThis->cRefs);
434 pThis->cRefs--;
435
436 if (pThis->cRefs)
437 {
438 BOOL fRc = SetEvent(pThis->hEvent);
439 AssertMsg(fRc, ("%d\n", GetLastError())); NOREF(fRc);
440
441 RTCritSectLeave(&pThis->CritSect);
442 }
443 else
444 rtLocalIpcServerWinDestroy(pThis);
445
446 return VINF_SUCCESS;
447}
448
449
450RTDECL(int) RTLocalIpcServerListen(RTLOCALIPCSERVER hServer, PRTLOCALIPCSESSION phClientSession)
451{
452 /*
453 * Validate input.
454 */
455 PRTLOCALIPCSERVERINT pThis = (PRTLOCALIPCSERVERINT)hServer;
456 AssertPtrReturn(pThis, VERR_INVALID_HANDLE);
457 AssertReturn(pThis->u32Magic == RTLOCALIPCSERVER_MAGIC, VERR_INVALID_MAGIC);
458
459 /*
460 * Enter the critsect before inspecting the object further.
461 */
462 int rc;
463 RTCritSectEnter(&pThis->CritSect);
464 if (pThis->fCancelled)
465 {
466 pThis->fCancelled = false;
467 rc = VERR_CANCELLED;
468 RTCritSectLeave(&pThis->CritSect);
469 }
470 else
471 {
472 pThis->cRefs++;
473 ResetEvent(pThis->hEvent);
474 RTCritSectLeave(&pThis->CritSect);
475
476 /*
477 * Try connect a client. We need to use overlapped I/O here because
478 * of the cancellation done by RTLocalIpcServerCancel and RTLocalIpcServerDestroy.
479 */
480 SetLastError(NO_ERROR);
481 BOOL fRc = ConnectNamedPipe(pThis->hNmPipe, &pThis->OverlappedIO);
482 DWORD dwErr = fRc ? NO_ERROR : GetLastError();
483 if ( !fRc
484 && dwErr == ERROR_IO_PENDING)
485 {
486 WaitForSingleObject(pThis->hEvent, INFINITE);
487 DWORD dwIgnored;
488 fRc = GetOverlappedResult(pThis->hNmPipe, &pThis->OverlappedIO, &dwIgnored, FALSE /* bWait*/);
489 dwErr = fRc ? NO_ERROR : GetLastError();
490 }
491
492 RTCritSectEnter(&pThis->CritSect);
493 if ( !pThis->fCancelled /* Event signalled but not cancelled? */
494 && pThis->u32Magic == RTLOCALIPCSERVER_MAGIC)
495 {
496 /*
497 * Still alive, some error or an actual client.
498 *
499 * If it's the latter we'll have to create a new pipe instance that
500 * replaces the current one for the server. The current pipe instance
501 * will be assigned to the client session.
502 */
503 if ( fRc
504 || dwErr == ERROR_PIPE_CONNECTED)
505 {
506 HANDLE hNmPipe;
507 rc = rtLocalIpcServerWinCreatePipeInstance(&hNmPipe, pThis->szName, false /* fFirst */);
508 if (RT_SUCCESS(rc))
509 {
510 HANDLE hNmPipeSession = pThis->hNmPipe; /* consumed */
511 pThis->hNmPipe = hNmPipe;
512 rc = rtLocalIpcWinCreateSession(phClientSession, hNmPipeSession);
513 }
514 else
515 {
516 /*
517 * We failed to create a new instance for the server, disconnect
518 * the client and fail. Don't try service the client here.
519 */
520 fRc = DisconnectNamedPipe(pThis->hNmPipe);
521 AssertMsg(fRc, ("%d\n", GetLastError()));
522 }
523 }
524 else
525 rc = RTErrConvertFromWin32(dwErr);
526 }
527 else
528 {
529 /*
530 * Cancelled or destroyed.
531 *
532 * Cancel the overlapped io if it didn't complete (must be done
533 * in the this thread) or disconnect the client.
534 */
535 if ( fRc
536 || dwErr == ERROR_PIPE_CONNECTED)
537 fRc = DisconnectNamedPipe(pThis->hNmPipe);
538 else if (dwErr == ERROR_IO_PENDING)
539 fRc = CancelIo(pThis->hNmPipe);
540 else
541 fRc = TRUE;
542 AssertMsg(fRc, ("%d\n", GetLastError()));
543 rc = VERR_CANCELLED;
544 }
545
546 pThis->cRefs--;
547 if (pThis->cRefs)
548 RTCritSectLeave(&pThis->CritSect);
549 else
550 rtLocalIpcServerWinDestroy(pThis);
551 }
552
553 return rc;
554}
555
556
557RTDECL(int) RTLocalIpcServerCancel(RTLOCALIPCSERVER hServer)
558{
559 /*
560 * Validate input.
561 */
562 PRTLOCALIPCSERVERINT pThis = (PRTLOCALIPCSERVERINT)hServer;
563 AssertPtrReturn(pThis, VERR_INVALID_HANDLE);
564 AssertReturn(pThis->u32Magic == RTLOCALIPCSERVER_MAGIC, VERR_INVALID_MAGIC);
565
566 /*
567 * Enter the critical section, then set the cancellation flag
568 * and signal the event (to wake up anyone in/at WaitForSingleObject).
569 */
570 int rc = RTCritSectEnter(&pThis->CritSect);
571 if (RT_SUCCESS(rc))
572 {
573 ASMAtomicUoWriteBool(&pThis->fCancelled, true);
574 BOOL fRc = SetEvent(pThis->hEvent);
575 AssertMsg(fRc, ("%d\n", GetLastError())); NOREF(fRc);
576
577 rc = RTCritSectLeave(&pThis->CritSect);
578 }
579
580 return rc;
581}
582
583
584/**
585 * Create a session instance.
586 *
587 * @returns IPRT status code.
588 *
589 * @param phClientSession Where to store the session handle on success.
590 * @param hNmPipeSession The named pipe handle. This will be consumed by this session, meaning on failure
591 * to create the session it will be closed.
592 */
593static int rtLocalIpcWinCreateSession(PRTLOCALIPCSESSION phClientSession, HANDLE hNmPipeSession)
594{
595 AssertPtrReturn(phClientSession, VERR_INVALID_POINTER);
596 AssertReturn(hNmPipeSession != INVALID_HANDLE_VALUE, VERR_INVALID_HANDLE);
597
598 int rc;
599
600 /*
601 * Allocate and initialize the session instance data.
602 */
603 PRTLOCALIPCSESSIONINT pThis = (PRTLOCALIPCSESSIONINT)RTMemAlloc(sizeof(*pThis));
604 if (pThis)
605 {
606 pThis->u32Magic = RTLOCALIPCSESSION_MAGIC;
607 pThis->cRefs = 1; /* our ref */
608 pThis->fCancelled = false;
609 pThis->fIOPending = false;
610 pThis->fZeroByteRead = false;
611 pThis->hNmPipe = hNmPipeSession;
612
613 rc = RTCritSectInit(&pThis->CritSect);
614 if (RT_SUCCESS(rc))
615 {
616 pThis->hEvent = CreateEvent(NULL /*lpEventAttributes*/, TRUE /*bManualReset*/,
617 FALSE /*bInitialState*/, NULL /*lpName*/);
618 if (pThis->hEvent != NULL)
619 {
620 RT_ZERO(pThis->OverlappedIO);
621 pThis->OverlappedIO.Internal = STATUS_PENDING;
622 pThis->OverlappedIO.hEvent = pThis->hEvent;
623
624 *phClientSession = pThis;
625 return VINF_SUCCESS;
626 }
627
628 /* bail out */
629 rc = RTErrConvertFromWin32(GetLastError());
630 RTCritSectDelete(&pThis->CritSect);
631 }
632 RTMemFree(pThis);
633 }
634 else
635 rc = VERR_NO_MEMORY;
636
637 BOOL fRc = CloseHandle(hNmPipeSession);
638 AssertMsg(fRc, ("%d\n", GetLastError())); NOREF(fRc);
639 return rc;
640}
641
642RTDECL(int) RTLocalIpcSessionConnect(PRTLOCALIPCSESSION phSession, const char *pszName, uint32_t fFlags)
643{
644 AssertPtrReturn(phSession, VERR_INVALID_POINTER);
645 AssertPtrReturn(pszName, VERR_INVALID_POINTER);
646 AssertReturn(*pszName, VERR_INVALID_PARAMETER);
647 AssertReturn(!fFlags, VERR_INVALID_PARAMETER); /* Flags currently unused, must be 0. */
648
649 PRTLOCALIPCSESSIONINT pThis = (PRTLOCALIPCSESSIONINT)RTMemAlloc(sizeof(*pThis));
650 if (!pThis)
651 return VERR_NO_MEMORY;
652
653 pThis->u32Magic = RTLOCALIPCSESSION_MAGIC;
654 pThis->cRefs = 1; /* The one we return. */
655 pThis->fIOPending = false;
656 pThis->fZeroByteRead = false;
657 pThis->fCancelled = false;
658 pThis->pbBounceBuf = NULL;
659 pThis->cbBounceBufAlloc = 0;
660 pThis->cbBounceBufUsed = 0;
661
662 int rc = RTCritSectInit(&pThis->CritSect);
663 if (RT_SUCCESS(rc))
664 {
665 pThis->hEvent = CreateEvent(NULL /*lpEventAttributes*/, TRUE /*bManualReset*/,
666 FALSE /*bInitialState*/, NULL /*lpName*/);
667 if (pThis->hEvent != NULL)
668 {
669 RT_ZERO(pThis->OverlappedIO);
670 pThis->OverlappedIO.Internal = STATUS_PENDING;
671 pThis->OverlappedIO.hEvent = pThis->hEvent;
672
673 PSECURITY_DESCRIPTOR pSecDesc;
674 rc = rtLocalIpcServerWinAllocSecurityDescriptior(&pSecDesc, false /* Client */);
675 if (RT_SUCCESS(rc))
676 {
677 char *pszPipe;
678 if (RTStrAPrintf(&pszPipe, "%s%s", RTLOCALIPC_WIN_PREFIX, pszName))
679 {
680 SECURITY_ATTRIBUTES SecAttrs;
681 SecAttrs.nLength = sizeof(SECURITY_ATTRIBUTES);
682 SecAttrs.lpSecurityDescriptor = pSecDesc;
683 SecAttrs.bInheritHandle = FALSE;
684
685 HANDLE hPipe = CreateFile(pszPipe, /* pipe name */
686 GENERIC_READ /* read and write access */
687 | GENERIC_WRITE,
688 0, /* no sharing */
689 &SecAttrs, /* lpSecurityAttributes */
690 OPEN_EXISTING, /* opens existing pipe */
691 FILE_FLAG_OVERLAPPED, /* default attributes */
692 NULL); /* no template file */
693 RTStrFree(pszPipe);
694
695 if (hPipe != INVALID_HANDLE_VALUE)
696 {
697 LocalFree(pSecDesc);
698
699 pThis->hNmPipe = hPipe;
700 *phSession = pThis;
701 return VINF_SUCCESS;
702 }
703 else
704 rc = RTErrConvertFromWin32(GetLastError());
705 }
706 else
707 rc = VERR_NO_MEMORY;
708
709 LocalFree(pSecDesc);
710 }
711
712 BOOL fRc = CloseHandle(pThis->hEvent);
713 AssertMsg(fRc, ("%d\n", GetLastError())); NOREF(fRc);
714 }
715 else
716 rc = RTErrConvertFromWin32(GetLastError());
717
718 int rc2 = RTCritSectDelete(&pThis->CritSect);
719 AssertRC(rc2);
720 }
721
722 RTMemFree(pThis);
723 return rc;
724}
725
726
727/**
728 * Call when the reference count reaches 0.
729 * Caller owns the critsect.
730 * @param pThis The instance to destroy.
731 */
732static void rtLocalIpcSessionWinDestroy(PRTLOCALIPCSESSIONINT pThis)
733{
734 BOOL fRc = CloseHandle(pThis->hNmPipe);
735 AssertMsg(fRc, ("%d\n", GetLastError())); NOREF(fRc);
736 pThis->hNmPipe = INVALID_HANDLE_VALUE;
737
738 fRc = CloseHandle(pThis->hEvent);
739 AssertMsg(fRc, ("%d\n", GetLastError())); NOREF(fRc);
740 pThis->hEvent = NULL;
741
742 RTCritSectLeave(&pThis->CritSect);
743 RTCritSectDelete(&pThis->CritSect);
744
745 RTMemFree(pThis);
746}
747
748
749RTDECL(int) RTLocalIpcSessionClose(RTLOCALIPCSESSION hSession)
750{
751 /*
752 * Validate input.
753 */
754 if (hSession == NIL_RTLOCALIPCSESSION)
755 return VINF_SUCCESS;
756 PRTLOCALIPCSESSIONINT pThis = (PRTLOCALIPCSESSIONINT)hSession;
757 AssertPtrReturn(pThis, VERR_INVALID_HANDLE);
758 AssertReturn(pThis->u32Magic == RTLOCALIPCSESSION_MAGIC, VERR_INVALID_MAGIC);
759
760 /*
761 * Cancel any thread currently busy using the session,
762 * leaving the cleanup to it.
763 */
764 RTCritSectEnter(&pThis->CritSect);
765 ASMAtomicUoWriteU32(&pThis->u32Magic, ~RTLOCALIPCSESSION_MAGIC);
766 ASMAtomicUoWriteBool(&pThis->fCancelled, true);
767 pThis->cRefs--;
768
769 if (pThis->cRefs > 0)
770 {
771 BOOL fRc = SetEvent(pThis->hEvent);
772 AssertMsg(fRc, ("%d\n", GetLastError())); NOREF(fRc);
773
774 RTCritSectLeave(&pThis->CritSect);
775 }
776 else
777 rtLocalIpcSessionWinDestroy(pThis);
778
779 return VINF_SUCCESS;
780}
781
782
783RTDECL(int) RTLocalIpcSessionRead(RTLOCALIPCSESSION hSession, void *pvBuffer, size_t cbBuffer, size_t *pcbRead)
784{
785 PRTLOCALIPCSESSIONINT pThis = (PRTLOCALIPCSESSIONINT)hSession;
786 AssertPtrReturn(pThis, VERR_INVALID_HANDLE);
787 AssertReturn(pThis->u32Magic == RTLOCALIPCSESSION_MAGIC, VERR_INVALID_HANDLE);
788 AssertPtrReturn(pvBuffer, VERR_INVALID_POINTER);
789 /* pcbRead is optional. */
790
791 int rc = RTCritSectEnter(&pThis->CritSect);
792 if (RT_SUCCESS(rc))
793 {
794 /* No concurrent readers, sorry. */
795 if (pThis->cRefs == 1)
796 {
797 pThis->cRefs++;
798
799 /*
800 * If pcbRead is non-NULL this indicates the maximum number of bytes to read.
801 * If pcbRead is NULL then this is the exact number of bytes to read.
802 */
803 size_t cbToRead = pcbRead ? *pcbRead : cbBuffer;
804 size_t cbTotalRead = 0;
805 while (cbToRead > 0)
806 {
807 /*
808 * Kick of a an overlapped read. It should return immediately if
809 * there is bytes in the buffer. If not, we'll cancel it and see
810 * what we get back.
811 */
812 rc = ResetEvent(pThis->OverlappedIO.hEvent); Assert(rc == TRUE);
813 DWORD cbRead = 0;
814 pThis->fIOPending = true;
815 RTCritSectLeave(&pThis->CritSect);
816
817 if (ReadFile(pThis->hNmPipe, pvBuffer,
818 cbToRead <= ~(DWORD)0 ? (DWORD)cbToRead : ~(DWORD)0,
819 &cbRead, &pThis->OverlappedIO))
820 rc = VINF_SUCCESS;
821 else if (GetLastError() == ERROR_IO_PENDING)
822 {
823 WaitForSingleObject(pThis->OverlappedIO.hEvent, INFINITE);
824 if (GetOverlappedResult(pThis->hNmPipe, &pThis->OverlappedIO,
825 &cbRead, TRUE /*fWait*/))
826 rc = VINF_SUCCESS;
827 else
828 {
829 DWORD dwErr = GetLastError();
830 AssertMsgFailed(("err=%ld\n", dwErr));
831 rc = RTErrConvertFromWin32(dwErr);
832 }
833 }
834 else
835 {
836 DWORD dwErr = GetLastError();
837 AssertMsgFailed(("err2=%ld\n", dwErr));
838 rc = RTErrConvertFromWin32(dwErr);
839 }
840
841 RTCritSectEnter(&pThis->CritSect);
842 pThis->fIOPending = false;
843 if (RT_FAILURE(rc))
844 break;
845
846 /* Advance. */
847 cbToRead -= cbRead;
848 cbTotalRead += cbRead;
849 pvBuffer = (uint8_t *)pvBuffer + cbRead;
850 }
851
852 if (pcbRead)
853 {
854 *pcbRead = cbTotalRead;
855 if ( RT_FAILURE(rc)
856 && cbTotalRead
857 && rc != VERR_INVALID_POINTER)
858 rc = VINF_SUCCESS;
859 }
860
861 pThis->cRefs--;
862 }
863 else
864 rc = VERR_WRONG_ORDER;
865 RTCritSectLeave(&pThis->CritSect);
866 }
867
868 return rc;
869}
870
871
872/**
873 * Common worker for handling I/O completion.
874 *
875 * This is used by RTLocalIpcSessionClose and RTLocalIpcSessionWrite.
876 *
877 * @returns IPRT status code.
878 * @param pThis The pipe instance handle.
879 */
880static int rtLocalIpcSessionWriteCheckCompletion(PRTLOCALIPCSESSIONINT pThis)
881{
882 int rc;
883 DWORD dwRc = WaitForSingleObject(pThis->OverlappedIO.hEvent, 0);
884 if (dwRc == WAIT_OBJECT_0)
885 {
886 DWORD cbWritten = 0;
887 if (GetOverlappedResult(pThis->hNmPipe, &pThis->OverlappedIO, &cbWritten, TRUE))
888 {
889 for (;;)
890 {
891 if (cbWritten >= pThis->cbBounceBufUsed)
892 {
893 pThis->fIOPending = false;
894 rc = VINF_SUCCESS;
895 break;
896 }
897
898 /* resubmit the remainder of the buffer - can this actually happen? */
899 memmove(&pThis->pbBounceBuf[0], &pThis->pbBounceBuf[cbWritten], pThis->cbBounceBufUsed - cbWritten);
900 rc = ResetEvent(pThis->OverlappedIO.hEvent); Assert(rc == TRUE);
901 if (!WriteFile(pThis->hNmPipe, pThis->pbBounceBuf, (DWORD)pThis->cbBounceBufUsed,
902 &cbWritten, &pThis->OverlappedIO))
903 {
904 DWORD dwErr = GetLastError();
905 if (dwErr == ERROR_IO_PENDING)
906 rc = VINF_TRY_AGAIN;
907 else
908 {
909 pThis->fIOPending = false;
910 if (dwErr == ERROR_NO_DATA)
911 rc = VERR_BROKEN_PIPE;
912 else
913 rc = RTErrConvertFromWin32(dwErr);
914 }
915 break;
916 }
917 Assert(cbWritten > 0);
918 }
919 }
920 else
921 {
922 pThis->fIOPending = false;
923 rc = RTErrConvertFromWin32(GetLastError());
924 }
925 }
926 else if (dwRc == WAIT_TIMEOUT)
927 rc = VINF_TRY_AGAIN;
928 else
929 {
930 pThis->fIOPending = false;
931 if (dwRc == WAIT_ABANDONED)
932 rc = VERR_INVALID_HANDLE;
933 else
934 rc = RTErrConvertFromWin32(GetLastError());
935 }
936 return rc;
937}
938
939
940RTDECL(int) RTLocalIpcSessionWrite(RTLOCALIPCSESSION hSession, const void *pvBuffer, size_t cbBuffer)
941{
942 PRTLOCALIPCSESSIONINT pThis = (PRTLOCALIPCSESSIONINT)hSession;
943 AssertPtrReturn(pThis, VERR_INVALID_HANDLE);
944 AssertReturn(pThis->u32Magic == RTLOCALIPCSESSION_MAGIC, VERR_INVALID_HANDLE);
945 AssertPtrReturn(pvBuffer, VERR_INVALID_POINTER);
946 AssertReturn(cbBuffer, VERR_INVALID_PARAMETER);
947
948 int rc = RTCritSectEnter(&pThis->CritSect);
949 if (RT_SUCCESS(rc))
950 {
951 /* No concurrent writers, sorry. */
952 if (pThis->cRefs == 1)
953 {
954 pThis->cRefs++;
955
956 /*
957 * If I/O is pending, wait for it to complete.
958 */
959 if (pThis->fIOPending)
960 {
961 rc = rtLocalIpcSessionWriteCheckCompletion(pThis);
962 while (rc == VINF_TRY_AGAIN)
963 {
964 Assert(pThis->fIOPending);
965 HANDLE hEvent = pThis->OverlappedIO.hEvent;
966 RTCritSectLeave(&pThis->CritSect);
967 WaitForSingleObject(pThis->OverlappedIO.hEvent, INFINITE);
968 RTCritSectEnter(&pThis->CritSect);
969 }
970 }
971 if (RT_SUCCESS(rc))
972 {
973 Assert(!pThis->fIOPending);
974
975 /*
976 * Try write everything.
977 * No bounce buffering, cUsers protects us.
978 */
979 size_t cbTotalWritten = 0;
980 while (cbBuffer > 0)
981 {
982 BOOL fRc = ResetEvent(pThis->OverlappedIO.hEvent); Assert(fRc == TRUE);
983 pThis->fIOPending = true;
984 RTCritSectLeave(&pThis->CritSect);
985
986 DWORD cbWritten = 0;
987 fRc = WriteFile(pThis->hNmPipe, pvBuffer,
988 cbBuffer <= ~(DWORD)0 ? (DWORD)cbBuffer : ~(DWORD)0,
989 &cbWritten, &pThis->OverlappedIO);
990 if (fRc)
991 {
992 rc = VINF_SUCCESS;
993 }
994 else
995 {
996 DWORD dwErr = GetLastError();
997 if (dwErr == ERROR_IO_PENDING)
998 {
999 DWORD dwRc = WaitForSingleObject(pThis->OverlappedIO.hEvent, INFINITE);
1000 if (dwRc == WAIT_OBJECT_0)
1001 {
1002 if (GetOverlappedResult(pThis->hNmPipe, &pThis->OverlappedIO, &cbWritten, TRUE /*fWait*/))
1003 rc = VINF_SUCCESS;
1004 else
1005 rc = RTErrConvertFromWin32(GetLastError());
1006 }
1007 else if (dwRc == WAIT_TIMEOUT)
1008 rc = VERR_TIMEOUT;
1009 else if (dwRc == WAIT_ABANDONED)
1010 rc = VERR_INVALID_HANDLE;
1011 else
1012 rc = RTErrConvertFromWin32(GetLastError());
1013 }
1014 else if (dwErr == ERROR_NO_DATA)
1015 rc = VERR_BROKEN_PIPE;
1016 else
1017 rc = RTErrConvertFromWin32(dwErr);
1018 }
1019
1020 RTCritSectEnter(&pThis->CritSect);
1021 pThis->fIOPending = false;
1022 if (RT_FAILURE(rc))
1023 break;
1024
1025 /* Advance. */
1026 pvBuffer = (char const *)pvBuffer + cbWritten;
1027 cbTotalWritten += cbWritten;
1028 cbBuffer -= cbWritten;
1029 }
1030 }
1031
1032 pThis->cRefs--;
1033 }
1034 else
1035 rc = VERR_WRONG_ORDER;
1036 RTCritSectLeave(&pThis->CritSect);
1037 }
1038
1039 return rc;
1040}
1041
1042
1043RTDECL(int) RTLocalIpcSessionFlush(RTLOCALIPCSESSION hSession)
1044{
1045 /* No flushing on Windows needed since RTLocalIpcSessionWrite will block until
1046 * all data was written (or an error occurred). */
1047 /** @todo Implement this as soon as we want an explicit asynchronous version of
1048 * RTLocalIpcSessionWrite on Windows. */
1049 return VINF_SUCCESS;
1050}
1051
1052
1053RTDECL(int) RTLocalIpcSessionWaitForData(RTLOCALIPCSESSION hSession, uint32_t cMillies)
1054{
1055 PRTLOCALIPCSESSIONINT pThis = (PRTLOCALIPCSESSIONINT)hSession;
1056 AssertPtrReturn(pThis, VERR_INVALID_HANDLE);
1057 AssertReturn(pThis->u32Magic == RTLOCALIPCSESSION_MAGIC, VERR_INVALID_HANDLE);
1058
1059 uint64_t const StartMsTS = RTTimeMilliTS();
1060
1061 int rc = RTCritSectEnter(&pThis->CritSect);
1062 if (RT_FAILURE(rc))
1063 return rc;
1064 for (unsigned iLoop = 0;; iLoop++)
1065 {
1066 HANDLE hWait = INVALID_HANDLE_VALUE;
1067
1068 if (pThis->fIOPending)
1069 hWait = pThis->OverlappedIO.hEvent;
1070 else
1071 {
1072 /* Peek at the pipe buffer and see how many bytes it contains. */
1073 DWORD cbAvailable;
1074 BOOL fRc = PeekNamedPipe(pThis->hNmPipe, NULL, 0, NULL, &cbAvailable, NULL);
1075 if ( fRc
1076 && cbAvailable)
1077 {
1078 rc = VINF_SUCCESS;
1079 break;
1080 }
1081 else if (!fRc)
1082 {
1083 rc = RTErrConvertFromWin32(GetLastError());
1084 break;
1085 }
1086
1087 /* Start a zero byte read operation that we can wait on. */
1088 if (cMillies == 0)
1089 {
1090 rc = VERR_TIMEOUT;
1091 break;
1092 }
1093 AssertBreakStmt(pThis->cRefs == 1, rc = VERR_WRONG_ORDER);
1094 fRc = ResetEvent(pThis->OverlappedIO.hEvent); Assert(fRc == TRUE);
1095 DWORD cbRead = 0;
1096 if (ReadFile(pThis->hNmPipe, pThis->abBuf, 0, &cbRead, &pThis->OverlappedIO))
1097 {
1098 rc = VINF_SUCCESS;
1099 if (iLoop > 10)
1100 RTThreadYield();
1101 }
1102 else if (GetLastError() == ERROR_IO_PENDING)
1103 {
1104 pThis->cRefs++;
1105 pThis->fIOPending = true;
1106 pThis->fZeroByteRead = true;
1107 hWait = pThis->OverlappedIO.hEvent;
1108 }
1109 else
1110 rc = RTErrConvertFromWin32(GetLastError());
1111 }
1112
1113 if (RT_FAILURE(rc))
1114 break;
1115
1116 /*
1117 * Check for timeout.
1118 */
1119 DWORD cMsMaxWait = INFINITE;
1120 if ( cMillies != RT_INDEFINITE_WAIT
1121 && ( hWait != INVALID_HANDLE_VALUE
1122 || iLoop > 10)
1123 )
1124 {
1125 uint64_t cElapsed = RTTimeMilliTS() - StartMsTS;
1126 if (cElapsed >= cMillies)
1127 {
1128 rc = VERR_TIMEOUT;
1129 break;
1130 }
1131 cMsMaxWait = cMillies - (uint32_t)cElapsed;
1132 }
1133
1134 /*
1135 * Wait.
1136 */
1137 if (hWait != INVALID_HANDLE_VALUE)
1138 {
1139 RTCritSectLeave(&pThis->CritSect);
1140
1141 DWORD dwRc = WaitForSingleObject(hWait, cMsMaxWait);
1142 if (dwRc == WAIT_OBJECT_0)
1143 rc = VINF_SUCCESS;
1144 else if (dwRc == WAIT_TIMEOUT)
1145 rc = VERR_TIMEOUT;
1146 else if (dwRc == WAIT_ABANDONED)
1147 rc = VERR_INVALID_HANDLE;
1148 else
1149 rc = RTErrConvertFromWin32(GetLastError());
1150
1151 if ( RT_FAILURE(rc)
1152 && pThis->u32Magic != RTLOCALIPCSESSION_MAGIC)
1153 return rc;
1154
1155 int rc2 = RTCritSectEnter(&pThis->CritSect);
1156 AssertRC(rc2);
1157 if (pThis->fZeroByteRead)
1158 {
1159 Assert(pThis->cRefs);
1160 pThis->cRefs--;
1161 pThis->fIOPending = false;
1162
1163 if (rc != VINF_SUCCESS)
1164 {
1165 BOOL fRc = CancelIo(pThis->hNmPipe);
1166 Assert(fRc == TRUE);
1167 }
1168
1169 DWORD cbRead = 0;
1170 BOOL fRc = GetOverlappedResult(pThis->hNmPipe, &pThis->OverlappedIO, &cbRead, TRUE /*fWait*/);
1171 if ( !fRc
1172 && RT_SUCCESS(rc))
1173 {
1174 DWORD dwRc = GetLastError();
1175 if (dwRc == ERROR_OPERATION_ABORTED)
1176 rc = VERR_CANCELLED;
1177 else
1178 rc = RTErrConvertFromWin32(dwRc);
1179 }
1180 }
1181
1182 if (RT_FAILURE(rc))
1183 break;
1184 }
1185 }
1186
1187 int rc2 = RTCritSectLeave(&pThis->CritSect);
1188 if (RT_SUCCESS(rc))
1189 rc = rc2;
1190
1191 return rc;
1192}
1193
1194
1195RTDECL(int) RTLocalIpcSessionCancel(RTLOCALIPCSESSION hSession)
1196{
1197 PRTLOCALIPCSESSIONINT pThis = (PRTLOCALIPCSESSIONINT)hSession;
1198 AssertPtrReturn(pThis, VERR_INVALID_HANDLE);
1199 AssertReturn(pThis->u32Magic == RTLOCALIPCSESSION_MAGIC, VERR_INVALID_HANDLE);
1200
1201 /*
1202 * Enter the critical section, then set the cancellation flag
1203 * and signal the event (to wake up anyone in/at WaitForSingleObject).
1204 */
1205 int rc = RTCritSectEnter(&pThis->CritSect);
1206 if (RT_SUCCESS(rc))
1207 {
1208 ASMAtomicUoWriteBool(&pThis->fCancelled, true);
1209 BOOL fRc = SetEvent(pThis->hEvent);
1210 AssertMsg(fRc, ("%d\n", GetLastError())); NOREF(fRc);
1211
1212 RTCritSectLeave(&pThis->CritSect);
1213 }
1214
1215 return rc;
1216}
1217
1218
1219RTDECL(int) RTLocalIpcSessionQueryProcess(RTLOCALIPCSESSION hSession, PRTPROCESS pProcess)
1220{
1221 return VERR_NOT_SUPPORTED;
1222}
1223
1224
1225RTDECL(int) RTLocalIpcSessionQueryUserId(RTLOCALIPCSESSION hSession, PRTUID pUid)
1226{
1227 return VERR_NOT_SUPPORTED;
1228}
1229
1230
1231RTDECL(int) RTLocalIpcSessionQueryGroupId(RTLOCALIPCSESSION hSession, PRTUID pUid)
1232{
1233 return VERR_NOT_SUPPORTED;
1234}
1235
Note: See TracBrowser for help on using the repository browser.

© 2024 Oracle Support Privacy / Do Not Sell My Info Terms of Use Trademark Policy Automated Access Etiquette