mirror of
https://github.com/vdr-projects/vdr.git
synced 2025-03-01 10:50:46 +00:00
- Fixed displaying colored button texts that are too long. - Suppressing replay progress display when replaying a DVD. - Updated channels.conf.cable (thanks to Uwe Scheffler). - Updated French OSD texts (thanks to Jean-Claude Repetto). - Improved AC3 decoding when replaying DVDs (thanks to Matjaz Thaler). - Fixed handling DVB card indexes when using only one card in a multi-card system. - Changed the 'Eject DVD' button text to a simple 'Eject' (the German text was too long...). - Made the font file generation more stable (thanks to Artur Skawina). - Changed the default value for the "DiSEqC" setup parameter to "off". - The new command line option '-E' can be used to define where the EPG data shall be written to. This is especially useful if VDR runs in a system that turns off the video disk when it is not used, and therefore needs to write the EPG file to a ramdisk (or turn off writing it alltogether). See 'vdr --help' for details. - Making sure the disk is up and running before starting recording (this is important for systems that turn off the video disk when it is not used). - Added the "Jump" function in replay mode (thanks to Stefan Huelswitt). See the description of the "Red" key in MANUAL under "Replay Control" for details. - Fixed displaying editing marks when toggling a mark in "pause" mode. - If there is no free DVB device to record, the log message will now be given only once. - Made I/O more robust by handling EINTR (thanks to Werner Fink).
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;
|
|
}
|
|
}
|