annotate constant-q-cpp/test/TestWindow.cpp @ 372:af71cbdab621 tip

Update bqvec code
author Chris Cannam
date Tue, 19 Nov 2019 10:13:32 +0000
parents 5d0a2ebb4d17
children
rev   line source
Chris@366 1 /* -*- c-basic-offset: 4 indent-tabs-mode: nil -*- vi:set ts=8 sts=4 sw=4: */
Chris@366 2
Chris@366 3 #include "dsp/Window.h"
Chris@366 4 #include "dsp/KaiserWindow.h"
Chris@366 5 #include "dsp/SincWindow.h"
Chris@366 6
Chris@366 7 #include <iostream>
Chris@366 8
Chris@366 9 #define BOOST_TEST_DYN_LINK
Chris@366 10 #define BOOST_TEST_MAIN
Chris@366 11
Chris@366 12 #include <boost/test/unit_test.hpp>
Chris@366 13
Chris@366 14 BOOST_AUTO_TEST_SUITE(TestWindow)
Chris@366 15
Chris@366 16 using std::cout;
Chris@366 17 using std::endl;
Chris@366 18
Chris@366 19 #define COMPARE_ARRAY(a, b) \
Chris@366 20 for (int cmp_i = 0; cmp_i < (int)(sizeof(a)/sizeof(a[0])); ++cmp_i) { \
Chris@366 21 BOOST_CHECK_SMALL(a[cmp_i] - b[cmp_i], 1e-4); \
Chris@366 22 }
Chris@366 23
Chris@366 24 void
Chris@366 25 testSymmetric(double *d, int n)
Chris@366 26 {
Chris@366 27 for (int i = 0; i <= n/2; ++i) {
Chris@366 28 BOOST_CHECK_CLOSE(d[i], d[n-i-1], 1e-10);
Chris@366 29 }
Chris@366 30 }
Chris@366 31
Chris@366 32 BOOST_AUTO_TEST_CASE(periodic)
Chris@366 33 {
Chris@366 34 // We can't actually test whether a function is periodic, given
Chris@366 35 // only one cycle of it! But we can make sure that all but the
Chris@366 36 // first sample is symmetric, which is what a symmetric window
Chris@366 37 // becomes when generated in periodic mode
Chris@366 38 double d[9];
Chris@366 39 for (int n = 8; n <= 9; ++n) {
Chris@366 40 for (int wt = (int)FirstWindow; wt <= (int)LastWindow; ++wt) {
Chris@366 41 for (int i = 0; i < n; ++i) d[i] = 1.0;
Chris@366 42 Window<double> w((WindowType)wt, n);
Chris@366 43 w.cut(d);
Chris@366 44 testSymmetric(d + 1, n - 1);
Chris@366 45 }
Chris@366 46 }
Chris@366 47 }
Chris@366 48
Chris@366 49 template <int N>
Chris@366 50 void testWindow(WindowType type, const double expected[N])
Chris@366 51 {
Chris@366 52 double d[N];
Chris@366 53 for (int i = 0; i < N; ++i) d[i] = 1.0;
Chris@366 54 Window<double> w(type, N);
Chris@366 55 w.cut(d);
Chris@366 56 COMPARE_ARRAY(d, expected);
Chris@366 57
Chris@366 58 double d0[N], d1[N];
Chris@366 59 for (int i = 0; i < N; ++i) d0[i] = 0.5 + (1.0 / (N * 2)) * (i + 1);
Chris@366 60 w.cut(d0, d1);
Chris@366 61 for (int i = 0; i < N; ++i) {
Chris@366 62 BOOST_CHECK_SMALL(d1[i] - d0[i] * expected[i], 1e-4);
Chris@366 63 }
Chris@366 64 }
Chris@366 65
Chris@366 66 BOOST_AUTO_TEST_CASE(bartlett)
Chris@366 67 {
Chris@366 68 double e1[] = { 1 };
Chris@366 69 testWindow<1>(BartlettWindow, e1);
Chris@366 70
Chris@366 71 double e2[] = { 0, 0 };
Chris@366 72 testWindow<2>(BartlettWindow, e2);
Chris@366 73
Chris@366 74 double e3[] = { 0, 2./3., 2./3. };
Chris@366 75 testWindow<3>(BartlettWindow, e3);
Chris@366 76
Chris@366 77 double e4[] = { 0, 1./2., 1., 1./2. };
Chris@366 78 testWindow<4>(BartlettWindow, e4);
Chris@366 79
Chris@366 80 double e5[] = { 0, 1./2., 1., 1., 1./2. };
Chris@366 81 testWindow<5>(BartlettWindow, e5);
Chris@366 82
Chris@366 83 double e6[] = { 0, 1./3., 2./3., 1., 2./3., 1./3. };
Chris@366 84 testWindow<6>(BartlettWindow, e6);
Chris@366 85 }
Chris@366 86
Chris@366 87 BOOST_AUTO_TEST_CASE(hamming)
Chris@366 88 {
Chris@366 89 double e1[] = { 1 };
Chris@366 90 testWindow<1>(HammingWindow, e1);
Chris@366 91
Chris@366 92 double e10[] = {
Chris@366 93 0.0800, 0.1679, 0.3979, 0.6821, 0.9121,
Chris@366 94 1.0000, 0.9121, 0.6821, 0.3979, 0.1679
Chris@366 95 };
Chris@366 96 testWindow<10>(HammingWindow, e10);
Chris@366 97 }
Chris@366 98
Chris@366 99 BOOST_AUTO_TEST_CASE(hann)
Chris@366 100 {
Chris@366 101 double e1[] = { 1 };
Chris@366 102 testWindow<1>(HanningWindow, e1);
Chris@366 103
Chris@366 104 double e10[] = {
Chris@366 105 0, 0.0955, 0.3455, 0.6545, 0.9045,
Chris@366 106 1.0000, 0.9045, 0.6545, 0.3455, 0.0955,
Chris@366 107 };
Chris@366 108 testWindow<10>(HanningWindow, e10);
Chris@366 109 }
Chris@366 110
Chris@366 111 BOOST_AUTO_TEST_CASE(blackman)
Chris@366 112 {
Chris@366 113 double e1[] = { 1 };
Chris@366 114 testWindow<1>(BlackmanWindow, e1);
Chris@366 115
Chris@366 116 double e10[] = {
Chris@366 117 0, 0.0402, 0.2008, 0.5098, 0.8492,
Chris@366 118 1.0000, 0.8492, 0.5098, 0.2008, 0.0402,
Chris@366 119 };
Chris@366 120 testWindow<10>(BlackmanWindow, e10);
Chris@366 121 }
Chris@366 122
Chris@366 123 BOOST_AUTO_TEST_CASE(blackmanHarris)
Chris@366 124 {
Chris@366 125 double e1[] = { 1 };
Chris@366 126 testWindow<1>(BlackmanHarrisWindow, e1);
Chris@366 127
Chris@366 128 double e10[] = {
Chris@366 129 0.0001, 0.0110, 0.1030, 0.3859, 0.7938,
Chris@366 130 1.0000, 0.7938, 0.3859, 0.1030, 0.0110,
Chris@366 131 };
Chris@366 132 testWindow<10>(BlackmanHarrisWindow, e10);
Chris@366 133 }
Chris@366 134
Chris@366 135 BOOST_AUTO_TEST_CASE(kaiser_4_10)
Chris@366 136 {
Chris@366 137 double d[10];
Chris@366 138 for (int i = 0; i < 10; ++i) d[i] = 1.0;
Chris@366 139 double e[] = {
Chris@366 140 0.0885, 0.2943, 0.5644, 0.8216, 0.9789,
Chris@366 141 0.9789, 0.8216, 0.5644, 0.2943, 0.0885
Chris@366 142 };
Chris@366 143 KaiserWindow::Parameters p;
Chris@366 144 p.length = 10;
Chris@366 145 p.beta = 4;
Chris@366 146 KaiserWindow k(p);
Chris@366 147 k.cut(d);
Chris@366 148 COMPARE_ARRAY(d, e);
Chris@366 149 }
Chris@366 150
Chris@366 151 BOOST_AUTO_TEST_CASE(kaiser_2p5_11)
Chris@366 152 {
Chris@366 153 double d[11];
Chris@366 154 for (int i = 0; i < 11; ++i) d[i] = 1.0;
Chris@366 155 double e[] = {
Chris@366 156 0.3040, 0.5005, 0.6929, 0.8546, 0.9622,
Chris@366 157 1.0000, 0.9622, 0.8546, 0.6929, 0.5005, 0.3040
Chris@366 158 };
Chris@366 159 KaiserWindow::Parameters p;
Chris@366 160 p.length = 11;
Chris@366 161 p.beta = 2.5;
Chris@366 162 KaiserWindow k(p);
Chris@366 163 k.cut(d);
Chris@366 164 COMPARE_ARRAY(d, e);
Chris@366 165 }
Chris@366 166
Chris@366 167 //!!! todo: tests for kaiser with attenuation and bandwidth parameters
Chris@366 168
Chris@366 169 template <int N>
Chris@366 170 void testSinc(double p, const double expected[N])
Chris@366 171 {
Chris@366 172 double d[N];
Chris@366 173 for (int i = 0; i < N; ++i) d[i] = 1.0;
Chris@366 174 SincWindow w(N, p);
Chris@366 175 w.cut(d);
Chris@366 176 COMPARE_ARRAY(d, expected);
Chris@366 177
Chris@366 178 double d0[N], d1[N];
Chris@366 179 for (int i = 0; i < N; ++i) d0[i] = 0.5 + (1.0 / (N * 2)) * (i + 1);
Chris@366 180 w.cut(d0, d1);
Chris@366 181 for (int i = 0; i < N; ++i) {
Chris@366 182 BOOST_CHECK_SMALL(d1[i] - d0[i] * expected[i], 1e-4);
Chris@366 183 }
Chris@366 184 }
Chris@366 185
Chris@366 186 BOOST_AUTO_TEST_CASE(sinc)
Chris@366 187 {
Chris@366 188 double e1[] = { 0, 0, 1, 0, 0 };
Chris@366 189 testSinc<5>(1, e1);
Chris@366 190
Chris@366 191 double e2[] = { 0, 0, 1, 0, 0 };
Chris@366 192 testSinc<5>(2, e2);
Chris@366 193
Chris@366 194 double e3[] = { -0.2122, 0.0, 0.6366, 1.0, 0.6366, 0, -0.2122 };
Chris@366 195 testSinc<7>(4, e3);
Chris@366 196
Chris@366 197 double e4[] = { -0.2122, 0, 0.6366, 1, 0.6366, 0 };
Chris@366 198 testSinc<6>(4, e4);
Chris@366 199 }
Chris@366 200
Chris@366 201 BOOST_AUTO_TEST_SUITE_END()
Chris@366 202