/* This file is part of Ingen. * Copyright (C) 2007-2009 David Robillard <http://drobilla.net> * * Ingen is free software; you can redistribute it and/or modify it under the * terms of the GNU General Public License as published by the Free Software * Foundation; either version 2 of the License, or (at your option) any later * version. * * Ingen 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 General Public License for details. * * You should have received a copy of the GNU General Public License along * with this program; if not, write to the Free Software Foundation, Inc., * 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA */ #ifndef INGEN_ENGINE_OSCENGINERECEIVER_HPP #define INGEN_ENGINE_OSCENGINERECEIVER_HPP #include <stdint.h> #include <lo/lo.h> #include "QueuedEngineInterface.hpp" #include "Request.hpp" #include "ingen-config.h" namespace Ingen { class JackDriver; class NodeFactory; class PatchImpl; /* Some boilerplate killing macros... */ #define LO_HANDLER_ARGS const char* path, const char* types, lo_arg** argv, int argc, lo_message msg /* Defines a static handler to be passed to lo_add_method, which is a trivial * wrapper around a non-static method that does the real work. Makes a whoole * lot of ugly boiler plate go away */ #define LO_HANDLER(name) \ int _##name##_cb (LO_HANDLER_ARGS);\ inline static int name##_cb(LO_HANDLER_ARGS, void* myself)\ { return ((OSCEngineReceiver*)myself)->_##name##_cb(path, types, argv, argc, msg); } /* FIXME: Make this receive and preprocess in the same thread? */ /** Receives OSC messages from liblo. * * This inherits from QueuedEngineInterface and calls it's own functions * via OSC. It's not actually a directly callable EngineInterface (it's * callable via OSC...) so it should be implemented-as-a (privately inherit) * QueuedEngineInterface, but it needs to be public so it's an EventSource * the Driver can use. This probably should be fixed somehow.. * * \ingroup engine */ class OSCEngineReceiver : public QueuedEngineInterface { public: OSCEngineReceiver(Engine& engine, size_t queue_size, uint16_t port); ~OSCEngineReceiver(); void activate_source(); void deactivate_source(); private: struct ReceiveThread : public Raul::Thread { explicit ReceiveThread(OSCEngineReceiver& receiver) : _receiver(receiver) {} virtual void _run(); private: OSCEngineReceiver& _receiver; }; friend class ReceiveThread; ReceiveThread* _receive_thread; #ifdef LIBLO_BUNDLES static int bundle_start_cb(lo_timetag time, void* myself) { return ((OSCEngineReceiver*)myself)->_bundle_start_cb(time); } static int bundle_end_cb(void* myself) { return ((OSCEngineReceiver*)myself)->_bundle_end_cb(); } int _bundle_start_cb(lo_timetag time); int _bundle_end_cb(); #endif static void error_cb(int num, const char* msg, const char* path); static int set_response_address_cb(LO_HANDLER_ARGS, void* myself); static int generic_cb(LO_HANDLER_ARGS, void* myself); static int unknown_cb(LO_HANDLER_ARGS, void* myself); LO_HANDLER(quit); LO_HANDLER(ping); LO_HANDLER(ping_slow); LO_HANDLER(register_client); LO_HANDLER(unregister_client); LO_HANDLER(load_plugins); LO_HANDLER(engine_activate); LO_HANDLER(engine_deactivate); LO_HANDLER(get); LO_HANDLER(put); LO_HANDLER(move); LO_HANDLER(del); LO_HANDLER(connect); LO_HANDLER(disconnect); LO_HANDLER(disconnect_all); LO_HANDLER(note_on); LO_HANDLER(note_off); LO_HANDLER(all_notes_off); LO_HANDLER(learn); LO_HANDLER(set_property); LO_HANDLER(property_set); LO_HANDLER(request_property); lo_server _server; }; } // namespace Ingen #endif // INGEN_ENGINE_OSCENGINERECEIVER_HPP