笃行者的笔记 https://passport2.21ic.com/?581776 [收藏] [复制] [RSS]

日志

FFT C语言实现

已有 2473 次阅读2011-11-24 20:26 |系统分类:DSP| FFT, 算法

最近在做传感信号处理遇到了循环相关问题,快速循环相关算法,先将两组信号FFT变换到频域,进行复数乘积,再IFFT逆变回来即可。傅立叶变换是信号处理的基本模块,这里将这段代码拿出来分享一下,希望对同行有所用。

Dit2FFT.c文件
/********************************************************************************

*File name        : dit2fft.c     
*Deion    : FFT calculation. Decimation in time FFT, radix-2. 

*

*Current version: v2.3 -- get a simple version, remove unnecessary code

*History version: 

*                 v2.2 -- modify the sin/cos table data, and the static

*                 v2.1 -- modify the  name

*                 v2.0 -- separate the bit-reverse block

*                  v1.9 -- rewrite the bit-reverse block

*                 v1.8 -- rewrite the twiddle factor generation, and test fft,ifft

*                  v1.7 -- fix the twiddle factor bug, and test the four 

*                          combinations

*                  v1.6 -- adjust the DIT2_FLOAT to DIT2_FLOAT32

*                 v1.5 -- change DIT2_UINT16 to DIT2_UINT, and extend its scope

*                 v1.4 -- change the name of s

*                 v1.3 -- add the sin/cos table

*                   v1.2 -- add ifft

*                  v1.1 -- add bit-reverse in-place

*                  v1.0 -- achieve the basic  of FFT

*Author            : Wiracle

*Date            :2008-10-23

********************************************************************************/

 
/*inlcude the head file*/

/* the NULL define needed*/

//#include <stdlib.h>

/*

* caculate the magnitude and phase

* -- sqrt()

* -- atan()

*/

#include <math.h>

#include "Alg_Dit2FFT.h"

 
/*the sin(x)/cos(x) table*/

DIT2_FLOAT sin_table[21] = {

0.0,                   // sin(-2*pi/(2^0))     
0.0,                   // sin(-2*pi/(2^1)) 

-1.0,                   // sin(-2*pi/(2^2)) 

-0.70710678118655,       // sin(-2*pi/(2^3)) 

-0.38268343236509,       // sin(-2*pi/(2^4))     
-0.19509032201613,       // sin(-2*pi/(2^5))     
-0.09801714032956,       // sin(-2*pi/(2^6))     
-0.04906767432742,       // sin(-2*pi/(2^7))     
-0.02454122852291,       // sin(-2*pi/(2^8))     
-0.01227153828572,       // sin(-2*pi/(2^9))     
-0.00613588464915,       // sin(-2*pi/(2^10))

-0.00306795676297,       // sin(-2*pi/(2^11))

-0.00153398018628,       // sin(-2*pi/(2^12))

-0.00076699031874,       // sin(-2*pi/(2^13))

-0.00038349518757,       // sin(-2*pi/(2^14))

-0.00019174759731,       // sin(-2*pi/(2^15))

-0.00009587379910,       // sin(-2*pi/(2^16))

-0.00004793689960,       // sin(-2*pi/(2^17))

-0.00002396844981,       // sin(-2*pi/(2^18))

-0.00001198422491,       // sin(-2*pi/(2^19))

-0.00000599211245       // sin(-2*pi/(2^20))

};

 
DIT2_FLOAT cos_table[21] = {

1.0,                   //cos(-2*pi/(2^0))     
-1.0,                   //cos(-2*pi/(2^1))

0.0,                   //cos(-2*pi/(2^2))

0.70710678118655,       //cos(-2*pi/(2^3))

0.92387953251129,       //cos(-2*pi/(2^4))     
0.98078528040323,       //cos(-2*pi/(2^5))     
0.99518472667220,       //cos(-2*pi/(2^6))     
0.99879545620517,       //cos(-2*pi/(2^7))     
0.99969881869620,       //cos(-2*pi/(2^8))     
0.99992470183914,       //cos(-2*pi/(2^9))     
0.99998117528260,       //cos(-2*pi/(2^10))     
0.99999529380958,       //cos(-2*pi/(2^11))     
0.99999882345170,       //cos(-2*pi/(2^12))     
0.99999970586288,       //cos(-2*pi/(2^13))     
0.99999992646572,       //cos(-2*pi/(2^14))     
0.99999998161643,       //cos(-2*pi/(2^15))     
0.99999999540411,       //cos(-2*pi/(2^16))     
0.99999999885103,       //cos(-2*pi/(2^17))     
0.99999999971276,       //cos(-2*pi/(2^18))     
0.99999999992819,       //cos(-2*pi/(2^19))     
0.99999999998205       //cos(-2*pi/(2^20))     
};

