/[svn]/linuxsampler/trunk/src/engines/gig/EngineChannel.cpp
ViewVC logotype

Diff of /linuxsampler/trunk/src/engines/gig/EngineChannel.cpp

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

revision 1653 by schoenebeck, Wed Jan 30 01:51:46 2008 UTC revision 1924 by persson, Sun Jun 28 16:43:38 2009 UTC
# Line 3  Line 3 
3   *   LinuxSampler - modular, streaming capable sampler                     *   *   LinuxSampler - modular, streaming capable sampler                     *
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 - 2009 Christian Schoenebeck                       *
7   *                                                                         *   *                                                                         *
8   *   This program is free software; you can redistribute it and/or modify  *   *   This program is free software; you can redistribute it and/or modify  *
9   *   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 24  Line 24 
24  #include "EngineChannel.h"  #include "EngineChannel.h"
25    
26  #include "../../common/global_private.h"  #include "../../common/global_private.h"
27    #include "../../Sampler.h"
28    
29  namespace LinuxSampler { namespace gig {  namespace LinuxSampler { namespace gig {
30    
31      EngineChannel::EngineChannel() :      EngineChannel::EngineChannel() :
32          InstrumentChangeCommandReader(InstrumentChangeCommand) {          InstrumentChangeCommandReader(InstrumentChangeCommand),
33            virtualMidiDevicesReader_AudioThread(virtualMidiDevices),
34            virtualMidiDevicesReader_MidiThread(virtualMidiDevices)
35        {
36          pMIDIKeyInfo = new midi_key_info_t[128];          pMIDIKeyInfo = new midi_key_info_t[128];
37          pEngine      = NULL;          pEngine      = NULL;
38          pInstrument  = NULL;          pInstrument  = NULL;
# Line 56  namespace LinuxSampler { namespace gig { Line 60  namespace LinuxSampler { namespace gig {
60          SoloMode       = false;          SoloMode       = false;
61          PortamentoMode = false;          PortamentoMode = false;
62          PortamentoTime = CONFIG_PORTAMENTO_TIME_DEFAULT;          PortamentoTime = CONFIG_PORTAMENTO_TIME_DEFAULT;
63    
64            // reset the instrument change command struct (need to be done
65            // twice, as it is double buffered)
66            {
67                instrument_change_command_t& cmd = InstrumentChangeCommand.GetConfigForUpdate();
68                cmd.pDimRegionsInUse = NULL;
69                cmd.pInstrument = NULL;
70                cmd.bChangeInstrument = false;
71            }
72            {
73                instrument_change_command_t& cmd = InstrumentChangeCommand.SwitchConfig();
74                cmd.pDimRegionsInUse = NULL;
75                cmd.pInstrument = NULL;
76                cmd.bChangeInstrument = false;
77            }
78      }      }
79    
80      EngineChannel::~EngineChannel() {      EngineChannel::~EngineChannel() {
81          DisconnectAudioOutputDevice();          DisconnectAudioOutputDevice();
82    
83            // In case the channel was removed before the instrument was
84            // fully loaded, try to give back instrument again (see bug #113)
85            instrument_change_command_t& cmd = ChangeInstrument(NULL);
86            if (cmd.pInstrument) {
87                    Engine::instruments.HandBack(cmd.pInstrument, this);
88            }
89            ///////
90    
91          if (pEventQueue) delete pEventQueue;          if (pEventQueue) delete pEventQueue;
92          if (pActiveKeys) delete pActiveKeys;          if (pActiveKeys) delete pActiveKeys;
93          if (pMIDIKeyInfo) delete[] pMIDIKeyInfo;          if (pMIDIKeyInfo) delete[] pMIDIKeyInfo;
# Line 293  namespace LinuxSampler { namespace gig { Line 321  namespace LinuxSampler { namespace gig {
321              cmd.bChangeInstrument = false;              cmd.bChangeInstrument = false;
322          }          }
323    
324            if (pInstrument != NULL) {
325                pInstrument = NULL;
326                InstrumentStat = -1;
327                InstrumentIdx  = -1;
328                InstrumentIdxName = "";
329                InstrumentFile = "";
330                bStatusChanged = true;
331            }
332    
333          for (uint i = 0; i < 128; i++) {          for (uint i = 0; i < 128; i++) {
334              pMIDIKeyInfo[i].pActiveVoices = new RTList<Voice>(pEngine->pVoicePool);              pMIDIKeyInfo[i].pActiveVoices = new RTList<Voice>(pEngine->pVoicePool);
335              pMIDIKeyInfo[i].pEvents       = new RTList<Event>(pEngine->pEventPool);              pMIDIKeyInfo[i].pEvents       = new RTList<Event>(pEngine->pEventPool);
# Line 316  namespace LinuxSampler { namespace gig { Line 353  namespace LinuxSampler { namespace gig {
353      void EngineChannel::DisconnectAudioOutputDevice() {      void EngineChannel::DisconnectAudioOutputDevice() {
354          if (pEngine) { // if clause to prevent disconnect loops          if (pEngine) { // if clause to prevent disconnect loops
355    
356                ResetInternal();
357    
358              // delete the structures used for instrument change              // delete the structures used for instrument change
359              RTList< ::gig::DimensionRegion*>* d = InstrumentChangeCommand.GetConfigForUpdate().pDimRegionsInUse;              RTList< ::gig::DimensionRegion*>* d = InstrumentChangeCommand.GetConfigForUpdate().pDimRegionsInUse;
360              if (d) delete d;              if (d) delete d;
361              EngineChannel::instrument_change_command_t& cmd = InstrumentChangeCommand.SwitchConfig();              EngineChannel::instrument_change_command_t& cmd = InstrumentChangeCommand.SwitchConfig();
362              d = cmd.pDimRegionsInUse;              d = cmd.pDimRegionsInUse;
363                if (d) delete d;
364    
365              if (cmd.pInstrument) {              if (cmd.pInstrument) {
366                  // release the currently loaded instrument                  // release the currently loaded instrument
367                  Engine::instruments.HandBackInstrument(cmd.pInstrument, this, d);                  Engine::instruments.HandBack(cmd.pInstrument, this);
             }  
             if (d) delete d;  
   
             // release all active dimension regions to resource  
             // manager  
             RTList<uint>::Iterator iuiKey = pActiveKeys->first();  
             RTList<uint>::Iterator end    = pActiveKeys->end();  
             while (iuiKey != end) { // iterate through all active keys  
                 midi_key_info_t* pKey = &pMIDIKeyInfo[*iuiKey];  
                 ++iuiKey;  
   
                 RTList<Voice>::Iterator itVoice     = pKey->pActiveVoices->first();  
                 RTList<Voice>::Iterator itVoicesEnd = pKey->pActiveVoices->end();  
                 for (; itVoice != itVoicesEnd; ++itVoice) { // iterate through all voices on this key  
                     Engine::instruments.HandBackDimReg(itVoice->pDimRgn);  
                 }  
368              }              }
369    
             ResetInternal();  
370              if (pEvents) {              if (pEvents) {
371                  delete pEvents;                  delete pEvents;
372                  pEvents = NULL;                  pEvents = NULL;
# Line 358  namespace LinuxSampler { namespace gig { Line 381  namespace LinuxSampler { namespace gig {
381                      pMIDIKeyInfo[i].pEvents = NULL;                      pMIDIKeyInfo[i].pEvents = NULL;
382                  }                  }
383              }              }
             Engine* oldEngine = pEngine;  
384              AudioOutputDevice* oldAudioDevice = pEngine->pAudioOutputDevice;              AudioOutputDevice* oldAudioDevice = pEngine->pAudioOutputDevice;
385              pEngine = NULL;              pEngine = NULL;
386              Engine::FreeEngine(this, oldAudioDevice);              Engine::FreeEngine(this, oldAudioDevice);
# Line 448  namespace LinuxSampler { namespace gig { Line 470  namespace LinuxSampler { namespace gig {
470          }          }
471          fxSends.push_back(pFxSend);          fxSends.push_back(pFxSend);
472          if (pEngine) pEngine->Enable();          if (pEngine) pEngine->Enable();
473          fireFxSendCountChanged(iSamplerChannelIndex, GetFxSendCount());          fireFxSendCountChanged(GetSamplerChannel()->Index(), GetFxSendCount());
474    
475          return pFxSend;          return pFxSend;
476      }      }
# Line 487  namespace LinuxSampler { namespace gig { Line 509  namespace LinuxSampler { namespace gig {
509              }              }
510          }          }
511          if (pEngine) pEngine->Enable();          if (pEngine) pEngine->Enable();
512          fireFxSendCountChanged(iSamplerChannelIndex, GetFxSendCount());          fireFxSendCountChanged(GetSamplerChannel()->Index(), GetFxSendCount());
513      }      }
514    
515      /**      /**
# Line 508  namespace LinuxSampler { namespace gig { Line 530  namespace LinuxSampler { namespace gig {
530              event.pEngineChannel      = this;              event.pEngineChannel      = this;
531              if (this->pEventQueue->write_space() > 0) this->pEventQueue->push(&event);              if (this->pEventQueue->write_space() > 0) this->pEventQueue->push(&event);
532              else dmsg(1,("EngineChannel: Input event queue full!"));              else dmsg(1,("EngineChannel: Input event queue full!"));
533              // inform instrument editor(s), if any ...              // inform connected virtual MIDI devices if any ...
534              pEngine->instruments.TrySendNoteOnToEditors(Key, Velocity, pInstrument);              // (e.g. virtual MIDI keyboard in instrument editor(s))
535                ArrayList<VirtualMidiDevice*>& devices =
536                    const_cast<ArrayList<VirtualMidiDevice*>&>(
537                        virtualMidiDevicesReader_MidiThread.Lock()
538                    );
539                for (int i = 0; i < devices.size(); i++) {
540                    devices[i]->SendNoteOnToDevice(Key, Velocity);
541                }
542                virtualMidiDevicesReader_MidiThread.Unlock();
543          }          }
544      }      }
545    
# Line 536  namespace LinuxSampler { namespace gig { Line 566  namespace LinuxSampler { namespace gig {
566              event.pEngineChannel      = this;              event.pEngineChannel      = this;
567              if (this->pEventQueue->write_space() > 0) this->pEventQueue->push(&event);              if (this->pEventQueue->write_space() > 0) this->pEventQueue->push(&event);
568              else dmsg(1,("EngineChannel: Input event queue full!"));              else dmsg(1,("EngineChannel: Input event queue full!"));
569              // inform instrument editor(s), if any ...              // inform connected virtual MIDI devices if any ...
570              pEngine->instruments.TrySendNoteOnToEditors(Key, Velocity, pInstrument);              // (e.g. virtual MIDI keyboard in instrument editor(s))
571                ArrayList<VirtualMidiDevice*>& devices =
572                    const_cast<ArrayList<VirtualMidiDevice*>&>(
573                        virtualMidiDevicesReader_MidiThread.Lock()
574                    );
575                for (int i = 0; i < devices.size(); i++) {
576                    devices[i]->SendNoteOnToDevice(Key, Velocity);
577                }
578                virtualMidiDevicesReader_MidiThread.Unlock();
579          }          }
580      }      }
581    
# Line 559  namespace LinuxSampler { namespace gig { Line 597  namespace LinuxSampler { namespace gig {
597              event.pEngineChannel      = this;              event.pEngineChannel      = this;
598              if (this->pEventQueue->write_space() > 0) this->pEventQueue->push(&event);              if (this->pEventQueue->write_space() > 0) this->pEventQueue->push(&event);
599              else dmsg(1,("EngineChannel: Input event queue full!"));              else dmsg(1,("EngineChannel: Input event queue full!"));
600              // inform instrument editor(s), if any ...              // inform connected virtual MIDI devices if any ...
601              pEngine->instruments.TrySendNoteOffToEditors(Key, Velocity, pInstrument);              // (e.g. virtual MIDI keyboard in instrument editor(s))
602                ArrayList<VirtualMidiDevice*>& devices =
603                    const_cast<ArrayList<VirtualMidiDevice*>&>(
604                        virtualMidiDevicesReader_MidiThread.Lock()
605                    );
606                for (int i = 0; i < devices.size(); i++) {
607                    devices[i]->SendNoteOffToDevice(Key, Velocity);
608                }
609                virtualMidiDevicesReader_MidiThread.Unlock();
610          }          }
611      }      }
612    
# Line 587  namespace LinuxSampler { namespace gig { Line 633  namespace LinuxSampler { namespace gig {
633              event.pEngineChannel      = this;              event.pEngineChannel      = this;
634              if (this->pEventQueue->write_space() > 0) this->pEventQueue->push(&event);              if (this->pEventQueue->write_space() > 0) this->pEventQueue->push(&event);
635              else dmsg(1,("EngineChannel: Input event queue full!"));              else dmsg(1,("EngineChannel: Input event queue full!"));
636              // inform instrument editor(s), if any ...              // inform connected virtual MIDI devices if any ...
637              pEngine->instruments.TrySendNoteOffToEditors(Key, Velocity, pInstrument);              // (e.g. virtual MIDI keyboard in instrument editor(s))
638                ArrayList<VirtualMidiDevice*>& devices =
639                    const_cast<ArrayList<VirtualMidiDevice*>&>(
640                        virtualMidiDevicesReader_MidiThread.Lock()
641                    );
642                for (int i = 0; i < devices.size(); i++) {
643                    devices[i]->SendNoteOffToDevice(Key, Velocity);
644                }
645                virtualMidiDevicesReader_MidiThread.Unlock();
646          }          }
647      }      }
648    
# Line 682  namespace LinuxSampler { namespace gig { Line 736  namespace LinuxSampler { namespace gig {
736          }          }
737      }      }
738    
739        /**
740         *  Will be called by the MIDIIn Thread to signal that a program
741         *  change should be performed. As a program change isn't
742         *  real-time safe, the actual change is performed by the disk
743         *  thread.
744         *
745         *  @param Program     - MIDI program change number
746         */
747        void EngineChannel::SendProgramChange(uint8_t Program) {
748            if (pEngine) {
749                pEngine->pDiskThread->OrderProgramChange(Program, this);
750            }
751        }
752    
753      void EngineChannel::ClearEventLists() {      void EngineChannel::ClearEventLists() {
754          pEvents->clear();          pEvents->clear();
755          // empty MIDI key specific event lists          // empty MIDI key specific event lists
# Line 702  namespace LinuxSampler { namespace gig { Line 770  namespace LinuxSampler { namespace gig {
770          MidiVolume     = 1.0;          MidiVolume     = 1.0;
771          GlobalPanLeft  = 1.0f;          GlobalPanLeft  = 1.0f;
772          GlobalPanRight = 1.0f;          GlobalPanRight = 1.0f;
773            iLastPanRequest = 64;
774          GlobalTranspose = 0;          GlobalTranspose = 0;
775          // set all MIDI controller values to zero          // set all MIDI controller values to zero
776          memset(ControllerTable, 0x00, 129);          memset(ControllerTable, 0x00, 129);
# Line 728  namespace LinuxSampler { namespace gig { Line 797  namespace LinuxSampler { namespace gig {
797       *                  current audio cycle       *                  current audio cycle
798       */       */
799      void EngineChannel::ImportEvents(uint Samples) {      void EngineChannel::ImportEvents(uint Samples) {
800            // import events from pure software MIDI "devices"
801            // (e.g. virtual keyboard in instrument editor)
802            {
803                const int FragmentPos = 0; // randomly chosen, we don't care about jitter for virtual MIDI devices
804                Event event = pEngine->pEventGenerator->CreateEvent(FragmentPos);
805                VirtualMidiDevice::event_t devEvent; // the event format we get from the virtual MIDI device
806                // as we're going to (carefully) write some status to the
807                // synchronized struct, we cast away the const
808                ArrayList<VirtualMidiDevice*>& devices =
809                    const_cast<ArrayList<VirtualMidiDevice*>&>(virtualMidiDevicesReader_AudioThread.Lock());
810                // iterate through all virtual MIDI devices
811                for (int i = 0; i < devices.size(); i++) {
812                    VirtualMidiDevice* pDev = devices[i];
813                    // I think we can simply flush the whole FIFO(s), the user shouldn't be so fast ;-)
814                    while (pDev->GetMidiEventFromDevice(devEvent)) {
815                        event.Type =
816                            (devEvent.Type == VirtualMidiDevice::EVENT_TYPE_NOTEON) ?
817                                Event::type_note_on : Event::type_note_off;
818                        event.Param.Note.Key      = devEvent.Key;
819                        event.Param.Note.Velocity = devEvent.Velocity;
820                        event.pEngineChannel      = this;
821                        // copy event to internal event list
822                        if (pEvents->poolIsEmpty()) {
823                            dmsg(1,("Event pool emtpy!\n"));
824                            goto exitVirtualDevicesLoop;
825                        }
826                        *pEvents->allocAppend() = event;
827                    }
828                }
829            }
830            exitVirtualDevicesLoop:
831            virtualMidiDevicesReader_AudioThread.Unlock();
832    
833            // import events from the regular MIDI devices
834          RingBuffer<Event,false>::NonVolatileReader eventQueueReader = pEventQueue->get_non_volatile_reader();          RingBuffer<Event,false>::NonVolatileReader eventQueueReader = pEventQueue->get_non_volatile_reader();
835          Event* pEvent;          Event* pEvent;
836          while (true) {          while (true) {
# Line 773  namespace LinuxSampler { namespace gig { Line 876  namespace LinuxSampler { namespace gig {
876          if (pEngine) pEngine->Enable();          if (pEngine) pEngine->Enable();
877      }      }
878    
879        void EngineChannel::Connect(VirtualMidiDevice* pDevice) {
880            // double buffer ... double work ...
881            {
882                ArrayList<VirtualMidiDevice*>& devices = virtualMidiDevices.GetConfigForUpdate();
883                devices.add(pDevice);
884            }
885            {
886                ArrayList<VirtualMidiDevice*>& devices = virtualMidiDevices.SwitchConfig();
887                devices.add(pDevice);
888            }
889        }
890    
891        void EngineChannel::Disconnect(VirtualMidiDevice* pDevice) {
892            // double buffer ... double work ...
893            {
894                ArrayList<VirtualMidiDevice*>& devices = virtualMidiDevices.GetConfigForUpdate();
895                devices.remove(pDevice);
896            }
897            {
898                ArrayList<VirtualMidiDevice*>& devices = virtualMidiDevices.SwitchConfig();
899                devices.remove(pDevice);
900            }
901        }
902    
903      float EngineChannel::Volume() {      float EngineChannel::Volume() {
904          return GlobalVolume;          return GlobalVolume;
905      }      }
# Line 782  namespace LinuxSampler { namespace gig { Line 909  namespace LinuxSampler { namespace gig {
909          bStatusChanged = true; // status of engine channel has changed, so set notify flag          bStatusChanged = true; // status of engine channel has changed, so set notify flag
910      }      }
911    
912        float EngineChannel::Pan() {
913            return float(iLastPanRequest - 64) / 64.0f;
914        }
915    
916        void EngineChannel::Pan(float f) {
917            int iMidiPan = int(f * 64.0f) + 64;
918            if (iMidiPan > 127) iMidiPan = 127;
919            else if (iMidiPan < 0) iMidiPan = 0;
920            GlobalPanLeft  = Engine::PanCurve[128 - iMidiPan];
921            GlobalPanRight = Engine::PanCurve[iMidiPan];
922            iLastPanRequest = iMidiPan;
923        }
924    
925      uint EngineChannel::Channels() {      uint EngineChannel::Channels() {
926          return 2;          return 2;
927      }      }
# Line 806  namespace LinuxSampler { namespace gig { Line 946  namespace LinuxSampler { namespace gig {
946          return LS_GIG_ENGINE_NAME;          return LS_GIG_ENGINE_NAME;
947      }      }
948    
949        void EngineChannel::ClearDimRegionsInUse() {
950            {
951                instrument_change_command_t& cmd = InstrumentChangeCommand.GetConfigForUpdate();
952                if(cmd.pDimRegionsInUse != NULL) cmd.pDimRegionsInUse->clear();
953            }
954            {
955                instrument_change_command_t& cmd = InstrumentChangeCommand.SwitchConfig();
956                if(cmd.pDimRegionsInUse != NULL) cmd.pDimRegionsInUse->clear();
957            }
958        }
959    
960        void EngineChannel::ResetDimRegionsInUse() {
961            {
962                instrument_change_command_t& cmd = InstrumentChangeCommand.GetConfigForUpdate();
963                if(cmd.pDimRegionsInUse != NULL) {
964                    delete cmd.pDimRegionsInUse;
965                    cmd.pDimRegionsInUse = new RTList< ::gig::DimensionRegion*>(pEngine->pDimRegionPool[0]);
966                }
967            }
968            {
969                instrument_change_command_t& cmd = InstrumentChangeCommand.SwitchConfig();
970                if(cmd.pDimRegionsInUse != NULL) {
971                    delete cmd.pDimRegionsInUse;
972                    cmd.pDimRegionsInUse = new RTList< ::gig::DimensionRegion*>(pEngine->pDimRegionPool[1]);
973                }
974            }
975        }
976    
977  }} // namespace LinuxSampler::gig  }} // namespace LinuxSampler::gig

Legend:
Removed from v.1653  
changed lines
  Added in v.1924

  ViewVC Help
Powered by ViewVC