aboutsummaryrefslogtreecommitdiffstats
path: root/src/system/EventPipe.cxx
diff options
context:
space:
mode:
authorMax Kellermann <max@duempel.org>2013-08-07 11:52:26 +0200
committerMax Kellermann <max@duempel.org>2013-08-07 11:52:26 +0200
commitf2ce8c3b62cedd0ae7cba1c3b9e2d1cec9a750a8 (patch)
treeb6eff1b3ed6d07e2b7358bdb4a12590d375ff531 /src/system/EventPipe.cxx
parent4223657ab8f9120c4175437645034392c451b97c (diff)
downloadmpd-f2ce8c3b62cedd0ae7cba1c3b9e2d1cec9a750a8.tar.gz
mpd-f2ce8c3b62cedd0ae7cba1c3b9e2d1cec9a750a8.tar.xz
mpd-f2ce8c3b62cedd0ae7cba1c3b9e2d1cec9a750a8.zip
event/EventFD: move to libsystem
Diffstat (limited to 'src/system/EventPipe.cxx')
-rw-r--r--src/system/EventPipe.cxx180
1 files changed, 180 insertions, 0 deletions
diff --git a/src/system/EventPipe.cxx b/src/system/EventPipe.cxx
new file mode 100644
index 000000000..e1a38c503
--- /dev/null
+++ b/src/system/EventPipe.cxx
@@ -0,0 +1,180 @@
+/*
+ * Copyright (C) 2003-2013 The Music Player Daemon Project
+ * http://www.musicpd.org
+ *
+ * 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.
+ */
+
+#include "config.h"
+#include "EventPipe.hxx"
+#include "system/fd_util.h"
+#include "system/FatalError.hxx"
+#include "gcc.h"
+
+#include <assert.h>
+#include <unistd.h>
+
+#ifdef WIN32
+#include <ws2tcpip.h>
+#include <winsock2.h>
+#include <cstring> /* for memset() */
+#endif
+
+#ifdef WIN32
+static bool PoorSocketPair(int fd[2]);
+#endif
+
+EventPipe::EventPipe()
+{
+#ifdef WIN32
+ bool success = PoorSocketPair(fds);
+#else
+ bool success = pipe_cloexec_nonblock(fds) >= 0;
+#endif
+ if (!success)
+ FatalSystemError("pipe() has failed");
+}
+
+EventPipe::~EventPipe()
+{
+#ifdef WIN32
+ closesocket(fds[0]);
+ closesocket(fds[1]);
+#else
+ close(fds[0]);
+ close(fds[1]);
+#endif
+}
+
+bool
+EventPipe::Read()
+{
+ assert(fds[0] >= 0);
+ assert(fds[1] >= 0);
+
+ char buffer[256];
+#ifdef WIN32
+ return recv(fds[0], buffer, sizeof(buffer), 0) > 0;
+#else
+ return read(fds[0], buffer, sizeof(buffer)) > 0;
+#endif
+}
+
+void
+EventPipe::Write()
+{
+ assert(fds[0] >= 0);
+ assert(fds[1] >= 0);
+
+#ifdef WIN32
+ send(fds[1], "", 1, 0);
+#else
+ gcc_unused ssize_t nbytes = write(fds[1], "", 1);
+#endif
+}
+
+#ifdef WIN32
+
+static void SafeCloseSocket(SOCKET s)
+{
+ int error = WSAGetLastError();
+ closesocket(s);
+ WSASetLastError(error);
+}
+
+/* Our poor man's socketpair() implementation
+ * Due to limited protocol/address family support and primitive error handling
+ * it's better to keep this as a private implementation detail of EventPipe
+ * rather than wide-available API.
+ */
+static bool PoorSocketPair(int fd[2])
+{
+ assert (fd != nullptr);
+
+ SOCKET listen_socket = socket(AF_INET, SOCK_STREAM, IPPROTO_TCP);
+ if (listen_socket == INVALID_SOCKET)
+ return false;
+
+ sockaddr_in address;
+ std::memset(&address, 0, sizeof(address));
+ address.sin_family = AF_INET;
+ address.sin_addr.s_addr = htonl(INADDR_LOOPBACK);
+
+ int ret = bind(listen_socket,
+ reinterpret_cast<sockaddr*>(&address),
+ sizeof(address));
+
+ if (ret < 0) {
+ SafeCloseSocket(listen_socket);
+ return false;
+ }
+
+ ret = listen(listen_socket, 1);
+
+ if (ret < 0) {
+ SafeCloseSocket(listen_socket);
+ return false;
+ }
+
+ int address_len = sizeof(address);
+ ret = getsockname(listen_socket,
+ reinterpret_cast<sockaddr*>(&address),
+ &address_len);
+
+ if (ret < 0) {
+ SafeCloseSocket(listen_socket);
+ return false;
+ }
+
+ SOCKET socket0 = socket(AF_INET, SOCK_STREAM, IPPROTO_TCP);
+ if (socket0 == INVALID_SOCKET) {
+ SafeCloseSocket(listen_socket);
+ return false;
+ }
+
+ ret = connect(socket0,
+ reinterpret_cast<sockaddr*>(&address),
+ sizeof(address));
+
+ if (ret < 0) {
+ SafeCloseSocket(listen_socket);
+ SafeCloseSocket(socket0);
+ return false;
+ }
+
+ SOCKET socket1 = accept(listen_socket, nullptr, nullptr);
+ if (socket1 == INVALID_SOCKET) {
+ SafeCloseSocket(listen_socket);
+ SafeCloseSocket(socket0);
+ return false;
+ }
+
+ SafeCloseSocket(listen_socket);
+
+ u_long non_block = 1;
+ if (ioctlsocket(socket0, FIONBIO, &non_block) < 0
+ || ioctlsocket(socket1, FIONBIO, &non_block) < 0) {
+ SafeCloseSocket(socket0);
+ SafeCloseSocket(socket1);
+ return false;
+ }
+
+ fd[0] = static_cast<int>(socket0);
+ fd[1] = static_cast<int>(socket1);
+
+ return true;
+}
+
+#endif