comparison data/model/FFTModel.cpp @ 936:0c1d6de8f44b

Merge from branch warnfix_no_size_t
author Chris Cannam
date Wed, 18 Jun 2014 13:51:16 +0100
parents 59e7fe1b1003
children a8aed8a85e09
comparison
equal deleted inserted replaced
917:49618f39ff09 936:0c1d6de8f44b
29 #endif 29 #endif
30 30
31 FFTModel::FFTModel(const DenseTimeValueModel *model, 31 FFTModel::FFTModel(const DenseTimeValueModel *model,
32 int channel, 32 int channel,
33 WindowType windowType, 33 WindowType windowType,
34 size_t windowSize, 34 int windowSize,
35 size_t windowIncrement, 35 int windowIncrement,
36 size_t fftSize, 36 int fftSize,
37 bool polar, 37 bool polar,
38 StorageAdviser::Criteria criteria, 38 StorageAdviser::Criteria criteria,
39 size_t fillFromColumn) : 39 int fillFromColumn) :
40 //!!! ZoomConstraint! 40 //!!! ZoomConstraint!
41 m_server(0), 41 m_server(0),
42 m_xshift(0), 42 m_xshift(0),
43 m_yshift(0) 43 m_yshift(0)
44 { 44 {
54 criteria, 54 criteria,
55 fillFromColumn); 55 fillFromColumn);
56 56
57 if (!m_server) return; // caller should check isOK() 57 if (!m_server) return; // caller should check isOK()
58 58
59 size_t xratio = windowIncrement / m_server->getWindowIncrement(); 59 int xratio = windowIncrement / m_server->getWindowIncrement();
60 size_t yratio = m_server->getFFTSize() / fftSize; 60 int yratio = m_server->getFFTSize() / fftSize;
61 61
62 while (xratio > 1) { 62 while (xratio > 1) {
63 if (xratio & 0x1) { 63 if (xratio & 0x1) {
64 cerr << "ERROR: FFTModel: Window increment ratio " 64 cerr << "ERROR: FFTModel: Window increment ratio "
65 << windowIncrement << " / " 65 << windowIncrement << " / "
103 103
104 FFTDataServer * 104 FFTDataServer *
105 FFTModel::getServer(const DenseTimeValueModel *model, 105 FFTModel::getServer(const DenseTimeValueModel *model,
106 int channel, 106 int channel,
107 WindowType windowType, 107 WindowType windowType,
108 size_t windowSize, 108 int windowSize,
109 size_t windowIncrement, 109 int windowIncrement,
110 size_t fftSize, 110 int fftSize,
111 bool polar, 111 bool polar,
112 StorageAdviser::Criteria criteria, 112 StorageAdviser::Criteria criteria,
113 size_t fillFromColumn) 113 int fillFromColumn)
114 { 114 {
115 // Obviously, an FFT model of channel C (where C != -1) of an 115 // Obviously, an FFT model of channel C (where C != -1) of an
116 // aggregate model is the same as the FFT model of the appropriate 116 // aggregate model is the same as the FFT model of the appropriate
117 // channel of whichever model that aggregate channel is drawn 117 // channel of whichever model that aggregate channel is drawn
118 // from. We should use that model here, in case we already have 118 // from. We should use that model here, in case we already have
157 polar, 157 polar,
158 criteria, 158 criteria,
159 fillFromColumn); 159 fillFromColumn);
160 } 160 }
161 161
162 size_t 162 int
163 FFTModel::getSampleRate() const 163 FFTModel::getSampleRate() const
164 { 164 {
165 return isOK() ? m_server->getModel()->getSampleRate() : 0; 165 return isOK() ? m_server->getModel()->getSampleRate() : 0;
166 } 166 }
167 167
168 FFTModel::Column 168 FFTModel::Column
169 FFTModel::getColumn(size_t x) const 169 FFTModel::getColumn(int x) const
170 { 170 {
171 Profiler profiler("FFTModel::getColumn", false); 171 Profiler profiler("FFTModel::getColumn", false);
172 172
173 Column result; 173 Column result;
174 174
175 result.clear(); 175 result.clear();
176 size_t h = getHeight(); 176 int h = getHeight();
177 result.reserve(h); 177 result.reserve(h);
178 178
179 #ifdef __GNUC__ 179 #ifdef __GNUC__
180 float magnitudes[h]; 180 float magnitudes[h];
181 #else 181 #else
182 float *magnitudes = (float *)alloca(h * sizeof(float)); 182 float *magnitudes = (float *)alloca(h * sizeof(float));
183 #endif 183 #endif
184 184
185 if (m_server->getMagnitudesAt(x << m_xshift, magnitudes)) { 185 if (m_server->getMagnitudesAt(x << m_xshift, magnitudes)) {
186 186
187 for (size_t y = 0; y < h; ++y) { 187 for (int y = 0; y < h; ++y) {
188 result.push_back(magnitudes[y]); 188 result.push_back(magnitudes[y]);
189 } 189 }
190 190
191 } else { 191 } else {
192 for (size_t i = 0; i < h; ++i) result.push_back(0.f); 192 for (int i = 0; i < h; ++i) result.push_back(0.f);
193 } 193 }
194 194
195 return result; 195 return result;
196 } 196 }
197 197
198 QString 198 QString
199 FFTModel::getBinName(size_t n) const 199 FFTModel::getBinName(int n) const
200 { 200 {
201 size_t sr = getSampleRate(); 201 int sr = getSampleRate();
202 if (!sr) return ""; 202 if (!sr) return "";
203 QString name = tr("%1 Hz").arg((n * sr) / ((getHeight()-1) * 2)); 203 QString name = tr("%1 Hz").arg((n * sr) / ((getHeight()-1) * 2));
204 return name; 204 return name;
205 } 205 }
206 206
207 bool 207 bool
208 FFTModel::estimateStableFrequency(size_t x, size_t y, float &frequency) 208 FFTModel::estimateStableFrequency(int x, int y, float &frequency)
209 { 209 {
210 if (!isOK()) return false; 210 if (!isOK()) return false;
211 211
212 size_t sampleRate = m_server->getModel()->getSampleRate(); 212 int sampleRate = m_server->getModel()->getSampleRate();
213 213
214 size_t fftSize = m_server->getFFTSize() >> m_yshift; 214 int fftSize = m_server->getFFTSize() >> m_yshift;
215 frequency = (float(y) * sampleRate) / fftSize; 215 frequency = (float(y) * sampleRate) / fftSize;
216 216
217 if (x+1 >= getWidth()) return false; 217 if (x+1 >= getWidth()) return false;
218 218
219 // At frequency f, a phase shift of 2pi (one cycle) happens in 1/f sec. 219 // At frequency f, a phase shift of 2pi (one cycle) happens in 1/f sec.
226 // = 2pi * (h * b) / w. 226 // = 2pi * (h * b) / w.
227 227
228 float oldPhase = getPhaseAt(x, y); 228 float oldPhase = getPhaseAt(x, y);
229 float newPhase = getPhaseAt(x+1, y); 229 float newPhase = getPhaseAt(x+1, y);
230 230
231 size_t incr = getResolution(); 231 int incr = getResolution();
232 232
233 float expectedPhase = oldPhase + (2.0 * M_PI * y * incr) / fftSize; 233 float expectedPhase = oldPhase + (2.0 * M_PI * y * incr) / fftSize;
234 234
235 float phaseError = princargf(newPhase - expectedPhase); 235 float phaseError = princargf(newPhase - expectedPhase);
236 236
245 245
246 return true; 246 return true;
247 } 247 }
248 248
249 FFTModel::PeakLocationSet 249 FFTModel::PeakLocationSet
250 FFTModel::getPeaks(PeakPickType type, size_t x, size_t ymin, size_t ymax) 250 FFTModel::getPeaks(PeakPickType type, int x, int ymin, int ymax)
251 { 251 {
252 Profiler profiler("FFTModel::getPeaks"); 252 Profiler profiler("FFTModel::getPeaks");
253 253
254 FFTModel::PeakLocationSet peaks; 254 FFTModel::PeakLocationSet peaks;
255 if (!isOK()) return peaks; 255 if (!isOK()) return peaks;
268 float values[n]; 268 float values[n];
269 #else 269 #else
270 float *values = (float *)alloca(n * sizeof(float)); 270 float *values = (float *)alloca(n * sizeof(float));
271 #endif 271 #endif
272 getMagnitudesAt(x, values, minbin, maxbin - minbin + 1); 272 getMagnitudesAt(x, values, minbin, maxbin - minbin + 1);
273 for (size_t bin = ymin; bin <= ymax; ++bin) { 273 for (int bin = ymin; bin <= ymax; ++bin) {
274 if (bin == minbin || bin == maxbin) continue; 274 if (bin == minbin || bin == maxbin) continue;
275 if (values[bin - minbin] > values[bin - minbin - 1] && 275 if (values[bin - minbin] > values[bin - minbin - 1] &&
276 values[bin - minbin] > values[bin - minbin + 1]) { 276 values[bin - minbin] > values[bin - minbin + 1]) {
277 peaks.insert(bin); 277 peaks.insert(bin);
278 } 278 }
289 // For peak picking we use a moving median window, picking the 289 // For peak picking we use a moving median window, picking the
290 // highest value within each continuous region of values that 290 // highest value within each continuous region of values that
291 // exceed the median. For pitch adaptivity, we adjust the window 291 // exceed the median. For pitch adaptivity, we adjust the window
292 // size to a roughly constant pitch range (about four tones). 292 // size to a roughly constant pitch range (about four tones).
293 293
294 size_t sampleRate = getSampleRate(); 294 int sampleRate = getSampleRate();
295 295
296 std::deque<float> window; 296 std::deque<float> window;
297 std::vector<size_t> inrange; 297 std::vector<int> inrange;
298 float dist = 0.5; 298 float dist = 0.5;
299 299
300 size_t medianWinSize = getPeakPickWindowSize(type, sampleRate, ymin, dist); 300 int medianWinSize = getPeakPickWindowSize(type, sampleRate, ymin, dist);
301 size_t halfWin = medianWinSize/2; 301 int halfWin = medianWinSize/2;
302 302
303 size_t binmin; 303 int binmin;
304 if (ymin > halfWin) binmin = ymin - halfWin; 304 if (ymin > halfWin) binmin = ymin - halfWin;
305 else binmin = 0; 305 else binmin = 0;
306 306
307 size_t binmax; 307 int binmax;
308 if (ymax + halfWin < values.size()) binmax = ymax + halfWin; 308 if (ymax + halfWin < values.size()) binmax = ymax + halfWin;
309 else binmax = values.size()-1; 309 else binmax = values.size()-1;
310 310
311 size_t prevcentre = 0; 311 int prevcentre = 0;
312 312
313 for (size_t bin = binmin; bin <= binmax; ++bin) { 313 for (int bin = binmin; bin <= binmax; ++bin) {
314 314
315 float value = values[bin]; 315 float value = values[bin];
316 316
317 window.push_back(value); 317 window.push_back(value);
318 318
319 // so-called median will actually be the dist*100'th percentile 319 // so-called median will actually be the dist*100'th percentile
320 medianWinSize = getPeakPickWindowSize(type, sampleRate, bin, dist); 320 medianWinSize = getPeakPickWindowSize(type, sampleRate, bin, dist);
321 halfWin = medianWinSize/2; 321 halfWin = medianWinSize/2;
322 322
323 while (window.size() > medianWinSize) { 323 while ((int)window.size() > medianWinSize) {
324 window.pop_front(); 324 window.pop_front();
325 } 325 }
326 326
327 size_t actualSize = window.size(); 327 int actualSize = window.size();
328 328
329 if (type == MajorPitchAdaptivePeaks) { 329 if (type == MajorPitchAdaptivePeaks) {
330 if (ymax + halfWin < values.size()) binmax = ymax + halfWin; 330 if (ymax + halfWin < values.size()) binmax = ymax + halfWin;
331 else binmax = values.size()-1; 331 else binmax = values.size()-1;
332 } 332 }
333 333
334 std::deque<float> sorted(window); 334 std::deque<float> sorted(window);
335 std::sort(sorted.begin(), sorted.end()); 335 std::sort(sorted.begin(), sorted.end());
336 float median = sorted[int(sorted.size() * dist)]; 336 float median = sorted[int(sorted.size() * dist)];
337 337
338 size_t centrebin = 0; 338 int centrebin = 0;
339 if (bin > actualSize/2) centrebin = bin - actualSize/2; 339 if (bin > actualSize/2) centrebin = bin - actualSize/2;
340 340
341 while (centrebin > prevcentre || bin == binmin) { 341 while (centrebin > prevcentre || bin == binmin) {
342 342
343 if (centrebin > prevcentre) ++prevcentre; 343 if (centrebin > prevcentre) ++prevcentre;
348 inrange.push_back(centrebin); 348 inrange.push_back(centrebin);
349 } 349 }
350 350
351 if (centre <= median || centrebin+1 == values.size()) { 351 if (centre <= median || centrebin+1 == values.size()) {
352 if (!inrange.empty()) { 352 if (!inrange.empty()) {
353 size_t peakbin = 0; 353 int peakbin = 0;
354 float peakval = 0.f; 354 float peakval = 0.f;
355 for (size_t i = 0; i < inrange.size(); ++i) { 355 for (int i = 0; i < (int)inrange.size(); ++i) {
356 if (i == 0 || values[inrange[i]] > peakval) { 356 if (i == 0 || values[inrange[i]] > peakval) {
357 peakval = values[inrange[i]]; 357 peakval = values[inrange[i]];
358 peakbin = inrange[i]; 358 peakbin = inrange[i];
359 } 359 }
360 } 360 }
370 } 370 }
371 371
372 return peaks; 372 return peaks;
373 } 373 }
374 374
375 size_t 375 int
376 FFTModel::getPeakPickWindowSize(PeakPickType type, size_t sampleRate, 376 FFTModel::getPeakPickWindowSize(PeakPickType type, int sampleRate,
377 size_t bin, float &percentile) const 377 int bin, float &percentile) const
378 { 378 {
379 percentile = 0.5; 379 percentile = 0.5;
380 if (type == MajorPeaks) return 10; 380 if (type == MajorPeaks) return 10;
381 if (bin == 0) return 3; 381 if (bin == 0) return 3;
382 382
383 size_t fftSize = m_server->getFFTSize() >> m_yshift; 383 int fftSize = m_server->getFFTSize() >> m_yshift;
384 float binfreq = (sampleRate * bin) / fftSize; 384 float binfreq = (sampleRate * bin) / fftSize;
385 float hifreq = Pitch::getFrequencyForPitch(73, 0, binfreq); 385 float hifreq = Pitch::getFrequencyForPitch(73, 0, binfreq);
386 386
387 int hibin = lrintf((hifreq * fftSize) / sampleRate); 387 int hibin = lrintf((hifreq * fftSize) / sampleRate);
388 int medianWinSize = hibin - bin; 388 int medianWinSize = hibin - bin;
392 392
393 return medianWinSize; 393 return medianWinSize;
394 } 394 }
395 395
396 FFTModel::PeakSet 396 FFTModel::PeakSet
397 FFTModel::getPeakFrequencies(PeakPickType type, size_t x, 397 FFTModel::getPeakFrequencies(PeakPickType type, int x,
398 size_t ymin, size_t ymax) 398 int ymin, int ymax)
399 { 399 {
400 Profiler profiler("FFTModel::getPeakFrequencies"); 400 Profiler profiler("FFTModel::getPeakFrequencies");
401 401
402 PeakSet peaks; 402 PeakSet peaks;
403 if (!isOK()) return peaks; 403 if (!isOK()) return peaks;
404 PeakLocationSet locations = getPeaks(type, x, ymin, ymax); 404 PeakLocationSet locations = getPeaks(type, x, ymin, ymax);
405 405
406 size_t sampleRate = getSampleRate(); 406 int sampleRate = getSampleRate();
407 size_t fftSize = m_server->getFFTSize() >> m_yshift; 407 int fftSize = m_server->getFFTSize() >> m_yshift;
408 size_t incr = getResolution(); 408 int incr = getResolution();
409 409
410 // This duplicates some of the work of estimateStableFrequency to 410 // This duplicates some of the work of estimateStableFrequency to
411 // allow us to retrieve the phases in two separate vertical 411 // allow us to retrieve the phases in two separate vertical
412 // columns, instead of jumping back and forth between columns x and 412 // columns, instead of jumping back and forth between columns x and
413 // x+1, which may be significantly slower if re-seeking is needed 413 // x+1, which may be significantly slower if re-seeking is needed
416 for (PeakLocationSet::iterator i = locations.begin(); 416 for (PeakLocationSet::iterator i = locations.begin();
417 i != locations.end(); ++i) { 417 i != locations.end(); ++i) {
418 phases.push_back(getPhaseAt(x, *i)); 418 phases.push_back(getPhaseAt(x, *i));
419 } 419 }
420 420
421 size_t phaseIndex = 0; 421 int phaseIndex = 0;
422 for (PeakLocationSet::iterator i = locations.begin(); 422 for (PeakLocationSet::iterator i = locations.begin();
423 i != locations.end(); ++i) { 423 i != locations.end(); ++i) {
424 float oldPhase = phases[phaseIndex]; 424 float oldPhase = phases[phaseIndex];
425 float newPhase = getPhaseAt(x+1, *i); 425 float newPhase = getPhaseAt(x+1, *i);
426 float expectedPhase = oldPhase + (2.0 * M_PI * *i * incr) / fftSize; 426 float expectedPhase = oldPhase + (2.0 * M_PI * *i * incr) / fftSize;