VirtualBox

source: vbox/trunk/src/VBox/Runtime/r3/posix/localipc-posix.cpp@ 76881

Last change on this file since 76881 was 76553, checked in by vboxsync, 6 years ago

scm --update-copyright-year

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 35.0 KB
Line 
1/* $Id: localipc-posix.cpp 76553 2019-01-01 01:45:53Z vboxsync $ */
2/** @file
3 * IPRT - Local IPC Server & Client, Posix.
4 */
5
6/*
7 * Copyright (C) 2006-2019 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/*********************************************************************************************************************************
29* Header Files *
30*********************************************************************************************************************************/
31#define LOG_GROUP RTLOGGROUP_LOCALIPC
32#include "internal/iprt.h"
33#include <iprt/localipc.h>
34
35#include <iprt/asm.h>
36#include <iprt/assert.h>
37#include <iprt/ctype.h>
38#include <iprt/critsect.h>
39#include <iprt/err.h>
40#include <iprt/mem.h>
41#include <iprt/log.h>
42#include <iprt/poll.h>
43#include <iprt/socket.h>
44#include <iprt/string.h>
45#include <iprt/time.h>
46
47#include <sys/types.h>
48#include <sys/socket.h>
49#include <sys/un.h>
50#ifndef RT_OS_OS2
51# include <sys/poll.h>
52#endif
53#include <errno.h>
54#include <fcntl.h>
55#include <signal.h>
56#include <unistd.h>
57
58#include "internal/magics.h"
59#include "internal/path.h"
60#include "internal/socket.h"
61
62
63/*********************************************************************************************************************************
64* Structures and Typedefs *
65*********************************************************************************************************************************/
66/**
67 * Local IPC service instance, POSIX.
68 */
69typedef struct RTLOCALIPCSERVERINT
70{
71 /** The magic (RTLOCALIPCSERVER_MAGIC). */
72 uint32_t u32Magic;
73 /** The creation flags. */
74 uint32_t fFlags;
75 /** Critical section protecting the structure. */
76 RTCRITSECT CritSect;
77 /** The number of references to the instance. */
78 uint32_t volatile cRefs;
79 /** Indicates that there is a pending cancel request. */
80 bool volatile fCancelled;
81 /** The server socket. */
82 RTSOCKET hSocket;
83 /** Thread currently listening for clients. */
84 RTTHREAD hListenThread;
85 /** The name we bound the server to (native charset encoding). */
86 struct sockaddr_un Name;
87} RTLOCALIPCSERVERINT;
88/** Pointer to a local IPC server instance (POSIX). */
89typedef RTLOCALIPCSERVERINT *PRTLOCALIPCSERVERINT;
90
91
92/**
93 * Local IPC session instance, POSIX.
94 */
95typedef struct RTLOCALIPCSESSIONINT
96{
97 /** The magic (RTLOCALIPCSESSION_MAGIC). */
98 uint32_t u32Magic;
99 /** Critical section protecting the structure. */
100 RTCRITSECT CritSect;
101 /** The number of references to the instance. */
102 uint32_t volatile cRefs;
103 /** Indicates that there is a pending cancel request. */
104 bool volatile fCancelled;
105 /** Set if this is the server side, clear if the client. */
106 bool fServerSide;
107 /** The client socket. */
108 RTSOCKET hSocket;
109 /** Thread currently doing read related activites. */
110 RTTHREAD hWriteThread;
111 /** Thread currently doing write related activies. */
112 RTTHREAD hReadThread;
113} RTLOCALIPCSESSIONINT;
114/** Pointer to a local IPC session instance (Windows). */
115typedef RTLOCALIPCSESSIONINT *PRTLOCALIPCSESSIONINT;
116
117
118/** Local IPC name prefix for portable names. */
119#define RTLOCALIPC_POSIX_NAME_PREFIX "/tmp/.iprt-localipc-"
120
121
122/**
123 * Validates the user specified name.
124 *
125 * @returns IPRT status code.
126 * @param pszName The name to validate.
127 * @param fNative Whether it's a native name or a portable name.
128 */
129static int rtLocalIpcPosixValidateName(const char *pszName, bool fNative)
130{
131 AssertPtrReturn(pszName, VERR_INVALID_POINTER);
132 AssertReturn(*pszName, VERR_INVALID_NAME);
133
134 if (!fNative)
135 {
136 for (;;)
137 {
138 char ch = *pszName++;
139 if (!ch)
140 break;
141 AssertReturn(!RT_C_IS_CNTRL(ch), VERR_INVALID_NAME);
142 AssertReturn((unsigned)ch < 0x80, VERR_INVALID_NAME);
143 AssertReturn(ch != '\\', VERR_INVALID_NAME);
144 AssertReturn(ch != '/', VERR_INVALID_NAME);
145 }
146 }
147 else
148 {
149 int rc = RTStrValidateEncoding(pszName);
150 AssertRCReturn(rc, rc);
151 }
152
153 return VINF_SUCCESS;
154}
155
156
157/**
158 * Constructs a local (unix) domain socket name.
159 *
160 * @returns IPRT status code.
161 * @param pAddr The address structure to construct the name in.
162 * @param pcbAddr Where to return the address size.
163 * @param pszName The user specified name (valid).
164 * @param fNative Whether it's a native name or a portable name.
165 */
166static int rtLocalIpcPosixConstructName(struct sockaddr_un *pAddr, uint8_t *pcbAddr, const char *pszName, bool fNative)
167{
168 const char *pszNativeName;
169 int rc = rtPathToNative(&pszNativeName, pszName, NULL /*pszBasePath not support*/);
170 if (RT_SUCCESS(rc))
171 {
172 size_t cchNativeName = strlen(pszNativeName);
173 size_t cbFull = !fNative ? cchNativeName + sizeof(RTLOCALIPC_POSIX_NAME_PREFIX) : cchNativeName + 1;
174 if (cbFull <= sizeof(pAddr->sun_path))
175 {
176 RT_ZERO(*pAddr);
177#ifdef RT_OS_OS2 /* Size must be exactly right on OS/2. */
178 *pcbAddr = sizeof(*pAddr);
179#else
180 *pcbAddr = RT_UOFFSETOF(struct sockaddr_un, sun_path) + (uint8_t)cbFull;
181#endif
182#ifdef HAVE_SUN_LEN_MEMBER
183 pAddr->sun_len = *pcbAddr;
184#endif
185 pAddr->sun_family = AF_LOCAL;
186
187 if (!fNative)
188 {
189 memcpy(pAddr->sun_path, RTLOCALIPC_POSIX_NAME_PREFIX, sizeof(RTLOCALIPC_POSIX_NAME_PREFIX) - 1);
190 memcpy(&pAddr->sun_path[sizeof(RTLOCALIPC_POSIX_NAME_PREFIX) - 1], pszNativeName, cchNativeName + 1);
191 }
192 else
193 memcpy(pAddr->sun_path, pszNativeName, cchNativeName + 1);
194 }
195 else
196 rc = VERR_FILENAME_TOO_LONG;
197 rtPathFreeNative(pszNativeName, pszName);
198 }
199 return rc;
200}
201
202
203
204RTDECL(int) RTLocalIpcServerCreate(PRTLOCALIPCSERVER phServer, const char *pszName, uint32_t fFlags)
205{
206 /*
207 * Parameter validation.
208 */
209 AssertPtrReturn(phServer, VERR_INVALID_POINTER);
210 *phServer = NIL_RTLOCALIPCSERVER;
211 AssertReturn(!(fFlags & ~RTLOCALIPC_FLAGS_VALID_MASK), VERR_INVALID_FLAGS);
212 int rc = rtLocalIpcPosixValidateName(pszName, RT_BOOL(fFlags & RTLOCALIPC_FLAGS_NATIVE_NAME));
213 if (RT_SUCCESS(rc))
214 {
215 /*
216 * Allocate memory for the instance and initialize it.
217 */
218 PRTLOCALIPCSERVERINT pThis = (PRTLOCALIPCSERVERINT)RTMemAllocZ(sizeof(*pThis));
219 if (pThis)
220 {
221 pThis->u32Magic = RTLOCALIPCSERVER_MAGIC;
222 pThis->fFlags = fFlags;
223 pThis->cRefs = 1;
224 pThis->fCancelled = false;
225 pThis->hListenThread = NIL_RTTHREAD;
226 rc = RTCritSectInit(&pThis->CritSect);
227 if (RT_SUCCESS(rc))
228 {
229 /*
230 * Create the local (unix) socket and bind to it.
231 */
232 rc = rtSocketCreate(&pThis->hSocket, AF_LOCAL, SOCK_STREAM, 0 /*iProtocol*/);
233 if (RT_SUCCESS(rc))
234 {
235 RTSocketSetInheritance(pThis->hSocket, false /*fInheritable*/);
236 signal(SIGPIPE, SIG_IGN); /* Required on solaris, at least. */
237
238 uint8_t cbAddr;
239 rc = rtLocalIpcPosixConstructName(&pThis->Name, &cbAddr, pszName,
240 RT_BOOL(fFlags & RTLOCALIPC_FLAGS_NATIVE_NAME));
241 if (RT_SUCCESS(rc))
242 {
243 rc = rtSocketBindRawAddr(pThis->hSocket, &pThis->Name, cbAddr);
244 if (rc == VERR_NET_ADDRESS_IN_USE)
245 {
246 unlink(pThis->Name.sun_path);
247 rc = rtSocketBindRawAddr(pThis->hSocket, &pThis->Name, cbAddr);
248 }
249 if (RT_SUCCESS(rc))
250 {
251 rc = rtSocketListen(pThis->hSocket, 16);
252 if (RT_SUCCESS(rc))
253 {
254 LogFlow(("RTLocalIpcServerCreate: Created %p (%s)\n", pThis, pThis->Name.sun_path));
255 *phServer = pThis;
256 return VINF_SUCCESS;
257 }
258 unlink(pThis->Name.sun_path);
259 }
260 }
261 RTSocketRelease(pThis->hSocket);
262 }
263 RTCritSectDelete(&pThis->CritSect);
264 }
265 RTMemFree(pThis);
266 }
267 else
268 rc = VERR_NO_MEMORY;
269 }
270 Log(("RTLocalIpcServerCreate: failed, rc=%Rrc\n", rc));
271 return rc;
272}
273
274
275/**
276 * Retains a reference to the server instance.
277 *
278 * @returns
279 * @param pThis The server instance.
280 */
281DECLINLINE(void) rtLocalIpcServerRetain(PRTLOCALIPCSERVERINT pThis)
282{
283 uint32_t cRefs = ASMAtomicIncU32(&pThis->cRefs);
284 Assert(cRefs < UINT32_MAX / 2 && cRefs); RT_NOREF_PV(cRefs);
285}
286
287
288/**
289 * Server instance destructor.
290 *
291 * @returns VINF_OBJECT_DESTROYED
292 * @param pThis The server instance.
293 */
294static int rtLocalIpcServerDtor(PRTLOCALIPCSERVERINT pThis)
295{
296 pThis->u32Magic = ~RTLOCALIPCSERVER_MAGIC;
297 if (RTSocketRelease(pThis->hSocket) == 0)
298 Log(("rtLocalIpcServerDtor: Released socket\n"));
299 else
300 Log(("rtLocalIpcServerDtor: Socket still has references (impossible?)\n"));
301 RTCritSectDelete(&pThis->CritSect);
302 unlink(pThis->Name.sun_path);
303 RTMemFree(pThis);
304 return VINF_OBJECT_DESTROYED;
305}
306
307
308/**
309 * Releases a reference to the server instance.
310 *
311 * @returns VINF_SUCCESS if only release, VINF_OBJECT_DESTROYED if destroyed.
312 * @param pThis The server instance.
313 */
314DECLINLINE(int) rtLocalIpcServerRelease(PRTLOCALIPCSERVERINT pThis)
315{
316 uint32_t cRefs = ASMAtomicDecU32(&pThis->cRefs);
317 Assert(cRefs < UINT32_MAX / 2);
318 if (!cRefs)
319 return rtLocalIpcServerDtor(pThis);
320 return VINF_SUCCESS;
321}
322
323
324/**
325 * The core of RTLocalIpcServerCancel, used by both the destroy and cancel APIs.
326 *
327 * @returns IPRT status code
328 * @param pThis The server instance.
329 */
330static int rtLocalIpcServerCancel(PRTLOCALIPCSERVERINT pThis)
331{
332 RTCritSectEnter(&pThis->CritSect);
333 pThis->fCancelled = true;
334 Log(("rtLocalIpcServerCancel:\n"));
335 if (pThis->hListenThread != NIL_RTTHREAD)
336 RTThreadPoke(pThis->hListenThread);
337 RTCritSectLeave(&pThis->CritSect);
338 return VINF_SUCCESS;
339}
340
341
342
343RTDECL(int) RTLocalIpcServerDestroy(RTLOCALIPCSERVER hServer)
344{
345 /*
346 * Validate input.
347 */
348 if (hServer == NIL_RTLOCALIPCSERVER)
349 return VINF_SUCCESS;
350 PRTLOCALIPCSERVERINT pThis = (PRTLOCALIPCSERVERINT)hServer;
351 AssertPtrReturn(pThis, VERR_INVALID_HANDLE);
352 AssertReturn(pThis->u32Magic == RTLOCALIPCSERVER_MAGIC, VERR_INVALID_HANDLE);
353
354 /*
355 * Invalidate the server, releasing the caller's reference to the instance
356 * data and making sure any other thread in the listen API will wake up.
357 */
358 AssertReturn(ASMAtomicCmpXchgU32(&pThis->u32Magic, ~RTLOCALIPCSERVER_MAGIC, RTLOCALIPCSERVER_MAGIC), VERR_WRONG_ORDER);
359
360 rtLocalIpcServerCancel(pThis);
361 return rtLocalIpcServerRelease(pThis);
362}
363
364
365RTDECL(int) RTLocalIpcServerCancel(RTLOCALIPCSERVER hServer)
366{
367 /*
368 * Validate input.
369 */
370 PRTLOCALIPCSERVERINT pThis = (PRTLOCALIPCSERVERINT)hServer;
371 AssertPtrReturn(pThis, VERR_INVALID_HANDLE);
372 AssertReturn(pThis->u32Magic == RTLOCALIPCSERVER_MAGIC, VERR_INVALID_HANDLE);
373
374 /*
375 * Do the job.
376 */
377 rtLocalIpcServerRetain(pThis);
378 rtLocalIpcServerCancel(pThis);
379 rtLocalIpcServerRelease(pThis);
380 return VINF_SUCCESS;
381}
382
383
384RTDECL(int) RTLocalIpcServerListen(RTLOCALIPCSERVER hServer, PRTLOCALIPCSESSION phClientSession)
385{
386 /*
387 * Validate input.
388 */
389 PRTLOCALIPCSERVERINT pThis = (PRTLOCALIPCSERVERINT)hServer;
390 AssertPtrReturn(pThis, VERR_INVALID_HANDLE);
391 AssertReturn(pThis->u32Magic == RTLOCALIPCSERVER_MAGIC, VERR_INVALID_HANDLE);
392
393 /*
394 * Begin listening.
395 */
396 rtLocalIpcServerRetain(pThis);
397 int rc = RTCritSectEnter(&pThis->CritSect);
398 if (RT_SUCCESS(rc))
399 {
400 if (pThis->hListenThread == NIL_RTTHREAD)
401 {
402 pThis->hListenThread = RTThreadSelf();
403
404 /*
405 * The listening retry loop.
406 */
407 for (;;)
408 {
409 if (!pThis->fCancelled)
410 {
411 rc = RTCritSectLeave(&pThis->CritSect);
412 AssertRCBreak(rc);
413
414 struct sockaddr_un Addr;
415 size_t cbAddr = sizeof(Addr);
416 RTSOCKET hClient;
417 Log(("RTLocalIpcServerListen: Calling rtSocketAccept...\n"));
418 rc = rtSocketAccept(pThis->hSocket, &hClient, (struct sockaddr *)&Addr, &cbAddr);
419 Log(("RTLocalIpcServerListen: rtSocketAccept returns %Rrc.\n", rc));
420
421 int rc2 = RTCritSectEnter(&pThis->CritSect);
422 AssertRCBreakStmt(rc2, rc = RT_SUCCESS(rc) ? rc2 : rc);
423
424 if (RT_SUCCESS(rc))
425 {
426 /*
427 * Create a client session.
428 */
429 PRTLOCALIPCSESSIONINT pSession = (PRTLOCALIPCSESSIONINT)RTMemAllocZ(sizeof(*pSession));
430 if (pSession)
431 {
432 pSession->u32Magic = RTLOCALIPCSESSION_MAGIC;
433 pSession->cRefs = 1;
434 pSession->fCancelled = false;
435 pSession->fServerSide = true;
436 pSession->hSocket = hClient;
437 pSession->hReadThread = NIL_RTTHREAD;
438 pSession->hWriteThread = NIL_RTTHREAD;
439 rc = RTCritSectInit(&pSession->CritSect);
440 if (RT_SUCCESS(rc))
441 {
442 Log(("RTLocalIpcServerListen: Returning new client session: %p\n", pSession));
443 *phClientSession = pSession;
444 break;
445 }
446
447 RTMemFree(pSession);
448 }
449 else
450 rc = VERR_NO_MEMORY;
451 }
452 else if ( rc != VERR_INTERRUPTED
453 && rc != VERR_TRY_AGAIN)
454 break;
455 }
456 else
457 {
458 rc = VERR_CANCELLED;
459 break;
460 }
461 }
462
463 pThis->hListenThread = NIL_RTTHREAD;
464 }
465 else
466 {
467 AssertFailed();
468 rc = VERR_RESOURCE_BUSY;
469 }
470 int rc2 = RTCritSectLeave(&pThis->CritSect);
471 AssertStmt(RT_SUCCESS(rc2), rc = RT_SUCCESS(rc) ? rc2 : rc);
472 }
473 rtLocalIpcServerRelease(pThis);
474
475 Log(("RTLocalIpcServerListen: returns %Rrc\n", rc));
476 return rc;
477}
478
479
480RTDECL(int) RTLocalIpcSessionConnect(PRTLOCALIPCSESSION phSession, const char *pszName, uint32_t fFlags)
481{
482 /*
483 * Parameter validation.
484 */
485 AssertPtrReturn(phSession, VERR_INVALID_POINTER);
486 *phSession = NIL_RTLOCALIPCSESSION;
487
488 AssertReturn(!(fFlags & ~RTLOCALIPC_C_FLAGS_VALID_MASK), VERR_INVALID_FLAGS);
489
490 int rc = rtLocalIpcPosixValidateName(pszName, RT_BOOL(fFlags & RTLOCALIPC_C_FLAGS_NATIVE_NAME));
491 if (RT_SUCCESS(rc))
492 {
493 /*
494 * Allocate memory for the instance and initialize it.
495 */
496 PRTLOCALIPCSESSIONINT pThis = (PRTLOCALIPCSESSIONINT)RTMemAllocZ(sizeof(*pThis));
497 if (pThis)
498 {
499 pThis->u32Magic = RTLOCALIPCSESSION_MAGIC;
500 pThis->cRefs = 1;
501 pThis->fCancelled = false;
502 pThis->fServerSide = false;
503 pThis->hSocket = NIL_RTSOCKET;
504 pThis->hReadThread = NIL_RTTHREAD;
505 pThis->hWriteThread = NIL_RTTHREAD;
506 rc = RTCritSectInit(&pThis->CritSect);
507 if (RT_SUCCESS(rc))
508 {
509 /*
510 * Create the local (unix) socket and try connect to the server.
511 */
512 rc = rtSocketCreate(&pThis->hSocket, AF_LOCAL, SOCK_STREAM, 0 /*iProtocol*/);
513 if (RT_SUCCESS(rc))
514 {
515 RTSocketSetInheritance(pThis->hSocket, false /*fInheritable*/);
516 signal(SIGPIPE, SIG_IGN); /* Required on solaris, at least. */
517
518 struct sockaddr_un Addr;
519 uint8_t cbAddr;
520 rc = rtLocalIpcPosixConstructName(&Addr, &cbAddr, pszName, RT_BOOL(fFlags & RTLOCALIPC_C_FLAGS_NATIVE_NAME));
521 if (RT_SUCCESS(rc))
522 {
523 rc = rtSocketConnectRaw(pThis->hSocket, &Addr, cbAddr);
524 if (RT_SUCCESS(rc))
525 {
526 *phSession = pThis;
527 Log(("RTLocalIpcSessionConnect: Returns new session %p\n", pThis));
528 return VINF_SUCCESS;
529 }
530 }
531 RTCritSectDelete(&pThis->CritSect);
532 }
533 }
534 RTMemFree(pThis);
535 }
536 else
537 rc = VERR_NO_MEMORY;
538 }
539 Log(("RTLocalIpcSessionConnect: returns %Rrc\n", rc));
540 return rc;
541}
542
543
544/**
545 * Retains a reference to the session instance.
546 *
547 * @param pThis The server instance.
548 */
549DECLINLINE(void) rtLocalIpcSessionRetain(PRTLOCALIPCSESSIONINT pThis)
550{
551 uint32_t cRefs = ASMAtomicIncU32(&pThis->cRefs);
552 Assert(cRefs < UINT32_MAX / 2 && cRefs); RT_NOREF_PV(cRefs);
553}
554
555
556RTDECL(uint32_t) RTLocalIpcSessionRetain(RTLOCALIPCSESSION hSession)
557{
558 PRTLOCALIPCSESSIONINT pThis = (PRTLOCALIPCSESSIONINT)hSession;
559 AssertPtrReturn(pThis, UINT32_MAX);
560 AssertReturn(pThis->u32Magic == RTLOCALIPCSESSION_MAGIC, UINT32_MAX);
561
562 uint32_t cRefs = ASMAtomicIncU32(&pThis->cRefs);
563 Assert(cRefs < UINT32_MAX / 2 && cRefs);
564 return cRefs;
565}
566
567
568/**
569 * Session instance destructor.
570 *
571 * @returns VINF_OBJECT_DESTROYED
572 * @param pThis The server instance.
573 */
574static int rtLocalIpcSessionDtor(PRTLOCALIPCSESSIONINT pThis)
575{
576 pThis->u32Magic = ~RTLOCALIPCSESSION_MAGIC;
577 if (RTSocketRelease(pThis->hSocket) == 0)
578 Log(("rtLocalIpcSessionDtor: Released socket\n"));
579 else
580 Log(("rtLocalIpcSessionDtor: Socket still has references (impossible?)\n"));
581 RTCritSectDelete(&pThis->CritSect);
582 RTMemFree(pThis);
583 return VINF_OBJECT_DESTROYED;
584}
585
586
587/**
588 * Releases a reference to the session instance.
589 *
590 * @returns VINF_SUCCESS or VINF_OBJECT_DESTROYED as appropriate.
591 * @param pThis The session instance.
592 */
593DECLINLINE(int) rtLocalIpcSessionRelease(PRTLOCALIPCSESSIONINT pThis)
594{
595 uint32_t cRefs = ASMAtomicDecU32(&pThis->cRefs);
596 Assert(cRefs < UINT32_MAX / 2);
597 if (!cRefs)
598 return rtLocalIpcSessionDtor(pThis);
599 Log(("rtLocalIpcSessionRelease: %u refs left\n", cRefs));
600 return VINF_SUCCESS;
601}
602
603
604RTDECL(uint32_t) RTLocalIpcSessionRelease(RTLOCALIPCSESSION hSession)
605{
606 if (hSession == NIL_RTLOCALIPCSESSION)
607 return 0;
608
609 PRTLOCALIPCSESSIONINT pThis = (PRTLOCALIPCSESSIONINT)hSession;
610 AssertPtrReturn(pThis, UINT32_MAX);
611 AssertReturn(pThis->u32Magic == RTLOCALIPCSESSION_MAGIC, UINT32_MAX);
612
613 uint32_t cRefs = ASMAtomicDecU32(&pThis->cRefs);
614 Assert(cRefs < UINT32_MAX / 2);
615 if (cRefs)
616 Log(("RTLocalIpcSessionRelease: %u refs left\n", cRefs));
617 else
618 rtLocalIpcSessionDtor(pThis);
619 return cRefs;
620}
621
622
623/**
624 * The core of RTLocalIpcSessionCancel, used by both the destroy and cancel APIs.
625 *
626 * @returns IPRT status code
627 * @param pThis The session instance.
628 */
629static int rtLocalIpcSessionCancel(PRTLOCALIPCSESSIONINT pThis)
630{
631 RTCritSectEnter(&pThis->CritSect);
632 pThis->fCancelled = true;
633 Log(("rtLocalIpcSessionCancel:\n"));
634 if (pThis->hReadThread != NIL_RTTHREAD)
635 RTThreadPoke(pThis->hReadThread);
636 if (pThis->hWriteThread != NIL_RTTHREAD)
637 RTThreadPoke(pThis->hWriteThread);
638 RTCritSectLeave(&pThis->CritSect);
639 return VINF_SUCCESS;
640}
641
642
643RTDECL(int) RTLocalIpcSessionClose(RTLOCALIPCSESSION hSession)
644{
645 /*
646 * Validate input.
647 */
648 if (hSession == NIL_RTLOCALIPCSESSION)
649 return VINF_SUCCESS;
650 PRTLOCALIPCSESSIONINT pThis = hSession;
651 AssertPtrReturn(pThis, VERR_INVALID_HANDLE);
652 AssertReturn(pThis->u32Magic == RTLOCALIPCSESSION_MAGIC, VERR_INVALID_HANDLE);
653
654 /*
655 * Invalidate the session, releasing the caller's reference to the instance
656 * data and making sure any other thread in the listen API will wake up.
657 */
658 Log(("RTLocalIpcSessionClose:\n"));
659
660 rtLocalIpcSessionCancel(pThis);
661 return rtLocalIpcSessionRelease(pThis);
662}
663
664
665RTDECL(int) RTLocalIpcSessionCancel(RTLOCALIPCSESSION hSession)
666{
667 /*
668 * Validate input.
669 */
670 PRTLOCALIPCSESSIONINT pThis = hSession;
671 AssertPtrReturn(pThis, VERR_INVALID_HANDLE);
672 AssertReturn(pThis->u32Magic == RTLOCALIPCSESSION_MAGIC, VERR_INVALID_HANDLE);
673
674 /*
675 * Do the job.
676 */
677 rtLocalIpcSessionRetain(pThis);
678 rtLocalIpcSessionCancel(pThis);
679 rtLocalIpcSessionRelease(pThis);
680 return VINF_SUCCESS;
681}
682
683
684/**
685 * Checks if the socket has has a HUP condition after reading zero bytes.
686 *
687 * @returns true if HUP, false if no.
688 * @param pThis The IPC session handle.
689 */
690static bool rtLocalIpcPosixHasHup(PRTLOCALIPCSESSIONINT pThis)
691{
692 int fdNative = RTSocketToNative(pThis->hSocket);
693
694#if !defined(RT_OS_OS2) && !defined(RT_OS_SOLARIS)
695 struct pollfd PollFd;
696 RT_ZERO(PollFd);
697 PollFd.fd = fdNative;
698 PollFd.events = POLLHUP | POLLERR;
699 if (poll(&PollFd, 1, 0) <= 0)
700 return false;
701 if (!(PollFd.revents & (POLLHUP | POLLERR)))
702 return false;
703#else /* RT_OS_OS2 || RT_OS_SOLARIS */
704 /*
705 * OS/2: No native poll, do zero byte send to check for EPIPE.
706 * Solaris: We don't get POLLHUP.
707 */
708 uint8_t bDummy;
709 ssize_t rcSend = send(fdNative, &bDummy, 0, 0);
710 if (rcSend >= 0 || (errno != EPIPE && errno != ECONNRESET))
711 return false;
712#endif /* RT_OS_OS2 || RT_OS_SOLARIS */
713
714 /*
715 * We've established EPIPE. Now make sure there aren't any last bytes to
716 * read that came in between the recv made by the caller and the disconnect.
717 */
718 uint8_t bPeek;
719 ssize_t rcRecv = recv(fdNative, &bPeek, 1, MSG_DONTWAIT | MSG_PEEK);
720 return rcRecv <= 0;
721}
722
723
724RTDECL(int) RTLocalIpcSessionRead(RTLOCALIPCSESSION hSession, void *pvBuf, size_t cbToRead, size_t *pcbRead)
725{
726 /*
727 * Validate input.
728 */
729 PRTLOCALIPCSESSIONINT pThis = hSession;
730 AssertPtrReturn(pThis, VERR_INVALID_HANDLE);
731 AssertReturn(pThis->u32Magic == RTLOCALIPCSESSION_MAGIC, VERR_INVALID_HANDLE);
732
733 /*
734 * Do the job.
735 */
736 rtLocalIpcSessionRetain(pThis);
737
738 int rc = RTCritSectEnter(&pThis->CritSect);
739 if (RT_SUCCESS(rc))
740 {
741 if (pThis->hReadThread == NIL_RTTHREAD)
742 {
743 pThis->hReadThread = RTThreadSelf();
744
745 for (;;)
746 {
747 if (!pThis->fCancelled)
748 {
749 rc = RTCritSectLeave(&pThis->CritSect);
750 AssertRCBreak(rc);
751
752 rc = RTSocketRead(pThis->hSocket, pvBuf, cbToRead, pcbRead);
753
754 /* Detect broken pipe. */
755 if (rc == VINF_SUCCESS)
756 {
757 if (!pcbRead || *pcbRead)
758 { /* likely */ }
759 else if (rtLocalIpcPosixHasHup(pThis))
760 rc = VERR_BROKEN_PIPE;
761 }
762 else if (rc == VERR_NET_CONNECTION_RESET_BY_PEER || rc == VERR_NET_SHUTDOWN)
763 rc = VERR_BROKEN_PIPE;
764
765 int rc2 = RTCritSectEnter(&pThis->CritSect);
766 AssertRCBreakStmt(rc2, rc = RT_SUCCESS(rc) ? rc2 : rc);
767
768 if ( rc == VERR_INTERRUPTED
769 || rc == VERR_TRY_AGAIN)
770 continue;
771 }
772 else
773 rc = VERR_CANCELLED;
774 break;
775 }
776
777 pThis->hReadThread = NIL_RTTHREAD;
778 }
779 int rc2 = RTCritSectLeave(&pThis->CritSect);
780 AssertStmt(RT_SUCCESS(rc2), rc = RT_SUCCESS(rc) ? rc2 : rc);
781 }
782
783 rtLocalIpcSessionRelease(pThis);
784 return rc;
785}
786
787
788RTDECL(int) RTLocalIpcSessionReadNB(RTLOCALIPCSESSION hSession, void *pvBuf, size_t cbToRead, size_t *pcbRead)
789{
790 /*
791 * Validate input.
792 */
793 PRTLOCALIPCSESSIONINT pThis = hSession;
794 AssertPtrReturn(pThis, VERR_INVALID_HANDLE);
795 AssertReturn(pThis->u32Magic == RTLOCALIPCSESSION_MAGIC, VERR_INVALID_HANDLE);
796
797 /*
798 * Do the job.
799 */
800 rtLocalIpcSessionRetain(pThis);
801
802 int rc = RTCritSectEnter(&pThis->CritSect);
803 if (RT_SUCCESS(rc))
804 {
805 if (pThis->hReadThread == NIL_RTTHREAD)
806 {
807 pThis->hReadThread = RTThreadSelf(); /* not really required, but whatever. */
808
809 for (;;)
810 {
811 if (!pThis->fCancelled)
812 {
813 rc = RTSocketReadNB(pThis->hSocket, pvBuf, cbToRead, pcbRead);
814
815 /* Detect broken pipe. */
816 if (rc == VINF_SUCCESS)
817 {
818 if (!pcbRead || *pcbRead)
819 { /* likely */ }
820 else if (rtLocalIpcPosixHasHup(pThis))
821 rc = VERR_BROKEN_PIPE;
822 }
823 else if (rc == VERR_NET_CONNECTION_RESET_BY_PEER || rc == VERR_NET_SHUTDOWN)
824 rc = VERR_BROKEN_PIPE;
825
826 if (rc == VERR_INTERRUPTED)
827 continue;
828 }
829 else
830 rc = VERR_CANCELLED;
831 break;
832 }
833
834 pThis->hReadThread = NIL_RTTHREAD;
835 }
836 int rc2 = RTCritSectLeave(&pThis->CritSect);
837 AssertStmt(RT_SUCCESS(rc2), rc = RT_SUCCESS(rc) ? rc2 : rc);
838 }
839
840 rtLocalIpcSessionRelease(pThis);
841 return rc;
842}
843
844
845RTDECL(int) RTLocalIpcSessionWrite(RTLOCALIPCSESSION hSession, const void *pvBuf, size_t cbToWrite)
846{
847 /*
848 * Validate input.
849 */
850 PRTLOCALIPCSESSIONINT pThis = hSession;
851 AssertPtrReturn(pThis, VERR_INVALID_HANDLE);
852 AssertReturn(pThis->u32Magic == RTLOCALIPCSESSION_MAGIC, VERR_INVALID_HANDLE);
853
854 /*
855 * Do the job.
856 */
857 rtLocalIpcSessionRetain(pThis);
858
859 int rc = RTCritSectEnter(&pThis->CritSect);
860 if (RT_SUCCESS(rc))
861 {
862 if (pThis->hWriteThread == NIL_RTTHREAD)
863 {
864 pThis->hWriteThread = RTThreadSelf();
865
866 for (;;)
867 {
868 if (!pThis->fCancelled)
869 {
870 rc = RTCritSectLeave(&pThis->CritSect);
871 AssertRCBreak(rc);
872
873 rc = RTSocketWrite(pThis->hSocket, pvBuf, cbToWrite);
874
875 int rc2 = RTCritSectEnter(&pThis->CritSect);
876 AssertRCBreakStmt(rc2, rc = RT_SUCCESS(rc) ? rc2 : rc);
877
878 if ( rc == VERR_INTERRUPTED
879 || rc == VERR_TRY_AGAIN)
880 continue;
881 }
882 else
883 rc = VERR_CANCELLED;
884 break;
885 }
886
887 pThis->hWriteThread = NIL_RTTHREAD;
888 }
889 int rc2 = RTCritSectLeave(&pThis->CritSect);
890 AssertStmt(RT_SUCCESS(rc2), rc = RT_SUCCESS(rc) ? rc2 : rc);
891 }
892
893 rtLocalIpcSessionRelease(pThis);
894 return rc;
895}
896
897
898RTDECL(int) RTLocalIpcSessionFlush(RTLOCALIPCSESSION hSession)
899{
900 /*
901 * Validate input.
902 */
903 PRTLOCALIPCSESSIONINT pThis = hSession;
904 AssertPtrReturn(pThis, VERR_INVALID_HANDLE);
905 AssertReturn(pThis->u32Magic == RTLOCALIPCSESSION_MAGIC, VERR_INVALID_HANDLE);
906
907 /*
908 * This is a no-op because apparently write doesn't return until the
909 * result is read. At least that's what the reply to a 2003-04-08 LKML
910 * posting title "fsync() on unix domain sockets?" indicates.
911 *
912 * For conformity, make sure there isn't any active writes concurrent to this call.
913 */
914 rtLocalIpcSessionRetain(pThis);
915
916 int rc = RTCritSectEnter(&pThis->CritSect);
917 if (RT_SUCCESS(rc))
918 {
919 if (pThis->hWriteThread == NIL_RTTHREAD)
920 rc = RTCritSectLeave(&pThis->CritSect);
921 else
922 {
923 rc = RTCritSectLeave(&pThis->CritSect);
924 if (RT_SUCCESS(rc))
925 rc = VERR_RESOURCE_BUSY;
926 }
927 }
928
929 rtLocalIpcSessionRelease(pThis);
930 return rc;
931}
932
933
934RTDECL(int) RTLocalIpcSessionWaitForData(RTLOCALIPCSESSION hSession, uint32_t cMillies)
935{
936 /*
937 * Validate input.
938 */
939 PRTLOCALIPCSESSIONINT pThis = hSession;
940 AssertPtrReturn(pThis, VERR_INVALID_HANDLE);
941 AssertReturn(pThis->u32Magic == RTLOCALIPCSESSION_MAGIC, VERR_INVALID_HANDLE);
942
943 /*
944 * Do the job.
945 */
946 rtLocalIpcSessionRetain(pThis);
947
948 int rc = RTCritSectEnter(&pThis->CritSect);
949 if (RT_SUCCESS(rc))
950 {
951 if (pThis->hReadThread == NIL_RTTHREAD)
952 {
953 pThis->hReadThread = RTThreadSelf();
954 uint64_t const msStart = RTTimeMilliTS();
955 RTMSINTERVAL const cMsOriginalTimeout = cMillies;
956
957 for (;;)
958 {
959 if (!pThis->fCancelled)
960 {
961 rc = RTCritSectLeave(&pThis->CritSect);
962 AssertRCBreak(rc);
963
964 uint32_t fEvents = 0;
965#ifdef RT_OS_OS2
966 /* This doesn't give us any error condition on hangup, so use HUP check. */
967 Log(("RTLocalIpcSessionWaitForData: Calling RTSocketSelectOneEx...\n"));
968 rc = RTSocketSelectOneEx(pThis->hSocket, RTPOLL_EVT_READ | RTPOLL_EVT_ERROR, &fEvents, cMillies);
969 Log(("RTLocalIpcSessionWaitForData: RTSocketSelectOneEx returns %Rrc, fEvents=%#x\n", rc, fEvents));
970 if (RT_SUCCESS(rc) && fEvents == RTPOLL_EVT_READ && rtLocalIpcPosixHasHup(pThis))
971 rc = VERR_BROKEN_PIPE;
972#else
973/** @todo RTSocketPoll? */
974 /* POLLHUP will be set on hangup. */
975 struct pollfd PollFd;
976 RT_ZERO(PollFd);
977 PollFd.fd = RTSocketToNative(pThis->hSocket);
978 PollFd.events = POLLHUP | POLLERR | POLLIN;
979 Log(("RTLocalIpcSessionWaitForData: Calling poll...\n"));
980 int cFds = poll(&PollFd, 1, cMillies == RT_INDEFINITE_WAIT ? -1 : cMillies);
981 if (cFds >= 1)
982 {
983 /* Linux & Darwin sets both POLLIN and POLLHUP when the pipe is
984 broken and but no more data to read. Google hints at NetBSD
985 returning more sane values (POLLIN till no more data, then
986 POLLHUP). Solairs OTOH, doesn't ever seem to return POLLHUP. */
987 fEvents = RTPOLL_EVT_READ;
988 if ( (PollFd.revents & (POLLHUP | POLLERR))
989 && !(PollFd.revents & POLLIN))
990 fEvents = RTPOLL_EVT_ERROR;
991# if defined(RT_OS_SOLARIS)
992 else if (PollFd.revents & POLLIN)
993# else
994 else if ((PollFd.revents & (POLLIN | POLLHUP)) == (POLLIN | POLLHUP))
995# endif
996 {
997 /* Check if there is actually data available. */
998 uint8_t bPeek;
999 ssize_t rcRecv = recv(PollFd.fd, &bPeek, 1, MSG_DONTWAIT | MSG_PEEK);
1000 if (rcRecv <= 0)
1001 fEvents = RTPOLL_EVT_ERROR;
1002 }
1003 rc = VINF_SUCCESS;
1004 }
1005 else if (rc == 0)
1006 rc = VERR_TIMEOUT;
1007 else
1008 rc = RTErrConvertFromErrno(errno);
1009 Log(("RTLocalIpcSessionWaitForData: poll returns %u (rc=%d), revents=%#x\n", cFds, rc, PollFd.revents));
1010#endif
1011
1012 int rc2 = RTCritSectEnter(&pThis->CritSect);
1013 AssertRCBreakStmt(rc2, rc = RT_SUCCESS(rc) ? rc2 : rc);
1014
1015 if (RT_SUCCESS(rc))
1016 {
1017 if (pThis->fCancelled)
1018 rc = VERR_CANCELLED;
1019 else if (fEvents & RTPOLL_EVT_ERROR)
1020 rc = VERR_BROKEN_PIPE;
1021 }
1022 else if ( rc == VERR_INTERRUPTED
1023 || rc == VERR_TRY_AGAIN)
1024 {
1025 /* Recalc cMillies. */
1026 if (cMsOriginalTimeout != RT_INDEFINITE_WAIT)
1027 {
1028 uint64_t cMsElapsed = RTTimeMilliTS() - msStart;
1029 cMillies = cMsElapsed >= cMsOriginalTimeout ? 0 : cMsOriginalTimeout - (RTMSINTERVAL)cMsElapsed;
1030 }
1031 continue;
1032 }
1033 }
1034 else
1035 rc = VERR_CANCELLED;
1036 break;
1037 }
1038
1039 pThis->hReadThread = NIL_RTTHREAD;
1040 }
1041 int rc2 = RTCritSectLeave(&pThis->CritSect);
1042 AssertStmt(RT_SUCCESS(rc2), rc = RT_SUCCESS(rc) ? rc2 : rc);
1043 }
1044
1045 rtLocalIpcSessionRelease(pThis);
1046 return rc;
1047}
1048
1049
1050RTDECL(int) RTLocalIpcSessionQueryProcess(RTLOCALIPCSESSION hSession, PRTPROCESS pProcess)
1051{
1052 RT_NOREF_PV(hSession); RT_NOREF_PV(pProcess);
1053 return VERR_NOT_SUPPORTED;
1054}
1055
1056
1057RTDECL(int) RTLocalIpcSessionQueryUserId(RTLOCALIPCSESSION hSession, PRTUID pUid)
1058{
1059 RT_NOREF_PV(hSession); RT_NOREF_PV(pUid);
1060 return VERR_NOT_SUPPORTED;
1061}
1062
1063
1064RTDECL(int) RTLocalIpcSessionQueryGroupId(RTLOCALIPCSESSION hSession, PRTGID pGid)
1065{
1066 RT_NOREF_PV(hSession); RT_NOREF_PV(pGid);
1067 return VERR_NOT_SUPPORTED;
1068}
1069
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