forked from BoysTownOrg/chapro
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathgha_demo.c
145 lines (128 loc) · 3.64 KB
/
gha_demo.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
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
// gha_demo.c - demonstrate GHA processing (FIR+AGC)
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <math.h>
#include <sigpro.h>
#include "chapro.h"
/***********************************************************/
static float *
audio_read(char *fn, float *fs, int *nwav)
{
VAR *vl;
vl = sp_wav_read(fn, 0, 0, fs);
if (vl == NULL) {
fprintf(stderr, "can't open %s\n", fn);
return (NULL);
}
*nwav = vl[0].rows * vl[0].cols;
return (vl[0].data);
}
static void
set_spl(float *x, int n, double rms_lev, double spl_ref)
{
float scl;
double xx, rms, smsq, lev;
int i;
smsq = 0;
for (i = 0; i < n; i++) {
xx = x[i];
smsq += xx * xx;
}
rms = sqrt(smsq / n);
lev = 20 * log10(rms / spl_ref);
scl = (float) pow(10,(rms_lev - lev) / 20);
for (i = 0; i < n; i++) {
x[i] *= scl;
}
}
static void
save_mat(char *fn, float fs, float *x, float *y, int n)
{
float *t;
int i;
static VAR *vl;
t = (float *) calloc(n, sizeof(float));
for (i = 0; i < n; i++) {
t[i] = i / fs;
}
vl = sp_var_alloc(3);
sp_var_add(vl, "t", t, n, 1, "f4");
sp_var_add(vl, "x", x, n, 1, "f4");
sp_var_add(vl, "y", y, n, 1, "f4");
sp_mat_save(fn, vl);
sp_var_clear(vl);
free(t);
}
static void
WDRC(CHA_PTR cp, float *x, float *y, int n, int nc)
{
float *ww, *xx, *yy, *zz;
int i, cs, nk;
ww = (float *) calloc(n, sizeof(float));
xx = (float *) calloc(n, sizeof(float));
yy = (float *) calloc(n * nc * 2, sizeof(float));
zz = (float *) calloc(n * nc * 2, sizeof(float));
// process FIR+AGC
cs = CHA_IVAR[_cs]; // chunk size
nk = n / cs; // number of chunks
for (i = 0; i < nk; i++) {
cha_agc_input(cp, x + i * cs, xx, cs);
cha_firfb_analyze(cp, xx, yy, cs);
cha_agc_channel(cp, yy, zz, cs);
cha_firfb_synthesize(cp, zz, ww, cs);
cha_agc_output(cp, ww, y + i * cs, cs);
}
free(ww);
free(xx);
free(yy);
free(zz);
}
static double
amplify(float *x, float *y, int n, double fs, CHA_DSL *dsl)
{
int nc;
static int nw = 256; // window size
static int cs = 32; // chunk size
static int wt = 0; // window type: 0=Hamming, 1=Blackman
static void *cp[NPTR] = {0};
static CHA_WDRC agc = {1, 50, 24000, 119, 0, 105, 10, 105};
nc = dsl->nchannel;
cha_firfb_prepare(cp, dsl->cross_freq, nc, fs, nw, wt, cs);
cha_agc_prepare(cp, dsl, &agc);
sp_tic();
WDRC(cp, x, y, n, nc);
return (sp_toc());
}
/***********************************************************/
int
main(int ac, char *av[])
{
double t1, t2;
float fs, *x, *y;
int n = 0;
static char *ifn = "test/cat.wav";
static char *ofn = "test/gha_demo.mat";
static double spl_ref = 1.1219e-6;
static double rms_lev = 65;
// DSL prescription - (first subject, left ear) from LD_RX.mat
static CHA_DSL dsl = {5, 50, 119, 0, 8,
{317.1666,502.9734,797.6319,1264.9,2005.9,3181.1,5044.7},
{-13.5942,-16.5909,-3.7978,6.6176,11.3050,23.7183,35.8586,37.3885},
{0.7,0.9,1,1.1,1.2,1.4,1.6,1.7},
{32.2,26.5,26.7,26.7,29.8,33.6,34.3,32.7},
{78.7667,88.2,90.7,92.8333,98.2,103.3,101.9,99.8}
};
x = audio_read(ifn, &fs, &n);
y = (float *) calloc(n, sizeof(float));
set_spl(x, n, rms_lev, spl_ref);
t1 = amplify(x, y, n, fs, &dsl);
save_mat(ofn, fs, x, y, n);
t2 = n / fs;
printf("%s -> %s\n", ifn, ofn);
printf("speed_ratio: ");
printf("(wave_time/wall_time) = (%.3f/%.3f) ", t2, t1);
printf("= %.1f\n", t2 / t1);
free(y);
return (0);
}