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
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
|
#ifndef INCLUDE_DISTILLER_H
#define INCLUDE_DISTILLER_H
/*
* This is the Loris C++ Class Library, implementing analysis,
* manipulation, and synthesis of digitized sounds using the Reassigned
* Bandwidth-Enhanced Additive Sound Model.
*
* Loris is Copyright (c) 1999-2010 by Kelly Fitz and Lippold Haken
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation; either version 2 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY, without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program; if not, write to the Free Software
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
*
*
* Distiller.h
*
* Definition of class Distiller.
*
* Kelly Fitz, 20 Oct 1999
* loris@cerlsoundgroup.org
*
* http://www.cerlsoundgroup.org/Loris/
*
*/
#include "Partial.h"
#include "PartialList.h"
#include "PartialUtils.h"
#include "Notifier.h" // for debugging only
#include <algorithm>
// begin namespace
namespace Loris {
// ---------------------------------------------------------------------------
// class Distiller
//
//! Class Distiller represents an algorithm for "distilling" a group of
//! Partials that logically represent a single component into a single
//! Partial.
//!
//! The sound morphing algorithm in Loris requires that Partials in a
//! given source be labeled uniquely, that is, no two Partials can have
//! the same label. The Distiller enforces this condition. All Partials
//! identified with a particular frequency channel (see Channelizer), and,
//! therefore, having a common label, are distilled into a single Partial,
//! leaving at most a single Partial per frequency channel and label.
//! Channels that contain no Partials are not represented in the distilled
//! data. Partials that are not labeled, that is, Partials having label 0,
//! are are left unmodified at the end of the Partial sequence.
//!
//! Distillation modifies the Partial container (a PartialList). All
//! Partials in the distilled range having a common label are replaced by
//! a single Partial in the distillation process. Only labeled
//! Partials are affected by distillation.
//
class Distiller
{
// -- instance variables --
double _fadeTime, _gapTime; // distillation parameters
// -- public interface --
public:
// -- global defaults and constants --
enum
{
//! Default time in milliseconds over which Partials joined by
//! distillation fade to and from zero amplitude. Divide by
//! 1000 to use as a member function parameter. This parameter
//! should be the same in Distiller, Sieve, and Collator.
DefaultFadeTimeMs = 5,
//! Default minimum duration in milliseconds of the silent
//! (zero-amplitude) gap between two Partials joined by
//! distillation. Divide by 1000 to use as a member function
//! parameter. This parameter should be the same in Distiller,
//! Sieve, and Collator.
DefaultSilentTimeMs = 1
};
// -- construction --
//! Construct a new Distiller using the specified fade time
//! for gaps between Partials. When two non-overlapping Partials
//! are distilled into a single Partial, the distilled Partial
//! fades out at the end of the earlier Partial and back in again
//! at the onset of the later one. The fade time is the time over
//! which these fades occur. By default, use a 1 ms fade time.
//! The gap time is the additional time over which a Partial faded
//! out must remain at zero amplitude before it can fade back in.
//! By default, use a gap time of one tenth of a millisecond, to
//! prevent a pair of arbitrarily close null Breakpoints being
//! inserted.
//!
//! \param partialFadeTime is the time (in seconds) over
//! which Partials joined by distillation fade to
//! and from zero amplitude. (Default is
//! Distiller::DefaultFadeTime).
//! \param partialSilentTime is the minimum duration (in seconds)
//! of the silent (zero-amplitude) gap between two
//! Partials joined by distillation. (Default is
//! Distiller::DefaultSilentTime).
explicit
Distiller( double partialFadeTime = Distiller::DefaultFadeTimeMs/1000.0,
double partialSilentTime = Distiller::DefaultSilentTimeMs/1000.0 );
// Use compiler-generated copy, assign, and destroy.
// -- distillation --
//! Distill labeled Partials in a collection leaving only a single
//! Partial per non-zero label.
//!
//! Unlabeled (zero-labeled) Partials are left unmodified at
//! the end of the distilled Partials.
//!
//! Return an iterator refering to the position of the first unlabeled Partial,
//! or the end of the distilled collection if there are no unlabeled Partials.
//! Since distillation is in-place, the Partials collection may be smaller
//! (fewer Partials) after distillation, and any iterators on the collection
//! may be invalidated.
//!
//! \post All labeled Partials in the collection are uniquely-labeled,
//! and all unlabeled Partials have been moved to the end of the
//! sequence.
//! \param partials is the collection of Partials to distill in-place
//! \return the position of the end of the range of distilled Partials,
//! which is either the end of the collection, or the position
//! or the first unlabeled Partial.
//!
//! If compiled with NO_TEMPLATE_MEMBERS defined, then partials
//! must be a PartialList, otherwise it can be any container type
//! storing Partials that supports at least bidirectional iterators.
//!
//! \sa Distiller::distill( Container & partials )
#if ! defined(NO_TEMPLATE_MEMBERS)
template< typename Container >
typename Container::iterator distill( Container & partials );
#else
inline
PartialList::iterator distill( PartialList & partials );
#endif
//! Function call operator: same as distill( PartialList & partials ).
#if ! defined(NO_TEMPLATE_MEMBERS)
template< typename Container >
typename Container::iterator operator() ( Container & partials );
#else
PartialList::iterator operator() ( PartialList & partials );
#endif
//! Static member that constructs an instance and applies
//! it to a sequence of Partials.
//!
//! \post All labeled Partials in the collection are uniquely-labeled,
//! and all unlabeled Partials have been moved to the end of the
//! sequence.
//! \param partials is the collection of Partials to distill in-place
//! \param partialFadeTime is the time (in seconds) over
//! which Partials joined by distillation fade to
//! and from zero amplitude.
//! \param partialSilentTime is the minimum duration (in seconds)
//! of the silent (zero-amplitude) gap between two
//! Partials joined by distillation. (Default is
//! Distiller::DefaultSilentTime).
//! \return the position of the end of the range of distilled Partials,
//! which is either the end of the collection, or the position
//! or the first unlabeled Partial.
//!
//! If compiled with NO_TEMPLATE_MEMBERS defined, then partials
//! must be a PartialList, otherwise it can be any container type
//! storing Partials that supports at least bidirectional iterators.
#if ! defined(NO_TEMPLATE_MEMBERS)
template< typename Container >
static typename Container::iterator
distill( Container & partials, double partialFadeTime,
double partialSilentTime = DefaultSilentTimeMs/1000.0 );
#else
static inline PartialList::iterator
distill( PartialList & partials, double partialFadeTime,
double partialSilentTime = DefaultSilentTimeMs/1000.0 );
#endif
private:
// -- helpers --
//! Distill labeled Partials in a PartialList leaving only a single
//! Partial per non-zero label.
//!
//! Unlabeled (zero-labeled) Partials are left unmodified at
//! the end of the distilled Partials.
//!
//! Return an iterator refering to the position of the first unlabeled Partial,
//! or the end of the distilled collection if there are no unlabeled Partials.
//! Since distillation is in-place, the Partials collection may be smaller
//! (fewer Partials) after distillation, and any iterators on the collection
//! may be invalidated.
//!
//! \post All labeled Partials in the collection are uniquely-labeled,
//! and all unlabeled Partials have been moved to the end of the
//! sequence.
//! \param partials is the collection of Partials to distill in-place
//! \return the position of the end of the range of distilled Partials,
//! which is either the end of the collection, or the position
//! or the first unlabeled Partial.
PartialList::iterator distill_list( PartialList & partials );
//! Distill a list of Partials having a common label
//! into a single Partial with that label, and append it
//! to the distilled collection. If an empty list of Partials
//! is passed, then an empty Partial having the specified
//! label is appended.
void distillOne( PartialList & partials, Partial::label_type label,
PartialList & distilled );
}; // end of class Distiller
// ---------------------------------------------------------------------------
// distill
// ---------------------------------------------------------------------------
//! Distill labeled Partials in a collection leaving only a single
//! Partial per non-zero label.
//!
//! Unlabeled (zero-labeled) Partials are left unmodified at
//! the end of the distilled Partials.
//!
//! Return an iterator refering to the position of the first unlabeled Partial,
//! or the end of the distilled collection if there are no unlabeled Partials.
//! Since distillation is in-place, the Partials collection may be smaller
//! (fewer Partials) after distillation, and any iterators on the collection
//! may be invalidated.
//!
//! \post All labeled Partials in the collection are uniquely-labeled,
//! and all unlabeled Partials have been moved to the end of the
//! sequence.
//! \param partials is the collection of Partials to distill in-place
//! \return the position of the end of the range of distilled Partials,
//! which is either the end of the collection, or the position
//! or the first unlabeled Partial.
//!
//! If compiled with NO_TEMPLATE_MEMBERS defined, then partials
//! must be a PartialList, otherwise it can be any container type
//! storing Partials that supports at least bidirectional iterators.
//!
//
#if ! defined(NO_TEMPLATE_MEMBERS)
template< typename Container >
typename Container::iterator Distiller::distill( Container & partials )
{
// This can be done so much more easily and
// efficiently on a list than on other containers
// that it is worth copying the Partials to a
// list for distillation, and then transfering
// them back.
//
// See below for a specialization for the case
// of the Container being a list, so no copy
// is needed.
PartialList pl( partials.begin(), partials.end() );
PartialList::iterator it = distill_list( pl );
// pl has distilled Partials at beginning, and
// unlabeled Partials at end:
typename Container::iterator beginUnlabeled =
std::copy( pl.begin(), it, partials.begin() );
typename Container::iterator endUnlabeled =
std::copy( it, pl.end(), beginUnlabeled );
partials.erase( endUnlabeled, partials.end() );
return beginUnlabeled;
}
// specialization for PartialList container
template< >
inline
PartialList::iterator Distiller::distill( PartialList & partials )
{
debugger << "using PartialList version of distill to avoid copying" << endl;
return distill_list( partials );
}
#else
inline
PartialList::iterator Distiller::distill( PartialList & partials )
{
return distill_list( partials );
}
#endif
// ---------------------------------------------------------------------------
// Function call operator
// ---------------------------------------------------------------------------
//! Function call operator: same as distill( PartialList & partials ).
//!
//! \sa Distiller::distill( Container & partials )
//
#if ! defined(NO_TEMPLATE_MEMBERS)
template< typename Container >
typename Container::iterator Distiller::operator()( Container & partials )
#else
inline
PartialList::iterator Distiller::operator()( PartialList & partials )
#endif
{
return distill( partials );
}
// ---------------------------------------------------------------------------
// distill
// ---------------------------------------------------------------------------
//! Static member that constructs an instance and applies
//! it to a sequence of Partials.
//!
//! \post All labeled Partials in the collection are uniquely-labeled,
//! and all unlabeled Partials have been moved to the end of the
//! sequence.
//! \param partials is the collection of Partials to distill in-place
//! \param partialFadeTime is the time (in seconds) over
//! which Partials joined by distillation fade to
//! and from zero amplitude.
//! \param partialSilentTime is the minimum duration (in seconds)
//! of the silent (zero-amplitude) gap between two
//! Partials joined by distillation. (Default is
//! Distiller::DefaultSilentTime).
//! \return the position of the end of the range of distilled Partials,
//! which is either the end of the collection, or the position
//! or the first unlabeled Partial.
//!
//! If compiled with NO_TEMPLATE_MEMBERS defined, then partials
//! must be a PartialList, otherwise it can be any container type
//! storing Partials that supports at least bidirectional iterators.
//
#if ! defined(NO_TEMPLATE_MEMBERS)
template< typename Container >
typename Container::iterator
Distiller::distill( Container & partials, double partialFadeTime,
double partialSilentTime )
#else
inline
PartialList::iterator
Distiller::distill( PartialList & partials, double partialFadeTime,
double partialSilentTime )
#endif
{
Distiller instance( partialFadeTime, partialSilentTime );
return instance.distill( partials );
}
} // end of namespace Loris
#endif /* ndef INCLUDE_DISTILLER_H */
|