socket_base.hpp 6.72 KB
Newer Older
1
/*
2 3
    Copyright (c) 2007-2011 iMatix Corporation
    Copyright (c) 2007-2011 Other contributors as noted in the AUTHORS file
4 5 6 7

    This file is part of 0MQ.

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

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

#ifndef __ZMQ_SOCKET_BASE_HPP_INCLUDED__
#define __ZMQ_SOCKET_BASE_HPP_INCLUDED__

24
#include <string>
25

26
#include "own.hpp"
27
#include "array.hpp"
Martin Sustrik's avatar
Martin Sustrik committed
28
#include "stdint.hpp"
29
#include "poller.hpp"
30
#include "atomic_counter.hpp"
31
#include "i_poll_events.hpp"
32
#include "mailbox.hpp"
33
#include "stdint.hpp"
34
#include "pipe.hpp"
35 36 37 38

namespace zmq
{

39
    class socket_base_t :
40
        public own_t,
41
        public array_item_t <>,
42 43
        public i_poll_events,
        public i_pipe_events
44
    {
45 46
        friend class reaper_t;

47 48
    public:

49 50 51
        //  Returns false if object is not a socket.
        bool check_tag ();

52 53
        //  Create a socket of a specified type.
        static socket_base_t *create (int type_, class ctx_t *parent_,
Martin Sustrik's avatar
Martin Sustrik committed
54
            uint32_t tid_);
55

56 57
        //  Returns the mailbox associated with this socket.
        mailbox_t *get_mailbox ();
58 59 60 61

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

63
        //  Interface for communication with the API layer.
64 65
        int setsockopt (int option_, const void *optval_, size_t optvallen_);
        int getsockopt (int option_, void *optval_, size_t *optvallen_);
66 67
        int bind (const char *addr_);
        int connect (const char *addr_);
68 69
        int send (class msg_t *msg_, int flags_);
        int recv (class msg_t *msg_, int flags_);
70
        int close ();
71

72 73 74 75 76
        //  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 ();

77 78 79 80 81 82 83 84 85
        //  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
86

87 88 89
        //  i_pipe_events interface implementation.
        void read_activated (pipe_t *pipe_);
        void write_activated (pipe_t *pipe_);
90
        void hiccuped (pipe_t *pipe_);
91
        void terminated (pipe_t *pipe_);
92

93 94
    protected:

Martin Sustrik's avatar
Martin Sustrik committed
95
        socket_base_t (class ctx_t *parent_, uint32_t tid_);
96 97
        virtual ~socket_base_t ();

98 99
        //  Concrete algorithms for the x- methods are to be defined by
        //  individual socket types.
100
        virtual void xattach_pipe (class pipe_t *pipe_) = 0;
101

102 103 104
        //  The default implementation assumes there are no specific socket
        //  options for the particular socket type. If not so, overload this
        //  method.
105
        virtual int xsetsockopt (int option_, const void *optval_,
106 107 108 109
            size_t optvallen_);

        //  The default implementation assumes that send is not supported.
        virtual bool xhas_out ();
110
        virtual int xsend (class msg_t *msg_, int flags_);
111 112 113

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

116 117 118
        //  i_pipe_events will be forwarded to these functions.
        virtual void xread_activated (pipe_t *pipe_);
        virtual void xwrite_activated (pipe_t *pipe_);
119
        virtual void xhiccuped (pipe_t *pipe_);
120
        virtual void xterminated (pipe_t *pipe_) = 0;
121

122 123 124
        //  Delay actual destruction of the socket.
        void process_destroy ();

125 126
    private:

127 128 129 130
        //  To be called after processing commands or invoking any command
        //  handlers explicitly. If required, it will deallocate the socket.
        void check_destroy ();

131 132 133 134
        //  Moves the flags from the message to local variables,
        //  to be later retrieved by getsockopt.
        void extract_flags (msg_t *msg_);

135 136 137
        //  Used to check whether the object is a socket.
        uint32_t tag;

138 139
        //  If true, associated context was already terminated.
        bool ctx_terminated;
140

141 142 143 144 145
        //  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;

146 147 148 149
        //  Parse URI string.
        int parse_uri (const char *uri_, std::string &protocol_,
            std::string &address_);

150 151 152
        //  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_);
153

154
        //  Register the pipe with this socket.
155
        void attach_pipe (class pipe_t *pipe_);
156

157 158
        //  Processes commands sent to this socket (if any). If timeout is -1,
        //  returns only after at least one command was processed.
159 160
        //  If throttle argument is true, commands are processed at most once
        //  in a predefined time period.
161
        int process_commands (int timeout_, bool throttle_);
162

163
        //  Handlers for incoming commands.
164
        void process_stop ();
165
        void process_bind (class pipe_t *pipe_);
166
        void process_unplug ();
167
        void process_term (int linger_);
168

169 170
        //  Socket's mailbox object.
        mailbox_t mailbox;
171

172 173 174 175
        //  List of attached pipes.
        typedef array_t <pipe_t, 3> pipes_t;
        pipes_t pipes;

176 177 178 179
        //  Reaper's poller and handle of this socket within it.
        poller_t *poller;
        poller_t::handle_t handle;

180
        //  Timestamp of when commands were processed the last time.
Martin Sustrik's avatar
Martin Sustrik committed
181
        uint64_t last_tsc;
182

Martin Sustrik's avatar
Martin Sustrik committed
183 184 185
        //  Number of messages received since last command processing.
        int ticks;

186 187 188
        //  True if the last message received had LABEL flag set.
        bool rcvlabel;

189 190 191
        //  True if the last message received had COMMAND flag set.
        bool rcvcmd;

192
        //  True if the last message received had MORE flag set.
193 194
        bool rcvmore;

195
        socket_base_t (const socket_base_t&);
196
        const socket_base_t &operator = (const socket_base_t&);
197 198 199 200 201
    };

}

#endif