Mercurial > hg > easaier-soundaccess
comparison data/fileio/CodedAudioFileReader.cpp @ 0:fc9323a41f5a
start base : Sonic Visualiser sv1-1.0rc1
author | lbajardsilogic |
---|---|
date | Fri, 11 May 2007 09:08:14 +0000 |
parents | |
children |
comparison
equal
deleted
inserted
replaced
-1:000000000000 | 0:fc9323a41f5a |
---|---|
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 This file copyright 2006 Chris Cannam. | |
8 | |
9 This program is free software; you can redistribute it and/or | |
10 modify it under the terms of the GNU General Public License as | |
11 published by the Free Software Foundation; either version 2 of the | |
12 License, or (at your option) any later version. See the file | |
13 COPYING included with this distribution for more information. | |
14 */ | |
15 | |
16 #include "CodedAudioFileReader.h" | |
17 | |
18 #include "WavFileReader.h" | |
19 #include "base/TempDirectory.h" | |
20 #include "base/Exceptions.h" | |
21 #include "base/Profiler.h" | |
22 | |
23 #include <iostream> | |
24 #include <QDir> | |
25 | |
26 CodedAudioFileReader::CodedAudioFileReader(CacheMode cacheMode) : | |
27 m_cacheMode(cacheMode), | |
28 m_initialised(false), | |
29 m_cacheFileWritePtr(0), | |
30 m_cacheFileReader(0), | |
31 m_cacheWriteBuffer(0), | |
32 m_cacheWriteBufferIndex(0), | |
33 m_cacheWriteBufferSize(16384) | |
34 { | |
35 } | |
36 | |
37 CodedAudioFileReader::~CodedAudioFileReader() | |
38 { | |
39 if (m_cacheFileWritePtr) sf_close(m_cacheFileWritePtr); | |
40 if (m_cacheFileReader) delete m_cacheFileReader; | |
41 if (m_cacheWriteBuffer) delete[] m_cacheWriteBuffer; | |
42 | |
43 if (m_cacheFileName != "") { | |
44 if (!QFile(m_cacheFileName).remove()) { | |
45 std::cerr << "WARNING: CodedAudioFileReader::~CodedAudioFileReader: Failed to delete cache file \"" << m_cacheFileName.toStdString() << "\"" << std::endl; | |
46 } | |
47 } | |
48 } | |
49 | |
50 void | |
51 CodedAudioFileReader::initialiseDecodeCache() | |
52 { | |
53 if (m_cacheMode == CacheInTemporaryFile) { | |
54 | |
55 m_cacheWriteBuffer = new float[m_cacheWriteBufferSize * m_channelCount]; | |
56 m_cacheWriteBufferIndex = 0; | |
57 | |
58 try { | |
59 QDir dir(TempDirectory::getInstance()->getPath()); | |
60 m_cacheFileName = dir.filePath(QString("decoded_%1.wav") | |
61 .arg((intptr_t)this)); | |
62 | |
63 SF_INFO fileInfo; | |
64 fileInfo.samplerate = m_sampleRate; | |
65 fileInfo.channels = m_channelCount; | |
66 fileInfo.format = SF_FORMAT_WAV | SF_FORMAT_FLOAT; | |
67 | |
68 m_cacheFileWritePtr = sf_open(m_cacheFileName.toLocal8Bit(), | |
69 SFM_WRITE, &fileInfo); | |
70 | |
71 if (!m_cacheFileWritePtr) { | |
72 std::cerr << "CodedAudioFileReader::initialiseDecodeCache: failed to open cache file \"" << m_cacheFileName.toStdString() << "\" (" << m_channelCount << " channels, sample rate " << m_sampleRate << " for writing, falling back to in-memory cache" << std::endl; | |
73 m_cacheMode = CacheInMemory; | |
74 } | |
75 } catch (DirectoryCreationFailed f) { | |
76 std::cerr << "CodedAudioFileReader::initialiseDecodeCache: failed to create temporary directory! Falling back to in-memory cache" << std::endl; | |
77 m_cacheMode = CacheInMemory; | |
78 } | |
79 } | |
80 | |
81 if (m_cacheMode == CacheInMemory) { | |
82 m_data.clear(); | |
83 } | |
84 | |
85 m_initialised = true; | |
86 } | |
87 | |
88 void | |
89 CodedAudioFileReader::addSampleToDecodeCache(float sample) | |
90 { | |
91 if (!m_initialised) return; | |
92 | |
93 switch (m_cacheMode) { | |
94 | |
95 case CacheInTemporaryFile: | |
96 | |
97 m_cacheWriteBuffer[m_cacheWriteBufferIndex++] = sample; | |
98 | |
99 if (m_cacheWriteBufferIndex == | |
100 m_cacheWriteBufferSize * m_channelCount) { | |
101 | |
102 //!!! check for return value! out of disk space, etc! | |
103 sf_writef_float(m_cacheFileWritePtr, | |
104 m_cacheWriteBuffer, | |
105 m_cacheWriteBufferSize); | |
106 | |
107 m_cacheWriteBufferIndex = 0; | |
108 } | |
109 break; | |
110 | |
111 case CacheInMemory: | |
112 m_data.push_back(sample); | |
113 break; | |
114 } | |
115 } | |
116 | |
117 void | |
118 CodedAudioFileReader::finishDecodeCache() | |
119 { | |
120 Profiler profiler("CodedAudioFileReader::finishDecodeCache", true); | |
121 | |
122 if (!m_initialised) { | |
123 std::cerr << "WARNING: CodedAudioFileReader::finishDecodeCache: Cache was never initialised!" << std::endl; | |
124 return; | |
125 } | |
126 | |
127 switch (m_cacheMode) { | |
128 | |
129 case CacheInTemporaryFile: | |
130 | |
131 if (m_cacheWriteBufferIndex > 0) { | |
132 //!!! check for return value! out of disk space, etc! | |
133 sf_writef_float(m_cacheFileWritePtr, | |
134 m_cacheWriteBuffer, | |
135 m_cacheWriteBufferIndex / m_channelCount); | |
136 } | |
137 | |
138 if (m_cacheWriteBuffer) { | |
139 delete[] m_cacheWriteBuffer; | |
140 m_cacheWriteBuffer = 0; | |
141 } | |
142 | |
143 m_cacheWriteBufferIndex = 0; | |
144 | |
145 sf_close(m_cacheFileWritePtr); | |
146 m_cacheFileWritePtr = 0; | |
147 | |
148 m_cacheFileReader = new WavFileReader(m_cacheFileName); | |
149 | |
150 if (!m_cacheFileReader->isOK()) { | |
151 std::cerr << "ERROR: CodedAudioFileReader::finishDecodeCache: Failed to construct WAV file reader for temporary file: " << m_cacheFileReader->getError().toStdString() << std::endl; | |
152 delete m_cacheFileReader; | |
153 m_cacheFileReader = 0; | |
154 } | |
155 break; | |
156 | |
157 case CacheInMemory: | |
158 // nothing to do | |
159 break; | |
160 } | |
161 } | |
162 | |
163 void | |
164 CodedAudioFileReader::getInterleavedFrames(size_t start, size_t count, | |
165 SampleBlock &frames) const | |
166 { | |
167 if (!m_initialised) return; | |
168 | |
169 switch (m_cacheMode) { | |
170 | |
171 case CacheInTemporaryFile: | |
172 if (m_cacheFileReader) { | |
173 m_cacheFileReader->getInterleavedFrames(start, count, frames); | |
174 } | |
175 break; | |
176 | |
177 case CacheInMemory: | |
178 { | |
179 frames.clear(); | |
180 if (!isOK()) return; | |
181 if (count == 0) return; | |
182 | |
183 // slownessabounds | |
184 | |
185 for (size_t i = start; i < start + count; ++i) { | |
186 for (size_t ch = 0; ch < m_channelCount; ++ch) { | |
187 size_t index = i * m_channelCount + ch; | |
188 if (index >= m_data.size()) return; | |
189 frames.push_back(m_data[index]); | |
190 } | |
191 } | |
192 } | |
193 } | |
194 } | |
195 |