socket_base.hpp 7.24 KB
Newer Older
1
/*
2
    Copyright (c) 2007-2012 iMatix Corporation
Martin Sustrik's avatar
Martin Sustrik committed
3
    Copyright (c) 2009-2011 250bpm s.r.o.
4
    Copyright (c) 2011 VMware, Inc.
5
    Copyright (c) 2007-2011 Other contributors as noted in the AUTHORS file
6 7 8 9

    This file is part of 0MQ.

    0MQ is free software; you can redistribute it and/or modify it under
10
    the terms of the GNU Lesser General Public License as published by
11 12 13 14 15 16
    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
17
    GNU Lesser General Public License for more details.
18

19
    You should have received a copy of the GNU Lesser General Public License
20 21 22 23 24 25
    along with this program.  If not, see <http://www.gnu.org/licenses/>.
*/

#ifndef __ZMQ_SOCKET_BASE_HPP_INCLUDED__
#define __ZMQ_SOCKET_BASE_HPP_INCLUDED__

26
#include <string>
27
#include <map>
28

29
#include "own.hpp"
30
#include "array.hpp"
Martin Sustrik's avatar
Martin Sustrik committed
31
#include "stdint.hpp"
32
#include "poller.hpp"
33
#include "atomic_counter.hpp"
34
#include "i_poll_events.hpp"
35
#include "mailbox.hpp"
36
#include "stdint.hpp"
37
#include "clock.hpp"
38
#include "pipe.hpp"
39 40 41 42

