cannam@126
|
1 /*
|
cannam@126
|
2 ** Copyright (c) 2002-2016, Erik de Castro Lopo <erikd@mega-nerd.com>
|
cannam@126
|
3 ** All rights reserved.
|
cannam@126
|
4 **
|
cannam@126
|
5 ** This code is released under 2-clause BSD license. Please see the
|
cannam@126
|
6 ** file at : https://github.com/erikd/libsamplerate/blob/master/COPYING
|
cannam@126
|
7 */
|
cannam@126
|
8
|
cannam@126
|
9 #include <stdio.h>
|
cannam@126
|
10 #include <stdlib.h>
|
cannam@126
|
11
|
cannam@126
|
12 #include <samplerate.h>
|
cannam@126
|
13
|
cannam@126
|
14 #include "util.h"
|
cannam@126
|
15
|
cannam@126
|
16 #define BUFFER_LEN 2048
|
cannam@126
|
17 #define CB_READ_LEN 256
|
cannam@126
|
18
|
cannam@126
|
19 static void process_reset_test (int converter) ;
|
cannam@126
|
20 static void callback_reset_test (int converter) ;
|
cannam@126
|
21
|
cannam@126
|
22 static float data_one [BUFFER_LEN] ;
|
cannam@126
|
23 static float data_zero [BUFFER_LEN] ;
|
cannam@126
|
24
|
cannam@126
|
25 int
|
cannam@126
|
26 main (void)
|
cannam@126
|
27 {
|
cannam@126
|
28 puts ("") ;
|
cannam@126
|
29
|
cannam@126
|
30 process_reset_test (SRC_ZERO_ORDER_HOLD) ;
|
cannam@126
|
31 process_reset_test (SRC_LINEAR) ;
|
cannam@126
|
32 process_reset_test (SRC_SINC_FASTEST) ;
|
cannam@126
|
33
|
cannam@126
|
34 callback_reset_test (SRC_ZERO_ORDER_HOLD) ;
|
cannam@126
|
35 callback_reset_test (SRC_LINEAR) ;
|
cannam@126
|
36 callback_reset_test (SRC_SINC_FASTEST) ;
|
cannam@126
|
37
|
cannam@126
|
38 puts ("") ;
|
cannam@126
|
39
|
cannam@126
|
40 return 0 ;
|
cannam@126
|
41 } /* main */
|
cannam@126
|
42
|
cannam@126
|
43 static void
|
cannam@126
|
44 process_reset_test (int converter)
|
cannam@126
|
45 { static float output [BUFFER_LEN] ;
|
cannam@126
|
46
|
cannam@126
|
47 SRC_STATE *src_state ;
|
cannam@126
|
48 SRC_DATA src_data ;
|
cannam@126
|
49 int k, error ;
|
cannam@126
|
50
|
cannam@126
|
51 printf ("\tprocess_reset_test (%-28s) ....... ", src_get_name (converter)) ;
|
cannam@126
|
52 fflush (stdout) ;
|
cannam@126
|
53
|
cannam@126
|
54 for (k = 0 ; k < BUFFER_LEN ; k++)
|
cannam@126
|
55 { data_one [k] = 1.0 ;
|
cannam@126
|
56 data_zero [k] = 0.0 ;
|
cannam@126
|
57 } ;
|
cannam@126
|
58
|
cannam@126
|
59 /* Get a converter. */
|
cannam@126
|
60 if ((src_state = src_new (converter, 1, &error)) == NULL)
|
cannam@126
|
61 { printf ("\n\nLine %d : src_new() failed : %s.\n\n", __LINE__, src_strerror (error)) ;
|
cannam@126
|
62 exit (1) ;
|
cannam@126
|
63 } ;
|
cannam@126
|
64
|
cannam@126
|
65 /* Process a bunch of 1.0 valued samples. */
|
cannam@126
|
66 src_data.data_in = data_one ;
|
cannam@126
|
67 src_data.data_out = output ;
|
cannam@126
|
68 src_data.input_frames = BUFFER_LEN ;
|
cannam@126
|
69 src_data.output_frames = BUFFER_LEN ;
|
cannam@126
|
70 src_data.src_ratio = 0.9 ;
|
cannam@126
|
71 src_data.end_of_input = 1 ;
|
cannam@126
|
72
|
cannam@126
|
73 if ((error = src_process (src_state, &src_data)) != 0)
|
cannam@126
|
74 { printf ("\n\nLine %d : src_simple () returned error : %s\n\n", __LINE__, src_strerror (error)) ;
|
cannam@126
|
75 exit (1) ;
|
cannam@126
|
76 } ;
|
cannam@126
|
77
|
cannam@126
|
78 /* Reset the state of the converter.*/
|
cannam@126
|
79 src_reset (src_state) ;
|
cannam@126
|
80
|
cannam@126
|
81 /* Now process some zero data. */
|
cannam@126
|
82 src_data.data_in = data_zero ;
|
cannam@126
|
83 src_data.data_out = output ;
|
cannam@126
|
84 src_data.input_frames = BUFFER_LEN ;
|
cannam@126
|
85 src_data.output_frames = BUFFER_LEN ;
|
cannam@126
|
86 src_data.src_ratio = 0.9 ;
|
cannam@126
|
87 src_data.end_of_input = 1 ;
|
cannam@126
|
88
|
cannam@126
|
89 if ((error = src_process (src_state, &src_data)) != 0)
|
cannam@126
|
90 { printf ("\n\nLine %d : src_simple () returned error : %s\n\n", __LINE__, src_strerror (error)) ;
|
cannam@126
|
91 exit (1) ;
|
cannam@126
|
92 } ;
|
cannam@126
|
93
|
cannam@126
|
94 /* Finally make sure that the output data is zero ie reset was sucessful. */
|
cannam@126
|
95 for (k = 0 ; k < BUFFER_LEN / 2 ; k++)
|
cannam@126
|
96 if (output [k] != 0.0)
|
cannam@126
|
97 { printf ("\n\nLine %d : output [%d] should be 0.0, is %f.\n", __LINE__, k, output [k]) ;
|
cannam@126
|
98 exit (1) ;
|
cannam@126
|
99 } ;
|
cannam@126
|
100
|
cannam@126
|
101 /* Make sure that this function has been exported. */
|
cannam@126
|
102 src_set_ratio (src_state, 1.0) ;
|
cannam@126
|
103
|
cannam@126
|
104 /* Delete converter. */
|
cannam@126
|
105 src_state = src_delete (src_state) ;
|
cannam@126
|
106
|
cannam@126
|
107 puts ("ok") ;
|
cannam@126
|
108 } /* process_reset_test */
|
cannam@126
|
109
|
cannam@126
|
110 /*==============================================================================
|
cannam@126
|
111 */
|
cannam@126
|
112
|
cannam@126
|
113 typedef struct
|
cannam@126
|
114 { int channels ;
|
cannam@126
|
115 long count, total ;
|
cannam@126
|
116 float *data ;
|
cannam@126
|
117 } TEST_CB_DATA ;
|
cannam@126
|
118
|
cannam@126
|
119 static long
|
cannam@126
|
120 test_callback_func (void *cb_data, float **data)
|
cannam@126
|
121 { TEST_CB_DATA *pcb_data ;
|
cannam@126
|
122
|
cannam@126
|
123 long frames ;
|
cannam@126
|
124
|
cannam@126
|
125 if ((pcb_data = cb_data) == NULL)
|
cannam@126
|
126 return 0 ;
|
cannam@126
|
127
|
cannam@126
|
128 if (data == NULL)
|
cannam@126
|
129 return 0 ;
|
cannam@126
|
130
|
cannam@126
|
131 if (pcb_data->total - pcb_data->count > 0)
|
cannam@126
|
132 frames = pcb_data->total - pcb_data->count ;
|
cannam@126
|
133 else
|
cannam@126
|
134 frames = 0 ;
|
cannam@126
|
135
|
cannam@126
|
136 *data = pcb_data->data + pcb_data->count ;
|
cannam@126
|
137 pcb_data->count += frames ;
|
cannam@126
|
138
|
cannam@126
|
139 return frames ;
|
cannam@126
|
140 } /* test_callback_func */
|
cannam@126
|
141
|
cannam@126
|
142 static void
|
cannam@126
|
143 callback_reset_test (int converter)
|
cannam@126
|
144 { static TEST_CB_DATA test_callback_data ;
|
cannam@126
|
145
|
cannam@126
|
146 static float output [BUFFER_LEN] ;
|
cannam@126
|
147
|
cannam@126
|
148 SRC_STATE *src_state ;
|
cannam@126
|
149
|
cannam@126
|
150 double src_ratio = 1.1 ;
|
cannam@126
|
151 long read_count, read_total ;
|
cannam@126
|
152 int k, error ;
|
cannam@126
|
153
|
cannam@126
|
154 printf ("\tcallback_reset_test (%-28s) ....... ", src_get_name (converter)) ;
|
cannam@126
|
155 fflush (stdout) ;
|
cannam@126
|
156
|
cannam@126
|
157 for (k = 0 ; k < ARRAY_LEN (data_one) ; k++)
|
cannam@126
|
158 { data_one [k] = 1.0 ;
|
cannam@126
|
159 data_zero [k] = 0.0 ;
|
cannam@126
|
160 } ;
|
cannam@126
|
161
|
cannam@126
|
162 if ((src_state = src_callback_new (test_callback_func, converter, 1, &error, &test_callback_data)) == NULL)
|
cannam@126
|
163 { printf ("\n\nLine %d : %s\n\n", __LINE__, src_strerror (error)) ;
|
cannam@126
|
164 exit (1) ;
|
cannam@126
|
165 } ;
|
cannam@126
|
166
|
cannam@126
|
167 /* Process a bunch of 1.0 valued samples. */
|
cannam@126
|
168 test_callback_data.channels = 1 ;
|
cannam@126
|
169 test_callback_data.count = 0 ;
|
cannam@126
|
170 test_callback_data.total = ARRAY_LEN (data_one) ;
|
cannam@126
|
171 test_callback_data.data = data_one ;
|
cannam@126
|
172
|
cannam@126
|
173 read_total = 0 ;
|
cannam@126
|
174 do
|
cannam@126
|
175 { read_count = (ARRAY_LEN (output) - read_total > CB_READ_LEN) ? CB_READ_LEN : ARRAY_LEN (output) - read_total ;
|
cannam@126
|
176 read_count = src_callback_read (src_state, src_ratio, read_count, output + read_total) ;
|
cannam@126
|
177 read_total += read_count ;
|
cannam@126
|
178 }
|
cannam@126
|
179 while (read_count > 0) ;
|
cannam@126
|
180
|
cannam@126
|
181 /* Check for errors. */
|
cannam@126
|
182 if ((error = src_error (src_state)) != 0)
|
cannam@126
|
183 { printf ("\n\nLine %d : %s\n\n", __LINE__, src_strerror (error)) ;
|
cannam@126
|
184 exit (1) ;
|
cannam@126
|
185 } ;
|
cannam@126
|
186
|
cannam@126
|
187 /* Reset the state of the converter.*/
|
cannam@126
|
188 src_reset (src_state) ;
|
cannam@126
|
189
|
cannam@126
|
190 /* Process a bunch of 0.0 valued samples. */
|
cannam@126
|
191 test_callback_data.channels = 1 ;
|
cannam@126
|
192 test_callback_data.count = 0 ;
|
cannam@126
|
193 test_callback_data.total = ARRAY_LEN (data_zero) ;
|
cannam@126
|
194 test_callback_data.data = data_zero ;
|
cannam@126
|
195
|
cannam@126
|
196 /* Now process some zero data. */
|
cannam@126
|
197 read_total = 0 ;
|
cannam@126
|
198 do
|
cannam@126
|
199 { read_count = (ARRAY_LEN (output) - read_total > CB_READ_LEN) ? CB_READ_LEN : ARRAY_LEN (output) - read_total ;
|
cannam@126
|
200 read_count = src_callback_read (src_state, src_ratio, read_count, output + read_total) ;
|
cannam@126
|
201 read_total += read_count ;
|
cannam@126
|
202 }
|
cannam@126
|
203 while (read_count > 0) ;
|
cannam@126
|
204
|
cannam@126
|
205 /* Check for errors. */
|
cannam@126
|
206 if ((error = src_error (src_state)) != 0)
|
cannam@126
|
207 { printf ("\n\nLine %d : %s\n\n", __LINE__, src_strerror (error)) ;
|
cannam@126
|
208 exit (1) ;
|
cannam@126
|
209 } ;
|
cannam@126
|
210
|
cannam@126
|
211 /* Finally make sure that the output data is zero ie reset was sucessful. */
|
cannam@126
|
212 for (k = 0 ; k < BUFFER_LEN / 2 ; k++)
|
cannam@126
|
213 if (output [k] != 0.0)
|
cannam@126
|
214 { printf ("\n\nLine %d : output [%d] should be 0.0, is %f.\n\n", __LINE__, k, output [k]) ;
|
cannam@126
|
215 save_oct_float ("output.dat", data_one, ARRAY_LEN (data_one), output, ARRAY_LEN (output)) ;
|
cannam@126
|
216 exit (1) ;
|
cannam@126
|
217 } ;
|
cannam@126
|
218
|
cannam@126
|
219 /* Make sure that this function has been exported. */
|
cannam@126
|
220 src_set_ratio (src_state, 1.0) ;
|
cannam@126
|
221
|
cannam@126
|
222 /* Delete converter. */
|
cannam@126
|
223 src_state = src_delete (src_state) ;
|
cannam@126
|
224
|
cannam@126
|
225 puts ("ok") ;
|
cannam@126
|
226 } /* callback_reset_test */
|
cannam@126
|
227
|
cannam@126
|
228
|