diff options
author | Paul Davis <paul@linuxaudiosystems.com> | 2008-06-02 21:41:35 +0000 |
---|---|---|
committer | Paul Davis <paul@linuxaudiosystems.com> | 2008-06-02 21:41:35 +0000 |
commit | 449aab3c465bbbf66d221fac3d7ea559f1720357 (patch) | |
tree | 6843cc40c88250a132acac701271f1504cd2df04 /libs/ardour/io.cc | |
parent | 9c0d7d72d70082a54f823cd44c0ccda5da64bb6f (diff) |
rollback to 3428, before the mysterious removal of libs/* at 3431/3432
git-svn-id: svn://localhost/ardour2/branches/3.0@3435 d708f5d6-7413-0410-9779-e7cbd77b26cf
Diffstat (limited to 'libs/ardour/io.cc')
-rw-r--r-- | libs/ardour/io.cc | 2612 |
1 files changed, 2612 insertions, 0 deletions
diff --git a/libs/ardour/io.cc b/libs/ardour/io.cc new file mode 100644 index 0000000000..5d4b41cf32 --- /dev/null +++ b/libs/ardour/io.cc @@ -0,0 +1,2612 @@ +/* + Copyright (C) 2000-2006 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 <fstream> +#include <algorithm> +#include <unistd.h> +#include <locale.h> +#include <errno.h> + +#include <sigc++/bind.h> + +#include <glibmm/thread.h> + +#include <pbd/xml++.h> +#include <pbd/replace_all.h> +#include <pbd/unknown_type.h> + +#include <ardour/audioengine.h> +#include <ardour/io.h> +#include <ardour/route.h> +#include <ardour/port.h> +#include <ardour/audio_port.h> +#include <ardour/midi_port.h> +#include <ardour/auto_bundle.h> +#include <ardour/session.h> +#include <ardour/cycle_timer.h> +#include <ardour/panner.h> +#include <ardour/buffer_set.h> +#include <ardour/meter.h> +#include <ardour/amp.h> + +#include "i18n.h" + +#include <cmath> + +/* + A bug in OS X's cmath that causes isnan() and isinf() to be + "undeclared". the following works around that +*/ + +#if defined(__APPLE__) && defined(__MACH__) +extern "C" int isnan (double); +extern "C" int isinf (double); +#endif + +#define BLOCK_PROCESS_CALLBACK() Glib::Mutex::Lock em (_session.engine().process_lock()) + +using namespace std; +using namespace ARDOUR; +using namespace PBD; + +const string IO::state_node_name = "IO"; +bool IO::connecting_legal = false; +bool IO::ports_legal = false; +bool IO::panners_legal = false; +sigc::signal<void> IO::Meter; +sigc::signal<int> IO::ConnectingLegal; +sigc::signal<int> IO::PortsLegal; +sigc::signal<int> IO::PannersLegal; +sigc::signal<void,ChanCount> IO::PortCountChanged; +sigc::signal<int> IO::PortsCreated; + +Glib::StaticMutex IO::m_meter_signal_lock = GLIBMM_STATIC_MUTEX_INIT; + +/* this is a default mapper of [0 .. 1.0] control values to a gain coefficient. + others can be imagined. +*/ + +#if 0 +static gain_t direct_control_to_gain (double fract) { + /* XXX Marcus writes: this doesn't seem right to me. but i don't have a better answer ... */ + /* this maxes at +6dB */ + return pow (2.0,(sqrt(sqrt(sqrt(fract)))*198.0-192.0)/6.0); +} + +static double direct_gain_to_control (gain_t gain) { + /* XXX Marcus writes: this doesn't seem right to me. but i don't have a better answer ... */ + if (gain == 0) return 0.0; + + return pow((6.0*log(gain)/log(2.0)+192.0)/198.0, 8.0); +} +#endif + +/** @param default_type The type of port that will be created by ensure_io + * and friends if no type is explicitly requested (to avoid breakage). + */ +IO::IO (Session& s, const string& name, + int input_min, int input_max, int output_min, int output_max, + DataType default_type, bool public_ports) + : Automatable (s, name), + _output_buffers (new BufferSet()), + _active(true), + _default_type (default_type), + _public_ports (public_ports), + _input_minimum (ChanCount::ZERO), + _input_maximum (ChanCount::INFINITE), + _output_minimum (ChanCount::ZERO), + _output_maximum (ChanCount::INFINITE) +{ + _panner = new Panner (name, _session); + _meter = new PeakMeter (_session); + + if (input_min > 0) { + _input_minimum = ChanCount(_default_type, input_min); + } + if (input_max >= 0) { + _input_maximum = ChanCount(_default_type, input_max); + } + if (output_min > 0) { + _output_minimum = ChanCount(_default_type, output_min); + } + if (output_max >= 0) { + _output_maximum = ChanCount(_default_type, output_max); + } + + _gain = 1.0; + _desired_gain = 1.0; + pending_state_node = 0; + no_panner_reset = false; + _phase_invert = false; + deferred_state = 0; + + boost::shared_ptr<AutomationList> gl( + new AutomationList(Parameter(GainAutomation), 0.0, 2.0, 1.0)); + + _gain_control = boost::shared_ptr<GainControl>( + new GainControl(X_("gaincontrol"), *this, gl)); + + add_control(_gain_control); + + apply_gain_automation = false; + + { + // IO::Meter is emitted from another thread so the + // Meter signal must be protected. + Glib::Mutex::Lock guard (m_meter_signal_lock); + m_meter_connection = Meter.connect (mem_fun (*this, &IO::meter)); + } + + // Connect to our own PortCountChanged signal to connect output buffers + IO::PortCountChanged.connect (mem_fun (*this, &IO::attach_buffers)); + + _session.add_controllable (_gain_control); + + create_bundles_for_inputs_and_outputs (); +} + +IO::IO (Session& s, const XMLNode& node, DataType dt) + : Automatable (s, "unnamed io"), + _output_buffers (new BufferSet()), + _active(true), + _default_type (dt) +{ + _meter = new PeakMeter (_session); + _public_ports = true; // XXX get this from node + _panner = 0; + deferred_state = 0; + no_panner_reset = false; + _desired_gain = 1.0; + _gain = 1.0; + + apply_gain_automation = false; + + boost::shared_ptr<AutomationList> gl( + new AutomationList(Parameter(GainAutomation), 0.0, 2.0, 1.0)); + + _gain_control = boost::shared_ptr<GainControl>( + new GainControl(X_("gaincontrol"), *this, gl)); + + add_control(_gain_control); + + set_state (node); + + { + // IO::Meter is emitted from another thread so the + // Meter signal must be protected. + Glib::Mutex::Lock guard (m_meter_signal_lock); + m_meter_connection = Meter.connect (mem_fun (*this, &IO::meter)); + } + + // Connect to our own PortCountChanged signal to connect output buffers + IO::PortCountChanged.connect (mem_fun (*this, &IO::attach_buffers)); + + _session.add_controllable (_gain_control); + + create_bundles_for_inputs_and_outputs (); +} + +IO::~IO () +{ + Glib::Mutex::Lock guard (m_meter_signal_lock); + Glib::Mutex::Lock lm (io_lock); + + BLOCK_PROCESS_CALLBACK (); + + for (PortSet::iterator i = _inputs.begin(); i != _inputs.end(); ++i) { + _session.engine().unregister_port (*i); + } + + for (PortSet::iterator i = _outputs.begin(); i != _outputs.end(); ++i) { + _session.engine().unregister_port (*i); + } + + m_meter_connection.disconnect(); + + delete _meter; + delete _panner; + delete _output_buffers; +} + +void +IO::silence (nframes_t nframes, nframes_t offset) +{ + /* io_lock, not taken: function must be called from Session::process() calltree */ + + for (PortSet::iterator i = _outputs.begin(); i != _outputs.end(); ++i) { + i->get_buffer().silence (nframes, offset); + } +} + +/** Deliver bufs to the IO's output ports + * + * This function should automatically do whatever it necessary to correctly deliver bufs + * to the outputs, eg applying gain or pan or whatever else needs to be done. + */ +void +IO::deliver_output (BufferSet& bufs, nframes_t start_frame, nframes_t end_frame, nframes_t nframes, nframes_t offset) +{ + // FIXME: type specific code doesn't actually need to be here, it will go away in time + + /* ********** AUDIO ********** */ + + // Apply gain if gain automation isn't playing + if ( ! apply_gain_automation) { + + gain_t dg = _gain; // desired gain + + { + Glib::Mutex::Lock dm (declick_lock, Glib::TRY_LOCK); + + if (dm.locked()) { + dg = _desired_gain; + } + + } + + if (dg != _gain || dg != 1.0) + Amp::run_in_place(bufs, nframes, _gain, dg, _phase_invert); + } + + // Use the panner to distribute audio to output port buffers + if (_panner && !_panner->empty() && !_panner->bypassed()) { + _panner->distribute (bufs, output_buffers(), start_frame, end_frame, nframes, offset); + } else { + const DataType type = DataType::AUDIO; + + // Copy any audio 1:1 to outputs + + BufferSet::iterator o = output_buffers().begin(type); + BufferSet::iterator i = bufs.begin(type); + BufferSet::iterator prev = i; + + while (i != bufs.end(type) && o != output_buffers().end (type)) { + o->read_from(*i, nframes, offset); + prev = i; + ++i; + ++o; + } + + /* extra outputs get a copy of the last buffer */ + + while (o != output_buffers().end(type)) { + o->read_from(*prev, nframes, offset); + ++o; + } + } + + /* ********** MIDI ********** */ + + // No MIDI, we're done here + if (bufs.count().n_midi() == 0) { + return; + } + + const DataType type = DataType::MIDI; + + // Copy any MIDI 1:1 to outputs + assert(bufs.count().n_midi() == output_buffers().count().n_midi()); + BufferSet::iterator o = output_buffers().begin(type); + for (BufferSet::iterator i = bufs.begin(type); i != bufs.end(type); ++i, ++o) { + o->read_from(*i, nframes, offset); + } +} + +void +IO::collect_input (BufferSet& outs, nframes_t nframes, nframes_t offset) +{ + assert(outs.available() >= n_inputs()); + + if (n_inputs() == ChanCount::ZERO) + return; + + outs.set_count(n_inputs()); + + for (DataType::iterator t = DataType::begin(); t != DataType::end(); ++t) { + + BufferSet::iterator o = outs.begin(*t); + for (PortSet::iterator i = _inputs.begin(*t); i != _inputs.end(*t); ++i, ++o) { + o->read_from(i->get_buffer(), nframes, offset); + } + + } +} + +void +IO::just_meter_input (nframes_t start_frame, nframes_t end_frame, + nframes_t nframes, nframes_t offset) +{ + BufferSet& bufs = _session.get_scratch_buffers (n_inputs()); + + collect_input (bufs, nframes, offset); + + _meter->run_in_place(bufs, start_frame, end_frame, nframes, offset); +} + + +void +IO::check_bundles_connected_to_inputs () +{ + check_bundles (_bundles_connected_to_inputs, inputs()); +} + +void +IO::check_bundles_connected_to_outputs () +{ + check_bundles (_bundles_connected_to_outputs, outputs()); +} + +void +IO::check_bundles (std::vector<UserBundleInfo>& list, const PortSet& ports) +{ + std::vector<UserBundleInfo> new_list; + + for (std::vector<UserBundleInfo>::iterator i = list.begin(); i != list.end(); ++i) { + + uint32_t const N = i->bundle->nchannels (); + + if (ports.num_ports() < N) { + continue; + } + + bool ok = true; + for (uint32_t j = 0; j < N; ++j) { + /* Every port on bundle channel j must be connected to our input j */ + PortList const pl = i->bundle->channel_ports (j); + for (uint32_t k = 0; k < pl.size(); ++k) { + if (ports.port(j)->connected_to (pl[k]) == false) { + ok = false; + break; + } + } + + if (ok == false) { + break; + } + } + + if (ok) { + new_list.push_back (*i); + } else { + i->configuration_will_change.disconnect (); + i->configuration_has_changed.disconnect (); + i->ports_will_change.disconnect (); + i->ports_have_changed.disconnect (); + } + } + + list = new_list; +} + + +int +IO::disconnect_input (Port* our_port, string other_port, void* src) +{ + if (other_port.length() == 0 || our_port == 0) { + return 0; + } + + { + BLOCK_PROCESS_CALLBACK (); + + { + Glib::Mutex::Lock lm (io_lock); + + /* check that our_port is really one of ours */ + + if ( ! _inputs.contains(our_port)) { + return -1; + } + + /* disconnect it from the source */ + + if (_session.engine().disconnect (other_port, our_port->name())) { + error << string_compose(_("IO: cannot disconnect input port %1 from %2"), our_port->name(), other_port) << endmsg; + return -1; + } + + check_bundles_connected_to_inputs (); + } + } + + input_changed (ConnectionsChanged, src); /* EMIT SIGNAL */ + _session.set_dirty (); + + return 0; +} + +int +IO::connect_input (Port* our_port, string other_port, void* src) +{ + if (other_port.length() == 0 || our_port == 0) { + return 0; + } + + { + BLOCK_PROCESS_CALLBACK (); + + { + Glib::Mutex::Lock lm (io_lock); + + /* check that our_port is really one of ours */ + + if ( ! _inputs.contains(our_port) ) { + return -1; + } + + /* connect it to the source */ + + if (_session.engine().connect (other_port, our_port->name())) { + return -1; + } + } + } + + input_changed (ConnectionsChanged, src); /* EMIT SIGNAL */ + _session.set_dirty (); + return 0; +} + +int +IO::disconnect_output (Port* our_port, string other_port, void* src) +{ + if (other_port.length() == 0 || our_port == 0) { + return 0; + } + + { + BLOCK_PROCESS_CALLBACK (); + + { + Glib::Mutex::Lock lm (io_lock); + + /* check that our_port is really one of ours */ + + if ( ! _outputs.contains(our_port) ) { + return -1; + } + + /* disconnect it from the destination */ + + if (_session.engine().disconnect (our_port->name(), other_port)) { + error << string_compose(_("IO: cannot disconnect output port %1 from %2"), our_port->name(), other_port) << endmsg; + return -1; + } + + check_bundles_connected_to_outputs (); + } + } + + output_changed (ConnectionsChanged, src); /* EMIT SIGNAL */ + _session.set_dirty (); + return 0; +} + +int +IO::connect_output (Port* our_port, string other_port, void* src) +{ + if (other_port.length() == 0 || our_port == 0) { + return 0; + } + + { + BLOCK_PROCESS_CALLBACK (); + + + { + Glib::Mutex::Lock lm (io_lock); + + /* check that our_port is really one of ours */ + + if ( ! _outputs.contains(our_port) ) { + return -1; + } + + /* connect it to the destination */ + + if (_session.engine().connect (our_port->name(), other_port)) { + return -1; + } + } + } + + output_changed (ConnectionsChanged, src); /* EMIT SIGNAL */ + _session.set_dirty (); + return 0; +} + +int +IO::set_input (Port* other_port, void* src) +{ + /* this removes all but one ports, and connects that one port + to the specified source. + */ + + if (_input_minimum.n_total() > 1) { + /* sorry, you can't do this */ + return -1; + } + + if (other_port == 0) { + if (_input_minimum == ChanCount::ZERO) { + return ensure_inputs (ChanCount::ZERO, false, true, src); + } else { + return -1; + } + } + + if (ensure_inputs (ChanCount(other_port->type(), 1), true, true, src)) { + return -1; + } + + return connect_input (_inputs.port(0), other_port->name(), src); +} + +int +IO::remove_output_port (Port* port, void* src) +{ + IOChange change (NoChange); + + { + BLOCK_PROCESS_CALLBACK (); + + + { + Glib::Mutex::Lock lm (io_lock); + + if (n_outputs() <= _output_minimum) { + /* sorry, you can't do this */ + return -1; + } + + if (_outputs.remove(port)) { + change = IOChange (change|ConfigurationChanged); + + if (port->connected()) { + change = IOChange (change|ConnectionsChanged); + } + + _session.engine().unregister_port (*port); + check_bundles_connected_to_outputs (); + + setup_peak_meters (); + reset_panner (); + } + } + + PortCountChanged (n_outputs()); /* EMIT SIGNAL */ + } + + if (change == ConnectionsChanged) { + setup_bundles_for_inputs_and_outputs (); + } + + if (change != NoChange) { + output_changed (change, src); + _session.set_dirty (); + return 0; + } + + return -1; +} + +/** Add an output port. + * + * @param destination Name of input port to connect new port to. + * @param src Source for emitted ConfigurationChanged signal. + * @param type Data type of port. Default value (NIL) will use this IO's default type. + */ +int +IO::add_output_port (string destination, void* src, DataType type) +{ + Port* our_port; + + if (type == DataType::NIL) + type = _default_type; + + { + BLOCK_PROCESS_CALLBACK (); + + + { + Glib::Mutex::Lock lm (io_lock); + + if (n_outputs() >= _output_maximum) { + return -1; + } + + /* Create a new output port */ + + string portname = build_legal_port_name (type, false); + + if ((our_port = _session.engine().register_output_port (type, portname, _public_ports)) == 0) { + error << string_compose(_("IO: cannot register output port %1"), portname) << endmsg; + return -1; + } + + _outputs.add (our_port); + setup_peak_meters (); + reset_panner (); + } + + PortCountChanged (n_outputs()); /* EMIT SIGNAL */ + } + + if (destination.length()) { + if (_session.engine().connect (our_port->name(), destination)) { + return -1; + } + } + + // pan_changed (src); /* EMIT SIGNAL */ + output_changed (ConfigurationChanged, src); /* EMIT SIGNAL */ + setup_bundles_for_inputs_and_outputs (); + _session.set_dirty (); + + return 0; +} + +int +IO::remove_input_port (Port* port, void* src) +{ + IOChange change (NoChange); + + { + BLOCK_PROCESS_CALLBACK (); + + + { + Glib::Mutex::Lock lm (io_lock); + + if (n_inputs() <= _input_minimum) { + /* sorry, you can't do this */ + return -1; + } + + if (_inputs.remove(port)) { + change = IOChange (change|ConfigurationChanged); + + if (port->connected()) { + change = IOChange (change|ConnectionsChanged); + } + + _session.engine().unregister_port (*port); + check_bundles_connected_to_inputs (); + + setup_peak_meters (); + reset_panner (); + } + } + + PortCountChanged (n_inputs ()); /* EMIT SIGNAL */ + } + + if (change == ConfigurationChanged) { + setup_bundles_for_inputs_and_outputs (); + } + + if (change != NoChange) { + input_changed (change, src); + _session.set_dirty (); + return 0; + } + + return -1; +} + + +/** Add an input port. + * + * @param type Data type of port. The appropriate port type, and @ref Port will be created. + * @param destination Name of input port to connect new port to. + * @param src Source for emitted ConfigurationChanged signal. + */ +int +IO::add_input_port (string source, void* src, DataType type) +{ + Port* our_port; + + if (type == DataType::NIL) + type = _default_type; + + { + BLOCK_PROCESS_CALLBACK (); + + { + Glib::Mutex::Lock lm (io_lock); + + if (n_inputs() >= _input_maximum) { + return -1; + } + + /* Create a new input port */ + + string portname = build_legal_port_name (type, true); + + if ((our_port = _session.engine().register_input_port (type, portname, _public_ports)) == 0) { + error << string_compose(_("IO: cannot register input port %1"), portname) << endmsg; + return -1; + } + + _inputs.add (our_port); + setup_peak_meters (); + reset_panner (); + } + + PortCountChanged (n_inputs()); /* EMIT SIGNAL */ + } + + if (source.length()) { + + if (_session.engine().connect (source, our_port->name())) { + return -1; + } + } + + // pan_changed (src); /* EMIT SIGNAL */ + input_changed (ConfigurationChanged, src); /* EMIT SIGNAL */ + setup_bundles_for_inputs_and_outputs (); + _session.set_dirty (); + + return 0; +} + +int +IO::disconnect_inputs (void* src) +{ + { + BLOCK_PROCESS_CALLBACK (); + + { + Glib::Mutex::Lock lm (io_lock); + + for (PortSet::iterator i = _inputs.begin(); i != _inputs.end(); ++i) { + _session.engine().disconnect (*i); + } + + check_bundles_connected_to_inputs (); + } + } + + input_changed (ConnectionsChanged, src); /* EMIT SIGNAL */ + + return 0; +} + +int +IO::disconnect_outputs (void* src) +{ + { + BLOCK_PROCESS_CALLBACK (); + + { + Glib::Mutex::Lock lm (io_lock); + + for (PortSet::iterator i = _outputs.begin(); i != _outputs.end(); ++i) { + _session.engine().disconnect (*i); + } + + check_bundles_connected_to_outputs (); + } + } + + output_changed (ConnectionsChanged, src); /* EMIT SIGNAL */ + _session.set_dirty (); + + return 0; +} + +bool +IO::ensure_inputs_locked (ChanCount count, bool clear, void* src) +{ + Port* input_port = 0; + bool changed = false; + + + for (DataType::iterator t = DataType::begin(); t != DataType::end(); ++t) { + + const size_t n = count.get(*t); + + /* remove unused ports */ + for (size_t i = n_inputs().get(*t); i > n; --i) { + input_port = _inputs.port(*t, i-1); + + assert(input_port); + _inputs.remove(input_port); + _session.engine().unregister_port (*input_port); + + changed = true; + } + + /* create any necessary new ports */ + while (n_inputs().get(*t) < n) { + + string portname = build_legal_port_name (*t, true); + + try { + + if ((input_port = _session.engine().register_input_port (*t, portname, _public_ports)) == 0) { + error << string_compose(_("IO: cannot register input port %1"), portname) << endmsg; + return -1; + } + } + + catch (AudioEngine::PortRegistrationFailure& err) { + setup_peak_meters (); + reset_panner (); + /* pass it on */ + throw AudioEngine::PortRegistrationFailure(); + } + + _inputs.add (input_port); + changed = true; + } + } + + if (changed) { + check_bundles_connected_to_inputs (); + setup_peak_meters (); + reset_panner (); + PortCountChanged (n_inputs()); /* EMIT SIGNAL */ + _session.set_dirty (); + } + + if (clear) { + /* disconnect all existing ports so that we get a fresh start */ + for (PortSet::iterator i = _inputs.begin(); i != _inputs.end(); ++i) { + _session.engine().disconnect (*i); + } + } + + return changed; +} + +/** Attach output_buffers to port buffers. + * + * Connected to IO's own PortCountChanged signal. + */ +void +IO::attach_buffers(ChanCount ignored) +{ + _output_buffers->attach_buffers(_outputs); +} + +int +IO::ensure_io (ChanCount in, ChanCount out, bool clear, void* src) +{ + bool in_changed = false; + bool out_changed = false; + bool need_pan_reset = false; + + in = min (_input_maximum, in); + + out = min (_output_maximum, out); + + if (in == n_inputs() && out == n_outputs() && !clear) { + return 0; + } + + { + BLOCK_PROCESS_CALLBACK (); + Glib::Mutex::Lock lm (io_lock); + + Port* port; + + if (n_outputs() != out) { + need_pan_reset = true; + } + + for (DataType::iterator t = DataType::begin(); t != DataType::end(); ++t) { + + const size_t nin = in.get(*t); + const size_t nout = out.get(*t); + + Port* output_port = 0; + Port* input_port = 0; + + /* remove unused output ports */ + for (size_t i = n_outputs().get(*t); i > nout; --i) { + output_port = _outputs.port(*t, i-1); + + assert(output_port); + _outputs.remove(output_port); + _session.engine().unregister_port (*output_port); + + out_changed = true; + } + + /* remove unused input ports */ + for (size_t i = n_inputs().get(*t); i > nin; --i) { + input_port = _inputs.port(*t, i-1); + + assert(input_port); + _inputs.remove(input_port); + _session.engine().unregister_port (*input_port); + + in_changed = true; + } + + /* create any necessary new input ports */ + + while (n_inputs().get(*t) < nin) { + + string portname = build_legal_port_name (*t, true); + + try { + if ((port = _session.engine().register_input_port (*t, portname, _public_ports)) == 0) { + error << string_compose(_("IO: cannot register input port %1"), portname) << endmsg; + return -1; + } + } + + catch (AudioEngine::PortRegistrationFailure& err) { + setup_peak_meters (); + reset_panner (); + /* pass it on */ + throw AudioEngine::PortRegistrationFailure(); + } + + _inputs.add (port); + in_changed = true; + } + + /* create any necessary new output ports */ + + while (n_outputs().get(*t) < nout) { + + string portname = build_legal_port_name (*t, false); + + try { + if ((port = _session.engine().register_output_port (*t, portname, _public_ports)) == 0) { + error << string_compose(_("IO: cannot register output port %1"), portname) << endmsg; + return -1; + } + } + + catch (AudioEngine::PortRegistrationFailure& err) { + setup_peak_meters (); + reset_panner (); + /* pass it on */ + throw AudioEngine::PortRegistrationFailure (); + } + + _outputs.add (port); + out_changed = true; + } + } + + if (clear) { + + /* disconnect all existing ports so that we get a fresh start */ + + for (PortSet::iterator i = _inputs.begin(); i != _inputs.end(); ++i) { + _session.engine().disconnect (*i); + } + + for (PortSet::iterator i = _outputs.begin(); i != _outputs.end(); ++i) { + _session.engine().disconnect (*i); + } + } + + if (in_changed || out_changed) { + setup_peak_meters (); + reset_panner (); + } + } + + if (out_changed) { + check_bundles_connected_to_outputs (); + output_changed (ConfigurationChanged, src); /* EMIT SIGNAL */ + } + + if (in_changed) { + check_bundles_connected_to_inputs (); + input_changed (ConfigurationChanged, src); /* EMIT SIGNAL */ + } + + if (in_changed || out_changed) { + PortCountChanged (max (n_outputs(), n_inputs())); /* EMIT SIGNAL */ + setup_bundles_for_inputs_and_outputs (); + _session.set_dirty (); + } + + return 0; +} + +int +IO::ensure_inputs (ChanCount count, bool clear, bool lockit, void* src) +{ + bool changed = false; + + count = min (_input_maximum, count); + + if (count == n_inputs() && !clear) { + return 0; + } + + if (lockit) { + BLOCK_PROCESS_CALLBACK (); + Glib::Mutex::Lock im (io_lock); + changed = ensure_inputs_locked (count, clear, src); + } else { + changed = ensure_inputs_locked (count, clear, src); + } + + if (changed) { + input_changed (ConfigurationChanged, src); /* EMIT SIGNAL */ + setup_bundles_for_inputs_and_outputs (); + _session.set_dirty (); + } + return 0; +} + +bool +IO::ensure_outputs_locked (ChanCount count, bool clear, void* src) +{ + Port* output_port = 0; + bool changed = false; + bool need_pan_reset = false; + + if (n_outputs() != count) { + need_pan_reset = true; + } + + for (DataType::iterator t = DataType::begin(); t != DataType::end(); ++t) { + + const size_t n = count.get(*t); + + /* remove unused ports */ + for (size_t i = n_outputs().get(*t); i > n; --i) { + output_port = _outputs.port(*t, i-1); + + assert(output_port); + _outputs.remove(output_port); + _session.engine().unregister_port (*output_port); + + changed = true; + } + + /* create any necessary new ports */ + while (n_outputs().get(*t) < n) { + + string portname = build_legal_port_name (*t, false); + + if ((output_port = _session.engine().register_output_port (*t, portname, _public_ports)) == 0) { + error << string_compose(_("IO: cannot register output port %1"), portname) << endmsg; + return -1; + } + + _outputs.add (output_port); + changed = true; + setup_peak_meters (); + + if (need_pan_reset) { + reset_panner (); + } + } + } + + if (changed) { + check_bundles_connected_to_outputs (); + PortCountChanged (n_outputs()); /* EMIT SIGNAL */ + _session.set_dirty (); + } + + if (clear) { + /* disconnect all existing ports so that we get a fresh start */ + for (PortSet::iterator i = _outputs.begin(); i != _outputs.end(); ++i) { + _session.engine().disconnect (*i); + } + } + + return changed; +} + +int +IO::ensure_outputs (ChanCount count, bool clear, bool lockit, void* src) +{ + bool changed = false; + + if (_output_maximum < ChanCount::INFINITE) { + count = min (_output_maximum, count); + if (count == n_outputs() && !clear) { + return 0; + } + } + + /* XXX caller should hold io_lock, but generally doesn't */ + + if (lockit) { + BLOCK_PROCESS_CALLBACK (); + Glib::Mutex::Lock im (io_lock); + changed = ensure_outputs_locked (count, clear, src); + } else { + changed = ensure_outputs_locked (count, clear, src); + } + + if (changed) { + output_changed (ConfigurationChanged, src); /* EMIT SIGNAL */ + setup_bundles_for_inputs_and_outputs (); + } + + return 0; +} + +gain_t +IO::effective_gain () const +{ + if (_gain_control->list()->automation_playback()) { + return _gain_control->get_value(); + } else { + return _desired_gain; + } +} + +void +IO::reset_panner () +{ + if (panners_legal) { + if (!no_panner_reset) { + _panner->reset (n_outputs().n_audio(), pans_required()); + } + } else { + panner_legal_c.disconnect (); + panner_legal_c = PannersLegal.connect (mem_fun (*this, &IO::panners_became_legal)); + } +} + +int +IO::panners_became_legal () +{ + _panner->reset (n_outputs().n_audio(), pans_required()); + _panner->load (); // automation + panner_legal_c.disconnect (); + return 0; +} + +void +IO::defer_pan_reset () +{ + no_panner_reset = true; +} + +void +IO::allow_pan_reset () +{ + no_panner_reset = false; + reset_panner (); +} + + +XMLNode& +IO::get_state (void) +{ + return state (true); +} + +XMLNode& +IO::state (bool full_state) +{ + XMLNode* node = new XMLNode (state_node_name); + char buf[64]; + string str; + vector<string>::iterator ci; + int n; + LocaleGuard lg (X_("POSIX")); + Glib::Mutex::Lock lm (io_lock); + + node->add_property("name", _name); + id().print (buf, sizeof (buf)); + node->add_property("id", buf); + + for ( + std::vector<UserBundleInfo>::iterator i = _bundles_connected_to_inputs.begin(); + i != _bundles_connected_to_inputs.end(); + ++i + ) + { + XMLNode* n = new XMLNode ("InputBundle"); + n->add_property ("name", i->bundle->name ()); + node->add_child_nocopy (*n); + } + + for ( + std::vector<UserBundleInfo>::iterator i = _bundles_connected_to_outputs.begin(); + i != _bundles_connected_to_outputs.end(); + ++i + ) + { + XMLNode* n = new XMLNode ("OutputBundle"); + n->add_property ("name", i->bundle->name ()); + node->add_child_nocopy (*n); + } + + str = ""; + + for (PortSet::iterator i = _inputs.begin(); i != _inputs.end(); ++i) { + + vector<string> connections; + + if (i->get_connections (connections)) { + + str += '{'; + + for (n = 0, ci = connections.begin(); ci != connections.end(); ++ci, ++n) { + if (n) { + str += ','; + } + + /* if its a connection to our own port, + return only the port name, not the + whole thing. this allows connections + to be re-established even when our + client name is different. + */ + + str += _session.engine().make_port_name_relative (*ci); + } + + str += '}'; + + } else { + str += "{}"; + } + } + + node->add_property ("inputs", str); + + str = ""; + + for (PortSet::iterator i = _outputs.begin(); i != _outputs.end(); ++i) { + + vector<string> connections; + + if (i->get_connections (connections)) { + + str += '{'; + + for (n = 0, ci = connections.begin(); ci != connections.end(); ++ci, ++n) { + if (n) { + str += ','; + } + + str += _session.engine().make_port_name_relative (*ci); + } + + str += '}'; + + } else { + str += "{}"; + } + } + + node->add_property ("outputs", str); + + node->add_child_nocopy (_panner->state (full_state)); + node->add_child_nocopy (_gain_control->get_state ()); + + snprintf (buf, sizeof(buf), "%2.12f", gain()); + node->add_property ("gain", buf); + + /* To make backwards compatibility a bit easier, write ChanCount::INFINITE to the session file + as -1. + */ + + int const in_max = _input_maximum == ChanCount::INFINITE ? -1 : _input_maximum.get(_default_type); + int const out_max = _output_maximum == ChanCount::INFINITE ? -1 : _output_maximum.get(_default_type); + + snprintf (buf, sizeof(buf)-1, "%d,%d,%d,%d", _input_minimum.get(_default_type), in_max, _output_minimum.get(_default_type), out_max); + + node->add_property ("iolimits", buf); + + /* automation */ + + if (full_state) + node->add_child_nocopy (get_automation_state()); + + return *node; +} + +int +IO::set_state (const XMLNode& node) +{ + const XMLProperty* prop; + XMLNodeConstIterator iter; + LocaleGuard lg (X_("POSIX")); + + /* force use of non-localized representation of decimal point, + since we use it a lot in XML files and so forth. + */ + + if (node.name() != state_node_name) { + error << string_compose(_("incorrect XML node \"%1\" passed to IO object"), node.name()) << endmsg; + return -1; + } + + if ((prop = node.property ("name")) != 0) { + _name = prop->value(); + /* used to set panner name with this, but no more */ + } + + if ((prop = node.property ("id")) != 0) { + _id = prop->value (); + } + + int in_min = -1; + int in_max = -1; + int out_min = -1; + int out_max = -1; + + if ((prop = node.property ("iolimits")) != 0) { + sscanf (prop->value().c_str(), "%d,%d,%d,%d", + &in_min, &in_max, &out_min, &out_max); + + /* Correct for the difference between the way we write things to session files and the + way things are described by ChanCount; see comments in io.h about what the different + ChanCount values mean. */ + + if (in_min < 0) { + _input_minimum = ChanCount::ZERO; + } else { + _input_minimum = ChanCount (_default_type, in_min); + } + + if (in_max < 0) { + _input_maximum = ChanCount::INFINITE; + } else { + _input_maximum = ChanCount (_default_type, in_max); + } + + if (out_min < 0) { + _output_minimum = ChanCount::ZERO; + } else { + _output_minimum = ChanCount (_default_type, out_min); + } + + if (out_max < 0) { + _output_maximum = ChanCount::INFINITE; + } else { + _output_maximum = ChanCount (_default_type, out_max); + } + } + + if ((prop = node.property ("gain")) != 0) { + set_gain (atof (prop->value().c_str()), this); + _gain = _desired_gain; + } + + if ((prop = node.property ("automation-state")) != 0 || (prop = node.property ("automation-style")) != 0) { + /* old school automation handling */ + } + + for (iter = node.children().begin(); iter != node.children().end(); ++iter) { + + if ((*iter)->name() == "Panner") { + if (_panner == 0) { + _panner = new Panner (_name, _session); + } + _panner->set_state (**iter); + } + + if ((*iter)->name() == X_("Automation")) { + + set_automation_state (*(*iter), Parameter(GainAutomation)); + } + + if ((*iter)->name() == X_("controllable")) { + if ((prop = (*iter)->property("name")) != 0 && prop->value() == "gaincontrol") { + _gain_control->set_state (**iter); + } + } + } + + if (ports_legal) { + + if (create_ports (node)) { + return -1; + } + + } else { + + port_legal_c = PortsLegal.connect (mem_fun (*this, &IO::ports_became_legal)); + } + + if (panners_legal) { + reset_panner (); + } else { + panner_legal_c = PannersLegal.connect (mem_fun (*this, &IO::panners_became_legal)); + } + + if (connecting_legal) { + + if (make_connections (node)) { + return -1; + } + + } else { + + connection_legal_c = ConnectingLegal.connect (mem_fun (*this, &IO::connecting_became_legal)); + } + + if (!ports_legal || !connecting_legal) { + pending_state_node = new XMLNode (node); + } + + return 0; +} + +int +IO::load_automation (string path) +{ + string fullpath; + ifstream in; + char line[128]; + uint32_t linecnt = 0; + float version; + LocaleGuard lg (X_("POSIX")); + + fullpath = _session.automation_dir(); + fullpath += path; + + in.open (fullpath.c_str()); + + if (!in) { + fullpath = _session.automation_dir(); + fullpath += _session.snap_name(); + fullpath += '-'; + fullpath += path; + + in.open (fullpath.c_str()); + + if (!in) { + error << string_compose(_("%1: cannot open automation event file \"%2\""), _name, fullpath) << endmsg; + return -1; + } + } + + clear_automation (); + + while (in.getline (line, sizeof(line), '\n')) { + char type; + nframes_t when; + double value; + + if (++linecnt == 1) { + if (memcmp (line, "version", 7) == 0) { + if (sscanf (line, "version %f", &version) != 1) { + error << string_compose(_("badly formed version number in automation event file \"%1\""), path) << endmsg; + return -1; + } + } else { + error << string_compose(_("no version information in automation event file \"%1\""), path) << endmsg; + return -1; + } + + continue; + } + + if (sscanf (line, "%c %" PRIu32 " %lf", &type, &when, &value) != 3) { + warning << string_compose(_("badly formatted automation event record at line %1 of %2 (ignored)"), linecnt, path) << endmsg; + continue; + } + + switch (type) { + case 'g': + _gain_control->list()->fast_simple_add (when, value); + break; + + case 's': + break; + + case 'm': + break; + + case 'p': + /* older (pre-1.0) versions of ardour used this */ + break; + + default: + warning << _("dubious automation event found (and ignored)") << endmsg; + } + } + + return 0; +} + +int +IO::connecting_became_legal () +{ + int ret; + + if (pending_state_node == 0) { + fatal << _("IO::connecting_became_legal() called without a pending state node") << endmsg; + /*NOTREACHED*/ + return -1; + } + + connection_legal_c.disconnect (); + + ret = make_connections (*pending_state_node); + + if (ports_legal) { + delete pending_state_node; + pending_state_node = 0; + } + + return ret; +} +int +IO::ports_became_legal () +{ + int ret; + + if (pending_state_node == 0) { + fatal << _("IO::ports_became_legal() called without a pending state node") << endmsg; + /*NOTREACHED*/ + return -1; + } + + port_legal_c.disconnect (); + + ret = create_ports (*pending_state_node); + + if (connecting_legal) { + delete pending_state_node; + pending_state_node = 0; + } + + return ret; +} + +int +IO::create_ports (const XMLNode& node) +{ + XMLProperty const * prop; + int num_inputs = 0; + int num_outputs = 0; + + if ((prop = node.property ("inputs")) != 0) { + num_inputs = count (prop->value().begin(), prop->value().end(), '{'); + } else if ((prop = node.property ("outputs")) != 0) { + num_outputs = count (prop->value().begin(), prop->value().end(), '{'); + } + + no_panner_reset = true; + + // FIXME: audio-only + if (ensure_io (ChanCount(DataType::AUDIO, num_inputs), ChanCount(DataType::AUDIO, num_outputs), true, this)) { + error << string_compose(_("%1: cannot create I/O ports"), _name) << endmsg; + return -1; + } + + no_panner_reset = false; + + set_deferred_state (); + + PortsCreated(); + return 0; +} + + +int +IO::make_connections (const XMLNode& node) +{ + XMLProperty const * prop; + + if ((prop = node.property ("inputs")) != 0) { + if (set_inputs (prop->value())) { + error << string_compose(_("improper input channel list in XML node (%1)"), prop->value()) << endmsg; + return -1; + } + } + + + if ((prop = node.property ("outputs")) != 0) { + if (set_outputs (prop->value())) { + error << string_compose(_("improper output channel list in XML node (%1)"), prop->value()) << endmsg; + return -1; + } + } + + for (XMLNodeConstIterator i = node.children().begin(); i != node.children().end(); ++i) { + + if ((*i)->name() == "InputBundle") { + XMLProperty const * prop = (*i)->property ("name"); + if (prop) { + boost::shared_ptr<Bundle> b = _session.bundle_by_name (prop->value()); + if (b) { + connect_input_ports_to_bundle (b, this); + } else { + error << string_compose(_("Unknown bundle \"%1\" listed for input of %2"), prop->value(), _name) << endmsg; + } + } + + } else if ((*i)->name() == "OutputBundle") { + XMLProperty const * prop = (*i)->property ("name"); + if (prop) { + boost::shared_ptr<Bundle> b = _session.bundle_by_name (prop->value()); + if (b) { + connect_output_ports_to_bundle (b, this); + } else { + error << string_compose(_("Unknown bundle \"%1\" listed for output of %2"), prop->value(), _name) << endmsg; + } + } + } + } + + return 0; +} + +int +IO::set_inputs (const string& str) +{ + vector<string> ports; + int i; + int n; + uint32_t nports; + + if ((nports = count (str.begin(), str.end(), '{')) == 0) { + return 0; + } + + // FIXME: audio-only + if (ensure_inputs (ChanCount(DataType::AUDIO, nports), true, true, this)) { + return -1; + } + + string::size_type start, end, ostart; + + ostart = 0; + start = 0; + end = 0; + i = 0; + + while ((start = str.find_first_of ('{', ostart)) != string::npos) { + start += 1; + + if ((end = str.find_first_of ('}', start)) == string::npos) { + error << string_compose(_("IO: badly formed string in XML node for inputs \"%1\""), str) << endmsg; + return -1; + } + + if ((n = parse_io_string (str.substr (start, end - start), ports)) < 0) { + error << string_compose(_("bad input string in XML node \"%1\""), str) << endmsg; + + return -1; + + } else if (n > 0) { + + for (int x = 0; x < n; ++x) { + connect_input (input (i), ports[x], this); + } + } + + ostart = end+1; + i++; + } + + return 0; +} + +int +IO::set_outputs (const string& str) +{ + vector<string> ports; + int i; + int n; + uint32_t nports; + + if ((nports = count (str.begin(), str.end(), '{')) == 0) { + return 0; + } + + // FIXME: audio-only + if (ensure_outputs (ChanCount(DataType::AUDIO, nports), true, true, this)) { + return -1; + } + + string::size_type start, end, ostart; + + ostart = 0; + start = 0; + end = 0; + i = 0; + + while ((start = str.find_first_of ('{', ostart)) != string::npos) { + start += 1; + + if ((end = str.find_first_of ('}', start)) == string::npos) { + error << string_compose(_("IO: badly formed string in XML node for outputs \"%1\""), str) << endmsg; + return -1; + } + + if ((n = parse_io_string (str.substr (start, end - start), ports)) < 0) { + error << string_compose(_("IO: bad output string in XML node \"%1\""), str) << endmsg; + + return -1; + + } else if (n > 0) { + + for (int x = 0; x < n; ++x) { + connect_output (output (i), ports[x], this); + } + } + + ostart = end+1; + i++; + } + + return 0; +} + +int +IO::parse_io_string (const string& str, vector<string>& ports) +{ + string::size_type pos, opos; + + if (str.length() == 0) { + return 0; + } + + pos = 0; + opos = 0; + + ports.clear (); + + while ((pos = str.find_first_of (',', opos)) != string::npos) { + ports.push_back (str.substr (opos, pos - opos)); + opos = pos + 1; + } + + if (opos < str.length()) { + ports.push_back (str.substr(opos)); + } + + return ports.size(); +} + +int +IO::parse_gain_string (const string& str, vector<string>& ports) +{ + string::size_type pos, opos; + + pos = 0; + opos = 0; + ports.clear (); + + while ((pos = str.find_first_of (',', opos)) != string::npos) { + ports.push_back (str.substr (opos, pos - opos)); + opos = pos + 1; + } + + if (opos < str.length()) { + ports.push_back (str.substr(opos)); + } + + return ports.size(); +} + +bool +IO::set_name (const string& requested_name) +{ + if (requested_name == _name) { + return true; + } + + string name; + Route *rt; + if ( (rt = dynamic_cast<Route *>(this))) { + name = Route::ensure_track_or_route_name(requested_name, _session); + } else { + name = requested_name; + } + + + /* replace all colons in the name. i wish we didn't have to do this */ + + if (replace_all (name, ":", "-")) { + warning << _("you cannot use colons to name objects with I/O connections") << endmsg; + } + + for (PortSet::iterator i = _inputs.begin(); i != _inputs.end(); ++i) { + string current_name = i->short_name(); + current_name.replace (current_name.find (_name), _name.length(), name); + i->set_name (current_name); + } + + for (PortSet::iterator i = _outputs.begin(); i != _outputs.end(); ++i) { + string current_name = i->short_name(); + current_name.replace (current_name.find (_name), _name.length(), name); + i->set_name (current_name); + } + + bool const r = SessionObject::set_name(name); + + setup_bundles_for_inputs_and_outputs (); + + return r; +} + +void +IO::set_input_minimum (ChanCount n) +{ + _input_minimum = n; +} + +void +IO::set_input_maximum (ChanCount n) +{ + _input_maximum = n; +} + +void +IO::set_output_minimum (ChanCount n) +{ + _output_minimum = n; +} + +void +IO::set_output_maximum (ChanCount n) +{ + _output_maximum = n; +} + +void +IO::set_port_latency (nframes_t nframes) +{ + Glib::Mutex::Lock lm (io_lock); + + for (PortSet::iterator i = _outputs.begin(); i != _outputs.end(); ++i) { + i->set_latency (nframes); + } +} + +nframes_t +IO::output_latency () const +{ + nframes_t max_latency; + nframes_t latency; + + max_latency = 0; + + /* io lock not taken - must be protected by other means */ + + for (PortSet::const_iterator i = _outputs.begin(); i != _outputs.end(); ++i) { + if ((latency = _session.engine().get_port_total_latency (*i)) > max_latency) { + max_latency = latency; + } + } + + return max_latency; +} + +nframes_t +IO::input_latency () const +{ + nframes_t max_latency; + nframes_t latency; + + max_latency = 0; + + /* io lock not taken - must be protected by other means */ + + for (PortSet::const_iterator i = _inputs.begin(); i != _inputs.end(); ++i) { + if ((latency = _session.engine().get_port_total_latency (*i)) > max_latency) { + max_latency = latency; + } + } + + return max_latency; +} + +int +IO::connect_input_ports_to_bundle (boost::shared_ptr<Bundle> c, void* src) +{ + { + BLOCK_PROCESS_CALLBACK (); + Glib::Mutex::Lock lm2 (io_lock); + + /* Connect to the bundle, not worrying about any connections + that are already made. */ + + uint32_t const channels = c->nchannels (); + + for (uint32_t n = 0; n < channels; ++n) { + const PortList& pl = c->channel_ports (n); + + for (PortList::const_iterator i = pl.begin(); i != pl.end(); ++i) { + + if (!_inputs.port(n)->connected_to (*i)) { + + if (_session.engine().connect (*i, _inputs.port(n)->name())) { + return -1; + } + } + + } + } + + /* If this is a UserBundle, make a note of what we've done */ + + boost::shared_ptr<UserBundle> ub = boost::dynamic_pointer_cast<UserBundle> (c); + if (ub) { + + /* See if we already know about this one */ + std::vector<UserBundleInfo>::iterator i = _bundles_connected_to_inputs.begin(); + while (i != _bundles_connected_to_inputs.end() && i->bundle != ub) { + ++i; + } + + if (i == _bundles_connected_to_inputs.end()) { + /* We don't, so make a note */ + _bundles_connected_to_inputs.push_back (UserBundleInfo (this, ub)); + } + } + } + + input_changed (IOChange (ConfigurationChanged|ConnectionsChanged), src); /* EMIT SIGNAL */ + return 0; +} + +int +IO::connect_output_ports_to_bundle (boost::shared_ptr<Bundle> c, void* src) +{ + { + BLOCK_PROCESS_CALLBACK (); + Glib::Mutex::Lock lm2 (io_lock); + + /* Connect to the bundle, not worrying about any connections + that are already made. */ + + uint32_t const channels = c->nchannels (); + + for (uint32_t n = 0; n < channels; ++n) { + + const PortList& pl = c->channel_ports (n); + + for (PortList::const_iterator i = pl.begin(); i != pl.end(); ++i) { + + if (!_outputs.port(n)->connected_to (*i)) { + + if (_session.engine().connect (_outputs.port(n)->name(), *i)) { + return -1; + } + } + } + } + + /* If this is a UserBundle, make a note of what we've done */ + + boost::shared_ptr<UserBundle> ub = boost::dynamic_pointer_cast<UserBundle> (c); + if (ub) { + + /* See if we already know about this one */ + std::vector<UserBundleInfo>::iterator i = _bundles_connected_to_outputs.begin(); + while (i != _bundles_connected_to_outputs.end() && i->bundle != ub) { + ++i; + } + + if (i == _bundles_connected_to_outputs.end()) { + /* We don't, so make a note */ + _bundles_connected_to_outputs.push_back (UserBundleInfo (this, ub)); + } + } + } + + output_changed (IOChange (ConnectionsChanged|ConfigurationChanged), src); /* EMIT SIGNAL */ + + return 0; +} + +int +IO::disable_connecting () +{ + connecting_legal = false; + return 0; +} + +int +IO::enable_connecting () +{ + connecting_legal = true; + return ConnectingLegal (); +} + +int +IO::disable_ports () +{ + ports_legal = false; + return 0; +} + +int +IO::enable_ports () +{ + ports_legal = true; + return PortsLegal (); +} + +int +IO::disable_panners (void) +{ + panners_legal = false; + return 0; +} + +int +IO::reset_panners () +{ + panners_legal = true; + return PannersLegal (); +} + +void +IO::bundle_configuration_will_change () +{ + //XXX +// connect_input_ports_to_bundle (_input_bundle, this); +} + +void +IO::bundle_configuration_has_changed () +{ + //XXX +// connect_input_ports_to_bundle (_input_bundle, this); +} + +void +IO::bundle_ports_will_change (int ignored) +{ +//XXX +// connect_output_ports_to_bundle (_output_bundle, this); +} + +void +IO::bundle_ports_have_changed (int ignored) +{ + //XXX +// connect_output_ports_to_bundle (_output_bundle, this); +} + +void +IO::GainControl::set_value (float val) +{ + // max gain at about +6dB (10.0 ^ ( 6 dB * 0.05)) + if (val > 1.99526231f) + val = 1.99526231f; + + _user_value = val; + _io.set_gain (val, this); + + Changed(); /* EMIT SIGNAL */ +} + +float +IO::GainControl::get_value (void) const +{ + return AutomationControl::get_value(); +} + +void +IO::setup_peak_meters() +{ + ChanCount max_streams = std::max(_inputs.count(), _outputs.count()); + _meter->configure_io(max_streams, max_streams); +} + +/** + Update the peak meters. + + The meter signal lock is taken to prevent modification of the + Meter signal while updating the meters, taking the meter signal + lock prior to taking the io_lock ensures that all IO will remain + valid while metering. +*/ +void +IO::update_meters() +{ + Glib::Mutex::Lock guard (m_meter_signal_lock); + Meter(); /* EMIT SIGNAL */ +} + +void +IO::meter () +{ + // FIXME: Ugly. Meter should manage the lock, if it's necessary + + Glib::Mutex::Lock lm (io_lock); // READER: meter thread. + _meter->meter(); +} + +void +IO::clear_automation () +{ + Automatable::clear_automation (); // clears gain automation + _panner->clear_automation (); +} + +void +IO::set_parameter_automation_state (Parameter param, AutoState state) +{ + // XXX: would be nice to get rid of this special hack + + if (param.type() == GainAutomation) { + + bool changed = false; + + { + Glib::Mutex::Lock lm (_automation_lock); + + boost::shared_ptr<AutomationList> gain_auto = _gain_control->list(); + + if (state != gain_auto->automation_state()) { + changed = true; + _last_automation_snapshot = 0; + gain_auto->set_automation_state (state); + + if (state != Off) { + // FIXME: shouldn't this use Curve? + set_gain (gain_auto->eval (_session.transport_frame()), this); + } + } + } + + if (changed) { + _session.set_dirty (); + } + + } else { + Automatable::set_parameter_automation_state(param, state); + } +} + +void +IO::inc_gain (gain_t factor, void *src) +{ + if (_desired_gain == 0.0f) + set_gain (0.000001f + (0.000001f * factor), src); + else + set_gain (_desired_gain + (_desired_gain * factor), src); +} + +void +IO::set_gain (gain_t val, void *src) +{ + // max gain at about +6dB (10.0 ^ ( 6 dB * 0.05)) + if (val > 1.99526231f) + val = 1.99526231f; + + if (src != _gain_control.get()) { + _gain_control->set_value(val); + // bit twisty, this will come back and call us again + // (this keeps control in sync with reality) + return; + } + + { + Glib::Mutex::Lock dm (declick_lock); + _desired_gain = val; + } + + if (_session.transport_stopped()) { + _gain = val; + } + + if (_session.transport_stopped() && src != 0 && src != this && _gain_control->list()->automation_write()) { + _gain_control->list()->add (_session.transport_frame(), val); + + } + + _session.set_dirty(); +} + +void +IO::start_pan_touch (uint32_t which) +{ + if (which < _panner->size()) { + (*_panner)[which]->pan_control()->list()->start_touch(); + } +} + +void +IO::end_pan_touch (uint32_t which) +{ + if (which < _panner->size()) { + (*_panner)[which]->pan_control()->list()->stop_touch(); + } + +} + +void +IO::automation_snapshot (nframes_t now, bool force) +{ + Automatable::automation_snapshot (now, force); + + if (_last_automation_snapshot > now || (now - _last_automation_snapshot) > _automation_interval) { + _panner->snapshot (now); + } + + _panner->snapshot (now); + _last_automation_snapshot = now; +} + +void +IO::transport_stopped (nframes_t frame) +{ + _gain_control->list()->reposition_for_rt_add (frame); + + if (_gain_control->list()->automation_state() != Off) { + + /* the src=0 condition is a special signal to not propagate + automation gain changes into the mix group when locating. + */ + + // FIXME: shouldn't this use Curve? + set_gain (_gain_control->list()->eval (frame), 0); + } + + _panner->transport_stopped (frame); +} + +string +IO::build_legal_port_name (DataType type, bool in) +{ + const int name_size = jack_port_name_size(); + int limit; + string suffix; + int maxports; + + if (type == DataType::AUDIO) { + suffix = _("audio"); + } else if (type == DataType::MIDI) { + suffix = _("midi"); + } else { + throw unknown_type(); + } + + if (in) { + suffix += _("_in"); + maxports = _input_maximum.get(type); + } else { + suffix += _("_out"); + maxports = _output_maximum.get(type); + } + + if (maxports == 1) { + // allow space for the slash + the suffix + limit = name_size - _session.engine().client_name().length() - (suffix.length() + 1); + char buf[name_size+1]; + snprintf (buf, name_size+1, ("%.*s/%s"), limit, _name.c_str(), suffix.c_str()); + return string (buf); + } + + // allow up to 4 digits for the output port number, plus the slash, suffix and extra space + + limit = name_size - _session.engine().client_name().length() - (suffix.length() + 5); + + char buf1[name_size+1]; + char buf2[name_size+1]; + + snprintf (buf1, name_size+1, ("%.*s/%s"), limit, _name.c_str(), suffix.c_str()); + + int port_number; + + if (in) { + port_number = find_input_port_hole (buf1); + } else { + port_number = find_output_port_hole (buf1); + } + + snprintf (buf2, name_size+1, "%s %d", buf1, port_number); + + return string (buf2); +} + +int32_t +IO::find_input_port_hole (const char* base) +{ + /* CALLER MUST HOLD IO LOCK */ + + uint32_t n; + + if (_inputs.empty()) { + return 1; + } + + /* we only allow up to 4 characters for the port number + */ + + for (n = 1; n < 9999; ++n) { + char buf[jack_port_name_size()]; + PortSet::iterator i = _inputs.begin(); + + snprintf (buf, jack_port_name_size(), _("%s %u"), base, n); + + for ( ; i != _inputs.end(); ++i) { + if (i->short_name() == buf) { + break; + } + } + + if (i == _inputs.end()) { + break; + } + } + return n; +} + +int32_t +IO::find_output_port_hole (const char* base) +{ + /* CALLER MUST HOLD IO LOCK */ + + uint32_t n; + + if (_outputs.empty()) { + return 1; + } + + /* we only allow up to 4 characters for the port number + */ + + for (n = 1; n < 9999; ++n) { + char buf[jack_port_name_size()]; + PortSet::iterator i = _outputs.begin(); + + snprintf (buf, jack_port_name_size(), _("%s %u"), base, n); + + for ( ; i != _outputs.end(); ++i) { + if (i->short_name() == buf) { + break; + } + } + + if (i == _outputs.end()) { + break; + } + } + + return n; +} + +void +IO::set_active (bool yn) +{ + _active = yn; + active_changed(); /* EMIT SIGNAL */ +} + + +AudioPort* +IO::audio_input(uint32_t n) const +{ + return dynamic_cast<AudioPort*>(input(n)); +} + +AudioPort* +IO::audio_output(uint32_t n) const +{ + return dynamic_cast<AudioPort*>(output(n)); +} + +MidiPort* +IO::midi_input(uint32_t n) const +{ + return dynamic_cast<MidiPort*>(input(n)); +} + +MidiPort* +IO::midi_output(uint32_t n) const +{ + return dynamic_cast<MidiPort*>(output(n)); +} + +void +IO::set_phase_invert (bool yn, void *src) +{ + if (_phase_invert != yn) { + _phase_invert = yn; + // phase_invert_changed (src); /* EMIT SIGNAL */ + } +} + +void +IO::set_denormal_protection (bool yn, void *src) +{ + if (_denormal_protection != yn) { + _denormal_protection = yn; + // denormal_protection_changed (src); /* EMIT SIGNAL */ + } +} + +void +IO::update_port_total_latencies () +{ + /* io_lock, not taken: function must be called from Session::process() calltree */ + + for (PortSet::iterator i = _inputs.begin(); i != _inputs.end(); ++i) { + _session.engine().update_total_latency (*i); + } + + for (PortSet::iterator i = _outputs.begin(); i != _outputs.end(); ++i) { + _session.engine().update_total_latency (*i); + } +} + + +/** + * Setup bundles that describe our inputs and outputs. + */ + +void +IO::setup_bundles_for_inputs_and_outputs () +{ + char buf[32]; + + snprintf(buf, sizeof (buf), _("%s in"), _name.c_str()); + _bundle_for_inputs->set_name (buf); + uint32_t const ni = inputs().num_ports(); + _bundle_for_inputs->set_channels (ni); + for (uint32_t i = 0; i < ni; ++i) { + _bundle_for_inputs->set_port (i, inputs().port(i)->name()); + } + + snprintf(buf, sizeof (buf), _("%s out"), _name.c_str()); + _bundle_for_outputs->set_name (buf); + uint32_t const no = outputs().num_ports(); + _bundle_for_outputs->set_channels (no); + for (uint32_t i = 0; i < no; ++i) { + _bundle_for_outputs->set_port (i, outputs().port(i)->name()); + } +} + + +/** + * Create and setup bundles that describe our inputs and outputs. + */ + +void +IO::create_bundles_for_inputs_and_outputs () +{ + _bundle_for_inputs = boost::shared_ptr<AutoBundle> (new AutoBundle (true)); + _bundle_for_outputs = boost::shared_ptr<AutoBundle> (new AutoBundle (false)); + setup_bundles_for_inputs_and_outputs (); +} + +/** Add a bundle to a list if is connected to our inputs. + * @param b Bundle to check. + * @param bundles List to add to. + */ +void +IO::maybe_add_input_bundle_to_list (boost::shared_ptr<Bundle> b, std::vector<boost::shared_ptr<Bundle> >* bundles) +{ + boost::shared_ptr<AutoBundle> ab = boost::dynamic_pointer_cast<AutoBundle, Bundle> (b); + if (ab == 0 || ab->ports_are_outputs() == false) { + return; + } + + if (ab->nchannels () != n_inputs().n_total ()) { + return; + } + + for (uint32_t i = 0; i < n_inputs().n_total (); ++i) { + + PortList const & pl = b->channel_ports (i); + + if (pl.empty()) { + return; + } + + if (!input(i)->connected_to (pl[0])) { + return; + } + } + + bundles->push_back (b); +} + +/** @return Bundles connected to our inputs */ +std::vector<boost::shared_ptr<Bundle> > +IO::bundles_connected_to_inputs () +{ + std::vector<boost::shared_ptr<Bundle> > bundles; + + /* User bundles */ + for (std::vector<UserBundleInfo>::iterator i = _bundles_connected_to_inputs.begin(); i != _bundles_connected_to_inputs.end(); ++i) { + bundles.push_back (i->bundle); + } + + /* Auto bundles */ + _session.foreach_bundle ( + sigc::bind (sigc::mem_fun (*this, &IO::maybe_add_input_bundle_to_list), &bundles) + ); + + return bundles; +} + + +/** Add a bundle to a list if is connected to our outputs. + * @param b Bundle to check. + * @param bundles List to add to. + */ +void +IO::maybe_add_output_bundle_to_list (boost::shared_ptr<Bundle> b, std::vector<boost::shared_ptr<Bundle> >* bundles) +{ + boost::shared_ptr<AutoBundle> ab = boost::dynamic_pointer_cast<AutoBundle, Bundle> (b); + if (ab == 0 || ab->ports_are_inputs() == false) { + return; + } + + if (ab->nchannels () != n_outputs().n_total ()) { + return; + } + + for (uint32_t i = 0; i < n_outputs().n_total (); ++i) { + + PortList const & pl = b->channel_ports (i); + + if (pl.empty()) { + return; + } + + if (!output(i)->connected_to (pl[0])) { + return; + } + } + + bundles->push_back (b); +} + + +/* @return Bundles connected to our outputs */ +std::vector<boost::shared_ptr<Bundle> > +IO::bundles_connected_to_outputs () +{ + std::vector<boost::shared_ptr<Bundle> > bundles; + + /* User bundles */ + for (std::vector<UserBundleInfo>::iterator i = _bundles_connected_to_outputs.begin(); i != _bundles_connected_to_outputs.end(); ++i) { + bundles.push_back (i->bundle); + } + + /* Auto bundles */ + _session.foreach_bundle ( + sigc::bind (sigc::mem_fun (*this, &IO::maybe_add_output_bundle_to_list), &bundles) + ); + + return bundles; +} + + +IO::UserBundleInfo::UserBundleInfo (IO* io, boost::shared_ptr<UserBundle> b) +{ + bundle = b; + configuration_will_change = b->ConfigurationWillChange.connect ( + sigc::mem_fun (*io, &IO::bundle_configuration_will_change) + ); + configuration_has_changed = b->ConfigurationHasChanged.connect ( + sigc::mem_fun (*io, &IO::bundle_configuration_has_changed) + ); + ports_will_change = b->PortsWillChange.connect ( + sigc::mem_fun (*io, &IO::bundle_ports_will_change) + ); + ports_have_changed = b->PortsHaveChanged.connect ( + sigc::mem_fun (*io, &IO::bundle_ports_have_changed) + ); +} + |