static DIT2_UINT DIT2_Idx(DIT2_UINT in);

/********************************************************************************

* name  : DIT2_Rvs

*Input          : DIT2_FFT pointer

*Deion    : Bitreverse the input data, including real and imag part

*    typedef  struct DIT2_FFT_STRUCT

*    {

*        DIT2_FLOAT *real_in;    --input and output buffer

*        DIT2_FLOAT *imag_in;    --input and output buffer(can be NULL)

*        DIT2_FLOAT *real_out;      --unused

*        DIT2_FLOAT *imag_out;      --unused

*        DIT2_FLOAT *mag;           --unused

*        DIT2_FLOAT *phase;         --unused

*        DIT2_UINT size;         --the FFT size, must be 2^N

*        DIT2_UINT stage;           --unused

*        DIT2_UINT transform;       --unused

*    }DIT2_FFT;

********************************************************************************/

#pragma CODE_SECTION(DIT2_Rvs,"ramfuncs");

void DIT2_Rvs(DIT2_FFT *fft)

{

    DIT2_UINT i,j,k,n;

    DIT2_FLOAT tw_real,tw_imag;

 
    /*para check*/

    /*

    if ((fft->real_in == NULL) || (fft->imag_in == NULL))

    {

        while (1);

    }

    */

/*

* 1.loop along the array -- i

* 2.find the bit-reversed index -- k

* 3.check if i < k, if so, swap the X and X[k]

*/

    for (i = 0; i < fft->size; i++)

    {

        k = 0;

        n = i;

        for (j = 0; j < fft->stage; j++)

        {

            k = (k<<1) | (n&1);

            n >>= 1;

        }

 
        if (i < k)

        {

            tw_real = fft->real_in;

            tw_imag = fft->imag_in;

            fft->real_in = fft->real_in[k];

            fft->imag_in = fft->imag_in[k];

            fft->real_in[k] = tw_real;

            fft->imag_in[k] = tw_imag;

        }

    }

}

 
 
/********************************************************************************

* name  : DIT2_Fft

*Input          : DIT2_FFT *fft

*Deion    : used when uncommenting DIT2_BIT_INVERSE_IN_PLACE which achives 

*                 the bit-reversed in-place. As the following, the real part of 

*                 the inputsequence buffer is pointed by real_in, where the 

*                 output--magnitude info is also in. The imaginary part of the 

*                 input sequence buffer is pointed by image_in, where the output--

*                 phase info is alse in.

*    typedef  struct DIT2_FFT_STRUCT

*    {

*        DIT2_FLOAT *real_in;    --input and output buffer

*        DIT2_FLOAT *imag_in;    --input and output buffer

*        DIT2_FLOAT *real_out;      --unused

*        DIT2_FLOAT *imag_out;      --unused

*        DIT2_FLOAT *mag;           --unused

*        DIT2_FLOAT *phase;         --unused

*        DIT2_UINT size;      --the FFT size, must be 2^N

*        DIT2_UINT stage;     --N

*        DIT2_UINT transform; --DIT2_FORWARD_TRANSFORM or DIT2_INVERSE_TRANSFORM

*    }DIT2_FFT;

********************************************************************************/

#pragma CODE_SECTION(DIT2_Fft,"ramfuncs");

void DIT2_Fft(DIT2_FFT *fft)

