/* This file is part of Ingen. Copyright 2007-2012 David Robillard Ingen is free software: you can redistribute it and/or modify it under the terms of the GNU Affero General Public License as published by the Free Software Foundation, either version 3 of the License, or 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 Affero General Public License for details. You should have received a copy of the GNU Affero General Public License along with Ingen. If not, see . */ #include #include #include #include "raul/Maid.hpp" #include "raul/Path.hpp" #include "ClientBroadcaster.hpp" #include "Connect.hpp" #include "ConnectionImpl.hpp" #include "DuplexPort.hpp" #include "Engine.hpp" #include "EngineStore.hpp" #include "InputPort.hpp" #include "OutputPort.hpp" #include "PatchImpl.hpp" #include "PortImpl.hpp" #include "ProcessContext.hpp" #include "types.hpp" using namespace std; using namespace Raul; namespace Ingen { namespace Server { namespace Events { Connect::Connect(Engine& engine, Interface* client, int32_t id, SampleCount timestamp, const Path& src_port_path, const Path& dst_port_path) : Event(engine, client, id, timestamp) , _src_port_path(src_port_path) , _dst_port_path(dst_port_path) , _patch(NULL) , _src_output_port(NULL) , _dst_input_port(NULL) , _compiled_patch(NULL) , _buffers(NULL) {} void Connect::pre_process() { Glib::RWLock::ReaderLock rlock(_engine.engine_store()->lock()); PortImpl* src_port = _engine.engine_store()->find_port(_src_port_path); PortImpl* dst_port = _engine.engine_store()->find_port(_dst_port_path); if (!src_port || !dst_port) { _status = PORT_NOT_FOUND; Event::pre_process(); return; } _dst_input_port = dynamic_cast(dst_port); _src_output_port = dynamic_cast(src_port); if (!_dst_input_port || !_src_output_port) { _status = DIRECTION_MISMATCH; Event::pre_process(); return; } NodeImpl* const src_node = src_port->parent_node(); NodeImpl* const dst_node = dst_port->parent_node(); if (!src_node || !dst_node) { _status = PARENT_NOT_FOUND; Event::pre_process(); return; } if (src_node->parent() != dst_node->parent() && src_node != dst_node->parent() && src_node->parent() != dst_node) { _status = PARENT_DIFFERS; Event::pre_process(); return; } if (!ConnectionImpl::can_connect(_src_output_port, _dst_input_port)) { _status = TYPE_MISMATCH; Event::pre_process(); return; } // Connection to a patch port from inside the patch if (src_node->parent_patch() != dst_node->parent_patch()) { assert(src_node->parent() == dst_node || dst_node->parent() == src_node); if (src_node->parent() == dst_node) _patch = dynamic_cast(dst_node); else _patch = dynamic_cast(src_node); // Connection from a patch input to a patch output (pass through) } else if (src_node == dst_node && dynamic_cast(src_node)) { _patch = dynamic_cast(src_node); // Normal connection between nodes with the same parent } else { _patch = src_node->parent_patch(); } if (_patch->has_connection(_src_output_port, _dst_input_port)) { _status = EXISTS; Event::pre_process(); return; } _connection = SharedPtr( new ConnectionImpl(_src_output_port, _dst_input_port)); rlock.release(); { Glib::RWLock::ReaderLock wlock(_engine.engine_store()->lock()); /* Need to be careful about patch port connections here and adding a node's parent as a dependant/provider, or adding a patch as its own provider... */ if (src_node != dst_node && src_node->parent() == dst_node->parent()) { dst_node->providers().push_back(src_node); src_node->dependants().push_back(dst_node); } _patch->add_connection(_connection); _dst_input_port->increment_num_connections(); } _buffers = new Raul::Array(_dst_input_port->poly()); _dst_input_port->get_buffers(*_engine.buffer_factory(), _buffers, _dst_input_port->poly()); if (_patch->enabled()) _compiled_patch = _patch->compile(); Event::pre_process(); } void Connect::execute(ProcessContext& context) { Event::execute(context); if (_status == SUCCESS) { // This must be inserted here, since they're actually used by the audio thread _dst_input_port->add_connection(_connection.get()); assert(_buffers); _engine.maid()->push(_dst_input_port->set_buffers(_buffers)); _dst_input_port->connect_buffers(); _engine.maid()->push(_patch->compiled_patch()); _patch->compiled_patch(_compiled_patch); } } void Connect::post_process() { respond(_status); if (!_status) { _engine.broadcaster()->connect(_src_port_path, _dst_port_path); } } } // namespace Events } // namespace Server } // namespace Ingen