X-Git-Url: https://main.carlh.net/gitweb/?p=dcpomatic.git;a=blobdiff_plain;f=src%2Flib%2Faudio_analysis.cc;h=13917cb5f6d693f3987b0c8adb85526a6426747c;hp=1863e38eeae85bf633c79151dc335fd4e3d9be09;hb=5eb8b5c3a1566aef638e9d9df03b88d320735092;hpb=a8a0dfd1b21de6c0facf965ab119833ff6f790bf diff --git a/src/lib/audio_analysis.cc b/src/lib/audio_analysis.cc index 1863e38ee..13917cb5f 100644 --- a/src/lib/audio_analysis.cc +++ b/src/lib/audio_analysis.cc @@ -1,5 +1,5 @@ /* - Copyright (C) 2012-2015 Carl Hetherington + Copyright (C) 2012-2018 Carl Hetherington This file is part of DCP-o-matic. @@ -21,9 +21,9 @@ #include "audio_analysis.h" #include "cross.h" #include "util.h" -#include "raw_convert.h" #include "playlist.h" #include "audio_content.h" +#include #include #include #include @@ -39,9 +39,16 @@ using std::string; using std::vector; using std::cout; using std::max; +using std::pair; +using std::make_pair; using std::list; using boost::shared_ptr; +using boost::optional; using boost::dynamic_pointer_cast; +using dcp::raw_convert; +using namespace dcpomatic; + +int const AudioAnalysis::_current_state_version = 3; AudioAnalysis::AudioAnalysis (int channels) { @@ -53,6 +60,11 @@ AudioAnalysis::AudioAnalysis (boost::filesystem::path filename) cxml::Document f ("AudioAnalysis"); f.read_file (filename); + if (f.optional_number_child("Version").get_value_or(1) < _current_state_version) { + /* Too old. Throw an exception so that this analysis is re-run. */ + throw OldFormatError ("Audio analysis file is too old"); + } + BOOST_FOREACH (cxml::NodePtr i, f.node_children ("Channel")) { vector channel; @@ -63,23 +75,24 @@ AudioAnalysis::AudioAnalysis (boost::filesystem::path filename) _data.push_back (channel); } - _sample_peak = f.optional_number_child ("Peak"); - if (!_sample_peak) { - /* New key */ - _sample_peak = f.optional_number_child ("SamplePeak"); + BOOST_FOREACH (cxml::ConstNodePtr i, f.node_children ("SamplePeak")) { + _sample_peak.push_back ( + PeakTime ( + dcp::raw_convert(i->content()), DCPTime(i->number_attribute("Time")) + ) + ); } - if (f.optional_number_child ("PeakTime")) { - _sample_peak_time = DCPTime (f.number_child ("PeakTime")); - } else if (f.optional_number_child ("SamplePeakTime")) { - _sample_peak_time = DCPTime (f.number_child ("SamplePeakTime")); + BOOST_FOREACH (cxml::ConstNodePtr i, f.node_children ("TruePeak")) { + _true_peak.push_back (dcp::raw_convert (i->content ())); } - _true_peak = f.optional_number_child ("TruePeak"); _integrated_loudness = f.optional_number_child ("IntegratedLoudness"); _loudness_range = f.optional_number_child ("LoudnessRange"); _analysis_gain = f.optional_number_child ("AnalysisGain"); + _samples_per_point = f.number_child ("SamplesPerPoint"); + _sample_rate = f.number_child ("SampleRate"); } void @@ -115,6 +128,8 @@ AudioAnalysis::write (boost::filesystem::path filename) shared_ptr doc (new xmlpp::Document); xmlpp::Element* root = doc->create_root_node ("AudioAnalysis"); + root->add_child("Version")->add_child_text (raw_convert (_current_state_version)); + BOOST_FOREACH (vector& i, _data) { xmlpp::Element* channel = root->add_child ("Channel"); BOOST_FOREACH (AudioPoint& j, i) { @@ -122,13 +137,14 @@ AudioAnalysis::write (boost::filesystem::path filename) } } - if (_sample_peak) { - root->add_child("SamplePeak")->add_child_text (raw_convert (_sample_peak.get ())); - root->add_child("SamplePeakTime")->add_child_text (raw_convert (_sample_peak_time.get().get ())); + for (size_t i = 0; i < _sample_peak.size(); ++i) { + xmlpp::Element* n = root->add_child("SamplePeak"); + n->add_child_text (raw_convert (_sample_peak[i].peak)); + n->set_attribute ("Time", raw_convert (_sample_peak[i].time.get())); } - if (_true_peak) { - root->add_child("TruePeak")->add_child_text (raw_convert (_true_peak.get ())); + BOOST_FOREACH (float i, _true_peak) { + root->add_child("TruePeak")->add_child_text (raw_convert (i)); } if (_integrated_loudness) { @@ -143,6 +159,9 @@ AudioAnalysis::write (boost::filesystem::path filename) root->add_child("AnalysisGain")->add_child_text (raw_convert (_analysis_gain.get ())); } + root->add_child("SamplesPerPoint")->add_child_text (raw_convert (_samples_per_point)); + root->add_child("SampleRate")->add_child_text (raw_convert (_sample_rate)); + doc->write_to_file_formatted (filename.string ()); } @@ -160,3 +179,36 @@ AudioAnalysis::gain_correction (shared_ptr playlist) return 0.0f; } + +/** @return Peak across all channels, and the channel number it is on */ +pair +AudioAnalysis::overall_sample_peak () const +{ + DCPOMATIC_ASSERT (!_sample_peak.empty ()); + + optional pt; + int c = 0; + + for (size_t i = 0; i < _sample_peak.size(); ++i) { + if (!pt || _sample_peak[i].peak > pt->peak) { + pt = _sample_peak[i]; + c = i; + } + } + + return make_pair (pt.get(), c); +} + +optional +AudioAnalysis::overall_true_peak () const +{ + optional p; + + BOOST_FOREACH (float i, _true_peak) { + if (!p || i > *p) { + p = i; + } + } + + return p; +}