Stop SndfileDecoder with no audio emitting lots of silence.
[dcpomatic.git] / src / lib / sndfile_decoder.cc
1 /*
2     Copyright (C) 2012 Carl Hetherington <cth@carlh.net>
3
4     This program is free software; you can redistribute it and/or modify
5     it under the terms of the GNU General Public License as published by
6     the Free Software Foundation; either version 2 of the License, or
7     (at your option) any later version.
8
9     This program is distributed in the hope that it will be useful,
10     but WITHOUT ANY WARRANTY; without even the implied warranty of
11     MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
12     GNU General Public License for more details.
13
14     You should have received a copy of the GNU General Public License
15     along with this program; if not, write to the Free Software
16     Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
17
18 */
19
20 #include <iostream>
21 #include <sndfile.h>
22 #include "sndfile_decoder.h"
23 #include "film.h"
24 #include "exceptions.h"
25
26 #include "i18n.h"
27
28 using std::vector;
29 using std::string;
30 using std::stringstream;
31 using std::min;
32 using std::cout;
33 using boost::shared_ptr;
34 using boost::optional;
35
36 SndfileDecoder::SndfileDecoder (shared_ptr<Film> f, DecodeOptions o)
37         : Decoder (f, o)
38         , AudioDecoder (f, o)
39         , _done (0)
40         , _frames (0)
41 {
42         _done = 0;
43         _frames = 0;
44         
45         vector<string> const files = _film->external_audio ();
46
47         int N = 0;
48         for (size_t i = 0; i < files.size(); ++i) {
49                 if (!files[i].empty()) {
50                         N = i + 1;
51                 }
52         }
53
54         if (N == 0) {
55                 return;
56         }
57
58         bool first = true;
59         
60         for (size_t i = 0; i < (size_t) N; ++i) {
61                 if (files[i].empty ()) {
62                         _sndfiles.push_back (0);
63                 } else {
64                         SF_INFO info;
65                         SNDFILE* s = sf_open (files[i].c_str(), SFM_READ, &info);
66                         if (!s) {
67                                 throw DecodeError (_("could not open external audio file for reading"));
68                         }
69
70                         if (info.channels != 1) {
71                                 throw DecodeError (_("external audio files must be mono"));
72                         }
73                         
74                         _sndfiles.push_back (s);
75
76                         if (first) {
77                                 shared_ptr<SndfileStream> st (
78                                         new SndfileStream (
79                                                 info.samplerate, av_get_default_channel_layout (N)
80                                                 )
81                                         );
82                                 
83                                 _audio_streams.push_back (st);
84                                 _audio_stream = st;
85                                 _frames = info.frames;
86                                 first = false;
87                         } else {
88                                 if (info.frames != _frames) {
89                                         throw DecodeError (_("external audio files have differing lengths"));
90                                 }
91                         }
92                 }
93         }
94 }
95
96 bool
97 SndfileDecoder::pass ()
98 {
99         /* Do things in half second blocks as I think there may be limits
100            to what FFmpeg (and in particular the resampler) can cope with.
101         */
102         sf_count_t const block = _audio_stream->sample_rate() / 2;
103         shared_ptr<AudioBuffers> audio (new AudioBuffers (_audio_stream->channels(), block));
104         sf_count_t const this_time = min (block, _frames - _done);
105         bool have_sound = false;
106         for (size_t i = 0; i < _sndfiles.size(); ++i) {
107                 if (!_sndfiles[i]) {
108                         audio->make_silent (i);
109                 } else {
110                         sf_read_float (_sndfiles[i], audio->data(i), this_time);
111                         have_sound = true;
112                 }
113         }
114
115         if (!have_sound) {
116                 return true;
117         }
118
119         audio->set_frames (this_time);
120         Audio (audio, double(_done) / _audio_stream->sample_rate());
121         _done += this_time;
122
123         return (_done == _frames);
124 }
125
126 SndfileDecoder::~SndfileDecoder ()
127 {
128         for (size_t i = 0; i < _sndfiles.size(); ++i) {
129                 if (_sndfiles[i]) {
130                         sf_close (_sndfiles[i]);
131                 }
132         }
133 }
134
135 shared_ptr<SndfileStream>
136 SndfileStream::create ()
137 {
138         return shared_ptr<SndfileStream> (new SndfileStream);
139 }
140
141 shared_ptr<SndfileStream>
142 SndfileStream::create (string t, optional<int> v)
143 {
144         if (!v) {
145                 /* version < 1; no type in the string, and there's only FFmpeg streams anyway */
146                 return shared_ptr<SndfileStream> ();
147         }
148
149         stringstream s (t);
150         string type;
151         s >> type;
152         if (type != N_("external")) {
153                 return shared_ptr<SndfileStream> ();
154         }
155
156         return shared_ptr<SndfileStream> (new SndfileStream (t, v));
157 }
158
159 SndfileStream::SndfileStream (string t, optional<int> v)
160 {
161         assert (v);
162
163         stringstream s (t);
164         string type;
165         s >> type >> _sample_rate >> _channel_layout;
166 }
167
168 SndfileStream::SndfileStream ()
169 {
170
171 }
172
173 string
174 SndfileStream::to_string () const
175 {
176         return String::compose (N_("external %1 %2"), _sample_rate, _channel_layout);
177 }