proxy.cpp 4.81 KB
Newer Older
Pieter Hintjens's avatar
Pieter Hintjens committed
1
/*
2
    Copyright (c) 2007-2013 Contributors as noted in the AUTHORS file
Pieter Hintjens's avatar
Pieter Hintjens committed
3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20

    This file is part of 0MQ.

    0MQ is free software; you can redistribute it and/or modify it under
    the terms of the GNU Lesser General Public License as published by
    the Free Software Foundation; either version 3 of the License, or
    (at your option) any later version.

    0MQ 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 <stddef.h>
Pieter Hintjens's avatar
Pieter Hintjens committed
21
#include "platform.hpp"
22
#include "proxy.hpp"
Pieter Hintjens's avatar
Pieter Hintjens committed
23 24
#include "likely.hpp"

Pieter Hintjens's avatar
Pieter Hintjens committed
25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47
#if defined ZMQ_FORCE_SELECT
#define ZMQ_POLL_BASED_ON_SELECT
#elif defined ZMQ_FORCE_POLL
#define ZMQ_POLL_BASED_ON_POLL
#elif defined ZMQ_HAVE_LINUX || defined ZMQ_HAVE_FREEBSD ||\
    defined ZMQ_HAVE_OPENBSD || defined ZMQ_HAVE_SOLARIS ||\
    defined ZMQ_HAVE_OSX || defined ZMQ_HAVE_QNXNTO ||\
    defined ZMQ_HAVE_HPUX || defined ZMQ_HAVE_AIX ||\
    defined ZMQ_HAVE_NETBSD
#define ZMQ_POLL_BASED_ON_POLL
#elif defined ZMQ_HAVE_WINDOWS || defined ZMQ_HAVE_OPENVMS ||\
     defined ZMQ_HAVE_CYGWIN
#define ZMQ_POLL_BASED_ON_SELECT
#endif

//  On AIX platform, poll.h has to be included first to get consistent
//  definition of pollfd structure (AIX uses 'reqevents' and 'retnevents'
//  instead of 'events' and 'revents' and defines macros to map from POSIX-y
//  names to AIX-specific names).
#if defined ZMQ_POLL_BASED_ON_POLL
#include <poll.h>
#endif

48 49 50 51 52
// These headers end up pulling in zmq.h somewhere in their include
// dependency chain
#include "socket_base.hpp"
#include "err.hpp"

Pieter Hintjens's avatar
Pieter Hintjens committed
53 54 55 56
// zmq.h must be included *after* poll.h for AIX to build properly
#include "../include/zmq.h"


57 58 59 60
int zmq::proxy (
    class socket_base_t *frontend_,
    class socket_base_t *backend_,
    class socket_base_t *capture_)
Pieter Hintjens's avatar
Pieter Hintjens committed
61 62 63
{
    msg_t msg;
    int rc = msg.init ();
Pieter Hintjens's avatar
Pieter Hintjens committed
64
    if (rc != 0)
Pieter Hintjens's avatar
Pieter Hintjens committed
65 66
        return -1;

67
    //  The algorithm below assumes ratio of requests and replies processed
Pieter Hintjens's avatar
Pieter Hintjens committed
68
    //  under full load to be 1:1.
69

Pieter Hintjens's avatar
Pieter Hintjens committed
70
    int more;
Pieter Hintjens's avatar
Pieter Hintjens committed
71
    size_t moresz;
Pieter Hintjens's avatar
Pieter Hintjens committed
72
    zmq_pollitem_t items [] = {
73 74
        { frontend_, 0, ZMQ_POLLIN, 0 },
        { backend_, 0, ZMQ_POLLIN, 0 }
Pieter Hintjens's avatar
Pieter Hintjens committed
75
    };
Pieter Hintjens's avatar
Pieter Hintjens committed
76 77 78
    while (true) {
        //  Wait while there are either requests or replies to process.
        rc = zmq_poll (&items [0], 2, -1);
Pieter Hintjens's avatar
Pieter Hintjens committed
79
        if (unlikely (rc < 0))
Pieter Hintjens's avatar
Pieter Hintjens committed
80 81
            return -1;

82
        //  Process a request
Pieter Hintjens's avatar
Pieter Hintjens committed
83 84
        if (items [0].revents & ZMQ_POLLIN) {
            while (true) {
85
                rc = frontend_->recv (&msg, 0);
Pieter Hintjens's avatar
Pieter Hintjens committed
86
                if (unlikely (rc < 0))
Pieter Hintjens's avatar
Pieter Hintjens committed
87
                    return -1;
88 89

                moresz = sizeof more;
90
                rc = frontend_->getsockopt (ZMQ_RCVMORE, &more, &moresz);
Pieter Hintjens's avatar
Pieter Hintjens committed
91
                if (unlikely (rc < 0))
Pieter Hintjens's avatar
Pieter Hintjens committed
92
                    return -1;
93

94 95 96 97 98 99 100 101 102 103 104 105 106 107
                //  Copy message to capture socket if any
                if (capture_) {
                    msg_t ctrl;
                    rc = ctrl.init ();
                    if (unlikely (rc < 0))
                        return -1;
                    rc = ctrl.copy (msg);
                    if (unlikely (rc < 0))
                        return -1;
                    rc = capture_->send (&ctrl, more? ZMQ_SNDMORE: 0);
                    if (unlikely (rc < 0))
                        return -1;
                }
                rc = backend_->send (&msg, more? ZMQ_SNDMORE: 0);
Pieter Hintjens's avatar
Pieter Hintjens committed
108
                if (unlikely (rc < 0))
Pieter Hintjens's avatar
Pieter Hintjens committed
109
                    return -1;
Pieter Hintjens's avatar
Pieter Hintjens committed
110
                if (more == 0)
Pieter Hintjens's avatar
Pieter Hintjens committed
111 112 113
                    break;
            }
        }
114
        //  Process a reply
Pieter Hintjens's avatar
Pieter Hintjens committed
115 116
        if (items [1].revents & ZMQ_POLLIN) {
            while (true) {
117
                rc = backend_->recv (&msg, 0);
Pieter Hintjens's avatar
Pieter Hintjens committed
118
                if (unlikely (rc < 0))
Pieter Hintjens's avatar
Pieter Hintjens committed
119
                    return -1;
120 121

                moresz = sizeof more;
122
                rc = backend_->getsockopt (ZMQ_RCVMORE, &more, &moresz);
Pieter Hintjens's avatar
Pieter Hintjens committed
123
                if (unlikely (rc < 0))
Pieter Hintjens's avatar
Pieter Hintjens committed
124
                    return -1;
125

126 127 128 129 130 131 132 133 134 135 136 137 138 139
                //  Copy message to capture socket if any
                if (capture_) {
                    msg_t ctrl;
                    rc = ctrl.init ();
                    if (unlikely (rc < 0))
                        return -1;
                    rc = ctrl.copy (msg);
                    if (unlikely (rc < 0))
                        return -1;
                    rc = capture_->send (&ctrl, more? ZMQ_SNDMORE: 0);
                    if (unlikely (rc < 0))
                        return -1;
                }
                rc = frontend_->send (&msg, more? ZMQ_SNDMORE: 0);
Pieter Hintjens's avatar
Pieter Hintjens committed
140
                if (unlikely (rc < 0))
Pieter Hintjens's avatar
Pieter Hintjens committed
141
                    return -1;
Pieter Hintjens's avatar
Pieter Hintjens committed
142
                if (more == 0)
Pieter Hintjens's avatar
Pieter Hintjens committed
143 144 145 146 147 148 149
                    break;
            }
        }

    }
    return 0;
}