Blame libcelt/psy.c

Packit 664db3
/* (C) 2007 Jean-Marc Valin, CSIRO
Packit 664db3
*/
Packit 664db3
/*
Packit 664db3
   Redistribution and use in source and binary forms, with or without
Packit 664db3
   modification, are permitted provided that the following conditions
Packit 664db3
   are met:
Packit 664db3
   
Packit 664db3
   - Redistributions of source code must retain the above copyright
Packit 664db3
   notice, this list of conditions and the following disclaimer.
Packit 664db3
   
Packit 664db3
   - Redistributions in binary form must reproduce the above copyright
Packit 664db3
   notice, this list of conditions and the following disclaimer in the
Packit 664db3
   documentation and/or other materials provided with the distribution.
Packit 664db3
   
Packit 664db3
   - Neither the name of the Xiph.org Foundation nor the names of its
Packit 664db3
   contributors may be used to endorse or promote products derived from
Packit 664db3
   this software without specific prior written permission.
Packit 664db3
   
Packit 664db3
   THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
Packit 664db3
   ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
Packit 664db3
   LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
Packit 664db3
   A PARTICULAR PURPOSE ARE DISCLAIMED.  IN NO EVENT SHALL THE FOUNDATION OR
Packit 664db3
   CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
Packit 664db3
   EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
Packit 664db3
   PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
Packit 664db3
   PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF
Packit 664db3
   LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
Packit 664db3
   NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
Packit 664db3
   SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
Packit 664db3
*/
Packit 664db3
Packit 664db3
#ifdef HAVE_CONFIG_H
Packit 664db3
#include "config.h"
Packit 664db3
#endif
Packit 664db3
Packit 664db3
#include "psy.h"
Packit 664db3
#include <math.h>
Packit 664db3
#include "os_support.h"
Packit 664db3
#include "arch.h"
Packit 664db3
#include "stack_alloc.h"
Packit 664db3
#include "mathops.h"
Packit 664db3
Packit 664db3
/* The Vorbis freq<->Bark mapping */
Packit 664db3
#define toBARK(n)   (13.1f*atan(.00074f*(n))+2.24f*atan((n)*(n)*1.85e-8f)+1e-4f*(n))
Packit 664db3
#define fromBARK(z) (102.f*(z)-2.f*pow(z,2.f)+.4f*pow(z,3.f)+pow(1.46f,z)-1.f)
Packit 664db3
Packit 664db3
#ifndef STATIC_MODES
Packit 664db3
/* Psychoacoustic spreading function. The idea here is compute a first order 
Packit 664db3
   recursive filter. The filter coefficient is frequency dependent and 
Packit 664db3
   chosen such that we have a -10dB/Bark slope on the right side and a -25dB/Bark
Packit 664db3
   slope on the left side. */
