diff --git a/Makefile.am b/Makefile.am index 7bfb35aa..e6ecfcc4 100644 --- a/Makefile.am +++ b/Makefile.am @@ -1,7 +1,7 @@ ACLOCAL_AMFLAGS = -I config -SUBDIRS = src doc perf devices -DIST_SUBDIRS = src doc perf devices builds/msvc +SUBDIRS = src doc perf devices zmqd +DIST_SUBDIRS = src doc perf devices zmqd builds/msvc EXTRA_DIST = \ $(top_srcdir)/foreign/openpgm/@pgm_basename@.tar.gz \ diff --git a/configure.in b/configure.in index 8d2f27d4..7ed5b1c1 100644 --- a/configure.in +++ b/configure.in @@ -382,7 +382,7 @@ AC_OUTPUT(Makefile src/Makefile doc/Makefile perf/Makefile src/libzmq.pc \ devices/Makefile devices/zmq_forwarder/Makefile \ devices/zmq_streamer/Makefile devices/zmq_queue/Makefile \ - builds/msvc/Makefile) + zmqd/Makefile builds/msvc/Makefile) # On Linux patch libtool to delete hardcoded paths (rpath). case "${host_os}" in diff --git a/zmqd/Makefile.am b/zmqd/Makefile.am new file mode 100644 index 00000000..32a657bc --- /dev/null +++ b/zmqd/Makefile.am @@ -0,0 +1,8 @@ +INCLUDES = -I$(top_srcdir)/include + +bin_PROGRAMS = zmqd + +zmqd_LDADD = $(top_builddir)/src/libzmq.la +zmqd_SOURCES = zmqd.cpp + + diff --git a/zmqd/zmqd.cpp b/zmqd/zmqd.cpp new file mode 100644 index 00000000..5b27ac26 --- /dev/null +++ b/zmqd/zmqd.cpp @@ -0,0 +1,368 @@ +/* + Copyright (c) 2007-2010 iMatix Corporation + + This file is part of 0MQ. + + 0MQ is free software; you can redistribute it and/or modify it under + the terms of the Lesser GNU 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 + Lesser GNU General Public License for more details. + + You should have received a copy of the Lesser GNU General Public License + along with this program. If not, see . +*/ + +#include +#include +#include + +#include "../include/zmq.hpp" +#include "../foreign/xmlParser/xmlParser.cpp" + +namespace { + + class device_cfg_t + { + + enum endpoint_direction {connect, bind}; + + typedef std::pair sock_details_t; + + typedef std::vector vsock_dets_t; + + public: + + explicit device_cfg_t(int type) + : device_type(type) , context(0) + , in_socket(0), out_socket(0) + { + } + + virtual ~device_cfg_t() + { + delete out_socket; + delete in_socket; + } + + bool init(XMLNode& device) + { + + XMLNode in_node = device.getChildNode ("in"); + if (in_node.isEmpty ()) { + fprintf (stderr, "'in' node is missing in the configuration file\n"); + return false; + } + + XMLNode out_node = device.getChildNode ("out"); + if (out_node.isEmpty ()) { + fprintf (stderr, "'out' node is missing in the configuration file\n"); + return false; + } + + if (!process_node(in_node,true,device_cfg_t::bind)) + return false; + if (!process_node(in_node,true,device_cfg_t::connect)) + return false; + if (!process_node(out_node,false,device_cfg_t::bind)) + return false; + if (!process_node(out_node,false,device_cfg_t::connect)) + return false; + + return true; + } + + void set_context(zmq::context_t* context_) + { + context = context_; + } + + zmq::context_t *get_context() const + { + return context; + } + + virtual bool make_sockets() = 0; + + bool set_up_connections() + { + for (vsock_dets_t::const_iterator i = in.begin() ; i != in.end(); + ++i) { + + switch (i->first) + { + case device_cfg_t::connect : + in_socket->connect(i->second.c_str()); + break; + case device_cfg_t::bind : + in_socket->bind(i->second.c_str()); + } + + } + + for (vsock_dets_t::const_iterator i = out.begin() ; i != out.end(); + ++i) { + + switch (i->first) + { + case device_cfg_t::connect : + out_socket->connect(i->second.c_str()); + break; + case device_cfg_t::bind : + out_socket->bind(i->second.c_str()); + } + + } + return true; + } + + void run() + { + zmq::device(device_type, *in_socket, *out_socket); + } + + + protected: + + bool make_sockets(int in_type, int out_type) + { + in_socket = new (std::nothrow) zmq::socket_t(*context, in_type); + if (!in_socket) + return false; + out_socket = new (std::nothrow) zmq::socket_t(*context, out_type); + if (!out_socket) { + return false; + } + return true; + } + + int process_node(XMLNode& target_, bool in_, + device_cfg_t::endpoint_direction ept_) + { + + const char * name = + (ept_ == device_cfg_t::connect) ? "connect" : "bind"; + int n = 0; + while (true) { + XMLNode connect = target_.getChildNode (name, n); + if (connect.isEmpty ()) + break; + const char *addr = connect.getAttribute ("addr"); + if (!addr) { + fprintf (stderr, "'%s' node is missing 'addr' attribute\n", + name); + return 0; + } + + if (in_) + in.push_back( sock_details_t(ept_, addr)); + else + out.push_back( sock_details_t(ept_, addr)); + + n++; + } + + return 1; + } + + + protected: + + int device_type; + zmq::context_t* context; + vsock_dets_t in; + vsock_dets_t out; + zmq::socket_t* in_socket; + zmq::socket_t* out_socket; + + private: + void operator = (device_cfg_t const &); + device_cfg_t(device_cfg_t const &); + }; + + + + class queue_device_cfg_t : public device_cfg_t + { + public: + queue_device_cfg_t() + : device_cfg_t(ZMQ_QUEUE) + {} + virtual bool make_sockets(){ + return device_cfg_t::make_sockets(ZMQ_XREP, ZMQ_XREQ); + } + }; + + + class streamer_device_cfg_t : public device_cfg_t + { + public: + streamer_device_cfg_t() + : device_cfg_t(ZMQ_STREAMER) + {} + virtual bool make_sockets () { + return device_cfg_t::make_sockets(ZMQ_UPSTREAM, ZMQ_DOWNSTREAM); + } + }; + + class forwarder_device_cfg_t : public device_cfg_t + { + public: + forwarder_device_cfg_t() + : device_cfg_t(ZMQ_FORWARDER) + {} + virtual bool make_sockets() { + if (!device_cfg_t::make_sockets(ZMQ_SUB, ZMQ_PUB) ) { + return false; + } + in_socket->setsockopt (ZMQ_SUBSCRIBE, "", 0); + return true; + } + }; + + + device_cfg_t* make_device_config(XMLNode& device) + { + const char *dev_type = device.getAttribute ("type"); + + if (!dev_type) { + fprintf (stderr, "'device' node is missing 'type' attribute\n"); + return NULL; + } + + if (strcmp (dev_type, "forwarder") == 0) { + return new (std::nothrow) forwarder_device_cfg_t; + } + else if (strcmp (dev_type, "streamer") == 0) { + return new (std::nothrow) streamer_device_cfg_t; + } + else if (strcmp (dev_type, "queue") == 0) { + return new (std::nothrow) queue_device_cfg_t; + } + + fprintf (stderr, "type attribute in the device configuration file " + "should be named 'forwarder', 'streamer' or 'queue'\n"); + + return NULL; + } + + + extern "C" void* worker_function(void *arg) + { + + if (!arg) { + fprintf (stderr, "arg is null, returning \n"); + return 0; + } + + std::auto_ptr cfg ( (device_cfg_t*) arg ); + + zmq::context_t* ctx = cfg->get_context(); + + if (!ctx) { + fprintf (stderr, "no context, returning \n"); + return 0; + } + + if (! cfg->make_sockets()) { + fprintf (stderr, "failed to make sockets, returning \n"); + return 0; + } + + + if (! cfg->set_up_connections()) { + fprintf (stderr, "failed to set up connections, returning \n"); + return 0; + } + + cfg->run(); + + return 0; + + } + + +} + + +int main (int argc, char *argv []) +{ + if (argc != 2) { + fprintf (stderr, "usage: zmqd \n"); + return 1; + } + + XMLNode root = XMLNode::parseFile (argv [1]); + + if (root.isEmpty ()) { + fprintf (stderr, "configuration file not found or not an XML file\n"); + return 1; + } + + if (strcmp (root.getName (), "config") != 0) { + fprintf (stderr, "root element in the configuration file should be " + "named 'config'\n"); + return 1; + } + + + std::vector vdev; + + while (true) { + + XMLNode device = root.getChildNode ("device", vdev.size()); + + if (device.isEmpty()) + break; + + device_cfg_t* dev = make_device_config(device); + + if (!dev) { + fprintf(stderr, "failed to create device config\n"); + return 1; + } + + if (! dev->init(device) ) { + + fprintf(stderr,"error with initialising device configuration\n"); + delete dev; + return 1; + } + + vdev.push_back(dev); + } + + std::vector::size_type num_devices = vdev.size(); + + if ( num_devices == 0 ) { + fprintf(stderr,"no devices in the config file\n"); + return 1; + } + + + zmq::context_t ctx (num_devices,1); + + + for (unsigned int i = 0 ; i < num_devices ; ++i) { + + vdev[i]->set_context(&ctx); + + if (i) { + pthread_t worker; + int rc = pthread_create (&worker, NULL, &worker_function, + (void*) vdev[i]); + assert (rc == 0); + } + } + + + worker_function((void*)vdev[0]); + + + return 0; +} +