fft.c
上传用户:riyaled888
上传日期:2009-03-27
资源大小:7338k
文件大小:7k
源码类别:

多媒体

开发平台:

MultiPlatform

  1. /*****************************************************************************
  2.  * fft.c: Iterative implementation of a FFT
  3.  *****************************************************************************
  4.  * $Id: fft.c 6961 2004-03-05 17:34:23Z sam $
  5.  *
  6.  * Mainly taken from XMMS's code
  7.  *
  8.  * Authors: Richard Boulton <richard@tartarus.org>
  9.  *          Ralph Loader <suckfish@ihug.co.nz>
  10.  *
  11.  * This program is free software; you can redistribute it and/or modify
  12.  * it under the terms of the GNU General Public License as published by
  13.  * the Free Software Foundation; either version 2 of the License, or
  14.  * (at your option) any later version.
  15.  *
  16.  * This program is distributed in the hope that it will be useful,
  17.  * but WITHOUT ANY WARRANTY; without even the implied warranty of
  18.  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
  19.  * GNU General Public License for more details.
  20.  *
  21.  * You should have received a copy of the GNU General Public License
  22.  * along with this program; if not, write to the Free Software
  23.  * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA  02111, USA.
  24.  *****************************************************************************/
  25. #include "fft.h"
  26. #include <stdlib.h>
  27. #include <math.h>
  28. #ifndef PI
  29.  #ifdef M_PI
  30.   #define PI M_PI
  31.  #else
  32.   #define PI            3.14159265358979323846  /* pi */
  33.  #endif
  34. #endif
  35. /******************************************************************************
  36.  * Local prototypes
  37.  *****************************************************************************/
  38. static void fft_prepare(const sound_sample *input, float * re, float * im);
  39. static void fft_calculate(float * re, float * im);
  40. static void fft_output(const float *re, const float *im, float *output);
  41. static int reverseBits(unsigned int initial);
  42. /* Table to speed up bit reverse copy */
  43. static unsigned int bitReverse[FFT_BUFFER_SIZE];
  44. /* The next two tables could be made to use less space in memory, since they
  45.  * overlap hugely, but hey. */
  46. static float sintable[FFT_BUFFER_SIZE / 2];
  47. static float costable[FFT_BUFFER_SIZE / 2];
  48. /*****************************************************************************
  49.  * These functions are the ones called externally
  50.  *****************************************************************************/
  51. /*
  52.  * Initialisation routine - sets up tables and space to work in.
  53.  * Returns a pointer to internal state, to be used when performing calls.
  54.  * On error, returns NULL.
  55.  * The pointer should be freed when it is finished with, by fft_close().
  56.  */
  57. fft_state *visual_fft_init(void)
  58. {
  59.     fft_state *p_state;
  60.     unsigned int i;
  61.     p_state = (fft_state *) malloc (sizeof(fft_state));
  62.     if(! p_state )
  63.         return NULL;
  64.     for(i = 0; i < FFT_BUFFER_SIZE; i++)
  65.     {
  66.         bitReverse[i] = reverseBits(i);
  67.     }
  68.     for(i = 0; i < FFT_BUFFER_SIZE / 2; i++)
  69.     {
  70.         float j = 2 * PI * i / FFT_BUFFER_SIZE;
  71.         costable[i] = cos(j);
  72.         sintable[i] = sin(j);
  73.     }
  74.     return p_state;
  75. }
  76. /*
  77.  * Do all the steps of the FFT, taking as input sound data (as described in
  78.  * sound.h) and returning the intensities of each frequency as floats in the
  79.  * range 0 to ((FFT_BUFFER_SIZE / 2) * 32768) ^ 2
  80.  *
  81.  * The input array is assumed to have FFT_BUFFER_SIZE elements,
  82.  * and the output array is assumed to have (FFT_BUFFER_SIZE / 2 + 1) elements.
  83.  * state is a (non-NULL) pointer returned by visual_fft_init.
  84.  */
  85. void fft_perform(const sound_sample *input, float *output, fft_state *state) {
  86.     /* Convert data from sound format to be ready for FFT */
  87.     fft_prepare(input, state->real, state->imag);
  88.     /* Do the actual FFT */
  89.     fft_calculate(state->real, state->imag);
  90.     /* Convert the FFT output into intensities */
  91.     fft_output(state->real, state->imag, output);
  92. }
  93. /*
  94.  * Free the state.
  95.  */
  96. void fft_close(fft_state *state) {
  97.     if(state) free(state);
  98. }
  99. /*****************************************************************************
  100.  * These functions are called from the other ones
  101.  *****************************************************************************/
  102. /*
  103.  * Prepare data to perform an FFT on
  104.  */
  105. static void fft_prepare(const sound_sample *input, float * re, float * im) {
  106.     unsigned int i;
  107.     float *p_real = re;
  108.     float *p_imag = im;
  109.     /* Get input, in reverse bit order */
  110.     for(i = 0; i < FFT_BUFFER_SIZE; i++)
  111.     {
  112.         *p_real++ = input[bitReverse[i]];
  113.         *p_imag++ = 0;
  114.     }
  115. }
  116. /*
  117.  * Take result of an FFT and calculate the intensities of each frequency
  118.  * Note: only produces half as many data points as the input had.
  119.  */
  120. static void fft_output(const float * re, const float * im, float *output)
  121. {
  122.     float *p_output = output;
  123.     const float *p_real   = re;
  124.     const float *p_imag   = im;
  125.     float *p_end    = output + FFT_BUFFER_SIZE / 2;
  126.     while(p_output <= p_end)
  127.     {
  128.         *p_output = (*p_real * *p_real) + (*p_imag * *p_imag);
  129.         p_output++; p_real++; p_imag++;
  130.     }
  131.     /* Do divisions to keep the constant and highest frequency terms in scale
  132.      * with the other terms. */
  133.     *output /= 4;
  134.     *p_end /= 4;
  135. }
  136. /*
  137.  * Actually perform the FFT
  138.  */
  139. static void fft_calculate(float * re, float * im)
  140. {
  141.     unsigned int i, j, k;
  142.     unsigned int exchanges;
  143.     float fact_real, fact_imag;
  144.     float tmp_real, tmp_imag;
  145.     unsigned int factfact;
  146.     /* Set up some variables to reduce calculation in the loops */
  147.     exchanges = 1;
  148.     factfact = FFT_BUFFER_SIZE / 2;
  149.     /* Loop through the divide and conquer steps */
  150.     for(i = FFT_BUFFER_SIZE_LOG; i != 0; i--) {
  151.         /* In this step, we have 2 ^ (i - 1) exchange groups, each with
  152.          * 2 ^ (FFT_BUFFER_SIZE_LOG - i) exchanges
  153.          */
  154.         /* Loop through the exchanges in a group */
  155.         for(j = 0; j != exchanges; j++) {
  156.             /* Work out factor for this exchange
  157.              * factor ^ (exchanges) = -1
  158.              * So, real = cos(j * PI / exchanges),
  159.              *     imag = sin(j * PI / exchanges)
  160.              */
  161.             fact_real = costable[j * factfact];
  162.             fact_imag = sintable[j * factfact];
  163.             /* Loop through all the exchange groups */
  164.             for(k = j; k < FFT_BUFFER_SIZE; k += exchanges << 1) {
  165.                 int k1 = k + exchanges;
  166.                 tmp_real = fact_real * re[k1] - fact_imag * im[k1];
  167.                 tmp_imag = fact_real * im[k1] + fact_imag * re[k1];
  168.                 re[k1] = re[k] - tmp_real;
  169.                 im[k1] = im[k] - tmp_imag;
  170.                 re[k]  += tmp_real;
  171.                 im[k]  += tmp_imag;
  172.             }
  173.         }
  174.         exchanges <<= 1;
  175.         factfact >>= 1;
  176.     }
  177. }
  178. static int reverseBits(unsigned int initial)
  179. {
  180.     unsigned int reversed = 0, loop;
  181.     for(loop = 0; loop < FFT_BUFFER_SIZE_LOG; loop++) {
  182.         reversed <<= 1;
  183.         reversed += (initial & 1);
  184.         initial >>= 1;
  185.     }
  186.     return reversed;
  187. }