summaryrefslogtreecommitdiff
path: root/libs/ardour/io.cc
diff options
context:
space:
mode:
Diffstat (limited to 'libs/ardour/io.cc')
-rw-r--r--libs/ardour/io.cc2821
1 files changed, 2821 insertions, 0 deletions
diff --git a/libs/ardour/io.cc b/libs/ardour/io.cc
new file mode 100644
index 0000000000..efee6fc397
--- /dev/null
+++ b/libs/ardour/io.cc
@@ -0,0 +1,2821 @@
+/*
+ Copyright (C) 2000 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.
+
+ $Id$
+*/
+
+#include <fstream>
+#include <algorithm>
+#include <unistd.h>
+#include <locale.h>
+
+#include <sigc++/bind.h>
+
+#include <pbd/lockmonitor.h>
+#include <pbd/xml++.h>
+
+#include <ardour/audioengine.h>
+#include <ardour/io.h>
+#include <ardour/port.h>
+#include <ardour/connection.h>
+#include <ardour/session.h>
+#include <ardour/cycle_timer.h>
+#include <ardour/panner.h>
+#include <ardour/dB.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
+
+
+using namespace std;
+using namespace ARDOUR;
+//using namespace sigc;
+
+static float current_automation_version_number = 1.0;
+
+jack_nframes_t IO::_automation_interval = 0;
+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::GrabPeakPower;
+sigc::signal<int> IO::ConnectingLegal;
+sigc::signal<int> IO::PortsLegal;
+sigc::signal<int> IO::PannersLegal;
+sigc::signal<void,uint32_t> IO::MoreOutputs;
+sigc::signal<int> IO::PortsCreated;
+
+/* this is a default mapper of MIDI control values to a gain coefficient.
+ others can be imagined. see IO::set_midi_to_gain_function().
+*/
+
+static gain_t direct_midi_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_midi (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);
+}
+
+static bool sort_ports_by_name (Port* a, Port* b)
+{
+ return a->name() < b->name();
+}
+
+
+IO::IO (Session& s, string name,
+
+ int input_min, int input_max, int output_min, int output_max)
+ : _session (s),
+ _name (name),
+ _midi_gain_control (*this, _session.midi_port()),
+ _gain_automation_curve (0.0, 2.0, 1.0),
+ _input_minimum (input_min),
+ _input_maximum (input_max),
+ _output_minimum (output_min),
+ _output_maximum (output_max)
+{
+ _id = new_id();
+ _panner = new Panner (name, _session);
+ _gain = 1.0;
+ _desired_gain = 1.0;
+ _input_connection = 0;
+ _output_connection = 0;
+ pending_state_node = 0;
+ _ninputs = 0;
+ _noutputs = 0;
+ no_panner_reset = false;
+ deferred_state = 0;
+
+ _midi_gain_control.midi_to_gain = direct_midi_to_gain;
+ _midi_gain_control.gain_to_midi = direct_gain_to_midi;
+
+ apply_gain_automation = false;
+
+ last_automation_snapshot = 0;
+
+ _gain_automation_state = Off;
+ _gain_automation_style = Absolute;
+
+ GrabPeakPower.connect (mem_fun (*this, &IO::grab_peak_power));
+}
+
+IO::~IO ()
+{
+ LockMonitor lm (io_lock, __LINE__, __FILE__);
+ vector<Port *>::iterator i;
+
+ for (i = _inputs.begin(); i != _inputs.end(); ++i) {
+ _session.engine().unregister_port (*i);
+ }
+
+ for (i = _outputs.begin(); i != _outputs.end(); ++i) {
+ _session.engine().unregister_port (*i);
+ }
+}
+
+void
+IO::silence (jack_nframes_t nframes, jack_nframes_t offset)
+{
+ /* io_lock, not taken: function must be called from Session::process() calltree */
+
+ for (vector<Port *>::iterator i = _outputs.begin(); i != _outputs.end(); ++i) {
+ (*i)->silence (nframes, offset);
+ }
+}
+
+void
+IO::apply_declick (vector<Sample *>& bufs, uint32_t nbufs, jack_nframes_t nframes, gain_t initial, gain_t target, bool invert_polarity)
+{
+ jack_nframes_t declick = min ((jack_nframes_t)4096, nframes);
+ gain_t delta;
+ Sample *buffer;
+ double fractional_shift;
+ double fractional_pos;
+
+ fractional_shift = -1.0/declick;
+
+ if (target < initial) {
+ /* fade out: remove more and more of delta from initial */
+ delta = -(initial - target);
+ } else {
+ /* fade in: add more and more of delta from initial */
+ delta = target - initial;
+ }
+
+ for (uint32_t n = 0; n < nbufs; ++n) {
+
+ buffer = bufs[n];
+ fractional_pos = 1.0;
+
+ if (invert_polarity) {
+ for (jack_nframes_t nx = 0; nx < declick; ++nx) {
+ buffer[nx] *= -(initial + (delta * (0.5 + 0.5 * cos (M_PI * fractional_pos))));
+ fractional_pos += fractional_shift;
+ }
+ } else {
+ for (jack_nframes_t nx = 0; nx < declick; ++nx) {
+ buffer[nx] *= (initial + (delta * (0.5 + 0.5 * cos (M_PI * fractional_pos))));
+ fractional_pos += fractional_shift;
+ }
+ }
+
+ /* now ensure the rest of the buffer has the target value
+ applied, if necessary.
+ */
+
+ if (declick != nframes) {
+
+ if (invert_polarity) {
+ target = -target;
+ }
+
+ if (target == 0.0) {
+ memset (&buffer[declick], 0, sizeof (Sample) * (nframes - declick));
+ } else if (target != 1.0) {
+ for (jack_nframes_t nx = declick; nx < nframes; ++nx) {
+ buffer[nx] *= target;
+ }
+ }
+ }
+ }
+}
+
+void
+IO::pan_automated (vector<Sample*>& bufs, uint32_t nbufs, jack_nframes_t start, jack_nframes_t end, jack_nframes_t nframes, jack_nframes_t offset)
+{
+ Sample* dst;
+
+ /* io_lock, not taken: function must be called from Session::process() calltree */
+
+ if (_noutputs == 0) {
+ return;
+ }
+
+ if (_noutputs == 1) {
+
+ dst = output(0)->get_buffer (nframes) + offset;
+
+ for (uint32_t n = 0; n < nbufs; ++n) {
+ if (bufs[n] != dst) {
+ memcpy (dst, bufs[n], sizeof (Sample) * nframes);
+ }
+ }
+
+ output(0)->mark_silence (false);
+
+ return;
+ }
+
+ uint32_t o;
+ vector<Port *>::iterator out;
+ vector<Sample *>::iterator in;
+ Panner::iterator pan;
+ Sample* obufs[_noutputs];
+
+ /* the terrible silence ... */
+
+ for (out = _outputs.begin(), o = 0; out != _outputs.end(); ++out, ++o) {
+ obufs[o] = (*out)->get_buffer (nframes) + offset;
+ memset (obufs[o], 0, sizeof (Sample) * nframes);
+ (*out)->mark_silence (false);
+ }
+
+ uint32_t n;
+
+ for (pan = _panner->begin(), n = 0; n < nbufs; ++n, ++pan) {
+ (*pan)->distribute_automated (bufs[n], obufs, start, end, nframes, _session.pan_automation_buffer());
+ }
+}
+
+void
+IO::pan (vector<Sample*>& bufs, uint32_t nbufs, jack_nframes_t nframes, jack_nframes_t offset, gain_t gain_coeff)
+{
+ Sample* dst;
+ Sample* src;
+
+ /* io_lock, not taken: function must be called from Session::process() calltree */
+
+ if (_noutputs == 0) {
+ return;
+ }
+
+ /* the panner can be empty if there are no inputs to the
+ route, but still outputs
+ */
+
+ if (_panner->bypassed() || _panner->empty()) {
+ deliver_output_no_pan (bufs, nbufs, nframes, offset);
+ return;
+ }
+
+ if (_noutputs == 1) {
+
+ dst = output(0)->get_buffer (nframes) + offset;
+
+ if (gain_coeff == 0.0f) {
+
+ /* only one output, and gain was zero, so make it silent */
+
+ memset (dst, 0, sizeof (Sample) * nframes);
+
+ } else if (gain_coeff == 1.0f){
+
+ /* mix all buffers into the output */
+
+ uint32_t n;
+
+ memcpy (dst, bufs[0], sizeof (Sample) * nframes);
+
+ for (n = 1; n < nbufs; ++n) {
+ src = bufs[n];
+
+ for (jack_nframes_t n = 0; n < nframes; ++n) {
+ dst[n] += src[n];
+ }
+ }
+
+ output(0)->mark_silence (false);
+
+ } else {
+
+ /* mix all buffers into the output, scaling them all by the gain */
+
+ uint32_t n;
+
+ src = bufs[0];
+
+ for (jack_nframes_t n = 0; n < nframes; ++n) {
+ dst[n] = src[n] * gain_coeff;
+ }
+
+ for (n = 1; n < nbufs; ++n) {
+ src = bufs[n];
+
+ for (jack_nframes_t n = 0; n < nframes; ++n) {
+ dst[n] += src[n] * gain_coeff;
+ }
+ }
+
+ output(0)->mark_silence (false);
+ }
+
+ return;
+ }
+
+ uint32_t o;
+ vector<Port *>::iterator out;
+ vector<Sample *>::iterator in;
+ Panner::iterator pan;
+ Sample* obufs[_noutputs];
+
+ /* the terrible silence ... */
+
+ /* XXX this is wasteful but i see no way to avoid it */
+
+ for (out = _outputs.begin(), o = 0; out != _outputs.end(); ++out, ++o) {
+ obufs[o] = (*out)->get_buffer (nframes) + offset;
+ memset (obufs[o], 0, sizeof (Sample) * nframes);
+ (*out)->mark_silence (false);
+ }
+
+ uint32_t n;
+
+ for (pan = _panner->begin(), n = 0; n < nbufs; ++n) {
+ Panner::iterator tmp;
+
+ tmp = pan;
+ ++tmp;
+
+ (*pan)->distribute (bufs[n], obufs, gain_coeff, nframes);
+
+ if (tmp != _panner->end()) {
+ pan = tmp;
+ }
+ }
+}
+
+void
+IO::deliver_output (vector<Sample *>& bufs, uint32_t nbufs, jack_nframes_t nframes, jack_nframes_t offset)
+{
+ /* io_lock, not taken: function must be called from Session::process() calltree */
+
+ if (_noutputs == 0) {
+ return;
+ }
+
+ if (_panner->bypassed()) {
+ deliver_output_no_pan (bufs, nbufs, nframes, offset);
+ return;
+ }
+
+
+ gain_t dg;
+
+ {
+ TentativeLockMonitor dm (declick_lock, __LINE__, __FILE__);
+
+ if (dm.locked()) {
+ dg = _desired_gain;
+ } else {
+ dg = _gain;
+ }
+ }
+
+ if (dg != _gain) {
+ apply_declick (bufs, nbufs, nframes, _gain, dg, false);
+ _gain = dg;
+ }
+
+ /* simple, non-automation panning to outputs */
+
+ if (_session.transport_speed() > 1.5f || _session.transport_speed() < -1.5f) {
+ pan (bufs, nbufs, nframes, offset, _gain * speed_quietning);
+ } else {
+ pan (bufs, nbufs, nframes, offset, _gain);
+ }
+}
+
+void
+IO::deliver_output_no_pan (vector<Sample *>& bufs, uint32_t nbufs, jack_nframes_t nframes, jack_nframes_t offset)
+{
+ /* io_lock, not taken: function must be called from Session::process() calltree */
+
+ if (_noutputs == 0) {
+ return;
+ }
+
+ gain_t dg;
+ gain_t old_gain;
+
+ if (apply_gain_automation) {
+
+ /* gain has already been applied by automation code. do nothing here except
+ speed quietning.
+ */
+
+ old_gain = _gain;
+ _gain = 1.0f;
+ dg = _gain;
+
+ } else {
+
+ TentativeLockMonitor dm (declick_lock, __LINE__, __FILE__);
+
+ if (dm.locked()) {
+ dg = _desired_gain;
+ } else {
+ dg = _gain;
+ }
+ }
+
+ Sample* src;
+ Sample* dst;
+ uint32_t i;
+ vector<Port*>::iterator o;
+ vector<Sample*> outs;
+ gain_t actual_gain;
+
+ if (dg != _gain) {
+ /* unlikely condition */
+ for (o = _outputs.begin(), i = 0; o != _outputs.end(); ++o, ++i) {
+ outs.push_back ((*o)->get_buffer (nframes) + offset);
+ }
+ }
+
+ /* reduce nbufs to the index of the last input buffer */
+
+ nbufs--;
+
+ if (_session.transport_speed() > 1.5f || _session.transport_speed() < -1.5f) {
+ actual_gain = _gain * speed_quietning;
+ } else {
+ actual_gain = _gain;
+ }
+
+ for (o = _outputs.begin(), i = 0; o != _outputs.end(); ++o, ++i) {
+
+ dst = (*o)->get_buffer (nframes) + offset;
+ src = bufs[min(nbufs,i)];
+
+ if (dg != _gain || actual_gain == 1.0f) {
+ memcpy (dst, src, sizeof (Sample) * nframes);
+ } else if (actual_gain == 0.0f) {
+ memset (dst, 0, sizeof (Sample) * nframes);
+ } else {
+ for (jack_nframes_t x = 0; x < nframes; ++x) {
+ dst[x] = src[x] * actual_gain;
+ }
+ }
+
+ (*o)->mark_silence (false);
+ }
+
+ if (dg != _gain) {
+ apply_declick (outs, outs.size(), nframes, _gain, dg, false);
+ _gain = dg;
+ }
+
+ if (apply_gain_automation) {
+ _gain = old_gain;
+ }
+}
+
+void
+IO::collect_input (vector<Sample *>& bufs, uint32_t nbufs, jack_nframes_t nframes, jack_nframes_t offset)
+{
+ /* io_lock, not taken: function must be called from Session::process() calltree */
+
+ vector<Port *>::iterator i;
+ uint32_t n;
+ Sample *last = 0;
+
+ /* we require that bufs.size() >= 1 */
+
+ for (n = 0, i = _inputs.begin(); n < nbufs; ++i, ++n) {
+ if (i == _inputs.end()) {
+ break;
+ }
+
+ /* XXX always read the full extent of the port buffer that
+ we need. One day, we may use jack_port_get_buffer_at_offset()
+ or something similar. For now, this simple hack will
+ have to do.
+
+ Hack? Why yes .. we only need to read nframes-worth of
+ data, but the data we want is at `offset' within the
+ buffer.
+ */
+
+ last = (*i)->get_buffer (nframes+offset) + offset;
+ // the dest buffer's offset has already been applied
+ memcpy (bufs[n], last, sizeof (Sample) * nframes);
+ }
+
+ /* fill any excess outputs with the last input */
+
+ while (n < nbufs && last) {
+ // the dest buffer's offset has already been applied
+ memcpy (bufs[n], last, sizeof (Sample) * nframes);
+ ++n;
+ }
+}
+
+void
+IO::just_meter_input (jack_nframes_t start_frame, jack_nframes_t end_frame,
+ jack_nframes_t nframes, jack_nframes_t offset)
+{
+ vector<Sample*>& bufs = _session.get_passthru_buffers ();
+ uint32_t nbufs = n_process_buffers ();
+
+ collect_input (bufs, nbufs, nframes, offset);
+
+ for (uint32_t n = 0; n < nbufs; ++n) {
+ _peak_power[n] = Session::compute_peak (bufs[n], nframes, _peak_power[n]);
+ }
+}
+
+void
+IO::drop_input_connection ()
+{
+ _input_connection = 0;
+ input_connection_configuration_connection.disconnect();
+ input_connection_connection_connection.disconnect();
+ _session.set_dirty ();
+}
+
+void
+IO::drop_output_connection ()
+{
+ _output_connection = 0;
+ output_connection_configuration_connection.disconnect();
+ output_connection_connection_connection.disconnect();
+ _session.set_dirty ();
+}
+
+int
+IO::disconnect_input (Port* our_port, string other_port, void* src)
+{
+ if (other_port.length() == 0 || our_port == 0) {
+ return 0;
+ }
+
+ {
+ LockMonitor em (_session.engine().process_lock(), __LINE__, __FILE__);
+
+ {
+ LockMonitor lm (io_lock, __LINE__, __FILE__);
+
+ /* check that our_port is really one of ours */
+
+ if (find (_inputs.begin(), _inputs.end(), our_port) == _inputs.end()) {
+ return -1;
+ }
+
+ /* disconnect it from the source */
+
+ if (_session.engine().disconnect (other_port, our_port->name())) {
+ error << compose(_("IO: cannot disconnect input port %1 from %2"), our_port->name(), other_port) << endmsg;
+ return -1;
+ }
+
+ drop_input_connection();
+ }
+ }
+
+ 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;
+ }
+
+ {
+ LockMonitor em(_session.engine().process_lock(), __LINE__, __FILE__);
+
+ {
+ LockMonitor lm (io_lock, __LINE__, __FILE__);
+
+ /* check that our_port is really one of ours */
+
+ if (find (_inputs.begin(), _inputs.end(), our_port) == _inputs.end()) {
+ return -1;
+ }
+
+ /* connect it to the source */
+
+ if (_session.engine().connect (other_port, our_port->name())) {
+ return -1;
+ }
+
+ drop_input_connection ();
+ }
+ }
+
+ 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;
+ }
+
+ {
+ LockMonitor em(_session.engine().process_lock(), __LINE__, __FILE__);
+
+ {
+ LockMonitor lm (io_lock, __LINE__, __FILE__);
+
+ if (find (_outputs.begin(), _outputs.end(), our_port) == _outputs.end()) {
+ return -1;
+ }
+
+ /* disconnect it from the destination */
+
+ if (_session.engine().disconnect (our_port->name(), other_port)) {
+ error << compose(_("IO: cannot disconnect output port %1 from %2"), our_port->name(), other_port) << endmsg;
+ return -1;
+ }
+
+ drop_output_connection ();
+ }
+ }
+
+ 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;
+ }
+
+ {
+ LockMonitor em(_session.engine().process_lock(), __LINE__, __FILE__);
+
+ {
+ LockMonitor lm (io_lock, __LINE__, __FILE__);
+
+ /* check that our_port is really one of ours */
+
+ if (find (_outputs.begin(), _outputs.end(), our_port) == _outputs.end()) {
+ return -1;
+ }
+
+ /* connect it to the destination */
+
+ if (_session.engine().connect (our_port->name(), other_port)) {
+ return -1;
+ }
+
+ drop_output_connection ();
+ }
+ }
+
+ 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 > 1 || _input_minimum == 0) {
+ /* sorry, you can't do this */
+ return -1;
+ }
+
+ if (other_port == 0) {
+ if (_input_minimum < 0) {
+ return ensure_inputs (0, false, true, src);
+ } else {
+ return -1;
+ }
+ }
+
+ if (ensure_inputs (1, true, true, src)) {
+ return -1;
+ }
+
+ return connect_input (_inputs.front(), other_port->name(), src);
+}
+
+int
+IO::remove_output_port (Port* port, void* src)
+{
+ IOChange change (NoChange);
+
+ {
+ LockMonitor em(_session.engine().process_lock(), __LINE__, __FILE__);
+
+ {
+ LockMonitor lm (io_lock, __LINE__, __FILE__);
+
+ if (_noutputs - 1 == (uint32_t) _output_minimum) {
+ /* sorry, you can't do this */
+ return -1;
+ }
+
+ for (vector<Port *>::iterator i = _outputs.begin(); i != _outputs.end(); ++i) {
+ if (*i == port) {
+ change = IOChange (change|ConfigurationChanged);
+ if (port->connected()) {
+ change = IOChange (change|ConnectionsChanged);
+ }
+
+ _session.engine().unregister_port (*i);
+ _outputs.erase (i);
+ _noutputs--;
+ drop_output_connection ();
+
+ break;
+ }
+ }
+
+ if (change != NoChange) {
+ setup_peak_meters ();
+ reset_panner ();
+ }
+ }
+ }
+
+ if (change != NoChange) {
+ output_changed (change, src); /* EMIT SIGNAL */
+ _session.set_dirty ();
+ return 0;
+ }
+
+ return -1;
+}
+
+int
+IO::add_output_port (string destination, void* src)
+{
+ Port* our_port;
+ char buf[64];
+
+ {
+ LockMonitor em(_session.engine().process_lock(), __LINE__, __FILE__);
+
+ {
+ LockMonitor lm (io_lock, __LINE__, __FILE__);
+
+ if (_output_maximum >= 0 && (int) _noutputs == _output_maximum) {
+ return -1;
+ }
+
+ /* Create a new output port */
+
+ if (_output_maximum == 1) {
+ snprintf (buf, sizeof (buf), _("%s/out"), _name.c_str());
+ } else {
+ snprintf (buf, sizeof (buf), _("%s/out %u"), _name.c_str(), find_output_port_hole());
+ }
+
+ if ((our_port = _session.engine().register_audio_output_port (buf)) == 0) {
+ error << compose(_("IO: cannot register output port %1"), buf) << endmsg;
+ return -1;
+ }
+
+ _outputs.push_back (our_port);
+ sort (_outputs.begin(), _outputs.end(), sort_ports_by_name);
+ ++_noutputs;
+ drop_output_connection ();
+ setup_peak_meters ();
+ reset_panner ();
+ }
+
+ MoreOutputs (_noutputs); /* 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 */
+ _session.set_dirty ();
+ return 0;
+}
+
+int
+IO::remove_input_port (Port* port, void* src)
+{
+ IOChange change (NoChange);
+
+ {
+ LockMonitor em(_session.engine().process_lock(), __LINE__, __FILE__);
+
+ {
+ LockMonitor lm (io_lock, __LINE__, __FILE__);
+
+ if (((int)_ninputs - 1) < _input_minimum) {
+ /* sorry, you can't do this */
+ return -1;
+ }
+ for (vector<Port *>::iterator i = _inputs.begin(); i != _inputs.end(); ++i) {
+
+ if (*i == port) {
+ change = IOChange (change|ConfigurationChanged);
+
+ if (port->connected()) {
+ change = IOChange (change|ConnectionsChanged);
+ }
+
+ _session.engine().unregister_port (*i);
+ _inputs.erase (i);
+ _ninputs--;
+ drop_input_connection ();
+
+ break;
+ }
+ }
+
+ if (change != NoChange) {
+ setup_peak_meters ();
+ reset_panner ();
+ }
+ }
+ }
+
+ if (change != NoChange) {
+ input_changed (change, src);
+ _session.set_dirty ();
+ return 0;
+ }
+
+ return -1;
+}
+
+int
+IO::add_input_port (string source, void* src)
+{
+ Port* our_port;
+ char buf[64];
+
+ {
+ LockMonitor em (_session.engine().process_lock(), __LINE__, __FILE__);
+
+ {
+ LockMonitor lm (io_lock, __LINE__, __FILE__);
+
+ if (_input_maximum >= 0 && (int) _ninputs == _input_maximum) {
+ return -1;
+ }
+
+ /* Create a new input port */
+
+ if (_input_maximum == 1) {
+ snprintf (buf, sizeof (buf), _("%s/in"), _name.c_str());
+ } else {
+ snprintf (buf, sizeof (buf), _("%s/in %u"), _name.c_str(), find_input_port_hole());
+ }
+
+ if ((our_port = _session.engine().register_audio_input_port (buf)) == 0) {
+ error << compose(_("IO: cannot register input port %1"), buf) << endmsg;
+ return -1;
+ }
+
+ _inputs.push_back (our_port);
+ sort (_inputs.begin(), _inputs.end(), sort_ports_by_name);
+ ++_ninputs;
+ drop_input_connection ();
+ setup_peak_meters ();
+ reset_panner ();
+ }
+
+ MoreOutputs (_ninputs); /* 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 */
+ _session.set_dirty ();
+
+ return 0;
+}
+
+int
+IO::disconnect_inputs (void* src)
+{
+ {
+ LockMonitor em (_session.engine().process_lock(), __LINE__, __FILE__);
+
+ {
+ LockMonitor lm (io_lock, __LINE__, __FILE__);
+
+ for (vector<Port *>::iterator i = _inputs.begin(); i != _inputs.end(); ++i) {
+ _session.engine().disconnect (*i);
+ }
+
+ drop_input_connection ();
+ }
+ }
+ input_changed (ConnectionsChanged, src); /* EMIT SIGNAL */
+ return 0;
+}
+
+int
+IO::disconnect_outputs (void* src)
+{
+ {
+ LockMonitor em (_session.engine().process_lock(), __LINE__, __FILE__);
+
+ {
+ LockMonitor lm (io_lock, __LINE__, __FILE__);
+
+ for (vector<Port *>::iterator i = _outputs.begin(); i != _outputs.end(); ++i) {
+ _session.engine().disconnect (*i);
+ }
+
+ drop_output_connection ();
+ }
+ }
+
+ output_changed (ConnectionsChanged, src); /* EMIT SIGNAL */
+ _session.set_dirty ();
+ return 0;
+}
+
+bool
+IO::ensure_inputs_locked (uint32_t n, bool clear, void* src)
+{
+ Port* input_port;
+ bool changed = false;
+ bool reduced = false;
+
+ /* remove unused ports */
+
+ while (_ninputs > n) {
+ _session.engine().unregister_port (_inputs.back());
+ _inputs.pop_back();
+ _ninputs--;
+ reduced = true;
+ changed = true;
+ }
+
+ /* create any necessary new ports */
+
+ while (_ninputs < n) {
+
+ char buf[64];
+
+ /* Create a new input port */
+
+ if (_input_maximum == 1) {
+ snprintf (buf, sizeof (buf), _("%s/in"), _name.c_str());
+ }
+ else {
+ snprintf (buf, sizeof (buf), _("%s/in %u"), _name.c_str(), find_input_port_hole());
+ }
+
+ try {
+
+ if ((input_port = _session.engine().register_audio_input_port (buf)) == 0) {
+ error << compose(_("IO: cannot register input port %1"), buf) << endmsg;
+ return -1;
+ }
+ }
+
+ catch (AudioEngine::PortRegistrationFailure& err) {
+ setup_peak_meters ();
+ reset_panner ();
+ /* pass it on */
+ throw err;
+ }
+
+ _inputs.push_back (input_port);
+ sort (_inputs.begin(), _inputs.end(), sort_ports_by_name);
+ ++_ninputs;
+ changed = true;
+ }
+
+ if (changed) {
+ drop_input_connection ();
+ setup_peak_meters ();
+ reset_panner ();
+ MoreOutputs (_ninputs); /* EMIT SIGNAL */
+ _session.set_dirty ();
+ }
+
+ if (clear) {
+ /* disconnect all existing ports so that we get a fresh start */
+
+ for (vector<Port *>::iterator i = _inputs.begin(); i != _inputs.end(); ++i) {
+ _session.engine().disconnect (*i);
+ }
+ }
+
+ return changed;
+}
+
+int
+IO::ensure_io (uint32_t nin, uint32_t nout, bool clear, void* src)
+{
+ bool in_changed = false;
+ bool out_changed = false;
+ bool in_reduced = false;
+ bool out_reduced = false;
+ bool need_pan_reset;
+
+ if (_input_maximum >= 0) {
+ nin = min (_input_maximum, (int) nin);
+ }
+
+ if (_output_maximum >= 0) {
+ nout = min (_output_maximum, (int) nout);
+ }
+
+ if (nin == _ninputs && nout == _noutputs && !clear) {
+ return 0;
+ }
+
+ {
+ LockMonitor em (_session.engine().process_lock(), __LINE__, __FILE__);
+ LockMonitor lm (io_lock, __LINE__, __FILE__);
+
+ Port* port;
+
+ if (_noutputs == nout) {
+ need_pan_reset = false;
+ } else {
+ need_pan_reset = true;
+ }
+
+ /* remove unused ports */
+
+ while (_ninputs > nin) {
+ _session.engine().unregister_port (_inputs.back());
+ _inputs.pop_back();
+ _ninputs--;
+ in_reduced = true;
+ in_changed = true;
+ }
+
+ while (_noutputs > nout) {
+ _session.engine().unregister_port (_outputs.back());
+ _outputs.pop_back();
+ _noutputs--;
+ out_reduced = true;
+ out_changed = true;
+ }
+
+ /* create any necessary new ports */
+
+ while (_ninputs < nin) {
+
+ char buf[64];
+
+ /* Create a new input port */
+
+ if (_input_maximum == 1) {
+ snprintf (buf, sizeof (buf), _("%s/in"), _name.c_str());
+ }
+ else {
+ snprintf (buf, sizeof (buf), _("%s/in %u"), _name.c_str(), find_input_port_hole());
+ }
+
+ try {
+ if ((port = _session.engine().register_audio_input_port (buf)) == 0) {
+ error << compose(_("IO: cannot register input port %1"), buf) << endmsg;
+ return -1;
+ }
+ }
+
+ catch (AudioEngine::PortRegistrationFailure& err) {
+ setup_peak_meters ();
+ reset_panner ();
+ /* pass it on */
+ throw err;
+ }
+
+ _inputs.push_back (port);
+ ++_ninputs;
+ in_changed = true;
+ }
+
+ /* create any necessary new ports */
+
+ while (_noutputs < nout) {
+
+ char buf[64];
+
+ /* Create a new output port */
+
+ if (_output_maximum == 1) {
+ snprintf (buf, sizeof (buf), _("%s/out"), _name.c_str());
+ } else {
+ snprintf (buf, sizeof (buf), _("%s/out %u"), _name.c_str(), find_output_port_hole());
+ }
+
+ try {
+ if ((port = _session.engine().register_audio_output_port (buf)) == 0) {
+ error << compose(_("IO: cannot register output port %1"), buf) << endmsg;
+ return -1;
+ }
+ }
+
+ catch (AudioEngine::PortRegistrationFailure& err) {
+ setup_peak_meters ();
+ reset_panner ();
+ /* pass it on */
+ throw err;
+ }
+
+ _outputs.push_back (port);
+ ++_noutputs;
+ out_changed = true;
+ }
+
+ if (clear) {
+
+ /* disconnect all existing ports so that we get a fresh start */
+
+ for (vector<Port *>::iterator i = _inputs.begin(); i != _inputs.end(); ++i) {
+ _session.engine().disconnect (*i);
+ }
+
+ for (vector<Port *>::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) {
+ sort (_outputs.begin(), _outputs.end(), sort_ports_by_name);
+ drop_output_connection ();
+ output_changed (ConfigurationChanged, src); /* EMIT SIGNAL */
+ }
+
+ if (in_changed) {
+ sort (_inputs.begin(), _inputs.end(), sort_ports_by_name);
+ drop_input_connection ();
+ input_changed (ConfigurationChanged, src); /* EMIT SIGNAL */
+ }
+
+ if (in_changed || out_changed) {
+ MoreOutputs (max (_noutputs, _ninputs)); /* EMIT SIGNAL */
+ _session.set_dirty ();
+ }
+
+ return 0;
+}
+
+int
+IO::ensure_inputs (uint32_t n, bool clear, bool lockit, void* src)
+{
+ bool changed = false;
+
+ if (_input_maximum >= 0) {
+ n = min (_input_maximum, (int) n);
+
+ if (n == _ninputs && !clear) {
+ return 0;
+ }
+ }
+
+ if (lockit) {
+ LockMonitor em (_session.engine().process_lock(), __LINE__, __FILE__);
+ changed = ensure_inputs_locked (n, clear, src);
+ } else {
+ changed = ensure_inputs_locked (n, clear, src);
+ }
+
+ if (changed) {
+ input_changed (ConfigurationChanged, src); /* EMIT SIGNAL */
+ _session.set_dirty ();
+ }
+
+ return 0;
+}
+
+bool
+IO::ensure_outputs_locked (uint32_t n, bool clear, void* src)
+{
+ Port* output_port;
+ bool changed = false;
+ bool reduced = false;
+ bool need_pan_reset;
+
+ if (_noutputs == n) {
+ need_pan_reset = false;
+ } else {
+ need_pan_reset = true;
+ }
+
+ /* remove unused ports */
+
+ while (_noutputs > n) {
+
+ _session.engine().unregister_port (_outputs.back());
+ _outputs.pop_back();
+ _noutputs--;
+ reduced = true;
+ changed = true;
+ }
+
+ /* create any necessary new ports */
+
+ while (_noutputs < n) {
+
+ char buf[64];
+
+ /* Create a new output port */
+
+ if (_output_maximum == 1) {
+ snprintf (buf, sizeof (buf), _("%s/out"), _name.c_str());
+ } else {
+ snprintf (buf, sizeof (buf), _("%s/out %u"), _name.c_str(), find_output_port_hole());
+ }
+
+ if ((output_port = _session.engine().register_audio_output_port (buf)) == 0) {
+ error << compose(_("IO: cannot register output port %1"), buf) << endmsg;
+ return -1;
+ }
+
+ _outputs.push_back (output_port);
+ sort (_outputs.begin(), _outputs.end(), sort_ports_by_name);
+ ++_noutputs;
+ changed = true;
+ setup_peak_meters ();
+
+ if (need_pan_reset) {
+ reset_panner ();
+ }
+ }
+
+ if (changed) {
+ drop_output_connection ();
+ MoreOutputs (_noutputs); /* EMIT SIGNAL */
+ _session.set_dirty ();
+ }
+
+ if (clear) {
+ /* disconnect all existing ports so that we get a fresh start */
+
+ for (vector<Port *>::iterator i = _outputs.begin(); i != _outputs.end(); ++i) {
+ _session.engine().disconnect (*i);
+ }
+ }
+
+ return changed;
+}
+
+int
+IO::ensure_outputs (uint32_t n, bool clear, bool lockit, void* src)
+{
+ bool changed = false;
+
+ if (_output_maximum >= 0) {
+ n = min (_output_maximum, (int) n);
+ if (n == _noutputs && !clear) {
+ return 0;
+ }
+ }
+
+ /* XXX caller should hold io_lock, but generally doesn't */
+
+ if (lockit) {
+ LockMonitor em (_session.engine().process_lock(), __LINE__, __FILE__);
+ changed = ensure_outputs_locked (n, clear, src);
+ } else {
+ changed = ensure_outputs_locked (n, clear, src);
+ }
+
+ if (changed) {
+ output_changed (ConfigurationChanged, src); /* EMIT SIGNAL */
+ }
+
+ return 0;
+}
+
+gain_t
+IO::effective_gain () const
+{
+ if (gain_automation_playback()) {
+ return _effective_gain;
+ } else {
+ return _desired_gain;
+ }
+}
+
+void
+IO::reset_panner ()
+{
+ if (panners_legal) {
+ if (!no_panner_reset) {
+ _panner->reset (_noutputs, 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 (_noutputs, 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[32];
+ string str;
+ bool need_ins = true;
+ bool need_outs = true;
+ LocaleGuard lg (X_("POSIX"));
+ LockMonitor lm (io_lock, __LINE__, __FILE__);
+
+ node->add_property("name", _name);
+ snprintf (buf, sizeof(buf), "%" PRIu64, id());
+ node->add_property("id", buf);
+
+ str = "";
+
+ if (_input_connection) {
+ node->add_property ("input-connection", _input_connection->name());
+ need_ins = false;
+ }
+
+ if (_output_connection) {
+ node->add_property ("output-connection", _output_connection->name());
+ need_outs = false;
+ }
+
+ if (need_ins) {
+ for (vector<Port *>::iterator i = _inputs.begin(); i != _inputs.end(); ++i) {
+
+ const char **connections = (*i)->get_connections();
+
+ if (connections && connections[0]) {
+ str += '{';
+
+ for (int n = 0; connections && connections[n]; ++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 (connections[n]);
+ }
+
+ str += '}';
+
+ free (connections);
+ }
+ else {
+ str += "{}";
+ }
+ }
+
+ node->add_property ("inputs", str);
+ }
+
+ if (need_outs) {
+ str = "";
+
+ for (vector<Port *>::iterator i = _outputs.begin(); i != _outputs.end(); ++i) {
+
+ const char **connections = (*i)->get_connections();
+
+ if (connections && connections[0]) {
+
+ str += '{';
+
+ for (int n = 0; connections[n]; ++n) {
+ if (n) {
+ str += ',';
+ }
+
+ str += _session.engine().make_port_name_relative (connections[n]);
+ }
+
+ str += '}';
+
+ free (connections);
+ }
+ else {
+ str += "{}";
+ }
+ }
+
+ node->add_property ("outputs", str);
+ }
+
+ node->add_child_nocopy (_panner->state (full_state));
+
+ snprintf (buf, sizeof(buf), "%2.12f", gain());
+ node->add_property ("gain", buf);
+
+ snprintf (buf, sizeof(buf)-1, "%d,%d,%d,%d",
+ _input_minimum,
+ _input_maximum,
+ _output_minimum,
+ _output_maximum);
+
+ node->add_property ("iolimits", buf);
+
+ /* MIDI control */
+
+ MIDI::channel_t chn;
+ MIDI::eventType ev;
+ MIDI::byte additional;
+ XMLNode* midi_node = 0;
+ XMLNode* child;
+
+ if (_midi_gain_control.get_control_info (chn, ev, additional)) {
+
+ midi_node = node->add_child ("MIDI");
+
+ child = midi_node->add_child ("gain");
+ set_midi_node_info (child, ev, chn, additional);
+ }
+
+ /* automation */
+
+ if (full_state) {
+ snprintf (buf, sizeof (buf), "0x%x", (int) _gain_automation_curve.automation_state());
+ } else {
+ /* never store anything except Off for automation state in a template */
+ snprintf (buf, sizeof (buf), "0x%x", ARDOUR::Off);
+ }
+ node->add_property ("automation-state", buf);
+ snprintf (buf, sizeof (buf), "0x%x", (int) _gain_automation_curve.automation_style());
+ node->add_property ("automation-style", buf);
+
+ /* XXX same for pan etc. */
+
+ return *node;
+}
+
+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::set_state (const XMLNode& node)
+{
+ const XMLProperty* prop;
+ XMLNodeConstIterator iter;
+ XMLNodeList midi_kids;
+ 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 << compose(_("incorrect XML node \"%1\" passed to IO object"), node.name()) << endmsg;
+ return -1;
+ }
+
+ if ((prop = node.property ("name")) != 0) {
+ _name = prop->value();
+ _panner->set_name (_name);
+ }
+
+ if ((prop = node.property ("id")) != 0) {
+ sscanf (prop->value().c_str(), "%llu", &_id);
+ }
+
+ if ((prop = node.property ("iolimits")) != 0) {
+ sscanf (prop->value().c_str(), "%d,%d,%d,%d",
+ &_input_minimum,
+ &_input_maximum,
+ &_output_minimum,
+ &_output_maximum);
+ }
+
+ if ((prop = node.property ("gain")) != 0) {
+ set_gain (atof (prop->value().c_str()), this);
+ _gain = _desired_gain;
+ }
+
+ for (iter = node.children().begin(); iter != node.children().end(); ++iter) {
+ if ((*iter)->name() == "Panner") {
+ _panner->set_state (**iter);
+ }
+ }
+
+ midi_kids = node.children ("MIDI");
+
+ for (iter = midi_kids.begin(); iter != midi_kids.end(); ++iter) {
+
+ XMLNodeList kids;
+ XMLNodeConstIterator miter;
+ XMLNode* child;
+
+ kids = (*iter)->children ();
+
+ for (miter = kids.begin(); miter != kids.end(); ++miter) {
+
+ child =* miter;
+
+ if (child->name() == "gain") {
+
+ MIDI::eventType ev = MIDI::on; /* initialize to keep gcc happy */
+ MIDI::byte additional = 0; /* ditto */
+ MIDI::channel_t chn = 0; /* ditto */
+
+ if (get_midi_node_info (child, ev, chn, additional)) {
+ _midi_gain_control.set_control_type (chn, ev, additional);
+ } else {
+ error << compose(_("MIDI gain control specification for %1 is incomplete, so it has been ignored"), _name) << endmsg;
+ }
+ }
+ }
+ }
+
+ if ((prop = node.property ("automation-state")) != 0) {
+
+ long int x;
+ x = strtol (prop->value().c_str(), 0, 16);
+ set_gain_automation_state (AutoState (x));
+ }
+
+ if ((prop = node.property ("automation-style")) != 0) {
+
+ long int x;
+ x = strtol (prop->value().c_str(), 0, 16);
+ set_gain_automation_style (AutoStyle (x));
+ }
+
+ 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::create_ports (const XMLNode& node)
+{
+ const XMLProperty* prop;
+ int num_inputs = 0;
+ int num_outputs = 0;
+
+ if ((prop = node.property ("input-connection")) != 0) {
+
+ Connection* c = _session.connection_by_name (prop->value());
+
+ if (c == 0) {
+ error << compose(_("Unknown connection \"%1\" listed for input of %2"), prop->value(), _name) << endmsg;
+
+ if ((c = _session.connection_by_name (_("in 1"))) == 0) {
+ error << _("No input connections available as a replacement")
+ << endmsg;
+ return -1;
+ } else {
+ info << compose (_("Connection %1 was not available - \"in 1\" used instead"), prop->value())
+ << endmsg;
+ }
+ }
+
+ num_inputs = c->nports();
+
+ } else if ((prop = node.property ("inputs")) != 0) {
+
+ num_inputs = count (prop->value().begin(), prop->value().end(), '{');
+ }
+
+ if ((prop = node.property ("output-connection")) != 0) {
+ Connection* c = _session.connection_by_name (prop->value());
+
+ if (c == 0) {
+ error << compose(_("Unknown connection \"%1\" listed for output of %2"), prop->value(), _name) << endmsg;
+
+ if ((c = _session.connection_by_name (_("out 1"))) == 0) {
+ error << _("No output connections available as a replacement")
+ << endmsg;
+ return -1;
+ } else {
+ info << compose (_("Connection %1 was not available - \"out 1\" used instead"), prop->value())
+ << endmsg;
+ }
+ }
+
+ num_outputs = c->nports ();
+
+ } else if ((prop = node.property ("outputs")) != 0) {
+ num_outputs = count (prop->value().begin(), prop->value().end(), '{');
+ }
+
+ no_panner_reset = true;
+
+ if (ensure_io (num_inputs, num_outputs, true, this)) {
+ error << compose(_("%1: cannot create I/O ports"), _name) << endmsg;
+ return -1;
+ }
+
+ no_panner_reset = false;
+
+ set_deferred_state ();
+
+ PortsCreated();
+ return 0;
+}
+
+bool
+IO::get_midi_node_info (XMLNode * node, MIDI::eventType & ev, MIDI::channel_t & chan, MIDI::byte & additional)
+{
+ bool ok = true;
+ const XMLProperty* prop;
+ int xx;
+
+ if ((prop = node->property ("event")) != 0) {
+ sscanf (prop->value().c_str(), "0x%x", &xx);
+ ev = (MIDI::eventType) xx;
+ } else {
+ ok = false;
+ }
+
+ if (ok && ((prop = node->property ("channel")) != 0)) {
+ sscanf (prop->value().c_str(), "%d", &xx);
+ chan = (MIDI::channel_t) xx;
+ } else {
+ ok = false;
+ }
+
+ if (ok && ((prop = node->property ("additional")) != 0)) {
+ sscanf (prop->value().c_str(), "0x%x", &xx);
+ additional = (MIDI::byte) xx;
+ }
+
+ return ok;
+}
+
+bool
+IO::set_midi_node_info (XMLNode * node, MIDI::eventType ev, MIDI::channel_t chan, MIDI::byte additional)
+{
+ char buf[32];
+
+ snprintf (buf, sizeof(buf), "0x%x", ev);
+ node->add_property ("event", buf);
+ snprintf (buf, sizeof(buf), "%d", chan);
+ node->add_property ("channel", buf);
+ snprintf (buf, sizeof(buf), "0x%x", additional);
+ node->add_property ("additional", buf);
+
+ return true;
+}
+
+
+int
+IO::make_connections (const XMLNode& node)
+{
+ const XMLProperty* prop;
+
+ if ((prop = node.property ("input-connection")) != 0) {
+ Connection* c = _session.connection_by_name (prop->value());
+
+ if (c == 0) {
+ error << compose(_("Unknown connection \"%1\" listed for input of %2"), prop->value(), _name) << endmsg;
+
+ if ((c = _session.connection_by_name (_("in 1"))) == 0) {
+ error << _("No input connections available as a replacement")
+ << endmsg;
+ return -1;
+ } else {
+ info << compose (_("Connection %1 was not available - \"in 1\" used instead"), prop->value())
+ << endmsg;
+ }
+ }
+
+ use_input_connection (*c, this);
+
+ } else if ((prop = node.property ("inputs")) != 0) {
+ if (set_inputs (prop->value())) {
+ error << compose(_("improper input channel list in XML node (%1)"), prop->value()) << endmsg;
+ return -1;
+ }
+ }
+
+ if ((prop = node.property ("output-connection")) != 0) {
+ Connection* c = _session.connection_by_name (prop->value());
+
+ if (c == 0) {
+ error << compose(_("Unknown connection \"%1\" listed for output of %2"), prop->value(), _name) << endmsg;
+
+ if ((c = _session.connection_by_name (_("out 1"))) == 0) {
+ error << _("No output connections available as a replacement")
+ << endmsg;
+ return -1;
+ } else {
+ info << compose (_("Connection %1 was not available - \"out 1\" used instead"), prop->value())
+ << endmsg;
+ }
+ }
+
+ use_output_connection (*c, this);
+
+ } else if ((prop = node.property ("outputs")) != 0) {
+ if (set_outputs (prop->value())) {
+ error << compose(_("improper output channel list in XML node (%1)"), prop->value()) << endmsg;
+ return -1;
+ }
+ }
+
+ 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;
+ }
+
+ if (ensure_inputs (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 << 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 << 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;
+ }
+
+ if (ensure_outputs (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 << 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 << 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();
+}
+
+int
+IO::set_name (string name, void* src)
+{
+ if (name == _name) {
+ return 0;
+ }
+
+ for (vector<Port *>::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 (vector<Port *>::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);
+ }
+
+ _name = name;
+ name_changed (src); /* EMIT SIGNAL */
+
+ return 0;
+}
+
+void
+IO::set_input_minimum (int n)
+{
+ _input_minimum = n;
+}
+
+void
+IO::set_input_maximum (int n)
+{
+ _input_maximum = n;
+}
+
+void
+IO::set_output_minimum (int n)
+{
+ _output_minimum = n;
+}
+
+void
+IO::set_output_maximum (int n)
+{
+ _output_maximum = n;
+}
+
+void
+IO::set_port_latency (jack_nframes_t nframes)
+{
+ LockMonitor lm (io_lock, __LINE__, __FILE__);
+
+ for (vector<Port *>::iterator i = _outputs.begin(); i != _outputs.end(); ++i) {
+ (*i)->set_latency (nframes);
+ }
+}
+
+jack_nframes_t
+IO::output_latency () const
+{
+ jack_nframes_t max_latency;
+ jack_nframes_t latency;
+
+ max_latency = 0;
+
+ /* io lock not taken - must be protected by other means */
+
+ for (vector<Port *>::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;
+}
+
+jack_nframes_t
+IO::input_latency () const
+{
+ jack_nframes_t max_latency;
+ jack_nframes_t latency;
+
+ max_latency = 0;
+
+ /* io lock not taken - must be protected by other means */
+
+ for (vector<Port *>::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::use_input_connection (Connection& c, void* src)
+{
+ uint32_t limit;
+
+ {
+ LockMonitor lm (_session.engine().process_lock(), __LINE__, __FILE__);
+ LockMonitor lm2 (io_lock, __LINE__, __FILE__);
+
+ limit = c.nports();
+
+ drop_input_connection ();
+
+ if (ensure_inputs (limit, false, false, src)) {
+ return -1;
+ }
+
+ /* first pass: check the current state to see what's correctly
+ connected, and drop anything that we don't want.
+ */
+
+ for (uint32_t n = 0; n < limit; ++n) {
+ const Connection::PortList& pl = c.port_connections (n);
+
+ for (Connection::PortList::const_iterator i = pl.begin(); i != pl.end(); ++i) {
+
+ if (!_inputs[n]->connected_to ((*i))) {
+
+ /* clear any existing connections */
+
+ _session.engine().disconnect (_inputs[n]);
+
+ } else if (_inputs[n]->connected() > 1) {
+
+ /* OK, it is connected to the port we want,
+ but its also connected to other ports.
+ Change that situation.
+ */
+
+ /* XXX could be optimized to not drop
+ the one we want.
+ */
+
+ _session.engine().disconnect (_inputs[n]);
+
+ }
+ }
+ }
+
+ /* second pass: connect all requested ports where necessary */
+
+ for (uint32_t n = 0; n < limit; ++n) {
+ const Connection::PortList& pl = c.port_connections (n);
+
+ for (Connection::PortList::const_iterator i = pl.begin(); i != pl.end(); ++i) {
+
+ if (!_inputs[n]->connected_to ((*i))) {
+
+ if (_session.engine().connect (*i, _inputs[n]->name())) {
+ return -1;
+ }
+ }
+
+ }
+ }
+
+ _input_connection = &c;
+
+ input_connection_configuration_connection = c.ConfigurationChanged.connect
+ (mem_fun (*this, &IO::input_connection_configuration_changed));
+ input_connection_connection_connection = c.ConnectionsChanged.connect
+ (mem_fun (*this, &IO::input_connection_connection_changed));
+ }
+
+ input_changed (IOChange (ConfigurationChanged|ConnectionsChanged), src); /* EMIT SIGNAL */
+ return 0;
+}
+
+int
+IO::use_output_connection (Connection& c, void* src)
+{
+ uint32_t limit;
+
+ {
+ LockMonitor lm (_session.engine().process_lock(), __LINE__, __FILE__);
+ LockMonitor lm2 (io_lock, __LINE__, __FILE__);
+
+ limit = c.nports();
+
+ drop_output_connection ();
+
+ if (ensure_outputs (limit, false, false, src)) {
+ return -1;
+ }
+
+ /* first pass: check the current state to see what's correctly
+ connected, and drop anything that we don't want.
+ */
+
+ for (uint32_t n = 0; n < limit; ++n) {
+
+ const Connection::PortList& pl = c.port_connections (n);
+
+ for (Connection::PortList::const_iterator i = pl.begin(); i != pl.end(); ++i) {
+
+ if (!_outputs[n]->connected_to ((*i))) {
+
+ /* clear any existing connections */
+
+ _session.engine().disconnect (_outputs[n]);
+
+ } else if (_outputs[n]->connected() > 1) {
+
+ /* OK, it is connected to the port we want,
+ but its also connected to other ports.
+ Change that situation.
+ */
+
+ /* XXX could be optimized to not drop
+ the one we want.
+ */
+
+ _session.engine().disconnect (_outputs[n]);
+ }
+ }
+ }
+
+ /* second pass: connect all requested ports where necessary */
+
+ for (uint32_t n = 0; n < limit; ++n) {
+
+ const Connection::PortList& pl = c.port_connections (n);
+
+ for (Connection::PortList::const_iterator i = pl.begin(); i != pl.end(); ++i) {
+
+ if (!_outputs[n]->connected_to ((*i))) {
+
+ if (_session.engine().connect (_outputs[n]->name(), *i)) {
+ return -1;
+ }
+ }
+ }
+ }
+
+ _output_connection = &c;
+
+ output_connection_configuration_connection = c.ConfigurationChanged.connect
+ (mem_fun (*this, &IO::output_connection_configuration_changed));
+ output_connection_connection_connection = c.ConnectionsChanged.connect
+ (mem_fun (*this, &IO::output_connection_connection_changed));
+ }
+
+ 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::input_connection_connection_changed (int ignored)
+{
+ use_input_connection (*_input_connection, this);
+}
+
+void
+IO::input_connection_configuration_changed ()
+{
+ use_input_connection (*_input_connection, this);
+}
+
+void
+IO::output_connection_connection_changed (int ignored)
+{
+ use_output_connection (*_output_connection, this);
+}
+
+void
+IO::output_connection_configuration_changed ()
+{
+ use_output_connection (*_output_connection, this);
+}
+
+IO::MIDIGainControl::MIDIGainControl (IO& i, MIDI::Port* port)
+ : MIDI::Controllable (port, 0), io (i), setting(false)
+{
+ midi_to_gain = 0;
+ gain_to_midi = 0;
+ setting = false;
+ last_written = 0; /* XXX need a good out-of-bound-value */
+}
+
+void
+IO::MIDIGainControl::set_value (float val)
+{
+ if (midi_to_gain == 0) return;
+
+ setting = true;
+ io.set_gain (midi_to_gain (val), this);
+ setting = false;
+}
+
+void
+IO::MIDIGainControl::send_feedback (gain_t gain)
+{
+ if (!setting && get_midi_feedback() && gain_to_midi) {
+ MIDI::byte val = (MIDI::byte) (gain_to_midi (gain) * 127.0);
+ MIDI::channel_t ch = 0;
+ MIDI::eventType ev = MIDI::none;
+ MIDI::byte additional = 0;
+ MIDI::EventTwoBytes data;
+
+ if (get_control_info (ch, ev, additional)) {
+ data.controller_number = additional;
+ data.value = val;
+
+ io._session.send_midi_message (get_port(), ev, ch, data);
+ }
+ //send_midi_feedback (gain_to_midi (gain));
+ }
+}
+
+MIDI::byte*
+IO::MIDIGainControl::write_feedback (MIDI::byte* buf, int32_t& bufsize, gain_t val, bool force)
+{
+ if (get_midi_feedback() && gain_to_midi && bufsize > 2) {
+ MIDI::channel_t ch = 0;
+ MIDI::eventType ev = MIDI::none;
+ MIDI::byte additional = 0;
+ MIDI::byte gm;
+
+ if (get_control_info (ch, ev, additional)) {
+ gm = (MIDI::byte) (gain_to_midi (val) * 127.0);
+
+ if (gm != last_written) {
+ *buf++ = (0xF0 & ev) | (0xF & ch);
+ *buf++ = additional; /* controller number */
+ *buf++ = gm;
+ last_written = gm;
+ bufsize -= 3;
+ }
+ }
+ }
+
+ return buf;
+}
+
+void
+IO::reset_peak_meters ()
+{
+ uint32_t limit = max (_ninputs, _noutputs);
+
+ for (uint32_t i = 0; i < limit; ++i) {
+ _peak_power[i] = 0;
+ }
+}
+
+void
+IO::setup_peak_meters ()
+{
+ uint32_t limit = max (_ninputs, _noutputs);
+
+ while (_peak_power.size() < limit) {
+ _peak_power.push_back (0);
+ _stored_peak_power.push_back (0);
+ }
+}
+
+UndoAction
+IO::get_memento() const
+{
+ return sigc::bind (mem_fun (*(const_cast<IO *>(this)), &StateManager::use_state), _current_state_id);
+}
+
+Change
+IO::restore_state (StateManager::State& state)
+{
+ return Change (0);
+}
+
+StateManager::State*
+IO::state_factory (std::string why) const
+{
+ StateManager::State* state = new StateManager::State (why);
+ return state;
+}
+
+void
+IO::send_state_changed ()
+{
+ return;
+}
+
+void
+IO::grab_peak_power ()
+{
+ LockMonitor lm (io_lock, __LINE__, __FILE__);
+
+ uint32_t limit = max (_ninputs, _noutputs);
+
+ for (uint32_t n = 0; n < limit; ++n) {
+ /* XXX should we use atomic exchange here ? */
+ _stored_peak_power[n] = _peak_power[n];
+ _peak_power[n] = 0;
+ }
+}
+
+void
+IO::reset_midi_control (MIDI::Port* port, bool on)
+{
+ MIDI::channel_t chn;
+ MIDI::eventType ev;
+ MIDI::byte extra;
+
+ _midi_gain_control.get_control_info (chn, ev, extra);
+ if (!on) {
+ chn = -1;
+ }
+ _midi_gain_control.midi_rebind (port, chn);
+
+ _panner->reset_midi_control (port, on);
+}
+
+
+int
+IO::save_automation (const string& path)
+{
+ string fullpath;
+ ofstream out;
+
+ fullpath = _session.automation_dir();
+ fullpath += path;
+
+ out.open (fullpath.c_str());
+
+ if (!out) {
+ error << compose(_("%1: could not open automation event file \"%2\""), _name, fullpath) << endmsg;
+ return -1;
+ }
+
+ out << X_("version ") << current_automation_version_number << endl;
+
+ /* XXX use apply_to_points to get thread safety */
+
+ for (AutomationList::iterator i = _gain_automation_curve.begin(); i != _gain_automation_curve.end(); ++i) {
+ out << "g " << (jack_nframes_t) floor ((*i)->when) << ' ' << (*i)->value << endl;
+ }
+
+ _panner->save ();
+
+ return 0;
+}
+
+int
+IO::load_automation (const 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 << compose(_("%1: cannot open automation event file \"%2\""), _name, fullpath) << endmsg;
+ return -1;
+ }
+ }
+
+ clear_automation ();
+
+ while (in.getline (line, sizeof(line), '\n')) {
+ char type;
+ jack_nframes_t when;
+ double value;
+
+ if (++linecnt == 1) {
+ if (memcmp (line, "version", 7) == 0) {
+ if (sscanf (line, "version %f", &version) != 1) {
+ error << compose(_("badly formed version number in automation event file \"%1\""), path) << endmsg;
+ return -1;
+ }
+ } else {
+ error << compose(_("no version information in automation event file \"%1\""), path) << endmsg;
+ return -1;
+ }
+
+ if (version != current_automation_version_number) {
+ error << compose(_("mismatched automation event file version (%1)"), version) << endmsg;
+ return -1;
+ }
+
+ continue;
+ }
+
+ if (sscanf (line, "%c %" PRIu32 " %lf", &type, &when, &value) != 3) {
+ warning << compose(_("badly formatted automation event record at line %1 of %2 (ignored)"), linecnt, path) << endmsg;
+ continue;
+ }
+
+ switch (type) {
+ case 'g':
+ _gain_automation_curve.add (when, value, true);
+ 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;
+ }
+ }
+
+ _gain_automation_curve.save_state (_("loaded from disk"));
+
+ return 0;
+}
+
+void
+IO::clear_automation ()
+{
+ LockMonitor lm (automation_lock, __LINE__, __FILE__);
+ _gain_automation_curve.clear ();
+ _panner->clear_automation ();
+}
+
+void
+IO::set_gain_automation_state (AutoState state)
+{
+ bool changed = false;
+
+ {
+ LockMonitor lm (automation_lock, __LINE__, __FILE__);
+
+ if (state != _gain_automation_curve.automation_state()) {
+ changed = true;
+ last_automation_snapshot = 0;
+ _gain_automation_curve.set_automation_state (state);
+
+ if (state != Off) {
+ set_gain (_gain_automation_curve.eval (_session.transport_frame()), this);
+ }
+ }
+ }
+
+ if (changed) {
+ _session.set_dirty ();
+ gain_automation_state_changed (); /* EMIT SIGNAL */
+ }
+}
+
+void
+IO::set_gain_automation_style (AutoStyle style)
+{
+ bool changed = false;
+
+ {
+ LockMonitor lm (automation_lock, __LINE__, __FILE__);
+
+ if (style != _gain_automation_curve.automation_style()) {
+ changed = true;
+ _gain_automation_curve.set_automation_style (style);
+ }
+ }
+
+ if (changed) {
+ gain_automation_style_changed (); /* EMIT SIGNAL */
+ }
+}
+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;
+
+ {
+ LockMonitor dm (declick_lock, __LINE__, __FILE__);
+ _desired_gain = val;
+ }
+
+ if (_session.transport_stopped()) {
+ _effective_gain = val;
+ _gain = val;
+ }
+
+ gain_changed (src);
+
+ if (_session.get_midi_feedback()) {
+ _midi_gain_control.send_feedback (_desired_gain);
+ }
+
+ if (_session.transport_stopped() && src != 0 && src != this && gain_automation_recording()) {
+ _gain_automation_curve.add (_session.transport_frame(), val);
+
+ }
+
+ _session.set_dirty();
+}
+
+void
+IO::send_all_midi_feedback ()
+{
+ if (_session.get_midi_feedback()) {
+ _midi_gain_control.send_feedback (_effective_gain);
+
+ // panners
+ _panner->send_all_midi_feedback();
+ }
+}
+
+MIDI::byte*
+IO::write_midi_feedback (MIDI::byte* buf, int32_t& bufsize)
+{
+ if (_session.get_midi_feedback()) {
+ if (gain_automation_playback ()) {
+ buf = _midi_gain_control.write_feedback (buf, bufsize, _effective_gain);
+ }
+ buf = _panner->write_midi_feedback (buf, bufsize);
+ }
+
+ return buf;
+}
+
+void
+IO::start_gain_touch ()
+{
+ _gain_automation_curve.start_touch ();
+}
+
+void
+IO::end_gain_touch ()
+{
+ _gain_automation_curve.stop_touch ();
+}
+
+void
+IO::start_pan_touch (uint32_t which)
+{
+ if (which < _panner->size()) {
+ (*_panner)[which]->automation().start_touch();
+ }
+}
+
+void
+IO::end_pan_touch (uint32_t which)
+{
+ if (which < _panner->size()) {
+ (*_panner)[which]->automation().stop_touch();
+ }
+
+}
+
+void
+IO::automation_snapshot (jack_nframes_t now)
+{
+ if (last_automation_snapshot > now || (now - last_automation_snapshot) > _automation_interval) {
+
+ if (gain_automation_recording()) {
+ _gain_automation_curve.rt_add (now, gain());
+ }
+
+ _panner->snapshot (now);
+
+ last_automation_snapshot = now;
+ }
+}
+
+void
+IO::transport_stopped (jack_nframes_t frame)
+{
+ _gain_automation_curve.reposition_for_rt_add (frame);
+
+ if (_gain_automation_curve.automation_state() != Off) {
+
+ if (gain_automation_recording()) {
+ _gain_automation_curve.save_state (_("automation write/touch"));
+ }
+
+ /* the src=0 condition is a special signal to not propagate
+ automation gain changes into the mix group when locating.
+ */
+
+ set_gain (_gain_automation_curve.eval (frame), 0);
+ }
+
+ _panner->transport_stopped (frame);
+}
+
+int32_t
+IO::find_input_port_hole ()
+{
+ /* CALLER MUST HOLD IO LOCK */
+
+ uint32_t n;
+
+ if (_inputs.empty()) {
+ return 1;
+ }
+
+ for (n = 1; n < UINT_MAX; ++n) {
+ char buf[jack_port_name_size()];
+ vector<Port*>::iterator i;
+
+ snprintf (buf, jack_port_name_size(), _("%s/in %u"), _name.c_str(), n);
+
+ for (i = _inputs.begin(); i != _inputs.end(); ++i) {
+ if ((*i)->short_name() == buf) {
+ break;
+ }
+ }
+
+ if (i == _inputs.end()) {
+ break;
+ }
+ }
+ return n;
+}
+
+int32_t
+IO::find_output_port_hole ()
+{
+ /* CALLER MUST HOLD IO LOCK */
+
+ uint32_t n;
+
+ if (_outputs.empty()) {
+ return 1;
+ }
+
+ for (n = 1; n < UINT_MAX; ++n) {
+ char buf[jack_port_name_size()];
+ vector<Port*>::iterator i;
+
+ snprintf (buf, jack_port_name_size(), _("%s/out %u"), _name.c_str(), n);
+
+ for (i = _outputs.begin(); i != _outputs.end(); ++i) {
+ if ((*i)->short_name() == buf) {
+ break;
+ }
+ }
+
+ if (i == _outputs.end()) {
+ break;
+ }
+ }
+
+ return n;
+}