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

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

Parent Directory Parent Directory | Revision Log Revision Log


Revision 2127 - (hide annotations) (download)
Wed Sep 22 18:59:16 2010 UTC (13 years, 7 months ago) by persson
File size: 6950 byte(s)
* fixed crash when deleting a sampler channel or changing engine type
  while an instrument load was in progress

1 schoenebeck 411 /***************************************************************************
2     * *
3     * LinuxSampler - modular, streaming capable sampler *
4     * *
5 iliev 2012 * Copyright (C) 2003,2004 by Benno Senoner and Christian Schoenebeck *
6 persson 2114 * Copyright (C) 2005-2008 Christian Schoenebeck *
7     * Copyright (C) 2009-2010 Christian Schoenebeck and Grigor Iliev *
8 schoenebeck 411 * *
9     * This program is free software; you can redistribute it and/or modify *
10     * it under the terms of the GNU General Public License as published by *
11     * the Free Software Foundation; either version 2 of the License, or *
12     * (at your option) any later version. *
13     * *
14     * This program is distributed in the hope that it will be useful, *
15     * but WITHOUT ANY WARRANTY; without even the implied warranty of *
16     * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the *
17     * GNU General Public License for more details. *
18     * *
19     * You should have received a copy of the GNU General Public License *
20     * along with this program; if not, write to the Free Software *
21     * Foundation, Inc., 59 Temple Place, Suite 330, Boston, *
22     * MA 02111-1307 USA *
23     ***************************************************************************/
24    
25     #include "EngineChannel.h"
26 iliev 2012 #include "Engine.h"
27 schoenebeck 411
28 persson 438 namespace LinuxSampler { namespace gig {
29 iliev 2012 EngineChannel::EngineChannel() {
30    
31 schoenebeck 411 }
32    
33     EngineChannel::~EngineChannel() {
34 schoenebeck 460 DisconnectAudioOutputDevice();
35 iliev 1826 // In case the channel was removed before the instrument was
36     // fully loaded, try to give back instrument again (see bug #113)
37 iliev 2012 InstrumentChangeCmd< ::gig::DimensionRegion, ::gig::Instrument>& cmd = ChangeInstrument(NULL);
38 iliev 1826 if (cmd.pInstrument) {
39 persson 2127 Engine::instruments.HandBack(cmd.pInstrument, this);
40 iliev 1826 }
41     ///////
42 schoenebeck 411 }
43    
44 iliev 2012 AbstractEngine::Format EngineChannel::GetEngineFormat() { return AbstractEngine::GIG; }
45 schoenebeck 660
46 iliev 2012 /** This method is not thread safe! */
47 schoenebeck 411 void EngineChannel::ResetInternal() {
48     CurrentKeyDimension = 0;
49 iliev 2012 EngineChannelBase<Voice, ::gig::DimensionRegion, ::gig::Instrument>::ResetInternal();
50 schoenebeck 411 }
51    
52     /**
53 iliev 2012 * Will be called by the MIDIIn Thread to signal that a program
54     * change should be performed. As a program change isn't
55     * real-time safe, the actual change is performed by the disk
56     * thread.
57 schoenebeck 411 *
58 iliev 2012 * @param Program - MIDI program change number
59 schoenebeck 411 */
60 iliev 2012 void EngineChannel::SendProgramChange(uint8_t Program) {
61     Engine* engine = dynamic_cast<Engine*>(pEngine);
62     if(engine == NULL) return;
63    
64     if(engine->GetDiskThread()) {
65     engine->GetDiskThread()->OrderProgramChange(Program, this);
66     } else {
67     // TODO:
68     }
69 schoenebeck 411 }
70    
71     /**
72     * Load an instrument from a .gig file. PrepareLoadInstrument() has to
73     * be called first to provide the information which instrument to load.
74     * This method will then actually start to load the instrument and block
75     * the calling thread until loading was completed.
76     *
77     * @see PrepareLoadInstrument()
78     */
79     void EngineChannel::LoadInstrument() {
80 iliev 2012 InstrumentResourceManager* pInstrumentManager = dynamic_cast<InstrumentResourceManager*>(pEngine->GetInstrumentManager());
81    
82 persson 1646 // make sure we don't trigger any new notes with an old
83     // instrument
84 iliev 2012 InstrumentChangeCmd< ::gig::DimensionRegion, ::gig::Instrument>& cmd = ChangeInstrument(0);
85 persson 1646 if (cmd.pInstrument) {
86     // give old instrument back to instrument manager, but
87     // keep the dimension regions and samples that are in use
88 iliev 2012 pInstrumentManager->HandBackInstrument(cmd.pInstrument, this, cmd.pRegionsInUse);
89 schoenebeck 411 }
90 iliev 2012 cmd.pRegionsInUse->clear();
91 schoenebeck 411
92     // delete all key groups
93 persson 2114 DeleteGroupEventLists();
94 schoenebeck 411
95     // request gig instrument from instrument manager
96 persson 1038 ::gig::Instrument* newInstrument;
97 schoenebeck 411 try {
98 schoenebeck 947 InstrumentManager::instrument_id_t instrid;
99     instrid.FileName = InstrumentFile;
100     instrid.Index = InstrumentIdx;
101 iliev 2012
102     newInstrument = pInstrumentManager->Borrow(instrid, this);
103 persson 1038 if (!newInstrument) {
104 schoenebeck 1212 throw InstrumentManagerException("resource was not created");
105 schoenebeck 411 }
106     }
107     catch (RIFF::Exception e) {
108     InstrumentStat = -2;
109 iliev 1309 StatusChanged(true);
110 schoenebeck 411 String msg = "gig::Engine error: Failed to load instrument, cause: " + e.Message;
111 schoenebeck 880 throw Exception(msg);
112 schoenebeck 411 }
113 schoenebeck 1212 catch (InstrumentManagerException e) {
114 schoenebeck 411 InstrumentStat = -3;
115 iliev 1309 StatusChanged(true);
116 schoenebeck 411 String msg = "gig::Engine error: Failed to load instrument, cause: " + e.Message();
117 schoenebeck 880 throw Exception(msg);
118 schoenebeck 411 }
119     catch (...) {
120     InstrumentStat = -4;
121 iliev 1309 StatusChanged(true);
122 schoenebeck 880 throw Exception("gig::Engine error: Failed to load instrument, cause: Unknown exception while trying to parse gig file.");
123 schoenebeck 411 }
124    
125 persson 2043 RoundRobinIndex = 0;
126     for (int i = 0 ; i < 128 ; i++) pMIDIKeyInfo[i].pRoundRobinIndex = NULL;
127    
128     // rebuild ActiveKeyGroups map with key groups of current
129     // instrument and set the round robin pointers to use one
130     // counter for each region
131     int region = 0;
132     for (::gig::Region* pRegion = newInstrument->GetFirstRegion(); pRegion; pRegion = newInstrument->GetNextRegion()) {
133 persson 2114 AddGroup(pRegion->KeyGroup);
134 schoenebeck 411
135 persson 2043 RoundRobinIndexes[region] = 0;
136     for (int iKey = pRegion->KeyRange.low; iKey <= pRegion->KeyRange.high; iKey++) {
137     pMIDIKeyInfo[iKey].pRoundRobinIndex = &RoundRobinIndexes[region];
138     }
139     region++;
140     }
141    
142 persson 1038 InstrumentIdxName = newInstrument->pInfo->Name;
143 schoenebeck 411 InstrumentStat = 100;
144    
145 persson 1646 ChangeInstrument(newInstrument);
146    
147 iliev 1298 StatusChanged(true);
148 schoenebeck 411 }
149    
150 iliev 2012 void EngineChannel::ProcessKeySwitchChange(int key) {
151     // Change key dimension value if key is in keyswitching area
152 persson 1646 {
153 iliev 2012 if (key >= pInstrument->DimensionKeyRange.low && key <= pInstrument->DimensionKeyRange.high)
154     CurrentKeyDimension = float(key - pInstrument->DimensionKeyRange.low) /
155     (pInstrument->DimensionKeyRange.high - pInstrument->DimensionKeyRange.low + 1);
156 persson 1646 }
157 schoenebeck 411 }
158    
159     }} // namespace LinuxSampler::gig

  ViewVC Help
Powered by ViewVC