mirror of
https://github.com/VDR4Arch/vdr.git
synced 2023-10-10 13:36:52 +02:00
219 lines
5.7 KiB
C
219 lines
5.7 KiB
C
|
/*
|
||
|
* imdct.c
|
||
|
*
|
||
|
* Copyright (C) Aaron Holtzman - May 1999
|
||
|
*
|
||
|
* This file is part of ac3dec, a free Dolby AC-3 stream decoder.
|
||
|
*
|
||
|
* ac3dec 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, or (at your option)
|
||
|
* any later version.
|
||
|
*
|
||
|
* ac3dec is distributed in the hope that it will be useful,
|
||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||
|
* GNU General Public License for more details.
|
||
|
*
|
||
|
* You should have received a copy of the GNU General Public License
|
||
|
* along with GNU Make; see the file COPYING. If not, write to
|
||
|
* the Free Software Foundation, 675 Mass Ave, Cambridge, MA 02139, USA.
|
||
|
*
|
||
|
*
|
||
|
*/
|
||
|
|
||
|
#include <stdlib.h>
|
||
|
#include <stdio.h>
|
||
|
#include <math.h>
|
||
|
#include "ac3.h"
|
||
|
#include "ac3_internal.h"
|
||
|
|
||
|
#include "downmix.h"
|
||
|
#include "imdct_c.h"
|
||
|
#include "srfft.h"
|
||
|
|
||
|
|
||
|
#define N 512
|
||
|
|
||
|
extern void (*imdct_do_512) (float data[],float delay[]);
|
||
|
extern void (*imdct_do_512_nol) (float data[], float delay[]);
|
||
|
extern void (*fft_64p) (complex_t *);
|
||
|
|
||
|
extern const int pm128[];
|
||
|
extern float window[];
|
||
|
extern complex_t buf[128];
|
||
|
|
||
|
extern void fft_64p_c (complex_t *);
|
||
|
extern void fft_128p_c (complex_t *);
|
||
|
|
||
|
static void imdct_do_512_c (float data[],float delay[]);
|
||
|
static void imdct_do_512_nol_c (float data[], float delay[]);
|
||
|
|
||
|
/* Twiddle factors for IMDCT */
|
||
|
static float xcos1[128] __attribute__((aligned(16)));
|
||
|
static float xsin1[128] __attribute__((aligned(16)));
|
||
|
|
||
|
|
||
|
int imdct_init_c (void)
|
||
|
{
|
||
|
int i;
|
||
|
float scale = 255.99609372;
|
||
|
|
||
|
imdct_do_512 = imdct_do_512_c;
|
||
|
imdct_do_512_nol = imdct_do_512_nol_c;
|
||
|
fft_64p = fft_64p_c;
|
||
|
|
||
|
/* Twiddle factors to turn IFFT into IMDCT */
|
||
|
|
||
|
for (i=0; i < 128; i++) {
|
||
|
xcos1[i] = cos(2.0f * M_PI * (8*i+1)/(8*N)) * scale;
|
||
|
xsin1[i] = sin(2.0f * M_PI * (8*i+1)/(8*N)) * scale;
|
||
|
}
|
||
|
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
|
||
|
static void imdct_do_512_c (float data[], float delay[])
|
||
|
{
|
||
|
int i, j;
|
||
|
float tmp_a_r, tmp_a_i;
|
||
|
float *data_ptr;
|
||
|
float *delay_ptr;
|
||
|
float *window_ptr;
|
||
|
|
||
|
// 512 IMDCT with source and dest data in 'data'
|
||
|
// Pre IFFT complex multiply plus IFFT complex conjugate
|
||
|
|
||
|
for( i=0; i < 128; i++) {
|
||
|
j = pm128[i];
|
||
|
//a = (data[256-2*j-1] - data[2*j]) * (xcos1[j] + xsin1[j]);
|
||
|
//c = data[2*j] * xcos1[j];
|
||
|
//b = data[256-2*j-1] * xsin1[j];
|
||
|
//buf1[i].re = a - b + c;
|
||
|
//buf1[i].im = b + c;
|
||
|
buf[i].re = (data[256-2*j-1] * xcos1[j]) - (data[2*j] * xsin1[j]);
|
||
|
buf[i].im = -1.0 * (data[2*j] * xcos1[j] + data[256-2*j-1] * xsin1[j]);
|
||
|
}
|
||
|
|
||
|
fft_128p_c (&buf[0]);
|
||
|
|
||
|
// Post IFFT complex multiply plus IFFT complex conjugate
|
||
|
for (i=0; i < 128; i++) {
|
||
|
tmp_a_r = buf[i].re;
|
||
|
tmp_a_i = buf[i].im;
|
||
|
//a = (tmp_a_r - tmp_a_i) * (xcos1[j] + xsin1[j]);
|
||
|
//b = tmp_a_r * xsin1[j];
|
||
|
//c = tmp_a_i * xcos1[j];
|
||
|
//buf[j].re = a - b + c;
|
||
|
//buf[j].im = b + c;
|
||
|
buf[i].re =(tmp_a_r * xcos1[i]) + (tmp_a_i * xsin1[i]);
|
||
|
buf[i].im =(tmp_a_r * xsin1[i]) - (tmp_a_i * xcos1[i]);
|
||
|
}
|
||
|
|
||
|
data_ptr = data;
|
||
|
delay_ptr = delay;
|
||
|
window_ptr = window;
|
||
|
|
||
|
// Window and convert to real valued signal
|
||
|
for (i=0; i< 64; i++) {
|
||
|
*data_ptr++ = -buf[64+i].im * *window_ptr++ + *delay_ptr++;
|
||
|
*data_ptr++ = buf[64-i-1].re * *window_ptr++ + *delay_ptr++;
|
||
|
}
|
||
|
|
||
|
for(i=0; i< 64; i++) {
|
||
|
*data_ptr++ = -buf[i].re * *window_ptr++ + *delay_ptr++;
|
||
|
*data_ptr++ = buf[128-i-1].im * *window_ptr++ + *delay_ptr++;
|
||
|
}
|
||
|
|
||
|
// The trailing edge of the window goes into the delay line
|
||
|
delay_ptr = delay;
|
||
|
|
||
|
for(i=0; i< 64; i++) {
|
||
|
*delay_ptr++ = -buf[64+i].re * *--window_ptr;
|
||
|
*delay_ptr++ = buf[64-i-1].im * *--window_ptr;
|
||
|
}
|
||
|
|
||
|
for(i=0; i<64; i++) {
|
||
|
*delay_ptr++ = buf[i].im * *--window_ptr;
|
||
|
*delay_ptr++ = -buf[128-i-1].re * *--window_ptr;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
|
||
|
static void imdct_do_512_nol_c (float data[], float delay[])
|
||
|
{
|
||
|
int i, j;
|
||
|
|
||
|
float tmp_a_i;
|
||
|
float tmp_a_r;
|
||
|
|
||
|
float *data_ptr;
|
||
|
float *delay_ptr;
|
||
|
float *window_ptr;
|
||
|
|
||
|
//
|
||
|
// 512 IMDCT with source and dest data in 'data'
|
||
|
//
|
||
|
|
||
|
// Pre IFFT complex multiply plus IFFT cmplx conjugate
|
||
|
|
||
|
for( i=0; i < 128; i++) {
|
||
|
/* z[i] = (X[256-2*i-1] + j * X[2*i]) * (xcos1[i] + j * xsin1[i]) */
|
||
|
j = pm128[i];
|
||
|
//a = (data[256-2*j-1] - data[2*j]) * (xcos1[j] + xsin1[j]);
|
||
|
//c = data[2*j] * xcos1[j];
|
||
|
//b = data[256-2*j-1] * xsin1[j];
|
||
|
//buf1[i].re = a - b + c;
|
||
|
|
||
|
//buf1[i].im = b + c;
|
||
|
buf[i].re = (data[256-2*j-1] * xcos1[j]) - (data[2*j] * xsin1[j]);
|
||
|
buf[i].im = -1.0 * (data[2*j] * xcos1[j] + data[256-2*j-1] * xsin1[j]);
|
||
|
}
|
||
|
|
||
|
fft_128p_c (&buf[0]);
|
||
|
|
||
|
/* Post IFFT complex multiply plus IFFT complex conjugate*/
|
||
|
for (i=0; i < 128; i++) {
|
||
|
/* y[n] = z[n] * (xcos1[n] + j * xsin1[n]) ; */
|
||
|
/* int j1 = i; */
|
||
|
tmp_a_r = buf[i].re;
|
||
|
tmp_a_i = buf[i].im;
|
||
|
//a = (tmp_a_r - tmp_a_i) * (xcos1[j] + xsin1[j]);
|
||
|
//b = tmp_a_r * xsin1[j];
|
||
|
//c = tmp_a_i * xcos1[j];
|
||
|
//buf[j].re = a - b + c;
|
||
|
//buf[j].im = b + c;
|
||
|
buf[i].re =(tmp_a_r * xcos1[i]) + (tmp_a_i * xsin1[i]);
|
||
|
buf[i].im =(tmp_a_r * xsin1[i]) - (tmp_a_i * xcos1[i]);
|
||
|
}
|
||
|
|
||
|
data_ptr = data;
|
||
|
delay_ptr = delay;
|
||
|
window_ptr = window;
|
||
|
|
||
|
/* Window and convert to real valued signal, no overlap here*/
|
||
|
for (i=0; i< 64; i++) {
|
||
|
*data_ptr++ = -buf[64+i].im * *window_ptr++;
|
||
|
*data_ptr++ = buf[64-i-1].re * *window_ptr++;
|
||
|
}
|
||
|
|
||
|
for(i=0; i< 64; i++) {
|
||
|
*data_ptr++ = -buf[i].re * *window_ptr++;
|
||
|
*data_ptr++ = buf[128-i-1].im * *window_ptr++;
|
||
|
}
|
||
|
|
||
|
/* The trailing edge of the window goes into the delay line */
|
||
|
delay_ptr = delay;
|
||
|
|
||
|
for(i=0; i< 64; i++) {
|
||
|
*delay_ptr++ = -buf[64+i].re * *--window_ptr;
|
||
|
*delay_ptr++ = buf[64-i-1].im * *--window_ptr;
|
||
|
}
|
||
|
|
||
|
for(i=0; i<64; i++) {
|
||
|
*delay_ptr++ = buf[i].im * *--window_ptr;
|
||
|
*delay_ptr++ = -buf[128-i-1].re * *--window_ptr;
|
||
|
}
|
||
|
}
|