[vlc-commits] Move Winsock poll() emulation to src/win32/

Rémi Denis-Courmont git at videolan.org
Sat Aug 20 13:11:24 CEST 2011


vlc | branch: master | Rémi Denis-Courmont <remi at remlab.net> | Sat Aug 20 13:58:04 2011 +0300| [45a540ed0cdd265c4f637d126fbd64fcc88cbc8f] | committer: Rémi Denis-Courmont

Move Winsock poll() emulation to src/win32/

> http://git.videolan.org/gitweb.cgi/vlc.git/?a=commit;h=45a540ed0cdd265c4f637d126fbd64fcc88cbc8f
---

 src/Makefile.am    |    5 ++-
 src/network/poll.c |  126 --------------------------------------------
 src/win32/poll.c   |  147 ++++++++++++++++++++++++++++++++++++++++++++++++++++
 3 files changed, 151 insertions(+), 127 deletions(-)

diff --git a/src/Makefile.am b/src/Makefile.am
index 87bfb70..cd5f5a5 100644
--- a/src/Makefile.am
+++ b/src/Makefile.am
@@ -280,6 +280,7 @@ SOURCES_libvlc_darwin = \
 	misc/atomic.c \
 	posix/filesystem.c \
 	posix/plugin.c \
+	network/poll.c \
 	posix/thread.c \
 	posix/darwin_specific.c \
 	$(NULL)
@@ -289,6 +290,7 @@ SOURCES_libvlc_linux = \
 	misc/atomic.c \
 	posix/filesystem.c \
 	posix/plugin.c \
+	network/poll.c \
 	posix/thread.c \
 	posix/linux_specific.c \
 	$(NULL)
@@ -298,6 +300,7 @@ SOURCES_libvlc_win32 = \
 	win32/atomic.c \
 	win32/filesystem.c \
 	win32/plugin.c \
+	win32/poll.c \
 	win32/thread.c \
 	win32/specific.c \
 	win32/winsock.c \
@@ -314,6 +317,7 @@ SOURCES_libvlc_other = \
 	posix/dirs.c \
 	misc/atomic.c \
 	posix/filesystem.c \
+	network/poll.c \
 	posix/thread.c \
 	posix/specific.c \
 	$(NULL)
@@ -421,7 +425,6 @@ SOURCES_libvlc_common = \
 	network/udp.c \
 	network/rootbind.c \
 	network/tls.c \
-	network/poll.c \
 	text/charset.c \
 	text/strings.c \
 	text/unicode.c \
diff --git a/src/network/poll.c b/src/network/poll.c
index c1ca900..24335c2 100644
--- a/src/network/poll.c
+++ b/src/network/poll.c
@@ -71,132 +71,6 @@ int vlc_poll (struct pollfd *fds, unsigned nfds, int timeout)
     abort ();
 }
 
