comparison data/model/FFTModel.cpp @ 1766:85b9b466a59f

Merge from branch by-id
author Chris Cannam
date Wed, 17 Jul 2019 14:24:51 +0100
parents 6d09d68165a4
children fadd9f8aaa27
comparison
equal deleted inserted replaced
1730:649ac57c5a2d 1766:85b9b466a59f
30 using namespace std; 30 using namespace std;
31 31
32 static HitCount inSmallCache("FFTModel: Small FFT cache"); 32 static HitCount inSmallCache("FFTModel: Small FFT cache");
33 static HitCount inSourceCache("FFTModel: Source data cache"); 33 static HitCount inSourceCache("FFTModel: Source data cache");
34 34
35 FFTModel::FFTModel(const DenseTimeValueModel *model, 35 FFTModel::FFTModel(ModelId modelId,
36 int channel, 36 int channel,
37 WindowType windowType, 37 WindowType windowType,
38 int windowSize, 38 int windowSize,
39 int windowIncrement, 39 int windowIncrement,
40 int fftSize) : 40 int fftSize) :
41 m_model(model), 41 m_model(modelId),
42 m_channel(channel), 42 m_channel(channel),
43 m_windowType(windowType), 43 m_windowType(windowType),
44 m_windowSize(windowSize), 44 m_windowSize(windowSize),
45 m_windowIncrement(windowIncrement), 45 m_windowIncrement(windowIncrement),
46 m_fftSize(fftSize), 46 m_fftSize(fftSize),
59 throw invalid_argument("FFTModel window size may not exceed FFT size"); 59 throw invalid_argument("FFTModel window size may not exceed FFT size");
60 } 60 }
61 61
62 m_fft.initFloat(); 62 m_fft.initFloat();
63 63
64 connect(model, SIGNAL(modelChanged()), 64 auto model = ModelById::getAs<DenseTimeValueModel>(m_model);
65 this, SIGNAL(modelChanged())); 65 if (model) {
66 connect(model, SIGNAL(modelChangedWithin(sv_frame_t, sv_frame_t)), 66 connect(model.get(), SIGNAL(modelChanged(ModelId)),
67 this, SIGNAL(modelChangedWithin(sv_frame_t, sv_frame_t))); 67 this, SIGNAL(modelChanged(ModelId)));
68 connect(model, SIGNAL(aboutToBeDeleted()), 68 connect(model.get(), SIGNAL(modelChangedWithin(ModelId, sv_frame_t, sv_frame_t)),
69 this, SLOT(sourceModelAboutToBeDeleted())); 69 this, SIGNAL(modelChangedWithin(ModelId, sv_frame_t, sv_frame_t)));
70 }
70 } 71 }
71 72
72 FFTModel::~FFTModel() 73 FFTModel::~FFTModel()
73 { 74 {
74 } 75 }
75 76
76 void 77 bool
77 FFTModel::sourceModelAboutToBeDeleted() 78 FFTModel::isOK() const
78 { 79 {
79 if (m_model) { 80 auto model = ModelById::getAs<DenseTimeValueModel>(m_model);
80 SVDEBUG << "FFTModel[" << this << "]::sourceModelAboutToBeDeleted(" << m_model << ")" << endl; 81 return (model && model->isOK());
81 m_model = nullptr; 82 }
82 } 83
84 int
85 FFTModel::getCompletion() const
86 {
87 int c = 100;
88 auto model = ModelById::getAs<DenseTimeValueModel>(m_model);
89 if (model) {
90 if (model->isReady(&c)) return 100;
91 }
92 return c;
93 }
94
95 sv_samplerate_t
96 FFTModel::getSampleRate() const
97 {
98 auto model = ModelById::getAs<DenseTimeValueModel>(m_model);
99 if (model) return model->getSampleRate();
100 else return 0;
83 } 101 }
84 102
85 int 103 int
86 FFTModel::getWidth() const 104 FFTModel::getWidth() const
87 { 105 {
88 if (!m_model) return 0; 106 auto model = ModelById::getAs<DenseTimeValueModel>(m_model);
89 return int((m_model->getEndFrame() - m_model->getStartFrame()) 107 if (!model) return 0;
108 return int((model->getEndFrame() - model->getStartFrame())
90 / m_windowIncrement) + 1; 109 / m_windowIncrement) + 1;
91 } 110 }
92 111
93 int 112 int
94 FFTModel::getHeight() const 113 FFTModel::getHeight() const
275 FFTModel::fvec 294 FFTModel::fvec
276 FFTModel::getSourceDataUncached(pair<sv_frame_t, sv_frame_t> range) const 295 FFTModel::getSourceDataUncached(pair<sv_frame_t, sv_frame_t> range) const
277 { 296 {
278 Profiler profiler("FFTModel::getSourceDataUncached"); 297 Profiler profiler("FFTModel::getSourceDataUncached");
279 298
280 if (!m_model) return {}; 299 auto model = ModelById::getAs<DenseTimeValueModel>(m_model);
300 if (!model) return {};
281 301
282 decltype(range.first) pfx = 0; 302 decltype(range.first) pfx = 0;
283 if (range.first < 0) { 303 if (range.first < 0) {
284 pfx = -range.first; 304 pfx = -range.first;
285 range = { 0, range.second }; 305 range = { 0, range.second };
286 } 306 }
287 307
288 auto data = m_model->getData(m_channel, 308 auto data = model->getData(m_channel,
289 range.first, 309 range.first,
290 range.second - range.first); 310 range.second - range.first);
291 311
292 if (data.empty()) { 312 if (data.empty()) {
293 SVDEBUG << "NOTE: empty source data for range (" << range.first << "," 313 SVDEBUG << "NOTE: empty source data for range (" << range.first << ","
294 << range.second << ") (model end frame " 314 << range.second << ") (model end frame "
295 << m_model->getEndFrame() << ")" << endl; 315 << model->getEndFrame() << ")" << endl;
296 } 316 }
297 317
298 // don't return a partial frame 318 // don't return a partial frame
299 data.resize(range.second - range.first, 0.f); 319 data.resize(range.second - range.first, 0.f);
300 320
302 vector<float> pad(pfx, 0.f); 322 vector<float> pad(pfx, 0.f);
303 data.insert(data.begin(), pad.begin(), pad.end()); 323 data.insert(data.begin(), pad.begin(), pad.end());
304 } 324 }
305 325
306 if (m_channel == -1) { 326 if (m_channel == -1) {
307 int channels = m_model->getChannelCount(); 327 int channels = model->getChannelCount();
308 if (channels > 1) { 328 if (channels > 1) {
309 int n = int(data.size()); 329 int n = int(data.size());
310 float factor = 1.f / float(channels); 330 float factor = 1.f / float(channels);
311 // use mean instead of sum for fft model input 331 // use mean instead of sum for fft model input
312 for (int i = 0; i < n; ++i) { 332 for (int i = 0; i < n; ++i) {