{

    DIT2_UINT i,j,k,n;

    DIT2_UINT block_size,block_end;

    DIT2_FLOAT cos1,sin1,cos2,sin2;

    DIT2_FLOAT tw_real,tw_imag;    //temperary t-w

    DIT2_INT16 ang_factor;

 
    /*para check*/

    /*

    if ((fft->imag_in == NULL) || (fft->real_in == NULL))

    {

        while (1);

    }

    */

 
    /*transform direction*/

    switch (fft->transform)

    {

        case DIT2_FORWARD_TRANSFORM:

            ang_factor = DIT2_FORWARD_TRANSFORM;

            break;

 
        case DIT2_INVERSE_TRANSFORM:

            ang_factor = DIT2_INVERSE_TRANSFORM;

            break;

 
        default:

            break;

    }

 
    /*fft*/

    block_end = 1;

/*

* The external loop:

*    find the cos(t-w)/sin(t-w) -- t-w = -2*pi/N

*/

    for (block_size = 2; block_size <= fft->size; block_size <<= 1)

    {

 
        i = DIT2_Idx(block_size);

        cos1 = cos_table;

        sin1 = sin_table;

 
        if (ang_factor == DIT2_INVERSE_TRANSFORM)

        {

            sin1 = -sin1;

        }

         
 
/*

* prepare the first t-w

*/

        for (i = 0; i < fft->size; i += block_size)

        {

            cos2 = 1;//cos(0)

            sin2 = 0;//sin(0)

 
            for (j = i, n = 0; n < block_end; j++, n++)

            {

                k = j + block_end;

                tw_real = fft->real_in[k]*cos2 - fft->imag_in[k]*sin2;

                tw_imag = fft->real_in[k]*sin2 + fft->imag_in[k]*cos2;

 
                /*lower elem*/

                fft->real_in[k] = fft->real_in[j] - tw_real;

                fft->imag_in[k] = fft->imag_in[j] - tw_imag;

 
                /*uper elem*/

                fft->real_in[j] = fft->real_in[j] + tw_real;

                fft->imag_in[j] = fft->imag_in[j] + tw_imag;

 
                /*adjust cos(t-w)/sin(t-w)*/

                tw_real = cos2*cos1 - sin2*sin1;//cos(n+1) = cos(n)cos(1) - sin(n)sin(1)

                tw_imag = cos2*sin1 + sin2*cos1;//sin(n+1) = sin(n)cos(1) + sin(1)cos(n)

 
                cos2 = tw_real;//cos2 = cos(n+1)

                sin2 = tw_imag;//sin2 = sin(n+1)

            }

        }

        block_end  = block_size;

    }

 
    /*inverse transform*/

    if (fft->transform == DIT2_INVERSE_TRANSFORM)

    {

        for (i = 0; i < fft->size; i++)

        {

            fft->real_in /= fft->size;

            fft->imag_in /= fft->size;

        }

    }

 
    return;

}

 
/********************************************************************************

* name  : DIT2_Mag

*Input          : DIT2_FFT pointer

*Deion    : calculate the magnitude and phase info of the output sequence,

*                 used when uncommenting the MACROA--

*                #define DIT2_BIT_INVERSE_IN_PLACE

*

*    typedef  struct DIT2_FFT_STRUCT

*    {

*        DIT2_FLOAT *real_in;   --output magnitude info 

*        DIT2_FLOAT *imag_in;   --output phase info

*        DIT2_FLOAT *real_out;      --unused

*        DIT2_FLOAT *imag_out;      --unused

*        DIT2_FLOAT *mag;           --unused

*        DIT2_FLOAT *phase;         --unused

*        DIT2_UINT size;      --the FFT size, must be 2^N

*        DIT2_UINT stage;         --unused

*        DIT2_UINT transform;     --unused

*    }DIT2_FFT;

********************************************************************************/

#pragma CODE_SECTION(DIT2_Mag,"ramfuncs");

void DIT2_Mag(DIT2_FFT *fft)

{

    DIT2_UINT i;

    DIT2_FLOAT  mag,rad;

 
    /*para check*/

    /*

    if ((fft->real_in == NULL) ||(fft->imag_in == NULL))

    {

        while (1);

    }*/

 
    for (i = 0; i < fft->size; i++)

    {

        mag = sqrt(fft->real_in * fft->real_in + fft->imag_in * fft->imag_in);

        rad = atan2(fft->imag_in,fft->real_in);

        fft->real_in = mag*2.0/fft->size;

        fft->imag_in = rad;

    }

 
    fft->real_in[0] /= 2.0;

 
    return;

}

 
/********************************************************************************

* name  : DIT2_Idx

*Input          : DIT2_UINT in -- the blocksize value

*Output         : the index in sin_table/cos_table

*Deion    : calculate the index in sin_table/cos_table to lookup the table

*                 according the current blocksize. Only used when uncomment the

*                 MACRO--#define DIT2_SIN_COS_TABALE

********************************************************************************/

#pragma CODE_SECTION(DIT2_Idx,"ramfuncs");

static DIT2_UINT DIT2_Idx(DIT2_UINT in)

{

    DIT2_UINT i;

 
    for (i=0; ; i++)

    {

        if (in & (1 << i))

        {

            return(i);

        }

    }

}


Dit2FFT.h文件
/********************************************************************************

*File name        : Alg_Dit2FFT.h     
*Deion    : FFT calculation. Decimation in time FFT, radix-2. 

*

*Current version: v2.3 -- get a simple version, remove unnecessary code

*History version: 

*                 v2.2 -- modify the sin/cos table data, and the static

*                 v2.1 -- modify the  name

*                 v2.0 -- separate the bit-reverse block

*                  v1.9 -- rewrite the bit-reverse block

*                 v1.8 -- rewrite the twiddle factor generation, and test fft,ifft

*                  v1.7 -- fix the twiddle factor bug, and test the four 

*                          combinations

*                  v1.6 -- adjust the DIT2_FLOAT to DIT2_FLOAT32

*                 v1.5 -- change DIT2_UINT16 to DIT2_UINT, and extend its scope

*                 v1.4 -- change the name of s

*                 v1.3 -- add the sin/cos table

*                   v1.2 -- add ifft

*                  v1.1 -- add bit-reverse in-place

*                  v1.0 -- achieve the basic  of FFT

*Author            : Wiracle

*Date            :2008-10-23

********************************************************************************/

 
#ifndef _ALG_DIT2FFT_H_

