-
Notifications
You must be signed in to change notification settings - Fork 185
/
Recorder.h
167 lines (135 loc) · 4.39 KB
/
Recorder.h
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
#ifndef STK_RECORDER_H
#define STK_RECORDER_H
#include "Instrmnt.h"
#include "Iir.h"
#include "DelayL.h"
#include "Noise.h"
#include "SineWave.h"
#include "ADSR.h"
namespace stk {
/***************************************************/
/*! \class Recorder
\brief A recorder / flute physical model.
This class implements a physical model of a recorder /
flute instrument, based on the paper "Sound production
in recorderlike instruments. II. A simulation model."
by M.P. Verge, A. Hirschberg and R. Causse, Journal of
the Acoustical Society of America, 1997.
Control Change Numbers:
- Softness = 2
- Noise Gain = 4
- Noise Cutoff = 16
- Vibrato Frequency = 11
- Vibrato Gain = 1
- Breath Pressure = 128
by Mathias Bredholt, McGill University.
Formatted for STK by Gary Scavone, 2023.
*/
/***************************************************/
class Recorder : public Instrmnt
{
public:
//! Class constructor.
Recorder( void );
//! Class destructor.
~Recorder( void );
//! Reset and clear all internal state.
void clear( void );
//! Set instrument parameters for a particular frequency.
void setFrequency( StkFloat val );
//! Apply breath velocity to instrument with given amplitude and rate of increase.
void startBlowing( StkFloat amplitude, StkFloat rate );
//! Decrease breath velocity with given rate of decrease.
void stopBlowing( StkFloat rate );
//! Start a note with the given frequency and amplitude.
void noteOn( StkFloat frequency, StkFloat amplitude );
//! Stop a note with the given amplitude (speed of decay).
void noteOff( StkFloat amplitude );
//! Perform the control change specified by \e number and \e value (0.0 - 128.0).
void controlChange( int number, StkFloat value );
//! Compute and return one output sample.
StkFloat tick( unsigned int channel = 0 );
//! Fill a channel of the StkFrames object with computed outputs.
/*!
The \c channel argument must be less than the number of
channels in the StkFrames argument (the first channel is specified
by 0). However, range checking is only performed if _STK_DEBUG_
is defined during compilation, in which case an out-of-range value
will trigger an StkError exception.
*/
StkFrames& tick( StkFrames& frames, unsigned int channel = 0 );
void setBlowPressure( StkFloat val );
void setVibratoGain( StkFloat val );
void setVibratoFrequency( StkFloat val );
void setNoiseGain( StkFloat val );
void setBreathCutoff( StkFloat val );
void setSoftness( StkFloat val );
private:
DelayL pinDelay_;
DelayL poutDelay_;
DelayL jetDelay_;
Iir radiation_filter_;
Iir visco_in_filter_;
Iir visco_out_filter_;
Iir jetFilter_;
Noise turb_;
Iir turbFilter_;
SineWave vibrato_;
ADSR adsr_;
//StkFloat M{ 0 };
//StkFloat maxPressure_( 0 );
double maxPressure_;
//StkFloat blow{ 0 };
StkFloat vibratoGain_;
StkFloat noiseGain_;
StkFloat breathCutoff_;
StkFloat outputGain_;
StkFloat psi_;
StkFloat poutL_;
StkFloat pout_;
StkFloat poutm1_;
StkFloat poutm2_;
StkFloat pin_;
StkFloat pinm1_;
StkFloat pinm2_;
StkFloat b1;
StkFloat b3;
StkFloat b4;
StkFloat Uj_;
StkFloat Ujm1_;
StkFloat Qj_;
StkFloat Qjm1_;
StkFloat Qjm2_;
StkFloat Q1_;
StkFloat Q1m1_;
StkFloat Q1m2_;
StkFloat Qp_;
StkFloat Qpm1_;
StkFloat pm_;
};
inline StkFrames& Recorder :: tick( StkFrames& frames, unsigned int channel )
{
unsigned int nChannels = lastFrame_.channels();
#if defined(_STK_DEBUG_)
if ( channel > frames.channels() - nChannels ) {
oStream_ << "Recorder::tick(): channel and StkFrames arguments are incompatible!";
handleError( StkError::FUNCTION_ARGUMENT );
}
#endif
StkFloat *samples = &frames[channel];
unsigned int j, hop = frames.channels() - nChannels;
if ( nChannels == 1 ) {
for ( unsigned int i=0; i<frames.frames(); i++, samples += hop )
*samples++ = tick();
}
else {
for ( unsigned int i=0; i<frames.frames(); i++, samples += hop ) {
*samples++ = tick();
for ( j=1; j<nChannels; j++ )
*samples++ = lastFrame_[j];
}
}
return frames;
}
} // stk namespace
#endif