Chris@148
|
1 /* -*- c-basic-offset: 4 indent-tabs-mode: nil -*- vi:set ts=8 sts=4 sw=4: */
|
Chris@148
|
2
|
Chris@148
|
3 /*
|
Chris@148
|
4 Sonic Visualiser
|
Chris@148
|
5 An audio file viewer and annotation editor.
|
Chris@148
|
6 Centre for Digital Music, Queen Mary, University of London.
|
Chris@202
|
7 This file copyright 2006 Chris Cannam and QMUL.
|
Chris@148
|
8
|
Chris@148
|
9 This program is free software; you can redistribute it and/or
|
Chris@148
|
10 modify it under the terms of the GNU General Public License as
|
Chris@148
|
11 published by the Free Software Foundation; either version 2 of the
|
Chris@148
|
12 License, or (at your option) any later version. See the file
|
Chris@148
|
13 COPYING included with this distribution for more information.
|
Chris@148
|
14 */
|
Chris@148
|
15
|
Chris@148
|
16 #ifndef _FFT_DATA_SERVER_H_
|
Chris@148
|
17 #define _FFT_DATA_SERVER_H_
|
Chris@148
|
18
|
Chris@148
|
19 #include "base/Window.h"
|
Chris@148
|
20 #include "base/Thread.h"
|
Chris@334
|
21 #include "base/StorageAdviser.h"
|
Chris@148
|
22
|
Chris@226
|
23 #include "FFTapi.h"
|
Chris@537
|
24 #include "FFTFileCacheReader.h"
|
Chris@537
|
25 #include "FFTFileCacheWriter.h"
|
Chris@537
|
26 #include "FFTMemoryCache.h"
|
Chris@148
|
27
|
Chris@148
|
28 #include <QMutex>
|
Chris@537
|
29 #include <QReadWriteLock>
|
Chris@537
|
30 #include <QReadLocker>
|
Chris@148
|
31 #include <QWaitCondition>
|
Chris@148
|
32 #include <QString>
|
Chris@148
|
33
|
Chris@148
|
34 #include <vector>
|
Chris@148
|
35 #include <deque>
|
Chris@148
|
36
|
Chris@148
|
37 class DenseTimeValueModel;
|
Chris@215
|
38 class Model;
|
Chris@148
|
39
|
Chris@148
|
40 class FFTDataServer
|
Chris@148
|
41 {
|
Chris@148
|
42 public:
|
Chris@148
|
43 static FFTDataServer *getInstance(const DenseTimeValueModel *model,
|
Chris@148
|
44 int channel,
|
Chris@148
|
45 WindowType windowType,
|
Chris@929
|
46 int windowSize,
|
Chris@929
|
47 int windowIncrement,
|
Chris@929
|
48 int fftSize,
|
Chris@148
|
49 bool polar,
|
Chris@334
|
50 StorageAdviser::Criteria criteria =
|
Chris@334
|
51 StorageAdviser::NoCriteria,
|
Chris@1045
|
52 sv_frame_t fillFromFrame = 0);
|
Chris@148
|
53
|
Chris@148
|
54 static FFTDataServer *getFuzzyInstance(const DenseTimeValueModel *model,
|
Chris@148
|
55 int channel,
|
Chris@148
|
56 WindowType windowType,
|
Chris@929
|
57 int windowSize,
|
Chris@929
|
58 int windowIncrement,
|
Chris@929
|
59 int fftSize,
|
Chris@148
|
60 bool polar,
|
Chris@334
|
61 StorageAdviser::Criteria criteria =
|
Chris@334
|
62 StorageAdviser::NoCriteria,
|
Chris@1045
|
63 sv_frame_t fillFromFrame = 0);
|
Chris@148
|
64
|
Chris@152
|
65 static void claimInstance(FFTDataServer *);
|
Chris@148
|
66 static void releaseInstance(FFTDataServer *);
|
Chris@148
|
67
|
Chris@215
|
68 static void modelAboutToBeDeleted(Model *);
|
Chris@215
|
69
|
Chris@148
|
70 const DenseTimeValueModel *getModel() const { return m_model; }
|
Chris@148
|
71 int getChannel() const { return m_channel; }
|
Chris@148
|
72 WindowType getWindowType() const { return m_windower.getType(); }
|
Chris@1081
|
73 int getWindowSize() const { return m_windowSize; }
|
Chris@1081
|
74 int getWindowIncrement() const { return m_windowIncrement; }
|
Chris@1081
|
75 int getFFTSize() const { return m_fftSize; }
|
Chris@148
|
76 bool getPolar() const { return m_polar; }
|
Chris@148
|
77
|
Chris@1081
|
78 int getWidth() const { return m_width; }
|
Chris@1081
|
79 int getHeight() const { return m_height; }
|
Chris@148
|
80
|
Chris@929
|
81 float getMagnitudeAt(int x, int y);
|
Chris@929
|
82 float getNormalizedMagnitudeAt(int x, int y);
|
Chris@929
|
83 float getMaximumMagnitudeAt(int x);
|
Chris@929
|
84 float getPhaseAt(int x, int y);
|
Chris@929
|
85 void getValuesAt(int x, int y, float &real, float &imaginary);
|
Chris@929
|
86 bool isColumnReady(int x);
|
Chris@148
|
87
|
Chris@929
|
88 bool getMagnitudesAt(int x, float *values, int minbin = 0, int count = 0, int step = 1);
|
Chris@929
|
89 bool getNormalizedMagnitudesAt(int x, float *values, int minbin = 0, int count = 0, int step = 1);
|
Chris@929
|
90 bool getPhasesAt(int x, float *values, int minbin = 0, int count = 0, int step = 1);
|
Chris@929
|
91 bool getValuesAt(int x, float *reals, float *imaginaries, int minbin = 0, int count = 0, int step = 1);
|
Chris@408
|
92
|
Chris@148
|
93 void suspend();
|
Chris@155
|
94 void suspendWrites();
|
Chris@154
|
95 void resume(); // also happens automatically if new data needed
|
Chris@148
|
96
|
Chris@148
|
97 // Convenience functions:
|
Chris@148
|
98
|
Chris@929
|
99 bool isLocalPeak(int x, int y) {
|
Chris@148
|
100 float mag = getMagnitudeAt(x, y);
|
Chris@148
|
101 if (y > 0 && mag < getMagnitudeAt(x, y - 1)) return false;
|
Chris@148
|
102 if (y < getHeight()-1 && mag < getMagnitudeAt(x, y + 1)) return false;
|
Chris@148
|
103 return true;
|
Chris@148
|
104 }
|
Chris@929
|
105 bool isOverThreshold(int x, int y, float threshold) {
|
Chris@148
|
106 return getMagnitudeAt(x, y) > threshold;
|
Chris@148
|
107 }
|
Chris@148
|
108
|
Chris@678
|
109 QString getError() const;
|
Chris@929
|
110 int getFillCompletion() const;
|
Chris@1038
|
111 sv_frame_t getFillExtent() const;
|
Chris@148
|
112
|
Chris@148
|
113 private:
|
Chris@148
|
114 FFTDataServer(QString fileBaseName,
|
Chris@148
|
115 const DenseTimeValueModel *model,
|
Chris@148
|
116 int channel,
|
Chris@148
|
117 WindowType windowType,
|
Chris@929
|
118 int windowSize,
|
Chris@929
|
119 int windowIncrement,
|
Chris@929
|
120 int fftSize,
|
Chris@148
|
121 bool polar,
|
Chris@334
|
122 StorageAdviser::Criteria criteria,
|
Chris@1045
|
123 sv_frame_t fillFromFrame = 0);
|
Chris@148
|
124
|
Chris@148
|
125 virtual ~FFTDataServer();
|
Chris@148
|
126
|
Chris@148
|
127 FFTDataServer(const FFTDataServer &); // not implemented
|
Chris@148
|
128 FFTDataServer &operator=(const FFTDataServer &); // not implemented
|
Chris@148
|
129
|
Chris@148
|
130 typedef float fftsample;
|
Chris@148
|
131
|
Chris@148
|
132 QString m_fileBaseName;
|
Chris@148
|
133 const DenseTimeValueModel *m_model;
|
Chris@148
|
134 int m_channel;
|
Chris@148
|
135
|
Chris@148
|
136 Window<fftsample> m_windower;
|
Chris@148
|
137
|
Chris@929
|
138 int m_windowSize;
|
Chris@929
|
139 int m_windowIncrement;
|
Chris@929
|
140 int m_fftSize;
|
Chris@148
|
141 bool m_polar;
|
Chris@148
|
142
|
Chris@929
|
143 int m_width;
|
Chris@929
|
144 int m_height;
|
Chris@929
|
145 int m_cacheWidth;
|
Chris@929
|
146 int m_cacheWidthPower;
|
Chris@929
|
147 int m_cacheWidthMask;
|
Chris@359
|
148
|
Chris@537
|
149 struct CacheBlock {
|
Chris@537
|
150 FFTMemoryCache *memoryCache;
|
Chris@537
|
151 typedef std::map<QThread *, FFTFileCacheReader *> ThreadReaderMap;
|
Chris@537
|
152 ThreadReaderMap fileCacheReader;
|
Chris@537
|
153 FFTFileCacheWriter *fileCacheWriter;
|
Chris@537
|
154 CacheBlock() : memoryCache(0), fileCacheWriter(0) { }
|
Chris@537
|
155 ~CacheBlock() {
|
Chris@537
|
156 delete memoryCache;
|
Chris@537
|
157 while (!fileCacheReader.empty()) {
|
Chris@537
|
158 delete fileCacheReader.begin()->second;
|
Chris@537
|
159 fileCacheReader.erase(fileCacheReader.begin());
|
Chris@537
|
160 }
|
Chris@537
|
161 delete fileCacheWriter;
|
Chris@537
|
162 }
|
Chris@537
|
163 };
|
Chris@537
|
164
|
Chris@537
|
165 typedef std::vector<CacheBlock *> CacheVector;
|
Chris@537
|
166 CacheVector m_caches;
|
Chris@537
|
167 QReadWriteLock m_cacheVectorLock; // locks cache lookup, not use
|
Chris@548
|
168 QMutex m_cacheCreationMutex; // solely to serialise makeCache() calls
|
Chris@537
|
169
|
Chris@929
|
170 FFTCacheReader *getCacheReader(int x, int &col) {
|
Chris@537
|
171 Profiler profiler("FFTDataServer::getCacheReader");
|
Chris@537
|
172 col = x & m_cacheWidthMask;
|
Chris@359
|
173 int c = x >> m_cacheWidthPower;
|
Chris@537
|
174 m_cacheVectorLock.lockForRead();
|
Chris@537
|
175 CacheBlock *cb(m_caches.at(c));
|
Chris@537
|
176 if (cb) {
|
Chris@547
|
177 if (cb->memoryCache) {
|
Chris@547
|
178 m_cacheVectorLock.unlock();
|
Chris@547
|
179 return cb->memoryCache;
|
Chris@547
|
180 }
|
Chris@537
|
181 if (cb->fileCacheWriter) {
|
Chris@537
|
182 QThread *me = QThread::currentThread();
|
Chris@537
|
183 CacheBlock::ThreadReaderMap &map = cb->fileCacheReader;
|
Chris@537
|
184 if (map.find(me) == map.end()) {
|
Chris@537
|
185 m_cacheVectorLock.unlock();
|
Chris@537
|
186 if (!makeCacheReader(c)) return 0;
|
Chris@537
|
187 return getCacheReader(x, col);
|
Chris@537
|
188 }
|
Chris@539
|
189 FFTCacheReader *reader = cb->fileCacheReader[me];
|
Chris@537
|
190 m_cacheVectorLock.unlock();
|
Chris@537
|
191 return reader;
|
Chris@537
|
192 }
|
Chris@537
|
193 // if cb exists but cb->fileCacheWriter doesn't, creation
|
Chris@537
|
194 // must have failed: don't try again
|
Chris@537
|
195 m_cacheVectorLock.unlock();
|
Chris@537
|
196 return 0;
|
Chris@537
|
197 }
|
Chris@537
|
198 m_cacheVectorLock.unlock();
|
Chris@1081
|
199 if (getError() != "") return 0;
|
Chris@537
|
200 if (!makeCache(c)) return 0;
|
Chris@537
|
201 return getCacheReader(x, col);
|
Chris@359
|
202 }
|
Chris@537
|
203
|
Chris@929
|
204 FFTCacheWriter *getCacheWriter(int x, int &col) {
|
Chris@537
|
205 Profiler profiler("FFTDataServer::getCacheWriter");
|
Chris@537
|
206 col = x & m_cacheWidthMask;
|
Chris@537
|
207 int c = x >> m_cacheWidthPower;
|
Chris@537
|
208 {
|
Chris@537
|
209 QReadLocker locker(&m_cacheVectorLock);
|
Chris@537
|
210 CacheBlock *cb(m_caches.at(c));
|
Chris@537
|
211 if (cb) {
|
Chris@537
|
212 if (cb->memoryCache) return cb->memoryCache;
|
Chris@537
|
213 if (cb->fileCacheWriter) return cb->fileCacheWriter;
|
Chris@537
|
214 // if cb exists, creation must have failed: don't try again
|
Chris@537
|
215 return 0;
|
Chris@537
|
216 }
|
Chris@537
|
217 }
|
Chris@537
|
218 if (!makeCache(c)) return 0;
|
Chris@537
|
219 return getCacheWriter(x, col);
|
Chris@537
|
220 }
|
Chris@537
|
221
|
Chris@929
|
222 bool haveCache(int x) {
|
Chris@359
|
223 int c = x >> m_cacheWidthPower;
|
Chris@548
|
224 return (m_caches.at(c) != 0);
|
Chris@359
|
225 }
|
Chris@148
|
226
|
Chris@537
|
227 bool makeCache(int c);
|
Chris@537
|
228 bool makeCacheReader(int c);
|
Chris@537
|
229
|
Chris@359
|
230 StorageAdviser::Criteria m_criteria;
|
Chris@359
|
231
|
Chris@929
|
232 void getStorageAdvice(int w, int h, bool &memory, bool &compact);
|
Chris@148
|
233
|
Chris@1081
|
234 mutable QMutex m_fftBuffersLock;
|
Chris@148
|
235 QWaitCondition m_condition;
|
Chris@148
|
236
|
Chris@148
|
237 fftsample *m_fftInput;
|
Chris@226
|
238 fftf_complex *m_fftOutput;
|
Chris@148
|
239 float *m_workbuffer;
|
Chris@226
|
240 fftf_plan m_fftPlan;
|
Chris@148
|
241
|
Chris@148
|
242 class FillThread : public Thread
|
Chris@148
|
243 {
|
Chris@148
|
244 public:
|
Chris@1045
|
245 FillThread(FFTDataServer &server, sv_frame_t fillFromFrame) :
|
Chris@148
|
246 m_server(server), m_extent(0), m_completion(0),
|
Chris@1045
|
247 m_fillFrom(fillFromFrame) { }
|
Chris@148
|
248
|
Chris@1038
|
249 sv_frame_t getExtent() const { return m_extent; }
|
Chris@929
|
250 int getCompletion() const { return m_completion ? m_completion : 1; }
|
Chris@1081
|
251 QString getError() const { return m_threadError; }
|
Chris@148
|
252 virtual void run();
|
Chris@148
|
253
|
Chris@148
|
254 protected:
|
Chris@148
|
255 FFTDataServer &m_server;
|
Chris@1038
|
256 sv_frame_t m_extent;
|
Chris@929
|
257 int m_completion;
|
Chris@1038
|
258 sv_frame_t m_fillFrom;
|
Chris@1081
|
259 QString m_threadError;
|
Chris@148
|
260 };
|
Chris@148
|
261
|
Chris@148
|
262 bool m_exiting;
|
Chris@148
|
263 bool m_suspended;
|
Chris@148
|
264 FillThread *m_fillThread;
|
Chris@678
|
265 QString m_error;
|
Chris@148
|
266
|
Chris@148
|
267 void deleteProcessingData();
|
Chris@929
|
268 void fillColumn(int x);
|
Chris@537
|
269 void fillComplete();
|
Chris@148
|
270
|
Chris@148
|
271 QString generateFileBasename() const;
|
Chris@148
|
272 static QString generateFileBasename(const DenseTimeValueModel *model,
|
Chris@148
|
273 int channel,
|
Chris@148
|
274 WindowType windowType,
|
Chris@929
|
275 int windowSize,
|
Chris@929
|
276 int windowIncrement,
|
Chris@929
|
277 int fftSize,
|
Chris@148
|
278 bool polar);
|
Chris@148
|
279
|
Chris@148
|
280 typedef std::pair<FFTDataServer *, int> ServerCountPair;
|
Chris@148
|
281 typedef std::map<QString, ServerCountPair> ServerMap;
|
Chris@215
|
282 typedef std::deque<FFTDataServer *> ServerQueue;
|
Chris@148
|
283
|
Chris@148
|
284 static ServerMap m_servers;
|
Chris@215
|
285 static ServerQueue m_releasedServers; // these are still in m_servers as well, with zero refcount
|
Chris@148
|
286 static QMutex m_serverMapMutex;
|
Chris@148
|
287 static FFTDataServer *findServer(QString); // call with serverMapMutex held
|
Chris@148
|
288 static void purgeLimbo(int maxSize = 3); // call with serverMapMutex held
|
Chris@216
|
289
|
Chris@216
|
290 static void claimInstance(FFTDataServer *, bool needLock);
|
Chris@216
|
291 static void releaseInstance(FFTDataServer *, bool needLock);
|
Chris@216
|
292
|
Chris@148
|
293 };
|
Chris@148
|
294
|
Chris@148
|
295 #endif
|