144 lines
4.6 KiB
C
144 lines
4.6 KiB
C
/*
|
|
* g722_1 - a library for the G.722.1 and Annex C codecs
|
|
*
|
|
* coef2sam.c
|
|
*
|
|
* Adapted by Steve Underwood <steveu@coppice.org> from the reference
|
|
* code supplied with ITU G.722.1, which is:
|
|
*
|
|
* © 2004 Polycom, Inc.
|
|
* All rights reserved.
|
|
*
|
|
* This program 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.
|
|
*
|
|
* $Id: coef2sam.c,v 1.10 2008/10/02 11:43:54 steveu Exp $
|
|
*/
|
|
|
|
/*! \file */
|
|
|
|
#if defined(HAVE_CONFIG_H)
|
|
#include <config.h>
|
|
#endif
|
|
|
|
#include <inttypes.h>
|
|
#include <stdlib.h>
|
|
|
|
#include "g722_1/g722_1.h"
|
|
|
|
#include "defs.h"
|
|
#include "coef2sam.h"
|
|
|
|
/*************************************************************************************
|
|
|
|
Purpose: Convert Reversed MLT (Modulated Lapped Transform) Coefficients to Samples
|
|
|
|
The "Reversed MLT" is an overlapped block transform which uses even symmetry
|
|
on the left, odd symmetry on the right and a Type IV DCT as the block transform.
|
|
It is thus similar to a MLT which uses odd symmetry on the left, even symmetry
|
|
on the right and a Type IV DST as the block transform. In fact, it is equivalent
|
|
to reversing the order of the samples, performing an MLT and then negating all
|
|
the even-numbered coefficients.
|
|
***************************************************************************/
|
|
|
|
#if defined(G722_1_USE_FIXED_POINT)
|
|
void rmlt_coefs_to_samples(int16_t coefs[],
|
|
int16_t old_samples[],
|
|
int16_t out_samples[],
|
|
int dct_length,
|
|
int16_t mag_shift)
|
|
{
|
|
int i;
|
|
int half_dct_length;
|
|
int last;
|
|
int16_t new_samples[MAX_DCT_LENGTH];
|
|
const int16_t *win;
|
|
int32_t sum;
|
|
|
|
half_dct_length = dct_length >> 1;
|
|
|
|
/* Perform a Type IV (inverse) DCT on the coefficients */
|
|
dct_type_iv_s(coefs, new_samples, dct_length);
|
|
|
|
if (mag_shift > 0)
|
|
{
|
|
for (i = 0; i < dct_length; i++)
|
|
new_samples[i] = shr(new_samples[i], mag_shift);
|
|
}
|
|
else if (mag_shift < 0)
|
|
{
|
|
mag_shift = negate(mag_shift);
|
|
for (i = 0; i < dct_length; i++)
|
|
new_samples[i] = shl(new_samples[i], mag_shift);
|
|
}
|
|
|
|
if (dct_length == DCT_LENGTH)
|
|
win = rmlt_to_samples_window;
|
|
else
|
|
win = max_rmlt_to_samples_window;
|
|
last = half_dct_length - 1;
|
|
for (i = 0; i < half_dct_length; i++)
|
|
{
|
|
/* Get the first half of the windowed samples */
|
|
sum = 0L;
|
|
sum = L_mac(sum, win[i], new_samples[last - i]);
|
|
sum = L_mac(sum, win[dct_length - i - 1], old_samples[i]);
|
|
out_samples[i] = xround(L_shl(sum, 2));
|
|
/* Get the second half of the windowed samples */
|
|
sum = 0L;
|
|
sum = L_mac(sum, win[half_dct_length + i], new_samples[i]);
|
|
sum = L_mac(sum, negate(win[last - i]), old_samples[last - i]);
|
|
out_samples[half_dct_length + i] = xround(L_shl(sum, 2));
|
|
}
|
|
|
|
/* Save the second half of the new samples for
|
|
next time, when they will be the old samples. */
|
|
for (i = 0; i < half_dct_length; i++)
|
|
old_samples[i] = new_samples[half_dct_length + i];
|
|
}
|
|
/*- End of function --------------------------------------------------------*/
|
|
#else
|
|
void rmlt_coefs_to_samples(float coefs[],
|
|
float old_samples[],
|
|
float out_samples[],
|
|
int dct_length)
|
|
{
|
|
int i;
|
|
int half_dct_length;
|
|
int last;
|
|
float new_samples[MAX_DCT_LENGTH];
|
|
const float *win;
|
|
float sum;
|
|
|
|
half_dct_length = dct_length >> 1;
|
|
|
|
/* Perform a Type IV (inverse) DCT on the coefficients */
|
|
dct_type_iv(coefs, new_samples, dct_length);
|
|
|
|
if (dct_length == DCT_LENGTH)
|
|
win = rmlt_to_samples_window;
|
|
else
|
|
win = max_rmlt_to_samples_window;
|
|
last = half_dct_length - 1;
|
|
for (i = 0; i < half_dct_length; i++)
|
|
{
|
|
/* Get the first half of the windowed samples */
|
|
sum = win[i]*new_samples[last - i];
|
|
sum += win[dct_length - i - 1]*old_samples[i];
|
|
out_samples[i] = sum;
|
|
/* Get the second half of the windowed samples */
|
|
sum = win[half_dct_length + i]*new_samples[i];
|
|
sum -= win[last - i]*old_samples[last - i];
|
|
out_samples[half_dct_length + i] = sum;
|
|
}
|
|
|
|
/* Save the second half of the new samples for next time, when they will
|
|
be the old samples. */
|
|
for (i = 0; i < half_dct_length; i++)
|
|
old_samples[i] = new_samples[half_dct_length + i];
|
|
}
|
|
/*- End of function --------------------------------------------------------*/
|
|
#endif
|
|
/*- End of file ------------------------------------------------------------*/
|