123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414 |
- /*
- Copyright (c) 2007-2016 Contributors as noted in the AUTHORS file
- This file is part of libzmq, the ZeroMQ core engine in C++.
- libzmq is free software; you can redistribute it and/or modify it under
- the terms of the GNU Lesser General Public License (LGPL) as published
- by the Free Software Foundation; either version 3 of the License, or
- (at your option) any later version.
- As a special exception, the Contributors give you permission to link
- this library with independent modules to produce an executable,
- regardless of the license terms of these independent modules, and to
- copy and distribute the resulting executable under terms of your choice,
- provided that you also meet, for each linked independent module, the
- terms and conditions of the license of that module. An independent
- module is a module which is not derived from or based on this library.
- If you modify this library, you must extend this exception to your
- version of the library.
- libzmq 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 Lesser General Public
- License for more details.
- You should have received a copy of the GNU Lesser General Public License
- along with this program. If not, see <http://www.gnu.org/licenses/>.
- */
- #include "precompiled.hpp"
- #include "poller.hpp"
- #include "polling_util.hpp"
- #if defined ZMQ_POLL_BASED_ON_POLL
- #if !defined ZMQ_HAVE_WINDOWS && !defined ZMQ_HAVE_AIX
- #include <poll.h>
- #endif
- #elif defined ZMQ_POLL_BASED_ON_SELECT
- #if defined ZMQ_HAVE_WINDOWS
- #elif defined ZMQ_HAVE_HPUX
- #include <sys/param.h>
- #include <sys/types.h>
- #include <sys/time.h>
- #elif defined ZMQ_HAVE_OPENVMS
- #include <sys/types.h>
- #include <sys/time.h>
- #elif defined ZMQ_HAVE_VXWORKS
- #include <sys/types.h>
- #include <sys/time.h>
- #include <sockLib.h>
- #include <strings.h>
- #else
- #include <sys/select.h>
- #endif
- #endif
- #include "signaler.hpp"
- #include "likely.hpp"
- #include "stdint.hpp"
- #include "config.hpp"
- #include "err.hpp"
- #include "fd.hpp"
- #include "ip.hpp"
- #include "tcp.hpp"
- #if !defined ZMQ_HAVE_WINDOWS
- #include <unistd.h>
- #include <netinet/tcp.h>
- #include <sys/types.h>
- #include <sys/socket.h>
- #endif
- #if !defined(ZMQ_HAVE_WINDOWS)
- // Helper to sleep for specific number of milliseconds (or until signal)
- //
- static int sleep_ms (unsigned int ms_)
- {
- if (ms_ == 0)
- return 0;
- #if defined ZMQ_HAVE_ANDROID
- usleep (ms_ * 1000);
- return 0;
- #elif defined ZMQ_HAVE_VXWORKS
- struct timespec ns_;
- ns_.tv_sec = ms_ / 1000;
- ns_.tv_nsec = ms_ % 1000 * 1000000;
- return nanosleep (&ns_, 0);
- #else
- return usleep (ms_ * 1000);
- #endif
- }
- // Helper to wait on close(), for non-blocking sockets, until it completes
- // If EAGAIN is received, will sleep briefly (1-100ms) then try again, until
- // the overall timeout is reached.
- //
- static int close_wait_ms (int fd_, unsigned int max_ms_ = 2000)
- {
- unsigned int ms_so_far = 0;
- const unsigned int min_step_ms = 1;
- const unsigned int max_step_ms = 100;
- const unsigned int step_ms =
- std::min (std::max (min_step_ms, max_ms_ / 10), max_step_ms);
- int rc = 0; // do not sleep on first attempt
- do {
- if (rc == -1 && errno == EAGAIN) {
- sleep_ms (step_ms);
- ms_so_far += step_ms;
- }
- rc = close (fd_);
- } while (ms_so_far < max_ms_ && rc == -1 && errno == EAGAIN);
- return rc;
- }
- #endif
- zmq::signaler_t::signaler_t ()
- {
- // Create the socketpair for signaling.
- if (make_fdpair (&_r, &_w) == 0) {
- unblock_socket (_w);
- unblock_socket (_r);
- }
- #ifdef HAVE_FORK
- pid = getpid ();
- #endif
- }
- // This might get run after some part of construction failed, leaving one or
- // both of _r and _w retired_fd.
- zmq::signaler_t::~signaler_t ()
- {
- #if defined ZMQ_HAVE_EVENTFD
- if (_r == retired_fd)
- return;
- int rc = close_wait_ms (_r);
- errno_assert (rc == 0);
- #elif defined ZMQ_HAVE_WINDOWS
- if (_w != retired_fd) {
- const struct linger so_linger = {1, 0};
- int rc = setsockopt (_w, SOL_SOCKET, SO_LINGER,
- reinterpret_cast<const char *> (&so_linger),
- sizeof so_linger);
- // Only check shutdown if WSASTARTUP was previously done
- if (rc == 0 || WSAGetLastError () != WSANOTINITIALISED) {
- wsa_assert (rc != SOCKET_ERROR);
- rc = closesocket (_w);
- wsa_assert (rc != SOCKET_ERROR);
- if (_r == retired_fd)
- return;
- rc = closesocket (_r);
- wsa_assert (rc != SOCKET_ERROR);
- }
- }
- #else
- if (_w != retired_fd) {
- int rc = close_wait_ms (_w);
- errno_assert (rc == 0);
- }
- if (_r != retired_fd) {
- int rc = close_wait_ms (_r);
- errno_assert (rc == 0);
- }
- #endif
- }
- zmq::fd_t zmq::signaler_t::get_fd () const
- {
- return _r;
- }
- void zmq::signaler_t::send ()
- {
- #if defined HAVE_FORK
- if (unlikely (pid != getpid ())) {
- //printf("Child process %d signaler_t::send returning without sending #1\n", getpid());
- return; // do not send anything in forked child context
- }
- #endif
- #if defined ZMQ_HAVE_EVENTFD
- const uint64_t inc = 1;
- ssize_t sz = write (_w, &inc, sizeof (inc));
- errno_assert (sz == sizeof (inc));
- #elif defined ZMQ_HAVE_WINDOWS
- const char dummy = 0;
- int nbytes;
- do {
- nbytes = ::send (_w, &dummy, sizeof (dummy), 0);
- wsa_assert (nbytes != SOCKET_ERROR);
- // wsa_assert does not abort on WSAEWOULDBLOCK. If we get this, we retry.
- } while (nbytes == SOCKET_ERROR);
- // Given the small size of dummy (should be 1) expect that send was able to send everything.
- zmq_assert (nbytes == sizeof (dummy));
- #elif defined ZMQ_HAVE_VXWORKS
- unsigned char dummy = 0;
- while (true) {
- ssize_t nbytes = ::send (_w, (char *) &dummy, sizeof (dummy), 0);
- if (unlikely (nbytes == -1 && errno == EINTR))
- continue;
- #if defined(HAVE_FORK)
- if (unlikely (pid != getpid ())) {
- //printf("Child process %d signaler_t::send returning without sending #2\n", getpid());
- errno = EINTR;
- break;
- }
- #endif
- zmq_assert (nbytes == sizeof dummy);
- break;
- }
- #else
- unsigned char dummy = 0;
- while (true) {
- ssize_t nbytes = ::send (_w, &dummy, sizeof (dummy), 0);
- if (unlikely (nbytes == -1 && errno == EINTR))
- continue;
- #if defined(HAVE_FORK)
- if (unlikely (pid != getpid ())) {
- //printf("Child process %d signaler_t::send returning without sending #2\n", getpid());
- errno = EINTR;
- break;
- }
- #endif
- zmq_assert (nbytes == sizeof dummy);
- break;
- }
- #endif
- }
- int zmq::signaler_t::wait (int timeout_) const
- {
- #ifdef HAVE_FORK
- if (unlikely (pid != getpid ())) {
- // we have forked and the file descriptor is closed. Emulate an interrupt
- // response.
- //printf("Child process %d signaler_t::wait returning simulating interrupt #1\n", getpid());
- errno = EINTR;
- return -1;
- }
- #endif
- #ifdef ZMQ_POLL_BASED_ON_POLL
- struct pollfd pfd;
- pfd.fd = _r;
- pfd.events = POLLIN;
- const int rc = poll (&pfd, 1, timeout_);
- if (unlikely (rc < 0)) {
- errno_assert (errno == EINTR);
- return -1;
- }
- if (unlikely (rc == 0)) {
- errno = EAGAIN;
- return -1;
- }
- #ifdef HAVE_FORK
- if (unlikely (pid != getpid ())) {
- // we have forked and the file descriptor is closed. Emulate an interrupt
- // response.
- //printf("Child process %d signaler_t::wait returning simulating interrupt #2\n", getpid());
- errno = EINTR;
- return -1;
- }
- #endif
- zmq_assert (rc == 1);
- zmq_assert (pfd.revents & POLLIN);
- return 0;
- #elif defined ZMQ_POLL_BASED_ON_SELECT
- optimized_fd_set_t fds (1);
- FD_ZERO (fds.get ());
- FD_SET (_r, fds.get ());
- struct timeval timeout;
- if (timeout_ >= 0) {
- timeout.tv_sec = timeout_ / 1000;
- timeout.tv_usec = timeout_ % 1000 * 1000;
- }
- #ifdef ZMQ_HAVE_WINDOWS
- int rc =
- select (0, fds.get (), NULL, NULL, timeout_ >= 0 ? &timeout : NULL);
- wsa_assert (rc != SOCKET_ERROR);
- #else
- int rc =
- select (_r + 1, fds.get (), NULL, NULL, timeout_ >= 0 ? &timeout : NULL);
- if (unlikely (rc < 0)) {
- errno_assert (errno == EINTR);
- return -1;
- }
- #endif
- if (unlikely (rc == 0)) {
- errno = EAGAIN;
- return -1;
- }
- zmq_assert (rc == 1);
- return 0;
- #else
- #error
- #endif
- }
- void zmq::signaler_t::recv ()
- {
- // Attempt to read a signal.
- #if defined ZMQ_HAVE_EVENTFD
- uint64_t dummy;
- ssize_t sz = read (_r, &dummy, sizeof (dummy));
- errno_assert (sz == sizeof (dummy));
- // If we accidentally grabbed the next signal(s) along with the current
- // one, return it back to the eventfd object.
- if (unlikely (dummy > 1)) {
- const uint64_t inc = dummy - 1;
- ssize_t sz2 = write (_w, &inc, sizeof (inc));
- errno_assert (sz2 == sizeof (inc));
- return;
- }
- zmq_assert (dummy == 1);
- #else
- unsigned char dummy;
- #if defined ZMQ_HAVE_WINDOWS
- const int nbytes =
- ::recv (_r, reinterpret_cast<char *> (&dummy), sizeof (dummy), 0);
- wsa_assert (nbytes != SOCKET_ERROR);
- #elif defined ZMQ_HAVE_VXWORKS
- ssize_t nbytes = ::recv (_r, (char *) &dummy, sizeof (dummy), 0);
- errno_assert (nbytes >= 0);
- #else
- ssize_t nbytes = ::recv (_r, &dummy, sizeof (dummy), 0);
- errno_assert (nbytes >= 0);
- #endif
- zmq_assert (nbytes == sizeof (dummy));
- zmq_assert (dummy == 0);
- #endif
- }
- int zmq::signaler_t::recv_failable ()
- {
- // Attempt to read a signal.
- #if defined ZMQ_HAVE_EVENTFD
- uint64_t dummy;
- ssize_t sz = read (_r, &dummy, sizeof (dummy));
- if (sz == -1) {
- errno_assert (errno == EAGAIN);
- return -1;
- }
- errno_assert (sz == sizeof (dummy));
- // If we accidentally grabbed the next signal(s) along with the current
- // one, return it back to the eventfd object.
- if (unlikely (dummy > 1)) {
- const uint64_t inc = dummy - 1;
- ssize_t sz2 = write (_w, &inc, sizeof (inc));
- errno_assert (sz2 == sizeof (inc));
- return 0;
- }
- zmq_assert (dummy == 1);
- #else
- unsigned char dummy;
- #if defined ZMQ_HAVE_WINDOWS
- const int nbytes =
- ::recv (_r, reinterpret_cast<char *> (&dummy), sizeof (dummy), 0);
- if (nbytes == SOCKET_ERROR) {
- const int last_error = WSAGetLastError ();
- if (last_error == WSAEWOULDBLOCK) {
- errno = EAGAIN;
- return -1;
- }
- wsa_assert (last_error == WSAEWOULDBLOCK);
- }
- #elif defined ZMQ_HAVE_VXWORKS
- ssize_t nbytes = ::recv (_r, (char *) &dummy, sizeof (dummy), 0);
- if (nbytes == -1) {
- if (errno == EAGAIN || errno == EWOULDBLOCK || errno == EINTR) {
- errno = EAGAIN;
- return -1;
- }
- errno_assert (errno == EAGAIN || errno == EWOULDBLOCK
- || errno == EINTR);
- }
- #else
- ssize_t nbytes = ::recv (_r, &dummy, sizeof (dummy), 0);
- if (nbytes == -1) {
- if (errno == EAGAIN || errno == EWOULDBLOCK || errno == EINTR) {
- errno = EAGAIN;
- return -1;
- }
- errno_assert (errno == EAGAIN || errno == EWOULDBLOCK
- || errno == EINTR);
- }
- #endif
- zmq_assert (nbytes == sizeof (dummy));
- zmq_assert (dummy == 0);
- #endif
- return 0;
- }
- bool zmq::signaler_t::valid () const
- {
- return _w != retired_fd;
- }
- #ifdef HAVE_FORK
- void zmq::signaler_t::forked ()
- {
- // Close file descriptors created in the parent and create new pair
- close (_r);
- close (_w);
- make_fdpair (&_r, &_w);
- }
- #endif
|