proxy.cpp 6.4 KB
Newer Older
Pieter Hintjens's avatar
Pieter Hintjens committed
1
/*
2
    Copyright (c) 2007-2014 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
int zmq::proxy (
    class socket_base_t *frontend_,
    class socket_base_t *backend_,
60 61
    class socket_base_t *capture_,
    class socket_base_t *control_)
Pieter Hintjens's avatar
Pieter Hintjens committed
62 63 64
{
    msg_t msg;
    int rc = msg.init ();
Pieter Hintjens's avatar
Pieter Hintjens committed
65
    if (rc != 0)
Pieter Hintjens's avatar
Pieter Hintjens committed
66 67
        return -1;

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

Pieter Hintjens's avatar
Pieter Hintjens committed
71
    int more;
Pieter Hintjens's avatar
Pieter Hintjens committed
72
    size_t moresz;
Pieter Hintjens's avatar
Pieter Hintjens committed
73
    zmq_pollitem_t items [] = {
74
        { frontend_, 0, ZMQ_POLLIN, 0 },
75 76
        { backend_, 0, ZMQ_POLLIN, 0 },
        { control_, 0, ZMQ_POLLIN, 0 }
Pieter Hintjens's avatar
Pieter Hintjens committed
77
    };
78
    int qt_poll_items = (control_ ? 3 : 2);
Pieter Hintjens's avatar
Pieter Hintjens committed
79 80 81 82 83 84 85 86 87

    //  Proxy can be in these three states
    enum {
        active,
        paused,
        terminated
    } state = active;

    while (state != terminated) {
Pieter Hintjens's avatar
Pieter Hintjens committed
88
        //  Wait while there are either requests or replies to process.
89
        rc = zmq_poll (&items [0], qt_poll_items, -1);
Pieter Hintjens's avatar
Pieter Hintjens committed
90
        if (unlikely (rc < 0))
Pieter Hintjens's avatar
Pieter Hintjens committed
91 92
            return -1;

93 94
        //  Process a control command if any
        if (control_ && items [2].revents & ZMQ_POLLIN) {
Pieter Hintjens's avatar
Pieter Hintjens committed
95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116
            rc = control_->recv (&msg, 0);
            if (unlikely (rc < 0))
                return -1;

            moresz = sizeof more;
            rc = control_->getsockopt (ZMQ_RCVMORE, &more, &moresz);
            if (unlikely (rc < 0) || more)
                return -1;

            //  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, 0);
                if (unlikely (rc < 0))
                    return -1;
            }
117
            if (msg.size () == 5 && memcmp (msg.data (), "PAUSE", 5) == 0)
Pieter Hintjens's avatar
Pieter Hintjens committed
118 119
                state = paused;
            else
120
            if (msg.size () == 6 && memcmp (msg.data (), "RESUME", 6) == 0)
Pieter Hintjens's avatar
Pieter Hintjens committed
121 122
                state = active;
            else
123
            if (msg.size () == 9 && memcmp (msg.data (), "TERMINATE", 9) == 0)
Pieter Hintjens's avatar
Pieter Hintjens committed
124 125 126 127
                state = terminated;
            else {
                //  This is an API error, we should assert
                puts ("E: invalid command sent to proxy");
128
                zmq_assert (false);
Pieter Hintjens's avatar
Pieter Hintjens committed
129
            }
130
        }
131
        //  Process a request
Pieter Hintjens's avatar
Pieter Hintjens committed
132 133
        if (state == active
        &&  items [0].revents & ZMQ_POLLIN) {
Pieter Hintjens's avatar
Pieter Hintjens committed
134
            while (true) {
135
                rc = frontend_->recv (&msg, 0);
Pieter Hintjens's avatar
Pieter Hintjens committed
136
                if (unlikely (rc < 0))
Pieter Hintjens's avatar
Pieter Hintjens committed
137
                    return -1;
138 139

                moresz = sizeof more;
140
                rc = frontend_->getsockopt (ZMQ_RCVMORE, &more, &moresz);
Pieter Hintjens's avatar
Pieter Hintjens committed
141
                if (unlikely (rc < 0))
Pieter Hintjens's avatar
Pieter Hintjens committed
142
                    return -1;
143

144 145 146 147 148 149 150 151 152 153 154 155 156 157
                //  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
158
                if (unlikely (rc < 0))
Pieter Hintjens's avatar
Pieter Hintjens committed
159
                    return -1;
Pieter Hintjens's avatar
Pieter Hintjens committed
160
                if (more == 0)
Pieter Hintjens's avatar
Pieter Hintjens committed
161 162 163
                    break;
            }
        }
164
        //  Process a reply
Pieter Hintjens's avatar
Pieter Hintjens committed
165 166
        if (state == active
        &&  items [1].revents & ZMQ_POLLIN) {
Pieter Hintjens's avatar
Pieter Hintjens committed
167
            while (true) {
168
                rc = backend_->recv (&msg, 0);
Pieter Hintjens's avatar
Pieter Hintjens committed
169 170 171 172 173 174 175 176 177 178 179 180
            if (unlikely (rc < 0))
                return -1;

            moresz = sizeof more;
            rc = backend_->getsockopt (ZMQ_RCVMORE, &more, &moresz);
            if (unlikely (rc < 0))
                return -1;

            //  Copy message to capture socket if any
            if (capture_) {
                msg_t ctrl;
                rc = ctrl.init ();
Pieter Hintjens's avatar
Pieter Hintjens committed
181
                if (unlikely (rc < 0))
Pieter Hintjens's avatar
Pieter Hintjens committed
182
                    return -1;
Pieter Hintjens's avatar
Pieter Hintjens committed
183
                rc = ctrl.copy (msg);
Pieter Hintjens's avatar
Pieter Hintjens committed
184
                if (unlikely (rc < 0))
Pieter Hintjens's avatar
Pieter Hintjens committed
185
                    return -1;
Pieter Hintjens's avatar
Pieter Hintjens committed
186
                rc = capture_->send (&ctrl, more? ZMQ_SNDMORE: 0);
Pieter Hintjens's avatar
Pieter Hintjens committed
187
                if (unlikely (rc < 0))
Pieter Hintjens's avatar
Pieter Hintjens committed
188
                    return -1;
Pieter Hintjens's avatar
Pieter Hintjens committed
189 190 191 192 193 194
            }
            rc = frontend_->send (&msg, more? ZMQ_SNDMORE: 0);
            if (unlikely (rc < 0))
                return -1;
            if (more == 0)
                break;
Pieter Hintjens's avatar
Pieter Hintjens committed
195 196 197 198 199
            }
        }
    }
    return 0;
}