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