-#elif defined (WIN32) || defined(__OS2__)
-
-#include <stdlib.h>
-#include <string.h>
-#include <errno.h>
-#ifndef __OS2__
-#ifdef FD_SETSIZE
-/* No, it's not as simple as #undef FD_SETSIZE */
-# error Header inclusion order compromised!
-#endif
-#define FD_SETSIZE 0
-#endif
-#include <vlc_network.h>
-
-#ifdef __OS2__
-#include <sys/time.h>
-#include <sys/select.h>
-
-#define SOCKET unsigned
-#endif
-
-int vlc_poll (struct pollfd *fds, unsigned nfds, int timeout)
-{
-    size_t setsize = sizeof (fd_set) + nfds * sizeof (SOCKET);
-    fd_set *rdset = malloc (setsize);
-    fd_set *wrset = malloc (setsize);
-    fd_set *exset = malloc (setsize);
-    struct timeval tv = { 0, 0 };
-    int val;
-
-    if (unlikely(rdset == NULL || wrset == NULL || exset == NULL))
-    {
-        free (rdset);
-        free (wrset);
-        free (exset);
-        errno = ENOMEM;
-        return -1;
-    }
-
-/* Winsock FD_SET uses FD_SETSIZE in its expansion */
-#undef FD_SETSIZE
-#define FD_SETSIZE (nfds)
-
-resume:
-    val = -1;
-    vlc_testcancel ();
-
-    FD_ZERO (rdset);
-    FD_ZERO (wrset);
-    FD_ZERO (exset);
-    for (unsigned i = 0; i < nfds; i++)
-    {
-        int fd = fds[i].fd;
-        if (val < fd)
-            val = fd;
-
-        /* With POSIX, FD_SET & FD_ISSET are not defined if fd is negative or
-         * bigger or equal than FD_SETSIZE. That is one of the reasons why VLC
-         * uses poll() rather than select(). Most POSIX systems implement
-         * fd_set has a bit field with no sanity checks. This is especially bad
-         * on systems (such as BSD) that have no process open files limit by
-         * default, such that it is quite feasible to get fd >= FD_SETSIZE.
-         * The next instructions will result in a buffer overflow if run on
-         * a POSIX system, and the later FD_ISSET would perform an undefined
-         * memory read.
-         *
-         * With Winsock, fd_set is a table of integers. This is awfully slow.
-         * However, FD_SET and FD_ISSET silently and safely discard excess
-         * entries. Here, overflow cannot happen anyway: fd_set of adequate
-         * size are allocated.
-         * Note that Vista has a much nicer WSAPoll(), but Mingw does not
-         * support it yet.
-         */
-        if (fds[i].events & POLLIN)
-            FD_SET ((SOCKET)fd, rdset);
-        if (fds[i].events & POLLOUT)
-            FD_SET ((SOCKET)fd, wrset);
-        if (fds[i].events & POLLPRI)
-            FD_SET ((SOCKET)fd, exset);
-    }
-
-#ifndef HAVE_ALERTABLE_SELECT
-# warning FIXME! Fix cancellation and remove this crap.
-    if ((timeout < 0) || (timeout > 50))
-    {
-        tv.tv_sec = 0;
-        tv.tv_usec = 50000;
-    }
-    else
-#endif
-    if (timeout >= 0)
-    {
-        div_t d = div (timeout, 1000);
-        tv.tv_sec = d.quot;
-        tv.tv_usec = d.rem * 1000;
-    }
-
-    val = select (val + 1, rdset, wrset, exset,
-                  /*(timeout >= 0) ?*/ &tv /*: NULL*/);
-
-#ifndef HAVE_ALERTABLE_SELECT
-    if (val == 0)
-    {
-        if (timeout > 0)
-            timeout -= (timeout > 50) ? 50 : timeout;
-        if (timeout != 0)
-            goto resume;
-    }
-#endif
-
-    if (val == -1)
-        return -1;
-
-    for (unsigned i = 0; i < nfds; i++)
-    {
-        int fd = fds[i].fd;
-        fds[i].revents = (FD_ISSET (fd, rdset) ? POLLIN : 0)
-                       | (FD_ISSET (fd, wrset) ? POLLOUT : 0)
-                       | (FD_ISSET (fd, exset) ? POLLPRI : 0);
-    }
-    free (exset);
-    free (wrset);
-    free (rdset);
-    return val;
-}
-
 #else
 # error poll() not implemented!
 #endif
