VirtualBox

source: vbox/trunk/src/VBox/Runtime/r3/posix/pipe-posix.cpp@ 34079

Last change on this file since 34079 was 33104, checked in by vboxsync, 14 years ago

RTPipeCreate: Use pipe2 on linux (added in 2.6.27) to avoid racing process creation.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 18.0 KB
Line 
1/* $Id: pipe-posix.cpp 33104 2010-10-13 12:46:59Z vboxsync $ */
2/** @file
3 * IPRT - Anonymous Pipes, POSIX Implementation.
4 */
5
6/*
7 * Copyright (C) 2010 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#include <iprt/pipe.h>
32#include "internal/iprt.h"
33
34#include <iprt/asm.h>
35#include <iprt/assert.h>
36#include <iprt/err.h>
37#include <iprt/mem.h>
38#include <iprt/string.h>
39#include <iprt/thread.h>
40#include "internal/magics.h"
41
42#include <errno.h>
43#include <fcntl.h>
44#include <limits.h>
45#include <unistd.h>
46#include <sys/poll.h>
47#include <sys/stat.h>
48#include <signal.h>
49#ifdef RT_OS_LINUX
50# include <sys/syscall.h>
51#endif
52
53
54/*******************************************************************************
55* Structures and Typedefs *
56*******************************************************************************/
57typedef struct RTPIPEINTERNAL
58{
59 /** Magic value (RTPIPE_MAGIC). */
60 uint32_t u32Magic;
61 /** The file descriptor. */
62 int fd;
63 /** Set if this is the read end, clear if it's the write end. */
64 bool fRead;
65 /** Atomically operated state variable.
66 *
67 * - Bits 0 thru 29 - Users of the new mode.
68 * - Bit 30 - The pipe mode, set indicates blocking.
69 * - Bit 31 - Set when we're switching the mode.
70 */
71 uint32_t volatile u32State;
72} RTPIPEINTERNAL;
73
74
75/*******************************************************************************
76* Defined Constants And Macros *
77*******************************************************************************/
78/** @name RTPIPEINTERNAL::u32State defines
79 * @{ */
80#define RTPIPE_POSIX_BLOCKING UINT32_C(0x40000000)
81#define RTPIPE_POSIX_SWITCHING UINT32_C(0x80000000)
82#define RTPIPE_POSIX_SWITCHING_BIT 31
83#define RTPIPE_POSIX_USERS_MASK UINT32_C(0x3fffffff)
84/** @} */
85
86
87
88/**
89 * Wrapper for calling pipe2() or pipe().
90 *
91 * When using pipe2() the returned handles are marked close-on-exec and does
92 * not risk racing process creation calls on other threads.
93 *
94 * @returns See pipe().
95 * @param paFds See pipe().
96 * @param piNewPipeSyscall Where to cache which call we should used. -1 if
97 * pipe(), 1 if pipe2(), 0 if not yet decided.
98 */
99static int my_pipe_wrapper(int *paFds, int *piNewPipeSyscall)
100{
101 if (*piNewPipeSyscall >= 0)
102 {
103#if defined(RT_OS_LINUX) && defined(__NR_pipe2)
104 long rc = syscall(__NR_pipe2, paFds, O_CLOEXEC);
105 if (rc >= 0)
106 {
107 if (*piNewPipeSyscall == 0)
108 *piNewPipeSyscall = 1;
109 return (int)rc;
110 }
111#endif
112 *piNewPipeSyscall = -1;
113 }
114
115 return pipe(paFds);
116}
117
118
119RTDECL(int) RTPipeCreate(PRTPIPE phPipeRead, PRTPIPE phPipeWrite, uint32_t fFlags)
120{
121 AssertPtrReturn(phPipeRead, VERR_INVALID_POINTER);
122 AssertPtrReturn(phPipeWrite, VERR_INVALID_POINTER);
123 AssertReturn(!(fFlags & ~RTPIPE_C_VALID_MASK), VERR_INVALID_PARAMETER);
124
125 /*
126 * Create the pipe and clear/set the close-on-exec flag as required.
127 */
128 int aFds[2] = {-1, -1};
129 static int s_iNewPipeSyscall = 0;
130 if (my_pipe_wrapper(aFds, &s_iNewPipeSyscall))
131 return RTErrConvertFromErrno(errno);
132
133 int rc = VINF_SUCCESS;
134 if (s_iNewPipeSyscall > 0)
135 {
136 /* created with close-on-exec set. */
137 if (fFlags & RTPIPE_C_INHERIT_READ)
138 {
139 if (fcntl(aFds[0], F_SETFD, 0))
140 rc = RTErrConvertFromErrno(errno);
141 }
142
143 if (fFlags & RTPIPE_C_INHERIT_WRITE)
144 {
145 if (fcntl(aFds[1], F_SETFD, 0))
146 rc = RTErrConvertFromErrno(errno);
147 }
148 }
149 else
150 {
151 /* created with close-on-exec cleared. */
152 if (!(fFlags & RTPIPE_C_INHERIT_READ))
153 {
154 if (fcntl(aFds[0], F_SETFD, FD_CLOEXEC))
155 rc = RTErrConvertFromErrno(errno);
156 }
157
158 if (!(fFlags & RTPIPE_C_INHERIT_WRITE))
159 {
160 if (fcntl(aFds[1], F_SETFD, FD_CLOEXEC))
161 rc = RTErrConvertFromErrno(errno);
162 }
163 }
164
165 if (RT_SUCCESS(rc))
166 {
167 /*
168 * Create the two handles.
169 */
170 RTPIPEINTERNAL *pThisR = (RTPIPEINTERNAL *)RTMemAlloc(sizeof(RTPIPEINTERNAL));
171 if (pThisR)
172 {
173 RTPIPEINTERNAL *pThisW = (RTPIPEINTERNAL *)RTMemAlloc(sizeof(RTPIPEINTERNAL));
174 if (pThisW)
175 {
176 pThisR->u32Magic = RTPIPE_MAGIC;
177 pThisW->u32Magic = RTPIPE_MAGIC;
178 pThisR->fd = aFds[0];
179 pThisW->fd = aFds[1];
180 pThisR->fRead = true;
181 pThisW->fRead = false;
182 pThisR->u32State = RTPIPE_POSIX_BLOCKING;
183 pThisW->u32State = RTPIPE_POSIX_BLOCKING;
184
185 *phPipeRead = pThisR;
186 *phPipeWrite = pThisW;
187
188 /*
189 * Before we leave, make sure to shut up SIGPIPE.
190 */
191 signal(SIGPIPE, SIG_IGN);
192 return VINF_SUCCESS;
193 }
194
195 RTMemFree(pThisR);
196 rc = VERR_NO_MEMORY;
197 }
198 else
199 rc = VERR_NO_MEMORY;
200 }
201
202 close(aFds[0]);
203 close(aFds[1]);
204 return rc;
205}
206
207
208RTDECL(int) RTPipeClose(RTPIPE hPipe)
209{
210 RTPIPEINTERNAL *pThis = hPipe;
211 if (pThis == NIL_RTPIPE)
212 return VINF_SUCCESS;
213 AssertPtrReturn(pThis, VERR_INVALID_PARAMETER);
214 AssertReturn(pThis->u32Magic == RTPIPE_MAGIC, VERR_INVALID_HANDLE);
215
216 /*
217 * Do the cleanup.
218 */
219 AssertReturn(ASMAtomicCmpXchgU32(&pThis->u32Magic, ~RTPIPE_MAGIC, RTPIPE_MAGIC), VERR_INVALID_HANDLE);
220
221 int fd = pThis->fd;
222 pThis->fd = -1;
223 close(fd);
224
225 if (ASMAtomicReadU32(&pThis->u32State) & RTPIPE_POSIX_USERS_MASK)
226 {
227 AssertFailed();
228 RTThreadSleep(1);
229 }
230
231 RTMemFree(pThis);
232
233 return VINF_SUCCESS;
234}
235
236RTDECL(int) RTPipeFromNative(PRTPIPE phPipe, RTHCINTPTR hNativePipe, uint32_t fFlags)
237{
238 AssertPtrReturn(phPipe, VERR_INVALID_POINTER);
239 AssertReturn(!(fFlags & ~RTPIPE_N_VALID_MASK), VERR_INVALID_PARAMETER);
240 AssertReturn(!!(fFlags & RTPIPE_N_READ) != !!(fFlags & RTPIPE_N_WRITE), VERR_INVALID_PARAMETER);
241
242 /*
243 * Get and validate the pipe handle info.
244 */
245 int hNative = (int)hNativePipe;
246 struct stat st;
247 AssertReturn(fstat(hNative, &st) == 0, RTErrConvertFromErrno(errno));
248 AssertMsgReturn(S_ISFIFO(st.st_mode) || S_ISSOCK(st.st_mode), ("%#x (%o)\n", st.st_mode, st.st_mode), VERR_INVALID_HANDLE);
249
250 int fFd = fcntl(hNative, F_GETFL, 0);
251 AssertReturn(fFd != -1, VERR_INVALID_HANDLE);
252 AssertMsgReturn( (fFd & O_ACCMODE) == (fFlags & RTPIPE_N_READ ? O_RDONLY : O_WRONLY)
253 || (fFd & O_ACCMODE) == O_RDWR /* Solaris creates bi-directional pipes. */
254 , ("%#x\n", fFd), VERR_INVALID_HANDLE);
255
256 /*
257 * Create the handle.
258 */
259 RTPIPEINTERNAL *pThis = (RTPIPEINTERNAL *)RTMemAlloc(sizeof(RTPIPEINTERNAL));
260 if (!pThis)
261 return VERR_NO_MEMORY;
262
263 pThis->u32Magic = RTPIPE_MAGIC;
264 pThis->fd = hNative;
265 pThis->fRead = !!(fFlags & RTPIPE_N_READ);
266 pThis->u32State = fFd & O_NONBLOCK ? 0 : RTPIPE_POSIX_BLOCKING;
267
268 /*
269 * Fix up inheritability and shut up SIGPIPE and we're done.
270 */
271 if (fcntl(hNative, F_SETFD, fFlags & RTPIPE_N_INHERIT ? 0 : FD_CLOEXEC) == 0)
272 {
273 signal(SIGPIPE, SIG_IGN);
274 *phPipe = pThis;
275 return VINF_SUCCESS;
276 }
277
278 int rc = RTErrConvertFromErrno(errno);
279 RTMemFree(pThis);
280 return rc;
281}
282
283
284RTDECL(RTHCINTPTR) RTPipeToNative(RTPIPE hPipe)
285{
286 RTPIPEINTERNAL *pThis = hPipe;
287 AssertPtrReturn(pThis, -1);
288 AssertReturn(pThis->u32Magic == RTPIPE_MAGIC, -1);
289
290 return pThis->fd;
291}
292
293
294/**
295 * Prepare blocking mode.
296 *
297 * @returns VINF_SUCCESS
298 * @retval VERR_WRONG_ORDER
299 * @retval VERR_INTERNAL_ERROR_4
300 *
301 * @param pThis The pipe handle.
302 */
303static int rtPipeTryBlocking(RTPIPEINTERNAL *pThis)
304{
305 /*
306 * Update the state.
307 */
308 for (;;)
309 {
310 uint32_t u32State = ASMAtomicReadU32(&pThis->u32State);
311 uint32_t const u32StateOld = u32State;
312 uint32_t const cUsers = (u32State & RTPIPE_POSIX_USERS_MASK);
313
314 if (u32State & RTPIPE_POSIX_BLOCKING)
315 {
316 AssertReturn(cUsers < RTPIPE_POSIX_USERS_MASK / 2, VERR_INTERNAL_ERROR_4);
317 u32State &= ~RTPIPE_POSIX_USERS_MASK;
318 u32State |= cUsers + 1;
319 if (ASMAtomicCmpXchgU32(&pThis->u32State, u32State, u32StateOld))
320 {
321 if (u32State & RTPIPE_POSIX_SWITCHING)
322 break;
323 return VINF_SUCCESS;
324 }
325 }
326 else if (cUsers == 0)
327 {
328 u32State = 1 | RTPIPE_POSIX_SWITCHING | RTPIPE_POSIX_BLOCKING;
329 if (ASMAtomicCmpXchgU32(&pThis->u32State, u32State, u32StateOld))
330 break;
331 }
332 else
333 return VERR_WRONG_ORDER;
334 ASMNopPause();
335 }
336
337 /*
338 * Do the switching.
339 */
340 int fFlags = fcntl(pThis->fd, F_GETFL, 0);
341 if (fFlags != -1)
342 {
343 if ( !(fFlags & O_NONBLOCK)
344 || fcntl(pThis->fd, F_SETFL, fFlags & ~O_NONBLOCK) != -1)
345 {
346 ASMAtomicBitClear(&pThis->u32State, RTPIPE_POSIX_SWITCHING_BIT);
347 return VINF_SUCCESS;
348 }
349 }
350
351 ASMAtomicDecU32(&pThis->u32State);
352 return RTErrConvertFromErrno(errno);
353}
354
355
356/**
357 * Prepare non-blocking mode.
358 *
359 * @returns VINF_SUCCESS
360 * @retval VERR_WRONG_ORDER
361 * @retval VERR_INTERNAL_ERROR_4
362 *
363 * @param pThis The pipe handle.
364 */
365static int rtPipeTryNonBlocking(RTPIPEINTERNAL *pThis)
366{
367 /*
368 * Update the state.
369 */
370 for (;;)
371 {
372 uint32_t u32State = ASMAtomicReadU32(&pThis->u32State);
373 uint32_t const u32StateOld = u32State;
374 uint32_t const cUsers = (u32State & RTPIPE_POSIX_USERS_MASK);
375
376 if (!(u32State & RTPIPE_POSIX_BLOCKING))
377 {
378 AssertReturn(cUsers < RTPIPE_POSIX_USERS_MASK / 2, VERR_INTERNAL_ERROR_4);
379 u32State &= ~RTPIPE_POSIX_USERS_MASK;
380 u32State |= cUsers + 1;
381 if (ASMAtomicCmpXchgU32(&pThis->u32State, u32State, u32StateOld))
382 {
383 if (u32State & RTPIPE_POSIX_SWITCHING)
384 break;
385 return VINF_SUCCESS;
386 }
387 }
388 else if (cUsers == 0)
389 {
390 u32State = 1 | RTPIPE_POSIX_SWITCHING;
391 if (ASMAtomicCmpXchgU32(&pThis->u32State, u32State, u32StateOld))
392 break;
393 }
394 else
395 return VERR_WRONG_ORDER;
396 ASMNopPause();
397 }
398
399 /*
400 * Do the switching.
401 */
402 int fFlags = fcntl(pThis->fd, F_GETFL, 0);
403 if (fFlags != -1)
404 {
405 if ( (fFlags & O_NONBLOCK)
406 || fcntl(pThis->fd, F_SETFL, fFlags | O_NONBLOCK) != -1)
407 {
408 ASMAtomicBitClear(&pThis->u32State, RTPIPE_POSIX_SWITCHING_BIT);
409 return VINF_SUCCESS;
410 }
411 }
412
413 ASMAtomicDecU32(&pThis->u32State);
414 return RTErrConvertFromErrno(errno);
415}
416
417
418/**
419 * Checks if the read pipe has a HUP condition.
420 *
421 * @returns true if HUP, false if no.
422 * @param pThis The pipe handle (read).
423 */
424static bool rtPipePosixHasHup(RTPIPEINTERNAL *pThis)
425{
426 Assert(pThis->fRead);
427
428 struct pollfd PollFd;
429 RT_ZERO(PollFd);
430 PollFd.fd = pThis->fd;
431 PollFd.events = POLLHUP;
432 return poll(&PollFd, 1, 0) >= 1
433 && (PollFd.revents & POLLHUP);
434}
435
436
437RTDECL(int) RTPipeRead(RTPIPE hPipe, void *pvBuf, size_t cbToRead, size_t *pcbRead)
438{
439 RTPIPEINTERNAL *pThis = hPipe;
440 AssertPtrReturn(pThis, VERR_INVALID_HANDLE);
441 AssertReturn(pThis->u32Magic == RTPIPE_MAGIC, VERR_INVALID_HANDLE);
442 AssertReturn(pThis->fRead, VERR_ACCESS_DENIED);
443 AssertPtr(pcbRead);
444 AssertPtr(pvBuf);
445
446 int rc = rtPipeTryNonBlocking(pThis);
447 if (RT_SUCCESS(rc))
448 {
449 ssize_t cbRead = read(pThis->fd, pvBuf, RT_MIN(cbToRead, SSIZE_MAX));
450 if (cbRead >= 0)
451 {
452 if (cbRead || !cbToRead || !rtPipePosixHasHup(pThis))
453 *pcbRead = cbRead;
454 else
455 rc = VERR_BROKEN_PIPE;
456 }
457 else if (errno == EAGAIN)
458 {
459 *pcbRead = 0;
460 rc = VINF_TRY_AGAIN;
461 }
462 else
463 rc = RTErrConvertFromErrno(errno);
464
465 ASMAtomicDecU32(&pThis->u32State);
466 }
467 return rc;
468}
469
470
471RTDECL(int) RTPipeReadBlocking(RTPIPE hPipe, void *pvBuf, size_t cbToRead, size_t *pcbRead)
472{
473 RTPIPEINTERNAL *pThis = hPipe;
474 AssertPtrReturn(pThis, VERR_INVALID_HANDLE);
475 AssertReturn(pThis->u32Magic == RTPIPE_MAGIC, VERR_INVALID_HANDLE);
476 AssertReturn(pThis->fRead, VERR_ACCESS_DENIED);
477 AssertPtr(pvBuf);
478
479 int rc = rtPipeTryBlocking(pThis);
480 if (RT_SUCCESS(rc))
481 {
482 size_t cbTotalRead = 0;
483 while (cbToRead > 0)
484 {
485 ssize_t cbRead = read(pThis->fd, pvBuf, RT_MIN(cbToRead, SSIZE_MAX));
486 if (cbRead < 0)
487 {
488 rc = RTErrConvertFromErrno(errno);
489 break;
490 }
491 if (!cbRead && rtPipePosixHasHup(pThis))
492 {
493 rc = VERR_BROKEN_PIPE;
494 break;
495 }
496
497 /* advance */
498 pvBuf = (char *)pvBuf + cbRead;
499 cbTotalRead += cbRead;
500 cbToRead -= cbRead;
501 }
502
503 if (pcbRead)
504 {
505 *pcbRead = cbTotalRead;
506 if ( RT_FAILURE(rc)
507 && cbTotalRead
508 && rc != VERR_INVALID_POINTER)
509 rc = VINF_SUCCESS;
510 }
511
512 ASMAtomicDecU32(&pThis->u32State);
513 }
514 return rc;
515}
516
517
518RTDECL(int) RTPipeWrite(RTPIPE hPipe, const void *pvBuf, size_t cbToWrite, size_t *pcbWritten)
519{
520 RTPIPEINTERNAL *pThis = hPipe;
521 AssertPtrReturn(pThis, VERR_INVALID_HANDLE);
522 AssertReturn(pThis->u32Magic == RTPIPE_MAGIC, VERR_INVALID_HANDLE);
523 AssertReturn(!pThis->fRead, VERR_ACCESS_DENIED);
524 AssertPtr(pcbWritten);
525 AssertPtr(pvBuf);
526
527 int rc = rtPipeTryNonBlocking(pThis);
528 if (RT_SUCCESS(rc))
529 {
530 if (cbToWrite)
531 {
532 ssize_t cbWritten = write(pThis->fd, pvBuf, RT_MIN(cbToWrite, SSIZE_MAX));
533 if (cbWritten >= 0)
534 *pcbWritten = cbWritten;
535 else if (errno == EAGAIN)
536 {
537 *pcbWritten = 0;
538 rc = VINF_TRY_AGAIN;
539 }
540 else
541 rc = RTErrConvertFromErrno(errno);
542 }
543 else
544 *pcbWritten = 0;
545
546 ASMAtomicDecU32(&pThis->u32State);
547 }
548 return rc;
549}
550
551
552RTDECL(int) RTPipeWriteBlocking(RTPIPE hPipe, const void *pvBuf, size_t cbToWrite, size_t *pcbWritten)
553{
554 RTPIPEINTERNAL *pThis = hPipe;
555 AssertPtrReturn(pThis, VERR_INVALID_HANDLE);
556 AssertReturn(pThis->u32Magic == RTPIPE_MAGIC, VERR_INVALID_HANDLE);
557 AssertReturn(!pThis->fRead, VERR_ACCESS_DENIED);
558 AssertPtr(pvBuf);
559 AssertPtrNull(pcbWritten);
560
561 int rc = rtPipeTryBlocking(pThis);
562 if (RT_SUCCESS(rc))
563 {
564 size_t cbTotalWritten = 0;
565 while (cbToWrite > 0)
566 {
567 ssize_t cbWritten = write(pThis->fd, pvBuf, RT_MIN(cbToWrite, SSIZE_MAX));
568 if (cbWritten < 0)
569 {
570 rc = RTErrConvertFromErrno(errno);
571 break;
572 }
573
574 /* advance */
575 pvBuf = (char const *)pvBuf + cbWritten;
576 cbTotalWritten += cbWritten;
577 cbToWrite -= cbWritten;
578 }
579
580 if (pcbWritten)
581 {
582 *pcbWritten = cbTotalWritten;
583 if ( RT_FAILURE(rc)
584 && cbTotalWritten
585 && rc != VERR_INVALID_POINTER)
586 rc = VINF_SUCCESS;
587 }
588
589 ASMAtomicDecU32(&pThis->u32State);
590 }
591 return rc;
592}
593
594
595RTDECL(int) RTPipeFlush(RTPIPE hPipe)
596{
597 RTPIPEINTERNAL *pThis = hPipe;
598 AssertPtrReturn(pThis, VERR_INVALID_HANDLE);
599 AssertReturn(pThis->u32Magic == RTPIPE_MAGIC, VERR_INVALID_HANDLE);
600 AssertReturn(!pThis->fRead, VERR_ACCESS_DENIED);
601
602 if (fsync(pThis->fd))
603 {
604 if (errno == EINVAL || errno == ENOTSUP)
605 return VERR_NOT_SUPPORTED;
606 return RTErrConvertFromErrno(errno);
607 }
608 return VINF_SUCCESS;
609}
610
611
612RTDECL(int) RTPipeSelectOne(RTPIPE hPipe, RTMSINTERVAL cMillies)
613{
614 RTPIPEINTERNAL *pThis = hPipe;
615 AssertPtrReturn(pThis, VERR_INVALID_HANDLE);
616 AssertReturn(pThis->u32Magic == RTPIPE_MAGIC, VERR_INVALID_HANDLE);
617
618 struct pollfd PollFd;
619 RT_ZERO(PollFd);
620 PollFd.fd = pThis->fd;
621 PollFd.events = POLLHUP | POLLERR;
622 if (pThis->fRead)
623 PollFd.events |= POLLIN | POLLPRI;
624 else
625 PollFd.events |= POLLOUT;
626
627 int timeout;
628 if ( cMillies == RT_INDEFINITE_WAIT
629 || cMillies >= INT_MAX /* lazy bird */)
630 timeout = -1;
631 else
632 timeout = cMillies;
633
634 int rc = poll(&PollFd, 1, 0);
635 if (rc == -1)
636 return RTErrConvertFromErrno(errno);
637 return rc > 0 ? VINF_SUCCESS : VERR_TIMEOUT;
638}
639
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