Mercurial > hg > svcore
diff data/model/test/TestWaveformOversampler.h @ 1551:4de4284d0596
Merge from branch zoom
author | Chris Cannam |
---|---|
date | Wed, 10 Oct 2018 08:44:15 +0100 |
parents | a53bf95f0bea |
children | 074b860a7828 |
line wrap: on
line diff
--- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/data/model/test/TestWaveformOversampler.h Wed Oct 10 08:44:15 2018 +0100 @@ -0,0 +1,251 @@ +/* -*- c-basic-offset: 4 indent-tabs-mode: nil -*- vi:set ts=8 sts=4 sw=4: */ + +/* + Sonic Visualiser + An audio file viewer and annotation editor. + Centre for Digital Music, Queen Mary, University of London. + + This program is free software; you can redistribute it and/or + modify it under the terms of the GNU General Public License as + published by the Free Software Foundation; either version 2 of the + License, or (at your option) any later version. See the file + COPYING included with this distribution for more information. +*/ + +#ifndef TEST_WAVEFORM_OVERSAMPLER_H +#define TEST_WAVEFORM_OVERSAMPLER_H + +#include "../WaveformOversampler.h" +#include "../WritableWaveFileModel.h" + +#include "../../../base/BaseTypes.h" + +#include <QObject> +#include <QtTest> + +class TestWaveformOversampler : public QObject +{ + Q_OBJECT + +public: + TestWaveformOversampler() { + m_source = floatvec_t(5000, 0.f); + m_source[0] = 1.f; + m_source[2500] = 0.5f; + m_source[2501] = -0.5f; + m_source[4999] = -1.f; + for (int i = 3000; i < 3900; ++i) { + m_source[i] = float(sin(double(i - 3000) * M_PI / 50.0)); + } + m_sourceModel = new WritableWaveFileModel(8000, 1); + const float *d = m_source.data(); + QVERIFY(m_sourceModel->addSamples(&d, m_source.size())); + m_sourceModel->writeComplete(); + } + + ~TestWaveformOversampler() { + delete m_sourceModel; + } + +private: + floatvec_t m_source; + WritableWaveFileModel *m_sourceModel; + + void compareStrided(floatvec_t obtained, floatvec_t expected, int stride) { + QCOMPARE(obtained.size(), expected.size() * stride); + float threshold = 1e-10f; + for (int i = 0; in_range_for(expected, i); ++i) { + if (fabsf(obtained[i * stride] - expected[i]) > threshold) { + std::cerr << "At position " << i * stride << ": " + << obtained[i * stride] << " != " << expected[i] + << std::endl; + QCOMPARE(obtained, expected); + } + } + } + + void compareVecs(floatvec_t obtained, floatvec_t expected) { + compareStrided(obtained, expected, 1); + } + + floatvec_t get(sv_frame_t sourceStartFrame, + sv_frame_t sourceFrameCount, + int oversampleBy) { + return WaveformOversampler::getOversampledData + (m_sourceModel, 0, + sourceStartFrame, sourceFrameCount, oversampleBy); + } + + void testVerbatim(sv_frame_t sourceStartFrame, + sv_frame_t sourceFrameCount, + int oversampleBy, + floatvec_t expected) { + floatvec_t output = + get(sourceStartFrame, sourceFrameCount, oversampleBy); + compareVecs(output, expected); + } + + void testStrided(sv_frame_t sourceStartFrame, + sv_frame_t sourceFrameCount, + int oversampleBy, + floatvec_t expected) { + // check only the values that are expected to be precisely the + // original samples + floatvec_t output = + get(sourceStartFrame, sourceFrameCount, oversampleBy); + compareStrided(output, expected, oversampleBy); + } + + floatvec_t sourceSubset(sv_frame_t start, sv_frame_t length) { + return floatvec_t(m_source.begin() + start, + m_source.begin() + start + length); + } + +private slots: + void testWholeVerbatim() { + testVerbatim(0, 5000, 1, m_source); + } + + void testSubsetsVerbatim() { + testVerbatim(0, 500, 1, sourceSubset(0, 500)); + testVerbatim(4500, 500, 1, sourceSubset(4500, 500)); + testVerbatim(2000, 1000, 1, sourceSubset(2000, 1000)); + } + + void testOverlapsVerbatim() { + // overlapping the start -> result should be zero-padded to + // preserve start frame + floatvec_t expected = sourceSubset(0, 400); + expected.insert(expected.begin(), 100, 0.f); + testVerbatim(-100, 500, 1, expected); + + // overlapping the end -> result should be truncated to + // preserve source length + expected = sourceSubset(4600, 400); + testVerbatim(4600, 500, 1, expected); + } + + void testWhole2x() { + testStrided(0, 5000, 2, m_source); + + // check for windowed sinc values between the original samples + floatvec_t output = get(0, 5000, 2); + QVERIFY(output[1] - 0.6358 < 0.0001); + QVERIFY(output[3] + 0.2099 < 0.0001); + } + + void testWhole3x() { + testStrided(0, 5000, 3, m_source); + + // check for windowed sinc values between the original samples + floatvec_t output = get(0, 5000, 3); + QVERIFY(output[1] > 0.7); + QVERIFY(output[2] > 0.4); + QVERIFY(output[4] < -0.1); + QVERIFY(output[5] < -0.1); + } + + void testWhole4x() { + testStrided(0, 5000, 4, m_source); + + // check for windowed sinc values between the original samples + floatvec_t output = get(0, 5000, 4); + QVERIFY(output[1] - 0.9000 < 0.0001); + QVERIFY(output[2] - 0.6358 < 0.0001); + QVERIFY(output[3] - 0.2993 < 0.0001); + QVERIFY(output[5] + 0.1787 < 0.0001); + QVERIFY(output[6] + 0.2099 < 0.0001); + QVERIFY(output[7] + 0.1267 < 0.0001); + + // alternate values at 2n should equal all values at n + output = get(0, 5000, 4); + floatvec_t half = get(0, 5000, 2); + compareStrided(output, half, 2); + } + + void testWhole8x() { + testStrided(0, 5000, 8, m_source); + + // alternate values at 2n should equal all values at n + floatvec_t output = get(0, 5000, 8); + floatvec_t half = get(0, 5000, 4); + compareStrided(output, half, 2); + } + + void testWhole10x() { + testStrided(0, 5000, 10, m_source); + + // alternate values at 2n should equal all values at n + floatvec_t output = get(0, 5000, 10); + floatvec_t half = get(0, 5000, 5); + compareStrided(output, half, 2); + } + + void testWhole16x() { + testStrided(0, 5000, 16, m_source); + + // alternate values at 2n should equal all values at n + floatvec_t output = get(0, 5000, 16); + floatvec_t half = get(0, 5000, 8); + compareStrided(output, half, 2); + } + + void testSubsets4x() { + testStrided(0, 500, 4, sourceSubset(0, 500)); + testStrided(4500, 500, 4, sourceSubset(4500, 500)); + testStrided(2000, 1000, 4, sourceSubset(2000, 1000)); + + // check for windowed sinc values between the original + // samples, even when the original sample that was the source + // of this sinc kernel is not within the requested range + floatvec_t output = get(1, 10, 4); + QVERIFY(output[0] < 0.0001); + QVERIFY(output[1] + 0.1787 < 0.0001); + QVERIFY(output[2] + 0.2099 < 0.0001); + QVERIFY(output[3] + 0.1267 < 0.0001); + + // and again at the end + output = get(4989, 10, 4); + QVERIFY(output[39] + 0.9000 < 0.0001); + QVERIFY(output[38] + 0.6358 < 0.0001); + QVERIFY(output[37] + 0.2993 < 0.0001); + QVERIFY(output[35] - 0.1787 < 0.0001); + QVERIFY(output[34] - 0.2099 < 0.0001); + QVERIFY(output[33] - 0.1267 < 0.0001); + } + + void testOverlaps4x() { + // overlapping the start -> result should be zero-padded to + // preserve start frame + floatvec_t expected = sourceSubset(0, 400); + expected.insert(expected.begin(), 100, 0.f); + testStrided(-100, 500, 4, expected); + + // overlapping the end -> result should be truncated to + // preserve source length + expected = sourceSubset(4600, 400); + testStrided(4600, 500, 4, expected); + } + + void testSubsets15x() { + testStrided(0, 500, 15, sourceSubset(0, 500)); + testStrided(4500, 500, 15, sourceSubset(4500, 500)); + testStrided(2000, 1000, 15, sourceSubset(2000, 1000)); + } + + void testOverlaps15x() { + // overlapping the start -> result should be zero-padded to + // preserve start frame + floatvec_t expected = sourceSubset(0, 400); + expected.insert(expected.begin(), 100, 0.f); + testStrided(-100, 500, 15, expected); + + // overlapping the end -> result should be truncated to + // preserve source length + expected = sourceSubset(4600, 400); + testStrided(4600, 500, 15, expected); + } +}; + + +#endif