VirtualBox

source: vbox/trunk/src/VBox/Runtime/r3/socket.cpp@ 40532

Last change on this file since 40532 was 39807, checked in by vboxsync, 13 years ago

duh2!

  • Property svn:eol-style set to native
  • Property svn:keywords set to Id
File size: 53.3 KB
Line 
1/* $Id: socket.cpp 39807 2012-01-19 10:43:42Z vboxsync $ */
2/** @file
3 * IPRT - Network Sockets.
4 */
5
6/*
7 * Copyright (C) 2006-2011 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#ifdef RT_OS_WINDOWS
32# include <winsock2.h>
33#else /* !RT_OS_WINDOWS */
34# include <errno.h>
35# include <sys/stat.h>
36# include <sys/socket.h>
37# include <netinet/in.h>
38# include <netinet/tcp.h>
39# include <arpa/inet.h>
40# ifdef IPRT_WITH_TCPIP_V6
41# include <netinet6/in6.h>
42# endif
43# include <sys/un.h>
44# include <netdb.h>
45# include <unistd.h>
46# include <fcntl.h>
47# include <sys/uio.h>
48#endif /* !RT_OS_WINDOWS */
49#include <limits.h>
50
51#include "internal/iprt.h"
52#include <iprt/socket.h>
53
54#include <iprt/alloca.h>
55#include <iprt/asm.h>
56#include <iprt/assert.h>
57#include <iprt/ctype.h>
58#include <iprt/err.h>
59#include <iprt/mempool.h>
60#include <iprt/poll.h>
61#include <iprt/string.h>
62#include <iprt/thread.h>
63#include <iprt/time.h>
64#include <iprt/mem.h>
65#include <iprt/sg.h>
66#include <iprt/log.h>
67
68#include "internal/magics.h"
69#include "internal/socket.h"
70
71
72/*******************************************************************************
73* Defined Constants And Macros *
74*******************************************************************************/
75/* non-standard linux stuff (it seems). */
76#ifndef MSG_NOSIGNAL
77# define MSG_NOSIGNAL 0
78#endif
79
80/* Windows has different names for SHUT_XXX. */
81#ifndef SHUT_RDWR
82# ifdef SD_BOTH
83# define SHUT_RDWR SD_BOTH
84# else
85# define SHUT_RDWR 2
86# endif
87#endif
88#ifndef SHUT_WR
89# ifdef SD_SEND
90# define SHUT_WR SD_SEND
91# else
92# define SHUT_WR 1
93# endif
94#endif
95#ifndef SHUT_RD
96# ifdef SD_RECEIVE
97# define SHUT_RD SD_RECEIVE
98# else
99# define SHUT_RD 0
100# endif
101#endif
102
103/* fixup backlevel OSes. */
104#if defined(RT_OS_OS2) || defined(RT_OS_WINDOWS)
105# define socklen_t int
106#endif
107
108/** How many pending connection. */
109#define RTTCP_SERVER_BACKLOG 10
110
111
112/*******************************************************************************
113* Structures and Typedefs *
114*******************************************************************************/
115/**
116 * Socket handle data.
117 *
118 * This is mainly required for implementing RTPollSet on Windows.
119 */
120typedef struct RTSOCKETINT
121{
122 /** Magic number (RTSOCKET_MAGIC). */
123 uint32_t u32Magic;
124 /** Exclusive user count.
125 * This is used to prevent two threads from accessing the handle concurrently.
126 * It can be higher than 1 if this handle is reference multiple times in a
127 * polling set (Windows). */
128 uint32_t volatile cUsers;
129 /** The native socket handle. */
130 RTSOCKETNATIVE hNative;
131 /** Indicates whether the handle has been closed or not. */
132 bool volatile fClosed;
133 /** Indicates whether the socket is operating in blocking or non-blocking mode
134 * currently. */
135 bool fBlocking;
136#ifdef RT_OS_WINDOWS
137 /** The event semaphore we've associated with the socket handle.
138 * This is WSA_INVALID_EVENT if not done. */
139 WSAEVENT hEvent;
140 /** The pollset currently polling this socket. This is NIL if no one is
141 * polling. */
142 RTPOLLSET hPollSet;
143 /** The events we're polling for. */
144 uint32_t fPollEvts;
145 /** The events we're currently subscribing to with WSAEventSelect.
146 * This is ZERO if we're currently not subscribing to anything. */
147 uint32_t fSubscribedEvts;
148 /** Saved events which are only posted once. */
149 uint32_t fEventsSaved;
150#endif /* RT_OS_WINDOWS */
151} RTSOCKETINT;
152
153
154/**
155 * Address union used internally for things like getpeername and getsockname.
156 */
157typedef union RTSOCKADDRUNION
158{
159 struct sockaddr Addr;
160 struct sockaddr_in IPv4;
161#ifdef IPRT_WITH_TCPIP_V6
162 struct sockaddr_in6 IPv6;
163#endif
164} RTSOCKADDRUNION;
165
166
167/**
168 * Get the last error as an iprt status code.
169 *
170 * @returns IPRT status code.
171 */
172DECLINLINE(int) rtSocketError(void)
173{
174#ifdef RT_OS_WINDOWS
175 return RTErrConvertFromWin32(WSAGetLastError());
176#else
177 return RTErrConvertFromErrno(errno);
178#endif
179}
180
181
182/**
183 * Resets the last error.
184 */
185DECLINLINE(void) rtSocketErrorReset(void)
186{
187#ifdef RT_OS_WINDOWS
188 WSASetLastError(0);
189#else
190 errno = 0;
191#endif
192}
193
194
195/**
196 * Get the last resolver error as an iprt status code.
197 *
198 * @returns iprt status code.
199 */
200int rtSocketResolverError(void)
201{
202#ifdef RT_OS_WINDOWS
203 return RTErrConvertFromWin32(WSAGetLastError());
204#else
205 switch (h_errno)
206 {
207 case HOST_NOT_FOUND:
208 return VERR_NET_HOST_NOT_FOUND;
209 case NO_DATA:
210 return VERR_NET_ADDRESS_NOT_AVAILABLE;
211 case NO_RECOVERY:
212 return VERR_IO_GEN_FAILURE;
213 case TRY_AGAIN:
214 return VERR_TRY_AGAIN;
215
216 default:
217 return VERR_UNRESOLVED_ERROR;
218 }
219#endif
220}
221
222
223/**
224 * Converts from a native socket address to a generic IPRT network address.
225 *
226 * @returns IPRT status code.
227 * @param pSrc The source address.
228 * @param cbSrc The size of the source address.
229 * @param pAddr Where to return the generic IPRT network
230 * address.
231 */
232static int rtSocketNetAddrFromAddr(RTSOCKADDRUNION const *pSrc, size_t cbSrc, PRTNETADDR pAddr)
233{
234 /*
235 * Convert the address.
236 */
237 if ( cbSrc == sizeof(struct sockaddr_in)
238 && pSrc->Addr.sa_family == AF_INET)
239 {
240 RT_ZERO(*pAddr);
241 pAddr->enmType = RTNETADDRTYPE_IPV4;
242 pAddr->uPort = RT_N2H_U16(pSrc->IPv4.sin_port);
243 pAddr->uAddr.IPv4.u = pSrc->IPv4.sin_addr.s_addr;
244 }
245#ifdef IPRT_WITH_TCPIP_V6
246 else if ( cbSrc == sizeof(struct sockaddr_in6)
247 && pSrc->Addr.sa_family == AF_INET6)
248 {
249 RT_ZERO(*pAddr);
250 pAddr->enmType = RTNETADDRTYPE_IPV6;
251 pAddr->uPort = RT_N2H_U16(pSrc->IPv6.sin6_port);
252 pAddr->uAddr.IPv6.au32[0] = pSrc->IPv6.sin6_addr.s6_addr32[0];
253 pAddr->uAddr.IPv6.au32[1] = pSrc->IPv6.sin6_addr.s6_addr32[1];
254 pAddr->uAddr.IPv6.au32[2] = pSrc->IPv6.sin6_addr.s6_addr32[2];
255 pAddr->uAddr.IPv6.au32[3] = pSrc->IPv6.sin6_addr.s6_addr32[3];
256 }
257#endif
258 else
259 return VERR_NET_ADDRESS_FAMILY_NOT_SUPPORTED;
260 return VINF_SUCCESS;
261}
262
263
264/**
265 * Converts from a generic IPRT network address to a native socket address.
266 *
267 * @returns IPRT status code.
268 * @param pAddr Pointer to the generic IPRT network address.
269 * @param pDst The source address.
270 * @param cbSrc The size of the source address.
271 * @param pcbAddr Where to store the size of the returned address.
272 * Optional
273 */
274static int rtSocketAddrFromNetAddr(PCRTNETADDR pAddr, RTSOCKADDRUNION *pDst, size_t cbDst, int *pcbAddr)
275{
276 RT_BZERO(pDst, cbDst);
277 if ( pAddr->enmType == RTNETADDRTYPE_IPV4
278 && cbDst >= sizeof(struct sockaddr_in))
279 {
280 pDst->Addr.sa_family = AF_INET;
281 pDst->IPv4.sin_port = RT_H2N_U16(pAddr->uPort);
282 pDst->IPv4.sin_addr.s_addr = pAddr->uAddr.IPv4.u;
283 if (pcbAddr)
284 *pcbAddr = sizeof(pDst->IPv4);
285 }
286#ifdef IPRT_WITH_TCPIP_V6
287 else if ( pAddr->enmType == RTNETADDRTYPE_IPV6
288 && cbDst >= sizeof(struct sockaddr_in6))
289 {
290 pDst->Addr.sa_family = AF_INET6;
291 pDst->IPv6.sin6_port = RT_H2N_U16(pAddr->uPort);
292 pSrc->IPv6.sin6_addr.s6_addr32[0] = pAddr->uAddr.IPv6.au32[0];
293 pSrc->IPv6.sin6_addr.s6_addr32[1] = pAddr->uAddr.IPv6.au32[1];
294 pSrc->IPv6.sin6_addr.s6_addr32[2] = pAddr->uAddr.IPv6.au32[2];
295 pSrc->IPv6.sin6_addr.s6_addr32[3] = pAddr->uAddr.IPv6.au32[3];
296 if (pcbAddr)
297 *pcbAddr = sizeof(pDst->IPv6);
298 }
299#endif
300 else
301 return VERR_NET_ADDRESS_FAMILY_NOT_SUPPORTED;
302 return VINF_SUCCESS;
303}
304
305
306/**
307 * Tries to lock the socket for exclusive usage by the calling thread.
308 *
309 * Call rtSocketUnlock() to unlock.
310 *
311 * @returns @c true if locked, @c false if not.
312 * @param pThis The socket structure.
313 */
314DECLINLINE(bool) rtSocketTryLock(RTSOCKETINT *pThis)
315{
316 return ASMAtomicCmpXchgU32(&pThis->cUsers, 1, 0);
317}
318
319
320/**
321 * Unlocks the socket.
322 *
323 * @param pThis The socket structure.
324 */
325DECLINLINE(void) rtSocketUnlock(RTSOCKETINT *pThis)
326{
327 ASMAtomicCmpXchgU32(&pThis->cUsers, 0, 1);
328}
329
330
331/**
332 * The slow path of rtSocketSwitchBlockingMode that does the actual switching.
333 *
334 * @returns IPRT status code.
335 * @param pThis The socket structure.
336 * @param fBlocking The desired mode of operation.
337 * @remarks Do not call directly.
338 */
339static int rtSocketSwitchBlockingModeSlow(RTSOCKETINT *pThis, bool fBlocking)
340{
341#ifdef RT_OS_WINDOWS
342 u_long uBlocking = fBlocking ? 0 : 1;
343 if (ioctlsocket(pThis->hNative, FIONBIO, &uBlocking))
344 return rtSocketError();
345
346#else
347 int fFlags = fcntl(pThis->hNative, F_GETFL, 0);
348 if (fFlags == -1)
349 return rtSocketError();
350
351 if (fBlocking)
352 fFlags &= ~O_NONBLOCK;
353 else
354 fFlags |= O_NONBLOCK;
355 if (fcntl(pThis->hNative, F_SETFL, fFlags) == -1)
356 return rtSocketError();
357#endif
358
359 pThis->fBlocking = fBlocking;
360 return VINF_SUCCESS;
361}
362
363
364/**
365 * Switches the socket to the desired blocking mode if necessary.
366 *
367 * The socket must be locked.
368 *
369 * @returns IPRT status code.
370 * @param pThis The socket structure.
371 * @param fBlocking The desired mode of operation.
372 */
373DECLINLINE(int) rtSocketSwitchBlockingMode(RTSOCKETINT *pThis, bool fBlocking)
374{
375 if (pThis->fBlocking != fBlocking)
376 return rtSocketSwitchBlockingModeSlow(pThis, fBlocking);
377 return VINF_SUCCESS;
378}
379
380
381/**
382 * Creates an IPRT socket handle for a native one.
383 *
384 * @returns IPRT status code.
385 * @param ppSocket Where to return the IPRT socket handle.
386 * @param hNative The native handle.
387 */
388int rtSocketCreateForNative(RTSOCKETINT **ppSocket, RTSOCKETNATIVE hNative)
389{
390 RTSOCKETINT *pThis = (RTSOCKETINT *)RTMemPoolAlloc(RTMEMPOOL_DEFAULT, sizeof(*pThis));
391 if (!pThis)
392 return VERR_NO_MEMORY;
393 pThis->u32Magic = RTSOCKET_MAGIC;
394 pThis->cUsers = 0;
395 pThis->hNative = hNative;
396 pThis->fClosed = false;
397 pThis->fBlocking = true;
398#ifdef RT_OS_WINDOWS
399 pThis->hEvent = WSA_INVALID_EVENT;
400 pThis->hPollSet = NIL_RTPOLLSET;
401 pThis->fPollEvts = 0;
402 pThis->fSubscribedEvts = 0;
403#endif
404 *ppSocket = pThis;
405 return VINF_SUCCESS;
406}
407
408
409RTDECL(int) RTSocketFromNative(PRTSOCKET phSocket, RTHCINTPTR uNative)
410{
411 AssertReturn(uNative != NIL_RTSOCKETNATIVE, VERR_INVALID_PARAMETER);
412#ifndef RT_OS_WINDOWS
413 AssertReturn(uNative >= 0, VERR_INVALID_PARAMETER);
414#endif
415 AssertPtrReturn(phSocket, VERR_INVALID_POINTER);
416 return rtSocketCreateForNative(phSocket, uNative);
417}
418
419
420/**
421 * Wrapper around socket().
422 *
423 * @returns IPRT status code.
424 * @param phSocket Where to store the handle to the socket on
425 * success.
426 * @param iDomain The protocol family (PF_XXX).
427 * @param iType The socket type (SOCK_XXX).
428 * @param iProtocol Socket parameter, usually 0.
429 */
430int rtSocketCreate(PRTSOCKET phSocket, int iDomain, int iType, int iProtocol)
431{
432 /*
433 * Create the socket.
434 */
435 RTSOCKETNATIVE hNative = socket(iDomain, iType, iProtocol);
436 if (hNative == NIL_RTSOCKETNATIVE)
437 return rtSocketError();
438
439 /*
440 * Wrap it.
441 */
442 int rc = rtSocketCreateForNative(phSocket, hNative);
443 if (RT_FAILURE(rc))
444 {
445#ifdef RT_OS_WINDOWS
446 closesocket(hNative);
447#else
448 close(hNative);
449#endif
450 }
451 return rc;
452}
453
454
455RTDECL(uint32_t) RTSocketRetain(RTSOCKET hSocket)
456{
457 RTSOCKETINT *pThis = hSocket;
458 AssertPtrReturn(pThis, UINT32_MAX);
459 AssertReturn(pThis->u32Magic == RTSOCKET_MAGIC, UINT32_MAX);
460 return RTMemPoolRetain(pThis);
461}
462
463
464/**
465 * Worker for RTSocketRelease and RTSocketClose.
466 *
467 * @returns IPRT status code.
468 * @param pThis The socket handle instance data.
469 * @param fDestroy Whether we're reaching ref count zero.
470 */
471static int rtSocketCloseIt(RTSOCKETINT *pThis, bool fDestroy)
472{
473 /*
474 * Invalidate the handle structure on destroy.
475 */
476 if (fDestroy)
477 {
478 Assert(ASMAtomicReadU32(&pThis->u32Magic) == RTSOCKET_MAGIC);
479 ASMAtomicWriteU32(&pThis->u32Magic, RTSOCKET_MAGIC_DEAD);
480 }
481
482 int rc = VINF_SUCCESS;
483 if (ASMAtomicCmpXchgBool(&pThis->fClosed, true, false))
484 {
485 /*
486 * Close the native handle.
487 */
488 RTSOCKETNATIVE hNative = pThis->hNative;
489 if (hNative != NIL_RTSOCKETNATIVE)
490 {
491 pThis->hNative = NIL_RTSOCKETNATIVE;
492
493#ifdef RT_OS_WINDOWS
494 if (closesocket(hNative))
495#else
496 if (close(hNative))
497#endif
498 {
499 rc = rtSocketError();
500#ifdef RT_OS_WINDOWS
501 AssertMsgFailed(("\"%s\": closesocket(%p) -> %Rrc\n", (uintptr_t)hNative, rc));
502#else
503 AssertMsgFailed(("\"%s\": close(%d) -> %Rrc\n", hNative, rc));
504#endif
505 }
506 }
507
508#ifdef RT_OS_WINDOWS
509 /*
510 * Close the event.
511 */
512 WSAEVENT hEvent = pThis->hEvent;
513 if (hEvent == WSA_INVALID_EVENT)
514 {
515 pThis->hEvent = WSA_INVALID_EVENT;
516 WSACloseEvent(hEvent);
517 }
518#endif
519 }
520
521 return rc;
522}
523
524
525RTDECL(uint32_t) RTSocketRelease(RTSOCKET hSocket)
526{
527 RTSOCKETINT *pThis = hSocket;
528 if (pThis == NIL_RTSOCKET)
529 return 0;
530 AssertPtrReturn(pThis, UINT32_MAX);
531 AssertReturn(pThis->u32Magic == RTSOCKET_MAGIC, UINT32_MAX);
532
533 /* get the refcount without killing it... */
534 uint32_t cRefs = RTMemPoolRefCount(pThis);
535 AssertReturn(cRefs != UINT32_MAX, UINT32_MAX);
536 if (cRefs == 1)
537 rtSocketCloseIt(pThis, true);
538
539 return RTMemPoolRelease(RTMEMPOOL_DEFAULT, pThis);
540}
541
542
543RTDECL(int) RTSocketClose(RTSOCKET hSocket)
544{
545 RTSOCKETINT *pThis = hSocket;
546 if (pThis == NIL_RTSOCKET)
547 return VINF_SUCCESS;
548 AssertPtrReturn(pThis, VERR_INVALID_HANDLE);
549 AssertReturn(pThis->u32Magic == RTSOCKET_MAGIC, VERR_INVALID_HANDLE);
550
551 uint32_t cRefs = RTMemPoolRefCount(pThis);
552 AssertReturn(cRefs != UINT32_MAX, UINT32_MAX);
553
554 int rc = rtSocketCloseIt(pThis, cRefs == 1);
555
556 RTMemPoolRelease(RTMEMPOOL_DEFAULT, pThis);
557 return rc;
558}
559
560
561RTDECL(RTHCUINTPTR) RTSocketToNative(RTSOCKET hSocket)
562{
563 RTSOCKETINT *pThis = hSocket;
564 AssertPtrReturn(pThis, RTHCUINTPTR_MAX);
565 AssertReturn(pThis->u32Magic == RTSOCKET_MAGIC, RTHCUINTPTR_MAX);
566 return (RTHCUINTPTR)pThis->hNative;
567}
568
569
570RTDECL(int) RTSocketSetInheritance(RTSOCKET hSocket, bool fInheritable)
571{
572 RTSOCKETINT *pThis = hSocket;
573 AssertPtrReturn(pThis, VERR_INVALID_HANDLE);
574 AssertReturn(pThis->u32Magic == RTSOCKET_MAGIC, VERR_INVALID_HANDLE);
575 AssertReturn(RTMemPoolRefCount(pThis) >= (pThis->cUsers ? 2U : 1U), VERR_CALLER_NO_REFERENCE);
576
577 int rc = VINF_SUCCESS;
578#ifdef RT_OS_WINDOWS
579 if (!SetHandleInformation((HANDLE)pThis->hNative, HANDLE_FLAG_INHERIT, fInheritable ? HANDLE_FLAG_INHERIT : 0))
580 rc = RTErrConvertFromWin32(GetLastError());
581#else
582 if (fcntl(pThis->hNative, F_SETFD, fInheritable ? 0 : FD_CLOEXEC) < 0)
583 rc = RTErrConvertFromErrno(errno);
584#endif
585
586 return rc;
587}
588
589static bool rtSocketIsIPv4Numerical(const char *pszAddress, PRTNETADDRIPV4 pAddr)
590{
591
592 /* Empty address resolves to the INADDR_ANY address (good for bind). */
593 if (!pszAddress || !*pszAddress)
594 {
595 pAddr->u = INADDR_ANY;
596 return true;
597 }
598
599 /* Four quads? */
600 char *psz = (char *)pszAddress;
601 for (int i = 0; i < 4; i++)
602 {
603 uint8_t u8;
604 int rc = RTStrToUInt8Ex(psz, &psz, 0, &u8);
605 if (rc != VINF_SUCCESS && rc != VWRN_TRAILING_CHARS)
606 return false;
607 if (*psz != (i < 3 ? '.' : '\0'))
608 return false;
609 psz++;
610
611 pAddr->au8[i] = u8; /* big endian */
612 }
613
614 return true;
615}
616
617RTDECL(int) RTSocketParseInetAddress(const char *pszAddress, unsigned uPort, PRTNETADDR pAddr)
618{
619 int rc;
620
621 /*
622 * Validate input.
623 */
624 AssertReturn(uPort > 0, VERR_INVALID_PARAMETER);
625 AssertPtrNullReturn(pszAddress, VERR_INVALID_POINTER);
626
627#ifdef RT_OS_WINDOWS
628 /*
629 * Initialize WinSock and check version.
630 */
631 WORD wVersionRequested = MAKEWORD(1, 1);
632 WSADATA wsaData;
633 rc = WSAStartup(wVersionRequested, &wsaData);
634 if (wsaData.wVersion != wVersionRequested)
635 {
636 AssertMsgFailed(("Wrong winsock version\n"));
637 return VERR_NOT_SUPPORTED;
638 }
639#endif
640
641 /*
642 * Resolve the address. Pretty crude at the moment, but we have to make
643 * sure to not ask the NT 4 gethostbyname about an IPv4 address as it may
644 * give a wrong answer.
645 */
646 /** @todo this only supports IPv4, and IPv6 support needs to be added.
647 * It probably needs to be converted to getaddrinfo(). */
648 RTNETADDRIPV4 IPv4Quad;
649 if (rtSocketIsIPv4Numerical(pszAddress, &IPv4Quad))
650 {
651 Log3(("rtSocketIsIPv4Numerical: %#x (%RTnaipv4)\n", pszAddress, IPv4Quad.u, IPv4Quad));
652 RT_ZERO(*pAddr);
653 pAddr->enmType = RTNETADDRTYPE_IPV4;
654 pAddr->uPort = uPort;
655 pAddr->uAddr.IPv4 = IPv4Quad;
656 return VINF_SUCCESS;
657 }
658
659 struct hostent *pHostEnt;
660 pHostEnt = gethostbyname(pszAddress);
661 if (!pHostEnt)
662 {
663 rc = rtSocketResolverError();
664 AssertMsgFailed(("Could not resolve '%s', rc=%Rrc\n", pszAddress, rc));
665 return rc;
666 }
667
668 if (pHostEnt->h_addrtype == AF_INET)
669 {
670 RT_ZERO(*pAddr);
671 pAddr->enmType = RTNETADDRTYPE_IPV4;
672 pAddr->uPort = uPort;
673 pAddr->uAddr.IPv4.u = ((struct in_addr *)pHostEnt->h_addr)->s_addr;
674 Log3(("gethostbyname: %s -> %#x (%RTnaipv4)\n", pszAddress, pAddr->uAddr.IPv4.u, pAddr->uAddr.IPv4));
675 }
676 else
677 return VERR_NET_ADDRESS_FAMILY_NOT_SUPPORTED;
678
679 return VINF_SUCCESS;
680}
681
682
683RTDECL(int) RTSocketRead(RTSOCKET hSocket, void *pvBuffer, size_t cbBuffer, size_t *pcbRead)
684{
685 /*
686 * Validate input.
687 */
688 RTSOCKETINT *pThis = hSocket;
689 AssertPtrReturn(pThis, VERR_INVALID_HANDLE);
690 AssertReturn(pThis->u32Magic == RTSOCKET_MAGIC, VERR_INVALID_HANDLE);
691 AssertReturn(cbBuffer > 0, VERR_INVALID_PARAMETER);
692 AssertPtr(pvBuffer);
693 AssertReturn(rtSocketTryLock(pThis), VERR_CONCURRENT_ACCESS);
694
695
696 int rc = rtSocketSwitchBlockingMode(pThis, true /* fBlocking */);
697 if (RT_FAILURE(rc))
698 return rc;
699
700 /*
701 * Read loop.
702 * If pcbRead is NULL we have to fill the entire buffer!
703 */
704 size_t cbRead = 0;
705 size_t cbToRead = cbBuffer;
706 for (;;)
707 {
708 rtSocketErrorReset();
709#ifdef RT_OS_WINDOWS
710 int cbNow = cbToRead >= INT_MAX/2 ? INT_MAX/2 : (int)cbToRead;
711#else
712 size_t cbNow = cbToRead;
713#endif
714 ssize_t cbBytesRead = recv(pThis->hNative, (char *)pvBuffer + cbRead, cbNow, MSG_NOSIGNAL);
715 if (cbBytesRead <= 0)
716 {
717 rc = rtSocketError();
718 Assert(RT_FAILURE_NP(rc) || cbBytesRead == 0);
719 if (RT_SUCCESS_NP(rc))
720 {
721 if (!pcbRead)
722 rc = VERR_NET_SHUTDOWN;
723 else
724 {
725 *pcbRead = 0;
726 rc = VINF_SUCCESS;
727 }
728 }
729 break;
730 }
731 if (pcbRead)
732 {
733 /* return partial data */
734 *pcbRead = cbBytesRead;
735 break;
736 }
737
738 /* read more? */
739 cbRead += cbBytesRead;
740 if (cbRead == cbBuffer)
741 break;
742
743 /* next */
744 cbToRead = cbBuffer - cbRead;
745 }
746
747 rtSocketUnlock(pThis);
748 return rc;
749}
750
751
752RTDECL(int) RTSocketReadFrom(RTSOCKET hSocket, void *pvBuffer, size_t cbBuffer, size_t *pcbRead, PRTNETADDR pSrcAddr)
753{
754 /*
755 * Validate input.
756 */
757 RTSOCKETINT *pThis = hSocket;
758 AssertPtrReturn(pThis, VERR_INVALID_HANDLE);
759 AssertReturn(pThis->u32Magic == RTSOCKET_MAGIC, VERR_INVALID_HANDLE);
760 AssertReturn(cbBuffer > 0, VERR_INVALID_PARAMETER);
761 AssertPtr(pvBuffer);
762 AssertPtr(pcbRead);
763 AssertReturn(rtSocketTryLock(pThis), VERR_CONCURRENT_ACCESS);
764
765 int rc = rtSocketSwitchBlockingMode(pThis, true /* fBlocking */);
766 if (RT_FAILURE(rc))
767 return rc;
768
769 /*
770 * Read data.
771 */
772 size_t cbRead = 0;
773 size_t cbToRead = cbBuffer;
774 rtSocketErrorReset();
775 RTSOCKADDRUNION u;
776#ifdef RT_OS_WINDOWS
777 int cbNow = cbToRead >= INT_MAX/2 ? INT_MAX/2 : (int)cbToRead;
778 int cbAddr = sizeof(u);
779#else
780 size_t cbNow = cbToRead;
781 socklen_t cbAddr = sizeof(u);
782#endif
783 ssize_t cbBytesRead = recvfrom(pThis->hNative, (char *)pvBuffer + cbRead, cbNow, MSG_NOSIGNAL, &u.Addr, &cbAddr);
784 if (cbBytesRead <= 0)
785 {
786 rc = rtSocketError();
787 Assert(RT_FAILURE_NP(rc) || cbBytesRead == 0);
788 if (RT_SUCCESS_NP(rc))
789 {
790 *pcbRead = 0;
791 rc = VINF_SUCCESS;
792 }
793 }
794 else
795 {
796 if (pSrcAddr)
797 rc = rtSocketNetAddrFromAddr(&u, cbAddr, pSrcAddr);
798 *pcbRead = cbBytesRead;
799 }
800
801 rtSocketUnlock(pThis);
802 return rc;
803}
804
805
806RTDECL(int) RTSocketWrite(RTSOCKET hSocket, const void *pvBuffer, size_t cbBuffer)
807{
808 /*
809 * Validate input.
810 */
811 RTSOCKETINT *pThis = hSocket;
812 AssertPtrReturn(pThis, VERR_INVALID_HANDLE);
813 AssertReturn(pThis->u32Magic == RTSOCKET_MAGIC, VERR_INVALID_HANDLE);
814 AssertReturn(rtSocketTryLock(pThis), VERR_CONCURRENT_ACCESS);
815
816 int rc = rtSocketSwitchBlockingMode(pThis, true /* fBlocking */);
817 if (RT_FAILURE(rc))
818 return rc;
819
820 /*
821 * Try write all at once.
822 */
823#ifdef RT_OS_WINDOWS
824 int cbNow = cbBuffer >= INT_MAX / 2 ? INT_MAX / 2 : (int)cbBuffer;
825#else
826 size_t cbNow = cbBuffer >= SSIZE_MAX ? SSIZE_MAX : cbBuffer;
827#endif
828 ssize_t cbWritten = send(pThis->hNative, (const char *)pvBuffer, cbNow, MSG_NOSIGNAL);
829 if (RT_LIKELY((size_t)cbWritten == cbBuffer && cbWritten >= 0))
830 rc = VINF_SUCCESS;
831 else if (cbWritten < 0)
832 rc = rtSocketError();
833 else
834 {
835 /*
836 * Unfinished business, write the remainder of the request. Must ignore
837 * VERR_INTERRUPTED here if we've managed to send something.
838 */
839 size_t cbSentSoFar = 0;
840 for (;;)
841 {
842 /* advance */
843 cbBuffer -= (size_t)cbWritten;
844 if (!cbBuffer)
845 break;
846 cbSentSoFar += (size_t)cbWritten;
847 pvBuffer = (char const *)pvBuffer + cbWritten;
848
849 /* send */
850#ifdef RT_OS_WINDOWS
851 cbNow = cbBuffer >= INT_MAX / 2 ? INT_MAX / 2 : (int)cbBuffer;
852#else
853 cbNow = cbBuffer >= SSIZE_MAX ? SSIZE_MAX : cbBuffer;
854#endif
855 cbWritten = send(pThis->hNative, (const char *)pvBuffer, cbNow, MSG_NOSIGNAL);
856 if (cbWritten >= 0)
857 AssertMsg(cbBuffer >= (size_t)cbWritten, ("Wrote more than we requested!!! cbWritten=%zu cbBuffer=%zu rtSocketError()=%d\n",
858 cbWritten, cbBuffer, rtSocketError()));
859 else
860 {
861 rc = rtSocketError();
862 if (rc != VERR_INTERNAL_ERROR || cbSentSoFar == 0)
863 break;
864 cbWritten = 0;
865 rc = VINF_SUCCESS;
866 }
867 }
868 }
869
870 rtSocketUnlock(pThis);
871 return rc;
872}
873
874
875RTDECL(int) RTSocketWriteTo(RTSOCKET hSocket, const void *pvBuffer, size_t cbBuffer, PCRTNETADDR pAddr)
876{
877 /*
878 * Validate input.
879 */
880 RTSOCKETINT *pThis = hSocket;
881 AssertPtrReturn(pThis, VERR_INVALID_HANDLE);
882 AssertReturn(pThis->u32Magic == RTSOCKET_MAGIC, VERR_INVALID_HANDLE);
883
884 /* no locking since UDP reads may be done concurrently to writes, and
885 * this is the normal use case of this code. */
886
887 int rc = rtSocketSwitchBlockingMode(pThis, true /* fBlocking */);
888 if (RT_FAILURE(rc))
889 return rc;
890
891 /* Figure out destination address. */
892 struct sockaddr *pSA = NULL;
893#ifdef RT_OS_WINDOWS
894 int cbSA = 0;
895#else
896 socklen_t cbSA = 0;
897#endif
898 RTSOCKADDRUNION u;
899 if (pAddr)
900 {
901 rc = rtSocketAddrFromNetAddr(pAddr, &u, sizeof(u), NULL);
902 if (RT_FAILURE(rc))
903 return rc;
904 pSA = &u.Addr;
905 cbSA = sizeof(u);
906 }
907
908 /*
909 * Must write all at once, otherwise it is a failure.
910 */
911#ifdef RT_OS_WINDOWS
912 int cbNow = cbBuffer >= INT_MAX / 2 ? INT_MAX / 2 : (int)cbBuffer;
913#else
914 size_t cbNow = cbBuffer >= SSIZE_MAX ? SSIZE_MAX : cbBuffer;
915#endif
916 ssize_t cbWritten = sendto(pThis->hNative, (const char *)pvBuffer, cbNow, MSG_NOSIGNAL, pSA, cbSA);
917 if (RT_LIKELY((size_t)cbWritten == cbBuffer && cbWritten >= 0))
918 rc = VINF_SUCCESS;
919 else if (cbWritten < 0)
920 rc = rtSocketError();
921 else
922 rc = VERR_TOO_MUCH_DATA;
923
924 rtSocketUnlock(pThis);
925 return rc;
926}
927
928
929RTDECL(int) RTSocketSgWrite(RTSOCKET hSocket, PCRTSGBUF pSgBuf)
930{
931 /*
932 * Validate input.
933 */
934 RTSOCKETINT *pThis = hSocket;
935 AssertPtrReturn(pThis, VERR_INVALID_HANDLE);
936 AssertReturn(pThis->u32Magic == RTSOCKET_MAGIC, VERR_INVALID_HANDLE);
937 AssertPtrReturn(pSgBuf, VERR_INVALID_PARAMETER);
938 AssertReturn(pSgBuf->cSegs > 0, VERR_INVALID_PARAMETER);
939 AssertReturn(rtSocketTryLock(pThis), VERR_CONCURRENT_ACCESS);
940
941 int rc = rtSocketSwitchBlockingMode(pThis, true /* fBlocking */);
942 if (RT_FAILURE(rc))
943 return rc;
944
945 /*
946 * Construct message descriptor (translate pSgBuf) and send it.
947 */
948 rc = VERR_NO_TMP_MEMORY;
949#ifdef RT_OS_WINDOWS
950 AssertCompileSize(WSABUF, sizeof(RTSGSEG));
951 AssertCompileMemberSize(WSABUF, buf, RT_SIZEOFMEMB(RTSGSEG, pvSeg));
952
953 LPWSABUF paMsg = (LPWSABUF)RTMemTmpAllocZ(pSgBuf->cSegs * sizeof(WSABUF));
954 if (paMsg)
955 {
956 for (unsigned i = 0; i < pSgBuf->cSegs; i++)
957 {
958 paMsg[i].buf = (char *)pSgBuf->paSegs[i].pvSeg;
959 paMsg[i].len = (u_long)pSgBuf->paSegs[i].cbSeg;
960 }
961
962 DWORD dwSent;
963 int hrc = WSASend(pThis->hNative, paMsg, pSgBuf->cSegs, &dwSent,
964 MSG_NOSIGNAL, NULL, NULL);
965 if (!hrc)
966 rc = VINF_SUCCESS;
967/** @todo check for incomplete writes */
968 else
969 rc = rtSocketError();
970
971 RTMemTmpFree(paMsg);
972 }
973
974#else /* !RT_OS_WINDOWS */
975 AssertCompileSize(struct iovec, sizeof(RTSGSEG));
976 AssertCompileMemberSize(struct iovec, iov_base, RT_SIZEOFMEMB(RTSGSEG, pvSeg));
977 AssertCompileMemberSize(struct iovec, iov_len, RT_SIZEOFMEMB(RTSGSEG, cbSeg));
978
979 struct iovec *paMsg = (struct iovec *)RTMemTmpAllocZ(pSgBuf->cSegs * sizeof(struct iovec));
980 if (paMsg)
981 {
982 for (unsigned i = 0; i < pSgBuf->cSegs; i++)
983 {
984 paMsg[i].iov_base = pSgBuf->paSegs[i].pvSeg;
985 paMsg[i].iov_len = pSgBuf->paSegs[i].cbSeg;
986 }
987
988 struct msghdr msgHdr;
989 RT_ZERO(msgHdr);
990 msgHdr.msg_iov = paMsg;
991 msgHdr.msg_iovlen = pSgBuf->cSegs;
992 ssize_t cbWritten = sendmsg(pThis->hNative, &msgHdr, MSG_NOSIGNAL);
993 if (RT_LIKELY(cbWritten >= 0))
994 rc = VINF_SUCCESS;
995/** @todo check for incomplete writes */
996 else
997 rc = rtSocketError();
998
999 RTMemTmpFree(paMsg);
1000 }
1001#endif /* !RT_OS_WINDOWS */
1002
1003 rtSocketUnlock(pThis);
1004 return rc;
1005}
1006
1007
1008RTDECL(int) RTSocketSgWriteL(RTSOCKET hSocket, size_t cSegs, ...)
1009{
1010 va_list va;
1011 va_start(va, cSegs);
1012 int rc = RTSocketSgWriteLV(hSocket, cSegs, va);
1013 va_end(va);
1014 return rc;
1015}
1016
1017
1018RTDECL(int) RTSocketSgWriteLV(RTSOCKET hSocket, size_t cSegs, va_list va)
1019{
1020 /*
1021 * Set up a S/G segment array + buffer on the stack and pass it
1022 * on to RTSocketSgWrite.
1023 */
1024 Assert(cSegs <= 16);
1025 PRTSGSEG paSegs = (PRTSGSEG)alloca(cSegs * sizeof(RTSGSEG));
1026 AssertReturn(paSegs, VERR_NO_TMP_MEMORY);
1027 for (size_t i = 0; i < cSegs; i++)
1028 {
1029 paSegs[i].pvSeg = va_arg(va, void *);
1030 paSegs[i].cbSeg = va_arg(va, size_t);
1031 }
1032
1033 RTSGBUF SgBuf;
1034 RTSgBufInit(&SgBuf, paSegs, cSegs);
1035 return RTSocketSgWrite(hSocket, &SgBuf);
1036}
1037
1038
1039RTDECL(int) RTSocketReadNB(RTSOCKET hSocket, void *pvBuffer, size_t cbBuffer, size_t *pcbRead)
1040{
1041 /*
1042 * Validate input.
1043 */
1044 RTSOCKETINT *pThis = hSocket;
1045 AssertPtrReturn(pThis, VERR_INVALID_HANDLE);
1046 AssertReturn(pThis->u32Magic == RTSOCKET_MAGIC, VERR_INVALID_HANDLE);
1047 AssertReturn(cbBuffer > 0, VERR_INVALID_PARAMETER);
1048 AssertPtr(pvBuffer);
1049 AssertPtrReturn(pcbRead, VERR_INVALID_PARAMETER);
1050 AssertReturn(rtSocketTryLock(pThis), VERR_CONCURRENT_ACCESS);
1051
1052 int rc = rtSocketSwitchBlockingMode(pThis, false /* fBlocking */);
1053 if (RT_FAILURE(rc))
1054 return rc;
1055
1056 rtSocketErrorReset();
1057#ifdef RT_OS_WINDOWS
1058 int cbNow = cbBuffer >= INT_MAX/2 ? INT_MAX/2 : (int)cbBuffer;
1059
1060 int cbRead = recv(pThis->hNative, (char *)pvBuffer, cbNow, MSG_NOSIGNAL);
1061 if (cbRead >= 0)
1062 {
1063 *pcbRead = cbRead;
1064 rc = VINF_SUCCESS;
1065 }
1066 else
1067 rc = rtSocketError();
1068
1069 if (rc == VERR_TRY_AGAIN)
1070 rc = VINF_TRY_AGAIN;
1071#else
1072 ssize_t cbRead = recv(pThis->hNative, pvBuffer, cbBuffer, MSG_NOSIGNAL);
1073 if (cbRead >= 0)
1074 *pcbRead = cbRead;
1075 else if (errno == EAGAIN)
1076 {
1077 *pcbRead = 0;
1078 rc = VINF_TRY_AGAIN;
1079 }
1080 else
1081 rc = rtSocketError();
1082#endif
1083
1084 rtSocketUnlock(pThis);
1085 return rc;
1086}
1087
1088
1089RTDECL(int) RTSocketWriteNB(RTSOCKET hSocket, const void *pvBuffer, size_t cbBuffer, size_t *pcbWritten)
1090{
1091 /*
1092 * Validate input.
1093 */
1094 RTSOCKETINT *pThis = hSocket;
1095 AssertPtrReturn(pThis, VERR_INVALID_HANDLE);
1096 AssertReturn(pThis->u32Magic == RTSOCKET_MAGIC, VERR_INVALID_HANDLE);
1097 AssertPtrReturn(pcbWritten, VERR_INVALID_PARAMETER);
1098 AssertReturn(rtSocketTryLock(pThis), VERR_CONCURRENT_ACCESS);
1099
1100 int rc = rtSocketSwitchBlockingMode(pThis, false /* fBlocking */);
1101 if (RT_FAILURE(rc))
1102 return rc;
1103
1104 rtSocketErrorReset();
1105#ifdef RT_OS_WINDOWS
1106 int cbNow = RT_MIN((int)cbBuffer, INT_MAX/2);
1107
1108 int cbWritten = send(pThis->hNative, (const char *)pvBuffer, cbNow, MSG_NOSIGNAL);
1109
1110 if (cbWritten >= 0)
1111 {
1112 *pcbWritten = cbWritten;
1113 rc = VINF_SUCCESS;
1114 }
1115 else
1116 rc = rtSocketError();
1117
1118 if (rc == VERR_TRY_AGAIN)
1119 rc = VINF_TRY_AGAIN;
1120#else
1121 ssize_t cbWritten = send(pThis->hNative, pvBuffer, cbBuffer, MSG_NOSIGNAL);
1122 if (cbWritten >= 0)
1123 *pcbWritten = cbWritten;
1124 else if (errno == EAGAIN)
1125 {
1126 *pcbWritten = 0;
1127 rc = VINF_TRY_AGAIN;
1128 }
1129 else
1130 rc = rtSocketError();
1131#endif
1132
1133 rtSocketUnlock(pThis);
1134 return rc;
1135}
1136
1137
1138RTDECL(int) RTSocketSgWriteNB(RTSOCKET hSocket, PCRTSGBUF pSgBuf, size_t *pcbWritten)
1139{
1140 /*
1141 * Validate input.
1142 */
1143 RTSOCKETINT *pThis = hSocket;
1144 AssertPtrReturn(pThis, VERR_INVALID_HANDLE);
1145 AssertReturn(pThis->u32Magic == RTSOCKET_MAGIC, VERR_INVALID_HANDLE);
1146 AssertPtrReturn(pSgBuf, VERR_INVALID_PARAMETER);
1147 AssertPtrReturn(pcbWritten, VERR_INVALID_PARAMETER);
1148 AssertReturn(pSgBuf->cSegs > 0, VERR_INVALID_PARAMETER);
1149 AssertReturn(rtSocketTryLock(pThis), VERR_CONCURRENT_ACCESS);
1150
1151 int rc = rtSocketSwitchBlockingMode(pThis, false /* fBlocking */);
1152 if (RT_FAILURE(rc))
1153 return rc;
1154
1155 unsigned cSegsToSend = 0;
1156 rc = VERR_NO_TMP_MEMORY;
1157#ifdef RT_OS_WINDOWS
1158 LPWSABUF paMsg = NULL;
1159
1160 RTSgBufMapToNative(paMsg, pSgBuf, WSABUF, buf, char *, len, u_long, cSegsToSend);
1161 if (paMsg)
1162 {
1163 DWORD dwSent = 0;
1164 int hrc = WSASend(pThis->hNative, paMsg, cSegsToSend, &dwSent,
1165 MSG_NOSIGNAL, NULL, NULL);
1166 if (!hrc)
1167 rc = VINF_SUCCESS;
1168 else
1169 rc = rtSocketError();
1170
1171 *pcbWritten = dwSent;
1172
1173 RTMemTmpFree(paMsg);
1174 }
1175
1176#else /* !RT_OS_WINDOWS */
1177 struct iovec *paMsg = NULL;
1178
1179 RTSgBufMapToNative(paMsg, pSgBuf, struct iovec, iov_base, void *, iov_len, size_t, cSegsToSend);
1180 if (paMsg)
1181 {
1182 struct msghdr msgHdr;
1183 RT_ZERO(msgHdr);
1184 msgHdr.msg_iov = paMsg;
1185 msgHdr.msg_iovlen = cSegsToSend;
1186 ssize_t cbWritten = sendmsg(pThis->hNative, &msgHdr, MSG_NOSIGNAL);
1187 if (RT_LIKELY(cbWritten >= 0))
1188 {
1189 rc = VINF_SUCCESS;
1190 *pcbWritten = cbWritten;
1191 }
1192 else
1193 rc = rtSocketError();
1194
1195 RTMemTmpFree(paMsg);
1196 }
1197#endif /* !RT_OS_WINDOWS */
1198
1199 rtSocketUnlock(pThis);
1200 return rc;
1201}
1202
1203
1204RTDECL(int) RTSocketSgWriteLNB(RTSOCKET hSocket, size_t cSegs, size_t *pcbWritten, ...)
1205{
1206 va_list va;
1207 va_start(va, pcbWritten);
1208 int rc = RTSocketSgWriteLVNB(hSocket, cSegs, pcbWritten, va);
1209 va_end(va);
1210 return rc;
1211}
1212
1213
1214RTDECL(int) RTSocketSgWriteLVNB(RTSOCKET hSocket, size_t cSegs, size_t *pcbWritten, va_list va)
1215{
1216 /*
1217 * Set up a S/G segment array + buffer on the stack and pass it
1218 * on to RTSocketSgWrite.
1219 */
1220 Assert(cSegs <= 16);
1221 PRTSGSEG paSegs = (PRTSGSEG)alloca(cSegs * sizeof(RTSGSEG));
1222 AssertReturn(paSegs, VERR_NO_TMP_MEMORY);
1223 for (size_t i = 0; i < cSegs; i++)
1224 {
1225 paSegs[i].pvSeg = va_arg(va, void *);
1226 paSegs[i].cbSeg = va_arg(va, size_t);
1227 }
1228
1229 RTSGBUF SgBuf;
1230 RTSgBufInit(&SgBuf, paSegs, cSegs);
1231 return RTSocketSgWriteNB(hSocket, &SgBuf, pcbWritten);
1232}
1233
1234
1235RTDECL(int) RTSocketSelectOne(RTSOCKET hSocket, RTMSINTERVAL cMillies)
1236{
1237 /*
1238 * Validate input.
1239 */
1240 RTSOCKETINT *pThis = hSocket;
1241 AssertPtrReturn(pThis, VERR_INVALID_HANDLE);
1242 AssertReturn(pThis->u32Magic == RTSOCKET_MAGIC, VERR_INVALID_HANDLE);
1243 AssertReturn(RTMemPoolRefCount(pThis) >= (pThis->cUsers ? 2U : 1U), VERR_CALLER_NO_REFERENCE);
1244 int const fdMax = (int)pThis->hNative + 1;
1245 AssertReturn(fdMax - 1 == pThis->hNative, VERR_INTERNAL_ERROR_5);
1246
1247 /*
1248 * Set up the file descriptor sets and do the select.
1249 */
1250 fd_set fdsetR;
1251 FD_ZERO(&fdsetR);
1252 FD_SET(pThis->hNative, &fdsetR);
1253
1254 fd_set fdsetE = fdsetR;
1255
1256 int rc;
1257 if (cMillies == RT_INDEFINITE_WAIT)
1258 rc = select(fdMax, &fdsetR, NULL, &fdsetE, NULL);
1259 else
1260 {
1261 struct timeval timeout;
1262 timeout.tv_sec = cMillies / 1000;
1263 timeout.tv_usec = (cMillies % 1000) * 1000;
1264 rc = select(fdMax, &fdsetR, NULL, &fdsetE, &timeout);
1265 }
1266 if (rc > 0)
1267 rc = VINF_SUCCESS;
1268 else if (rc == 0)
1269 rc = VERR_TIMEOUT;
1270 else
1271 rc = rtSocketError();
1272
1273 return rc;
1274}
1275
1276
1277RTDECL(int) RTSocketSelectOneEx(RTSOCKET hSocket, uint32_t fEvents, uint32_t *pfEvents,
1278 RTMSINTERVAL cMillies)
1279{
1280 /*
1281 * Validate input.
1282 */
1283 RTSOCKETINT *pThis = hSocket;
1284 AssertPtrReturn(pThis, VERR_INVALID_HANDLE);
1285 AssertReturn(pThis->u32Magic == RTSOCKET_MAGIC, VERR_INVALID_HANDLE);
1286 AssertPtrReturn(pfEvents, VERR_INVALID_PARAMETER);
1287 AssertReturn(!(fEvents & ~RTSOCKET_EVT_VALID_MASK), VERR_INVALID_PARAMETER);
1288 AssertReturn(RTMemPoolRefCount(pThis) >= (pThis->cUsers ? 2U : 1U), VERR_CALLER_NO_REFERENCE);
1289 int const fdMax = (int)pThis->hNative + 1;
1290 AssertReturn(fdMax - 1 == pThis->hNative, VERR_INTERNAL_ERROR_5);
1291
1292 *pfEvents = 0;
1293
1294 /*
1295 * Set up the file descriptor sets and do the select.
1296 */
1297 fd_set fdsetR;
1298 fd_set fdsetW;
1299 fd_set fdsetE;
1300 FD_ZERO(&fdsetR);
1301 FD_ZERO(&fdsetW);
1302 FD_ZERO(&fdsetE);
1303
1304 if (fEvents & RTSOCKET_EVT_READ)
1305 FD_SET(pThis->hNative, &fdsetR);
1306 if (fEvents & RTSOCKET_EVT_WRITE)
1307 FD_SET(pThis->hNative, &fdsetW);
1308 if (fEvents & RTSOCKET_EVT_ERROR)
1309 FD_SET(pThis->hNative, &fdsetE);
1310
1311 int rc;
1312 if (cMillies == RT_INDEFINITE_WAIT)
1313 rc = select(fdMax, &fdsetR, &fdsetW, &fdsetE, NULL);
1314 else
1315 {
1316 struct timeval timeout;
1317 timeout.tv_sec = cMillies / 1000;
1318 timeout.tv_usec = (cMillies % 1000) * 1000;
1319 rc = select(fdMax, &fdsetR, &fdsetW, &fdsetE, &timeout);
1320 }
1321 if (rc > 0)
1322 {
1323 if (FD_ISSET(pThis->hNative, &fdsetR))
1324 *pfEvents |= RTSOCKET_EVT_READ;
1325 if (FD_ISSET(pThis->hNative, &fdsetW))
1326 *pfEvents |= RTSOCKET_EVT_WRITE;
1327 if (FD_ISSET(pThis->hNative, &fdsetE))
1328 *pfEvents |= RTSOCKET_EVT_ERROR;
1329
1330 rc = VINF_SUCCESS;
1331 }
1332 else if (rc == 0)
1333 rc = VERR_TIMEOUT;
1334 else
1335 rc = rtSocketError();
1336
1337 return rc;
1338}
1339
1340
1341RTDECL(int) RTSocketShutdown(RTSOCKET hSocket, bool fRead, bool fWrite)
1342{
1343 /*
1344 * Validate input, don't lock it because we might want to interrupt a call
1345 * active on a different thread.
1346 */
1347 RTSOCKETINT *pThis = hSocket;
1348 AssertPtrReturn(pThis, VERR_INVALID_HANDLE);
1349 AssertReturn(pThis->u32Magic == RTSOCKET_MAGIC, VERR_INVALID_HANDLE);
1350 AssertReturn(RTMemPoolRefCount(pThis) >= (pThis->cUsers ? 2U : 1U), VERR_CALLER_NO_REFERENCE);
1351 AssertReturn(fRead || fWrite, VERR_INVALID_PARAMETER);
1352
1353 /*
1354 * Do the job.
1355 */
1356 int rc = VINF_SUCCESS;
1357 int fHow;
1358 if (fRead && fWrite)
1359 fHow = SHUT_RDWR;
1360 else if (fRead)
1361 fHow = SHUT_RD;
1362 else
1363 fHow = SHUT_WR;
1364 if (shutdown(pThis->hNative, fHow) == -1)
1365 rc = rtSocketError();
1366
1367 return rc;
1368}
1369
1370
1371RTDECL(int) RTSocketGetLocalAddress(RTSOCKET hSocket, PRTNETADDR pAddr)
1372{
1373 /*
1374 * Validate input.
1375 */
1376 RTSOCKETINT *pThis = hSocket;
1377 AssertPtrReturn(pThis, VERR_INVALID_HANDLE);
1378 AssertReturn(pThis->u32Magic == RTSOCKET_MAGIC, VERR_INVALID_HANDLE);
1379 AssertReturn(RTMemPoolRefCount(pThis) >= (pThis->cUsers ? 2U : 1U), VERR_CALLER_NO_REFERENCE);
1380
1381 /*
1382 * Get the address and convert it.
1383 */
1384 int rc;
1385 RTSOCKADDRUNION u;
1386#ifdef RT_OS_WINDOWS
1387 int cbAddr = sizeof(u);
1388#else
1389 socklen_t cbAddr = sizeof(u);
1390#endif
1391 RT_ZERO(u);
1392 if (getsockname(pThis->hNative, &u.Addr, &cbAddr) == 0)
1393 rc = rtSocketNetAddrFromAddr(&u, cbAddr, pAddr);
1394 else
1395 rc = rtSocketError();
1396
1397 return rc;
1398}
1399
1400
1401RTDECL(int) RTSocketGetPeerAddress(RTSOCKET hSocket, PRTNETADDR pAddr)
1402{
1403 /*
1404 * Validate input.
1405 */
1406 RTSOCKETINT *pThis = hSocket;
1407 AssertPtrReturn(pThis, VERR_INVALID_HANDLE);
1408 AssertReturn(pThis->u32Magic == RTSOCKET_MAGIC, VERR_INVALID_HANDLE);
1409 AssertReturn(RTMemPoolRefCount(pThis) >= (pThis->cUsers ? 2U : 1U), VERR_CALLER_NO_REFERENCE);
1410
1411 /*
1412 * Get the address and convert it.
1413 */
1414 int rc;
1415 RTSOCKADDRUNION u;
1416#ifdef RT_OS_WINDOWS
1417 int cbAddr = sizeof(u);
1418#else
1419 socklen_t cbAddr = sizeof(u);
1420#endif
1421 RT_ZERO(u);
1422 if (getpeername(pThis->hNative, &u.Addr, &cbAddr) == 0)
1423 rc = rtSocketNetAddrFromAddr(&u, cbAddr, pAddr);
1424 else
1425 rc = rtSocketError();
1426
1427 return rc;
1428}
1429
1430
1431
1432/**
1433 * Wrapper around bind.
1434 *
1435 * @returns IPRT status code.
1436 * @param hSocket The socket handle.
1437 * @param pAddr The address to bind to.
1438 */
1439int rtSocketBind(RTSOCKET hSocket, PCRTNETADDR pAddr)
1440{
1441 /*
1442 * Validate input.
1443 */
1444 RTSOCKETINT *pThis = hSocket;
1445 AssertPtrReturn(pThis, VERR_INVALID_HANDLE);
1446 AssertReturn(pThis->u32Magic == RTSOCKET_MAGIC, VERR_INVALID_HANDLE);
1447 AssertReturn(rtSocketTryLock(pThis), VERR_CONCURRENT_ACCESS);
1448
1449 RTSOCKADDRUNION u;
1450 int cbAddr;
1451 int rc = rtSocketAddrFromNetAddr(pAddr, &u, sizeof(u), &cbAddr);
1452 if (RT_SUCCESS(rc))
1453 {
1454 if (bind(pThis->hNative, &u.Addr, cbAddr) != 0)
1455 rc = rtSocketError();
1456 }
1457
1458 rtSocketUnlock(pThis);
1459 return rc;
1460}
1461
1462
1463/**
1464 * Wrapper around listen.
1465 *
1466 * @returns IPRT status code.
1467 * @param hSocket The socket handle.
1468 * @param cMaxPending The max number of pending connections.
1469 */
1470int rtSocketListen(RTSOCKET hSocket, int cMaxPending)
1471{
1472 /*
1473 * Validate input.
1474 */
1475 RTSOCKETINT *pThis = hSocket;
1476 AssertPtrReturn(pThis, VERR_INVALID_HANDLE);
1477 AssertReturn(pThis->u32Magic == RTSOCKET_MAGIC, VERR_INVALID_HANDLE);
1478 AssertReturn(rtSocketTryLock(pThis), VERR_CONCURRENT_ACCESS);
1479
1480 int rc = VINF_SUCCESS;
1481 if (listen(pThis->hNative, cMaxPending) != 0)
1482 rc = rtSocketError();
1483
1484 rtSocketUnlock(pThis);
1485 return rc;
1486}
1487
1488
1489/**
1490 * Wrapper around accept.
1491 *
1492 * @returns IPRT status code.
1493 * @param hSocket The socket handle.
1494 * @param phClient Where to return the client socket handle on
1495 * success.
1496 * @param pAddr Where to return the client address.
1497 * @param pcbAddr On input this gives the size buffer size of what
1498 * @a pAddr point to. On return this contains the
1499 * size of what's stored at @a pAddr.
1500 */
1501int rtSocketAccept(RTSOCKET hSocket, PRTSOCKET phClient, struct sockaddr *pAddr, size_t *pcbAddr)
1502{
1503 /*
1504 * Validate input.
1505 * Only lock the socket temporarily while we get the native handle, so that
1506 * we can safely shutdown and destroy the socket from a different thread.
1507 */
1508 RTSOCKETINT *pThis = hSocket;
1509 AssertPtrReturn(pThis, VERR_INVALID_HANDLE);
1510 AssertReturn(pThis->u32Magic == RTSOCKET_MAGIC, VERR_INVALID_HANDLE);
1511 AssertReturn(rtSocketTryLock(pThis), VERR_CONCURRENT_ACCESS);
1512
1513 /*
1514 * Call accept().
1515 */
1516 rtSocketErrorReset();
1517 int rc = VINF_SUCCESS;
1518#ifdef RT_OS_WINDOWS
1519 int cbAddr = (int)*pcbAddr;
1520#else
1521 socklen_t cbAddr = *pcbAddr;
1522#endif
1523 RTSOCKETNATIVE hNativeClient = accept(pThis->hNative, pAddr, &cbAddr);
1524 if (hNativeClient != NIL_RTSOCKETNATIVE)
1525 {
1526 *pcbAddr = cbAddr;
1527
1528 /*
1529 * Wrap the client socket.
1530 */
1531 rc = rtSocketCreateForNative(phClient, hNativeClient);
1532 if (RT_FAILURE(rc))
1533 {
1534#ifdef RT_OS_WINDOWS
1535 closesocket(hNativeClient);
1536#else
1537 close(hNativeClient);
1538#endif
1539 }
1540 }
1541 else
1542 rc = rtSocketError();
1543
1544 rtSocketUnlock(pThis);
1545 return rc;
1546}
1547
1548
1549/**
1550 * Wrapper around connect.
1551 *
1552 * @returns IPRT status code.
1553 * @param hSocket The socket handle.
1554 * @param pAddr The socket address to connect to.
1555 */
1556int rtSocketConnect(RTSOCKET hSocket, PCRTNETADDR pAddr)
1557{
1558 /*
1559 * Validate input.
1560 */
1561 RTSOCKETINT *pThis = hSocket;
1562 AssertPtrReturn(pThis, VERR_INVALID_HANDLE);
1563 AssertReturn(pThis->u32Magic == RTSOCKET_MAGIC, VERR_INVALID_HANDLE);
1564 AssertReturn(rtSocketTryLock(pThis), VERR_CONCURRENT_ACCESS);
1565
1566 RTSOCKADDRUNION u;
1567 int cbAddr;
1568 int rc = rtSocketAddrFromNetAddr(pAddr, &u, sizeof(u), &cbAddr);
1569 if (RT_SUCCESS(rc))
1570 {
1571 if (connect(pThis->hNative, &u.Addr, cbAddr) != 0)
1572 rc = rtSocketError();
1573 }
1574
1575 rtSocketUnlock(pThis);
1576 return rc;
1577}
1578
1579
1580/**
1581 * Wrapper around setsockopt.
1582 *
1583 * @returns IPRT status code.
1584 * @param hSocket The socket handle.
1585 * @param iLevel The protocol level, e.g. IPPORTO_TCP.
1586 * @param iOption The option, e.g. TCP_NODELAY.
1587 * @param pvValue The value buffer.
1588 * @param cbValue The size of the value pointed to by pvValue.
1589 */
1590int rtSocketSetOpt(RTSOCKET hSocket, int iLevel, int iOption, void const *pvValue, int cbValue)
1591{
1592 /*
1593 * Validate input.
1594 */
1595 RTSOCKETINT *pThis = hSocket;
1596 AssertPtrReturn(pThis, VERR_INVALID_HANDLE);
1597 AssertReturn(pThis->u32Magic == RTSOCKET_MAGIC, VERR_INVALID_HANDLE);
1598 AssertReturn(rtSocketTryLock(pThis), VERR_CONCURRENT_ACCESS);
1599
1600 int rc = VINF_SUCCESS;
1601 if (setsockopt(pThis->hNative, iLevel, iOption, (const char *)pvValue, cbValue) != 0)
1602 rc = rtSocketError();
1603
1604 rtSocketUnlock(pThis);
1605 return rc;
1606}
1607
1608#ifdef RT_OS_WINDOWS
1609
1610/**
1611 * Internal RTPollSetAdd helper that returns the handle that should be added to
1612 * the pollset.
1613 *
1614 * @returns Valid handle on success, INVALID_HANDLE_VALUE on failure.
1615 * @param hSocket The socket handle.
1616 * @param fEvents The events we're polling for.
1617 * @param ph where to put the primary handle.
1618 */
1619int rtSocketPollGetHandle(RTSOCKET hSocket, uint32_t fEvents, PHANDLE ph)
1620{
1621 RTSOCKETINT *pThis = hSocket;
1622 AssertPtrReturn(pThis, VERR_INVALID_HANDLE);
1623 AssertReturn(pThis->u32Magic == RTSOCKET_MAGIC, VERR_INVALID_HANDLE);
1624 AssertReturn(rtSocketTryLock(pThis), VERR_CONCURRENT_ACCESS);
1625
1626 int rc = VINF_SUCCESS;
1627 if (pThis->hEvent != WSA_INVALID_EVENT)
1628 *ph = pThis->hEvent;
1629 else
1630 {
1631 *ph = pThis->hEvent = WSACreateEvent();
1632 if (pThis->hEvent == WSA_INVALID_EVENT)
1633 rc = rtSocketError();
1634 }
1635
1636 rtSocketUnlock(pThis);
1637 return rc;
1638}
1639
1640
1641/**
1642 * Undos the harm done by WSAEventSelect.
1643 *
1644 * @returns IPRT status code.
1645 * @param pThis The socket handle.
1646 */
1647static int rtSocketPollClearEventAndRestoreBlocking(RTSOCKETINT *pThis)
1648{
1649 int rc = VINF_SUCCESS;
1650 if (pThis->fSubscribedEvts)
1651 {
1652 if (WSAEventSelect(pThis->hNative, WSA_INVALID_EVENT, 0) == 0)
1653 {
1654 pThis->fSubscribedEvts = 0;
1655
1656 /*
1657 * Switch back to blocking mode if that was the state before the
1658 * operation.
1659 */
1660 if (pThis->fBlocking)
1661 {
1662 u_long fNonBlocking = 0;
1663 int rc2 = ioctlsocket(pThis->hNative, FIONBIO, &fNonBlocking);
1664 if (rc2 != 0)
1665 {
1666 rc = rtSocketError();
1667 AssertMsgFailed(("%Rrc; rc2=%d\n", rc, rc2));
1668 }
1669 }
1670 }
1671 else
1672 {
1673 rc = rtSocketError();
1674 AssertMsgFailed(("%Rrc\n", rc));
1675 }
1676 }
1677 return rc;
1678}
1679
1680
1681/**
1682 * Updates the mask of events we're subscribing to.
1683 *
1684 * @returns IPRT status code.
1685 * @param pThis The socket handle.
1686 * @param fEvents The events we want to subscribe to.
1687 */
1688static int rtSocketPollUpdateEvents(RTSOCKETINT *pThis, uint32_t fEvents)
1689{
1690 LONG fNetworkEvents = 0;
1691 if (fEvents & RTPOLL_EVT_READ)
1692 fNetworkEvents |= FD_READ;
1693 if (fEvents & RTPOLL_EVT_WRITE)
1694 fNetworkEvents |= FD_WRITE;
1695 if (fEvents & RTPOLL_EVT_ERROR)
1696 fNetworkEvents |= FD_CLOSE;
1697 LogFlowFunc(("fNetworkEvents=%#x\n", fNetworkEvents));
1698 if (WSAEventSelect(pThis->hNative, pThis->hEvent, fNetworkEvents) == 0)
1699 {
1700 pThis->fSubscribedEvts = fEvents;
1701 return VINF_SUCCESS;
1702 }
1703
1704 int rc = rtSocketError();
1705 AssertMsgFailed(("fNetworkEvents=%#x rc=%Rrc\n", fNetworkEvents, rtSocketError()));
1706 return rc;
1707}
1708
1709
1710/**
1711 * Checks for pending events.
1712 *
1713 * @returns Event mask or 0.
1714 * @param pThis The socket handle.
1715 * @param fEvents The desired events.
1716 */
1717static uint32_t rtSocketPollCheck(RTSOCKETINT *pThis, uint32_t fEvents)
1718{
1719 int rc = VINF_SUCCESS;
1720 uint32_t fRetEvents = 0;
1721
1722 LogFlowFunc(("pThis=%#p fEvents=%#x\n", pThis, fEvents));
1723
1724 /* Make sure WSAEnumNetworkEvents returns what we want. */
1725 if ((pThis->fSubscribedEvts & fEvents) != fEvents)
1726 rc = rtSocketPollUpdateEvents(pThis, pThis->fSubscribedEvts | fEvents);
1727
1728 /* Get the event mask, ASSUMES that WSAEnumNetworkEvents doesn't clear stuff. */
1729 WSANETWORKEVENTS NetEvts;
1730 RT_ZERO(NetEvts);
1731 if (WSAEnumNetworkEvents(pThis->hNative, pThis->hEvent, &NetEvts) == 0)
1732 {
1733 if ( (NetEvts.lNetworkEvents & FD_READ)
1734 && (fEvents & RTPOLL_EVT_READ)
1735 && NetEvts.iErrorCode[FD_READ_BIT] == 0)
1736 fRetEvents |= RTPOLL_EVT_READ;
1737
1738 if ( (NetEvts.lNetworkEvents & FD_WRITE)
1739 && (fEvents & RTPOLL_EVT_WRITE)
1740 && NetEvts.iErrorCode[FD_WRITE_BIT] == 0)
1741 fRetEvents |= RTPOLL_EVT_WRITE;
1742
1743 if (fEvents & RTPOLL_EVT_ERROR)
1744 {
1745 if (NetEvts.lNetworkEvents & FD_CLOSE)
1746 fRetEvents |= RTPOLL_EVT_ERROR;
1747 else
1748 for (uint32_t i = 0; i < FD_MAX_EVENTS; i++)
1749 if ( (NetEvts.lNetworkEvents & (1L << i))
1750 && NetEvts.iErrorCode[i] != 0)
1751 fRetEvents |= RTPOLL_EVT_ERROR;
1752 }
1753 }
1754 else
1755 rc = rtSocketError();
1756
1757 /* Fall back on select if we hit an error above. */
1758 if (RT_FAILURE(rc))
1759 {
1760 /** @todo */
1761 }
1762
1763 LogFlowFunc(("fRetEvents=%#x\n", fRetEvents));
1764 return fRetEvents;
1765}
1766
1767
1768/**
1769 * Internal RTPoll helper that polls the socket handle and, if @a fNoWait is
1770 * clear, starts whatever actions we've got running during the poll call.
1771 *
1772 * @returns 0 if no pending events, actions initiated if @a fNoWait is clear.
1773 * Event mask (in @a fEvents) and no actions if the handle is ready
1774 * already.
1775 * UINT32_MAX (asserted) if the socket handle is busy in I/O or a
1776 * different poll set.
1777 *
1778 * @param hSocket The socket handle.
1779 * @param hPollSet The poll set handle (for access checks).
1780 * @param fEvents The events we're polling for.
1781 * @param fFinalEntry Set if this is the final entry for this handle
1782 * in this poll set. This can be used for dealing
1783 * with duplicate entries.
1784 * @param fNoWait Set if it's a zero-wait poll call. Clear if
1785 * we'll wait for an event to occur.
1786 *
1787 * @remarks There is a potential race wrt duplicate handles when @a fNoWait is
1788 * @c true, we don't currently care about that oddity...
1789 */
1790uint32_t rtSocketPollStart(RTSOCKET hSocket, RTPOLLSET hPollSet, uint32_t fEvents, bool fFinalEntry, bool fNoWait)
1791{
1792 RTSOCKETINT *pThis = hSocket;
1793 AssertPtrReturn(pThis, UINT32_MAX);
1794 AssertReturn(pThis->u32Magic == RTSOCKET_MAGIC, UINT32_MAX);
1795 if (rtSocketTryLock(pThis))
1796 pThis->hPollSet = hPollSet;
1797 else
1798 {
1799 AssertReturn(pThis->hPollSet == hPollSet, UINT32_MAX);
1800 ASMAtomicIncU32(&pThis->cUsers);
1801 }
1802
1803 /* (rtSocketPollCheck will reset the event object). */
1804 uint32_t fRetEvents = pThis->fEventsSaved;
1805 pThis->fEventsSaved = 0; /* Reset */
1806 fRetEvents |= rtSocketPollCheck(pThis, fEvents);
1807
1808 if ( !fRetEvents
1809 && !fNoWait)
1810 {
1811 pThis->fPollEvts |= fEvents;
1812 if ( fFinalEntry
1813 && pThis->fSubscribedEvts != pThis->fPollEvts)
1814 {
1815 int rc = rtSocketPollUpdateEvents(pThis, pThis->fPollEvts);
1816 if (RT_FAILURE(rc))
1817 {
1818 pThis->fPollEvts = 0;
1819 fRetEvents = UINT32_MAX;
1820 }
1821 }
1822 }
1823
1824 if (fRetEvents || fNoWait)
1825 {
1826 if (pThis->cUsers == 1)
1827 {
1828 rtSocketPollClearEventAndRestoreBlocking(pThis);
1829 pThis->hPollSet = NIL_RTPOLLSET;
1830 }
1831 ASMAtomicDecU32(&pThis->cUsers);
1832 }
1833
1834 return fRetEvents;
1835}
1836
1837
1838/**
1839 * Called after a WaitForMultipleObjects returned in order to check for pending
1840 * events and stop whatever actions that rtSocketPollStart() initiated.
1841 *
1842 * @returns Event mask or 0.
1843 *
1844 * @param hSocket The socket handle.
1845 * @param fEvents The events we're polling for.
1846 * @param fFinalEntry Set if this is the final entry for this handle
1847 * in this poll set. This can be used for dealing
1848 * with duplicate entries. Only keep in mind that
1849 * this method is called in reverse order, so the
1850 * first call will have this set (when the entire
1851 * set was processed).
1852 * @param fHarvestEvents Set if we should check for pending events.
1853 */
1854uint32_t rtSocketPollDone(RTSOCKET hSocket, uint32_t fEvents, bool fFinalEntry, bool fHarvestEvents)
1855{
1856 RTSOCKETINT *pThis = hSocket;
1857 AssertPtrReturn(pThis, 0);
1858 AssertReturn(pThis->u32Magic == RTSOCKET_MAGIC, 0);
1859 Assert(pThis->cUsers > 0);
1860 Assert(pThis->hPollSet != NIL_RTPOLLSET);
1861
1862 /* Harvest events and clear the event mask for the next round of polling. */
1863 uint32_t fRetEvents = rtSocketPollCheck(pThis, fEvents);
1864 pThis->fPollEvts = 0;
1865
1866 /*
1867 * Save the write event if required.
1868 * It is only posted once and might get lost if the another source in the
1869 * pollset with a higher priority has pending events.
1870 */
1871 if ( !fHarvestEvents
1872 && fRetEvents)
1873 {
1874 pThis->fEventsSaved = fRetEvents;
1875 fRetEvents = 0;
1876 }
1877
1878 /* Make the socket blocking again and unlock the handle. */
1879 if (pThis->cUsers == 1)
1880 {
1881 rtSocketPollClearEventAndRestoreBlocking(pThis);
1882 pThis->hPollSet = NIL_RTPOLLSET;
1883 }
1884 ASMAtomicDecU32(&pThis->cUsers);
1885 return fRetEvents;
1886}
1887
1888#endif /* RT_OS_WINDOWS */
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