/[svn]/linuxsampler/trunk/src/engines/sfz/SfzSignalUnitRack.h
ViewVC logotype

Contents of /linuxsampler/trunk/src/engines/sfz/SfzSignalUnitRack.h

Parent Directory Parent Directory | Revision Log Revision Log


Revision 2237 - (show annotations) (download) (as text)
Fri Aug 12 13:07:05 2011 UTC (12 years, 8 months ago) by iliev
File MIME type: text/x-c++hdr
File size: 15689 byte(s)
* sfz engine: implemented opcodes pan_onccN, pan_smoothccN,
  pan_curveccN, egN_pan, egN_pan_curve, egN_pan_onccX,
  egN_pan_curveccX, pitch_veltrack
* sfz engine: when failed to parse a sfz file
  print the line number on which the error occurs

1 /***************************************************************************
2 * *
3 * LinuxSampler - modular, streaming capable sampler *
4 * *
5 * Copyright (C) 2011 Grigor Iliev *
6 * *
7 * This program is free software; you can redistribute it and/or modify *
8 * it under the terms of the GNU General Public License as published by *
9 * the Free Software Foundation; either version 2 of the License, or *
10 * (at your option) any later version. *
11 * *
12 * This program is distributed in the hope that it will be useful, *
13 * but WITHOUT ANY WARRANTY; without even the implied warranty of *
14 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the *
15 * GNU General Public License for more details. *
16 * *
17 * You should have received a copy of the GNU General Public License *
18 * along with this program; if not, write to the Free Software *
19 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, *
20 * MA 02111-1307 USA *
21 ***************************************************************************/
22
23 #ifndef __LS_SFZSIGNALUNITRACK_H__
24 #define __LS_SFZSIGNALUNITRACK_H__
25
26 #include "../common/SignalUnitRack.h"
27 #include "EG.h"
28 #include "EGADSR.h"
29 #include "../common/AbstractVoice.h"
30 #include "../common/PulseLFO.h"
31 #include "../common/SawLFO.h"
32 #include "../common/SineLFO.h"
33
34 namespace LinuxSampler { namespace sfz {
35 const int MaxUnitCount = 1000;
36 const int maxEgCount = 100; // Maximum number of v2 envelope generators
37 const int maxLfoCount = 100; // Maximum number of v2 LFOs
38
39 class Voice;
40 class SfzSignalUnitRack;
41
42 class SfzSignalUnit: public SignalUnit {
43 public:
44 Voice* pVoice;
45
46 SfzSignalUnit(SfzSignalUnitRack* rack);
47 SfzSignalUnit(const SfzSignalUnit& Unit): SignalUnit(Unit.pRack) { Copy(Unit); }
48
49 void Copy(const SfzSignalUnit& Unit) {
50 pVoice = Unit.pVoice;
51
52 SignalUnit::Copy(Unit);
53 }
54
55 double GetSampleRate();
56 float GetInfluence(ArrayList< ::sfz::CC>& cc);
57 };
58
59
60 class CCUnit: public CCSignalUnit {
61 public:
62 Voice* pVoice;
63
64 CCUnit(SfzSignalUnitRack* rack, Listener* l = NULL);
65
66 virtual void Trigger();
67
68 void SetCCs(::sfz::Array<int>& pCC);
69 void SetCCs(ArrayList< ::sfz::CC>& cc);
70
71 virtual void AddSmoothCC(uint8_t Controller, float Influence, short int Curve, float Smooth);
72
73 inline int GetCurveCount();
74 inline ::sfz::Curve* GetCurve(int idx);
75
76 double GetSampleRate();
77 };
78
79 class CurveCCUnit: public CCUnit {
80 public:
81 CurveCCUnit(SfzSignalUnitRack* rack, Listener* l = NULL): CCUnit(rack, l) { }
82
83 virtual float Normalize(uint8_t val, short int curve = -1) {
84 if (curve == -1) return val / 127.0f;
85 return GetCurve(curve)->v[val];
86 }
87 };
88
89
90
91 class SmoothCCUnit: public CurveCCUnit {
92 protected:
93 Smoother Smoothers[128];
94 public:
95 SmoothCCUnit(SfzSignalUnitRack* rack, Listener* l = NULL): CurveCCUnit(rack, l) { }
96
97 virtual void AddSmoothCC(uint8_t Controller, float Influence, short int Curve, float Smooth);
98 };
99
100
101 class XFInCCUnit: public CCUnit {
102 public:
103 XFInCCUnit(SfzSignalUnitRack* rack, Listener* l = NULL): CCUnit(rack, l) { }
104
105 virtual bool Active() { return Ctrls.size() > 0; }
106 virtual void Calculate();
107 virtual void SetCrossFadeCCs(::sfz::Array<int>& loCCs, ::sfz::Array<int>& hiCCs);
108 };
109
110
111 class XFOutCCUnit: public XFInCCUnit {
112 public:
113 XFOutCCUnit(SfzSignalUnitRack* rack, Listener* l = NULL): XFInCCUnit(rack, l) { }
114
115 virtual void Calculate();
116 };
117
118
119 template <class T>
120 class EGUnit: public SfzSignalUnit {
121 public:
122 ::sfz::EG* pEGInfo;
123 T EG;
124
125 EGUnit(SfzSignalUnitRack* rack): SfzSignalUnit(rack), pEGInfo(NULL) { }
126 EGUnit(const EGUnit& Unit): SfzSignalUnit(Unit) { Copy(Unit); }
127 void operator=(const EGUnit& Unit) { Copy(Unit); }
128
129 void Copy(const EGUnit& Unit) {
130 pEGInfo = Unit.pEGInfo;
131
132 SfzSignalUnit::Copy(Unit);
133 }
134
135 virtual bool Active() { return EG.active(); }
136 virtual float GetLevel() { return DelayStage() ? 0 : EG.getLevel(); }
137
138 virtual void EnterReleaseStage() { EG.update(EG::event_release, GetSampleRate()); }
139 virtual void CancelRelease() { EG.update(EG::event_cancel_release, GetSampleRate()); }
140
141 virtual void Increment() {
142 if (DelayStage()) return;
143
144 SfzSignalUnit::Increment();
145 if (!EG.active()) return;
146
147 switch (EG.getSegmentType()) {
148 case EG::segment_lin:
149 EG.processLin();
150 break;
151 case EG::segment_exp:
152 EG.processExp();
153 break;
154 case EG::segment_end:
155 EG.getLevel();
156 break; // noop
157 case EG::segment_pow:
158 EG.processPow();
159 break;
160 }
161
162 if (EG.active()) {
163 EG.increment(1);
164 if (!EG.toStageEndLeft()) EG.update(EG::event_stage_end, GetSampleRate());
165 }
166 }
167 };
168
169 class EGv1Unit: public EGUnit<EGADSR> {
170 public:
171 int depth;
172 EGv1Unit(SfzSignalUnitRack* rack): EGUnit<EGADSR>(rack), depth(0) { }
173 };
174
175 class EGv2Unit: public EGUnit< ::LinuxSampler::sfz::EG> {
176 protected:
177 ::sfz::EG egInfo;
178 public:
179 CCUnit suAmpOnCC;
180 CCUnit suVolOnCC;
181 CCUnit suPitchOnCC;
182 CCUnit suCutoffOnCC;
183 CCUnit suResOnCC;
184 CurveCCUnit suPanOnCC;
185
186 EGv2Unit(SfzSignalUnitRack* rack);
187 virtual void Trigger();
188 };
189
190 class PitchEGUnit: public EGv1Unit {
191 public:
192 PitchEGUnit(SfzSignalUnitRack* rack): EGv1Unit(rack) { }
193 virtual void Trigger();
194 };
195
196 class FilEGUnit: public EGv1Unit {
197 public:
198 FilEGUnit(SfzSignalUnitRack* rack): EGv1Unit(rack) { }
199 virtual void Trigger();
200 };
201
202 class AmpEGUnit: public EGv1Unit {
203 public:
204 AmpEGUnit(SfzSignalUnitRack* rack): EGv1Unit(rack) { }
205 virtual void Trigger();
206 };
207
208 class AbstractLfo {
209 public:
210 virtual float Render() = 0;
211 virtual void Update(const uint16_t& ExtControlValue) = 0;
212 virtual void Trigger(float Frequency, start_level_t StartLevel, uint16_t InternalDepth, uint16_t ExtControlDepth, bool FlipPhase, unsigned int SampleRate) = 0;
213 virtual void SetPhase(float phase) = 0;
214 virtual void SetFrequency(float Frequency, unsigned int SampleRate) = 0;
215 };
216
217 template <class T>
218 class LfoBase: public AbstractLfo, public T {
219 public:
220 LfoBase(float Max): T(Max) { }
221 virtual float Render() { return T::render(); }
222
223 virtual void Update(const uint16_t& ExtControlValue) { T::update(ExtControlValue); }
224
225 virtual void Trigger (
226 float Frequency, start_level_t StartLevel, uint16_t InternalDepth,
227 uint16_t ExtControlDepth, bool FlipPhase, unsigned int SampleRate
228 ) {
229 T::trigger(Frequency, StartLevel, InternalDepth, ExtControlDepth, FlipPhase, SampleRate);
230 }
231
232 virtual void SetPhase(float phase) { T::setPhase(phase); }
233
234 virtual void SetFrequency(float Frequency, unsigned int SampleRate) {
235 T::setFrequency(Frequency, SampleRate);
236 }
237 };
238
239 class LFOUnit;
240
241 class FadeEGUnit: public EGUnit<EGADSR> {
242 public:
243 FadeEGUnit(SfzSignalUnitRack* rack): EGUnit<EGADSR>(rack) { }
244 virtual void Trigger() { }
245 virtual void EnterReleaseStage() { }
246 virtual void CancelRelease() { }
247
248 friend class LFOUnit;
249 };
250
251 class LFOUnit: public SfzSignalUnit, public CCSignalUnit::Listener {
252 public:
253 ::sfz::LFO* pLfoInfo;
254 AbstractLfo* pLFO;
255 FadeEGUnit suFadeEG;
256 SmoothCCUnit suDepthOnCC;
257 SmoothCCUnit suFreqOnCC;
258
259 LFOUnit(SfzSignalUnitRack* rack);
260 LFOUnit(const LFOUnit& Unit);
261 void operator=(const LFOUnit& Unit) { Copy(Unit); }
262
263 void Copy(const LFOUnit& Unit) {
264 pLfoInfo = Unit.pLfoInfo;
265 suFadeEG = Unit.suFadeEG;
266
267 SfzSignalUnit::Copy(Unit);
268 }
269
270 virtual bool Active() { return pLfoInfo->freq > 0; }
271 virtual void Trigger();
272 virtual void Increment();
273 virtual float GetLevel() { return Level; }
274 virtual void ValueChanged(CCSignalUnit* pUnit);
275 };
276
277 class LFOv1Unit: public LFOUnit {
278 public:
279 ::sfz::LFO lfoInfo;
280 LfoBase<LFOSigned> lfo;
281
282 LFOv1Unit(SfzSignalUnitRack* rack): LFOUnit(rack), lfo(1200.0f) {
283 pLfoInfo = &lfoInfo; pLFO = &lfo;
284 }
285
286 virtual void Trigger();
287 };
288
289 class LFOv2Unit: public LFOUnit {
290 protected:
291 FixedArray<AbstractLfo*> lfos;
292 LfoBase<LFOSigned> lfo0; // triangle
293 LfoBase<SineLFO<range_signed> > lfo1; // sine
294 LfoBase<PulseLFO<range_unsigned, 750> > lfo2; // pulse 75%
295 LfoBase<SquareLFO<range_signed> > lfo3; // square
296 LfoBase<PulseLFO<range_unsigned, 250> > lfo4; // pulse 25%
297 LfoBase<PulseLFO<range_unsigned, 125> > lfo5; // pulse 12,5%
298 LfoBase<SawLFO<range_unsigned, true> > lfo6; // saw up
299 LfoBase<SawLFO<range_unsigned, false> > lfo7; // saw down
300
301
302 public:
303 SmoothCCUnit suVolOnCC;
304 SmoothCCUnit suPitchOnCC;
305 SmoothCCUnit suPanOnCC;
306 SmoothCCUnit suCutoffOnCC;
307 SmoothCCUnit suResOnCC;
308
309 LFOv2Unit(SfzSignalUnitRack* rack);
310
311 virtual void Trigger();
312 };
313
314 class AmpLFOUnit: public LFOv1Unit {
315 public:
316 AmpLFOUnit(SfzSignalUnitRack* rack): LFOv1Unit(rack) { }
317
318 virtual void Trigger();
319 };
320
321 class PitchLFOUnit: public LFOv1Unit {
322 public:
323 PitchLFOUnit(SfzSignalUnitRack* rack): LFOv1Unit(rack) { }
324
325 virtual void Trigger();
326 };
327
328 class FilLFOUnit: public LFOv1Unit {
329 public:
330 FilLFOUnit(SfzSignalUnitRack* rack): LFOv1Unit(rack) { }
331
332 virtual void Trigger();
333 };
334
335
336 class EndpointUnit: public EndpointSignalUnit {
337 private:
338 float xfCoeff; // crossfade coefficient
339 float pitchVeltrackRatio;
340
341 public:
342 Voice* pVoice;
343 XFInCCUnit suXFInCC;
344 XFOutCCUnit suXFOutCC;
345 SmoothCCUnit suPanOnCC;
346
347 EndpointUnit(SfzSignalUnitRack* rack);
348
349 virtual void Trigger();
350
351 /** The endpoint should be active until the volume EG is active. */
352 virtual bool Active();
353
354 virtual float GetVolume();
355 virtual float GetFilterCutoff();
356 virtual float GetPitch();
357 virtual float GetResonance();
358 virtual float GetPan();
359
360 SfzSignalUnitRack* const GetRack();
361
362 virtual float CalculateResonance(float res) {
363 return GetResonance() + res;
364 }
365
366 virtual float CalculateFilterCutoff(float cutoff);
367 };
368
369
370 class SfzSignalUnitRack : public SignalUnitRack {
371 private:
372 EndpointUnit suEndpoint;
373 AmpEGUnit suVolEG;
374 FilEGUnit suFilEG;
375 PitchEGUnit suPitchEG;
376
377 AmpLFOUnit suAmpLFO;
378 PitchLFOUnit suPitchLFO;
379 FilLFOUnit suFilLFO;
380
381 // SFZ v2
382
383 SmoothCCUnit suVolOnCC;
384
385 FixedArray<EGv2Unit*> EGs;
386
387 // used for optimization - contains only the ones that are modulating volume
388 FixedArray<EGv2Unit*> volEGs;
389
390 // used for optimization - contains only the ones that are modulating pitch
391 FixedArray<EGv2Unit*> pitchEGs;
392
393 // used for optimization - contains only the ones that are modulating filter cutoff
394 FixedArray<EGv2Unit*> filEGs;
395
396 // used for optimization - contains only the ones that are modulating resonance
397 FixedArray<EGv2Unit*> resEGs;
398
399 // used for optimization - contains only the ones that are modulating pitch
400 FixedArray<EGv2Unit*> panEGs;
401
402
403 FixedArray<LFOv2Unit*> LFOs;
404
405 // used for optimization - contains only the ones that are modulating volume
406 FixedArray<LFOv2Unit*> volLFOs;
407
408 // used for optimization - contains only the ones that are modulating pitch
409 FixedArray<LFOv2Unit*> pitchLFOs;
410
411 // used for optimization - contains only the ones that are modulating filter cutoff
412 FixedArray<LFOv2Unit*> filLFOs;
413
414 // used for optimization - contains only the ones that are modulating resonance
415 FixedArray<LFOv2Unit*> resLFOs;
416
417 // used for optimization - contains only the ones that are modulating pan
418 FixedArray<LFOv2Unit*> panLFOs;
419
420
421 public:
422 Voice* const pVoice;
423
424 /**
425 * @param Voice The voice to which this rack belongs.
426 */
427 SfzSignalUnitRack(Voice* voice);
428 ~SfzSignalUnitRack();
429
430 virtual EndpointSignalUnit* GetEndpointUnit();
431
432 virtual void Trigger();
433 virtual void EnterFadeOutStage();
434
435 friend class EndpointUnit;
436 };
437 }} // namespace LinuxSampler::sfz
438
439 #endif /* __LS_SFZSIGNALUNITRACK_H__ */
440

  ViewVC Help
Powered by ViewVC