comparison data/fileio/BQAFileReader.cpp @ 1605:d83ab62cdc28

Merge from branch bqaudiostream
author Chris Cannam
date Wed, 30 Jan 2019 14:57:12 +0000
parents 8aa1447fe27e
children 14747f24ad04
comparison
equal deleted inserted replaced
1586:841b2a3e606d 1605:d83ab62cdc28
1 /* -*- c-basic-offset: 4 indent-tabs-mode: nil -*- vi:set ts=8 sts=4 sw=4: */
2
3 /*
4 Sonic Visualiser
5 An audio file viewer and annotation editor.
6 Centre for Digital Music, Queen Mary, University of London.
7
8 This program is free software; you can redistribute it and/or
9 modify it under the terms of the GNU General Public License as
10 published by the Free Software Foundation; either version 2 of the
11 License, or (at your option) any later version. See the file
12 COPYING included with this distribution for more information.
13 */
14
15 #include "BQAFileReader.h"
16
17 #include <bqaudiostream/AudioReadStreamFactory.h>
18 #include <bqaudiostream/AudioReadStream.h>
19 #include <bqaudiostream/Exceptions.h>
20
21 #include "base/Profiler.h"
22 #include "base/ProgressReporter.h"
23
24 #include <QFileInfo>
25
26 using namespace std;
27
28 BQAFileReader::BQAFileReader(FileSource source,
29 DecodeMode decodeMode,
30 CacheMode mode,
31 sv_samplerate_t targetRate,
32 bool normalised,
33 ProgressReporter *reporter) :
34 CodedAudioFileReader(mode, targetRate, normalised),
35 m_source(source),
36 m_path(source.getLocalFilename()),
37 m_cancelled(false),
38 m_completion(0),
39 m_reporter(reporter),
40 m_decodeThread(0)
41 {
42 SVDEBUG << "BQAFileReader: local path: \"" << m_path
43 << "\", decode mode: " << decodeMode << " ("
44 << (decodeMode == DecodeAtOnce ? "DecodeAtOnce" : "DecodeThreaded")
45 << ")" << endl;
46
47 m_channelCount = 0;
48 m_fileRate = 0;
49
50 Profiler profiler("BQAFileReader::BQAFileReader");
51
52 try {
53 m_stream = breakfastquay::AudioReadStreamFactory::createReadStream
54 (m_path.toUtf8().data());
55 } catch (const std::exception &e) {
56 m_error = e.what();
57 SVDEBUG << "BQAFileReader: createReadStream failed: " << m_error << endl;
58 m_stream = 0;
59 return;
60 }
61
62 m_channelCount = int(m_stream->getChannelCount());
63 m_fileRate = sv_samplerate_t(m_stream->getSampleRate());
64 m_title = QString::fromUtf8(m_stream->getTrackName().c_str());
65 m_maker = QString::fromUtf8(m_stream->getArtistName().c_str());
66
67 initialiseDecodeCache();
68
69 if (decodeMode == DecodeAtOnce) {
70
71 if (m_reporter) {
72 connect(m_reporter, SIGNAL(cancelled()), this, SLOT(cancelled()));
73 m_reporter->setMessage
74 (tr("Decoding %1...").arg(QFileInfo(m_path).fileName()));
75 }
76
77 sv_frame_t blockSize = 65536;
78 floatvec_t block(blockSize * m_channelCount, 0.f);
79
80 while (true) {
81 try {
82 sv_frame_t retrieved =
83 m_stream->getInterleavedFrames(blockSize, block.data());
84
85 addSamplesToDecodeCache(block.data(), retrieved);
86
87 if (retrieved < blockSize) {
88 break;
89 }
90 } catch (const breakfastquay::InvalidFileFormat &f) {
91 m_error = f.what();
92 SVDEBUG << "BQAFileReader: init failed: " << m_error << endl;
93 break;
94 }
95
96 if (m_cancelled) break;
97 }
98
99 if (isDecodeCacheInitialised()) finishDecodeCache();
100 endSerialised();
101
102 if (m_reporter) m_reporter->setProgress(100);
103
104 delete m_stream;
105 m_stream = 0;
106
107 } else {
108
109 if (m_reporter) m_reporter->setProgress(100);
110
111 m_decodeThread = new DecodeThread(this);
112 m_decodeThread->start();
113 }
114 }
115
116 BQAFileReader::~BQAFileReader()
117 {
118 if (m_decodeThread) {
119 m_cancelled = true;
120 m_decodeThread->wait();
121 delete m_decodeThread;
122 }
123
124 delete m_stream;
125 }
126
127 void
128 BQAFileReader::cancelled()
129 {
130 m_cancelled = true;
131 }
132
133 void
134 BQAFileReader::DecodeThread::run()
135 {
136 if (m_reader->m_cacheMode == CacheInTemporaryFile) {
137 m_reader->startSerialised("BQAFileReader::Decode");
138 }
139
140 sv_frame_t blockSize = 65536;
141 floatvec_t block(blockSize * m_reader->getChannelCount(), 0.f);
142
143 while (true) {
144 try {
145 sv_frame_t retrieved =
146 m_reader->m_stream->getInterleavedFrames
147 (blockSize, block.data());
148
149 m_reader->addSamplesToDecodeCache(block.data(), retrieved);
150
151 if (retrieved < blockSize) {
152 break;
153 }
154 } catch (const breakfastquay::InvalidFileFormat &f) {
155 m_reader->m_error = f.what();
156 SVDEBUG << "BQAFileReader: decode failed: " << m_reader->m_error << endl;
157 break;
158 }
159
160 if (m_reader->m_cancelled) break;
161 }
162
163 if (m_reader->isDecodeCacheInitialised()) m_reader->finishDecodeCache();
164 m_reader->m_completion = 100;
165
166 m_reader->endSerialised();
167
168 delete m_reader->m_stream;
169 m_reader->m_stream = 0;
170 }
171
172 void
173 BQAFileReader::getSupportedExtensions(set<QString> &extensions)
174 {
175 vector<string> exts =
176 breakfastquay::AudioReadStreamFactory::getSupportedFileExtensions();
177 for (auto e: exts) {
178 extensions.insert(QString::fromUtf8(e.c_str()));
179 }
180 }
181
182 bool
183 BQAFileReader::supportsExtension(QString extension)
184 {
185 set<QString> extensions;
186 getSupportedExtensions(extensions);
187 return (extensions.find(extension.toLower()) != extensions.end());
188 }
189
190 bool
191 BQAFileReader::supportsContentType(QString type)
192 {
193 // extremely optimistic, but it's better than rejecting everything
194 //!!! todo: be more sensible
195 return (type.startsWith("audio/"));
196 }
197
198 bool
199 BQAFileReader::supports(FileSource &source)
200 {
201 return (supportsExtension(source.getExtension()) ||
202 supportsContentType(source.getContentType()));
203 }
204