diff options
| author | Martin Lucina <mato@kotelna.sk> | 2011-03-28 10:39:51 +0200 | 
|---|---|---|
| committer | Martin Lucina <martin@lucina.net> | 2012-01-23 08:53:37 +0100 | 
| commit | 3e20cb1b8a2b1ca222011df37334e5f4f88dd565 (patch) | |
| tree | 4a753775186bc7f583f1ceb3f9aa675b6f110596 /src/device.cpp | |
| parent | 3f0085ddbef1a44b6bb7a0b23af497d56e0025fa (diff) | |
| parent | e645fc2693acc796304498909786b7b47005b429 (diff) | |
Imported Debian patch 2.1.3-1debian/2.1.3-1
Diffstat (limited to 'src/device.cpp')
| -rw-r--r-- | src/device.cpp | 120 | 
1 files changed, 120 insertions, 0 deletions
| diff --git a/src/device.cpp b/src/device.cpp new file mode 100644 index 0000000..351283a --- /dev/null +++ b/src/device.cpp @@ -0,0 +1,120 @@ +/* +    Copyright (c) 2007-2011 iMatix Corporation +    Copyright (c) 2007-2011 Other contributors as noted in the AUTHORS file + +    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> + +#include "../include/zmq.h" + +#include "device.hpp" +#include "socket_base.hpp" +#include "likely.hpp" +#include "err.hpp" + +int zmq::device (class socket_base_t *insocket_, +        class socket_base_t *outsocket_) +{ +    zmq_msg_t msg; +    int rc = zmq_msg_init (&msg); + +    if (rc != 0) { +        return -1; +    } + +    int64_t more; +    size_t moresz; + +    zmq_pollitem_t items [2]; +    items [0].socket = insocket_; +    items [0].fd = 0; +    items [0].events = ZMQ_POLLIN; +    items [0].revents = 0; +    items [1].socket = outsocket_; +    items [1].fd = 0; +    items [1].events = ZMQ_POLLIN; +    items [1].revents = 0; + +    while (true) { + +        //  Wait while there are either requests or replies to process. +        rc = zmq_poll (&items [0], 2, -1); +        if (unlikely (rc < 0)) { +            return -1; +        } + +        //  The algorithm below asumes ratio of request and replies processed +        //  under full load to be 1:1. Although processing requests replies +        //  first is tempting it is suspectible to DoS attacks (overloading +        //  the system with unsolicited replies). + +        //  Process a request. +        if (items [0].revents & ZMQ_POLLIN) { +            while (true) { + +                rc = insocket_->recv (&msg, 0); +                if (unlikely (rc < 0)) { +                    return -1; +                } + +                moresz = sizeof (more); +                rc = insocket_->getsockopt (ZMQ_RCVMORE, &more, &moresz); +                if (unlikely (rc < 0)) { +                    return -1; +                } + +                rc = outsocket_->send (&msg, more ? ZMQ_SNDMORE : 0); +                if (unlikely (rc < 0)) { +                    return -1; +                } + +                if (!more) +                    break; +            } +        } + +        //  Process a reply. +        if (items [1].revents & ZMQ_POLLIN) { +            while (true) { + +                rc = outsocket_->recv (&msg, 0); +                if (unlikely (rc < 0)) { +                    return -1; +                } + +                moresz = sizeof (more); +                rc = outsocket_->getsockopt (ZMQ_RCVMORE, &more, &moresz); +                if (unlikely (rc < 0)) { +                    return -1; +                } + +                rc = insocket_->send (&msg, more ? ZMQ_SNDMORE : 0); +                if (unlikely (rc < 0)) { +                    return -1; +                } + +                if (!more) +                    break; +            } +        } + +    } + +    return 0; +} + | 
