/* Copyright (C) 2009 Paul Davis This program 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. This program 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 more 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., 675 Mass Ave, Cambridge, MA 02139, USA. */ #include #include "pbd/failed_constructor.h" #include "ardour/audio_buffer.h" #include "ardour/internal_return.h" #include "ardour/mute_master.h" #include "ardour/session.h" #include "ardour/internal_send.h" #include "ardour/audioengine.h" using namespace std; using namespace ARDOUR; PBD::Signal1 InternalReturn::CycleStart; InternalReturn::InternalReturn (Session& s) : Return (s, true) { CycleStart.connect_same_thread (*this, boost::bind (&InternalReturn::cycle_start, this, _1)); _display_to_user = false; } void InternalReturn::run (BufferSet& bufs, framepos_t /*start_frame*/, framepos_t /*end_frame*/, pframes_t nframes, bool) { if (!_active && !_pending_active) { return; } /* _sends is only modified with the process lock held, so this is ok, I think */ for (list::iterator i = _sends.begin(); i != _sends.end(); ++i) { if ((*i)->active ()) { bufs.merge_from ((*i)->get_buffers(), nframes); } } _active = _pending_active; } bool InternalReturn::configure_io (ChanCount in, ChanCount out) { IOProcessor::configure_io (in, out); allocate_buffers (_session.engine().frames_per_cycle()); return true; } int InternalReturn::set_block_size (pframes_t nframes) { allocate_buffers (nframes); return 0; } void InternalReturn::allocate_buffers (pframes_t nframes) { buffers.ensure_buffers (_configured_input, nframes); buffers.set_count (_configured_input); } void InternalReturn::add_send (InternalSend* send) { Glib::Mutex::Lock lm (_session.engine().process_lock()); _sends.push_back (send); } void InternalReturn::remove_send (InternalSend* send) { Glib::Mutex::Lock lm (_session.engine().process_lock()); _sends.remove (send); } void InternalReturn::cycle_start (pframes_t nframes) { /* called from process cycle - no lock necessary */ if (!_sends.empty ()) { /* don't bother with this if nobody is going to feed us anything */ buffers.silence (nframes, 0); } } XMLNode& InternalReturn::state (bool full) { XMLNode& node (Return::state (full)); /* override type */ node.add_property("type", "intreturn"); return node; } XMLNode& InternalReturn::get_state() { return state (true); } int InternalReturn::set_state (const XMLNode& node, int version) { return Return::set_state (node, version); } bool InternalReturn::can_support_io_configuration (const ChanCount& in, ChanCount& out) const { out = in; return true; }