/proc/self/cwd/libfaad/filtbank.c
Line | Count | Source (jump to first uncovered line) |
1 | | /* |
2 | | ** FAAD2 - Freeware Advanced Audio (AAC) Decoder including SBR decoding |
3 | | ** Copyright (C) 2003-2005 M. Bakker, Nero AG, http://www.nero.com |
4 | | ** |
5 | | ** This program is free software; you can redistribute it and/or modify |
6 | | ** it under the terms of the GNU General Public License as published by |
7 | | ** the Free Software Foundation; either version 2 of the License, or |
8 | | ** (at your option) any later version. |
9 | | ** |
10 | | ** This program is distributed in the hope that it will be useful, |
11 | | ** but WITHOUT ANY WARRANTY; without even the implied warranty of |
12 | | ** MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the |
13 | | ** GNU General Public License for more details. |
14 | | ** |
15 | | ** You should have received a copy of the GNU General Public License |
16 | | ** along with this program; if not, write to the Free Software |
17 | | ** Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. |
18 | | ** |
19 | | ** Any non-GPL usage of this software or parts of this software is strictly |
20 | | ** forbidden. |
21 | | ** |
22 | | ** The "appropriate copyright message" mentioned in section 2c of the GPLv2 |
23 | | ** must read: "Code from FAAD2 is copyright (c) Nero AG, www.nero.com" |
24 | | ** |
25 | | ** Commercial non-GPL licensing of this software is possible. |
26 | | ** For more info contact Nero AG through Mpeg4AAClicense@nero.com. |
27 | | ** |
28 | | ** $Id: filtbank.c,v 1.46 2009/01/26 23:51:15 menno Exp $ |
29 | | **/ |
30 | | |
31 | | #include "common.h" |
32 | | #include "structs.h" |
33 | | |
34 | | #include <stdlib.h> |
35 | | #ifdef _WIN32_WCE |
36 | | #define assert(x) |
37 | | #else |
38 | | #include <assert.h> |
39 | | #endif |
40 | | |
41 | | #include "filtbank.h" |
42 | | #include "syntax.h" |
43 | | #include "kbd_win.h" |
44 | | #include "sine_win.h" |
45 | | #include "mdct.h" |
46 | | |
47 | | |
48 | | fb_info *filter_bank_init(uint16_t frame_len) |
49 | 0 | { |
50 | 0 | uint16_t nshort = frame_len/8; |
51 | 0 | #ifdef LD_DEC |
52 | 0 | uint16_t frame_len_ld = frame_len/2; |
53 | 0 | #endif |
54 | |
|
55 | 0 | fb_info *fb = (fb_info*)faad_malloc(sizeof(fb_info)); |
56 | 0 | memset(fb, 0, sizeof(fb_info)); |
57 | | |
58 | | /* normal */ |
59 | 0 | fb->mdct256 = faad_mdct_init(2*nshort); |
60 | 0 | fb->mdct2048 = faad_mdct_init(2*frame_len); |
61 | 0 | #ifdef LD_DEC |
62 | | /* LD */ |
63 | 0 | fb->mdct1024 = faad_mdct_init(2*frame_len_ld); |
64 | 0 | #endif |
65 | |
|
66 | 0 | #ifdef ALLOW_SMALL_FRAMELENGTH |
67 | 0 | if (frame_len == 1024) |
68 | 0 | { |
69 | 0 | #endif |
70 | 0 | fb->long_window[0] = sine_long_1024; |
71 | 0 | fb->short_window[0] = sine_short_128; |
72 | 0 | fb->long_window[1] = kbd_long_1024; |
73 | 0 | fb->short_window[1] = kbd_short_128; |
74 | 0 | #ifdef LD_DEC |
75 | 0 | fb->ld_window[0] = sine_mid_512; |
76 | 0 | fb->ld_window[1] = ld_mid_512; |
77 | 0 | #endif |
78 | 0 | #ifdef ALLOW_SMALL_FRAMELENGTH |
79 | 0 | } else /* (frame_len == 960) */ { |
80 | 0 | fb->long_window[0] = sine_long_960; |
81 | 0 | fb->short_window[0] = sine_short_120; |
82 | 0 | fb->long_window[1] = kbd_long_960; |
83 | 0 | fb->short_window[1] = kbd_short_120; |
84 | 0 | #ifdef LD_DEC |
85 | 0 | fb->ld_window[0] = sine_mid_480; |
86 | 0 | fb->ld_window[1] = ld_mid_480; |
87 | 0 | #endif |
88 | 0 | } |
89 | 0 | #endif |
90 | |
|
91 | 0 | return fb; |
92 | 0 | } |
93 | | |
94 | | void filter_bank_end(fb_info *fb) |
95 | 0 | { |
96 | 0 | if (fb != NULL) |
97 | 0 | { |
98 | | #ifdef PROFILE |
99 | | printf("FB: %I64d cycles\n", fb->cycles); |
100 | | #endif |
101 | |
|
102 | 0 | faad_mdct_end(fb->mdct256); |
103 | 0 | faad_mdct_end(fb->mdct2048); |
104 | 0 | #ifdef LD_DEC |
105 | 0 | faad_mdct_end(fb->mdct1024); |
106 | 0 | #endif |
107 | |
|
108 | 0 | faad_free(fb); |
109 | 0 | } |
110 | 0 | } |
111 | | |
112 | | static INLINE void imdct_long(fb_info *fb, real_t *in_data, real_t *out_data, uint16_t len) |
113 | 0 | { |
114 | 0 | #ifdef LD_DEC |
115 | 0 | mdct_info *mdct = NULL; |
116 | |
|
117 | 0 | switch (len) |
118 | 0 | { |
119 | 0 | case 2048: |
120 | 0 | case 1920: |
121 | 0 | mdct = fb->mdct2048; |
122 | 0 | break; |
123 | 0 | case 1024: |
124 | 0 | case 960: |
125 | 0 | mdct = fb->mdct1024; |
126 | 0 | break; |
127 | 0 | } |
128 | | |
129 | 0 | faad_imdct(mdct, in_data, out_data); |
130 | | #else |
131 | | (void)len; |
132 | | faad_imdct(fb->mdct2048, in_data, out_data); |
133 | | #endif |
134 | 0 | } |
135 | | |
136 | | |
137 | | #ifdef LTP_DEC |
138 | | static INLINE void mdct(fb_info *fb, real_t *in_data, real_t *out_data, uint16_t len) |
139 | 0 | { |
140 | 0 | mdct_info *mdct = NULL; |
141 | |
|
142 | 0 | switch (len) |
143 | 0 | { |
144 | 0 | case 2048: |
145 | 0 | case 1920: |
146 | 0 | mdct = fb->mdct2048; |
147 | 0 | break; |
148 | 0 | case 256: |
149 | 0 | case 240: |
150 | 0 | mdct = fb->mdct256; |
151 | 0 | break; |
152 | 0 | #ifdef LD_DEC |
153 | 0 | case 1024: |
154 | 0 | case 960: |
155 | 0 | mdct = fb->mdct1024; |
156 | 0 | break; |
157 | 0 | #endif |
158 | 0 | } |
159 | | |
160 | 0 | faad_mdct(mdct, in_data, out_data); |
161 | 0 | } |
162 | | #endif |
163 | | |
164 | | void ifilter_bank(fb_info *fb, uint8_t window_sequence, uint8_t window_shape, |
165 | | uint8_t window_shape_prev, real_t *freq_in, |
166 | | real_t *time_out, real_t *overlap, |
167 | | uint8_t object_type, uint16_t frame_len) |
168 | 0 | { |
169 | 0 | int16_t i; |
170 | 0 | ALIGN real_t transf_buf[2*1024] = {0}; |
171 | |
|
172 | 0 | const real_t *window_long = NULL; |
173 | 0 | const real_t *window_long_prev = NULL; |
174 | 0 | const real_t *window_short = NULL; |
175 | 0 | const real_t *window_short_prev = NULL; |
176 | |
|
177 | 0 | uint16_t nlong = frame_len; |
178 | 0 | uint16_t nshort = frame_len/8; |
179 | 0 | uint16_t trans = nshort/2; |
180 | |
|
181 | 0 | uint16_t nflat_ls = (nlong-nshort)/2; |
182 | |
|
183 | | #ifdef PROFILE |
184 | | int64_t count = faad_get_ts(); |
185 | | #endif |
186 | | |
187 | | /* select windows of current frame and previous frame (Sine or KBD) */ |
188 | 0 | #ifdef LD_DEC |
189 | 0 | if (object_type == LD) |
190 | 0 | { |
191 | 0 | window_long = fb->ld_window[window_shape]; |
192 | 0 | window_long_prev = fb->ld_window[window_shape_prev]; |
193 | 0 | } else { |
194 | | #else |
195 | | (void)object_type; |
196 | | #endif |
197 | 0 | window_long = fb->long_window[window_shape]; |
198 | 0 | window_long_prev = fb->long_window[window_shape_prev]; |
199 | 0 | window_short = fb->short_window[window_shape]; |
200 | 0 | window_short_prev = fb->short_window[window_shape_prev]; |
201 | 0 | #ifdef LD_DEC |
202 | 0 | } |
203 | 0 | #endif |
204 | |
|
205 | | #if 0 |
206 | | for (i = 0; i < 1024; i++) |
207 | | { |
208 | | printf("%d\n", freq_in[i]); |
209 | | } |
210 | | #endif |
211 | |
|
212 | | #if 0 |
213 | | printf("%d %d\n", window_sequence, window_shape); |
214 | | #endif |
215 | |
|
216 | 0 | switch (window_sequence) |
217 | 0 | { |
218 | 0 | case ONLY_LONG_SEQUENCE: |
219 | | /* perform iMDCT */ |
220 | 0 | imdct_long(fb, freq_in, transf_buf, 2*nlong); |
221 | | |
222 | | /* add second half output of previous frame to windowed output of current frame */ |
223 | 0 | for (i = 0; i < nlong; i+=4) |
224 | 0 | { |
225 | 0 | time_out[i] = overlap[i] + MUL_F(transf_buf[i],window_long_prev[i]); |
226 | 0 | time_out[i+1] = overlap[i+1] + MUL_F(transf_buf[i+1],window_long_prev[i+1]); |
227 | 0 | time_out[i+2] = overlap[i+2] + MUL_F(transf_buf[i+2],window_long_prev[i+2]); |
228 | 0 | time_out[i+3] = overlap[i+3] + MUL_F(transf_buf[i+3],window_long_prev[i+3]); |
229 | 0 | } |
230 | | |
231 | | /* window the second half and save as overlap for next frame */ |
232 | 0 | for (i = 0; i < nlong; i+=4) |
233 | 0 | { |
234 | 0 | overlap[i] = MUL_F(transf_buf[nlong+i],window_long[nlong-1-i]); |
235 | 0 | overlap[i+1] = MUL_F(transf_buf[nlong+i+1],window_long[nlong-2-i]); |
236 | 0 | overlap[i+2] = MUL_F(transf_buf[nlong+i+2],window_long[nlong-3-i]); |
237 | 0 | overlap[i+3] = MUL_F(transf_buf[nlong+i+3],window_long[nlong-4-i]); |
238 | 0 | } |
239 | 0 | break; |
240 | | |
241 | 0 | case LONG_START_SEQUENCE: |
242 | | /* perform iMDCT */ |
243 | 0 | imdct_long(fb, freq_in, transf_buf, 2*nlong); |
244 | | |
245 | | /* add second half output of previous frame to windowed output of current frame */ |
246 | 0 | for (i = 0; i < nlong; i+=4) |
247 | 0 | { |
248 | 0 | time_out[i] = overlap[i] + MUL_F(transf_buf[i],window_long_prev[i]); |
249 | 0 | time_out[i+1] = overlap[i+1] + MUL_F(transf_buf[i+1],window_long_prev[i+1]); |
250 | 0 | time_out[i+2] = overlap[i+2] + MUL_F(transf_buf[i+2],window_long_prev[i+2]); |
251 | 0 | time_out[i+3] = overlap[i+3] + MUL_F(transf_buf[i+3],window_long_prev[i+3]); |
252 | 0 | } |
253 | | |
254 | | /* window the second half and save as overlap for next frame */ |
255 | | /* construct second half window using padding with 1's and 0's */ |
256 | 0 | for (i = 0; i < nflat_ls; i++) |
257 | 0 | overlap[i] = transf_buf[nlong+i]; |
258 | 0 | for (i = 0; i < nshort; i++) |
259 | 0 | overlap[nflat_ls+i] = MUL_F(transf_buf[nlong+nflat_ls+i],window_short[nshort-i-1]); |
260 | 0 | for (i = 0; i < nflat_ls; i++) |
261 | 0 | overlap[nflat_ls+nshort+i] = 0; |
262 | 0 | break; |
263 | | |
264 | 0 | case EIGHT_SHORT_SEQUENCE: |
265 | | /* perform iMDCT for each short block */ |
266 | 0 | faad_imdct(fb->mdct256, freq_in+0*nshort, transf_buf+2*nshort*0); |
267 | 0 | faad_imdct(fb->mdct256, freq_in+1*nshort, transf_buf+2*nshort*1); |
268 | 0 | faad_imdct(fb->mdct256, freq_in+2*nshort, transf_buf+2*nshort*2); |
269 | 0 | faad_imdct(fb->mdct256, freq_in+3*nshort, transf_buf+2*nshort*3); |
270 | 0 | faad_imdct(fb->mdct256, freq_in+4*nshort, transf_buf+2*nshort*4); |
271 | 0 | faad_imdct(fb->mdct256, freq_in+5*nshort, transf_buf+2*nshort*5); |
272 | 0 | faad_imdct(fb->mdct256, freq_in+6*nshort, transf_buf+2*nshort*6); |
273 | 0 | faad_imdct(fb->mdct256, freq_in+7*nshort, transf_buf+2*nshort*7); |
274 | | |
275 | | /* add second half output of previous frame to windowed output of current frame */ |
276 | 0 | for (i = 0; i < nflat_ls; i++) |
277 | 0 | time_out[i] = overlap[i]; |
278 | 0 | for(i = 0; i < nshort; i++) |
279 | 0 | { |
280 | 0 | time_out[nflat_ls+ i] = overlap[nflat_ls+ i] + MUL_F(transf_buf[nshort*0+i],window_short_prev[i]); |
281 | 0 | time_out[nflat_ls+1*nshort+i] = overlap[nflat_ls+nshort*1+i] + MUL_F(transf_buf[nshort*1+i],window_short[nshort-1-i]) + MUL_F(transf_buf[nshort*2+i],window_short[i]); |
282 | 0 | time_out[nflat_ls+2*nshort+i] = overlap[nflat_ls+nshort*2+i] + MUL_F(transf_buf[nshort*3+i],window_short[nshort-1-i]) + MUL_F(transf_buf[nshort*4+i],window_short[i]); |
283 | 0 | time_out[nflat_ls+3*nshort+i] = overlap[nflat_ls+nshort*3+i] + MUL_F(transf_buf[nshort*5+i],window_short[nshort-1-i]) + MUL_F(transf_buf[nshort*6+i],window_short[i]); |
284 | 0 | if (i < trans) |
285 | 0 | time_out[nflat_ls+4*nshort+i] = overlap[nflat_ls+nshort*4+i] + MUL_F(transf_buf[nshort*7+i],window_short[nshort-1-i]) + MUL_F(transf_buf[nshort*8+i],window_short[i]); |
286 | 0 | } |
287 | | |
288 | | /* window the second half and save as overlap for next frame */ |
289 | 0 | for(i = 0; i < nshort; i++) |
290 | 0 | { |
291 | 0 | if (i >= trans) |
292 | 0 | overlap[nflat_ls+4*nshort+i-nlong] = MUL_F(transf_buf[nshort*7+i],window_short[nshort-1-i]) + MUL_F(transf_buf[nshort*8+i],window_short[i]); |
293 | 0 | overlap[nflat_ls+5*nshort+i-nlong] = MUL_F(transf_buf[nshort*9+i],window_short[nshort-1-i]) + MUL_F(transf_buf[nshort*10+i],window_short[i]); |
294 | 0 | overlap[nflat_ls+6*nshort+i-nlong] = MUL_F(transf_buf[nshort*11+i],window_short[nshort-1-i]) + MUL_F(transf_buf[nshort*12+i],window_short[i]); |
295 | 0 | overlap[nflat_ls+7*nshort+i-nlong] = MUL_F(transf_buf[nshort*13+i],window_short[nshort-1-i]) + MUL_F(transf_buf[nshort*14+i],window_short[i]); |
296 | 0 | overlap[nflat_ls+8*nshort+i-nlong] = MUL_F(transf_buf[nshort*15+i],window_short[nshort-1-i]); |
297 | 0 | } |
298 | 0 | for (i = 0; i < nflat_ls; i++) |
299 | 0 | overlap[nflat_ls+nshort+i] = 0; |
300 | 0 | break; |
301 | | |
302 | 0 | case LONG_STOP_SEQUENCE: |
303 | | /* perform iMDCT */ |
304 | 0 | imdct_long(fb, freq_in, transf_buf, 2*nlong); |
305 | | |
306 | | /* add second half output of previous frame to windowed output of current frame */ |
307 | | /* construct first half window using padding with 1's and 0's */ |
308 | 0 | for (i = 0; i < nflat_ls; i++) |
309 | 0 | time_out[i] = overlap[i]; |
310 | 0 | for (i = 0; i < nshort; i++) |
311 | 0 | time_out[nflat_ls+i] = overlap[nflat_ls+i] + MUL_F(transf_buf[nflat_ls+i],window_short_prev[i]); |
312 | 0 | for (i = 0; i < nflat_ls; i++) |
313 | 0 | time_out[nflat_ls+nshort+i] = overlap[nflat_ls+nshort+i] + transf_buf[nflat_ls+nshort+i]; |
314 | | |
315 | | /* window the second half and save as overlap for next frame */ |
316 | 0 | for (i = 0; i < nlong; i++) |
317 | 0 | overlap[i] = MUL_F(transf_buf[nlong+i],window_long[nlong-1-i]); |
318 | 0 | break; |
319 | 0 | } |
320 | |
|
321 | | #if 0 |
322 | | for (i = 0; i < 1024; i++) |
323 | | { |
324 | | printf("%d\n", time_out[i]); |
325 | | //printf("0x%.8X\n", time_out[i]); |
326 | | } |
327 | | #endif |
328 | | |
329 | |
|
330 | | #ifdef PROFILE |
331 | | count = faad_get_ts() - count; |
332 | | fb->cycles += count; |
333 | | #endif |
334 | 0 | } |
335 | | |
336 | | |
337 | | #ifdef LTP_DEC |
338 | | /* only works for LTP -> no overlapping, no short blocks */ |
339 | | void filter_bank_ltp(fb_info *fb, uint8_t window_sequence, uint8_t window_shape, |
340 | | uint8_t window_shape_prev, real_t *in_data, real_t *out_mdct, |
341 | | uint8_t object_type, uint16_t frame_len) |
342 | 0 | { |
343 | 0 | int16_t i; |
344 | 0 | ALIGN real_t windowed_buf[2*1024] = {0}; |
345 | |
|
346 | 0 | const real_t *window_long = NULL; |
347 | 0 | const real_t *window_long_prev = NULL; |
348 | 0 | const real_t *window_short = NULL; |
349 | 0 | const real_t *window_short_prev = NULL; |
350 | |
|
351 | 0 | uint16_t nlong = frame_len; |
352 | 0 | uint16_t nshort = frame_len/8; |
353 | 0 | uint16_t nflat_ls = (nlong-nshort)/2; |
354 | |
|
355 | 0 | assert(window_sequence != EIGHT_SHORT_SEQUENCE); |
356 | | |
357 | 0 | #ifdef LD_DEC |
358 | 0 | if (object_type == LD) |
359 | 0 | { |
360 | 0 | window_long = fb->ld_window[window_shape]; |
361 | 0 | window_long_prev = fb->ld_window[window_shape_prev]; |
362 | 0 | } else { |
363 | 0 | #endif |
364 | 0 | window_long = fb->long_window[window_shape]; |
365 | 0 | window_long_prev = fb->long_window[window_shape_prev]; |
366 | 0 | window_short = fb->short_window[window_shape]; |
367 | 0 | window_short_prev = fb->short_window[window_shape_prev]; |
368 | 0 | #ifdef LD_DEC |
369 | 0 | } |
370 | 0 | #endif |
371 | |
|
372 | 0 | switch(window_sequence) |
373 | 0 | { |
374 | 0 | case ONLY_LONG_SEQUENCE: |
375 | 0 | for (i = nlong-1; i >= 0; i--) |
376 | 0 | { |
377 | 0 | windowed_buf[i] = MUL_F(in_data[i], window_long_prev[i]); |
378 | 0 | windowed_buf[i+nlong] = MUL_F(in_data[i+nlong], window_long[nlong-1-i]); |
379 | 0 | } |
380 | 0 | mdct(fb, windowed_buf, out_mdct, 2*nlong); |
381 | 0 | break; |
382 | | |
383 | 0 | case LONG_START_SEQUENCE: |
384 | 0 | for (i = 0; i < nlong; i++) |
385 | 0 | windowed_buf[i] = MUL_F(in_data[i], window_long_prev[i]); |
386 | 0 | for (i = 0; i < nflat_ls; i++) |
387 | 0 | windowed_buf[i+nlong] = in_data[i+nlong]; |
388 | 0 | for (i = 0; i < nshort; i++) |
389 | 0 | windowed_buf[i+nlong+nflat_ls] = MUL_F(in_data[i+nlong+nflat_ls], window_short[nshort-1-i]); |
390 | 0 | for (i = 0; i < nflat_ls; i++) |
391 | 0 | windowed_buf[i+nlong+nflat_ls+nshort] = 0; |
392 | 0 | mdct(fb, windowed_buf, out_mdct, 2*nlong); |
393 | 0 | break; |
394 | | |
395 | 0 | case LONG_STOP_SEQUENCE: |
396 | 0 | for (i = 0; i < nflat_ls; i++) |
397 | 0 | windowed_buf[i] = 0; |
398 | 0 | for (i = 0; i < nshort; i++) |
399 | 0 | windowed_buf[i+nflat_ls] = MUL_F(in_data[i+nflat_ls], window_short_prev[i]); |
400 | 0 | for (i = 0; i < nflat_ls; i++) |
401 | 0 | windowed_buf[i+nflat_ls+nshort] = in_data[i+nflat_ls+nshort]; |
402 | 0 | for (i = 0; i < nlong; i++) |
403 | 0 | windowed_buf[i+nlong] = MUL_F(in_data[i+nlong], window_long[nlong-1-i]); |
404 | 0 | mdct(fb, windowed_buf, out_mdct, 2*nlong); |
405 | 0 | break; |
406 | 0 | } |
407 | 0 | } |
408 | | #endif |