/* This file is part of Ingen. * Copyright (C) 2007-2009 Dave Robillard * * 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 CONNECTIONIMPL_H #define CONNECTIONIMPL_H #include #include #include #include "raul/Deletable.hpp" #include "interface/PortType.hpp" #include "interface/Connection.hpp" #include "PortImpl.hpp" #include "PortImpl.hpp" using namespace std; namespace Ingen { class PortImpl; class Buffer; class BufferFactory; /** Represents a single inbound connection for an InputPort. * * This can be a group of ports (ie coming from a polyphonic Node) or * a single Port. This class exists basically as an abstraction of mixing * down polyphonic inputs, so InputPort can just deal with mixing down * multiple connections (oblivious to the polyphonic situation of the * connection itself). * * \ingroup engine */ class ConnectionImpl : public Raul::Deletable, public Shared::Connection { public: ConnectionImpl(BufferFactory& bufs, PortImpl* src_port, PortImpl* dst_port); PortImpl* src_port() const { return _src_port; } PortImpl* dst_port() const { return _dst_port; } const Raul::Path src_port_path() const { return _src_port->path(); } const Raul::Path dst_port_path() const { return _dst_port->path(); } /** Used by some (recursive) events to prevent double disconnections */ bool pending_disconnection() { return _pending_disconnection; } void pending_disconnection(bool b) { _pending_disconnection = b; } void process(Context& context); /** Get the buffer for a particular voice. * A Connection is smart - it knows the destination port requesting the * buffer, and will return accordingly (e.g. the same buffer for every * voice in a mono->poly connection). */ inline SharedPtr buffer(uint32_t voice) const { if (must_mix()) { return _local_buffer; } else if ( ! _src_port->polyphonic()) { return _src_port->buffer(0); } else { return _src_port->buffer(voice); } } void set_buffer_size(BufferFactory& bufs, size_t size); void prepare_poly(BufferFactory& bufs, uint32_t poly); void apply_poly(Raul::Maid& maid, uint32_t poly); /** Returns true if this connection must mix down voices into a local buffer */ inline bool must_mix() const { return _src_port->poly() > _dst_port->poly(); } protected: void dump() const; BufferFactory& _bufs; PortImpl* const _src_port; PortImpl* const _dst_port; SharedPtr _local_buffer; bool _pending_disconnection; }; } // namespace Ingen #endif // CONNECTIONIMPL_H