/[svn]/linuxsampler/trunk/src/drivers/midi/MidiInputPort.cpp
ViewVC logotype

Diff of /linuxsampler/trunk/src/drivers/midi/MidiInputPort.cpp

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

revision 244 by schoenebeck, Fri Sep 17 01:01:11 2004 UTC revision 1887 by persson, Sat Apr 18 08:17:16 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 - 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 22  Line 23 
23    
24  #include "MidiInputPort.h"  #include "MidiInputPort.h"
25    
26    #include "../../common/global_private.h"
27    #include "MidiInstrumentMapper.h"
28    #include "../../Sampler.h"
29    #include "../../engines/EngineFactory.h"
30    #include "VirtualMidiDevice.h"
31    
32    #include <algorithm>
33    
34  namespace LinuxSampler {  namespace LinuxSampler {
35    
36  // *************** ParameterName ***************  // *************** ParameterName ***************
# Line 47  namespace LinuxSampler { Line 56  namespace LinuxSampler {
56          return std::vector<String>();          return std::vector<String>();
57      }      }
58    
59      void MidiInputPort::ParameterName::OnSetValue(String s) throw (LinuxSamplerException) {      void MidiInputPort::ParameterName::OnSetValue(String s) throw (Exception) {
60          return; /* FIXME: Nothing to do here */          return; /* FIXME: Nothing to do here */
61      }      }
62    
# Line 59  namespace LinuxSampler { Line 68  namespace LinuxSampler {
68      MidiInputPort::~MidiInputPort() {      MidiInputPort::~MidiInputPort() {
69          std::map<String,DeviceRuntimeParameter*>::iterator iter = Parameters.begin();          std::map<String,DeviceRuntimeParameter*>::iterator iter = Parameters.begin();
70          while (iter != Parameters.end()) {          while (iter != Parameters.end()) {
             Parameters.erase(iter);  
71              delete iter->second;              delete iter->second;
72              iter++;              iter++;
73          }          }
74            Parameters.clear();
75      }      }
76    
77      MidiInputPort::MidiInputPort(MidiInputDevice* pDevice, int portNumber) {      MidiInputPort::MidiInputPort(MidiInputDevice* pDevice, int portNumber)
78            : MidiChannelMapReader(MidiChannelMap),
79              SysexListenersReader(SysexListeners),
80              virtualMidiDevicesReader(virtualMidiDevices) {
81          this->pDevice = pDevice;          this->pDevice = pDevice;
82          this->portNumber = portNumber;          this->portNumber = portNumber;
83          Parameters["NAME"] = new ParameterName(this);          Parameters["NAME"] = new ParameterName(this);
# Line 84  namespace LinuxSampler { Line 96  namespace LinuxSampler {
96      }      }
97    
98      void MidiInputPort::DispatchNoteOn(uint8_t Key, uint8_t Velocity, uint MidiChannel) {      void MidiInputPort::DispatchNoteOn(uint8_t Key, uint8_t Velocity, uint MidiChannel) {
99          std::set<Engine*>::iterator engineiter = MidiChannelMap[MidiChannel].begin();          if (Key > 127 || Velocity > 127 || MidiChannel > 16) return;
100          std::set<Engine*>::iterator end        = MidiChannelMap[MidiChannel].end();          const MidiChannelMap_t& midiChannelMap = MidiChannelMapReader.Lock();
101          for (; engineiter != end; engineiter++) (*engineiter)->SendNoteOn(Key, Velocity);          // dispatch event for engines listening to the same MIDI channel
102            {
103                std::set<EngineChannel*>::iterator engineiter = midiChannelMap[MidiChannel].begin();
104                std::set<EngineChannel*>::iterator end        = midiChannelMap[MidiChannel].end();
105                for (; engineiter != end; engineiter++) (*engineiter)->SendNoteOn(Key, Velocity);
106            }
107            // dispatch event for engines listening to ALL MIDI channels
108            {
109                std::set<EngineChannel*>::iterator engineiter = midiChannelMap[midi_chan_all].begin();
110                std::set<EngineChannel*>::iterator end        = midiChannelMap[midi_chan_all].end();
111                for (; engineiter != end; engineiter++) (*engineiter)->SendNoteOn(Key, Velocity);
112            }
113            MidiChannelMapReader.Unlock();
114    
115            // dispatch event to all low priority MIDI listeners
116            const std::vector<VirtualMidiDevice*>& listeners =
117                virtualMidiDevicesReader.Lock();
118            for (int i = 0; i < listeners.size(); ++i)
119                listeners[i]->SendNoteOnToDevice(Key, Velocity);
120            virtualMidiDevicesReader.Unlock();
121        }
122    
123        void MidiInputPort::DispatchNoteOn(uint8_t Key, uint8_t Velocity, uint MidiChannel, int32_t FragmentPos) {
124            if (Key > 127 || Velocity > 127 || MidiChannel > 16) return;
125            const MidiChannelMap_t& midiChannelMap = MidiChannelMapReader.Lock();
126            // dispatch event for engines listening to the same MIDI channel
127            {
128                std::set<EngineChannel*>::iterator engineiter = midiChannelMap[MidiChannel].begin();
129                std::set<EngineChannel*>::iterator end        = midiChannelMap[MidiChannel].end();
130                for (; engineiter != end; engineiter++) (*engineiter)->SendNoteOn(Key, Velocity, FragmentPos);
131            }
132            // dispatch event for engines listening to ALL MIDI channels
133            {
134                std::set<EngineChannel*>::iterator engineiter = midiChannelMap[midi_chan_all].begin();
135                std::set<EngineChannel*>::iterator end        = midiChannelMap[midi_chan_all].end();
136                for (; engineiter != end; engineiter++) (*engineiter)->SendNoteOn(Key, Velocity, FragmentPos);
137            }
138            MidiChannelMapReader.Unlock();
139    
140            // dispatch event to all low priority MIDI listeners
141            const std::vector<VirtualMidiDevice*>& listeners =
142                virtualMidiDevicesReader.Lock();
143            for (int i = 0; i < listeners.size(); ++i)
144                listeners[i]->SendNoteOnToDevice(Key, Velocity);
145            virtualMidiDevicesReader.Unlock();
146      }      }
147    
148      void MidiInputPort::DispatchNoteOff(uint8_t Key, uint8_t Velocity, uint MidiChannel) {      void MidiInputPort::DispatchNoteOff(uint8_t Key, uint8_t Velocity, uint MidiChannel) {
149          std::set<Engine*>::iterator engineiter = MidiChannelMap[MidiChannel].begin();          if (Key > 127 || Velocity > 127 || MidiChannel > 16) return;
150          std::set<Engine*>::iterator end        = MidiChannelMap[MidiChannel].end();          const MidiChannelMap_t& midiChannelMap = MidiChannelMapReader.Lock();
151          for (; engineiter != end; engineiter++) (*engineiter)->SendNoteOff(Key, Velocity);          // dispatch event for engines listening to the same MIDI channel
152            {
153                std::set<EngineChannel*>::iterator engineiter = midiChannelMap[MidiChannel].begin();
154                std::set<EngineChannel*>::iterator end        = midiChannelMap[MidiChannel].end();
155                for (; engineiter != end; engineiter++) (*engineiter)->SendNoteOff(Key, Velocity);
156            }
157            // dispatch event for engines listening to ALL MIDI channels
158            {
159                std::set<EngineChannel*>::iterator engineiter = midiChannelMap[midi_chan_all].begin();
160                std::set<EngineChannel*>::iterator end        = midiChannelMap[midi_chan_all].end();
161                for (; engineiter != end; engineiter++) (*engineiter)->SendNoteOff(Key, Velocity);
162            }
163            MidiChannelMapReader.Unlock();
164    
165            // dispatch event to all low priority MIDI listeners
166            const std::vector<VirtualMidiDevice*>& listeners =
167                virtualMidiDevicesReader.Lock();
168            for (int i = 0; i < listeners.size(); ++i)
169                listeners[i]->SendNoteOffToDevice(Key, Velocity);
170            virtualMidiDevicesReader.Unlock();
171        }
172    
173        void MidiInputPort::DispatchNoteOff(uint8_t Key, uint8_t Velocity, uint MidiChannel, int32_t FragmentPos) {
174            if (Key > 127 || Velocity > 127 || MidiChannel > 16) return;
175            const MidiChannelMap_t& midiChannelMap = MidiChannelMapReader.Lock();
176            // dispatch event for engines listening to the same MIDI channel
177            {
178                std::set<EngineChannel*>::iterator engineiter = midiChannelMap[MidiChannel].begin();
179                std::set<EngineChannel*>::iterator end        = midiChannelMap[MidiChannel].end();
180                for (; engineiter != end; engineiter++) (*engineiter)->SendNoteOff(Key, Velocity, FragmentPos);
181            }
182            // dispatch event for engines listening to ALL MIDI channels
183            {
184                std::set<EngineChannel*>::iterator engineiter = midiChannelMap[midi_chan_all].begin();
185                std::set<EngineChannel*>::iterator end        = midiChannelMap[midi_chan_all].end();
186                for (; engineiter != end; engineiter++) (*engineiter)->SendNoteOff(Key, Velocity, FragmentPos);
187            }
188            MidiChannelMapReader.Unlock();
189    
190            // dispatch event to all low priority MIDI listeners
191            const std::vector<VirtualMidiDevice*>& listeners =
192                virtualMidiDevicesReader.Lock();
193            for (int i = 0; i < listeners.size(); ++i)
194                listeners[i]->SendNoteOffToDevice(Key, Velocity);
195            virtualMidiDevicesReader.Unlock();
196      }      }
197    
198      void MidiInputPort::DispatchPitchbend(int Pitch, uint MidiChannel) {      void MidiInputPort::DispatchPitchbend(int Pitch, uint MidiChannel) {
199          std::set<Engine*>::iterator engineiter = MidiChannelMap[MidiChannel].begin();          if (Pitch < -8192 || Pitch > 8191 || MidiChannel > 16) return;
200          std::set<Engine*>::iterator end        = MidiChannelMap[MidiChannel].end();          const MidiChannelMap_t& midiChannelMap = MidiChannelMapReader.Lock();
201          for (; engineiter != end; engineiter++) (*engineiter)->SendPitchbend(Pitch);          // dispatch event for engines listening to the same MIDI channel
202            {
203                std::set<EngineChannel*>::iterator engineiter = midiChannelMap[MidiChannel].begin();
204                std::set<EngineChannel*>::iterator end        = midiChannelMap[MidiChannel].end();
205                for (; engineiter != end; engineiter++) (*engineiter)->SendPitchbend(Pitch);
206            }
207            // dispatch event for engines listening to ALL MIDI channels
208            {
209                std::set<EngineChannel*>::iterator engineiter = midiChannelMap[midi_chan_all].begin();
210                std::set<EngineChannel*>::iterator end        = midiChannelMap[midi_chan_all].end();
211                for (; engineiter != end; engineiter++) (*engineiter)->SendPitchbend(Pitch);
212            }
213            MidiChannelMapReader.Unlock();
214        }
215    
216        void MidiInputPort::DispatchPitchbend(int Pitch, uint MidiChannel, int32_t FragmentPos) {
217            if (Pitch < -8192 || Pitch > 8191 || MidiChannel > 16) return;
218            const MidiChannelMap_t& midiChannelMap = MidiChannelMapReader.Lock();
219            // dispatch event for engines listening to the same MIDI channel
220            {
221                std::set<EngineChannel*>::iterator engineiter = midiChannelMap[MidiChannel].begin();
222                std::set<EngineChannel*>::iterator end        = midiChannelMap[MidiChannel].end();
223                for (; engineiter != end; engineiter++) (*engineiter)->SendPitchbend(Pitch, FragmentPos);
224            }
225            // dispatch event for engines listening to ALL MIDI channels
226            {
227                std::set<EngineChannel*>::iterator engineiter = midiChannelMap[midi_chan_all].begin();
228                std::set<EngineChannel*>::iterator end        = midiChannelMap[midi_chan_all].end();
229                for (; engineiter != end; engineiter++) (*engineiter)->SendPitchbend(Pitch, FragmentPos);
230            }
231            MidiChannelMapReader.Unlock();
232      }      }
233    
234      void MidiInputPort::DispatchControlChange(uint8_t Controller, uint8_t Value, uint MidiChannel) {      void MidiInputPort::DispatchControlChange(uint8_t Controller, uint8_t Value, uint MidiChannel) {
235          std::set<Engine*>::iterator engineiter = MidiChannelMap[MidiChannel].begin();          if (Controller > 128 || Value > 127 || MidiChannel > 16) return;
236          std::set<Engine*>::iterator end        = MidiChannelMap[MidiChannel].end();          const MidiChannelMap_t& midiChannelMap = MidiChannelMapReader.Lock();
237          for (; engineiter != end; engineiter++) (*engineiter)->SendControlChange(Controller, Value);          // dispatch event for engines listening to the same MIDI channel
238            {
239                std::set<EngineChannel*>::iterator engineiter = midiChannelMap[MidiChannel].begin();
240                std::set<EngineChannel*>::iterator end        = midiChannelMap[MidiChannel].end();
241                for (; engineiter != end; engineiter++) (*engineiter)->SendControlChange(Controller, Value);
242            }
243            // dispatch event for engines listening to ALL MIDI channels
244            {
245                std::set<EngineChannel*>::iterator engineiter = midiChannelMap[midi_chan_all].begin();
246                std::set<EngineChannel*>::iterator end        = midiChannelMap[midi_chan_all].end();
247                for (; engineiter != end; engineiter++) (*engineiter)->SendControlChange(Controller, Value);
248            }
249            MidiChannelMapReader.Unlock();
250        }
251    
252        void MidiInputPort::DispatchControlChange(uint8_t Controller, uint8_t Value, uint MidiChannel, int32_t FragmentPos) {
253            if (Controller > 128 || Value > 127 || MidiChannel > 16) return;
254            const MidiChannelMap_t& midiChannelMap = MidiChannelMapReader.Lock();
255            // dispatch event for engines listening to the same MIDI channel
256            {
257                std::set<EngineChannel*>::iterator engineiter = midiChannelMap[MidiChannel].begin();
258                std::set<EngineChannel*>::iterator end        = midiChannelMap[MidiChannel].end();
259                for (; engineiter != end; engineiter++) (*engineiter)->SendControlChange(Controller, Value, FragmentPos);
260            }
261            // dispatch event for engines listening to ALL MIDI channels
262            {
263                std::set<EngineChannel*>::iterator engineiter = midiChannelMap[midi_chan_all].begin();
264                std::set<EngineChannel*>::iterator end        = midiChannelMap[midi_chan_all].end();
265                for (; engineiter != end; engineiter++) (*engineiter)->SendControlChange(Controller, Value, FragmentPos);
266            }
267            MidiChannelMapReader.Unlock();
268      }      }
269    
270      void MidiInputPort::DispatchSysex(void* pData, uint Size) {      void MidiInputPort::DispatchSysex(void* pData, uint Size) {
271          for (uint MidiChannel = 0; MidiChannel <= 16; MidiChannel++) {          const std::set<Engine*> allEngines = SysexListenersReader.Lock();
272              std::set<Engine*>::iterator engineiter = MidiChannelMap[MidiChannel].begin();          // dispatch event to all engine instances
273              std::set<Engine*>::iterator end        = MidiChannelMap[MidiChannel].end();          std::set<Engine*>::iterator engineiter = allEngines.begin();
274              for (; engineiter != end; engineiter++) (*engineiter)->SendSysex(pData, Size);          std::set<Engine*>::iterator end        = allEngines.end();
275            for (; engineiter != end; engineiter++) (*engineiter)->SendSysex(pData, Size, this);
276            SysexListenersReader.Unlock();
277        }
278    
279        void MidiInputPort::DispatchProgramChange(uint8_t Program, uint MidiChannel) {
280            dmsg(1,("Received MIDI program change (prog=%d,ch=%d)\n",Program,MidiChannel));
281            if (Program > 127 || MidiChannel > 16) return;
282            if (!pDevice || !pDevice->pSampler) {
283                std::cerr << "MidiInputPort: ERROR, no sampler instance to handle program change."
284                          << "This is a bug, please report it!\n" << std::flush;
285                return;
286            }
287            std::vector<int> maps = MidiInstrumentMapper::Maps();
288            if (maps.empty()) return;
289    
290            const MidiChannelMap_t& midiChannelMap = MidiChannelMapReader.Lock();
291            // dispatch event for engines listening to the same MIDI channel
292            {
293                std::set<EngineChannel*>::iterator engineiter = midiChannelMap[MidiChannel].begin();
294                std::set<EngineChannel*>::iterator end        = midiChannelMap[MidiChannel].end();
295                for (; engineiter != end; engineiter++) {
296                    (*engineiter)->SetMidiProgram(Program);
297                    if ((*engineiter)->UsesNoMidiInstrumentMap()) continue;
298                    if (MidiInstrumentMapper::GetMapCount() == 0) continue;
299                    // retrieve the MIDI instrument map this engine channel is assigned to
300                    int iMapID = ((*engineiter)->UsesDefaultMidiInstrumentMap())
301                        ? MidiInstrumentMapper::GetDefaultMap() /*default*/ : (*engineiter)->GetMidiInstrumentMap();
302                    // is there an entry for this MIDI bank&prog pair in that map?
303                    midi_prog_index_t midiIndex;
304                    midiIndex.midi_bank_msb = (*engineiter)->GetMidiBankMsb();
305                    midiIndex.midi_bank_lsb = (*engineiter)->GetMidiBankLsb();
306                    midiIndex.midi_prog     = (*engineiter)->GetMidiProgram();
307                    optional<MidiInstrumentMapper::entry_t> mapping =
308                        MidiInstrumentMapper::GetEntry(iMapID, midiIndex);
309                    if (mapping) { // if mapping exists ...
310                        InstrumentManager::instrument_id_t id;
311                        id.FileName = mapping->InstrumentFile;
312                        id.Index    = mapping->InstrumentIndex;
313                        //TODO: we should switch the engine type here
314                        InstrumentManager::LoadInstrumentInBackground(id, *engineiter);
315                        (*engineiter)->Volume(mapping->Volume);
316                    }
317                }
318            }
319            // dispatch event for engines listening to ALL MIDI channels
320            {
321                std::set<EngineChannel*>::iterator engineiter = midiChannelMap[midi_chan_all].begin();
322                std::set<EngineChannel*>::iterator end        = midiChannelMap[midi_chan_all].end();
323                for (; engineiter != end; engineiter++) {
324                    (*engineiter)->SetMidiProgram(Program);
325                    if ((*engineiter)->UsesNoMidiInstrumentMap()) continue;
326                    if (MidiInstrumentMapper::GetMapCount() == 0) continue;
327                    // retrieve the MIDI instrument map this engine channel is assigned to
328                    int iMapID = ((*engineiter)->UsesDefaultMidiInstrumentMap())
329                        ? MidiInstrumentMapper::GetDefaultMap() /*default*/ : (*engineiter)->GetMidiInstrumentMap();
330                    // is there an entry for this MIDI bank&prog pair in that map?
331                    midi_prog_index_t midiIndex;
332                    midiIndex.midi_bank_msb = (*engineiter)->GetMidiBankMsb();
333                    midiIndex.midi_bank_lsb = (*engineiter)->GetMidiBankLsb();
334                    midiIndex.midi_prog     = (*engineiter)->GetMidiProgram();
335                    optional<MidiInstrumentMapper::entry_t> mapping =
336                        MidiInstrumentMapper::GetEntry(iMapID, midiIndex);
337                    if (mapping) { // if mapping exists ...
338                        InstrumentManager::instrument_id_t id;
339                        id.FileName = mapping->InstrumentFile;
340                        id.Index    = mapping->InstrumentIndex;
341                        //TODO: we should switch the engine type here
342                        InstrumentManager::LoadInstrumentInBackground(id, *engineiter);
343                        (*engineiter)->Volume(mapping->Volume);
344                    }
345                }
346            }
347            MidiChannelMapReader.Unlock();
348        }
349    
350        void MidiInputPort::DispatchBankSelectMsb(uint8_t BankMSB, uint MidiChannel) {
351            if (BankMSB > 127 || MidiChannel > 16) return;
352            if (!pDevice || !pDevice->pSampler) {
353                std::cerr << "MidiInputPort: ERROR, no sampler instance to handle bank select MSB."
354                          << "This is a bug, please report it!\n" << std::flush;
355                return;
356            }
357            const MidiChannelMap_t& midiChannelMap = MidiChannelMapReader.Lock();
358            // dispatch event for engines listening to the same MIDI channel
359            {
360                std::set<EngineChannel*>::iterator engineiter = midiChannelMap[MidiChannel].begin();
361                std::set<EngineChannel*>::iterator end        = midiChannelMap[MidiChannel].end();
362                // according to the MIDI specs, a bank select should not alter the patch
363                for (; engineiter != end; engineiter++) (*engineiter)->SetMidiBankMsb(BankMSB);
364            }
365            // dispatch event for engines listening to ALL MIDI channels
366            {
367                std::set<EngineChannel*>::iterator engineiter = midiChannelMap[midi_chan_all].begin();
368                std::set<EngineChannel*>::iterator end        = midiChannelMap[midi_chan_all].end();
369                // according to the MIDI specs, a bank select should not alter the patch
370                for (; engineiter != end; engineiter++) (*engineiter)->SetMidiBankMsb(BankMSB);
371            }
372            MidiChannelMapReader.Unlock();
373        }
374    
375        void MidiInputPort::DispatchBankSelectLsb(uint8_t BankLSB, uint MidiChannel) {
376            if (BankLSB > 127 || MidiChannel > 16) return;
377            if (!pDevice || !pDevice->pSampler) {
378                std::cerr << "MidiInputPort: ERROR, no sampler instance to handle bank select LSB."
379                          << "This is a bug, please report it!\n" << std::flush;
380                return;
381            }
382            const MidiChannelMap_t& midiChannelMap = MidiChannelMapReader.Lock();
383            // dispatch event for engines listening to the same MIDI channel
384            {
385                std::set<EngineChannel*>::iterator engineiter = midiChannelMap[MidiChannel].begin();
386                std::set<EngineChannel*>::iterator end        = midiChannelMap[MidiChannel].end();
387                // according to the MIDI specs, a bank select should not alter the patch
388                for (; engineiter != end; engineiter++) (*engineiter)->SetMidiBankLsb(BankLSB);
389            }
390            // dispatch event for engines listening to ALL MIDI channels
391            {
392                std::set<EngineChannel*>::iterator engineiter = midiChannelMap[midi_chan_all].begin();
393                std::set<EngineChannel*>::iterator end        = midiChannelMap[midi_chan_all].end();
394                // according to the MIDI specs, a bank select should not alter the patch
395                for (; engineiter != end; engineiter++) (*engineiter)->SetMidiBankLsb(BankLSB);
396            }
397            MidiChannelMapReader.Unlock();
398        }
399    
400        void MidiInputPort::DispatchRaw(uint8_t* pData) {
401            uint8_t channel = pData[0] & 0x0f;
402            switch (pData[0] & 0xf0) {
403            case 0x80:
404                DispatchNoteOff(pData[1], pData[2], channel);
405                break;
406            case 0x90:
407                if (pData[2]) {
408                    DispatchNoteOn(pData[1], pData[2], channel);
409                } else {
410                    DispatchNoteOff(pData[1], pData[2], channel);
411                }
412                break;
413            case 0xb0:
414                if (pData[1] == 0) {
415                    DispatchBankSelectMsb(pData[2], channel);
416                } else if (pData[1] == 32) {
417                    DispatchBankSelectLsb(pData[2], channel);
418                }
419                DispatchControlChange(pData[1], pData[2], channel);
420                break;
421            case 0xc0:
422                DispatchProgramChange(pData[1], channel);
423                break;
424            case 0xd0:
425                DispatchControlChange(128, pData[1], channel);
426                break;
427            case 0xe0:
428                DispatchPitchbend((pData[1] | pData[2] << 7) - 8192, channel);
429                break;
430          }          }
431      }      }
432    
433      void MidiInputPort::Connect(Engine* pEngine, midi_chan_t MidiChannel) {      void MidiInputPort::DispatchRaw(uint8_t* pData, int32_t FragmentPos) {
434            uint8_t channel = pData[0] & 0x0f;
435            switch (pData[0] & 0xf0) {
436            case 0x80:
437                DispatchNoteOff(pData[1], pData[2], channel, FragmentPos);
438                break;
439            case 0x90:
440                if (pData[2]) {
441                    DispatchNoteOn(pData[1], pData[2], channel, FragmentPos);
442                } else {
443                    DispatchNoteOff(pData[1], pData[2], channel, FragmentPos);
444                }
445                break;
446            case 0xb0:
447                if (pData[1] == 0) {
448                    DispatchBankSelectMsb(pData[2], channel);
449                } else if (pData[1] == 32) {
450                    DispatchBankSelectLsb(pData[2], channel);
451                }
452                DispatchControlChange(pData[1], pData[2], channel, FragmentPos);
453                break;
454            case 0xc0:
455                DispatchProgramChange(pData[1], channel);
456                break;
457            case 0xd0:
458                DispatchControlChange(128, pData[1], channel, FragmentPos);
459                break;
460            case 0xe0:
461                DispatchPitchbend((pData[1] | pData[2] << 7) - 8192, channel, FragmentPos);
462                break;
463            }
464        }
465    
466        void MidiInputPort::Connect(EngineChannel* pEngineChannel, midi_chan_t MidiChannel) {
467          if (MidiChannel < 0 || MidiChannel > 16)          if (MidiChannel < 0 || MidiChannel > 16)
468              throw MidiInputException("MIDI channel index out of bounds");              throw MidiInputException("MIDI channel index out of bounds");
         Disconnect(pEngine);  
         MidiChannelMap[MidiChannel].insert(pEngine);  
     }  
469    
470      void MidiInputPort::Disconnect(Engine* pEngine) {          // first check if desired connection is already established
471          try { for (int i = 0; i <= 16; i++) MidiChannelMap[i].erase(pEngine); }          MidiChannelMapMutex.Lock();
472            MidiChannelMap_t& midiChannelMap = MidiChannelMap.GetConfigForUpdate();
473            bool bAlreadyDone = midiChannelMap[MidiChannel].count(pEngineChannel);
474            MidiChannelMapMutex.Unlock();
475            if (bAlreadyDone) return;
476    
477            // remove all other connections of that engine channel (if any)
478            Disconnect(pEngineChannel);
479    
480            // register engine channel on the desired MIDI channel
481            MidiChannelMapMutex.Lock();
482            MidiChannelMap.GetConfigForUpdate()[MidiChannel].insert(pEngineChannel);
483            MidiChannelMap.SwitchConfig()[MidiChannel].insert(pEngineChannel);
484            MidiChannelMapMutex.Unlock();
485    
486            // inform engine channel about this connection
487            pEngineChannel->Connect(this, MidiChannel);
488    
489            // mark engine channel as changed
490            pEngineChannel->StatusChanged(true);
491        }
492    
493        void MidiInputPort::Disconnect(EngineChannel* pEngineChannel) {
494            if (!pEngineChannel) return;
495    
496            bool bChannelFound = false;
497    
498            // unregister engine channel from all MIDI channels
499            MidiChannelMapMutex.Lock();
500            try {
501                {
502                    MidiChannelMap_t& midiChannelMap = MidiChannelMap.GetConfigForUpdate();
503                    for (int i = 0; i <= 16; i++) {
504                        bChannelFound |= midiChannelMap[i].count(pEngineChannel);
505                        midiChannelMap[i].erase(pEngineChannel);
506                    }
507                }
508                // do the same update again, after switching to the other config
509                {
510                    MidiChannelMap_t& midiChannelMap = MidiChannelMap.SwitchConfig();
511                    for (int i = 0; i <= 16; i++) {
512                        bChannelFound |= midiChannelMap[i].count(pEngineChannel);
513                        midiChannelMap[i].erase(pEngineChannel);
514                    }
515                }
516            }
517          catch(...) { /* NOOP */ }          catch(...) { /* NOOP */ }
518            MidiChannelMapMutex.Unlock();
519    
520            // inform engine channel about the disconnection (if there is one)
521            if (bChannelFound) pEngineChannel->DisconnectMidiInputPort();
522    
523            // mark engine channel as changed
524            pEngineChannel->StatusChanged(true);
525        }
526    
527        SynchronizedConfig<std::set<LinuxSampler::Engine*> > MidiInputPort::SysexListeners;
528    
529        void MidiInputPort::AddSysexListener(Engine* engine) {
530            std::pair<std::set<Engine*>::iterator, bool> p = SysexListeners.GetConfigForUpdate().insert(engine);
531            if (p.second) SysexListeners.SwitchConfig().insert(engine);
532        }
533    
534        bool MidiInputPort::RemoveSysexListener(Engine* engine) {
535            int count = SysexListeners.GetConfigForUpdate().erase(engine);
536            if (count) SysexListeners.SwitchConfig().erase(engine);
537            return count;
538        }
539    
540        void MidiInputPort::Connect(VirtualMidiDevice* pDevice) {
541            virtualMidiDevicesMutex.Lock();
542            // double buffer ... double work ...
543            {
544                std::vector<VirtualMidiDevice*>& devices =
545                    virtualMidiDevices.GetConfigForUpdate();
546                devices.push_back(pDevice);
547            }
548            {
549                std::vector<VirtualMidiDevice*>& devices =
550                    virtualMidiDevices.SwitchConfig();
551                devices.push_back(pDevice);
552            }
553            virtualMidiDevicesMutex.Unlock();
554        }
555    
556        void MidiInputPort::Disconnect(VirtualMidiDevice* pDevice) {
557            virtualMidiDevicesMutex.Lock();
558            // double buffer ... double work ...
559            {
560                std::vector<VirtualMidiDevice*>& devices =
561                    virtualMidiDevices.GetConfigForUpdate();
562                devices.erase(std::find(devices.begin(), devices.end(), pDevice));
563            }
564            {
565                std::vector<VirtualMidiDevice*>& devices =
566                    virtualMidiDevices.SwitchConfig();
567                devices.erase(std::find(devices.begin(), devices.end(), pDevice));
568            }
569            virtualMidiDevicesMutex.Unlock();
570      }      }
571    
572  } // namespace LinuxSampler  } // namespace LinuxSampler

Legend:
Removed from v.244  
changed lines
  Added in v.1887

  ViewVC Help
Powered by ViewVC