/* Copyright (C) 2002 Paul Davis This program is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation; either version 2 of the License, or (at your option) any later version. This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with this program; if not, write to the Free Software Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA. */ #include #include "pbd/error.h" #include "pbd/enumwriter.h" #include "pbd/boost_debug.h" #include "evoral/Curve.hpp" #include "ardour/amp.h" #include "ardour/audio_buffer.h" #include "ardour/audio_diskstream.h" #include "ardour/audio_track.h" #include "ardour/audioplaylist.h" #include "ardour/audioregion.h" #include "ardour/audiosource.h" #include "ardour/buffer_set.h" #include "ardour/io_processor.h" #include "ardour/panner.h" #include "ardour/meter.h" #include "ardour/playlist_factory.h" #include "ardour/plugin_insert.h" #include "ardour/processor.h" #include "ardour/region_factory.h" #include "ardour/route_group_specialized.h" #include "ardour/session.h" #include "ardour/utils.h" #include "ardour/session_playlists.h" #include "ardour/delivery.h" #include "ardour/meter.h" #include "i18n.h" using namespace std; using namespace ARDOUR; using namespace PBD; AudioTrack::AudioTrack (Session& sess, string name, Route::Flag flag, TrackMode mode) : Track (sess, name, flag, mode) { } AudioTrack::~AudioTrack () { } void AudioTrack::use_new_diskstream () { AudioDiskstream::Flag dflags = AudioDiskstream::Flag (0); if (_flags & Hidden) { dflags = AudioDiskstream::Flag (dflags | AudioDiskstream::Hidden); } else { dflags = AudioDiskstream::Flag (dflags | AudioDiskstream::Recordable); } if (_mode == Destructive) { dflags = AudioDiskstream::Flag (dflags | AudioDiskstream::Destructive); } else if (_mode == NonLayered){ dflags = AudioDiskstream::Flag(dflags | AudioDiskstream::NonLayered); } AudioDiskstream* dsp (new AudioDiskstream (_session, name(), dflags)); boost::shared_ptr ds (dsp); ds->do_refill_with_alloc (); ds->set_block_size (_session.get_block_size ()); set_diskstream (ds); } void AudioTrack::set_diskstream (boost::shared_ptr ds) { Track::set_diskstream (ds); _diskstream->set_track (this); _diskstream->set_destructive (_mode == Destructive); _diskstream->set_non_layered (_mode == NonLayered); if (audio_diskstream()->deprecated_io_node) { if (!IO::connecting_legal) { IO::ConnectingLegal.connect_same_thread (*this, boost::bind (&AudioTrack::deprecated_use_diskstream_connections, this)); } else { deprecated_use_diskstream_connections (); } } _diskstream->set_record_enabled (false); _diskstream->monitor_input (false); DiskstreamChanged (); /* EMIT SIGNAL */ } boost::shared_ptr AudioTrack::audio_diskstream() const { return boost::dynamic_pointer_cast(_diskstream); } int AudioTrack::set_mode (TrackMode m) { if (m != _mode) { if (_diskstream->set_destructive (m == Destructive)) { return -1; } _diskstream->set_non_layered (m == NonLayered); _mode = m; TrackModeChanged (); /* EMIT SIGNAL */ } return 0; } bool AudioTrack::can_use_mode (TrackMode m, bool& bounce_required) { switch (m) { case NonLayered: case Normal: bounce_required = false; return true; case Destructive: default: return _diskstream->can_become_destructive (bounce_required); } } int AudioTrack::deprecated_use_diskstream_connections () { boost::shared_ptr diskstream = audio_diskstream(); if (diskstream->deprecated_io_node == 0) { return 0; } const XMLProperty* prop; XMLNode& node (*diskstream->deprecated_io_node); /* don't do this more than once. */ diskstream->deprecated_io_node = 0; if ((prop = node.property ("gain")) != 0) { _amp->set_gain (atof (prop->value().c_str()), this); } if ((prop = node.property ("input-connection")) != 0) { boost::shared_ptr c = _session.bundle_by_name (prop->value()); if (c == 0) { error << string_compose(_("Unknown bundle \"%1\" listed for input of %2"), prop->value(), _name) << endmsg; if ((c = _session.bundle_by_name (_("in 1"))) == 0) { error << _("No input bundles available as a replacement") << endmsg; return -1; } else { info << string_compose (_("Bundle %1 was not available - \"in 1\" used instead"), prop->value()) << endmsg; } } _input->connect_ports_to_bundle (c, this); } else if ((prop = node.property ("inputs")) != 0) { if (_input->set_ports (prop->value())) { error << string_compose(_("improper input channel list in XML node (%1)"), prop->value()) << endmsg; return -1; } } return 0; } int AudioTrack::set_state (const XMLNode& node, int version) { return _set_state (node, version, true); } int AudioTrack::_set_state (const XMLNode& node, int version, bool call_base) { const XMLProperty *prop; XMLNodeConstIterator iter; if (call_base) { if (Route::_set_state (node, version, call_base)) { return -1; } } if ((prop = node.property (X_("mode"))) != 0) { _mode = TrackMode (string_2_enum (prop->value(), _mode)); } else { _mode = Normal; } XMLNodeList nlist; XMLNodeConstIterator niter; XMLNode *child; nlist = node.children(); for (niter = nlist.begin(); niter != nlist.end(); ++niter){ child = *niter; if (child->name() == X_("recenable")) { _rec_enable_control->set_state (*child, version); _session.add_controllable (_rec_enable_control); } } if (version >= 3000) { if ((child = find_named_node (node, X_("Diskstream"))) != 0) { boost::shared_ptr ds (new AudioDiskstream (_session, *child)); ds->do_refill_with_alloc (); set_diskstream (ds); } } pending_state = const_cast (&node); if (_session.state_of_the_state() & Session::Loading) { _session.StateReady.connect_same_thread (*this, boost::bind (&AudioTrack::set_state_part_two, this)); } else { set_state_part_two (); } return 0; } XMLNode& AudioTrack::state (bool full_state) { XMLNode& root (Route::state(full_state)); XMLNode* freeze_node; char buf[64]; if (_freeze_record.playlist) { XMLNode* inode; freeze_node = new XMLNode (X_("freeze-info")); freeze_node->add_property ("playlist", _freeze_record.playlist->name()); freeze_node->add_property ("state", enum_2_string (_freeze_record.state)); for (vector::iterator i = _freeze_record.processor_info.begin(); i != _freeze_record.processor_info.end(); ++i) { inode = new XMLNode (X_("processor")); (*i)->id.print (buf, sizeof (buf)); inode->add_property (X_("id"), buf); inode->add_child_copy ((*i)->state); freeze_node->add_child_nocopy (*inode); } root.add_child_nocopy (*freeze_node); } root.add_property (X_("mode"), enum_2_string (_mode)); root.add_child_nocopy (_rec_enable_control->get_state()); root.add_child_nocopy (_diskstream->get_state ()); return root; } void AudioTrack::set_state_part_two () { XMLNode* fnode; XMLProperty* prop; LocaleGuard lg (X_("POSIX")); /* This is called after all session state has been restored but before have been made ports and connections are established. */ if (pending_state == 0) { return; } if ((fnode = find_named_node (*pending_state, X_("freeze-info"))) != 0) { _freeze_record.state = Frozen; for (vector::iterator i = _freeze_record.processor_info.begin(); i != _freeze_record.processor_info.end(); ++i) { delete *i; } _freeze_record.processor_info.clear (); if ((prop = fnode->property (X_("playlist"))) != 0) { boost::shared_ptr pl = _session.playlists->by_name (prop->value()); if (pl) { _freeze_record.playlist = boost::dynamic_pointer_cast (pl); } else { _freeze_record.playlist.reset (); _freeze_record.state = NoFreeze; return; } } if ((prop = fnode->property (X_("state"))) != 0) { _freeze_record.state = FreezeState (string_2_enum (prop->value(), _freeze_record.state)); } XMLNodeConstIterator citer; XMLNodeList clist = fnode->children(); for (citer = clist.begin(); citer != clist.end(); ++citer) { if ((*citer)->name() != X_("processor")) { continue; } if ((prop = (*citer)->property (X_("id"))) == 0) { continue; } FreezeRecordProcessorInfo* frii = new FreezeRecordProcessorInfo (*((*citer)->children().front()), boost::shared_ptr()); frii->id = prop->value (); _freeze_record.processor_info.push_back (frii); } } } int AudioTrack::roll (nframes_t nframes, framepos_t start_frame, framepos_t end_frame, int declick, bool can_record, bool rec_monitors_input, bool& need_butler) { Glib::RWLock::ReaderLock lm (_processor_lock, Glib::TRY_LOCK); if (!lm.locked()) { return 0; } int dret; Sample* b; Sample* tmpb; nframes_t transport_frame; boost::shared_ptr diskstream = audio_diskstream(); automation_snapshot (start_frame, false); if (n_outputs().n_total() == 0 && _processors.empty()) { return 0; } if (!_active) { silence (nframes); return 0; } transport_frame = _session.transport_frame(); if ((nframes = check_initial_delay (nframes, transport_frame)) == 0) { /* need to do this so that the diskstream sets its playback distance to zero, thus causing diskstream::commit to do nothing. */ return diskstream->process (transport_frame, 0, can_record, rec_monitors_input, need_butler); } _silent = false; _amp->apply_gain_automation(false); if ((dret = diskstream->process (transport_frame, nframes, can_record, rec_monitors_input, need_butler)) != 0) { silence (nframes); return dret; } /* special condition applies */ if (_meter_point == MeterInput) { _input->process_input (_meter, start_frame, end_frame, nframes); } if (diskstream->record_enabled() && !can_record && !_session.config.get_auto_input()) { /* not actually recording, but we want to hear the input material anyway, at least potentially (depending on monitoring options) */ passthru (start_frame, end_frame, nframes, false); } else if ((b = diskstream->playback_buffer(0)) != 0) { /* XXX is it true that the earlier test on n_outputs() means that we can avoid checking it again here? i think so, because changing the i/o configuration of an IO requires holding the AudioEngine lock, which we hold while in the process() tree. */ /* copy the diskstream data to all output buffers */ size_t limit = input_streams ().n_audio(); BufferSet& bufs = _session.get_scratch_buffers (); const size_t blimit = bufs.count().n_audio(); uint32_t n; uint32_t i; if (limit > blimit) { /* example case: auditioner configured for stereo output, but loaded with an 8 channel file. there are only 2 passthrough buffers, but n_process_buffers() will return 8. arbitrary decision: map all channels in the diskstream to the outputs available. */ float scaling = limit/blimit; for (i = 0, n = 1; i < blimit; ++i, ++n) { /* first time through just copy a channel into the output buffer. */ Sample* bb = bufs.get_audio (i).data(); for (nframes_t xx = 0; xx < nframes; ++xx) { bb[xx] = b[xx] * scaling; } if (n < diskstream->n_channels().n_audio()) { tmpb = diskstream->playback_buffer(n); if (tmpb!=0) { b = tmpb; } } } for (;i < limit; ++i, ++n) { /* for all remaining channels, sum with existing data in the output buffers */ bufs.get_audio (i%blimit).accumulate_with_gain_from (b, nframes, 0, scaling); if (n < diskstream->n_channels().n_audio()) { tmpb = diskstream->playback_buffer(n); if (tmpb!=0) { b = tmpb; } } } limit = blimit; } else { for (i = 0, n = 1; i < limit; ++i, ++n) { memcpy (bufs.get_audio (i).data(), b, sizeof (Sample) * nframes); if (n < diskstream->n_channels().n_audio()) { tmpb = diskstream->playback_buffer(n); if (tmpb!=0) { b = tmpb; } } } /* try to leave any MIDI buffers alone */ ChanCount chn; chn.set_audio (limit); chn.set_midi (_input->n_ports().n_midi()); bufs.set_count (chn); } /* don't waste time with automation if we're recording or we've just stopped (yes it can happen) */ if (!diskstream->record_enabled() && _session.transport_rolling()) { #ifdef XXX_MOVE_THIS_TO_AMP Glib::Mutex::Lock am (data().control_lock(), Glib::TRY_LOCK); if (am.locked() && gain_control()->automation_playback()) { _amp->apply_gain_automation( gain_control()->list()->curve().rt_safe_get_vector ( start_frame, end_frame, _session.gain_automation_buffer(), nframes)); } #endif } process_output_buffers (bufs, start_frame, end_frame, nframes, (!_session.get_record_enabled() || !Config->get_do_not_record_plugins()), declick); } else { /* problem with the diskstream; just be quiet for a bit */ silence (nframes); } return 0; } int AudioTrack::export_stuff (BufferSet& buffers, framepos_t start, framecnt_t nframes, bool enable_processing) { boost::scoped_array gain_buffer (new gain_t[nframes]); boost::scoped_array mix_buffer (new Sample[nframes]); boost::shared_ptr diskstream = audio_diskstream(); Glib::RWLock::ReaderLock rlock (_processor_lock); boost::shared_ptr apl = boost::dynamic_pointer_cast(diskstream->playlist()); assert(apl); assert ((framecnt_t) buffers.get_audio(0).capacity() >= nframes); if (apl->read (buffers.get_audio(0).data(), mix_buffer.get(), gain_buffer.get(), start, nframes) != nframes) { return -1; } assert(buffers.count().n_audio() >= 1); uint32_t n=1; Sample* b = buffers.get_audio(0).data(); BufferSet::audio_iterator bi = buffers.audio_begin(); ++bi; for ( ; bi != buffers.audio_end(); ++bi, ++n) { if (n < diskstream->n_channels().n_audio()) { if (apl->read (bi->data(), mix_buffer.get(), gain_buffer.get(), start, nframes, n) != nframes) { return -1; } b = bi->data(); } else { /* duplicate last across remaining buffers */ memcpy (bi->data(), b, sizeof (Sample) * nframes); } } // If no processing is required, there's no need to go any further. if (!enable_processing) { return 0; } /* note: only run processors during export. other layers in the machinery will already have checked that there are no external port processors. Also, don't run deliveries that write to real output ports, and don't run meters. */ for (ProcessorList::iterator i = _processors.begin(); i != _processors.end(); ++i) { boost::shared_ptr processor = boost::dynamic_pointer_cast (*i); boost::shared_ptr delivery = boost::dynamic_pointer_cast (*i); boost::shared_ptr meter = boost::dynamic_pointer_cast (*i); if (processor && (!delivery || !Delivery::role_requires_output_ports (delivery->role())) && !meter) { processor->run (buffers, start, start+nframes, nframes, true); } } return 0; } boost::shared_ptr AudioTrack::bounce (InterThreadInfo& itt) { vector > srcs; return _session.write_one_track (*this, _session.current_start_frame(), _session.current_end_frame(), false, srcs, itt); } boost::shared_ptr AudioTrack::bounce_range (nframes_t start, nframes_t end, InterThreadInfo& itt, bool enable_processing) { vector > srcs; return _session.write_one_track (*this, start, end, false, srcs, itt, enable_processing); } void AudioTrack::freeze_me (InterThreadInfo& itt) { vector > srcs; string new_playlist_name; boost::shared_ptr new_playlist; string dir; string region_name; boost::shared_ptr diskstream = audio_diskstream(); if ((_freeze_record.playlist = boost::dynamic_pointer_cast(diskstream->playlist())) == 0) { return; } uint32_t n = 1; while (n < (UINT_MAX-1)) { string candidate; candidate = string_compose ("%1", _freeze_record.playlist->name(), n); if (_session.playlists->by_name (candidate) == 0) { new_playlist_name = candidate; break; } ++n; } if (n == (UINT_MAX-1)) { error << string_compose (X_("There are too many frozen versions of playlist \"%1\"" " to create another one"), _freeze_record.playlist->name()) << endmsg; return; } boost::shared_ptr res; if ((res = _session.write_one_track (*this, _session.current_start_frame(), _session.current_end_frame(), true, srcs, itt)) == 0) { return; } _freeze_record.processor_info.clear (); { Glib::RWLock::ReaderLock lm (_processor_lock); for (ProcessorList::iterator r = _processors.begin(); r != _processors.end(); ++r) { boost::shared_ptr processor; if ((processor = boost::dynamic_pointer_cast(*r)) != 0) { FreezeRecordProcessorInfo* frii = new FreezeRecordProcessorInfo ((*r)->get_state(), processor); frii->id = processor->id(); _freeze_record.processor_info.push_back (frii); /* now deactivate the processor */ processor->deactivate (); _session.set_dirty (); } } } new_playlist = PlaylistFactory::create (DataType::AUDIO, _session, new_playlist_name, false); /* XXX need main outs automation state _freeze_record.pan_automation_state = _mainpanner->automation_state(); */ region_name = new_playlist_name; /* create a new region from all filesources, keep it private */ PropertyList plist; plist.add (Properties::start, 0); plist.add (Properties::length, srcs[0]->length(srcs[0]->timeline_position())); plist.add (Properties::name, region_name); plist.add (Properties::whole_file, true); boost::shared_ptr region (RegionFactory::create (srcs, plist, false)); new_playlist->set_orig_diskstream_id (_diskstream->id()); new_playlist->add_region (region, _session.current_start_frame()); new_playlist->set_frozen (true); region->set_locked (true); diskstream->use_playlist (boost::dynamic_pointer_cast(new_playlist)); diskstream->set_record_enabled (false); /* reset stuff that has already been accounted for in the freeze process */ set_gain (1.0, this); _amp->gain_control()->set_automation_state (Off); /* XXX need to use _main_outs _panner->set_automation_state (Off); */ _freeze_record.state = Frozen; FreezeChange(); /* EMIT SIGNAL */ } void AudioTrack::unfreeze () { if (_freeze_record.playlist) { audio_diskstream()->use_playlist (_freeze_record.playlist); { Glib::RWLock::ReaderLock lm (_processor_lock); // should this be a write lock? jlc for (ProcessorList::iterator i = _processors.begin(); i != _processors.end(); ++i) { for (vector::iterator ii = _freeze_record.processor_info.begin(); ii != _freeze_record.processor_info.end(); ++ii) { if ((*ii)->id == (*i)->id()) { (*i)->set_state (((*ii)->state), Stateful::current_state_version); break; } } } } _freeze_record.playlist.reset (); /* XXX need to use _main_outs _panner->set_automation_state (_freeze_record.pan_automation_state); */ } _freeze_record.state = UnFrozen; FreezeChange (); /* EMIT SIGNAL */ } boost::shared_ptr AudioTrack::write_source (uint32_t n) { boost::shared_ptr ds = boost::dynamic_pointer_cast (_diskstream); assert (ds); return ds->write_source (n); }