namespace zmq
{

43 44 45 46
    class ctx_t;
    class msg_t;
    class pipe_t;

47
    class socket_base_t :
48
        public own_t,
49
        public array_item_t <>,
50 51
        public i_poll_events,
        public i_pipe_events
52
    {
53 54
        friend class reaper_t;

55 56
    public:

57 58 59
        //  Returns false if object is not a socket.
        bool check_tag ();

60
        //  Create a socket of a specified type.
61
        static socket_base_t *create (int type_, zmq::ctx_t *parent_,
62
            uint32_t tid_, int sid_);
63

64 65
        //  Returns the mailbox associated with this socket.
        mailbox_t *get_mailbox ();
66 67 68 69

        //  Interrupt blocking call if the socket is stuck in one.
        //  This function can be called from a different thread!
        void stop ();
70

71
        //  Interface for communication with the API layer.
72 73
        int setsockopt (int option_, const void *optval_, size_t optvallen_);
        int getsockopt (int option_, void *optval_, size_t *optvallen_);
74 75
        int bind (const char *addr_);
        int connect (const char *addr_);
76
        int term_endpoint (const char *addr_);
77 78
        int send (zmq::msg_t *msg_, int flags_);
        int recv (zmq::msg_t *msg_, int flags_);
79
        int close ();
80

81 82 83 84 85
        //  These functions are used by the polling mechanism to determine
        //  which events are to be reported from this socket.
        bool has_in ();
        bool has_out ();

86 87 88 89 90 91 92 93 94
        //  Using this function reaper thread ask the socket to regiter with
        //  its poller.
        void start_reaping (poller_t *poller_);

        //  i_poll_events implementation. This interface is used when socket
        //  is handled by the poller in the reaper thread.
        void in_event ();
        void out_event ();
        void timer_event (int id_);
Martin Sustrik's avatar
Martin Sustrik committed
95

96 97 98
        //  i_pipe_events interface implementation.
        void read_activated (pipe_t *pipe_);
        void write_activated (pipe_t *pipe_);
99
        void hiccuped (pipe_t *pipe_);
100
        void terminated (pipe_t *pipe_);
skaller's avatar
skaller committed
101 102
        void lock();
        void unlock();
103 104 105

        void monitor_event (int event_, ...);

106 107
    protected:

108
        socket_base_t (zmq::ctx_t *parent_, uint32_t tid_, int sid_);
109 110
        virtual ~socket_base_t ();

111 112
        //  Concrete algorithms for the x- methods are to be defined by
        //  individual socket types.
113 114
        virtual void xattach_pipe (zmq::pipe_t *pipe_,
            bool icanhasall_ = false) = 0;
115

116 117 118
        //  The default implementation assumes there are no specific socket
        //  options for the particular socket type. If not so, overload this
        //  method.
119
        virtual int xsetsockopt (int option_, const void *optval_,
120 121 122 123
            size_t optvallen_);

        //  The default implementation assumes that send is not supported.
        virtual bool xhas_out ();
124
        virtual int xsend (zmq::msg_t *msg_, int flags_);
125 126 127

        //  The default implementation assumes that recv in not supported.
        virtual bool xhas_in ();
128
        virtual int xrecv (zmq::msg_t *msg_, int flags_);
Martin Sustrik's avatar
Martin Sustrik committed
129

130 131 132
        //  i_pipe_events will be forwarded to these functions.
        virtual void xread_activated (pipe_t *pipe_);
        virtual void xwrite_activated (pipe_t *pipe_);
133
        virtual void xhiccuped (pipe_t *pipe_);
134
        virtual void xterminated (pipe_t *pipe_) = 0;
135

136 137 138
        //  Delay actual destruction of the socket.
        void process_destroy ();

139
    private:
140 141 142 143 144 145
        //  Creates new endpoint ID and adds the endpoint to the map.
        void add_endpoint (const char *addr_, own_t *endpoint_);

        //  Map of open endpoints.
        typedef std::multimap <std::string, own_t *> endpoints_t;
        endpoints_t endpoints;
146

147 148 149 150
        //  To be called after processing commands or invoking any command
        //  handlers explicitly. If required, it will deallocate the socket.
        void check_destroy ();

151 152 153 154
        //  Moves the flags from the message to local variables,
        //  to be later retrieved by getsockopt.
        void extract_flags (msg_t *msg_);

155 156 157
        //  Used to check whether the object is a socket.
        uint32_t tag;

158 159
        //  If true, associated context was already terminated.
        bool ctx_terminated;
160

161 162 163 164 165
        //  If true, object should have been already destroyed. However,
        //  destruction is delayed while we unwind the stack to the point
        //  where it doesn't intersect the object being destroyed.
        bool destroyed;

166 167 168 169
        //  Parse URI string.
        int parse_uri (const char *uri_, std::string &protocol_,
            std::string &address_);

170 171 172
        //  Check whether transport protocol, as specified in connect or
        //  bind, is available and compatible with the socket type.
        int check_protocol (const std::string &protocol_);
173

174
        //  Register the pipe with this socket.
175
        void attach_pipe (zmq::pipe_t *pipe_, bool icanhasall_ = false);
176

177 178
        //  Processes commands sent to this socket (if any). If timeout is -1,
        //  returns only after at least one command was processed.
179 180
        //  If throttle argument is true, commands are processed at most once
        //  in a predefined time period.
181
        int process_commands (int timeout_, bool throttle_);
182

183
        //  Handlers for incoming commands.
184
        void process_stop ();
185
        void process_bind (zmq::pipe_t *pipe_);
186
        void process_term (int linger_);
187

188 189
        //  Socket's mailbox object.
        mailbox_t mailbox;
190

191 192 193 194
        //  List of attached pipes.
        typedef array_t <pipe_t, 3> pipes_t;
        pipes_t pipes;

195 196 197 198
        //  Reaper's poller and handle of this socket within it.
        poller_t *poller;
        poller_t::handle_t handle;

199
        //  Timestamp of when commands were processed the last time.
Martin Sustrik's avatar
Martin Sustrik committed
200
        uint64_t last_tsc;
201

Martin Sustrik's avatar
Martin Sustrik committed
202 203 204
        //  Number of messages received since last command processing.
        int ticks;

205
        //  True if the last message received had MORE flag set.
206 207
        bool rcvmore;

208 209 210
        //  Improves efficiency of time measurement.
        clock_t clock;

211
        socket_base_t (const socket_base_t&);
212
        const socket_base_t &operator = (const socket_base_t&);
skaller's avatar
skaller committed
213
        mutex_t sync;
214 215 216 217 218
    };

}

#endif