FastLED 3.9.15
Loading...
Searching...
No Matches
kiss_fftr.c
Go to the documentation of this file.
1/*
2 * Copyright (c) 2003-2004, Mark Borgerding. All rights reserved.
3 * This file is part of KISS FFT - https://github.com/mborgerding/kissfft
4 *
5 * SPDX-License-Identifier: BSD-3-Clause
6 * See COPYING file for more information.
7 */
8
9#include "kiss_fftr.h"
10#include "_kiss_fft_guts.h"
11
16#ifdef USE_SIMD
17 void * pad;
18#endif
19};
20
21kiss_fftr_cfg kiss_fftr_alloc(int nfft,int inverse_fft,void * mem,size_t * lenmem)
22{
23 int i;
24 kiss_fftr_cfg st = NULL;
25 size_t subsize = 0, memneeded;
26
27 if (nfft & 1) {
28 fprintf(stderr,"Real FFT optimization must be even.\n");
29 return NULL;
30 }
31 nfft >>= 1;
32
33 kiss_fft_alloc (nfft, inverse_fft, NULL, &subsize);
34 memneeded = sizeof(struct kiss_fftr_state) + subsize + sizeof(kiss_fft_cpx) * ( nfft * 3 / 2);
35
36 if (lenmem == NULL) {
37 st = (kiss_fftr_cfg) KISS_FFT_MALLOC (memneeded);
38 } else {
39 if (*lenmem >= memneeded)
40 st = (kiss_fftr_cfg) mem;
41 *lenmem = memneeded;
42 }
43 if (!st)
44 return NULL;
45
46 st->substate = (kiss_fft_cfg) (st + 1); /*just beyond kiss_fftr_state struct */
47
48 #pragma GCC diagnostic push
49 #pragma GCC diagnostic ignored "-Wcast-align"
50 // This cast alginment might be a problem, kiss_fft_cpx has alignment-2.
51 st->tmpbuf = (kiss_fft_cpx *) (((char *) st->substate) + subsize);
52 #pragma GCC diagnostic pop
53
54 st->super_twiddles = st->tmpbuf + nfft;
55 kiss_fft_alloc(nfft, inverse_fft, st->substate, &subsize);
56
57 for (i = 0; i < nfft/2; ++i) {
58 double phase =
59 -3.14159265358979323846264338327 * ((double) (i+1) / nfft + .5);
60 if (inverse_fft)
61 phase *= -1;
62 kf_cexp (st->super_twiddles+i,phase);
63 }
64 return st;
65}
66
67void kiss_fftr(kiss_fftr_cfg st,const kiss_fft_scalar *timedata,kiss_fft_cpx *freqdata)
68{
69 /* input buffer timedata is stored row-wise */
70 int k,ncfft;
71 kiss_fft_cpx fpnk,fpk,f1k,f2k,tw,tdc;
72
73 if ( st->substate->inverse) {
74 fprintf(stderr,"kiss fft usage error: improper alloc\n");
75 exit(1);
76 }
77
78 ncfft = st->substate->nfft;
79
80 /*perform the parallel fft of two real signals packed in real,imag*/
81 kiss_fft( st->substate , (const kiss_fft_cpx*)timedata, st->tmpbuf );
82 /* The real part of the DC element of the frequency spectrum in st->tmpbuf
83 * contains the sum of the even-numbered elements of the input time sequence
84 * The imag part is the sum of the odd-numbered elements
85 *
86 * The sum of tdc.r and tdc.i is the sum of the input time sequence.
87 * yielding DC of input time sequence
88 * The difference of tdc.r - tdc.i is the sum of the input (dot product) [1,-1,1,-1...
89 * yielding Nyquist bin of input time sequence
90 */
91
92 tdc.r = st->tmpbuf[0].r;
93 tdc.i = st->tmpbuf[0].i;
94 C_FIXDIV(tdc,2);
95 CHECK_OVERFLOW_OP(tdc.r ,+, tdc.i);
96 CHECK_OVERFLOW_OP(tdc.r ,-, tdc.i);
97 freqdata[0].r = tdc.r + tdc.i;
98 freqdata[ncfft].r = tdc.r - tdc.i;
99#ifdef USE_SIMD
100 freqdata[ncfft].i = freqdata[0].i = _mm_set1_ps(0);
101#else
102 freqdata[ncfft].i = freqdata[0].i = 0;
103#endif
104
105 for ( k=1;k <= ncfft/2 ; ++k ) {
106 fpk = st->tmpbuf[k];
107 fpnk.r = st->tmpbuf[ncfft-k].r;
108 fpnk.i = - st->tmpbuf[ncfft-k].i;
109 C_FIXDIV(fpk,2);
110 C_FIXDIV(fpnk,2);
111
112 C_ADD( f1k, fpk , fpnk );
113 C_SUB( f2k, fpk , fpnk );
114 C_MUL( tw , f2k , st->super_twiddles[k-1]);
115
116 freqdata[k].r = HALF_OF(f1k.r + tw.r);
117 freqdata[k].i = HALF_OF(f1k.i + tw.i);
118 freqdata[ncfft-k].r = HALF_OF(f1k.r - tw.r);
119 freqdata[ncfft-k].i = HALF_OF(tw.i - f1k.i);
120 }
121}
122
123void kiss_fftri(kiss_fftr_cfg st,const kiss_fft_cpx *freqdata,kiss_fft_scalar *timedata)
124{
125 /* input buffer timedata is stored row-wise */
126 int k, ncfft;
127
128 if (st->substate->inverse == 0) {
129 fprintf (stderr, "kiss fft usage error: improper alloc\n");
130 exit (1);
131 }
132
133 ncfft = st->substate->nfft;
134
135 st->tmpbuf[0].r = freqdata[0].r + freqdata[ncfft].r;
136 st->tmpbuf[0].i = freqdata[0].r - freqdata[ncfft].r;
137 C_FIXDIV(st->tmpbuf[0],2);
138
139 for (k = 1; k <= ncfft / 2; ++k) {
140 kiss_fft_cpx fk, fnkc, fek, fok, tmp;
141 fk = freqdata[k];
142 fnkc.r = freqdata[ncfft - k].r;
143 fnkc.i = -freqdata[ncfft - k].i;
144 C_FIXDIV( fk , 2 );
145 C_FIXDIV( fnkc , 2 );
146
147 C_ADD (fek, fk, fnkc);
148 C_SUB (tmp, fk, fnkc);
149 C_MUL (fok, tmp, st->super_twiddles[k-1]);
150 C_ADD (st->tmpbuf[k], fek, fok);
151 C_SUB (st->tmpbuf[ncfft - k], fek, fok);
152#ifdef USE_SIMD
153 st->tmpbuf[ncfft - k].i *= _mm_set1_ps(-1.0);
154#else
155 st->tmpbuf[ncfft - k].i *= -1;
156#endif
157 }
158 kiss_fft (st->substate, st->tmpbuf, (kiss_fft_cpx *) timedata);
159}
#define C_FIXDIV(c, div)
#define HALF_OF(x)
#define C_ADD(res, a, b)
#define CHECK_OVERFLOW_OP(a, op, b)
#define C_SUB(res, a, b)
#define C_MUL(m, a, b)
#define kf_cexp(x, phase)
void kiss_fft(kiss_fft_cfg cfg, const kiss_fft_cpx *fin, kiss_fft_cpx *fout)
Definition kiss_fft.c:379
kiss_fft_cfg kiss_fft_alloc(int nfft, int inverse_fft, void *mem, size_t *lenmem)
Definition kiss_fft.c:333
#define kiss_fft_scalar
Definition kiss_fft.h:63
#define KISS_FFT_MALLOC
Definition kiss_fft.h:53
struct kiss_fft_state * kiss_fft_cfg
Definition kiss_fft.h:77
kiss_fft_scalar r
Definition kiss_fft.h:73
kiss_fft_scalar i
Definition kiss_fft.h:74
void kiss_fftr(kiss_fftr_cfg st, const kiss_fft_scalar *timedata, kiss_fft_cpx *freqdata)
Definition kiss_fftr.c:67
void kiss_fftri(kiss_fftr_cfg st, const kiss_fft_cpx *freqdata, kiss_fft_scalar *timedata)
Definition kiss_fftr.c:123
kiss_fftr_cfg kiss_fftr_alloc(int nfft, int inverse_fft, void *mem, size_t *lenmem)
Definition kiss_fftr.c:21
kiss_fft_cfg substate
Definition kiss_fftr.c:13
kiss_fft_cpx * tmpbuf
Definition kiss_fftr.c:14
kiss_fft_cpx * super_twiddles
Definition kiss_fftr.c:15
struct kiss_fftr_state * kiss_fftr_cfg
Definition kiss_fftr.h:26