X-Git-Url: https://main.carlh.net/gitweb/?a=blobdiff_plain;f=libs%2Fardour%2Faudiofilesource.cc;h=9c1b969190c530d8cb2ec95b066b1b02b0c349a1;hb=ac9219a3c884b69352ff5ab0d13f30fb15cf8e6e;hp=f2f09b71fc8262e666d28f6382b2765e8154f90e;hpb=51693a3a5835a7c2deb16242d14d73fbb7881382;p=ardour.git diff --git a/libs/ardour/audiofilesource.cc b/libs/ardour/audiofilesource.cc index f2f09b71fc..9c1b969190 100644 --- a/libs/ardour/audiofilesource.cc +++ b/libs/ardour/audiofilesource.cc @@ -32,25 +32,25 @@ #include "pbd/convert.h" #include "pbd/basename.h" +#include "pbd/file_utils.h" #include "pbd/mountpoint.h" #include "pbd/stl_delete.h" #include "pbd/strsplit.h" #include "pbd/shortpath.h" +#include "pbd/stacktrace.h" #include "pbd/enumwriter.h" #include +#include #include #include -#include +#include #include "ardour/audiofilesource.h" #include "ardour/debug.h" -#include "ardour/sndfile_helpers.h" #include "ardour/sndfilesource.h" #include "ardour/session.h" -#include "ardour/session_directory.h" -#include "ardour/source_factory.h" #include "ardour/filename_extensions.h" // if these headers come before sigc++ is included @@ -68,55 +68,72 @@ using namespace ARDOUR; using namespace PBD; using namespace Glib; -ustring AudioFileSource::peak_dir = ""; +string AudioFileSource::peak_dir = ""; -sigc::signal AudioFileSource::HeaderPositionOffsetChanged; -uint64_t AudioFileSource::header_position_offset = 0; +PBD::Signal0 AudioFileSource::HeaderPositionOffsetChanged; +framecnt_t AudioFileSource::header_position_offset = 0; /* XXX maybe this too */ char AudioFileSource::bwf_serial_number[13] = "000000000000"; struct SizedSampleBuffer { - nframes_t size; - Sample* buf; + framecnt_t size; + Sample* buf; - SizedSampleBuffer (nframes_t sz) : size (sz) { - buf = new Sample[size]; - } + SizedSampleBuffer (framecnt_t sz) : size (sz) { + buf = new Sample[size]; + } - ~SizedSampleBuffer() { - delete [] buf; - } + ~SizedSampleBuffer() { + delete [] buf; + } }; -Glib::StaticPrivate thread_interleave_buffer = GLIBMM_STATIC_PRIVATE_INIT; +Glib::Threads::Private thread_interleave_buffer; -/** Constructor used for existing internal-to-session files. */ -AudioFileSource::AudioFileSource (Session& s, const ustring& path, bool embedded, Source::Flag flags) +/** Constructor used for existing external-to-session files. */ +AudioFileSource::AudioFileSource (Session& s, const string& path, Source::Flag flags) : Source (s, DataType::AUDIO, path, flags) , AudioSource (s, path) - , FileSource (s, DataType::AUDIO, path, embedded, flags) + /* note that external files have their own path as "origin" */ + , FileSource (s, DataType::AUDIO, path, path, flags) { - if (init (path, true)) { + if (init (_path, true)) { throw failed_constructor (); } } /** Constructor used for new internal-to-session files. */ -AudioFileSource::AudioFileSource (Session& s, const ustring& path, bool embedded, Source::Flag flags, +AudioFileSource::AudioFileSource (Session& s, const string& path, const string& origin, Source::Flag flags, SampleFormat /*samp_format*/, HeaderFormat /*hdr_format*/) : Source (s, DataType::AUDIO, path, flags) , AudioSource (s, path) - , FileSource (s, DataType::AUDIO, path, embedded, flags) + , FileSource (s, DataType::AUDIO, path, origin, flags) { - _is_embedded = false; + /* note that origin remains empty */ - if (init (path, false)) { + if (init (_path, false)) { throw failed_constructor (); } } -/** Constructor used for existing internal-to-session files. File must exist. */ +/** Constructor used for existing internal-to-session files during crash + * recovery. File must exist + */ +AudioFileSource::AudioFileSource (Session& s, const string& path, Source::Flag flags, bool /* ignored-exists-for-prototype differentiation */) + : Source (s, DataType::AUDIO, path, flags) + , AudioSource (s, path) + , FileSource (s, DataType::AUDIO, path, string(), flags) +{ + /* note that origin remains empty */ + + if (init (_path, true)) { + throw failed_constructor (); + } +} + + +/** Constructor used for existing internal-to-session files via XML. File must exist. */ AudioFileSource::AudioFileSource (Session& s, const XMLNode& node, bool must_exist) : Source (s, node) , AudioSource (s, node) @@ -126,7 +143,7 @@ AudioFileSource::AudioFileSource (Session& s, const XMLNode& node, bool must_exi throw failed_constructor (); } - if (init (_name, must_exist)) { + if (init (_path, must_exist)) { throw failed_constructor (); } } @@ -135,22 +152,21 @@ AudioFileSource::~AudioFileSource () { DEBUG_TRACE (DEBUG::Destruction, string_compose ("AudioFileSource destructor %1, removable? %2\n", _path, removable())); if (removable()) { - unlink (_path.c_str()); - unlink (peakpath.c_str()); + ::g_unlink (_path.c_str()); + ::g_unlink (peakpath.c_str()); } } int -AudioFileSource::init (const ustring& pathstr, bool must_exist) +AudioFileSource::init (const string& pathstr, bool must_exist) { - _peaks_built = false; return FileSource::init (pathstr, must_exist); } -ustring -AudioFileSource::peak_path (ustring audio_path) +string +AudioFileSource::peak_path (string audio_path) { - ustring base; + string base; base = PBD::basename_nosuffix (audio_path); base += '%'; @@ -159,10 +175,10 @@ AudioFileSource::peak_path (ustring audio_path) return _session.peak_path (base); } -ustring -AudioFileSource::find_broken_peakfile (ustring peak_path, ustring audio_path) +string +AudioFileSource::find_broken_peakfile (string peak_path, string audio_path) { - ustring str; + string str; /* check for the broken location in use by 2.0 for several months */ @@ -170,7 +186,7 @@ AudioFileSource::find_broken_peakfile (ustring peak_path, ustring audio_path) if (Glib::file_test (str, Glib::FILE_TEST_EXISTS)) { - if (is_embedded()) { + if (!within_session()) { /* it would be nice to rename it but the nature of the bug means that we can't reliably use it. @@ -200,21 +216,21 @@ AudioFileSource::find_broken_peakfile (ustring peak_path, ustring audio_path) return peak_path; } -ustring -AudioFileSource::broken_peak_path (ustring audio_path) +string +AudioFileSource::broken_peak_path (string audio_path) { - return _session.peak_path (audio_path); + return _session.peak_path (basename_nosuffix (audio_path)); } -ustring -AudioFileSource::old_peak_path (ustring audio_path) +string +AudioFileSource::old_peak_path (string audio_path) { /* XXX hardly bombproof! fix me */ struct stat stat_file; struct stat stat_mount; - ustring mp = mountpoint (audio_path); + string mp = mountpoint (audio_path); stat (audio_path.c_str(), &stat_file); stat (mp.c_str(), &stat_mount); @@ -223,10 +239,10 @@ AudioFileSource::old_peak_path (ustring audio_path) #ifdef __APPLE__ snprintf (buf, sizeof (buf), "%u-%u-%d.peak", stat_mount.st_ino, stat_file.st_ino, _channel); #else - snprintf (buf, sizeof (buf), "%ld-%ld-%d.peak", stat_mount.st_ino, stat_file.st_ino, _channel); + snprintf (buf, sizeof (buf), "%" PRId64 "-%" PRId64 "-%d.peak", (int64_t) stat_mount.st_ino, (int64_t) stat_file.st_ino, _channel); #endif - ustring res = peak_dir; + string res = peak_dir; res += buf; res += peakfile_suffix; @@ -234,17 +250,21 @@ AudioFileSource::old_peak_path (ustring audio_path) } bool -AudioFileSource::get_soundfile_info (ustring path, SoundFileInfo& _info, string& error_msg) +AudioFileSource::get_soundfile_info (string path, SoundFileInfo& _info, string& error_msg) { -#ifdef HAVE_COREAUDIO - if (CoreAudioSource::get_soundfile_info (path, _info, error_msg) == 0) { + /* try sndfile first because it gets timecode info from .wav (BWF) if it exists, + which at present, ExtAudioFile from Apple seems unable to do. + */ + + if (SndFileSource::get_soundfile_info (path, _info, error_msg) != 0) { return true; } -#endif // HAVE_COREAUDIO - if (SndFileSource::get_soundfile_info (path, _info, error_msg) != 0) { +#ifdef HAVE_COREAUDIO + if (CoreAudioSource::get_soundfile_info (path, _info, error_msg) == 0) { return true; } +#endif // HAVE_COREAUDIO return false; } @@ -256,6 +276,7 @@ AudioFileSource::get_state () char buf[32]; snprintf (buf, sizeof (buf), "%u", _channel); root.add_property (X_("channel"), buf); + root.add_property (X_("origin"), _origin); return root; } @@ -284,32 +305,24 @@ AudioFileSource::mark_streaming_write_completed () return; } - /* XXX notice that we're readers of _peaks_built - but we must hold a solid lock on PeaksReady. - */ - - Glib::Mutex::Lock lm (_lock); - - if (_peaks_built) { - PeaksReady (); /* EMIT SIGNAL */ - } + AudioSource::mark_streaming_write_completed (); } int AudioFileSource::move_dependents_to_trash() { - return ::unlink (peakpath.c_str()); + return ::g_unlink (peakpath.c_str()); } void -AudioFileSource::set_header_position_offset (nframes_t offset) +AudioFileSource::set_header_position_offset (framecnt_t offset) { header_position_offset = offset; HeaderPositionOffsetChanged (); } bool -AudioFileSource::is_empty (Session& /*s*/, ustring path) +AudioFileSource::is_empty (Session& /*s*/, string path) { SoundFileInfo info; string err; @@ -326,37 +339,52 @@ int AudioFileSource::setup_peakfile () { if (!(_flags & NoPeakFile)) { - return initialize_peakfile (_file_is_new, _path); + return initialize_peakfile (_path); } else { return 0; } } bool -AudioFileSource::safe_audio_file_extension(const ustring& file) +AudioFileSource::safe_audio_file_extension(const string& file) { const char* suffixes[] = { - ".wav", ".WAV", - ".aiff", ".AIFF", - ".caf", ".CAF", ".aif", ".AIF", + ".aifc", ".AIFC", + ".aiff", ".AIFF", ".amb", ".AMB", - ".snd", ".SND", ".au", ".AU", - ".raw", ".RAW", - ".sf", ".SF", + ".caf", ".CAF", ".cdr", ".CDR", + ".flac", ".FLAC", + ".htk", ".HTK", + ".iff", ".IFF", + ".mat", ".MAT", + ".oga", ".OGA", + ".ogg", ".OGG", + ".paf", ".PAF", + ".pvf", ".PVF", + ".sf", ".SF", ".smp", ".SMP", + ".snd", ".SND", ".maud", ".MAUD", + ".voc", ".VOC" ".vwe", ".VWE", - ".paf", ".PAF", - ".voc", ".VOC", - ".ogg", ".OGG", - ".flac", ".FLAC", + ".w64", ".W64", + ".wav", ".WAV", #ifdef HAVE_COREAUDIO - ".mp3", ".MP3", ".aac", ".AAC", + ".adts", ".ADTS", + ".ac3", ".AC3", + ".amr", ".AMR", + ".mpa", ".MPA", + ".mpeg", ".MPEG", + ".mp1", ".MP1", + ".mp2", ".MP2", + ".mp3", ".MP3", ".mp4", ".MP4", + ".m4a", ".M4A", + ".sd2", ".SD2", // libsndfile supports sd2 also, but the resource fork is required to open. #endif // HAVE_COREAUDIO }; @@ -370,7 +398,7 @@ AudioFileSource::safe_audio_file_extension(const ustring& file) } Sample* -AudioFileSource::get_interleave_buffer (nframes_t size) +AudioFileSource::get_interleave_buffer (framecnt_t size) { SizedSampleBuffer* ssb; @@ -386,3 +414,4 @@ AudioFileSource::get_interleave_buffer (nframes_t size) return ssb->buf; } +