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

Diff of /linuxsampler/trunk/src/engines/EngineChannelBase.h

Parent Directory Parent Directory | Revision Log Revision Log | View Patch Patch

revision 2659 by schoenebeck, Sun Jun 29 19:34:28 2014 UTC revision 2884 by schoenebeck, Wed Apr 20 15:22:58 2016 UTC
# Line 4  Line 4 
4   *                                                                         *   *                                                                         *
5   *   Copyright (C) 2003,2004 by Benno Senoner and Christian Schoenebeck    *   *   Copyright (C) 2003,2004 by Benno Senoner and Christian Schoenebeck    *
6   *   Copyright (C) 2005-2008 Christian Schoenebeck                         *   *   Copyright (C) 2005-2008 Christian Schoenebeck                         *
7   *   Copyright (C) 2009-2013 Christian Schoenebeck and Grigor Iliev        *   *   Copyright (C) 2009-2012 Christian Schoenebeck and Grigor Iliev        *
8     *   Copyright (C) 2012-2016 Christian Schoenebeck and Andreas Persson     *
9   *                                                                         *   *                                                                         *
10   *   This program is free software; you can redistribute it and/or modify  *   *   This program is free software; you can redistribute it and/or modify  *
11   *   it under the terms of the GNU General Public License as published by  *   *   it under the terms of the GNU General Public License as published by  *
# Line 49  namespace LinuxSampler { Line 50  namespace LinuxSampler {
50      };      };
51    
52      template<class V>      template<class V>
53      class VoicePool {      class NotePool {
54          public:          public:
55              virtual Pool<V>* GetVoicePool() = 0;              virtual Pool<V>* GetVoicePool() = 0;
56                virtual Pool< Note<V> >* GetNotePool() = 0;
57                virtual Pool<note_id_t>* GetNodeIDPool() = 0;
58      };      };
59    
60      template <class V /* Voice */, class R /* Region */, class I /* Instrument */>      template <class V /* Voice */, class R /* Region */, class I /* Instrument */>
61      class EngineChannelBase: public AbstractEngineChannel, public MidiKeyboardManager<V>, public ResourceConsumer<I> {      class EngineChannelBase: public AbstractEngineChannel, public MidiKeyboardManager<V>, public ResourceConsumer<I> {
62          public:          public:
63                typedef typename RTList< Note<V> >::Iterator NoteIterator;
64              typedef typename RTList<R*>::Iterator RTListRegionIterator;              typedef typename RTList<R*>::Iterator RTListRegionIterator;
65              typedef typename MidiKeyboardManager<V>::MidiKey MidiKey;              typedef typename MidiKeyboardManager<V>::MidiKey MidiKey;
66    
# Line 133  namespace LinuxSampler { Line 137  namespace LinuxSampler {
137                      LockGuard lock(EngineMutex);                      LockGuard lock(EngineMutex);
138                      pEngine = newEngine;                      pEngine = newEngine;
139                  }                  }
140                  ResetInternal();                  ResetInternal(false/*don't reset engine*/); // 'false' is error prone here, but the danger of recursion with 'true' would be worse, there could be a better solution though
141                  pEvents = new RTList<Event>(pEngine->pEventPool);                  pEvents = new RTList<Event>(pEngine->pEventPool);
142                    delayedEvents.pList = new RTList<Event>(pEngine->pEventPool);
143    
144                  RegionPools<R>* pRegionPool = dynamic_cast<RegionPools<R>*>(pEngine);                  RegionPools<R>* pRegionPool = dynamic_cast<RegionPools<R>*>(pEngine);
145                  // reset the instrument change command struct (need to be done                  // reset the instrument change command struct (need to be done
# Line 161  namespace LinuxSampler { Line 166  namespace LinuxSampler {
166                      bStatusChanged = true;                      bStatusChanged = true;
167                  }                  }
168    
169                  VoicePool<V>* pVoicePool = dynamic_cast<VoicePool<V>*>(pEngine);                  NotePool<V>* pNotePool = dynamic_cast<NotePool<V>*>(pEngine);
170                  MidiKeyboardManager<V>::AllocateActiveVoices(pVoicePool->GetVoicePool());                  MidiKeyboardManager<V>::AllocateActiveNotesLists(
171                  MidiKeyboardManager<V>::AllocateEvents(pEngine->pEventPool);                      pNotePool->GetNotePool(),
172                        pNotePool->GetVoicePool()
173                    );
174                    MidiKeyboardManager<V>::AllocateEventsLists(pEngine->pEventPool);
175    
176                  AudioDeviceChannelLeft  = 0;                  AudioDeviceChannelLeft  = 0;
177                  AudioDeviceChannelRight = 1;                  AudioDeviceChannelRight = 1;
# Line 184  namespace LinuxSampler { Line 192  namespace LinuxSampler {
192              virtual void DisconnectAudioOutputDevice() {              virtual void DisconnectAudioOutputDevice() {
193                  if (pEngine) { // if clause to prevent disconnect loops                  if (pEngine) { // if clause to prevent disconnect loops
194    
195                      ResetInternal();                      ResetInternal(false/*don't reset engine*/); // 'false' is error prone here, but the danger of recursion with 'true' would be worse, there could be a better solution though
196    
197                      DeleteRegionsInUse();                      DeleteRegionsInUse();
198                      UnloadScriptInUse();                      UnloadScriptInUse();
# Line 199  namespace LinuxSampler { Line 207  namespace LinuxSampler {
207                          delete pEvents;                          delete pEvents;
208                          pEvents = NULL;                          pEvents = NULL;
209                      }                      }
210                        if (delayedEvents.pList) {
211                            delete delayedEvents.pList;
212                            delayedEvents.pList = NULL;
213                        }
214    
215                      MidiKeyboardManager<V>::DeleteActiveVoices();                      MidiKeyboardManager<V>::DeleteActiveNotesLists();
216                      MidiKeyboardManager<V>::DeleteEvents();                      MidiKeyboardManager<V>::DeleteEventsLists();
217                      DeleteGroupEventLists();                      DeleteGroupEventLists();
218    
219                      AudioOutputDevice* oldAudioDevice = pEngine->pAudioOutputDevice;                      AudioOutputDevice* oldAudioDevice = pEngine->pAudioOutputDevice;
# Line 226  namespace LinuxSampler { Line 238  namespace LinuxSampler {
238                      virtual bool Process(MidiKey* pMidiKey) { pMidiKey->pEvents->clear(); return false; }                      virtual bool Process(MidiKey* pMidiKey) { pMidiKey->pEvents->clear(); return false; }
239              };              };
240    
241              void ClearEventLists() {              /**
242                 * Free all events of the current audio fragment cycle. Calling
243                 * this method will @b NOT free events scheduled past the current
244                 * fragment's boundary! (@see AbstractEngineChannel::delayedEvents).
245                 */
246                void ClearEventListsOfCurrentFragment() {
247                  pEvents->clear();                  pEvents->clear();
248                  // empty MIDI key specific event lists                  // empty MIDI key specific event lists
249                  ClearEventListsHandler handler;                  ClearEventListsHandler handler;
# Line 249  namespace LinuxSampler { Line 266  namespace LinuxSampler {
266              virtual void ResourceToBeUpdated(I* pResource, void*& pUpdateArg) OVERRIDE {              virtual void ResourceToBeUpdated(I* pResource, void*& pUpdateArg) OVERRIDE {
267                  dmsg(3,("EngineChannelBase: Received instrument update message.\n"));                  dmsg(3,("EngineChannelBase: Received instrument update message.\n"));
268                  if (pEngine) pEngine->DisableAndLock();                  if (pEngine) pEngine->DisableAndLock();
269                  ResetInternal();                  ResetInternal(false/*don't reset engine*/);
270                  this->pInstrument = NULL;                  this->pInstrument = NULL;
271              }              }
272    
# Line 271  namespace LinuxSampler { Line 288  namespace LinuxSampler {
288               */               */
289              virtual void OnResourceProgress(float fProgress) OVERRIDE {              virtual void OnResourceProgress(float fProgress) OVERRIDE {
290                  this->InstrumentStat = int(fProgress * 100.0f);                  this->InstrumentStat = int(fProgress * 100.0f);
291                  dmsg(7,("EngineChannelBase: progress %d%", InstrumentStat));                  dmsg(7,("EngineChannelBase: progress %d%%", InstrumentStat));
292                  bStatusChanged = true; // status of engine has changed, so set notify flag                  bStatusChanged = true; // status of engine has changed, so set notify flag
293              }              }
294    
# Line 283  namespace LinuxSampler { Line 300  namespace LinuxSampler {
300                  SetDiskStreamCount(handler.StreamCount);                  SetDiskStreamCount(handler.StreamCount);
301              }              }
302    
303                /**
304                 * Called by real-time instrument script functions to schedule a
305                 * new note (new note-on event and a new @c Note object linked to it)
306                 * @a delay microseconds in future.
307                 *
308                 * @b IMPORTANT: for the supplied @a delay to be scheduled
309                 * correctly, the passed @a pEvent must be assigned a valid
310                 * fragment time within the current audio fragment boundaries. That
311                 * fragment time will be used by this method as basis for
312                 * interpreting what "now" acutally is, and thus it will be used as
313                 * basis for calculating the precise scheduling time for @a delay.
314                 * The easiest way to achieve this is by copying a recent event
315                 * which happened within the current audio fragment cycle: i.e. the
316                 * original event which caused calling this method here, or by using
317                 * Event::copyTimefrom() method to only copy the time, without any
318                 * other event data.
319                 *
320                 * @param pEvent - note-on event to be scheduled in future (event
321                 *                 data will be copied)
322                 * @param delay - amount of microseconds in future (from now) when
323                 *                event shall be processed
324                 * @returns unique note ID of scheduled new note, or NULL on error
325                 */
326                note_id_t ScheduleNoteMicroSec(const Event* pEvent, int delay) OVERRIDE {
327                    // add (copied) note-on event into scheduler queue
328                    const event_id_t noteOnEventID = ScheduleEventMicroSec(pEvent, delay);
329                    if (!noteOnEventID) return 0; // error
330                    // get access to (copied) event on the scheduler queue
331                    RTList<Event>::Iterator itEvent = pEvents->fromID(noteOnEventID);
332                    // stick a new note to the (copied) event on the queue
333                    const note_id_t noteID = pEngine->LaunchNewNote(this, &*itEvent);
334                    return noteID;
335                }
336    
337                /**
338                 * Called by real-time instrument script functions to ignore the note
339                 * reflected by given note ID. The note's event will be freed immediately
340                 * to its event pool and this will prevent voices to be launched for the
341                 * note.
342                 *
343                 * NOTE: preventing a note by calling this method works only if the note
344                 * was launched within the current audio fragment cycle.
345                 *
346                 * @param id - unique ID of note to be dropped
347                 */
348                void IgnoreNote(note_id_t id) OVERRIDE {
349                    Pool< Note<V> >* pNotePool =
350                        dynamic_cast<NotePool<V>*>(pEngine)->GetNotePool();
351    
352                    NoteIterator itNote = pNotePool->fromID(id);
353                    if (!itNote) return; // note probably already released
354    
355                    // if the note already got active voices, then it is too late to drop it
356                    if (!itNote->pActiveVoices->isEmpty()) return;
357    
358                    // if the original (note-on) event is not available anymore, then it is too late to drop it
359                    RTList<Event>::Iterator itEvent = pEvents->fromID(itNote->eventID);
360                    if (!itEvent) return;
361    
362                    // drop the note
363                    pNotePool->free(itNote);
364    
365                    // drop the original event
366                    pEvents->free(itEvent);
367                }
368    
369              RTList<R*>* pRegionsInUse;     ///< temporary pointer into the instrument change command, used by the audio thread              RTList<R*>* pRegionsInUse;     ///< temporary pointer into the instrument change command, used by the audio thread
370              I* pInstrument;              I* pInstrument;
371    
# Line 370  namespace LinuxSampler { Line 453  namespace LinuxSampler {
453              SyncConfInstrChangeCmdReader InstrumentChangeCommandReader;              SyncConfInstrChangeCmdReader InstrumentChangeCommandReader;
454    
455              /** This method is not thread safe! */              /** This method is not thread safe! */
456              virtual void ResetInternal() {              virtual void ResetInternal(bool bResetEngine) OVERRIDE {
457                  AbstractEngineChannel::ResetInternal();                  AbstractEngineChannel::ResetInternal(bResetEngine);
458    
459                  MidiKeyboardManager<V>::Reset();                  MidiKeyboardManager<V>::Reset();
460              }              }

Legend:
Removed from v.2659  
changed lines
  Added in v.2884

  ViewVC Help
Powered by ViewVC