diff --git a/src/win32/poll.c b/src/win32/poll.c
new file mode 100644
index 0000000..4fa693f
--- /dev/null
+++ b/src/win32/poll.c
@@ -0,0 +1,147 @@
+/*****************************************************************************
+ * poll.c: poll() emulation for Winsock
+ *****************************************************************************
+ * Copyright © 2007 Rémi Denis-Courmont
+ *
+ * This program is free software; you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License as published by
+ * the Free Software Foundation; either version 2 of the License, or
+ * (at your option) any later version.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+ * GNU General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program; if not, write to the Free Software
+ * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston MA 02110-1301, USA.
+ *****************************************************************************/
+
+#ifdef HAVE_CONFIG_H
+# include "config.h"
+#endif
+
+#include <stdlib.h>
+#include <string.h>
+#include <errno.h>
+
+#include <vlc_common.h>
+
+#ifdef FD_SETSIZE
+/* Too late for #undef FD_SETSIZE to work: fd_set is already defined. */
+# error Header inclusion order compromised!
+#endif
+#define FD_SETSIZE 0
+#include <vlc_network.h>
+
+#ifdef __OS2__
+#include <sys/time.h>
+#include <sys/select.h>
+#define SOCKET unsigned
+#endif
+
+int vlc_poll (struct pollfd *fds, unsigned nfds, int timeout)
+{
+    size_t setsize = sizeof (fd_set) + nfds * sizeof (SOCKET);
+    fd_set *rdset = malloc (setsize);
+    fd_set *wrset = malloc (setsize);
+    fd_set *exset = malloc (setsize);
+    struct timeval tv = { 0, 0 };
+    int val;
+
+    if (unlikely(rdset == NULL || wrset == NULL || exset == NULL))
+    {
+        free (rdset);
+        free (wrset);
+        free (exset);
+        errno = ENOMEM;
+        return -1;
+    }
+
+/* Winsock FD_SET uses FD_SETSIZE in its expansion */
+#undef FD_SETSIZE
+#define FD_SETSIZE (nfds)
+
+resume:
+    val = -1;
+    vlc_testcancel ();
+
+    FD_ZERO (rdset);
+    FD_ZERO (wrset);
+    FD_ZERO (exset);
+    for (unsigned i = 0; i < nfds; i++)
+    {
+        int fd = fds[i].fd;
+        if (val < fd)
+            val = fd;
+
+        /* With POSIX, FD_SET & FD_ISSET are not defined if fd is negative or
+         * bigger or equal than FD_SETSIZE. That is one of the reasons why VLC
+         * uses poll() rather than select(). Most POSIX systems implement
+         * fd_set has a bit field with no sanity checks. This is especially bad
+         * on systems (such as BSD) that have no process open files limit by
+         * default, such that it is quite feasible to get fd >= FD_SETSIZE.
+         * The next instructions will result in a buffer overflow if run on
+         * a POSIX system, and the later FD_ISSET would perform an undefined
+         * memory read.
+         *
+         * With Winsock, fd_set is a table of integers. This is awfully slow.
+         * However, FD_SET and FD_ISSET silently and safely discard excess
+         * entries. Here, overflow cannot happen anyway: fd_set of adequate
+         * size are allocated.
+         * Note that Vista has a much nicer WSAPoll(), but Mingw does not
+         * support it yet.
+         */
+        if (fds[i].events & POLLIN)
+            FD_SET ((SOCKET)fd, rdset);
+        if (fds[i].events & POLLOUT)
+            FD_SET ((SOCKET)fd, wrset);
+        if (fds[i].events & POLLPRI)
+            FD_SET ((SOCKET)fd, exset);
+    }
+
+#ifndef HAVE_ALERTABLE_SELECT
+# warning FIXME! Fix cancellation and remove this crap.
+    if ((timeout < 0) || (timeout > 50))
+    {
+        tv.tv_sec = 0;
+        tv.tv_usec = 50000;
+    }
+    else
+#endif
+    if (timeout >= 0)
+    {
+        div_t d = div (timeout, 1000);
+        tv.tv_sec = d.quot;
+        tv.tv_usec = d.rem * 1000;
+    }
+
+    val = select (val + 1, rdset, wrset, exset,
+                  /*(timeout >= 0) ?*/ &tv /*: NULL*/);
+
+#ifndef HAVE_ALERTABLE_SELECT
+    if (val == 0)
+    {
+        if (timeout > 0)
+            timeout -= (timeout > 50) ? 50 : timeout;
+        if (timeout != 0)
+            goto resume;
+    }
+#endif
+
+    if (val == -1)
+        return -1;
+
+    for (unsigned i = 0; i < nfds; i++)
+    {
+        int fd = fds[i].fd;
+        fds[i].revents = (FD_ISSET (fd, rdset) ? POLLIN : 0)
+                       | (FD_ISSET (fd, wrset) ? POLLOUT : 0)
+                       | (FD_ISSET (fd, exset) ? POLLPRI : 0);
+    }
+    free (exset);
+    free (wrset);
+    free (rdset);
+    return val;
+}



More information about the vlc-commits mailing list