forked from BoysTownOrg/chapro
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathdciirfb_process.c
105 lines (95 loc) · 2.82 KB
/
dciirfb_process.c
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
// dciirfb_process.c - complex-double filterbank processing functions
#include <stdlib.h>
#include <string.h>
#include <math.h>
#include "chapro.h"
#define fmove(x,y,n) memmove(x,y,(n)*sizeof(float))
#define GTFO 4
/***********************************************************/
static __inline void
filter_tf(float *x, float *y, int cs, double *coef, double *hist, int op)
{
double xx, yyr, yyi, *zr, *zi, *br, *bi, *ar, *ai;
int i, ir, ii, j, no;
no = op - 1;
br = coef;
bi = br + op;
ar = bi + op;
ai = ar + op;
zr = hist;
zi = zr + op;
/* loop over time */
for (i = 0; i < cs; i++) {
xx = x[i];
yyr = (br[0] * xx) + zr[0];
yyi = (bi[0] * xx) + zi[0];
for (j = 1; j < no; j++) {
zr[j - 1] = (br[j] * xx) - (ar[j] * yyr - ai[j] * yyi) + zr[j];
zi[j - 1] = (bi[j] * xx) - (ar[j] * yyi + ai[j] * yyr) + zi[j];
}
zr[no - 1] = (br[no] * xx) - (ar[no] * yyr - ai[no] * yyi);
zi[no - 1] = (bi[no] * xx) - (ar[no] * yyi + ai[no] * yyr);
/* channel out */
ir = i * 2;
ii = ir + 1;
y[ir] = (float) yyr;
y[ii] = (float) yyi;
}
}
/***********************************************************/
FUNC(void)
cha_dciirfb_analyze(CHA_PTR cp, float *x, float *y, int cs)
{
double *br, *zr, *bkr, *zkr;
float *ydr, *ydi, *yk;
int i, ir, ii, d, m, k, n, nc, ns, op, no = GTFO, *dn;
dn = (int *) cp[_dn];
nc = CHA_IVAR[_nc];
ns = CHA_IVAR[_ns];
ydr = (float *) cp[_ydr];
ydi = ydr + ns * nc;
op = no + 1;
br = (double *) cp[_br];
zr = (double *) cp[_zr];
/* loop over filterbank channel */
#pragma omp parallel for
for (k = 0; k < nc; k++) {
yk = y + k * cs * 2;
bkr = br + k * op * 4;
zkr = zr + k * op * 4;
filter_tf(x, yk, cs, bkr, zkr, op);
for (i = 0; i < cs; i++) {
ir = i * 2;
ii = ir + 1;
/* peak shift */
if (dn) {
d = dn[k];
m = k * ns;
n = m + d;
fmove(ydr + m + 1, ydr + m, d);
fmove(ydi + m + 1, ydi + m, d);
ydr[m] = yk[ir];
ydi[m] = yk[ii];
yk[ir] = ydr[n];
yk[ii] = ydi[n];
}
}
}
}
// double CIIR-filterbank synthesis
FUNC(void)
cha_dciirfb_synthesize(CHA_PTR cp, float *x, float *y, int cs)
{
float sm;
int i, k, kr, nc;
nc = CHA_IVAR[_nc];
for (i = 0; i < cs; i++) {
sm = 0;
/* loop over filterbank channel */
for (k = 0; k < nc; k++) {
kr = (i + k * cs) * 2;
sm += x[kr];
}
y[i] = sm;
}
}