LCOV - code coverage report
Current view: top level - home/h/core/forks/m4-libzmq/src - ipc_listener.cpp (source / functions) Hit Total Coverage
Test: zeromq-4.2.0 Code Coverage Lines: 112 131 85.5 %
Date: 2016-05-09 Functions: 12 12 100.0 %
Legend: Lines: hit not hit

          Line data    Source code
       1             : /*
       2             :     Copyright (c) 2007-2016 Contributors as noted in the AUTHORS file
       3             : 
       4             :     This file is part of libzmq, the ZeroMQ core engine in C++.
       5             : 
       6             :     libzmq is free software; you can redistribute it and/or modify it under
       7             :     the terms of the GNU Lesser General Public License (LGPL) as published
       8             :     by the Free Software Foundation; either version 3 of the License, or
       9             :     (at your option) any later version.
      10             : 
      11             :     As a special exception, the Contributors give you permission to link
      12             :     this library with independent modules to produce an executable,
      13             :     regardless of the license terms of these independent modules, and to
      14             :     copy and distribute the resulting executable under terms of your choice,
      15             :     provided that you also meet, for each linked independent module, the
      16             :     terms and conditions of the license of that module. An independent
      17             :     module is a module which is not derived from or based on this library.
      18             :     If you modify this library, you must extend this exception to your
      19             :     version of the library.
      20             : 
      21             :     libzmq is distributed in the hope that it will be useful, but WITHOUT
      22             :     ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
      23             :     FITNESS FOR A PARTICULAR PURPOSE. See the GNU Lesser General Public
      24             :     License for more details.
      25             : 
      26             :     You should have received a copy of the GNU Lesser General Public License
      27             :     along with this program.  If not, see <http://www.gnu.org/licenses/>.
      28             : */
      29             : 
      30             : #include "precompiled.hpp"
      31             : #include "ipc_listener.hpp"
      32             : 
      33             : #if !defined ZMQ_HAVE_WINDOWS && !defined ZMQ_HAVE_OPENVMS
      34             : 
      35             : #include <new>
      36             : 
      37             : #include <string.h>
      38             : 
      39             : #include "stream_engine.hpp"
      40             : #include "ipc_address.hpp"
      41             : #include "io_thread.hpp"
      42             : #include "session_base.hpp"
      43             : #include "config.hpp"
      44             : #include "err.hpp"
      45             : #include "ip.hpp"
      46             : #include "socket_base.hpp"
      47             : 
      48             : #include <unistd.h>
      49             : #include <sys/socket.h>
      50             : #include <fcntl.h>
      51             : #include <sys/un.h>
      52             : #include <sys/stat.h>
      53             : 
      54             : #ifdef ZMQ_HAVE_LOCAL_PEERCRED
      55             : #   include <sys/types.h>
      56             : #   include <sys/ucred.h>
      57             : #endif
      58             : #ifdef ZMQ_HAVE_SO_PEERCRED
      59             : #   include <sys/types.h>
      60             : #   include <pwd.h>
      61             : #   include <grp.h>
      62             : #   if defined ZMQ_HAVE_OPENBSD
      63             : #       define ucred sockpeercred
      64             : #   endif
      65             : #endif
      66             : 
      67             : const char *zmq::ipc_listener_t::tmp_env_vars[] = {
      68             :   "TMPDIR",
      69             :   "TEMPDIR",
      70             :   "TMP",
      71             :   0  // Sentinel
      72             : };
      73             : 
      74           9 : int zmq::ipc_listener_t::create_wildcard_address(std::string& path_,
      75             :         std::string& file_)
      76             : {
      77             :     std::string tmp_path;
      78             : 
      79             :     // If TMPDIR, TEMPDIR, or TMP are available and are directories, create
      80             :     // the socket directory there.
      81           9 :     const char **tmp_env = tmp_env_vars;
      82          45 :     while ( tmp_path.empty() && *tmp_env != 0 ) {
      83          27 :         char *tmpdir = getenv(*tmp_env);
      84             :         struct stat statbuf;
      85             : 
      86             :         // Confirm it is actually a directory before trying to use
      87          27 :         if ( tmpdir != 0 && ::stat(tmpdir, &statbuf) == 0 && S_ISDIR(statbuf.st_mode) ) {
      88           0 :             tmp_path.assign(tmpdir);
      89           0 :             if ( *(tmp_path.rbegin()) != '/' ) {
      90           0 :                 tmp_path.push_back('/');
      91             :             }
      92             :         }
      93             : 
      94             :         // Try the next environment variable
      95          27 :         ++tmp_env;
      96             :     }
      97             : 
      98             :     // Append a directory name
      99           9 :     tmp_path.append("tmpXXXXXX");
     100             : 
     101             :     // We need room for tmp_path + trailing NUL
     102           9 :     std::vector<char> buffer(tmp_path.length()+1);
     103          18 :     strcpy(buffer.data(), tmp_path.c_str());
     104             : 
     105             : #ifdef HAVE_MKDTEMP
     106             :     // Create the directory.  POSIX requires that mkdtemp() creates the
     107             :     // directory with 0700 permissions, meaning the only possible race
     108             :     // with socket creation could be the same user.  However, since
     109             :     // each socket is created in a directory created by mkdtemp(), and
     110             :     // mkdtemp() guarantees a unique directory name, there will be no
     111             :     // collision.
     112           9 :     if ( mkdtemp(buffer.data()) == 0 ) {
     113             :         return -1;
     114             :     }
     115             : 
     116           9 :     path_.assign(buffer.data());
     117          18 :     file_.assign (path_ + "/socket");
     118             : #else
     119             :     // Silence -Wunused-parameter. #pragma and __attribute__((unused)) are not
     120             :     // very portable unfortunately...
     121             :     (void) path_;
     122             :     int fd = mkstemp (buffer.data());
     123             :     if (fd == -1)
     124             :          return -1;
     125             :     ::close (fd);
     126             : 
     127             :     file_.assign (buffer.data());
     128             : #endif
     129             : 
     130           9 :     return 0;
     131             : }
     132             : 
     133          63 : zmq::ipc_listener_t::ipc_listener_t (io_thread_t *io_thread_,
     134             :       socket_base_t *socket_, const options_t &options_) :
     135             :     own_t (io_thread_, options_),
     136             :     io_object_t (io_thread_),
     137             :     has_file (false),
     138             :     s (retired_fd),
     139         189 :     socket (socket_)
     140             : {
     141          63 : }
     142             : 
     143         378 : zmq::ipc_listener_t::~ipc_listener_t ()
     144             : {
     145          63 :     zmq_assert (s == retired_fd);
     146         126 : }
     147             : 
     148          63 : void zmq::ipc_listener_t::process_plug ()
     149             : {
     150             :     //  Start polling for incoming connections.
     151          63 :     handle = add_fd (s);
     152          63 :     set_pollin (handle);
     153          63 : }
     154             : 
     155          63 : void zmq::ipc_listener_t::process_term (int linger_)
     156             : {
     157          63 :     rm_fd (handle);
     158          63 :     close ();
     159          63 :     own_t::process_term (linger_);
     160          63 : }
     161             : 
     162          63 : void zmq::ipc_listener_t::in_event ()
     163             : {
     164          63 :     fd_t fd = accept ();
     165             : 
     166             :     //  If connection was reset by the peer in the meantime, just ignore it.
     167             :     //  TODO: Handle specific errors like ENFILE/EMFILE etc.
     168          63 :     if (fd == retired_fd) {
     169           9 :         socket->event_accept_failed (endpoint, zmq_errno());
     170          72 :         return;
     171             :     }
     172             : 
     173             :     //  Create the engine object for this connection.
     174             :     stream_engine_t *engine = new (std::nothrow)
     175          54 :         stream_engine_t (fd, options, endpoint);
     176          54 :     alloc_assert (engine);
     177             : 
     178             :     //  Choose I/O thread to run connecter in. Given that we are already
     179             :     //  running in an I/O thread, there must be at least one available.
     180          54 :     io_thread_t *io_thread = choose_io_thread (options.affinity);
     181          54 :     zmq_assert (io_thread);
     182             : 
     183             :     //  Create and launch a session object.
     184             :     session_base_t *session = session_base_t::create (io_thread, false, socket,
     185          54 :         options, NULL);
     186          54 :     errno_assert (session);
     187          54 :     session->inc_seqnum ();
     188          54 :     launch_child (session);
     189          54 :     send_attach (session, engine, false);
     190          54 :     socket->event_accepted (endpoint, fd);
     191             : }
     192             : 
     193          63 : int zmq::ipc_listener_t::get_address (std::string &addr_)
     194             : {
     195             :     struct sockaddr_storage ss;
     196             : #ifdef ZMQ_HAVE_HPUX
     197             :     int sl = sizeof (ss);
     198             : #else
     199          63 :     socklen_t sl = sizeof (ss);
     200             : #endif
     201          63 :     int rc = getsockname (s, (sockaddr *) &ss, &sl);
     202          63 :     if (rc != 0) {
     203             :         addr_.clear ();
     204           0 :         return rc;
     205             :     }
     206             : 
     207          63 :     ipc_address_t addr ((struct sockaddr *) &ss, sl);
     208          63 :     return addr.to_string (addr_);
     209             : }
     210             : 
     211          63 : int zmq::ipc_listener_t::set_address (const char *addr_)
     212             : {
     213             :     //  Create addr on stack for auto-cleanup
     214          63 :     std::string addr (addr_);
     215             : 
     216             :     //  Allow wildcard file
     217         117 :     if (options.use_fd == -1 && addr [0] == '*') {
     218           9 :         if ( create_wildcard_address(tmp_socket_dirname, addr) < 0 ) {
     219             :             return -1;
     220             :         }
     221             :     }
     222             : 
     223             :     //  Get rid of the file associated with the UNIX domain socket that
     224             :     //  may have been left behind by the previous run of the application.
     225             :     //  MUST NOT unlink if the FD is managed by the user, or it will stop
     226             :     //  working after the first client connects. The user will take care of
     227             :     //  cleaning up the file after the service is stopped.
     228          63 :     if (options.use_fd == -1) {
     229          54 :         ::unlink (addr.c_str());
     230             :     }
     231          63 :     filename.clear ();
     232             : 
     233             :     //  Initialise the address structure.
     234         126 :     ipc_address_t address;
     235          63 :     int rc = address.resolve (addr.c_str());
     236          63 :     if (rc != 0) {
     237           0 :         if ( !tmp_socket_dirname.empty() ) {
     238             :             // We need to preserve errno to return to the user
     239           0 :             int errno_ = errno;
     240           0 :             ::rmdir(tmp_socket_dirname.c_str ());
     241           0 :             tmp_socket_dirname.clear();
     242           0 :             errno = errno_;
     243             :         }
     244             :         return -1;
     245             :     }
     246             : 
     247          63 :     address.to_string (endpoint);
     248             : 
     249          63 :     if (options.use_fd != -1) {
     250           9 :         s = options.use_fd;
     251             :     } else {
     252             :         //  Create a listening socket.
     253          54 :         s = open_socket (AF_UNIX, SOCK_STREAM, 0);
     254          54 :         if (s == -1) {
     255           0 :             if ( !tmp_socket_dirname.empty() ) {
     256             :                 // We need to preserve errno to return to the user
     257           0 :                 int errno_ = errno;
     258           0 :                 ::rmdir(tmp_socket_dirname.c_str ());
     259           0 :                 tmp_socket_dirname.clear();
     260           0 :                 errno = errno_;
     261             :             }
     262             :             return -1;
     263             :         }
     264             : 
     265             :         //  Bind the socket to the file path.
     266          54 :         rc = bind (s, address.addr (), address.addrlen ());
     267          54 :         if (rc != 0)
     268             :             goto error;
     269             : 
     270             :         //  Listen for incoming connections.
     271          54 :         rc = listen (s, options.backlog);
     272          54 :         if (rc != 0)
     273             :             goto error;
     274             :     }
     275             : 
     276          63 :     filename.assign (addr.c_str());
     277          63 :     has_file = true;
     278             : 
     279          63 :     socket->event_listening (endpoint, s);
     280             :     return 0;
     281             : 
     282             : error:
     283           0 :     int err = errno;
     284           0 :     close ();
     285           0 :     errno = err;
     286           0 :     return -1;
     287             : }
     288             : 
     289          63 : int zmq::ipc_listener_t::close ()
     290             : {
     291          63 :     zmq_assert (s != retired_fd);
     292          63 :     int rc = ::close (s);
     293          63 :     errno_assert (rc == 0);
     294             : 
     295          63 :     s = retired_fd;
     296             : 
     297             :     //  If there's an underlying UNIX domain socket, get rid of the file it
     298             :     //  is associated with.
     299             :     //  MUST NOT unlink if the FD is managed by the user, or it will stop
     300             :     //  working after the first client connects. The user will take care of
     301             :     //  cleaning up the file after the service is stopped.
     302          63 :     if (has_file && options.use_fd == -1) {
     303          54 :         rc = 0;
     304             : 
     305         108 :         if ( !filename.empty () ) {
     306          54 :           rc = ::unlink(filename.c_str ());
     307             :         }
     308             : 
     309         105 :         if ( rc == 0 && !tmp_socket_dirname.empty() ) {
     310          18 :           rc = ::rmdir(tmp_socket_dirname.c_str ());
     311           9 :           tmp_socket_dirname.clear();
     312             :         }
     313             : 
     314          54 :         if (rc != 0) {
     315           3 :             socket->event_close_failed (endpoint, zmq_errno());
     316           3 :             return -1;
     317             :         }
     318             :     }
     319             : 
     320          60 :     socket->event_closed (endpoint, s);
     321          60 :     return 0;
     322             : }
     323             : 
     324             : #if defined ZMQ_HAVE_SO_PEERCRED
     325             : 
     326          63 : bool zmq::ipc_listener_t::filter (fd_t sock)
     327             : {
     328         183 :     if (options.ipc_uid_accept_filters.empty () &&
     329         114 :         options.ipc_pid_accept_filters.empty () &&
     330          51 :         options.ipc_gid_accept_filters.empty ())
     331             :         return true;
     332             : 
     333             :     struct ucred cred;
     334          21 :     socklen_t size = sizeof (cred);
     335             : 
     336          21 :     if (getsockopt (sock, SOL_SOCKET, SO_PEERCRED, &cred, &size))
     337             :         return false;
     338          81 :     if (options.ipc_uid_accept_filters.find (cred.uid) != options.ipc_uid_accept_filters.end () ||
     339          54 :             options.ipc_gid_accept_filters.find (cred.gid) != options.ipc_gid_accept_filters.end () ||
     340          30 :             options.ipc_pid_accept_filters.find (cred.pid) != options.ipc_pid_accept_filters.end ())
     341             :         return true;
     342             : 
     343             :     struct passwd *pw;
     344             :     struct group *gr;
     345             : 
     346          12 :     if (!(pw = getpwuid (cred.uid)))
     347             :         return false;
     348          42 :     for (options_t::ipc_gid_accept_filters_t::const_iterator it = options.ipc_gid_accept_filters.begin ();
     349          30 :             it != options.ipc_gid_accept_filters.end (); it++) {
     350           6 :         if (!(gr = getgrgid (*it)))
     351             :             continue;
     352           6 :         for (char **mem = gr->gr_mem; *mem; mem++) {
     353           6 :             if (!strcmp (*mem, pw->pw_name))
     354             :                 return true;
     355             :         }
     356             :     }
     357             :     return false;
     358             : }
     359             : 
     360             : #elif defined ZMQ_HAVE_LOCAL_PEERCRED
     361             : 
     362             : bool zmq::ipc_listener_t::filter (fd_t sock)
     363             : {
     364             :     if (options.ipc_uid_accept_filters.empty () &&
     365             :         options.ipc_gid_accept_filters.empty ())
     366             :         return true;
     367             : 
     368             :     struct xucred cred;
     369             :     socklen_t size = sizeof (cred);
     370             : 
     371             :     if (getsockopt (sock, 0, LOCAL_PEERCRED, &cred, &size))
     372             :         return false;
     373             :     if (cred.cr_version != XUCRED_VERSION)
     374             :         return false;
     375             :     if (options.ipc_uid_accept_filters.find (cred.cr_uid) != options.ipc_uid_accept_filters.end ())
     376             :         return true;
     377             :     for (int i = 0; i < cred.cr_ngroups; i++) {
     378             :         if (options.ipc_gid_accept_filters.find (cred.cr_groups[i]) != options.ipc_gid_accept_filters.end ())
     379             :             return true;
     380             :     }
     381             : 
     382             :     return false;
     383             : }
     384             : 
     385             : #endif
     386             : 
     387          63 : zmq::fd_t zmq::ipc_listener_t::accept ()
     388             : {
     389             :     //  Accept one connection and deal with different failure modes.
     390             :     //  The situation where connection cannot be accepted due to insufficient
     391             :     //  resources is considered valid and treated by ignoring the connection.
     392          63 :     zmq_assert (s != retired_fd);
     393          63 :     fd_t sock = ::accept (s, NULL, NULL);
     394          63 :     if (sock == -1) {
     395           0 :         errno_assert (errno == EAGAIN || errno == EWOULDBLOCK ||
     396             :             errno == EINTR || errno == ECONNABORTED || errno == EPROTO ||
     397             :             errno == ENFILE);
     398             :         return retired_fd;
     399             :     }
     400             : 
     401             :     //  Race condition can cause socket not to be closed (if fork happens
     402             :     //  between accept and this point).
     403             : #ifdef FD_CLOEXEC
     404          63 :     int rc = fcntl (sock, F_SETFD, FD_CLOEXEC);
     405          63 :     errno_assert (rc != -1);
     406             : #endif
     407             : 
     408             :     // IPC accept() filters
     409             : #if defined ZMQ_HAVE_SO_PEERCRED || defined ZMQ_HAVE_LOCAL_PEERCRED
     410          63 :     if (!filter (sock)) {
     411           9 :         int rc = ::close (sock);
     412           9 :         errno_assert (rc == 0);
     413             :         return retired_fd;
     414             :     }
     415             : #endif
     416             : 
     417             :     return sock;
     418             : }
     419             : 
     420             : #endif

Generated by: LCOV version 1.10