-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathssub.cpp
266 lines (198 loc) · 6.54 KB
/
ssub.cpp
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
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
#include "duet.h"
int merged_streams = 0; // Just for debug / printing (seems something's wrong: more merges than streams)
//#define OLD_MASK_BUILD
//#define OLD_PEAK_ASSIGN
// After giving one buffer in one call (new_buffer), at the end of the next (blocking) call, it can be released.
// Cannot be reassigned to a different output because it stores the previous buffer and it would have conflicts, thus all the data must be passed
size_t write_data(Buffers<real> &o, Buffers<real> *new_buffers, const size_t FFT_N, const size_t FFT_slide)
{
static int overlapping_buffers = 1; // how many buffers are in current use (state variable)
static Buffers<real> *a=new_buffers, *b=NULL;
static size_t i = 0, p = 0;
unsigned int buffers = o.buffers();
if (FFT_slide < FFT_N) // Up to 50% overlap
{
if (overlapping_buffers == 2)
{
b = new_buffers;
while (i < FFT_N)
{
//o[p] = a[i] + b[i-FFT_slide];
for (uint buf = 0; buf < buffers; ++buf)
(*o(buf))[p] = (*(*a)(buf))[i] + (*(*b)(buf))[i-FFT_slide];
++p;
++i;
}
// RELEASE(a)
i = FFT_N-FFT_slide;
a = b;
overlapping_buffers = 1;
}
// overlapping_buffers == 1
while (i < FFT_slide)
{
// o[p] = a[i]
for (uint buf = 0; buf < buffers; ++buf)
(*o(buf))[p] = (*(*a)(buf))[i];
++p;
++i;
}
overlapping_buffers = 2;
// Now wait for new call with new_buffer
}
else // No overlap
{
i = 0;
a = new_buffers;
while (i < FFT_slide) // == FFT_N
{
// o[p] = a[i]
for (uint buf = 0; buf < buffers; ++buf)
(*o(buf))[p] = (*(*a)(buf))[i];
++p;
++i;
}
}
return p;
}
struct Cfg
{
unsigned int FFT_N, FFT_slide;
real eta, beta, alpha;
};
/**
Arguments: prgm [FFT_N] [x1_wav] [x2_wav]
*/
void StationaryNoisePSD(Matrix<real> &X_history, Buffer<real> &NPSD, idx tb_start, idx tb_end)
{
static const idx K = NPSD.size(), FFT_N = K*2;
static Buffer<real> P(NPSD);
real *X = NULL;
NPSD.clear();
for (idx tb = tb_start; tb < tb_end; ++tb)
{
X = X_history(tb);
evenHC2power(FFT_N, X, P());
NPSD += P;
}
NPSD /= tb_end - tb_start;
}
void SSub(real *X, Buffer<real> &NPSD, const Cfg &cfg)
{
static const idx K = NPSD.size(), FFT_N = K*2;
for (idx k = 0; k < K; ++k)
{
std::complex<real> C(X[k], X[FFT_N-k]);
real Xk_power = std::norm(C);
real Yk_power = Xk_power - cfg.beta * NPSD[k];
if (Yk_power < 0)
{
X[k] = cfg.eta * X[k] ;
X[FFT_N-k] = cfg.eta * X[FFT_N-k];
}
else
{
std::complex<real> Ck ( std::polar<real>(std::sqrt(Yk_power), std::arg(C)) );
X[k] = Ck.real();
X[FFT_N-k] = Ck.imag();
}
}
}
int main(int argc, char **argv)
{
/* Name convention throughout this file:
i - input
o - output
m - magnitude
and capital letters for the frequency domain
*/
Options _o("presets.cfg", Quit, 1); // Choose the preset file.
Options o (_o("ssub_preset").c_str(), Quit, 1); // True configuration file (preset)
unsigned int FFT_N = o.i("FFT_N");
unsigned int FFT_slide = FFT_N * (o.i("FFT_slide_percentage")/100.0);
Cfg cfg;
cfg.FFT_N = FFT_N;
cfg.FFT_slide = FFT_slide;
cfg.beta = o.f("beta");
cfg.eta = o.f("eta");
cfg.alpha = o.f("alpha");
// Choose mic input files
Guarantee(argc == 3, "Usage: ssub <in> <out>");
std::string I_filepath = argv[1];
SndfileHandle I(I_filepath);
Guarantee(wav::ok(I) , "Input file doesn't exist.");
Guarantee(wav::mono(I), "Input files must be mono.");
const uint sample_rate_Hz = I.samplerate();
const idx samples = I.frames();
const real Ts = 1.0/(real)sample_rate_Hz;
Buffer<real> wav(samples);
I.read(wav(), samples);
printf(YELLOW "\nProcessing input file with %lu frames @ %u Hz.\n\n" NOCOLOR, samples, sample_rate_Hz);
Guarantee0(FFT_N % 2, "System implemented for FFTs with even size.");
Guarantee(FFT_slide <= FFT_N, "FFT_slide(%u) > FFT_N(%u)", FFT_slide, FFT_N);
const idx time_blocks = div_up<size_t>(samples-FFT_N,FFT_slide) + 1;
// Silence blocks
static const int Sb = o.i("initial_silence_blocks");
//// Storage allocation ///////
// Initialize the buffers all with the same characteristics and aligned for FFTW use.
Buffer<real> x(FFT_N, 0, fftw_malloc, fftw_free), X(x), M(FFT_N/2), NPSD(M);
// We're going to save at least one of the microphone transforms for all time blocks for the static heuristic reconstruction
Matrix<real> X_history(time_blocks, FFT_N);
Buffer<real> wav_out(time_blocks*FFT_slide+(FFT_N-FFT_slide), 0, fftw_malloc, fftw_free);
const real FFT_df = sample_rate_Hz / (real) FFT_N;
Buffer<real> f_axis(FFT_N/2);
f_axis.fill_range(0,0.5*(real)sample_rate_Hz);
fftw_plan xXplan, Xxplan;
cout << "Estimating FFT plan..." << endl;
cout << "The fast way!\n";
int FFT_flags = FFTW_ESTIMATE;
xXplan = fftw_plan_r2r_1d(FFT_N, x(), X(), FFTW_R2HC, FFT_flags);
Xxplan = fftw_plan_r2r_1d(FFT_N, X(), x(), FFTW_HC2R, FFT_flags);
cout << "DONE" << endl;
Buffer<real> W(FFT_N);
select_window(o("window"), W);
for (idx time_block = 0; time_block < time_blocks; ++time_block)
{
idx block_offset = time_block*FFT_slide;
for (idx i = 0; i < FFT_N; ++i)
{
idx t = i+block_offset;
x[i] = ( t < samples ? wav[t] * W[i] : 0 );
}
fftw_execute(xXplan);
// Keep the record of X1 for all time for later audio reconstruction
for (idx f = 0; f < FFT_N; ++f)
X_history(time_block,f) = X[f];
} // End of blockwise processing
// Estimate noise and subtract weakly several times.
for (int iterations = 0; iterations < o.i("iterations"); ++iterations)
{
if (o.i("use_final_silence"))
StationaryNoisePSD(X_history, NPSD, (long int)time_blocks-(long int)Sb, (long int)time_blocks);
else
StationaryNoisePSD(X_history, NPSD, 0, Sb);
for (idx tb = 0; tb < time_blocks; ++tb)
SSub(X_history(tb), NPSD, cfg);
}
for (idx tb = 0; tb < time_blocks; ++tb)
{
fftw_execute_r2r(Xxplan, X_history(tb), x());
wav_out.add_at(x, tb*FFT_slide);
}
wav_out /= FFT_N;
Gnuplot p;
/*
// Show amplitude envelope
wav_out.clear();
for (unsigned int tb=0; tb < time_blocks; ++tb)
wav_out.add_at(W,tb*FFT_slide);
*/
p.plot(wav_out(),wav_out.size(),"wav_out");
wait();
std::string wav_filepath(argv[2]);
printf("%s...", wav_filepath.c_str());
print_status( wav::write_mono(wav_filepath, wav_out(), samples, sample_rate_Hz) );
fftw_destroy_plan(xXplan);
fftw_destroy_plan(Xxplan);
return 0;
}