proxy.cpp 5.09 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>
21
#include "poller.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
//  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

33 34 35 36 37
// 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
38 39 40
// zmq.h must be included *after* poll.h for AIX to build properly
#include "../include/zmq.h"

41
int capture(
42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61
        class zmq::socket_base_t *capture_,
        zmq::msg_t& msg_,
        int more_ = 0)
{
    //  Copy message to capture socket if any
    if (capture_) {
        zmq::msg_t ctrl;
        int 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;
    }
    return 0;
}

62
int forward(
63 64 65
        class zmq::socket_base_t *from_,
        class zmq::socket_base_t *to_,
        class zmq::socket_base_t *capture_,
66
        zmq::msg_t& msg_)
67 68 69
{
    int more;
    size_t moresz;
70
    while (true) {
71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92
        int rc = from_->recv (&msg_, 0);
        if (unlikely (rc < 0))
            return -1;

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

        //  Copy message to capture socket if any
        rc = capture(capture_, msg_, more);
        if (unlikely (rc < 0))
            return -1;

        rc = to_->send (&msg_, more? ZMQ_SNDMORE: 0);
        if (unlikely (rc < 0))
            return -1;
        if (more == 0)
            break;
    }
    return 0;
}
Pieter Hintjens's avatar
Pieter Hintjens committed
93

94
int zmq::proxy (
95 96 97
    class socket_base_t *frontend_,
    class socket_base_t *backend_,
    class socket_base_t *capture_,
98
    class socket_base_t *control_)
Pieter Hintjens's avatar
Pieter Hintjens committed
99 100 101
{
    msg_t msg;
    int rc = msg.init ();
Pieter Hintjens's avatar
Pieter Hintjens committed
102
    if (rc != 0)
Pieter Hintjens's avatar
Pieter Hintjens committed
103 104
        return -1;

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

Pieter Hintjens's avatar
Pieter Hintjens committed
108
    int more;
Pieter Hintjens's avatar
Pieter Hintjens committed
109
    size_t moresz;
110 111 112 113 114 115
    zmq_pollitem_t items [] = {
        { frontend_, 0, ZMQ_POLLIN, 0 },
        { backend_, 0, ZMQ_POLLIN, 0 },
        { control_, 0, ZMQ_POLLIN, 0 }
    };
    int qt_poll_items = (control_ ? 3 : 2);
Pieter Hintjens's avatar
Pieter Hintjens committed
116 117 118 119 120 121 122 123 124

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

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

130
        //  Process a control command if any
131
        if (control_ && items [2].revents & ZMQ_POLLIN) {
Pieter Hintjens's avatar
Pieter Hintjens committed
132 133 134 135 136 137 138 139 140 141
            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
142 143 144 145
            rc = capture(capture_, msg);
            if (unlikely (rc < 0))
                return -1;

146
            if (msg.size () == 5 && memcmp (msg.data (), "PAUSE", 5) == 0)
Pieter Hintjens's avatar
Pieter Hintjens committed
147 148
                state = paused;
            else
149
            if (msg.size () == 6 && memcmp (msg.data (), "RESUME", 6) == 0)
Pieter Hintjens's avatar
Pieter Hintjens committed
150 151
                state = active;
            else
152
            if (msg.size () == 9 && memcmp (msg.data (), "TERMINATE", 9) == 0)
Pieter Hintjens's avatar
Pieter Hintjens committed
153 154 155 156
                state = terminated;
            else {
                //  This is an API error, we should assert
                puts ("E: invalid command sent to proxy");
157
                zmq_assert (false);
Pieter Hintjens's avatar
Pieter Hintjens committed
158
            }
159
        }
160 161 162
        //  Process a request
        if (state == active
        &&  items [0].revents & ZMQ_POLLIN) {
163
            rc = forward(frontend_, backend_, capture_,msg);
164 165 166 167 168 169
            if (unlikely (rc < 0))
                return -1;
        }
        //  Process a reply
        if (state == active
        &&  items [1].revents & ZMQ_POLLIN) {
170
            rc = forward(backend_, frontend_, capture_,msg);
171 172
            if (unlikely (rc < 0))
                return -1;
Pieter Hintjens's avatar
Pieter Hintjens committed
173 174 175 176
        }
    }
    return 0;
}