Packit 664db3
void psydecay_init(struct PsyDecay *decay, int len, celt_int32_t Fs)
Packit 664db3
{
Packit 664db3
   int i;
Packit 664db3
   celt_word16_t *decayR = (celt_word16_t*)celt_alloc(sizeof(celt_word16_t)*len);
Packit 664db3
   /*decay->decayL = celt_alloc(sizeof(celt_word16_t)*len);*/
Packit 664db3
   for (i=0;i
Packit 664db3
   {
Packit 664db3
      float f;
Packit 664db3
      float deriv;
Packit 664db3
      /* Real frequency (in Hz) */
Packit 664db3
      f = Fs*i*(1/(2.f*len));
Packit 664db3
      /* This is the derivative of the Vorbis freq->Bark function (see above) */
Packit 664db3
      deriv = (8.288e-8 * f)/(3.4225e-16 *f*f*f*f + 1) +  .009694/(5.476e-7 *f*f + 1) + 1e-4;
Packit 664db3
      /* Back to FFT bin units */
Packit 664db3
      deriv *= Fs*(1/(2.f*len));
Packit 664db3
      /* decay corresponding to -10dB/Bark */
Packit 664db3
      decayR[i] = Q15ONE*pow(.1f, deriv);
Packit 664db3
      /* decay corresponding to -25dB/Bark */
Packit 664db3
      /*decay->decayL[i] = Q15ONE*pow(0.0031623f, deriv);*/
Packit 664db3
      /*printf ("%f %f\n", decayL[i], decayR[i]);*/
Packit 664db3
   }
Packit 664db3
   decay->decayR = decayR;
Packit 664db3
}
Packit 664db3
Packit 664db3
void psydecay_clear(struct PsyDecay *decay)
Packit 664db3
{
Packit 664db3
   celt_free((celt_word16_t *)decay->decayR);
Packit 664db3
   /*celt_free(decay->decayL);*/
Packit 664db3
}
Packit 664db3
#endif
Packit 664db3
Packit 664db3
static void spreading_func(const struct PsyDecay *d, celt_word32_t * restrict psd, int len)
Packit 664db3
{
Packit 664db3
   int i;
Packit 664db3
   celt_word32_t mem;
Packit 664db3
   /*for (i=0;i
Packit 664db3
   /* Compute right slope (-10 dB/Bark) */
Packit 664db3
   mem=psd[0];
Packit 664db3
   for (i=0;i
Packit 664db3
   {
Packit 664db3
      /* psd = (1-decay)*psd + decay*mem */
Packit 664db3
      psd[i] = EPSILON + psd[i] + MULT16_32_Q15(d->decayR[i],mem-psd[i]);
Packit 664db3
      mem = psd[i];
Packit 664db3
   }
Packit 664db3
   /* Compute left slope (-25 dB/Bark) */
Packit 664db3
   mem=psd[len-1];
Packit 664db3
   for (i=len-1;i>=0;i--)
Packit 664db3
   {
Packit 664db3
      /* Left side has around twice the slope as the right side, so we just
Packit 664db3
         square the coef instead of storing two sets of decay coefs */
Packit 664db3
      celt_word16_t decayL = MULT16_16_Q15(d->decayR[i], d->decayR[i]);
Packit 664db3
      /* psd = (1-decay)*psd + decay*mem */
Packit 664db3
      psd[i] = EPSILON + psd[i] + MULT16_32_Q15(decayL,mem-psd[i]);
Packit 664db3
      mem = psd[i];
Packit 664db3
   }
Packit 664db3
   /*for (i=0;i
Packit 664db3
#if 0 /* Prints signal and mask energy per critical band */
Packit 664db3
   for (i=0;i<25;i++)
Packit 664db3
   {
Packit 664db3
      int start,end;
Packit 664db3
      int j;
Packit 664db3
      celt_word32_t Esig=0, Emask=0;
Packit 664db3
      start = (int)floor(fromBARK((float)i)*(2*len)/Fs);
Packit 664db3
      if (start<0)
Packit 664db3
         start = 0;
Packit 664db3
      end = (int)ceil(fromBARK((float)(i+1))*(2*len)/Fs);
Packit 664db3
      if (end<=start)
Packit 664db3
         end = start+1;
Packit 664db3
      if (end>len-1)
Packit 664db3
         end = len-1;
Packit 664db3
      for (j=start;j
Packit 664db3
      {
Packit 664db3
         Esig += psd[j];
Packit 664db3
         Emask += mask[j];
Packit 664db3
      }
Packit 664db3
      printf ("%f %f ", Esig, Emask);
Packit 664db3
   }
Packit 664db3
   printf ("\n");
Packit 664db3
#endif
Packit 664db3
}
Packit 664db3
Packit 664db3
/* Compute a marking threshold from the spectrum X. */
Packit 664db3
void compute_masking(const struct PsyDecay *decay, celt_word16_t *X, celt_mask_t * restrict mask, int len)
Packit 664db3
{
Packit 664db3
   int i;
Packit 664db3
   int N;
Packit 664db3
   N=len>>1;
Packit 664db3
   mask[0] = MULT16_16(X[0], X[0]);
Packit 664db3
   for (i=1;i
Packit 664db3
      mask[i] = ADD32(MULT16_16(X[i*2], X[i*2]), MULT16_16(X[i*2+1], X[i*2+1]));
Packit 664db3
   /* TODO: Do tone masking */
Packit 664db3
   /* Noise masking */
Packit 664db3
   spreading_func(decay, mask, N);
Packit 664db3
}
Packit 664db3
Packit 664db3
#ifdef EXP_PSY /* Not needed for now, but will be useful in the future */
Packit 664db3
void compute_mdct_masking(const struct PsyDecay *decay, celt_word32_t *X, celt_word16_t *tonality, celt_word16_t *long_window, celt_mask_t *mask, int len)
Packit 664db3
{
Packit 664db3
   int i;
Packit 664db3
   VARDECL(float, psd);
Packit 664db3
   SAVE_STACK;
Packit 664db3
   ALLOC(psd, len, float);
Packit 664db3
   for (i=0;i
Packit 664db3
      psd[i] = X[i]*X[i]*tonality[i];
Packit 664db3
   for (i=1;i
Packit 664db3
      mask[i] = .5*psd[i] + .25*(psd[i-1]+psd[i+1]);
Packit 664db3
   /*psd[0] = .5*mask[0]+.25*(mask[1]+mask[2]);*/
Packit 664db3
   mask[0] = .5*psd[0]+.5*psd[1];
Packit 664db3
   mask[len-1] = .5*(psd[len-1]+psd[len-2]);
Packit 664db3
   /* TODO: Do tone masking */
Packit 664db3
   /* Noise masking */
Packit 664db3
   spreading_func(decay, mask, len);
Packit 664db3
   RESTORE_STACK;  
Packit 664db3
}
Packit 664db3
Packit 664db3
void compute_tonality(const CELTMode *m, celt_word16_t * restrict X, celt_word16_t * mem, int len, celt_word16_t *tonality, int mdct_size)
Packit 664db3
{
Packit 664db3
   int i;
Packit 664db3
   celt_word16_t norm_1;
Packit 664db3
   celt_word16_t *mem2;
Packit 664db3
   int N = len>>2;
Packit 664db3
Packit 664db3
   mem2 = mem+2*N;
Packit 664db3
   X[0] = 0;
Packit 664db3
   X[1] = 0;
Packit 664db3
   tonality[0] = 1;
Packit 664db3
   for (i=1;i
Packit 664db3
   {
Packit 664db3
      celt_word16_t re, im, re2, im2;
Packit 664db3
      re = X[2*i];
Packit 664db3
      im = X[2*i+1];
Packit 664db3
      /* Normalise spectrum */
Packit 664db3
      norm_1 = celt_rsqrt(.01+MAC16_16(MULT16_16(re,re), im,im));
Packit 664db3
      re = MULT16_16(re, norm_1);
Packit 664db3
      im = MULT16_16(im, norm_1);
Packit 664db3
      /* Phase derivative */
Packit 664db3
      re2 = re*mem[2*i] + im*mem[2*i+1];
Packit 664db3
      im2 = im*mem[2*i] - re*mem[2*i+1];
Packit 664db3
      mem[2*i] = re;
Packit 664db3
      mem[2*i+1] = im;
Packit 664db3
      /* Phase second derivative */
Packit 664db3
      re = re2*mem2[2*i] + im2*mem2[2*i+1];
Packit 664db3
      im = im2*mem2[2*i] - re2*mem2[2*i+1];
Packit 664db3
      mem2[2*i] = re2;
Packit 664db3
      mem2[2*i+1] = im2;
Packit 664db3
      /*printf ("%f ", re);*/
Packit 664db3
      X[2*i] = re;
Packit 664db3
      X[2*i+1] = im;
Packit 664db3
   }
Packit 664db3
   /*printf ("\n");*/
Packit 664db3
   for (i=0;i
Packit 664db3
   {
Packit 664db3
      tonality[i] = 1.0-X[2*i]*X[2*i]*X[2*i];
Packit 664db3
      if (tonality[i]>1)
Packit 664db3
         tonality[i] = 1;
Packit 664db3
      if (tonality[i]<.02)
Packit 664db3
         tonality[i]=.02;
Packit 664db3
   }
Packit 664db3
}
Packit 664db3
#endif