/* * Copyright (C) 2011-2012 David Robillard * Copyright (C) 2011-2017 Paul Davis * Copyright (C) 2012-2016 Tim Mayberry * Copyright (C) 2014-2016 Robin Gareus * * 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., * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA. */ #ifdef WAF_BUILD #include "libardour-config.h" #endif #include #include #include #include #include "pbd/error.h" #include "ardour/audioplaylist.h" #include "ardour/audio_playlist_source.h" #include "ardour/audioregion.h" #include "ardour/filename_extensions.h" #include "ardour/session.h" #include "ardour/session_directory.h" #include "pbd/i18n.h" using namespace std; using namespace ARDOUR; using namespace PBD; AudioPlaylistSource::AudioPlaylistSource (Session& s, const ID& orig, const std::string& name, boost::shared_ptr p, uint32_t chn, sampleoffset_t begin, samplecnt_t len, Source::Flag flags) : Source (s, DataType::AUDIO, name) , PlaylistSource (s, orig, name, p, DataType::AUDIO, begin, len, flags) , AudioSource (s, name) , _playlist_channel (chn) { AudioSource::_length = len; ensure_buffers_for_level (_level, _session.sample_rate()); } AudioPlaylistSource::AudioPlaylistSource (Session& s, const XMLNode& node) : Source (s, node) , PlaylistSource (s, node) , AudioSource (s, node) { /* PlaylistSources are never writable, renameable or removable */ _flags = Flag (_flags & ~(Writable|CanRename|Removable|RemovableIfEmpty|RemoveAtDestroy)); /* ancestors have already called ::set_state() in their XML-based constructors. */ if (set_state (node, Stateful::loading_state_version, false)) { throw failed_constructor (); } AudioSource::_length = _playlist_length; } AudioPlaylistSource::~AudioPlaylistSource () { } XMLNode& AudioPlaylistSource::get_state () { XMLNode& node (AudioSource::get_state ()); /* merge PlaylistSource state */ PlaylistSource::add_state (node); node.set_property ("channel", _playlist_channel); return node; } int AudioPlaylistSource::set_state (const XMLNode& node, int version) { return set_state (node, version, true); } int AudioPlaylistSource::set_state (const XMLNode& node, int version, bool with_descendants) { if (with_descendants) { if (Source::set_state (node, version) || PlaylistSource::set_state (node, version) || AudioSource::set_state (node, version)) { return -1; } } pair extent = _playlist->get_extent(); AudioSource::_length = extent.second - extent.first; if (!node.get_property (X_("channel"), _playlist_channel)) { throw failed_constructor (); } ensure_buffers_for_level (_level, _session.sample_rate()); return 0; } samplecnt_t AudioPlaylistSource::read_unlocked (Sample* dst, samplepos_t start, samplecnt_t cnt) const { boost::shared_array sbuf; boost::shared_array gbuf; samplecnt_t to_read; samplecnt_t to_zero; /* we must be careful not to read beyond the end of our "section" of * the playlist, because otherwise we may read data that exists, but * is not supposed be part of our data. */ if (cnt > _playlist_length - start) { to_read = _playlist_length - start; to_zero = cnt - to_read; } else { to_read = cnt; to_zero = 0; } { /* Don't need to hold the lock for the actual read, and actually, we cannot, but we do want to interlock with any changes to the list of buffers caused by creating new nested playlists/sources */ Glib::Threads::Mutex::Lock lm (_level_buffer_lock); sbuf = _mixdown_buffers[_level-1]; gbuf = _gain_buffers[_level-1]; } boost::dynamic_pointer_cast(_playlist)->read (dst, sbuf.get(), gbuf.get(), start+_playlist_offset, to_read, _playlist_channel); if (to_zero) { memset (dst+to_read, 0, sizeof (Sample) * to_zero); } return cnt; } samplecnt_t AudioPlaylistSource::write_unlocked (Sample *, samplecnt_t) { fatal << string_compose (_("programming error: %1"), "AudioPlaylistSource::write() called - should be impossible") << endmsg; abort(); /*NOTREACHED*/ return 0; } bool AudioPlaylistSource::empty () const { return !_playlist || _playlist->empty(); } uint32_t AudioPlaylistSource::n_channels () const { /* use just the first region to decide */ if (empty()) { return 1; } boost::shared_ptr r = _playlist->region_list_property().front (); boost::shared_ptr ar = boost::dynamic_pointer_cast (r); return ar->audio_source()->n_channels (); } float AudioPlaylistSource::sample_rate () const { /* use just the first region to decide */ if (empty()) { _session.sample_rate (); } boost::shared_ptr r = _playlist->region_list_property().front (); boost::shared_ptr ar = boost::dynamic_pointer_cast (r); return ar->audio_source()->sample_rate (); } int AudioPlaylistSource::setup_peakfile () { _peak_path = Glib::build_filename (_session.session_directory().peak_path(), name() + ARDOUR::peakfile_suffix); return initialize_peakfile (string()); } string AudioPlaylistSource::construct_peak_filepath (const string& /*audio_path_*/, const bool /* in_session */, const bool /* old_peak_name */) const { return _peak_path; }