#define _ALG_DIT2FFT_H_

 
//#include "App_TypeDefDataStruct.h"

typedef  unsigned int       DIT2_UINT16;

typedef  int                DIT2_INT16;

typedef  unsigned long      DIT2_UINT32;

typedef  long               DIT2_INT32;

typedef  long long          DIT2_INT64;

typedef  unsigned long long DIT2_UINT64;

typedef  float              DIT2_FLOAT32;

typedef  long double        DIT2_FLOAT64;

/*config the type used */

typedef  DIT2_FLOAT32       DIT2_FLOAT;

typedef  DIT2_UINT32        DIT2_UINT;

typedef  DIT2_INT32         DIT2_INT;

 
//FFT运算数据结构

typedef struct st_DIT2_FFT

{

    DIT2_FLOAT32 *real_in;    //input and output buffer

    DIT2_FLOAT32 *imag_in;    //input and output buffer

    DIT2_UINT16 size;    //FFT size ,must be 2^N

    DIT2_UINT16 stage;    //蝶形次数

    DIT2_UINT16 transform; //0: FFT运算  1:IFFT运算

}DIT2_FFT;

 
 
 
/*choose the FFT direction*/

#define DIT2_FORWARD_TRANSFORM 0U

#define DIT2_INVERSE_TRANSFORM 1U

//#include "..\Sources_Application\App_TypeDefDataStruct.h"

 
extern DIT2_FLOAT sin_table[];

extern DIT2_FLOAT cos_table[];

 
/********************************************************************************

* name  : DIT2_Rvs

*Input          : DIT2_FFT pointer

*Deion    : Bitreverse the input data, including real and imag part

********************************************************************************/

extern void DIT2_Rvs(DIT2_FFT *fft);

 
/********************************************************************************

* name  : DIT2_Fft

*Input          : DIT2_FFT *fft

*Deion    : Calculate the FFT or IFFT

********************************************************************************/

extern void DIT2_Fft(DIT2_FFT *fft);

 
/********************************************************************************

* name  : DIT2_Mag

*Input          : DIT2_FFT pointer

*Deion    : calculate the magnitude and phase info of the output sequence,

********************************************************************************/

extern void DIT2_Mag(DIT2_FFT *fft);

 
#endif // end of #ifndef _ALG_DIT2FFT_H_


app_main.c  文件
// TI File $Revision: /main/14 $

// Checkin $Date: April 21, 2008   15:41:07 $

//###########################################################################

#include "DSP28x_Project.h"     // Device Headerfile and Examples Include File

#include "SoftTimer.h"

#include "Alg_Dit2FFT.h"

#include "stdlib.h"

 
#define PI 3.1415926

// Prototype statements for s found within this file.

interrupt void cpu_timer1_isr(void);

 
SOFT_TIMER_LINK timer_1;

SOFT_TIMER_LINK timer_2;

 
#pragma DATA_SECTION(sig_real,"ZONE7DATA");

float sig_real[8192];

#pragma DATA_SECTION(sig_imag,"ZONE7DATA");

float sig_imag[8192];

 
DIT2_FFT dit2_fft;

 
 
void init_fft(void);

void init_ifft(void);

 
void main(void)

{

    init_fft();

    DIT2_Rvs(&dit2_fft);

    DIT2_Fft(&dit2_fft);

    //DIT2_Mag(&dit2_fft);

    init_ifft();     
    DIT2_Rvs(&dit2_fft);

    DIT2_Fft(&dit2_fft);

    for(;;)

    {

 
    }

}

 
 
void init_fft(void)

{

    int i;

    float omg_o;

 
 
    dit2_fft.real_in = sig_real;

    dit2_fft.imag_in = sig_imag;

    dit2_fft.transform = 1;

    dit2_fft.size = 8192;

    dit2_fft.stage = 13;

 
    omg_o = 2*PI/dit2_fft.size;

 
    for(i=0;i<dit2_fft.size;i++)

    {

        sig_real = sin(2*omg_o*i) + 2*sin(50*omg_o*i);// + 4*sin(100*omg_o*i)+ 8*sin(1000*omg_o*i) ;

        sig_imag = 0;

    }

}

void init_ifft(void)

{

    dit2_fft.real_in = sig_real;

    dit2_fft.imag_in = sig_imag;

    dit2_fft.transform = 0;

    dit2_fft.size = 8192;

    dit2_fft.stage = 13;

}

 
 
 
 
 
 
//===========================================================================

// No more.

//===========================================================================


路过

鸡蛋

鲜花

握手

雷人

评论 (0 个评论)