/[svn]/linuxsampler/trunk/src/network/lscpserver.cpp
ViewVC logotype

Diff of /linuxsampler/trunk/src/network/lscpserver.cpp

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

revision 793 by iliev, Wed Oct 26 09:34:38 2005 UTC revision 2375 by schoenebeck, Thu Oct 4 17:45:22 2012 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 Christian Schoenebeck                              *   *   Copyright (C) 2005 - 2012 Christian Schoenebeck                       *
7   *                                                                         *   *                                                                         *
8   *   This library is free software; you can redistribute it and/or modify  *   *   This library 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 21  Line 21 
21   *   MA  02111-1307  USA                                                   *   *   MA  02111-1307  USA                                                   *
22   ***************************************************************************/   ***************************************************************************/
23    
24    #include <algorithm>
25    #include <string>
26    
27    #include "../common/File.h"
28  #include "lscpserver.h"  #include "lscpserver.h"
29  #include "lscpresultset.h"  #include "lscpresultset.h"
30  #include "lscpevent.h"  #include "lscpevent.h"
 //#include "../common/global.h"  
31    
32    #if defined(WIN32)
33    #include <windows.h>
34    #else
35  #include <fcntl.h>  #include <fcntl.h>
36    #endif
37    
38  #if HAVE_SQLITE3  #if ! HAVE_SQLITE3
39  # include "sqlite3.h"  #define DOESNT_HAVE_SQLITE3 "No database support. SQLITE3 was not installed when linuxsampler was built."
40  #endif  #endif
41    
42  #include "../engines/EngineFactory.h"  #include "../engines/EngineFactory.h"
43  #include "../engines/EngineChannelFactory.h"  #include "../engines/EngineChannelFactory.h"
44  #include "../drivers/audio/AudioOutputDeviceFactory.h"  #include "../drivers/audio/AudioOutputDeviceFactory.h"
45  #include "../drivers/midi/MidiInputDeviceFactory.h"  #include "../drivers/midi/MidiInputDeviceFactory.h"
46    #include "../effects/EffectFactory.h"
47    
48    namespace LinuxSampler {
49    
50    /**
51     * Returns a copy of the given string where all special characters are
52     * replaced by LSCP escape sequences ("\xHH"). This function shall be used
53     * to escape LSCP response fields in case the respective response field is
54     * actually defined as using escape sequences in the LSCP specs.
55     *
56     * @e Caution: DO NOT use this function for escaping path based responses,
57     * use the Path class (src/common/Path.h) for this instead!
58     */
59    static String _escapeLscpResponse(String txt) {
60        for (int i = 0; i < txt.length(); i++) {
61            const char c = txt.c_str()[i];
62            if (
63                !(c >= '0' && c <= '9') &&
64                !(c >= 'a' && c <= 'z') &&
65                !(c >= 'A' && c <= 'Z') &&
66                !(c == ' ') && !(c == '!') && !(c == '#') && !(c == '$') &&
67                !(c == '%') && !(c == '&') && !(c == '(') && !(c == ')') &&
68                !(c == '*') && !(c == '+') && !(c == ',') && !(c == '-') &&
69                !(c == '.') && !(c == '/') && !(c == ':') && !(c == ';') &&
70                !(c == '<') && !(c == '=') && !(c == '>') && !(c == '?') &&
71                !(c == '@') && !(c == '[') && !(c == ']') &&
72                !(c == '^') && !(c == '_') && !(c == '`') && !(c == '{') &&
73                !(c == '|') && !(c == '}') && !(c == '~')
74            ) {
75                // convert the "special" character into a "\xHH" LSCP escape sequence
76                char buf[5];
77                snprintf(buf, sizeof(buf), "\\x%02x", static_cast<unsigned char>(c));
78                txt.replace(i, 1, buf);
79                i += 3;
80            }
81        }
82        return txt;
83    }
84    
85  /**  /**
86   * Below are a few static members of the LSCPServer class.   * Below are a few static members of the LSCPServer class.
# Line 53  Line 98 
98  fd_set LSCPServer::fdSet;  fd_set LSCPServer::fdSet;
99  int LSCPServer::currentSocket = -1;  int LSCPServer::currentSocket = -1;
100  std::vector<yyparse_param_t> LSCPServer::Sessions = std::vector<yyparse_param_t>();  std::vector<yyparse_param_t> LSCPServer::Sessions = std::vector<yyparse_param_t>();
101    std::vector<yyparse_param_t>::iterator itCurrentSession = std::vector<yyparse_param_t>::iterator();
102  std::map<int,String> LSCPServer::bufferedNotifies = std::map<int,String>();  std::map<int,String> LSCPServer::bufferedNotifies = std::map<int,String>();
103  std::map<int,String> LSCPServer::bufferedCommands = std::map<int,String>();  std::map<int,String> LSCPServer::bufferedCommands = std::map<int,String>();
104  std::map< LSCPEvent::event_t, std::list<int> > LSCPServer::eventSubscriptions = std::map< LSCPEvent::event_t, std::list<int> >();  std::map< LSCPEvent::event_t, std::list<int> > LSCPServer::eventSubscriptions = std::map< LSCPEvent::event_t, std::list<int> >();
# Line 61  Mutex LSCPServer::NotifyBufferMutex = Mu Line 107  Mutex LSCPServer::NotifyBufferMutex = Mu
107  Mutex LSCPServer::SubscriptionMutex = Mutex();  Mutex LSCPServer::SubscriptionMutex = Mutex();
108  Mutex LSCPServer::RTNotifyMutex = Mutex();  Mutex LSCPServer::RTNotifyMutex = Mutex();
109    
110  LSCPServer::LSCPServer(Sampler* pSampler, long int addr, short int port) : Thread(true, false, 0, -4) {  LSCPServer::LSCPServer(Sampler* pSampler, long int addr, short int port) : Thread(true, false, 0, -4), eventHandler(this) {
111      SocketAddress.sin_family      = AF_INET;      SocketAddress.sin_family      = AF_INET;
112      SocketAddress.sin_addr.s_addr = addr;      SocketAddress.sin_addr.s_addr = addr;
113      SocketAddress.sin_port        = port;      SocketAddress.sin_port        = port;
114      this->pSampler = pSampler;      this->pSampler = pSampler;
115        LSCPEvent::RegisterEvent(LSCPEvent::event_audio_device_count, "AUDIO_OUTPUT_DEVICE_COUNT");
116        LSCPEvent::RegisterEvent(LSCPEvent::event_audio_device_info, "AUDIO_OUTPUT_DEVICE_INFO");
117        LSCPEvent::RegisterEvent(LSCPEvent::event_midi_device_count, "MIDI_INPUT_DEVICE_COUNT");
118        LSCPEvent::RegisterEvent(LSCPEvent::event_midi_device_info, "MIDI_INPUT_DEVICE_INFO");
119      LSCPEvent::RegisterEvent(LSCPEvent::event_channel_count, "CHANNEL_COUNT");      LSCPEvent::RegisterEvent(LSCPEvent::event_channel_count, "CHANNEL_COUNT");
120      LSCPEvent::RegisterEvent(LSCPEvent::event_voice_count, "VOICE_COUNT");      LSCPEvent::RegisterEvent(LSCPEvent::event_voice_count, "VOICE_COUNT");
121      LSCPEvent::RegisterEvent(LSCPEvent::event_stream_count, "STREAM_COUNT");      LSCPEvent::RegisterEvent(LSCPEvent::event_stream_count, "STREAM_COUNT");
122      LSCPEvent::RegisterEvent(LSCPEvent::event_buffer_fill, "BUFFER_FILL");      LSCPEvent::RegisterEvent(LSCPEvent::event_buffer_fill, "BUFFER_FILL");
123      LSCPEvent::RegisterEvent(LSCPEvent::event_channel_info, "CHANNEL_INFO");      LSCPEvent::RegisterEvent(LSCPEvent::event_channel_info, "CHANNEL_INFO");
124        LSCPEvent::RegisterEvent(LSCPEvent::event_fx_send_count, "FX_SEND_COUNT");
125        LSCPEvent::RegisterEvent(LSCPEvent::event_fx_send_info, "FX_SEND_INFO");
126        LSCPEvent::RegisterEvent(LSCPEvent::event_midi_instr_map_count, "MIDI_INSTRUMENT_MAP_COUNT");
127        LSCPEvent::RegisterEvent(LSCPEvent::event_midi_instr_map_info, "MIDI_INSTRUMENT_MAP_INFO");
128        LSCPEvent::RegisterEvent(LSCPEvent::event_midi_instr_count, "MIDI_INSTRUMENT_COUNT");
129        LSCPEvent::RegisterEvent(LSCPEvent::event_midi_instr_info, "MIDI_INSTRUMENT_INFO");
130        LSCPEvent::RegisterEvent(LSCPEvent::event_db_instr_dir_count, "DB_INSTRUMENT_DIRECTORY_COUNT");
131        LSCPEvent::RegisterEvent(LSCPEvent::event_db_instr_dir_info, "DB_INSTRUMENT_DIRECTORY_INFO");
132        LSCPEvent::RegisterEvent(LSCPEvent::event_db_instr_count, "DB_INSTRUMENT_COUNT");
133        LSCPEvent::RegisterEvent(LSCPEvent::event_db_instr_info, "DB_INSTRUMENT_INFO");
134        LSCPEvent::RegisterEvent(LSCPEvent::event_db_instrs_job_info, "DB_INSTRUMENTS_JOB_INFO");
135      LSCPEvent::RegisterEvent(LSCPEvent::event_misc, "MISCELLANEOUS");      LSCPEvent::RegisterEvent(LSCPEvent::event_misc, "MISCELLANEOUS");
136        LSCPEvent::RegisterEvent(LSCPEvent::event_total_stream_count, "TOTAL_STREAM_COUNT");
137      LSCPEvent::RegisterEvent(LSCPEvent::event_total_voice_count, "TOTAL_VOICE_COUNT");      LSCPEvent::RegisterEvent(LSCPEvent::event_total_voice_count, "TOTAL_VOICE_COUNT");
138        LSCPEvent::RegisterEvent(LSCPEvent::event_global_info, "GLOBAL_INFO");
139        LSCPEvent::RegisterEvent(LSCPEvent::event_channel_midi, "CHANNEL_MIDI");
140        LSCPEvent::RegisterEvent(LSCPEvent::event_device_midi, "DEVICE_MIDI");
141        LSCPEvent::RegisterEvent(LSCPEvent::event_fx_instance_count, "EFFECT_INSTANCE_COUNT");
142        LSCPEvent::RegisterEvent(LSCPEvent::event_fx_instance_info, "EFFECT_INSTANCE_INFO");
143        LSCPEvent::RegisterEvent(LSCPEvent::event_send_fx_chain_count, "SEND_EFFECT_CHAIN_COUNT");
144        LSCPEvent::RegisterEvent(LSCPEvent::event_send_fx_chain_info, "SEND_EFFECT_CHAIN_INFO");
145      hSocket = -1;      hSocket = -1;
146  }  }
147    
148  LSCPServer::~LSCPServer() {  LSCPServer::~LSCPServer() {
149        CloseAllConnections();
150        InstrumentManager::StopBackgroundThread();
151    #if defined(WIN32)
152        if (hSocket >= 0) closesocket(hSocket);
153    #else
154      if (hSocket >= 0) close(hSocket);      if (hSocket >= 0) close(hSocket);
155    #endif
156    }
157    
158    LSCPServer::EventHandler::EventHandler(LSCPServer* pParent) {
159        this->pParent = pParent;
160    }
161    
162    LSCPServer::EventHandler::~EventHandler() {
163        std::vector<midi_listener_entry> l = channelMidiListeners;
164        channelMidiListeners.clear();
165        for (int i = 0; i < l.size(); i++)
166            delete l[i].pMidiListener;
167    }
168    
169    void LSCPServer::EventHandler::ChannelCountChanged(int NewCount) {
170        LSCPServer::SendLSCPNotify(LSCPEvent(LSCPEvent::event_channel_count, NewCount));
171    }
172    
173    void LSCPServer::EventHandler::ChannelAdded(SamplerChannel* pChannel) {
174        pChannel->AddEngineChangeListener(this);
175    }
176    
177    void LSCPServer::EventHandler::ChannelToBeRemoved(SamplerChannel* pChannel) {
178        if (!pChannel->GetEngineChannel()) return;
179        EngineToBeChanged(pChannel->Index());
180    }
181    
182    void LSCPServer::EventHandler::EngineToBeChanged(int ChannelId) {
183        SamplerChannel* pSamplerChannel =
184            pParent->pSampler->GetSamplerChannel(ChannelId);
185        if (!pSamplerChannel) return;
186        EngineChannel* pEngineChannel =
187            pSamplerChannel->GetEngineChannel();
188        if (!pEngineChannel) return;
189        for (std::vector<midi_listener_entry>::iterator iter = channelMidiListeners.begin(); iter != channelMidiListeners.end(); ++iter) {
190            if ((*iter).pEngineChannel == pEngineChannel) {
191                VirtualMidiDevice* pMidiListener = (*iter).pMidiListener;
192                pEngineChannel->Disconnect(pMidiListener);
193                channelMidiListeners.erase(iter);
194                delete pMidiListener;
195                return;
196            }
197        }
198    }
199    
200    void LSCPServer::EventHandler::EngineChanged(int ChannelId) {
201        SamplerChannel* pSamplerChannel =
202            pParent->pSampler->GetSamplerChannel(ChannelId);
203        if (!pSamplerChannel) return;
204        EngineChannel* pEngineChannel =
205            pSamplerChannel->GetEngineChannel();
206        if (!pEngineChannel) return;
207        VirtualMidiDevice* pMidiListener = new VirtualMidiDevice;
208        pEngineChannel->Connect(pMidiListener);
209        midi_listener_entry entry = {
210            pSamplerChannel, pEngineChannel, pMidiListener
211        };
212        channelMidiListeners.push_back(entry);
213    }
214    
215    void LSCPServer::EventHandler::AudioDeviceCountChanged(int NewCount) {
216        LSCPServer::SendLSCPNotify(LSCPEvent(LSCPEvent::event_audio_device_count, NewCount));
217    }
218    
219    void LSCPServer::EventHandler::MidiDeviceCountChanged(int NewCount) {
220        LSCPServer::SendLSCPNotify(LSCPEvent(LSCPEvent::event_midi_device_count, NewCount));
221    }
222    
223    void LSCPServer::EventHandler::MidiDeviceToBeDestroyed(MidiInputDevice* pDevice) {
224        pDevice->RemoveMidiPortCountListener(this);
225        for (int i = 0; i < pDevice->PortCount(); ++i)
226            MidiPortToBeRemoved(pDevice->GetPort(i));
227    }
228    
229    void LSCPServer::EventHandler::MidiDeviceCreated(MidiInputDevice* pDevice) {
230        pDevice->AddMidiPortCountListener(this);
231        for (int i = 0; i < pDevice->PortCount(); ++i)
232            MidiPortAdded(pDevice->GetPort(i));
233    }
234    
235    void LSCPServer::EventHandler::MidiPortCountChanged(int NewCount) {
236        // yet unused
237    }
238    
239    void LSCPServer::EventHandler::MidiPortToBeRemoved(MidiInputPort* pPort) {
240        for (std::vector<device_midi_listener_entry>::iterator iter = deviceMidiListeners.begin(); iter != deviceMidiListeners.end(); ++iter) {
241            if ((*iter).pPort == pPort) {
242                VirtualMidiDevice* pMidiListener = (*iter).pMidiListener;
243                pPort->Disconnect(pMidiListener);
244                deviceMidiListeners.erase(iter);
245                delete pMidiListener;
246                return;
247            }
248        }
249    }
250    
251    void LSCPServer::EventHandler::MidiPortAdded(MidiInputPort* pPort) {
252        // find out the device ID
253        std::map<uint, MidiInputDevice*> devices =
254            pParent->pSampler->GetMidiInputDevices();
255        for (
256            std::map<uint, MidiInputDevice*>::iterator iter = devices.begin();
257            iter != devices.end(); ++iter
258        ) {
259            if (iter->second == pPort->GetDevice()) { // found
260                VirtualMidiDevice* pMidiListener = new VirtualMidiDevice;
261                pPort->Connect(pMidiListener);
262                device_midi_listener_entry entry = {
263                    pPort, pMidiListener, iter->first
264                };
265                deviceMidiListeners.push_back(entry);
266                return;
267            }
268        }
269    }
270    
271    void LSCPServer::EventHandler::MidiInstrumentCountChanged(int MapId, int NewCount) {
272        LSCPServer::SendLSCPNotify(LSCPEvent(LSCPEvent::event_midi_instr_count, MapId, NewCount));
273    }
274    
275    void LSCPServer::EventHandler::MidiInstrumentInfoChanged(int MapId, int Bank, int Program) {
276        LSCPServer::SendLSCPNotify(LSCPEvent(LSCPEvent::event_midi_instr_info, MapId, Bank, Program));
277    }
278    
279    void LSCPServer::EventHandler::MidiInstrumentMapCountChanged(int NewCount) {
280        LSCPServer::SendLSCPNotify(LSCPEvent(LSCPEvent::event_midi_instr_map_count, NewCount));
281    }
282    
283    void LSCPServer::EventHandler::MidiInstrumentMapInfoChanged(int MapId) {
284        LSCPServer::SendLSCPNotify(LSCPEvent(LSCPEvent::event_midi_instr_map_info, MapId));
285    }
286    
287    void LSCPServer::EventHandler::FxSendCountChanged(int ChannelId, int NewCount) {
288        LSCPServer::SendLSCPNotify(LSCPEvent(LSCPEvent::event_fx_send_count, ChannelId, NewCount));
289    }
290    
291    void LSCPServer::EventHandler::VoiceCountChanged(int ChannelId, int NewCount) {
292        LSCPServer::SendLSCPNotify(LSCPEvent(LSCPEvent::event_voice_count, ChannelId, NewCount));
293    }
294    
295    void LSCPServer::EventHandler::StreamCountChanged(int ChannelId, int NewCount) {
296        LSCPServer::SendLSCPNotify(LSCPEvent(LSCPEvent::event_stream_count, ChannelId, NewCount));
297    }
298    
299    void LSCPServer::EventHandler::BufferFillChanged(int ChannelId, String FillData) {
300        LSCPServer::SendLSCPNotify(LSCPEvent(LSCPEvent::event_buffer_fill, ChannelId, FillData));
301    }
302    
303    void LSCPServer::EventHandler::TotalVoiceCountChanged(int NewCount) {
304        LSCPServer::SendLSCPNotify(LSCPEvent(LSCPEvent::event_total_voice_count, NewCount));
305    }
306    
307    void LSCPServer::EventHandler::TotalStreamCountChanged(int NewCount) {
308        LSCPServer::SendLSCPNotify(LSCPEvent(LSCPEvent::event_total_stream_count, NewCount));
309    }
310    
311    #if HAVE_SQLITE3
312    void LSCPServer::DbInstrumentsEventHandler::DirectoryCountChanged(String Dir) {
313        LSCPServer::SendLSCPNotify(LSCPEvent(LSCPEvent::event_db_instr_dir_count, InstrumentsDb::toEscapedPath(Dir)));
314    }
315    
316    void LSCPServer::DbInstrumentsEventHandler::DirectoryInfoChanged(String Dir) {
317        LSCPServer::SendLSCPNotify(LSCPEvent(LSCPEvent::event_db_instr_dir_info, InstrumentsDb::toEscapedPath(Dir)));
318    }
319    
320    void LSCPServer::DbInstrumentsEventHandler::DirectoryNameChanged(String Dir, String NewName) {
321        Dir = "'" + InstrumentsDb::toEscapedPath(Dir) + "'";
322        NewName = "'" + InstrumentsDb::toEscapedPath(NewName) + "'";
323        LSCPServer::SendLSCPNotify(LSCPEvent(LSCPEvent::event_db_instr_dir_info, "NAME", Dir, NewName));
324    }
325    
326    void LSCPServer::DbInstrumentsEventHandler::InstrumentCountChanged(String Dir) {
327        LSCPServer::SendLSCPNotify(LSCPEvent(LSCPEvent::event_db_instr_count, InstrumentsDb::toEscapedPath(Dir)));
328    }
329    
330    void LSCPServer::DbInstrumentsEventHandler::InstrumentInfoChanged(String Instr) {
331        LSCPServer::SendLSCPNotify(LSCPEvent(LSCPEvent::event_db_instr_info, InstrumentsDb::toEscapedPath(Instr)));
332    }
333    
334    void LSCPServer::DbInstrumentsEventHandler::InstrumentNameChanged(String Instr, String NewName) {
335        Instr = "'" + InstrumentsDb::toEscapedPath(Instr) + "'";
336        NewName = "'" + InstrumentsDb::toEscapedPath(NewName) + "'";
337        LSCPServer::SendLSCPNotify(LSCPEvent(LSCPEvent::event_db_instr_info, "NAME", Instr, NewName));
338    }
339    
340    void LSCPServer::DbInstrumentsEventHandler::JobStatusChanged(int JobId) {
341        LSCPServer::SendLSCPNotify(LSCPEvent(LSCPEvent::event_db_instrs_job_info, JobId));
342    }
343    #endif // HAVE_SQLITE3
344    
345    void LSCPServer::RemoveListeners() {
346        pSampler->RemoveChannelCountListener(&eventHandler);
347        pSampler->RemoveAudioDeviceCountListener(&eventHandler);
348        pSampler->RemoveMidiDeviceCountListener(&eventHandler);
349        pSampler->RemoveVoiceCountListener(&eventHandler);
350        pSampler->RemoveStreamCountListener(&eventHandler);
351        pSampler->RemoveBufferFillListener(&eventHandler);
352        pSampler->RemoveTotalStreamCountListener(&eventHandler);
353        pSampler->RemoveTotalVoiceCountListener(&eventHandler);
354        pSampler->RemoveFxSendCountListener(&eventHandler);
355        MidiInstrumentMapper::RemoveMidiInstrumentCountListener(&eventHandler);
356        MidiInstrumentMapper::RemoveMidiInstrumentInfoListener(&eventHandler);
357        MidiInstrumentMapper::RemoveMidiInstrumentMapCountListener(&eventHandler);
358        MidiInstrumentMapper::RemoveMidiInstrumentMapInfoListener(&eventHandler);
359    #if HAVE_SQLITE3
360        InstrumentsDb::GetInstrumentsDb()->RemoveInstrumentsDbListener(&dbInstrumentsEventHandler);
361    #endif
362  }  }
363    
364  /**  /**
# Line 95  int LSCPServer::WaitUntilInitialized(lon Line 376  int LSCPServer::WaitUntilInitialized(lon
376  }  }
377    
378  int LSCPServer::Main() {  int LSCPServer::Main() {
379            #if defined(WIN32)
380            WSADATA wsaData;
381            int iResult;
382            iResult = WSAStartup(MAKEWORD(2,2), &wsaData);
383            if (iResult != 0) {
384                    std::cerr << "LSCPServer: WSAStartup failed: " << iResult << "\n";
385                    exit(EXIT_FAILURE);
386            }
387            #endif
388      hSocket = socket(AF_INET, SOCK_STREAM, 0);      hSocket = socket(AF_INET, SOCK_STREAM, 0);
389      if (hSocket < 0) {      if (hSocket < 0) {
390          std::cerr << "LSCPServer: Could not create server socket." << std::endl;          std::cerr << "LSCPServer: Could not create server socket." << std::endl;
# Line 108  int LSCPServer::Main() { Line 398  int LSCPServer::Main() {
398              if (bind(hSocket, (sockaddr*) &SocketAddress, sizeof(sockaddr_in)) < 0) {              if (bind(hSocket, (sockaddr*) &SocketAddress, sizeof(sockaddr_in)) < 0) {
399                  if (trial > LSCP_SERVER_BIND_TIMEOUT) {                  if (trial > LSCP_SERVER_BIND_TIMEOUT) {
400                      std::cerr << "gave up!" << std::endl;                      std::cerr << "gave up!" << std::endl;
401                        #if defined(WIN32)
402                        closesocket(hSocket);
403                        #else
404                      close(hSocket);                      close(hSocket);
405                        #endif
406                      //return -1;                      //return -1;
407                      exit(EXIT_FAILURE);                      exit(EXIT_FAILURE);
408                  }                  }
# Line 121  int LSCPServer::Main() { Line 415  int LSCPServer::Main() {
415      listen(hSocket, 1);      listen(hSocket, 1);
416      Initialized.Set(true);      Initialized.Set(true);
417    
418        // Registering event listeners
419        pSampler->AddChannelCountListener(&eventHandler);
420        pSampler->AddAudioDeviceCountListener(&eventHandler);
421        pSampler->AddMidiDeviceCountListener(&eventHandler);
422        pSampler->AddVoiceCountListener(&eventHandler);
423        pSampler->AddStreamCountListener(&eventHandler);
424        pSampler->AddBufferFillListener(&eventHandler);
425        pSampler->AddTotalStreamCountListener(&eventHandler);
426        pSampler->AddTotalVoiceCountListener(&eventHandler);
427        pSampler->AddFxSendCountListener(&eventHandler);
428        MidiInstrumentMapper::AddMidiInstrumentCountListener(&eventHandler);
429        MidiInstrumentMapper::AddMidiInstrumentInfoListener(&eventHandler);
430        MidiInstrumentMapper::AddMidiInstrumentMapCountListener(&eventHandler);
431        MidiInstrumentMapper::AddMidiInstrumentMapInfoListener(&eventHandler);
432    #if HAVE_SQLITE3
433        InstrumentsDb::GetInstrumentsDb()->AddInstrumentsDbListener(&dbInstrumentsEventHandler);
434    #endif
435      // now wait for client connections and handle their requests      // now wait for client connections and handle their requests
436      sockaddr_in client;      sockaddr_in client;
437      int length = sizeof(client);      int length = sizeof(client);
# Line 131  int LSCPServer::Main() { Line 442  int LSCPServer::Main() {
442      timeval timeout;      timeval timeout;
443    
444      while (true) {      while (true) {
445            #if CONFIG_PTHREAD_TESTCANCEL
446                    TestCancel();
447            #endif
448          // check if some engine channel's parameter / status changed, if so notify the respective LSCP event subscribers          // check if some engine channel's parameter / status changed, if so notify the respective LSCP event subscribers
449          {          {
450                EngineChannelFactory::EngineChannelsMutex.Lock();
451              std::set<EngineChannel*> engineChannels = EngineChannelFactory::EngineChannelInstances();              std::set<EngineChannel*> engineChannels = EngineChannelFactory::EngineChannelInstances();
452              std::set<EngineChannel*>::iterator itEngineChannel = engineChannels.begin();              std::set<EngineChannel*>::iterator itEngineChannel = engineChannels.begin();
453              std::set<EngineChannel*>::iterator itEnd           = engineChannels.end();              std::set<EngineChannel*>::iterator itEnd           = engineChannels.end();
454              for (; itEngineChannel != itEnd; ++itEngineChannel) {              for (; itEngineChannel != itEnd; ++itEngineChannel) {
455                  if ((*itEngineChannel)->StatusChanged()) {                  if ((*itEngineChannel)->StatusChanged()) {
456                      SendLSCPNotify(LSCPEvent(LSCPEvent::event_channel_info, (*itEngineChannel)->iSamplerChannelIndex));                      SendLSCPNotify(LSCPEvent(LSCPEvent::event_channel_info, (*itEngineChannel)->GetSamplerChannel()->Index()));
457                    }
458    
459                    for (int i = 0; i < (*itEngineChannel)->GetFxSendCount(); i++) {
460                        FxSend* fxs = (*itEngineChannel)->GetFxSend(i);
461                        if(fxs != NULL && fxs->IsInfoChanged()) {
462                            int chn = (*itEngineChannel)->GetSamplerChannel()->Index();
463                            LSCPServer::SendLSCPNotify(LSCPEvent(LSCPEvent::event_fx_send_info, chn, fxs->Id()));
464                            fxs->SetInfoChanged(false);
465                        }
466                    }
467                }
468                EngineChannelFactory::EngineChannelsMutex.Unlock();
469            }
470    
471            // check if MIDI data arrived on some engine channel
472            for (int i = 0; i < eventHandler.channelMidiListeners.size(); ++i) {
473                const EventHandler::midi_listener_entry entry =
474                    eventHandler.channelMidiListeners[i];
475                VirtualMidiDevice* pMidiListener = entry.pMidiListener;
476                if (pMidiListener->NotesChanged()) {
477                    for (int iNote = 0; iNote < 128; iNote++) {
478                        if (pMidiListener->NoteChanged(iNote)) {
479                            const bool bActive = pMidiListener->NoteIsActive(iNote);
480                            LSCPServer::SendLSCPNotify(
481                                LSCPEvent(
482                                    LSCPEvent::event_channel_midi,
483                                    entry.pSamplerChannel->Index(),
484                                    std::string(bActive ? "NOTE_ON" : "NOTE_OFF"),
485                                    iNote,
486                                    bActive ? pMidiListener->NoteOnVelocity(iNote)
487                                            : pMidiListener->NoteOffVelocity(iNote)
488                                )
489                            );
490                        }
491                    }
492                }
493            }
494    
495            // check if MIDI data arrived on some MIDI device
496            for (int i = 0; i < eventHandler.deviceMidiListeners.size(); ++i) {
497                const EventHandler::device_midi_listener_entry entry =
498                    eventHandler.deviceMidiListeners[i];
499                VirtualMidiDevice* pMidiListener = entry.pMidiListener;
500                if (pMidiListener->NotesChanged()) {
501                    for (int iNote = 0; iNote < 128; iNote++) {
502                        if (pMidiListener->NoteChanged(iNote)) {
503                            const bool bActive = pMidiListener->NoteIsActive(iNote);
504                            LSCPServer::SendLSCPNotify(
505                                LSCPEvent(
506                                    LSCPEvent::event_device_midi,
507                                    entry.uiDeviceID,
508                                    entry.pPort->GetPortNumber(),
509                                    std::string(bActive ? "NOTE_ON" : "NOTE_OFF"),
510                                    iNote,
511                                    bActive ? pMidiListener->NoteOnVelocity(iNote)
512                                            : pMidiListener->NoteOffVelocity(iNote)
513                                )
514                            );
515                        }
516                  }                  }
517              }              }
518          }          }
# Line 146  int LSCPServer::Main() { Line 520  int LSCPServer::Main() {
520          //Now let's deliver late notifies (if any)          //Now let's deliver late notifies (if any)
521          NotifyBufferMutex.Lock();          NotifyBufferMutex.Lock();
522          for (std::map<int,String>::iterator iterNotify = bufferedNotifies.begin(); iterNotify != bufferedNotifies.end(); iterNotify++) {          for (std::map<int,String>::iterator iterNotify = bufferedNotifies.begin(); iterNotify != bufferedNotifies.end(); iterNotify++) {
523    #ifdef MSG_NOSIGNAL
524                  send(iterNotify->first, iterNotify->second.c_str(), iterNotify->second.size(), MSG_NOSIGNAL);                  send(iterNotify->first, iterNotify->second.c_str(), iterNotify->second.size(), MSG_NOSIGNAL);
525                  bufferedNotifies.erase(iterNotify);  #else
526                    send(iterNotify->first, iterNotify->second.c_str(), iterNotify->second.size(), 0);
527    #endif
528          }          }
529            bufferedNotifies.clear();
530          NotifyBufferMutex.Unlock();          NotifyBufferMutex.Unlock();
531    
532          fd_set selectSet = fdSet;          fd_set selectSet = fdSet;
# Line 157  int LSCPServer::Main() { Line 535  int LSCPServer::Main() {
535    
536          int retval = select(maxSessions+1, &selectSet, NULL, NULL, &timeout);          int retval = select(maxSessions+1, &selectSet, NULL, NULL, &timeout);
537    
538          if (retval == 0)          if (retval == 0 || (retval == -1 && errno == EINTR))
539                  continue; //Nothing try again                  continue; //Nothing try again
540          if (retval == -1) {          if (retval == -1) {
541                  std::cerr << "LSCPServer: Socket select error." << std::endl;                  std::cerr << "LSCPServer: Socket select error." << std::endl;
542                    #if defined(WIN32)
543                    closesocket(hSocket);
544                    #else
545                  close(hSocket);                  close(hSocket);
546                    #endif
547                  exit(EXIT_FAILURE);                  exit(EXIT_FAILURE);
548          }          }
549    
# Line 173  int LSCPServer::Main() { Line 555  int LSCPServer::Main() {
555                          exit(EXIT_FAILURE);                          exit(EXIT_FAILURE);
556                  }                  }
557    
558                    #if defined(WIN32)
559                    u_long nonblock_io = 1;
560                    if( ioctlsocket(socket, FIONBIO, &nonblock_io) ) {
561                      std::cerr << "LSCPServer: ioctlsocket: set FIONBIO failed. Error " << WSAGetLastError() << std::endl;
562                      exit(EXIT_FAILURE);
563                    }
564            #else
565                    struct linger linger;
566                    linger.l_onoff = 1;
567                    linger.l_linger = 0;
568                    if(setsockopt(socket, SOL_SOCKET, SO_LINGER, &linger, sizeof(linger))) {
569                        std::cerr << "LSCPServer: Failed to set SO_LINGER\n";
570                    }
571    
572                  if (fcntl(socket, F_SETFL, O_NONBLOCK)) {                  if (fcntl(socket, F_SETFL, O_NONBLOCK)) {
573                          std::cerr << "LSCPServer: F_SETFL O_NONBLOCK failed." << std::endl;                          std::cerr << "LSCPServer: F_SETFL O_NONBLOCK failed." << std::endl;
574                          exit(EXIT_FAILURE);                          exit(EXIT_FAILURE);
575                  }                  }
576                    #endif
577    
578                  // Parser initialization                  // Parser initialization
579                  yyparse_param_t yyparse_param;                  yyparse_param_t yyparse_param;
# Line 200  int LSCPServer::Main() { Line 597  int LSCPServer::Main() {
597                                  int dummy; // just a temporary hack to fulfill the restart() function prototype                                  int dummy; // just a temporary hack to fulfill the restart() function prototype
598                                  restart(NULL, dummy); // restart the 'scanner'                                  restart(NULL, dummy); // restart the 'scanner'
599                                  currentSocket = (*iter).hSession;  //a hack                                  currentSocket = (*iter).hSession;  //a hack
600                                    itCurrentSession = iter; // another hack
601                                  dmsg(2,("LSCPServer: [%s]\n",bufferedCommands[currentSocket].c_str()));                                  dmsg(2,("LSCPServer: [%s]\n",bufferedCommands[currentSocket].c_str()));
602                                  if ((*iter).bVerbose) { // if echo mode enabled                                  if ((*iter).bVerbose) { // if echo mode enabled
603                                      AnswerClient(bufferedCommands[currentSocket]);                                      AnswerClient(bufferedCommands[currentSocket]);
604                                  }                                  }
605                                  int result = yyparse(&(*iter));                                  int result = yyparse(&(*iter));
606                                  currentSocket = -1;     //continuation of a hack                                  currentSocket = -1;     //continuation of a hack
607                                    itCurrentSession = Sessions.end(); // hack as well
608                                  dmsg(3,("LSCPServer: Done parsing on socket %d.\n", currentSocket));                                  dmsg(3,("LSCPServer: Done parsing on socket %d.\n", currentSocket));
609                                  if (result == LSCP_QUIT) { //Was it a quit command by any chance?                                  if (result == LSCP_QUIT) { //Was it a quit command by any chance?
610                                          CloseConnection(iter);                                          CloseConnection(iter);
# Line 233  void LSCPServer::CloseConnection( std::v Line 632  void LSCPServer::CloseConnection( std::v
632          NotifyMutex.Lock();          NotifyMutex.Lock();
633          bufferedCommands.erase(socket);          bufferedCommands.erase(socket);
634          bufferedNotifies.erase(socket);          bufferedNotifies.erase(socket);
635            #if defined(WIN32)
636            closesocket(socket);
637            #else
638          close(socket);          close(socket);
639            #endif
640          NotifyMutex.Unlock();          NotifyMutex.Unlock();
641  }  }
642    
643    void LSCPServer::CloseAllConnections() {
644        std::vector<yyparse_param_t>::iterator iter = Sessions.begin();
645        while(iter != Sessions.end()) {
646            CloseConnection(iter);
647            iter = Sessions.begin();
648        }
649    }
650    
651    void LSCPServer::LockRTNotify() {
652        RTNotifyMutex.Lock();
653    }
654    
655    void LSCPServer::UnlockRTNotify() {
656        RTNotifyMutex.Unlock();
657    }
658    
659  int LSCPServer::EventSubscribers( std::list<LSCPEvent::event_t> events ) {  int LSCPServer::EventSubscribers( std::list<LSCPEvent::event_t> events ) {
660          int subs = 0;          int subs = 0;
661          SubscriptionMutex.Lock();          SubscriptionMutex.Lock();
# Line 262  void LSCPServer::SendLSCPNotify( LSCPEve Line 681  void LSCPServer::SendLSCPNotify( LSCPEve
681          while (true) {          while (true) {
682                  if (NotifyMutex.Trylock()) {                  if (NotifyMutex.Trylock()) {
683                          for(;iter != end; iter++)                          for(;iter != end; iter++)
684    #ifdef MSG_NOSIGNAL
685                                  send(*iter, notify.c_str(), notify.size(), MSG_NOSIGNAL);                                  send(*iter, notify.c_str(), notify.size(), MSG_NOSIGNAL);
686    #else
687                                    send(*iter, notify.c_str(), notify.size(), 0);
688    #endif
689                          NotifyMutex.Unlock();                          NotifyMutex.Unlock();
690                          break;                          break;
691                  } else {                  } else {
# Line 294  extern int GetLSCPCommand( void *buf, in Line 717  extern int GetLSCPCommand( void *buf, in
717          return command.size();          return command.size();
718  }  }
719    
720    extern yyparse_param_t* GetCurrentYaccSession() {
721        return &(*itCurrentSession);
722    }
723    
724  /**  /**
725   * Will be called to try to read the command from the socket   * Will be called to try to read the command from the socket
726   * If command is read, it will return true. Otherwise false is returned.   * If command is read, it will return true. Otherwise false is returned.
# Line 304  bool LSCPServer::GetLSCPCommand( std::ve Line 731  bool LSCPServer::GetLSCPCommand( std::ve
731          char c;          char c;
732          int i = 0;          int i = 0;
733          while (true) {          while (true) {
734                    #if defined(WIN32)
735                    int result = recv(socket, (char *)&c, 1, 0); //Read one character at a time for now
736                    #else
737                  int result = recv(socket, (void *)&c, 1, 0); //Read one character at a time for now                  int result = recv(socket, (void *)&c, 1, 0); //Read one character at a time for now
738                    #endif
739                  if (result == 0) { //socket was selected, so 0 here means client has closed the connection                  if (result == 0) { //socket was selected, so 0 here means client has closed the connection
740                          CloseConnection(iter);                          CloseConnection(iter);
741                          break;                          break;
# Line 314  bool LSCPServer::GetLSCPCommand( std::ve Line 745  bool LSCPServer::GetLSCPCommand( std::ve
745                                  continue; //Ignore CR                                  continue; //Ignore CR
746                          if (c == '\n') {                          if (c == '\n') {
747                                  LSCPServer::SendLSCPNotify(LSCPEvent(LSCPEvent::event_misc, "Received \'" + bufferedCommands[socket] + "\' on socket", socket));                                  LSCPServer::SendLSCPNotify(LSCPEvent(LSCPEvent::event_misc, "Received \'" + bufferedCommands[socket] + "\' on socket", socket));
748                                  bufferedCommands[socket] += "\n";                                  bufferedCommands[socket] += "\r\n";
749                                  return true; //Complete command was read                                  return true; //Complete command was read
750                          }                          }
751                          bufferedCommands[socket] += c;                          bufferedCommands[socket] += c;
752                  }                  }
753                    #if defined(WIN32)
754                    if (result == SOCKET_ERROR) {
755                        int wsa_lasterror = WSAGetLastError();
756                            if (wsa_lasterror == WSAEWOULDBLOCK) //Would block, try again later.
757                                    return false;
758                            dmsg(2,("LSCPScanner: Socket error after recv() Error %d.\n", wsa_lasterror));
759                            CloseConnection(iter);
760                            break;
761                    }
762                    #else
763                  if (result == -1) {                  if (result == -1) {
764                          if (errno == EAGAIN) //Would block, try again later.                          if (errno == EAGAIN) //Would block, try again later.
765                                  return false;                                  return false;
# Line 357  bool LSCPServer::GetLSCPCommand( std::ve Line 798  bool LSCPServer::GetLSCPCommand( std::ve
798                          CloseConnection(iter);                          CloseConnection(iter);
799                          break;                          break;
800                  }                  }
801                    #endif
802          }          }
803          return false;          return false;
804  }  }
# Line 371  void LSCPServer::AnswerClient(String Ret Line 813  void LSCPServer::AnswerClient(String Ret
813      dmsg(2,("LSCPServer::AnswerClient(ReturnMessage=%s)", ReturnMessage.c_str()));      dmsg(2,("LSCPServer::AnswerClient(ReturnMessage=%s)", ReturnMessage.c_str()));
814      if (currentSocket != -1) {      if (currentSocket != -1) {
815              NotifyMutex.Lock();              NotifyMutex.Lock();
816    #ifdef MSG_NOSIGNAL
817              send(currentSocket, ReturnMessage.c_str(), ReturnMessage.size(), MSG_NOSIGNAL);              send(currentSocket, ReturnMessage.c_str(), ReturnMessage.size(), MSG_NOSIGNAL);
818    #else
819                send(currentSocket, ReturnMessage.c_str(), ReturnMessage.size(), 0);
820    #endif
821              NotifyMutex.Unlock();              NotifyMutex.Unlock();
822      }      }
823  }  }
# Line 415  String LSCPServer::CreateAudioOutputDevi Line 861  String LSCPServer::CreateAudioOutputDevi
861          AudioOutputDevice* pDevice = pSampler->CreateAudioOutputDevice(Driver, Parameters);          AudioOutputDevice* pDevice = pSampler->CreateAudioOutputDevice(Driver, Parameters);
862          // search for the created device to get its index          // search for the created device to get its index
863          int index = GetAudioOutputDeviceIndex(pDevice);          int index = GetAudioOutputDeviceIndex(pDevice);
864          if (index == -1) throw LinuxSamplerException("Internal error: could not find created audio output device.");          if (index == -1) throw Exception("Internal error: could not find created audio output device.");
865          result = index; // success          result = index; // success
866      }      }
867      catch (LinuxSamplerException e) {      catch (Exception e) {
868          result.Error(e);          result.Error(e);
869      }      }
870      return result.Produce();      return result.Produce();
# Line 431  String LSCPServer::CreateMidiInputDevice Line 877  String LSCPServer::CreateMidiInputDevice
877          MidiInputDevice* pDevice = pSampler->CreateMidiInputDevice(Driver, Parameters);          MidiInputDevice* pDevice = pSampler->CreateMidiInputDevice(Driver, Parameters);
878          // search for the created device to get its index          // search for the created device to get its index
879          int index = GetMidiInputDeviceIndex(pDevice);          int index = GetMidiInputDeviceIndex(pDevice);
880          if (index == -1) throw LinuxSamplerException("Internal error: could not find created midi input device.");          if (index == -1) throw Exception("Internal error: could not find created midi input device.");
881          result = index; // success          result = index; // success
882      }      }
883      catch (LinuxSamplerException e) {      catch (Exception e) {
884          result.Error(e);          result.Error(e);
885      }      }
886      return result.Produce();      return result.Produce();
# Line 445  String LSCPServer::DestroyAudioOutputDev Line 891  String LSCPServer::DestroyAudioOutputDev
891      LSCPResultSet result;      LSCPResultSet result;
892      try {      try {
893          std::map<uint, AudioOutputDevice*> devices = pSampler->GetAudioOutputDevices();          std::map<uint, AudioOutputDevice*> devices = pSampler->GetAudioOutputDevices();
894          if (!devices.count(DeviceIndex)) throw LinuxSamplerException("There is no audio output device with index " + ToString(DeviceIndex) + ".");          if (!devices.count(DeviceIndex)) throw Exception("There is no audio output device with index " + ToString(DeviceIndex) + ".");
895          AudioOutputDevice* pDevice = devices[DeviceIndex];          AudioOutputDevice* pDevice = devices[DeviceIndex];
896          pSampler->DestroyAudioOutputDevice(pDevice);          pSampler->DestroyAudioOutputDevice(pDevice);
897      }      }
898      catch (LinuxSamplerException e) {      catch (Exception e) {
899          result.Error(e);          result.Error(e);
900      }      }
901      return result.Produce();      return result.Produce();
# Line 460  String LSCPServer::DestroyMidiInputDevic Line 906  String LSCPServer::DestroyMidiInputDevic
906      LSCPResultSet result;      LSCPResultSet result;
907      try {      try {
908          std::map<uint, MidiInputDevice*> devices = pSampler->GetMidiInputDevices();          std::map<uint, MidiInputDevice*> devices = pSampler->GetMidiInputDevices();
909          if (!devices.count(DeviceIndex)) throw LinuxSamplerException("There is no audio output device with index " + ToString(DeviceIndex) + ".");          if (!devices.count(DeviceIndex)) throw Exception("There is no audio output device with index " + ToString(DeviceIndex) + ".");
910          MidiInputDevice* pDevice = devices[DeviceIndex];          MidiInputDevice* pDevice = devices[DeviceIndex];
911          pSampler->DestroyMidiInputDevice(pDevice);          pSampler->DestroyMidiInputDevice(pDevice);
912      }      }
913      catch (LinuxSamplerException e) {      catch (Exception e) {
914          result.Error(e);          result.Error(e);
915      }      }
916      return result.Produce();      return result.Produce();
917  }  }
918    
919    EngineChannel* LSCPServer::GetEngineChannel(uint uiSamplerChannel) {
920        SamplerChannel* pSamplerChannel = pSampler->GetSamplerChannel(uiSamplerChannel);
921        if (!pSamplerChannel) throw Exception("Invalid sampler channel number " + ToString(uiSamplerChannel));
922    
923        EngineChannel* pEngineChannel = pSamplerChannel->GetEngineChannel();
924        if (!pEngineChannel) throw Exception("There is no engine deployed on this sampler channel yet");
925    
926        return pEngineChannel;
927    }
928    
929  /**  /**
930   * Will be called by the parser to load an instrument.   * Will be called by the parser to load an instrument.
931   */   */
# Line 478  String LSCPServer::LoadInstrument(String Line 934  String LSCPServer::LoadInstrument(String
934      LSCPResultSet result;      LSCPResultSet result;
935      try {      try {
936          SamplerChannel* pSamplerChannel = pSampler->GetSamplerChannel(uiSamplerChannel);          SamplerChannel* pSamplerChannel = pSampler->GetSamplerChannel(uiSamplerChannel);
937          if (!pSamplerChannel) throw LinuxSamplerException("Invalid sampler channel number " + ToString(uiSamplerChannel));          if (!pSamplerChannel) throw Exception("Invalid sampler channel number " + ToString(uiSamplerChannel));
938          EngineChannel* pEngineChannel = pSamplerChannel->GetEngineChannel();          EngineChannel* pEngineChannel = pSamplerChannel->GetEngineChannel();
939          if (!pEngineChannel) throw LinuxSamplerException("No engine type assigned to sampler channel yet");          if (!pEngineChannel) throw Exception("No engine type assigned to sampler channel yet");
940          if (!pSamplerChannel->GetAudioOutputDevice())          if (!pSamplerChannel->GetAudioOutputDevice())
941              throw LinuxSamplerException("No audio output device connected to sampler channel");              throw Exception("No audio output device connected to sampler channel");
942          if (bBackground) {          if (bBackground) {
943              InstrumentLoader.StartNewLoad(Filename, uiInstrument, pEngineChannel);              InstrumentManager::instrument_id_t id;
944                id.FileName = Filename;
945                id.Index    = uiInstrument;
946                InstrumentManager::LoadInstrumentInBackground(id, pEngineChannel);
947          }          }
948          else {          else {
949              // tell the engine channel which instrument to load              // tell the engine channel which instrument to load
# Line 493  String LSCPServer::LoadInstrument(String Line 952  String LSCPServer::LoadInstrument(String
952              pEngineChannel->LoadInstrument();              pEngineChannel->LoadInstrument();
953          }          }
954      }      }
955      catch (LinuxSamplerException e) {      catch (Exception e) {
956           result.Error(e);           result.Error(e);
957      }      }
958      return result.Produce();      return result.Produce();
# Line 508  String LSCPServer::SetEngineType(String Line 967  String LSCPServer::SetEngineType(String
967      LSCPResultSet result;      LSCPResultSet result;
968      try {      try {
969          SamplerChannel* pSamplerChannel = pSampler->GetSamplerChannel(uiSamplerChannel);          SamplerChannel* pSamplerChannel = pSampler->GetSamplerChannel(uiSamplerChannel);
970          if (!pSamplerChannel) throw LinuxSamplerException("Invalid sampler channel number " + ToString(uiSamplerChannel));          if (!pSamplerChannel) throw Exception("Invalid sampler channel number " + ToString(uiSamplerChannel));
971          LockRTNotify();          LockRTNotify();
972          pSamplerChannel->SetEngineType(EngineName);          pSamplerChannel->SetEngineType(EngineName);
973          if(HasSoloChannel()) pSamplerChannel->GetEngineChannel()->SetMute(-1);          if(HasSoloChannel()) pSamplerChannel->GetEngineChannel()->SetMute(-1);
974          UnlockRTNotify();          UnlockRTNotify();
975      }      }
976      catch (LinuxSamplerException e) {      catch (Exception e) {
977           result.Error(e);           result.Error(e);
978      }      }
979      return result.Produce();      return result.Produce();
# Line 552  String LSCPServer::ListChannels() { Line 1011  String LSCPServer::ListChannels() {
1011   */   */
1012  String LSCPServer::AddChannel() {  String LSCPServer::AddChannel() {
1013      dmsg(2,("LSCPServer: AddChannel()\n"));      dmsg(2,("LSCPServer: AddChannel()\n"));
1014        LockRTNotify();
1015      SamplerChannel* pSamplerChannel = pSampler->AddSamplerChannel();      SamplerChannel* pSamplerChannel = pSampler->AddSamplerChannel();
1016        UnlockRTNotify();
1017      LSCPResultSet result(pSamplerChannel->Index());      LSCPResultSet result(pSamplerChannel->Index());
1018      return result.Produce();      return result.Produce();
1019  }  }
# Line 574  String LSCPServer::RemoveChannel(uint ui Line 1035  String LSCPServer::RemoveChannel(uint ui
1035   */   */
1036  String LSCPServer::GetAvailableEngines() {  String LSCPServer::GetAvailableEngines() {
1037      dmsg(2,("LSCPServer: GetAvailableEngines()\n"));      dmsg(2,("LSCPServer: GetAvailableEngines()\n"));
1038      LSCPResultSet result("1");      LSCPResultSet result;
1039        try {
1040            int n = EngineFactory::AvailableEngineTypes().size();
1041            result.Add(n);
1042        }
1043        catch (Exception e) {
1044            result.Error(e);
1045        }
1046      return result.Produce();      return result.Produce();
1047  }  }
1048    
# Line 583  String LSCPServer::GetAvailableEngines() Line 1051  String LSCPServer::GetAvailableEngines()
1051   */   */
1052  String LSCPServer::ListAvailableEngines() {  String LSCPServer::ListAvailableEngines() {
1053      dmsg(2,("LSCPServer: ListAvailableEngines()\n"));      dmsg(2,("LSCPServer: ListAvailableEngines()\n"));
1054      LSCPResultSet result("\'GIG\'");      LSCPResultSet result;
1055        try {
1056            String s = EngineFactory::AvailableEngineTypesAsString();
1057            result.Add(s);
1058        }
1059        catch (Exception e) {
1060            result.Error(e);
1061        }
1062      return result.Produce();      return result.Produce();
1063  }  }
1064    
# Line 594  String LSCPServer::ListAvailableEngines( Line 1069  String LSCPServer::ListAvailableEngines(
1069  String LSCPServer::GetEngineInfo(String EngineName) {  String LSCPServer::GetEngineInfo(String EngineName) {
1070      dmsg(2,("LSCPServer: GetEngineInfo(EngineName=%s)\n", EngineName.c_str()));      dmsg(2,("LSCPServer: GetEngineInfo(EngineName=%s)\n", EngineName.c_str()));
1071      LSCPResultSet result;      LSCPResultSet result;
1072        LockRTNotify();
1073      try {      try {
1074          Engine* pEngine = EngineFactory::Create(EngineName);          Engine* pEngine = EngineFactory::Create(EngineName);
1075          result.Add("DESCRIPTION", pEngine->Description());          result.Add("DESCRIPTION", _escapeLscpResponse(pEngine->Description()));
1076          result.Add("VERSION",     pEngine->Version());          result.Add("VERSION",     pEngine->Version());
1077          EngineFactory::Destroy(pEngine);          EngineFactory::Destroy(pEngine);
1078      }      }
1079      catch (LinuxSamplerException e) {      catch (Exception e) {
1080           result.Error(e);           result.Error(e);
1081      }      }
1082        UnlockRTNotify();
1083      return result.Produce();      return result.Produce();
1084  }  }
1085    
# Line 615  String LSCPServer::GetChannelInfo(uint u Line 1092  String LSCPServer::GetChannelInfo(uint u
1092      LSCPResultSet result;      LSCPResultSet result;
1093      try {      try {
1094          SamplerChannel* pSamplerChannel = pSampler->GetSamplerChannel(uiSamplerChannel);          SamplerChannel* pSamplerChannel = pSampler->GetSamplerChannel(uiSamplerChannel);
1095          if (!pSamplerChannel) throw LinuxSamplerException("Invalid sampler channel number " + ToString(uiSamplerChannel));          if (!pSamplerChannel) throw Exception("Invalid sampler channel number " + ToString(uiSamplerChannel));
1096          EngineChannel* pEngineChannel = pSamplerChannel->GetEngineChannel();          EngineChannel* pEngineChannel = pSamplerChannel->GetEngineChannel();
1097    
1098          //Defaults values          //Defaults values
# Line 629  String LSCPServer::GetChannelInfo(uint u Line 1106  String LSCPServer::GetChannelInfo(uint u
1106          String AudioRouting;          String AudioRouting;
1107          int Mute = 0;          int Mute = 0;
1108          bool Solo = false;          bool Solo = false;
1109            String MidiInstrumentMap = "NONE";
1110    
1111          if (pEngineChannel) {          if (pEngineChannel) {
1112              EngineName          = pEngineChannel->EngineName();              EngineName          = pEngineChannel->EngineName();
# Line 646  String LSCPServer::GetChannelInfo(uint u Line 1124  String LSCPServer::GetChannelInfo(uint u
1124              }              }
1125              Mute = pEngineChannel->GetMute();              Mute = pEngineChannel->GetMute();
1126              Solo = pEngineChannel->GetSolo();              Solo = pEngineChannel->GetSolo();
1127                if (pEngineChannel->UsesNoMidiInstrumentMap())
1128                    MidiInstrumentMap = "NONE";
1129                else if (pEngineChannel->UsesDefaultMidiInstrumentMap())
1130                    MidiInstrumentMap = "DEFAULT";
1131                else
1132                    MidiInstrumentMap = ToString(pEngineChannel->GetMidiInstrumentMap());
1133          }          }
1134    
1135          result.Add("ENGINE_NAME", EngineName);          result.Add("ENGINE_NAME", EngineName);
# Line 661  String LSCPServer::GetChannelInfo(uint u Line 1145  String LSCPServer::GetChannelInfo(uint u
1145          if (pSamplerChannel->GetMidiInputChannel() == midi_chan_all) result.Add("MIDI_INPUT_CHANNEL", "ALL");          if (pSamplerChannel->GetMidiInputChannel() == midi_chan_all) result.Add("MIDI_INPUT_CHANNEL", "ALL");
1146          else result.Add("MIDI_INPUT_CHANNEL", pSamplerChannel->GetMidiInputChannel());          else result.Add("MIDI_INPUT_CHANNEL", pSamplerChannel->GetMidiInputChannel());
1147    
1148            // convert the filename into the correct encoding as defined for LSCP
1149            // (especially in terms of special characters -> escape sequences)
1150            if (InstrumentFileName != "NONE" && InstrumentFileName != "") {
1151    #if WIN32
1152                InstrumentFileName = Path::fromWindows(InstrumentFileName).toLscp();
1153    #else
1154                // assuming POSIX
1155                InstrumentFileName = Path::fromPosix(InstrumentFileName).toLscp();
1156    #endif
1157            }
1158    
1159          result.Add("INSTRUMENT_FILE", InstrumentFileName);          result.Add("INSTRUMENT_FILE", InstrumentFileName);
1160          result.Add("INSTRUMENT_NR", InstrumentIndex);          result.Add("INSTRUMENT_NR", InstrumentIndex);
1161          result.Add("INSTRUMENT_NAME", InstrumentName);          result.Add("INSTRUMENT_NAME", _escapeLscpResponse(InstrumentName));
1162          result.Add("INSTRUMENT_STATUS", InstrumentStatus);          result.Add("INSTRUMENT_STATUS", InstrumentStatus);
1163          result.Add("MUTE", Mute == -1 ? "MUTED_BY_SOLO" : (Mute ? "true" : "false"));          result.Add("MUTE", Mute == -1 ? "MUTED_BY_SOLO" : (Mute ? "true" : "false"));
1164          result.Add("SOLO", Solo);          result.Add("SOLO", Solo);
1165            result.Add("MIDI_INSTRUMENT_MAP", MidiInstrumentMap);
1166      }      }
1167      catch (LinuxSamplerException e) {      catch (Exception e) {
1168           result.Error(e);           result.Error(e);
1169      }      }
1170      return result.Produce();      return result.Produce();
# Line 682  String LSCPServer::GetVoiceCount(uint ui Line 1178  String LSCPServer::GetVoiceCount(uint ui
1178      dmsg(2,("LSCPServer: GetVoiceCount(SamplerChannel=%d)\n", uiSamplerChannel));      dmsg(2,("LSCPServer: GetVoiceCount(SamplerChannel=%d)\n", uiSamplerChannel));
1179      LSCPResultSet result;      LSCPResultSet result;
1180      try {      try {
1181          SamplerChannel* pSamplerChannel = pSampler->GetSamplerChannel(uiSamplerChannel);          EngineChannel* pEngineChannel = GetEngineChannel(uiSamplerChannel);
1182          if (!pSamplerChannel) throw LinuxSamplerException("Invalid sampler channel number " + ToString(uiSamplerChannel));          if (!pEngineChannel->GetEngine()) throw Exception("No audio output device connected to sampler channel");
         EngineChannel* pEngineChannel = pSamplerChannel->GetEngineChannel();  
         if (!pEngineChannel) throw LinuxSamplerException("No engine loaded on sampler channel");  
         if (!pEngineChannel->GetEngine()) throw LinuxSamplerException("No audio output device connected to sampler channel");  
1183          result.Add(pEngineChannel->GetEngine()->VoiceCount());          result.Add(pEngineChannel->GetEngine()->VoiceCount());
1184      }      }
1185      catch (LinuxSamplerException e) {      catch (Exception e) {
1186           result.Error(e);           result.Error(e);
1187      }      }
1188      return result.Produce();      return result.Produce();
# Line 703  String LSCPServer::GetStreamCount(uint u Line 1196  String LSCPServer::GetStreamCount(uint u
1196      dmsg(2,("LSCPServer: GetStreamCount(SamplerChannel=%d)\n", uiSamplerChannel));      dmsg(2,("LSCPServer: GetStreamCount(SamplerChannel=%d)\n", uiSamplerChannel));
1197      LSCPResultSet result;      LSCPResultSet result;
1198      try {      try {
1199          SamplerChannel* pSamplerChannel = pSampler->GetSamplerChannel(uiSamplerChannel);          EngineChannel* pEngineChannel = GetEngineChannel(uiSamplerChannel);
1200          if (!pSamplerChannel) throw LinuxSamplerException("Invalid sampler channel number " + ToString(uiSamplerChannel));          if (!pEngineChannel->GetEngine()) throw Exception("No audio output device connected to sampler channel");
         EngineChannel* pEngineChannel = pSamplerChannel->GetEngineChannel();  
         if (!pEngineChannel) throw LinuxSamplerException("No engine type assigned to sampler channel");  
         if (!pEngineChannel->GetEngine()) throw LinuxSamplerException("No audio output device connected to sampler channel");  
1201          result.Add(pEngineChannel->GetEngine()->DiskStreamCount());          result.Add(pEngineChannel->GetEngine()->DiskStreamCount());
1202      }      }
1203      catch (LinuxSamplerException e) {      catch (Exception e) {
1204           result.Error(e);           result.Error(e);
1205      }      }
1206      return result.Produce();      return result.Produce();
# Line 724  String LSCPServer::GetBufferFill(fill_re Line 1214  String LSCPServer::GetBufferFill(fill_re
1214      dmsg(2,("LSCPServer: GetBufferFill(ResponseType=%d, SamplerChannel=%d)\n", ResponseType, uiSamplerChannel));      dmsg(2,("LSCPServer: GetBufferFill(ResponseType=%d, SamplerChannel=%d)\n", ResponseType, uiSamplerChannel));
1215      LSCPResultSet result;      LSCPResultSet result;
1216      try {      try {
1217          SamplerChannel* pSamplerChannel = pSampler->GetSamplerChannel(uiSamplerChannel);          EngineChannel* pEngineChannel = GetEngineChannel(uiSamplerChannel);
1218          if (!pSamplerChannel) throw LinuxSamplerException("Invalid sampler channel number " + ToString(uiSamplerChannel));          if (!pEngineChannel->GetEngine()) throw Exception("No audio output device connected to sampler channel");
         EngineChannel* pEngineChannel = pSamplerChannel->GetEngineChannel();  
         if (!pEngineChannel) throw LinuxSamplerException("No engine type assigned to sampler channel");  
         if (!pEngineChannel->GetEngine()) throw LinuxSamplerException("No audio output device connected to sampler channel");  
1219          if (!pEngineChannel->GetEngine()->DiskStreamSupported()) result.Add("NA");          if (!pEngineChannel->GetEngine()->DiskStreamSupported()) result.Add("NA");
1220          else {          else {
1221              switch (ResponseType) {              switch (ResponseType) {
# Line 739  String LSCPServer::GetBufferFill(fill_re Line 1226  String LSCPServer::GetBufferFill(fill_re
1226                      result.Add(pEngineChannel->GetEngine()->DiskStreamBufferFillPercentage());                      result.Add(pEngineChannel->GetEngine()->DiskStreamBufferFillPercentage());
1227                      break;                      break;
1228                  default:                  default:
1229                      throw LinuxSamplerException("Unknown fill response type");                      throw Exception("Unknown fill response type");
1230              }              }
1231          }          }
1232      }      }
1233      catch (LinuxSamplerException e) {      catch (Exception e) {
1234           result.Error(e);           result.Error(e);
1235      }      }
1236      return result.Produce();      return result.Produce();
# Line 756  String LSCPServer::GetAvailableAudioOutp Line 1243  String LSCPServer::GetAvailableAudioOutp
1243          int n = AudioOutputDeviceFactory::AvailableDrivers().size();          int n = AudioOutputDeviceFactory::AvailableDrivers().size();
1244          result.Add(n);          result.Add(n);
1245      }      }
1246      catch (LinuxSamplerException e) {      catch (Exception e) {
1247          result.Error(e);          result.Error(e);
1248      }      }
1249      return result.Produce();      return result.Produce();
# Line 769  String LSCPServer::ListAvailableAudioOut Line 1256  String LSCPServer::ListAvailableAudioOut
1256          String s = AudioOutputDeviceFactory::AvailableDriversAsString();          String s = AudioOutputDeviceFactory::AvailableDriversAsString();
1257          result.Add(s);          result.Add(s);
1258      }      }
1259      catch (LinuxSamplerException e) {      catch (Exception e) {
1260          result.Error(e);          result.Error(e);
1261      }      }
1262      return result.Produce();      return result.Produce();
# Line 782  String LSCPServer::GetAvailableMidiInput Line 1269  String LSCPServer::GetAvailableMidiInput
1269          int n = MidiInputDeviceFactory::AvailableDrivers().size();          int n = MidiInputDeviceFactory::AvailableDrivers().size();
1270          result.Add(n);          result.Add(n);
1271      }      }
1272      catch (LinuxSamplerException e) {      catch (Exception e) {
1273          result.Error(e);          result.Error(e);
1274      }      }
1275      return result.Produce();      return result.Produce();
# Line 795  String LSCPServer::ListAvailableMidiInpu Line 1282  String LSCPServer::ListAvailableMidiInpu
1282          String s = MidiInputDeviceFactory::AvailableDriversAsString();          String s = MidiInputDeviceFactory::AvailableDriversAsString();
1283          result.Add(s);          result.Add(s);
1284      }      }
1285      catch (LinuxSamplerException e) {      catch (Exception e) {
1286          result.Error(e);          result.Error(e);
1287      }      }
1288      return result.Produce();      return result.Produce();
# Line 815  String LSCPServer::GetMidiInputDriverInf Line 1302  String LSCPServer::GetMidiInputDriverInf
1302              for (;iter != parameters.end(); iter++) {              for (;iter != parameters.end(); iter++) {
1303                  if (s != "") s += ",";                  if (s != "") s += ",";
1304                  s += iter->first;                  s += iter->first;
1305                    delete iter->second;
1306              }              }
1307              result.Add("PARAMETERS", s);              result.Add("PARAMETERS", s);
1308          }          }
1309      }      }
1310      catch (LinuxSamplerException e) {      catch (Exception e) {
1311          result.Error(e);          result.Error(e);
1312      }      }
1313      return result.Produce();      return result.Produce();
# Line 839  String LSCPServer::GetAudioOutputDriverI Line 1327  String LSCPServer::GetAudioOutputDriverI
1327              for (;iter != parameters.end(); iter++) {              for (;iter != parameters.end(); iter++) {
1328                  if (s != "") s += ",";                  if (s != "") s += ",";
1329                  s += iter->first;                  s += iter->first;
1330                    delete iter->second;
1331              }              }
1332              result.Add("PARAMETERS", s);              result.Add("PARAMETERS", s);
1333          }          }
1334      }      }
1335      catch (LinuxSamplerException e) {      catch (Exception e) {
1336          result.Error(e);          result.Error(e);
1337      }      }
1338      return result.Produce();      return result.Produce();
# Line 869  String LSCPServer::GetMidiInputDriverPar Line 1358  String LSCPServer::GetMidiInputDriverPar
1358          if (oRangeMin)      result.Add("RANGE_MIN",     *oRangeMin);          if (oRangeMin)      result.Add("RANGE_MIN",     *oRangeMin);
1359          if (oRangeMax)      result.Add("RANGE_MAX",     *oRangeMax);          if (oRangeMax)      result.Add("RANGE_MAX",     *oRangeMax);
1360          if (oPossibilities) result.Add("POSSIBILITIES", *oPossibilities);          if (oPossibilities) result.Add("POSSIBILITIES", *oPossibilities);
1361            delete pParameter;
1362      }      }
1363      catch (LinuxSamplerException e) {      catch (Exception e) {
1364          result.Error(e);          result.Error(e);
1365      }      }
1366      return result.Produce();      return result.Produce();
# Line 896  String LSCPServer::GetAudioOutputDriverP Line 1386  String LSCPServer::GetAudioOutputDriverP
1386          if (oRangeMin)      result.Add("RANGE_MIN",     *oRangeMin);          if (oRangeMin)      result.Add("RANGE_MIN",     *oRangeMin);
1387          if (oRangeMax)      result.Add("RANGE_MAX",     *oRangeMax);          if (oRangeMax)      result.Add("RANGE_MAX",     *oRangeMax);
1388          if (oPossibilities) result.Add("POSSIBILITIES", *oPossibilities);          if (oPossibilities) result.Add("POSSIBILITIES", *oPossibilities);
1389            delete pParameter;
1390      }      }
1391      catch (LinuxSamplerException e) {      catch (Exception e) {
1392          result.Error(e);          result.Error(e);
1393      }      }
1394      return result.Produce();      return result.Produce();
# Line 910  String LSCPServer::GetAudioOutputDeviceC Line 1401  String LSCPServer::GetAudioOutputDeviceC
1401          uint count = pSampler->AudioOutputDevices();          uint count = pSampler->AudioOutputDevices();
1402          result.Add(count); // success          result.Add(count); // success
1403      }      }
1404      catch (LinuxSamplerException e) {      catch (Exception e) {
1405          result.Error(e);          result.Error(e);
1406      }      }
1407      return result.Produce();      return result.Produce();
# Line 923  String LSCPServer::GetMidiInputDeviceCou Line 1414  String LSCPServer::GetMidiInputDeviceCou
1414          uint count = pSampler->MidiInputDevices();          uint count = pSampler->MidiInputDevices();
1415          result.Add(count); // success          result.Add(count); // success
1416      }      }
1417      catch (LinuxSamplerException e) {      catch (Exception e) {
1418          result.Error(e);          result.Error(e);
1419      }      }
1420      return result.Produce();      return result.Produce();
# Line 942  String LSCPServer::GetAudioOutputDevices Line 1433  String LSCPServer::GetAudioOutputDevices
1433          }          }
1434          result.Add(s);          result.Add(s);
1435      }      }
1436      catch (LinuxSamplerException e) {      catch (Exception e) {
1437          result.Error(e);          result.Error(e);
1438      }      }
1439      return result.Produce();      return result.Produce();
# Line 961  String LSCPServer::GetMidiInputDevices() Line 1452  String LSCPServer::GetMidiInputDevices()
1452          }          }
1453          result.Add(s);          result.Add(s);
1454      }      }
1455      catch (LinuxSamplerException e) {      catch (Exception e) {
1456          result.Error(e);          result.Error(e);
1457      }      }
1458      return result.Produce();      return result.Produce();
# Line 972  String LSCPServer::GetAudioOutputDeviceI Line 1463  String LSCPServer::GetAudioOutputDeviceI
1463      LSCPResultSet result;      LSCPResultSet result;
1464      try {      try {
1465          std::map<uint,AudioOutputDevice*> devices = pSampler->GetAudioOutputDevices();          std::map<uint,AudioOutputDevice*> devices = pSampler->GetAudioOutputDevices();
1466          if (!devices.count(DeviceIndex)) throw LinuxSamplerException("There is no audio output device with index " + ToString(DeviceIndex) + ".");          if (!devices.count(DeviceIndex)) throw Exception("There is no audio output device with index " + ToString(DeviceIndex) + ".");
1467          AudioOutputDevice* pDevice = devices[DeviceIndex];          AudioOutputDevice* pDevice = devices[DeviceIndex];
1468          result.Add("DRIVER", pDevice->Driver());          result.Add("DRIVER", pDevice->Driver());
1469          std::map<String,DeviceCreationParameter*> parameters = pDevice->DeviceParameters();          std::map<String,DeviceCreationParameter*> parameters = pDevice->DeviceParameters();
# Line 981  String LSCPServer::GetAudioOutputDeviceI Line 1472  String LSCPServer::GetAudioOutputDeviceI
1472              result.Add(iter->first, iter->second->Value());              result.Add(iter->first, iter->second->Value());
1473          }          }
1474      }      }
1475      catch (LinuxSamplerException e) {      catch (Exception e) {
1476          result.Error(e);          result.Error(e);
1477      }      }
1478      return result.Produce();      return result.Produce();
# Line 992  String LSCPServer::GetMidiInputDeviceInf Line 1483  String LSCPServer::GetMidiInputDeviceInf
1483      LSCPResultSet result;      LSCPResultSet result;
1484      try {      try {
1485          std::map<uint,MidiInputDevice*> devices = pSampler->GetMidiInputDevices();          std::map<uint,MidiInputDevice*> devices = pSampler->GetMidiInputDevices();
1486          if (!devices.count(DeviceIndex)) throw LinuxSamplerException("There is no MIDI input device with index " + ToString(DeviceIndex) + ".");          if (!devices.count(DeviceIndex)) throw Exception("There is no MIDI input device with index " + ToString(DeviceIndex) + ".");
1487          MidiInputDevice* pDevice = devices[DeviceIndex];          MidiInputDevice* pDevice = devices[DeviceIndex];
1488          result.Add("DRIVER", pDevice->Driver());          result.Add("DRIVER", pDevice->Driver());
1489          std::map<String,DeviceCreationParameter*> parameters = pDevice->DeviceParameters();          std::map<String,DeviceCreationParameter*> parameters = pDevice->DeviceParameters();
# Line 1001  String LSCPServer::GetMidiInputDeviceInf Line 1492  String LSCPServer::GetMidiInputDeviceInf
1492              result.Add(iter->first, iter->second->Value());              result.Add(iter->first, iter->second->Value());
1493          }          }
1494      }      }
1495      catch (LinuxSamplerException e) {      catch (Exception e) {
1496          result.Error(e);          result.Error(e);
1497      }      }
1498      return result.Produce();      return result.Produce();
# Line 1012  String LSCPServer::GetMidiInputPortInfo( Line 1503  String LSCPServer::GetMidiInputPortInfo(
1503      try {      try {
1504          // get MIDI input device          // get MIDI input device
1505          std::map<uint,MidiInputDevice*> devices = pSampler->GetMidiInputDevices();          std::map<uint,MidiInputDevice*> devices = pSampler->GetMidiInputDevices();
1506          if (!devices.count(DeviceIndex)) throw LinuxSamplerException("There is no MIDI input device with index " + ToString(DeviceIndex) + ".");          if (!devices.count(DeviceIndex)) throw Exception("There is no MIDI input device with index " + ToString(DeviceIndex) + ".");
1507          MidiInputDevice* pDevice = devices[DeviceIndex];          MidiInputDevice* pDevice = devices[DeviceIndex];
1508    
1509          // get MIDI port          // get MIDI port
1510          MidiInputPort* pMidiInputPort = pDevice->GetPort(PortIndex);          MidiInputPort* pMidiInputPort = pDevice->GetPort(PortIndex);
1511          if (!pMidiInputPort) throw LinuxSamplerException("There is no MIDI input port with index " + ToString(PortIndex) + ".");          if (!pMidiInputPort) throw Exception("There is no MIDI input port with index " + ToString(PortIndex) + ".");
1512    
1513          // return the values of all MIDI port parameters          // return the values of all MIDI port parameters
1514          std::map<String,DeviceRuntimeParameter*> parameters = pMidiInputPort->PortParameters();          std::map<String,DeviceRuntimeParameter*> parameters = pMidiInputPort->PortParameters();
# Line 1026  String LSCPServer::GetMidiInputPortInfo( Line 1517  String LSCPServer::GetMidiInputPortInfo(
1517              result.Add(iter->first, iter->second->Value());              result.Add(iter->first, iter->second->Value());
1518          }          }
1519      }      }
1520      catch (LinuxSamplerException e) {      catch (Exception e) {
1521          result.Error(e);          result.Error(e);
1522      }      }
1523      return result.Produce();      return result.Produce();
# Line 1038  String LSCPServer::GetAudioOutputChannel Line 1529  String LSCPServer::GetAudioOutputChannel
1529      try {      try {
1530          // get audio output device          // get audio output device
1531          std::map<uint,AudioOutputDevice*> devices = pSampler->GetAudioOutputDevices();          std::map<uint,AudioOutputDevice*> devices = pSampler->GetAudioOutputDevices();
1532          if (!devices.count(DeviceId)) throw LinuxSamplerException("There is no audio output device with index " + ToString(DeviceId) + ".");          if (!devices.count(DeviceId)) throw Exception("There is no audio output device with index " + ToString(DeviceId) + ".");
1533          AudioOutputDevice* pDevice = devices[DeviceId];          AudioOutputDevice* pDevice = devices[DeviceId];
1534    
1535          // get audio channel          // get audio channel
1536          AudioChannel* pChannel = pDevice->Channel(ChannelId);          AudioChannel* pChannel = pDevice->Channel(ChannelId);
1537          if (!pChannel) throw LinuxSamplerException("Audio output device does not have audio channel " + ToString(ChannelId) + ".");          if (!pChannel) throw Exception("Audio output device does not have audio channel " + ToString(ChannelId) + ".");
1538    
1539          // return the values of all audio channel parameters          // return the values of all audio channel parameters
1540          std::map<String,DeviceRuntimeParameter*> parameters = pChannel->ChannelParameters();          std::map<String,DeviceRuntimeParameter*> parameters = pChannel->ChannelParameters();
# Line 1052  String LSCPServer::GetAudioOutputChannel Line 1543  String LSCPServer::GetAudioOutputChannel
1543              result.Add(iter->first, iter->second->Value());              result.Add(iter->first, iter->second->Value());
1544          }          }
1545      }      }
1546      catch (LinuxSamplerException e) {      catch (Exception e) {
1547          result.Error(e);          result.Error(e);
1548      }      }
1549      return result.Produce();      return result.Produce();
# Line 1064  String LSCPServer::GetMidiInputPortParam Line 1555  String LSCPServer::GetMidiInputPortParam
1555      try {      try {
1556          // get MIDI input device          // get MIDI input device
1557          std::map<uint,MidiInputDevice*> devices = pSampler->GetMidiInputDevices();          std::map<uint,MidiInputDevice*> devices = pSampler->GetMidiInputDevices();
1558          if (!devices.count(DeviceId)) throw LinuxSamplerException("There is no midi input device with index " + ToString(DeviceId) + ".");          if (!devices.count(DeviceId)) throw Exception("There is no midi input device with index " + ToString(DeviceId) + ".");
1559          MidiInputDevice* pDevice = devices[DeviceId];          MidiInputDevice* pDevice = devices[DeviceId];
1560    
1561          // get midi port          // get midi port
1562          MidiInputPort* pPort = pDevice->GetPort(PortId);          MidiInputPort* pPort = pDevice->GetPort(PortId);
1563          if (!pPort) throw LinuxSamplerException("Midi input device does not have port " + ToString(PortId) + ".");          if (!pPort) throw Exception("Midi input device does not have port " + ToString(PortId) + ".");
1564    
1565          // get desired port parameter          // get desired port parameter
1566          std::map<String,DeviceRuntimeParameter*> parameters = pPort->PortParameters();          std::map<String,DeviceRuntimeParameter*> parameters = pPort->PortParameters();
1567          if (!parameters.count(ParameterName)) throw LinuxSamplerException("Midi port does not provide a parameter '" + ParameterName + "'.");          if (!parameters.count(ParameterName)) throw Exception("Midi port does not provide a parameter '" + ParameterName + "'.");
1568          DeviceRuntimeParameter* pParameter = parameters[ParameterName];          DeviceRuntimeParameter* pParameter = parameters[ParameterName];
1569    
1570          // return all fields of this audio channel parameter          // return all fields of this audio channel parameter
# Line 1085  String LSCPServer::GetMidiInputPortParam Line 1576  String LSCPServer::GetMidiInputPortParam
1576          if (pParameter->RangeMax())      result.Add("RANGE_MAX",     *pParameter->RangeMax());          if (pParameter->RangeMax())      result.Add("RANGE_MAX",     *pParameter->RangeMax());
1577          if (pParameter->Possibilities()) result.Add("POSSIBILITIES", *pParameter->Possibilities());          if (pParameter->Possibilities()) result.Add("POSSIBILITIES", *pParameter->Possibilities());
1578      }      }
1579      catch (LinuxSamplerException e) {      catch (Exception e) {
1580          result.Error(e);          result.Error(e);
1581      }      }
1582      return result.Produce();      return result.Produce();
# Line 1097  String LSCPServer::GetAudioOutputChannel Line 1588  String LSCPServer::GetAudioOutputChannel
1588      try {      try {
1589          // get audio output device          // get audio output device
1590          std::map<uint,AudioOutputDevice*> devices = pSampler->GetAudioOutputDevices();          std::map<uint,AudioOutputDevice*> devices = pSampler->GetAudioOutputDevices();
1591          if (!devices.count(DeviceId)) throw LinuxSamplerException("There is no audio output device with index " + ToString(DeviceId) + ".");          if (!devices.count(DeviceId)) throw Exception("There is no audio output device with index " + ToString(DeviceId) + ".");
1592          AudioOutputDevice* pDevice = devices[DeviceId];          AudioOutputDevice* pDevice = devices[DeviceId];
1593    
1594          // get audio channel          // get audio channel
1595          AudioChannel* pChannel = pDevice->Channel(ChannelId);          AudioChannel* pChannel = pDevice->Channel(ChannelId);
1596          if (!pChannel) throw LinuxSamplerException("Audio output device does not have audio channel " + ToString(ChannelId) + ".");          if (!pChannel) throw Exception("Audio output device does not have audio channel " + ToString(ChannelId) + ".");
1597    
1598          // get desired audio channel parameter          // get desired audio channel parameter
1599          std::map<String,DeviceRuntimeParameter*> parameters = pChannel->ChannelParameters();          std::map<String,DeviceRuntimeParameter*> parameters = pChannel->ChannelParameters();
1600          if (!parameters.count(ParameterName)) throw LinuxSamplerException("Audio channel does not provide a parameter '" + ParameterName + "'.");          if (!parameters.count(ParameterName)) throw Exception("Audio channel does not provide a parameter '" + ParameterName + "'.");
1601          DeviceRuntimeParameter* pParameter = parameters[ParameterName];          DeviceRuntimeParameter* pParameter = parameters[ParameterName];
1602    
1603          // return all fields of this audio channel parameter          // return all fields of this audio channel parameter
# Line 1118  String LSCPServer::GetAudioOutputChannel Line 1609  String LSCPServer::GetAudioOutputChannel
1609          if (pParameter->RangeMax())      result.Add("RANGE_MAX",     *pParameter->RangeMax());          if (pParameter->RangeMax())      result.Add("RANGE_MAX",     *pParameter->RangeMax());
1610          if (pParameter->Possibilities()) result.Add("POSSIBILITIES", *pParameter->Possibilities());          if (pParameter->Possibilities()) result.Add("POSSIBILITIES", *pParameter->Possibilities());
1611      }      }
1612      catch (LinuxSamplerException e) {      catch (Exception e) {
1613          result.Error(e);          result.Error(e);
1614      }      }
1615      return result.Produce();      return result.Produce();
# Line 1130  String LSCPServer::SetAudioOutputChannel Line 1621  String LSCPServer::SetAudioOutputChannel
1621      try {      try {
1622          // get audio output device          // get audio output device
1623          std::map<uint,AudioOutputDevice*> devices = pSampler->GetAudioOutputDevices();          std::map<uint,AudioOutputDevice*> devices = pSampler->GetAudioOutputDevices();
1624          if (!devices.count(DeviceId)) throw LinuxSamplerException("There is no audio output device with index " + ToString(DeviceId) + ".");          if (!devices.count(DeviceId)) throw Exception("There is no audio output device with index " + ToString(DeviceId) + ".");
1625          AudioOutputDevice* pDevice = devices[DeviceId];          AudioOutputDevice* pDevice = devices[DeviceId];
1626    
1627          // get audio channel          // get audio channel
1628          AudioChannel* pChannel = pDevice->Channel(ChannelId);          AudioChannel* pChannel = pDevice->Channel(ChannelId);
1629          if (!pChannel) throw LinuxSamplerException("Audio output device does not have audio channel " + ToString(ChannelId) + ".");          if (!pChannel) throw Exception("Audio output device does not have audio channel " + ToString(ChannelId) + ".");
1630    
1631          // get desired audio channel parameter          // get desired audio channel parameter
1632          std::map<String,DeviceRuntimeParameter*> parameters = pChannel->ChannelParameters();          std::map<String,DeviceRuntimeParameter*> parameters = pChannel->ChannelParameters();
1633          if (!parameters.count(ParamKey)) throw LinuxSamplerException("Audio channel does not provide a parameter '" + ParamKey + "'.");          if (!parameters.count(ParamKey)) throw Exception("Audio channel does not provide a parameter '" + ParamKey + "'.");
1634          DeviceRuntimeParameter* pParameter = parameters[ParamKey];          DeviceRuntimeParameter* pParameter = parameters[ParamKey];
1635    
1636          // set new channel parameter value          // set new channel parameter value
1637          pParameter->SetValue(ParamVal);          pParameter->SetValue(ParamVal);
1638            LSCPServer::SendLSCPNotify(LSCPEvent(LSCPEvent::event_audio_device_info, DeviceId));
1639      }      }
1640      catch (LinuxSamplerException e) {      catch (Exception e) {
1641          result.Error(e);          result.Error(e);
1642      }      }
1643      return result.Produce();      return result.Produce();
# Line 1156  String LSCPServer::SetAudioOutputDeviceP Line 1648  String LSCPServer::SetAudioOutputDeviceP
1648      LSCPResultSet result;      LSCPResultSet result;
1649      try {      try {
1650          std::map<uint,AudioOutputDevice*> devices = pSampler->GetAudioOutputDevices();          std::map<uint,AudioOutputDevice*> devices = pSampler->GetAudioOutputDevices();
1651          if (!devices.count(DeviceIndex)) throw LinuxSamplerException("There is no audio output device with index " + ToString(DeviceIndex) + ".");          if (!devices.count(DeviceIndex)) throw Exception("There is no audio output device with index " + ToString(DeviceIndex) + ".");
1652          AudioOutputDevice* pDevice = devices[DeviceIndex];          AudioOutputDevice* pDevice = devices[DeviceIndex];
1653          std::map<String,DeviceCreationParameter*> parameters = pDevice->DeviceParameters();          std::map<String,DeviceCreationParameter*> parameters = pDevice->DeviceParameters();
1654          if (!parameters.count(ParamKey)) throw LinuxSamplerException("Audio output device " + ToString(DeviceIndex) + " does not have a device parameter '" + ParamKey + "'");          if (!parameters.count(ParamKey)) throw Exception("Audio output device " + ToString(DeviceIndex) + " does not have a device parameter '" + ParamKey + "'");
1655          parameters[ParamKey]->SetValue(ParamVal);          parameters[ParamKey]->SetValue(ParamVal);
1656            LSCPServer::SendLSCPNotify(LSCPEvent(LSCPEvent::event_audio_device_info, DeviceIndex));
1657      }      }
1658      catch (LinuxSamplerException e) {      catch (Exception e) {
1659          result.Error(e);          result.Error(e);
1660      }      }
1661      return result.Produce();      return result.Produce();
# Line 1173  String LSCPServer::SetMidiInputDevicePar Line 1666  String LSCPServer::SetMidiInputDevicePar
1666      LSCPResultSet result;      LSCPResultSet result;
1667      try {      try {
1668          std::map<uint,MidiInputDevice*> devices = pSampler->GetMidiInputDevices();          std::map<uint,MidiInputDevice*> devices = pSampler->GetMidiInputDevices();
1669          if (!devices.count(DeviceIndex)) throw LinuxSamplerException("There is no MIDI input device with index " + ToString(DeviceIndex) + ".");          if (!devices.count(DeviceIndex)) throw Exception("There is no MIDI input device with index " + ToString(DeviceIndex) + ".");
1670          MidiInputDevice* pDevice = devices[DeviceIndex];          MidiInputDevice* pDevice = devices[DeviceIndex];
1671          std::map<String,DeviceCreationParameter*> parameters = pDevice->DeviceParameters();          std::map<String,DeviceCreationParameter*> parameters = pDevice->DeviceParameters();
1672          if (!parameters.count(ParamKey)) throw LinuxSamplerException("MIDI input device " + ToString(DeviceIndex) + " does not have a device parameter '" + ParamKey + "'");          if (!parameters.count(ParamKey)) throw Exception("MIDI input device " + ToString(DeviceIndex) + " does not have a device parameter '" + ParamKey + "'");
1673          parameters[ParamKey]->SetValue(ParamVal);          parameters[ParamKey]->SetValue(ParamVal);
1674            LSCPServer::SendLSCPNotify(LSCPEvent(LSCPEvent::event_midi_device_info, DeviceIndex));
1675      }      }
1676      catch (LinuxSamplerException e) {      catch (Exception e) {
1677          result.Error(e);          result.Error(e);
1678      }      }
1679      return result.Produce();      return result.Produce();
# Line 1191  String LSCPServer::SetMidiInputPortParam Line 1685  String LSCPServer::SetMidiInputPortParam
1685      try {      try {
1686          // get MIDI input device          // get MIDI input device
1687          std::map<uint,MidiInputDevice*> devices = pSampler->GetMidiInputDevices();          std::map<uint,MidiInputDevice*> devices = pSampler->GetMidiInputDevices();
1688          if (!devices.count(DeviceIndex)) throw LinuxSamplerException("There is no MIDI input device with index " + ToString(DeviceIndex) + ".");          if (!devices.count(DeviceIndex)) throw Exception("There is no MIDI input device with index " + ToString(DeviceIndex) + ".");
1689          MidiInputDevice* pDevice = devices[DeviceIndex];          MidiInputDevice* pDevice = devices[DeviceIndex];
1690    
1691          // get MIDI port          // get MIDI port
1692          MidiInputPort* pMidiInputPort = pDevice->GetPort(PortIndex);          MidiInputPort* pMidiInputPort = pDevice->GetPort(PortIndex);
1693          if (!pMidiInputPort) throw LinuxSamplerException("There is no MIDI input port with index " + ToString(PortIndex) + ".");          if (!pMidiInputPort) throw Exception("There is no MIDI input port with index " + ToString(PortIndex) + ".");
1694    
1695          // set port parameter value          // set port parameter value
1696          std::map<String,DeviceRuntimeParameter*> parameters = pMidiInputPort->PortParameters();          std::map<String,DeviceRuntimeParameter*> parameters = pMidiInputPort->PortParameters();
1697          if (!parameters.count(ParamKey)) throw LinuxSamplerException("MIDI input device " + ToString(PortIndex) + " does not have a parameter '" + ParamKey + "'");          if (!parameters.count(ParamKey)) throw Exception("MIDI input device " + ToString(PortIndex) + " does not have a parameter '" + ParamKey + "'");
1698          parameters[ParamKey]->SetValue(ParamVal);          parameters[ParamKey]->SetValue(ParamVal);
1699            LSCPServer::SendLSCPNotify(LSCPEvent(LSCPEvent::event_midi_device_info, DeviceIndex));
1700      }      }
1701      catch (LinuxSamplerException e) {      catch (Exception e) {
1702          result.Error(e);          result.Error(e);
1703      }      }
1704      return result.Produce();      return result.Produce();
# Line 1218  String LSCPServer::SetAudioOutputChannel Line 1713  String LSCPServer::SetAudioOutputChannel
1713      LSCPResultSet result;      LSCPResultSet result;
1714      try {      try {
1715          SamplerChannel* pSamplerChannel = pSampler->GetSamplerChannel(uiSamplerChannel);          SamplerChannel* pSamplerChannel = pSampler->GetSamplerChannel(uiSamplerChannel);
1716          if (!pSamplerChannel) throw LinuxSamplerException("Invalid sampler channel number " + ToString(uiSamplerChannel));          if (!pSamplerChannel) throw Exception("Invalid sampler channel number " + ToString(uiSamplerChannel));
1717          EngineChannel* pEngineChannel = pSamplerChannel->GetEngineChannel();          EngineChannel* pEngineChannel = pSamplerChannel->GetEngineChannel();
1718          if (!pEngineChannel) throw LinuxSamplerException("No engine type yet assigned to sampler channel " + ToString(uiSamplerChannel));          if (!pEngineChannel) throw Exception("No engine type yet assigned to sampler channel " + ToString(uiSamplerChannel));
1719          if (!pSamplerChannel->GetAudioOutputDevice()) throw LinuxSamplerException("No audio output device connected to sampler channel " + ToString(uiSamplerChannel));          if (!pSamplerChannel->GetAudioOutputDevice()) throw Exception("No audio output device connected to sampler channel " + ToString(uiSamplerChannel));
1720          pEngineChannel->SetOutputChannel(ChannelAudioOutputChannel, AudioOutputDeviceInputChannel);          pEngineChannel->SetOutputChannel(ChannelAudioOutputChannel, AudioOutputDeviceInputChannel);
1721      }      }
1722      catch (LinuxSamplerException e) {      catch (Exception e) {
1723           result.Error(e);           result.Error(e);
1724      }      }
1725      return result.Produce();      return result.Produce();
# Line 1233  String LSCPServer::SetAudioOutputChannel Line 1728  String LSCPServer::SetAudioOutputChannel
1728  String LSCPServer::SetAudioOutputDevice(uint AudioDeviceId, uint uiSamplerChannel) {  String LSCPServer::SetAudioOutputDevice(uint AudioDeviceId, uint uiSamplerChannel) {
1729      dmsg(2,("LSCPServer: SetAudiotOutputDevice(AudioDeviceId=%d, SamplerChannel=%d)\n",AudioDeviceId,uiSamplerChannel));      dmsg(2,("LSCPServer: SetAudiotOutputDevice(AudioDeviceId=%d, SamplerChannel=%d)\n",AudioDeviceId,uiSamplerChannel));
1730      LSCPResultSet result;      LSCPResultSet result;
1731        LockRTNotify();
1732      try {      try {
1733          SamplerChannel* pSamplerChannel = pSampler->GetSamplerChannel(uiSamplerChannel);          SamplerChannel* pSamplerChannel = pSampler->GetSamplerChannel(uiSamplerChannel);
1734          if (!pSamplerChannel) throw LinuxSamplerException("Invalid sampler channel number " + ToString(uiSamplerChannel));          if (!pSamplerChannel) throw Exception("Invalid sampler channel number " + ToString(uiSamplerChannel));
1735          std::map<uint, AudioOutputDevice*> devices = pSampler->GetAudioOutputDevices();          std::map<uint, AudioOutputDevice*> devices = pSampler->GetAudioOutputDevices();
1736          if (!devices.count(AudioDeviceId)) throw LinuxSamplerException("There is no audio output device with index " + ToString(AudioDeviceId));          if (!devices.count(AudioDeviceId)) throw Exception("There is no audio output device with index " + ToString(AudioDeviceId));
1737          AudioOutputDevice* pDevice = devices[AudioDeviceId];          AudioOutputDevice* pDevice = devices[AudioDeviceId];
1738          pSamplerChannel->SetAudioOutputDevice(pDevice);          pSamplerChannel->SetAudioOutputDevice(pDevice);
1739      }      }
1740      catch (LinuxSamplerException e) {      catch (Exception e) {
1741           result.Error(e);           result.Error(e);
1742      }      }
1743        UnlockRTNotify();
1744      return result.Produce();      return result.Produce();
1745  }  }
1746    
1747  String LSCPServer::SetAudioOutputType(String AudioOutputDriver, uint uiSamplerChannel) {  String LSCPServer::SetAudioOutputType(String AudioOutputDriver, uint uiSamplerChannel) {
1748      dmsg(2,("LSCPServer: SetAudioOutputType(String AudioOutputDriver=%s, SamplerChannel=%d)\n",AudioOutputDriver.c_str(),uiSamplerChannel));      dmsg(2,("LSCPServer: SetAudioOutputType(String AudioOutputDriver=%s, SamplerChannel=%d)\n",AudioOutputDriver.c_str(),uiSamplerChannel));
1749      LSCPResultSet result;      LSCPResultSet result;
1750        LockRTNotify();
1751      try {      try {
1752          SamplerChannel* pSamplerChannel = pSampler->GetSamplerChannel(uiSamplerChannel);          SamplerChannel* pSamplerChannel = pSampler->GetSamplerChannel(uiSamplerChannel);
1753          if (!pSamplerChannel) throw LinuxSamplerException("Invalid sampler channel number " + ToString(uiSamplerChannel));          if (!pSamplerChannel) throw Exception("Invalid sampler channel number " + ToString(uiSamplerChannel));
1754          // Driver type name aliasing...          // Driver type name aliasing...
1755          if (AudioOutputDriver == "Alsa") AudioOutputDriver = "ALSA";          if (AudioOutputDriver == "Alsa") AudioOutputDriver = "ALSA";
1756          if (AudioOutputDriver == "Jack") AudioOutputDriver = "JACK";          if (AudioOutputDriver == "Jack") AudioOutputDriver = "JACK";
# Line 1274  String LSCPServer::SetAudioOutputType(St Line 1772  String LSCPServer::SetAudioOutputType(St
1772          }          }
1773          // Must have a device...          // Must have a device...
1774          if (pDevice == NULL)          if (pDevice == NULL)
1775              throw LinuxSamplerException("Internal error: could not create audio output device.");              throw Exception("Internal error: could not create audio output device.");
1776          // Set it as the current channel device...          // Set it as the current channel device...
1777          pSamplerChannel->SetAudioOutputDevice(pDevice);          pSamplerChannel->SetAudioOutputDevice(pDevice);
1778      }      }
1779      catch (LinuxSamplerException e) {      catch (Exception e) {
1780           result.Error(e);           result.Error(e);
1781      }      }
1782        UnlockRTNotify();
1783      return result.Produce();      return result.Produce();
1784  }  }
1785    
# Line 1289  String LSCPServer::SetMIDIInputPort(uint Line 1788  String LSCPServer::SetMIDIInputPort(uint
1788      LSCPResultSet result;      LSCPResultSet result;
1789      try {      try {
1790          SamplerChannel* pSamplerChannel = pSampler->GetSamplerChannel(uiSamplerChannel);          SamplerChannel* pSamplerChannel = pSampler->GetSamplerChannel(uiSamplerChannel);
1791          if (!pSamplerChannel) throw LinuxSamplerException("Invalid sampler channel number " + ToString(uiSamplerChannel));          if (!pSamplerChannel) throw Exception("Invalid sampler channel number " + ToString(uiSamplerChannel));
1792          pSamplerChannel->SetMidiInputPort(MIDIPort);          pSamplerChannel->SetMidiInputPort(MIDIPort);
1793      }      }
1794      catch (LinuxSamplerException e) {      catch (Exception e) {
1795           result.Error(e);           result.Error(e);
1796      }      }
1797      return result.Produce();      return result.Produce();
# Line 1303  String LSCPServer::SetMIDIInputChannel(u Line 1802  String LSCPServer::SetMIDIInputChannel(u
1802      LSCPResultSet result;      LSCPResultSet result;
1803      try {      try {
1804          SamplerChannel* pSamplerChannel = pSampler->GetSamplerChannel(uiSamplerChannel);          SamplerChannel* pSamplerChannel = pSampler->GetSamplerChannel(uiSamplerChannel);
1805          if (!pSamplerChannel) throw LinuxSamplerException("Invalid sampler channel number " + ToString(uiSamplerChannel));          if (!pSamplerChannel) throw Exception("Invalid sampler channel number " + ToString(uiSamplerChannel));
1806          pSamplerChannel->SetMidiInputChannel((midi_chan_t) MIDIChannel);          pSamplerChannel->SetMidiInputChannel((midi_chan_t) MIDIChannel);
1807      }      }
1808      catch (LinuxSamplerException e) {      catch (Exception e) {
1809           result.Error(e);           result.Error(e);
1810      }      }
1811      return result.Produce();      return result.Produce();
# Line 1317  String LSCPServer::SetMIDIInputDevice(ui Line 1816  String LSCPServer::SetMIDIInputDevice(ui
1816      LSCPResultSet result;      LSCPResultSet result;
1817      try {      try {
1818          SamplerChannel* pSamplerChannel = pSampler->GetSamplerChannel(uiSamplerChannel);          SamplerChannel* pSamplerChannel = pSampler->GetSamplerChannel(uiSamplerChannel);
1819          if (!pSamplerChannel) throw LinuxSamplerException("Invalid sampler channel number " + ToString(uiSamplerChannel));          if (!pSamplerChannel) throw Exception("Invalid sampler channel number " + ToString(uiSamplerChannel));
1820          std::map<uint, MidiInputDevice*> devices = pSampler->GetMidiInputDevices();          std::map<uint, MidiInputDevice*> devices = pSampler->GetMidiInputDevices();
1821          if (!devices.count(MIDIDeviceId)) throw LinuxSamplerException("There is no MIDI input device with index " + ToString(MIDIDeviceId));          if (!devices.count(MIDIDeviceId)) throw Exception("There is no MIDI input device with index " + ToString(MIDIDeviceId));
1822          MidiInputDevice* pDevice = devices[MIDIDeviceId];          MidiInputDevice* pDevice = devices[MIDIDeviceId];
1823          pSamplerChannel->SetMidiInputDevice(pDevice);          pSamplerChannel->SetMidiInputDevice(pDevice);
1824      }      }
1825      catch (LinuxSamplerException e) {      catch (Exception e) {
1826           result.Error(e);           result.Error(e);
1827      }      }
1828      return result.Produce();      return result.Produce();
# Line 1334  String LSCPServer::SetMIDIInputType(Stri Line 1833  String LSCPServer::SetMIDIInputType(Stri
1833      LSCPResultSet result;      LSCPResultSet result;
1834      try {      try {
1835          SamplerChannel* pSamplerChannel = pSampler->GetSamplerChannel(uiSamplerChannel);          SamplerChannel* pSamplerChannel = pSampler->GetSamplerChannel(uiSamplerChannel);
1836          if (!pSamplerChannel) throw LinuxSamplerException("Invalid sampler channel number " + ToString(uiSamplerChannel));          if (!pSamplerChannel) throw Exception("Invalid sampler channel number " + ToString(uiSamplerChannel));
1837          // Driver type name aliasing...          // Driver type name aliasing...
1838          if (MidiInputDriver == "Alsa") MidiInputDriver = "ALSA";          if (MidiInputDriver == "Alsa") MidiInputDriver = "ALSA";
1839          // Check if there's one MIDI input device already created          // Check if there's one MIDI input device already created
# Line 1354  String LSCPServer::SetMIDIInputType(Stri Line 1853  String LSCPServer::SetMIDIInputType(Stri
1853              pDevice = pSampler->CreateMidiInputDevice(MidiInputDriver, params);              pDevice = pSampler->CreateMidiInputDevice(MidiInputDriver, params);
1854              // Make it with at least one initial port.              // Make it with at least one initial port.
1855              std::map<String,DeviceCreationParameter*> parameters = pDevice->DeviceParameters();              std::map<String,DeviceCreationParameter*> parameters = pDevice->DeviceParameters();
             parameters["PORTS"]->SetValue("1");  
1856          }          }
1857          // Must have a device...          // Must have a device...
1858          if (pDevice == NULL)          if (pDevice == NULL)
1859              throw LinuxSamplerException("Internal error: could not create MIDI input device.");              throw Exception("Internal error: could not create MIDI input device.");
1860          // Set it as the current channel device...          // Set it as the current channel device...
1861          pSamplerChannel->SetMidiInputDevice(pDevice);          pSamplerChannel->SetMidiInputDevice(pDevice);
1862      }      }
1863      catch (LinuxSamplerException e) {      catch (Exception e) {
1864           result.Error(e);           result.Error(e);
1865      }      }
1866      return result.Produce();      return result.Produce();
# Line 1377  String LSCPServer::SetMIDIInput(uint MID Line 1875  String LSCPServer::SetMIDIInput(uint MID
1875      LSCPResultSet result;      LSCPResultSet result;
1876      try {      try {
1877          SamplerChannel* pSamplerChannel = pSampler->GetSamplerChannel(uiSamplerChannel);          SamplerChannel* pSamplerChannel = pSampler->GetSamplerChannel(uiSamplerChannel);
1878          if (!pSamplerChannel) throw LinuxSamplerException("Invalid sampler channel number " + ToString(uiSamplerChannel));          if (!pSamplerChannel) throw Exception("Invalid sampler channel number " + ToString(uiSamplerChannel));
1879          std::map<uint, MidiInputDevice*> devices =  pSampler->GetMidiInputDevices();          std::map<uint, MidiInputDevice*> devices =  pSampler->GetMidiInputDevices();
1880          if (!devices.count(MIDIDeviceId)) throw LinuxSamplerException("There is no MIDI input device with index " + ToString(MIDIDeviceId));          if (!devices.count(MIDIDeviceId)) throw Exception("There is no MIDI input device with index " + ToString(MIDIDeviceId));
1881          MidiInputDevice* pDevice = devices[MIDIDeviceId];          MidiInputDevice* pDevice = devices[MIDIDeviceId];
1882          pSamplerChannel->SetMidiInput(pDevice, MIDIPort, (midi_chan_t) MIDIChannel);          pSamplerChannel->SetMidiInput(pDevice, MIDIPort, (midi_chan_t) MIDIChannel);
1883      }      }
1884      catch (LinuxSamplerException e) {      catch (Exception e) {
1885           result.Error(e);           result.Error(e);
1886      }      }
1887      return result.Produce();      return result.Produce();
# Line 1397  String LSCPServer::SetVolume(double dVol Line 1895  String LSCPServer::SetVolume(double dVol
1895      dmsg(2,("LSCPServer: SetVolume(Volume=%f, SamplerChannel=%d)\n", dVolume, uiSamplerChannel));      dmsg(2,("LSCPServer: SetVolume(Volume=%f, SamplerChannel=%d)\n", dVolume, uiSamplerChannel));
1896      LSCPResultSet result;      LSCPResultSet result;
1897      try {      try {
1898          SamplerChannel* pSamplerChannel = pSampler->GetSamplerChannel(uiSamplerChannel);          EngineChannel* pEngineChannel = GetEngineChannel(uiSamplerChannel);
         if (!pSamplerChannel) throw LinuxSamplerException("Invalid sampler channel number " + ToString(uiSamplerChannel));  
         EngineChannel* pEngineChannel = pSamplerChannel->GetEngineChannel();  
         if (!pEngineChannel) throw LinuxSamplerException("No engine type assigned to sampler channel");  
1899          pEngineChannel->Volume(dVolume);          pEngineChannel->Volume(dVolume);
1900      }      }
1901      catch (LinuxSamplerException e) {      catch (Exception e) {
1902           result.Error(e);           result.Error(e);
1903      }      }
1904      return result.Produce();      return result.Produce();
# Line 1416  String LSCPServer::SetChannelMute(bool b Line 1911  String LSCPServer::SetChannelMute(bool b
1911      dmsg(2,("LSCPServer: SetChannelMute(bMute=%d,uiSamplerChannel=%d)\n",bMute,uiSamplerChannel));      dmsg(2,("LSCPServer: SetChannelMute(bMute=%d,uiSamplerChannel=%d)\n",bMute,uiSamplerChannel));
1912      LSCPResultSet result;      LSCPResultSet result;
1913      try {      try {
1914          SamplerChannel* pSamplerChannel = pSampler->GetSamplerChannel(uiSamplerChannel);          EngineChannel* pEngineChannel = GetEngineChannel(uiSamplerChannel);
         if (!pSamplerChannel) throw LinuxSamplerException("Invalid sampler channel number " + ToString(uiSamplerChannel));  
   
         EngineChannel* pEngineChannel = pSamplerChannel->GetEngineChannel();  
         if (!pEngineChannel) throw LinuxSamplerException("No engine type assigned to sampler channel");  
1915    
1916          if(!bMute) pEngineChannel->SetMute((HasSoloChannel() && !pEngineChannel->GetSolo()) ? -1 : 0);          if(!bMute) pEngineChannel->SetMute((HasSoloChannel() && !pEngineChannel->GetSolo()) ? -1 : 0);
1917          else pEngineChannel->SetMute(1);          else pEngineChannel->SetMute(1);
1918      } catch (LinuxSamplerException e) {      } catch (Exception e) {
1919          result.Error(e);          result.Error(e);
1920      }      }
1921      return result.Produce();      return result.Produce();
# Line 1437  String LSCPServer::SetChannelSolo(bool b Line 1928  String LSCPServer::SetChannelSolo(bool b
1928      dmsg(2,("LSCPServer: SetChannelSolo(bSolo=%d,uiSamplerChannel=%d)\n",bSolo,uiSamplerChannel));      dmsg(2,("LSCPServer: SetChannelSolo(bSolo=%d,uiSamplerChannel=%d)\n",bSolo,uiSamplerChannel));
1929      LSCPResultSet result;      LSCPResultSet result;
1930      try {      try {
1931          SamplerChannel* pSamplerChannel = pSampler->GetSamplerChannel(uiSamplerChannel);          EngineChannel* pEngineChannel = GetEngineChannel(uiSamplerChannel);
         if (!pSamplerChannel) throw LinuxSamplerException("Invalid sampler channel number " + ToString(uiSamplerChannel));  
   
         EngineChannel* pEngineChannel = pSamplerChannel->GetEngineChannel();  
         if (!pEngineChannel) throw LinuxSamplerException("No engine type assigned to sampler channel");  
1932    
1933          bool oldSolo = pEngineChannel->GetSolo();          bool oldSolo = pEngineChannel->GetSolo();
1934          bool hadSoloChannel = HasSoloChannel();          bool hadSoloChannel = HasSoloChannel();
1935            
1936          pEngineChannel->SetSolo(bSolo);          pEngineChannel->SetSolo(bSolo);
1937            
1938          if(!oldSolo && bSolo) {          if(!oldSolo && bSolo) {
1939              if(pEngineChannel->GetMute() == -1) pEngineChannel->SetMute(0);              if(pEngineChannel->GetMute() == -1) pEngineChannel->SetMute(0);
1940              if(!hadSoloChannel) MuteNonSoloChannels();              if(!hadSoloChannel) MuteNonSoloChannels();
1941          }          }
1942            
1943          if(oldSolo && !bSolo) {          if(oldSolo && !bSolo) {
1944              if(!HasSoloChannel()) UnmuteChannels();              if(!HasSoloChannel()) UnmuteChannels();
1945              else if(!pEngineChannel->GetMute()) pEngineChannel->SetMute(-1);              else if(!pEngineChannel->GetMute()) pEngineChannel->SetMute(-1);
1946          }          }
1947      } catch (LinuxSamplerException e) {      } catch (Exception e) {
1948          result.Error(e);          result.Error(e);
1949      }      }
1950      return result.Produce();      return result.Produce();
# Line 1510  void  LSCPServer::UnmuteChannels() { Line 1997  void  LSCPServer::UnmuteChannels() {
1997      }      }
1998  }  }
1999    
2000    String LSCPServer::AddOrReplaceMIDIInstrumentMapping(uint MidiMapID, uint MidiBank, uint MidiProg, String EngineType, String InstrumentFile, uint InstrumentIndex, float Volume, MidiInstrumentMapper::mode_t LoadMode, String Name, bool bModal) {
2001        dmsg(2,("LSCPServer: AddOrReplaceMIDIInstrumentMapping()\n"));
2002    
2003        midi_prog_index_t idx;
2004        idx.midi_bank_msb = (MidiBank >> 7) & 0x7f;
2005        idx.midi_bank_lsb = MidiBank & 0x7f;
2006        idx.midi_prog     = MidiProg;
2007    
2008        MidiInstrumentMapper::entry_t entry;
2009        entry.EngineName      = EngineType;
2010        entry.InstrumentFile  = InstrumentFile;
2011        entry.InstrumentIndex = InstrumentIndex;
2012        entry.LoadMode        = LoadMode;
2013        entry.Volume          = Volume;
2014        entry.Name            = Name;
2015    
2016        LSCPResultSet result;
2017        try {
2018            // PERSISTENT mapping commands might block for a long time, so in
2019            // that case we add/replace the mapping in another thread in case
2020            // the NON_MODAL argument was supplied, non persistent mappings
2021            // should return immediately, so we don't need to do that for them
2022            bool bInBackground = (entry.LoadMode == MidiInstrumentMapper::PERSISTENT && !bModal);
2023            MidiInstrumentMapper::AddOrReplaceEntry(MidiMapID, idx, entry, bInBackground);
2024        } catch (Exception e) {
2025            result.Error(e);
2026        }
2027        return result.Produce();
2028    }
2029    
2030    String LSCPServer::RemoveMIDIInstrumentMapping(uint MidiMapID, uint MidiBank, uint MidiProg) {
2031        dmsg(2,("LSCPServer: RemoveMIDIInstrumentMapping()\n"));
2032    
2033        midi_prog_index_t idx;
2034        idx.midi_bank_msb = (MidiBank >> 7) & 0x7f;
2035        idx.midi_bank_lsb = MidiBank & 0x7f;
2036        idx.midi_prog     = MidiProg;
2037    
2038        LSCPResultSet result;
2039        try {
2040            MidiInstrumentMapper::RemoveEntry(MidiMapID, idx);
2041        } catch (Exception e) {
2042            result.Error(e);
2043        }
2044        return result.Produce();
2045    }
2046    
2047    String LSCPServer::GetMidiInstrumentMappings(uint MidiMapID) {
2048        dmsg(2,("LSCPServer: GetMidiInstrumentMappings()\n"));
2049        LSCPResultSet result;
2050        try {
2051            result.Add(MidiInstrumentMapper::GetInstrumentCount(MidiMapID));
2052        } catch (Exception e) {
2053            result.Error(e);
2054        }
2055        return result.Produce();
2056    }
2057    
2058    
2059    String LSCPServer::GetAllMidiInstrumentMappings() {
2060        dmsg(2,("LSCPServer: GetAllMidiInstrumentMappings()\n"));
2061        LSCPResultSet result;
2062        try {
2063            result.Add(MidiInstrumentMapper::GetInstrumentCount());
2064        } catch (Exception e) {
2065            result.Error(e);
2066        }
2067        return result.Produce();
2068    }
2069    
2070    String LSCPServer::GetMidiInstrumentMapping(uint MidiMapID, uint MidiBank, uint MidiProg) {
2071        dmsg(2,("LSCPServer: GetMidiIstrumentMapping()\n"));
2072        LSCPResultSet result;
2073        try {
2074            MidiInstrumentMapper::entry_t entry = MidiInstrumentMapper::GetEntry(MidiMapID, MidiBank, MidiProg);
2075            // convert the filename into the correct encoding as defined for LSCP
2076            // (especially in terms of special characters -> escape sequences)
2077    #if WIN32
2078            const String instrumentFileName = Path::fromWindows(entry.InstrumentFile).toLscp();
2079    #else
2080            // assuming POSIX
2081            const String instrumentFileName = Path::fromPosix(entry.InstrumentFile).toLscp();
2082    #endif
2083    
2084            result.Add("NAME", _escapeLscpResponse(entry.Name));
2085            result.Add("ENGINE_NAME", entry.EngineName);
2086            result.Add("INSTRUMENT_FILE", instrumentFileName);
2087            result.Add("INSTRUMENT_NR", (int) entry.InstrumentIndex);
2088            String instrumentName;
2089            Engine* pEngine = EngineFactory::Create(entry.EngineName);
2090            if (pEngine) {
2091                if (pEngine->GetInstrumentManager()) {
2092                    InstrumentManager::instrument_id_t instrID;
2093                    instrID.FileName = entry.InstrumentFile;
2094                    instrID.Index    = entry.InstrumentIndex;
2095                    instrumentName = pEngine->GetInstrumentManager()->GetInstrumentName(instrID);
2096                }
2097                EngineFactory::Destroy(pEngine);
2098            }
2099            result.Add("INSTRUMENT_NAME", _escapeLscpResponse(instrumentName));
2100            switch (entry.LoadMode) {
2101                case MidiInstrumentMapper::ON_DEMAND:
2102                    result.Add("LOAD_MODE", "ON_DEMAND");
2103                    break;
2104                case MidiInstrumentMapper::ON_DEMAND_HOLD:
2105                    result.Add("LOAD_MODE", "ON_DEMAND_HOLD");
2106                    break;
2107                case MidiInstrumentMapper::PERSISTENT:
2108                    result.Add("LOAD_MODE", "PERSISTENT");
2109                    break;
2110                default:
2111                    throw Exception("entry reflects invalid LOAD_MODE, consider this as a bug!");
2112            }
2113            result.Add("VOLUME", entry.Volume);
2114        } catch (Exception e) {
2115            result.Error(e);
2116        }
2117        return result.Produce();
2118    }
2119    
2120    String LSCPServer::ListMidiInstrumentMappings(uint MidiMapID) {
2121        dmsg(2,("LSCPServer: ListMidiInstrumentMappings()\n"));
2122        LSCPResultSet result;
2123        try {
2124            String s;
2125            std::map<midi_prog_index_t,MidiInstrumentMapper::entry_t> mappings = MidiInstrumentMapper::Entries(MidiMapID);
2126            std::map<midi_prog_index_t,MidiInstrumentMapper::entry_t>::iterator iter = mappings.begin();
2127            for (; iter != mappings.end(); iter++) {
2128                if (s.size()) s += ",";
2129                s += "{" + ToString(MidiMapID) + ","
2130                         + ToString((int(iter->first.midi_bank_msb) << 7) | int(iter->first.midi_bank_lsb)) + ","
2131                         + ToString(int(iter->first.midi_prog)) + "}";
2132            }
2133            result.Add(s);
2134        } catch (Exception e) {
2135            result.Error(e);
2136        }
2137        return result.Produce();
2138    }
2139    
2140    String LSCPServer::ListAllMidiInstrumentMappings() {
2141        dmsg(2,("LSCPServer: ListAllMidiInstrumentMappings()\n"));
2142        LSCPResultSet result;
2143        try {
2144            std::vector<int> maps = MidiInstrumentMapper::Maps();
2145            String s;
2146            for (int i = 0; i < maps.size(); i++) {
2147                std::map<midi_prog_index_t,MidiInstrumentMapper::entry_t> mappings = MidiInstrumentMapper::Entries(maps[i]);
2148                std::map<midi_prog_index_t,MidiInstrumentMapper::entry_t>::iterator iter = mappings.begin();
2149                for (; iter != mappings.end(); iter++) {
2150                    if (s.size()) s += ",";
2151                    s += "{" + ToString(maps[i]) + ","
2152                             + ToString((int(iter->first.midi_bank_msb) << 7) | int(iter->first.midi_bank_lsb)) + ","
2153                             + ToString(int(iter->first.midi_prog)) + "}";
2154                }
2155            }
2156            result.Add(s);
2157        } catch (Exception e) {
2158            result.Error(e);
2159        }
2160        return result.Produce();
2161    }
2162    
2163    String LSCPServer::ClearMidiInstrumentMappings(uint MidiMapID) {
2164        dmsg(2,("LSCPServer: ClearMidiInstrumentMappings()\n"));
2165        LSCPResultSet result;
2166        try {
2167            MidiInstrumentMapper::RemoveAllEntries(MidiMapID);
2168        } catch (Exception e) {
2169            result.Error(e);
2170        }
2171        return result.Produce();
2172    }
2173    
2174    String LSCPServer::ClearAllMidiInstrumentMappings() {
2175        dmsg(2,("LSCPServer: ClearAllMidiInstrumentMappings()\n"));
2176        LSCPResultSet result;
2177        try {
2178            std::vector<int> maps = MidiInstrumentMapper::Maps();
2179            for (int i = 0; i < maps.size(); i++)
2180                MidiInstrumentMapper::RemoveAllEntries(maps[i]);
2181        } catch (Exception e) {
2182            result.Error(e);
2183        }
2184        return result.Produce();
2185    }
2186    
2187    String LSCPServer::AddMidiInstrumentMap(String MapName) {
2188        dmsg(2,("LSCPServer: AddMidiInstrumentMap()\n"));
2189        LSCPResultSet result;
2190        try {
2191            int MapID = MidiInstrumentMapper::AddMap(MapName);
2192            result = LSCPResultSet(MapID);
2193        } catch (Exception e) {
2194            result.Error(e);
2195        }
2196        return result.Produce();
2197    }
2198    
2199    String LSCPServer::RemoveMidiInstrumentMap(uint MidiMapID) {
2200        dmsg(2,("LSCPServer: RemoveMidiInstrumentMap()\n"));
2201        LSCPResultSet result;
2202        try {
2203            MidiInstrumentMapper::RemoveMap(MidiMapID);
2204        } catch (Exception e) {
2205            result.Error(e);
2206        }
2207        return result.Produce();
2208    }
2209    
2210    String LSCPServer::RemoveAllMidiInstrumentMaps() {
2211        dmsg(2,("LSCPServer: RemoveAllMidiInstrumentMaps()\n"));
2212        LSCPResultSet result;
2213        try {
2214            MidiInstrumentMapper::RemoveAllMaps();
2215        } catch (Exception e) {
2216            result.Error(e);
2217        }
2218        return result.Produce();
2219    }
2220    
2221    String LSCPServer::GetMidiInstrumentMaps() {
2222        dmsg(2,("LSCPServer: GetMidiInstrumentMaps()\n"));
2223        LSCPResultSet result;
2224        try {
2225            result.Add(MidiInstrumentMapper::Maps().size());
2226        } catch (Exception e) {
2227            result.Error(e);
2228        }
2229        return result.Produce();
2230    }
2231    
2232    String LSCPServer::ListMidiInstrumentMaps() {
2233        dmsg(2,("LSCPServer: ListMidiInstrumentMaps()\n"));
2234        LSCPResultSet result;
2235        try {
2236            std::vector<int> maps = MidiInstrumentMapper::Maps();
2237            String sList;
2238            for (int i = 0; i < maps.size(); i++) {
2239                if (sList != "") sList += ",";
2240                sList += ToString(maps[i]);
2241            }
2242            result.Add(sList);
2243        } catch (Exception e) {
2244            result.Error(e);
2245        }
2246        return result.Produce();
2247    }
2248    
2249    String LSCPServer::GetMidiInstrumentMap(uint MidiMapID) {
2250        dmsg(2,("LSCPServer: GetMidiInstrumentMap()\n"));
2251        LSCPResultSet result;
2252        try {
2253            result.Add("NAME", _escapeLscpResponse(MidiInstrumentMapper::MapName(MidiMapID)));
2254            result.Add("DEFAULT", MidiInstrumentMapper::GetDefaultMap() == MidiMapID);
2255        } catch (Exception e) {
2256            result.Error(e);
2257        }
2258        return result.Produce();
2259    }
2260    
2261    String LSCPServer::SetMidiInstrumentMapName(uint MidiMapID, String NewName) {
2262        dmsg(2,("LSCPServer: SetMidiInstrumentMapName()\n"));
2263        LSCPResultSet result;
2264        try {
2265            MidiInstrumentMapper::RenameMap(MidiMapID, NewName);
2266        } catch (Exception e) {
2267            result.Error(e);
2268        }
2269        return result.Produce();
2270    }
2271    
2272    /**
2273     * Set the MIDI instrument map the given sampler channel shall use for
2274     * handling MIDI program change messages. There are the following two
2275     * special (negative) values:
2276     *
2277     *    - (-1) :  set to NONE (ignore program changes)
2278     *    - (-2) :  set to DEFAULT map
2279     */
2280    String LSCPServer::SetChannelMap(uint uiSamplerChannel, int MidiMapID) {
2281        dmsg(2,("LSCPServer: SetChannelMap()\n"));
2282        LSCPResultSet result;
2283        try {
2284            EngineChannel* pEngineChannel = GetEngineChannel(uiSamplerChannel);
2285    
2286            if      (MidiMapID == -1) pEngineChannel->SetMidiInstrumentMapToNone();
2287            else if (MidiMapID == -2) pEngineChannel->SetMidiInstrumentMapToDefault();
2288            else                      pEngineChannel->SetMidiInstrumentMap(MidiMapID);
2289        } catch (Exception e) {
2290            result.Error(e);
2291        }
2292        return result.Produce();
2293    }
2294    
2295    String LSCPServer::CreateFxSend(uint uiSamplerChannel, uint MidiCtrl, String Name) {
2296        dmsg(2,("LSCPServer: CreateFxSend()\n"));
2297        LSCPResultSet result;
2298        try {
2299            EngineChannel* pEngineChannel = GetEngineChannel(uiSamplerChannel);
2300    
2301            FxSend* pFxSend = pEngineChannel->AddFxSend(MidiCtrl, Name);
2302            if (!pFxSend) throw Exception("Could not add FxSend, don't ask, I don't know why (probably a bug)");
2303    
2304            result = LSCPResultSet(pFxSend->Id()); // success
2305        } catch (Exception e) {
2306            result.Error(e);
2307        }
2308        return result.Produce();
2309    }
2310    
2311    String LSCPServer::DestroyFxSend(uint uiSamplerChannel, uint FxSendID) {
2312        dmsg(2,("LSCPServer: DestroyFxSend()\n"));
2313        LSCPResultSet result;
2314        try {
2315            EngineChannel* pEngineChannel = GetEngineChannel(uiSamplerChannel);
2316    
2317            FxSend* pFxSend = NULL;
2318            for (int i = 0; i < pEngineChannel->GetFxSendCount(); i++) {
2319                if (pEngineChannel->GetFxSend(i)->Id() == FxSendID) {
2320                    pFxSend = pEngineChannel->GetFxSend(i);
2321                    break;
2322                }
2323            }
2324            if (!pFxSend) throw Exception("There is no FxSend with that ID on the given sampler channel");
2325            pEngineChannel->RemoveFxSend(pFxSend);
2326        } catch (Exception e) {
2327            result.Error(e);
2328        }
2329        return result.Produce();
2330    }
2331    
2332    String LSCPServer::GetFxSends(uint uiSamplerChannel) {
2333        dmsg(2,("LSCPServer: GetFxSends()\n"));
2334        LSCPResultSet result;
2335        try {
2336            EngineChannel* pEngineChannel = GetEngineChannel(uiSamplerChannel);
2337    
2338            result.Add(pEngineChannel->GetFxSendCount());
2339        } catch (Exception e) {
2340            result.Error(e);
2341        }
2342        return result.Produce();
2343    }
2344    
2345    String LSCPServer::ListFxSends(uint uiSamplerChannel) {
2346        dmsg(2,("LSCPServer: ListFxSends()\n"));
2347        LSCPResultSet result;
2348        String list;
2349        try {
2350            EngineChannel* pEngineChannel = GetEngineChannel(uiSamplerChannel);
2351    
2352            for (int i = 0; i < pEngineChannel->GetFxSendCount(); i++) {
2353                FxSend* pFxSend = pEngineChannel->GetFxSend(i);
2354                if (list != "") list += ",";
2355                list += ToString(pFxSend->Id());
2356            }
2357            result.Add(list);
2358        } catch (Exception e) {
2359            result.Error(e);
2360        }
2361        return result.Produce();
2362    }
2363    
2364    FxSend* LSCPServer::GetFxSend(uint uiSamplerChannel, uint FxSendID) {
2365        EngineChannel* pEngineChannel = GetEngineChannel(uiSamplerChannel);
2366    
2367        FxSend* pFxSend = NULL;
2368        for (int i = 0; i < pEngineChannel->GetFxSendCount(); i++) {
2369            if (pEngineChannel->GetFxSend(i)->Id() == FxSendID) {
2370                pFxSend = pEngineChannel->GetFxSend(i);
2371                break;
2372            }
2373        }
2374        if (!pFxSend) throw Exception("There is no FxSend with that ID on the given sampler channel");
2375        return pFxSend;
2376    }
2377    
2378    String LSCPServer::GetFxSendInfo(uint uiSamplerChannel, uint FxSendID) {
2379        dmsg(2,("LSCPServer: GetFxSendInfo()\n"));
2380        LSCPResultSet result;
2381        try {
2382            EngineChannel* pEngineChannel = GetEngineChannel(uiSamplerChannel);
2383            FxSend* pFxSend = GetFxSend(uiSamplerChannel, FxSendID);
2384    
2385            // gather audio routing informations
2386            String AudioRouting;
2387            for (int chan = 0; chan < pEngineChannel->Channels(); chan++) {
2388                if (AudioRouting != "") AudioRouting += ",";
2389                AudioRouting += ToString(pFxSend->DestinationChannel(chan));
2390            }
2391    
2392            const String sEffectRouting =
2393                (pFxSend->DestinationEffectChain() >= 0 && pFxSend->DestinationEffectChainPosition() >= 0)
2394                    ? ToString(pFxSend->DestinationEffectChain()) + "," + ToString(pFxSend->DestinationEffectChainPosition())
2395                    : "NONE";
2396    
2397            // success
2398            result.Add("NAME", _escapeLscpResponse(pFxSend->Name()));
2399            result.Add("MIDI_CONTROLLER", pFxSend->MidiController());
2400            result.Add("LEVEL", ToString(pFxSend->Level()));
2401            result.Add("AUDIO_OUTPUT_ROUTING", AudioRouting);
2402            result.Add("EFFECT", sEffectRouting);
2403        } catch (Exception e) {
2404            result.Error(e);
2405        }
2406        return result.Produce();
2407    }
2408    
2409    String LSCPServer::SetFxSendName(uint uiSamplerChannel, uint FxSendID, String Name) {
2410        dmsg(2,("LSCPServer: SetFxSendName()\n"));
2411        LSCPResultSet result;
2412        try {
2413            FxSend* pFxSend = GetFxSend(uiSamplerChannel, FxSendID);
2414    
2415            pFxSend->SetName(Name);
2416            LSCPServer::SendLSCPNotify(LSCPEvent(LSCPEvent::event_fx_send_info, uiSamplerChannel, FxSendID));
2417        } catch (Exception e) {
2418            result.Error(e);
2419        }
2420        return result.Produce();
2421    }
2422    
2423    String LSCPServer::SetFxSendAudioOutputChannel(uint uiSamplerChannel, uint FxSendID, uint FxSendChannel, uint DeviceChannel) {
2424        dmsg(2,("LSCPServer: SetFxSendAudioOutputChannel()\n"));
2425        LSCPResultSet result;
2426        try {
2427            FxSend* pFxSend = GetFxSend(uiSamplerChannel, FxSendID);
2428    
2429            pFxSend->SetDestinationChannel(FxSendChannel, DeviceChannel);
2430            LSCPServer::SendLSCPNotify(LSCPEvent(LSCPEvent::event_fx_send_info, uiSamplerChannel, FxSendID));
2431        } catch (Exception e) {
2432            result.Error(e);
2433        }
2434        return result.Produce();
2435    }
2436    
2437    String LSCPServer::SetFxSendMidiController(uint uiSamplerChannel, uint FxSendID, uint MidiController) {
2438        dmsg(2,("LSCPServer: SetFxSendMidiController()\n"));
2439        LSCPResultSet result;
2440        try {
2441            FxSend* pFxSend = GetFxSend(uiSamplerChannel, FxSendID);
2442    
2443            pFxSend->SetMidiController(MidiController);
2444            LSCPServer::SendLSCPNotify(LSCPEvent(LSCPEvent::event_fx_send_info, uiSamplerChannel, FxSendID));
2445        } catch (Exception e) {
2446            result.Error(e);
2447        }
2448        return result.Produce();
2449    }
2450    
2451    String LSCPServer::SetFxSendLevel(uint uiSamplerChannel, uint FxSendID, double dLevel) {
2452        dmsg(2,("LSCPServer: SetFxSendLevel()\n"));
2453        LSCPResultSet result;
2454        try {
2455            FxSend* pFxSend = GetFxSend(uiSamplerChannel, FxSendID);
2456    
2457            pFxSend->SetLevel((float)dLevel);
2458            LSCPServer::SendLSCPNotify(LSCPEvent(LSCPEvent::event_fx_send_info, uiSamplerChannel, FxSendID));
2459        } catch (Exception e) {
2460            result.Error(e);
2461        }
2462        return result.Produce();
2463    }
2464    
2465    String LSCPServer::SetFxSendEffect(uint uiSamplerChannel, uint FxSendID, int iSendEffectChain, int iEffectChainPosition) {
2466        dmsg(2,("LSCPServer: SetFxSendEffect(%d,%d)\n", iSendEffectChain, iEffectChainPosition));
2467        LSCPResultSet result;
2468        try {
2469            FxSend* pFxSend = GetFxSend(uiSamplerChannel, FxSendID);
2470    
2471            pFxSend->SetDestinationEffect(iSendEffectChain, iEffectChainPosition);
2472            LSCPServer::SendLSCPNotify(LSCPEvent(LSCPEvent::event_fx_send_info, uiSamplerChannel, FxSendID));
2473        } catch (Exception e) {
2474            result.Error(e);
2475        }
2476        return result.Produce();
2477    }
2478    
2479    String LSCPServer::GetAvailableEffects() {
2480        dmsg(2,("LSCPServer: GetAvailableEffects()\n"));
2481        LSCPResultSet result;
2482        try {
2483            int n = EffectFactory::AvailableEffectsCount();
2484            result.Add(n);
2485        }
2486        catch (Exception e) {
2487            result.Error(e);
2488        }
2489        return result.Produce();
2490    }
2491    
2492    String LSCPServer::ListAvailableEffects() {
2493        dmsg(2,("LSCPServer: ListAvailableEffects()\n"));
2494        LSCPResultSet result;
2495        String list;
2496        try {
2497            //FIXME: for now we simply enumerate from 0 .. EffectFactory::AvailableEffectsCount() here, in future we should use unique IDs for effects during the whole sampler session. This issue comes into game when the user forces a reload of available effect plugins
2498            int n = EffectFactory::AvailableEffectsCount();
2499            for (int i = 0; i < n; i++) {
2500                if (i) list += ",";
2501                list += ToString(i);
2502            }
2503        }
2504        catch (Exception e) {
2505            result.Error(e);
2506        }
2507        result.Add(list);
2508        return result.Produce();
2509    }
2510    
2511    String LSCPServer::GetEffectInfo(int iEffectIndex) {
2512        dmsg(2,("LSCPServer: GetEffectInfo(%d)\n", iEffectIndex));
2513        LSCPResultSet result;
2514        try {
2515            EffectInfo* pEffectInfo = EffectFactory::GetEffectInfo(iEffectIndex);
2516            if (!pEffectInfo)
2517                throw Exception("There is no effect with index " + ToString(iEffectIndex));
2518    
2519            // convert the filename into the correct encoding as defined for LSCP
2520            // (especially in terms of special characters -> escape sequences)
2521    #if WIN32
2522            const String dllFileName = Path::fromWindows(pEffectInfo->Module()).toLscp();
2523    #else
2524            // assuming POSIX
2525            const String dllFileName = Path::fromPosix(pEffectInfo->Module()).toLscp();
2526    #endif
2527    
2528            result.Add("SYSTEM", pEffectInfo->EffectSystem());
2529            result.Add("MODULE", dllFileName);
2530            result.Add("NAME", _escapeLscpResponse(pEffectInfo->Name()));
2531            result.Add("DESCRIPTION", _escapeLscpResponse(pEffectInfo->Description()));
2532        }
2533        catch (Exception e) {
2534            result.Error(e);
2535        }
2536        return result.Produce();    
2537    }
2538    
2539    String LSCPServer::GetEffectInstanceInfo(int iEffectInstance) {
2540        dmsg(2,("LSCPServer: GetEffectInstanceInfo(%d)\n", iEffectInstance));
2541        LSCPResultSet result;
2542        try {
2543            Effect* pEffect = EffectFactory::GetEffectInstanceByID(iEffectInstance);
2544            if (!pEffect)
2545                throw Exception("There is no effect instance with ID " + ToString(iEffectInstance));
2546    
2547            EffectInfo* pEffectInfo = pEffect->GetEffectInfo();
2548    
2549            // convert the filename into the correct encoding as defined for LSCP
2550            // (especially in terms of special characters -> escape sequences)
2551    #if WIN32
2552            const String dllFileName = Path::fromWindows(pEffectInfo->Module()).toLscp();
2553    #else
2554            // assuming POSIX
2555            const String dllFileName = Path::fromPosix(pEffectInfo->Module()).toLscp();
2556    #endif
2557    
2558            result.Add("SYSTEM", pEffectInfo->EffectSystem());
2559            result.Add("MODULE", dllFileName);
2560            result.Add("NAME", _escapeLscpResponse(pEffectInfo->Name()));
2561            result.Add("DESCRIPTION", _escapeLscpResponse(pEffectInfo->Description()));
2562            result.Add("INPUT_CONTROLS", ToString(pEffect->InputControlCount()));
2563        }
2564        catch (Exception e) {
2565            result.Error(e);
2566        }
2567        return result.Produce();
2568    }
2569    
2570    String LSCPServer::GetEffectInstanceInputControlInfo(int iEffectInstance, int iInputControlIndex) {
2571        dmsg(2,("LSCPServer: GetEffectInstanceInputControlInfo(%d,%d)\n", iEffectInstance, iInputControlIndex));
2572        LSCPResultSet result;
2573        try {
2574            Effect* pEffect = EffectFactory::GetEffectInstanceByID(iEffectInstance);
2575            if (!pEffect)
2576                throw Exception("There is no effect instance with ID " + ToString(iEffectInstance));
2577    
2578            EffectControl* pEffectControl = pEffect->InputControl(iInputControlIndex);
2579            if (!pEffectControl)
2580                throw Exception(
2581                    "Effect instance " + ToString(iEffectInstance) +
2582                    " does not have an input control with index " +
2583                    ToString(iInputControlIndex)
2584                );
2585    
2586            result.Add("DESCRIPTION", _escapeLscpResponse(pEffectControl->Description()));
2587            result.Add("VALUE", pEffectControl->Value());
2588            if (pEffectControl->MinValue())
2589                 result.Add("RANGE_MIN", *pEffectControl->MinValue());
2590            if (pEffectControl->MaxValue())
2591                 result.Add("RANGE_MAX", *pEffectControl->MaxValue());
2592            if (!pEffectControl->Possibilities().empty())
2593                 result.Add("POSSIBILITIES", pEffectControl->Possibilities());
2594            if (pEffectControl->DefaultValue())
2595                 result.Add("DEFAULT", *pEffectControl->DefaultValue());
2596        } catch (Exception e) {
2597            result.Error(e);
2598        }
2599        return result.Produce();
2600    }
2601    
2602    String LSCPServer::SetEffectInstanceInputControlValue(int iEffectInstance, int iInputControlIndex, double dValue) {
2603        dmsg(2,("LSCPServer: SetEffectInstanceInputControlValue(%d,%d,%f)\n", iEffectInstance, iInputControlIndex, dValue));
2604        LSCPResultSet result;
2605        try {
2606            Effect* pEffect = EffectFactory::GetEffectInstanceByID(iEffectInstance);
2607            if (!pEffect)
2608                throw Exception("There is no effect instance with ID " + ToString(iEffectInstance));
2609    
2610            EffectControl* pEffectControl = pEffect->InputControl(iInputControlIndex);
2611            if (!pEffectControl)
2612                throw Exception(
2613                    "Effect instance " + ToString(iEffectInstance) +
2614                    " does not have an input control with index " +
2615                    ToString(iInputControlIndex)
2616                );
2617    
2618            pEffectControl->SetValue(dValue);
2619            LSCPServer::SendLSCPNotify(LSCPEvent(LSCPEvent::event_fx_instance_info, iEffectInstance));
2620        } catch (Exception e) {
2621            result.Error(e);
2622        }
2623        return result.Produce();
2624    }
2625    
2626    String LSCPServer::CreateEffectInstance(int iEffectIndex) {
2627        dmsg(2,("LSCPServer: CreateEffectInstance(%d)\n", iEffectIndex));
2628        LSCPResultSet result;
2629        try {
2630            EffectInfo* pEffectInfo = EffectFactory::GetEffectInfo(iEffectIndex);
2631            if (!pEffectInfo)
2632                throw Exception("There is no effect with index " + ToString(iEffectIndex));
2633            Effect* pEffect = EffectFactory::Create(pEffectInfo);
2634            result = pEffect->ID(); // success
2635            LSCPServer::SendLSCPNotify(LSCPEvent(LSCPEvent::event_fx_instance_count, EffectFactory::EffectInstancesCount()));
2636        } catch (Exception e) {
2637            result.Error(e);
2638        }
2639        return result.Produce();
2640    }
2641    
2642    String LSCPServer::CreateEffectInstance(String effectSystem, String module, String effectName) {
2643        dmsg(2,("LSCPServer: CreateEffectInstance('%s','%s','%s')\n", effectSystem.c_str(), module.c_str(), effectName.c_str()));
2644        LSCPResultSet result;
2645        try {
2646            // to allow loading the same LSCP session file on different systems
2647            // successfully, probably with different effect plugin DLL paths or even
2648            // running completely different operating systems, we do the following
2649            // for finding the right effect:
2650            //
2651            // first try to search for an exact match of the effect plugin DLL
2652            // (a.k.a 'module'), to avoid picking the wrong DLL with the same
2653            // effect name ...
2654            EffectInfo* pEffectInfo = EffectFactory::GetEffectInfo(effectSystem, module, effectName, EffectFactory::MODULE_MATCH_EXACTLY);
2655            // ... if no effect with exactly matchin DLL filename was found, then
2656            // try to lower the restrictions of matching the effect plugin DLL
2657            // filename and try again and again ...
2658            if (!pEffectInfo) {
2659                dmsg(2,("no exact module match, trying MODULE_IGNORE_PATH\n"));
2660                pEffectInfo = EffectFactory::GetEffectInfo(effectSystem, module, effectName, EffectFactory::MODULE_IGNORE_PATH);
2661            }
2662            if (!pEffectInfo) {
2663                dmsg(2,("no module match, trying MODULE_IGNORE_PATH | MODULE_IGNORE_CASE\n"));
2664                pEffectInfo = EffectFactory::GetEffectInfo(effectSystem, module, effectName, EffectFactory::MODULE_IGNORE_PATH | EffectFactory::MODULE_IGNORE_CASE);
2665            }
2666            if (!pEffectInfo) {
2667                dmsg(2,("no module match, trying MODULE_IGNORE_PATH | MODULE_IGNORE_CASE | MODULE_IGNORE_EXTENSION\n"));
2668                pEffectInfo = EffectFactory::GetEffectInfo(effectSystem, module, effectName, EffectFactory::MODULE_IGNORE_PATH | EffectFactory::MODULE_IGNORE_CASE | EffectFactory::MODULE_IGNORE_EXTENSION);
2669            }
2670            // ... if there was still no effect found, then completely ignore the
2671            // DLL plugin filename argument and just search for the matching effect
2672            // system type and effect name
2673            if (!pEffectInfo) {
2674                dmsg(2,("no module match, trying MODULE_IGNORE_ALL\n"));
2675                pEffectInfo = EffectFactory::GetEffectInfo(effectSystem, module, effectName, EffectFactory::MODULE_IGNORE_ALL);
2676            }
2677            if (!pEffectInfo)
2678                throw Exception("There is no such effect '" + effectSystem + "' '" + module + "' '" + effectName + "'");
2679    
2680            Effect* pEffect = EffectFactory::Create(pEffectInfo);
2681            result = LSCPResultSet(pEffect->ID());
2682            LSCPServer::SendLSCPNotify(LSCPEvent(LSCPEvent::event_fx_instance_count, EffectFactory::EffectInstancesCount()));
2683        } catch (Exception e) {
2684            result.Error(e);
2685        }
2686        return result.Produce();
2687    }
2688    
2689    String LSCPServer::DestroyEffectInstance(int iEffectInstance) {
2690        dmsg(2,("LSCPServer: DestroyEffectInstance(%d)\n", iEffectInstance));
2691        LSCPResultSet result;
2692        try {
2693            Effect* pEffect = EffectFactory::GetEffectInstanceByID(iEffectInstance);
2694            if (!pEffect)
2695                throw Exception("There is no effect instance with ID " + ToString(iEffectInstance));
2696            EffectFactory::Destroy(pEffect);
2697            LSCPServer::SendLSCPNotify(LSCPEvent(LSCPEvent::event_fx_instance_count, EffectFactory::EffectInstancesCount()));
2698        } catch (Exception e) {
2699            result.Error(e);
2700        }
2701        return result.Produce();
2702    }
2703    
2704    String LSCPServer::GetEffectInstances() {
2705        dmsg(2,("LSCPServer: GetEffectInstances()\n"));
2706        LSCPResultSet result;
2707        try {
2708            int n = EffectFactory::EffectInstancesCount();
2709            result.Add(n);
2710        } catch (Exception e) {
2711            result.Error(e);
2712        }
2713        return result.Produce();
2714    }
2715    
2716    String LSCPServer::ListEffectInstances() {
2717        dmsg(2,("LSCPServer: ListEffectInstances()\n"));
2718        LSCPResultSet result;
2719        String list;
2720        try {
2721            int n = EffectFactory::EffectInstancesCount();
2722            for (int i = 0; i < n; i++) {
2723                Effect* pEffect = EffectFactory::GetEffectInstance(i);
2724                if (i) list += ",";
2725                list += ToString(pEffect->ID());
2726            }
2727        } catch (Exception e) {
2728            result.Error(e);
2729        }
2730        result.Add(list);
2731        return result.Produce();
2732    }
2733    
2734    String LSCPServer::GetSendEffectChains(int iAudioOutputDevice) {
2735        dmsg(2,("LSCPServer: GetSendEffectChains(%d)\n", iAudioOutputDevice));
2736        LSCPResultSet result;
2737        try {
2738            std::map<uint,AudioOutputDevice*> devices = pSampler->GetAudioOutputDevices();
2739            if (!devices.count(iAudioOutputDevice))
2740                throw Exception("There is no audio output device with index " + ToString(iAudioOutputDevice) + ".");
2741            AudioOutputDevice* pDevice = devices[iAudioOutputDevice];
2742            int n = pDevice->SendEffectChainCount();
2743            result.Add(n);
2744        } catch (Exception e) {
2745            result.Error(e);
2746        }
2747        return result.Produce();
2748    }
2749    
2750    String LSCPServer::ListSendEffectChains(int iAudioOutputDevice) {
2751        dmsg(2,("LSCPServer: ListSendEffectChains(%d)\n", iAudioOutputDevice));
2752        LSCPResultSet result;
2753        String list;
2754        try {
2755            std::map<uint,AudioOutputDevice*> devices = pSampler->GetAudioOutputDevices();
2756            if (!devices.count(iAudioOutputDevice))
2757                throw Exception("There is no audio output device with index " + ToString(iAudioOutputDevice) + ".");
2758            AudioOutputDevice* pDevice = devices[iAudioOutputDevice];
2759            int n = pDevice->SendEffectChainCount();
2760            for (int i = 0; i < n; i++) {
2761                EffectChain* pEffectChain = pDevice->SendEffectChain(i);
2762                if (i) list += ",";
2763                list += ToString(pEffectChain->ID());
2764            }
2765        } catch (Exception e) {
2766            result.Error(e);
2767        }
2768        result.Add(list);
2769        return result.Produce();
2770    }
2771    
2772    String LSCPServer::AddSendEffectChain(int iAudioOutputDevice) {
2773        dmsg(2,("LSCPServer: AddSendEffectChain(%d)\n", iAudioOutputDevice));
2774        LSCPResultSet result;
2775        try {
2776            std::map<uint,AudioOutputDevice*> devices = pSampler->GetAudioOutputDevices();
2777            if (!devices.count(iAudioOutputDevice))
2778                throw Exception("There is no audio output device with index " + ToString(iAudioOutputDevice) + ".");
2779            AudioOutputDevice* pDevice = devices[iAudioOutputDevice];
2780            EffectChain* pEffectChain = pDevice->AddSendEffectChain();
2781            result = pEffectChain->ID();
2782            LSCPServer::SendLSCPNotify(LSCPEvent(LSCPEvent::event_send_fx_chain_count, iAudioOutputDevice, pDevice->SendEffectChainCount()));
2783        } catch (Exception e) {
2784            result.Error(e);
2785        }
2786        return result.Produce();
2787    }
2788    
2789    String LSCPServer::RemoveSendEffectChain(int iAudioOutputDevice, int iSendEffectChain) {
2790        dmsg(2,("LSCPServer: RemoveSendEffectChain(%d,%d)\n", iAudioOutputDevice, iSendEffectChain));
2791        LSCPResultSet result;
2792        try {
2793            std::map<uint,AudioOutputDevice*> devices = pSampler->GetAudioOutputDevices();
2794            if (!devices.count(iAudioOutputDevice))
2795                throw Exception("There is no audio output device with index " + ToString(iAudioOutputDevice) + ".");
2796    
2797            std::set<EngineChannel*> engineChannels = EngineChannelFactory::EngineChannelInstances();
2798            std::set<EngineChannel*>::iterator itEngineChannel = engineChannels.begin();
2799            std::set<EngineChannel*>::iterator itEnd           = engineChannels.end();
2800            for (; itEngineChannel != itEnd; ++itEngineChannel) {
2801                AudioOutputDevice* pDev = (*itEngineChannel)->GetAudioOutputDevice();
2802                if (pDev != NULL && pDev->deviceId() == iAudioOutputDevice) {
2803                    for (int i = 0; i < (*itEngineChannel)->GetFxSendCount(); i++) {
2804                        FxSend* fxs = (*itEngineChannel)->GetFxSend(i);
2805                        if(fxs != NULL && fxs->DestinationEffectChain() == iSendEffectChain) {
2806                            throw Exception("The effect chain is still in use by channel " + ToString((*itEngineChannel)->GetSamplerChannel()->Index()));
2807                        }
2808                    }
2809                }
2810            }
2811    
2812            AudioOutputDevice* pDevice = devices[iAudioOutputDevice];
2813            for (int i = 0; i < pDevice->SendEffectChainCount(); i++) {
2814                EffectChain* pEffectChain = pDevice->SendEffectChain(i);
2815                if (pEffectChain->ID() == iSendEffectChain) {
2816                    pDevice->RemoveSendEffectChain(i);
2817                    LSCPServer::SendLSCPNotify(LSCPEvent(LSCPEvent::event_send_fx_chain_count, iAudioOutputDevice, pDevice->SendEffectChainCount()));
2818                    return result.Produce();
2819                }
2820            }
2821            throw Exception(
2822                "There is no send effect chain with ID " +
2823                ToString(iSendEffectChain) + " for audio output device " +
2824                ToString(iAudioOutputDevice) + "."
2825            );
2826        } catch (Exception e) {
2827            result.Error(e);
2828        }
2829        return result.Produce();
2830    }
2831    
2832    static EffectChain* _getSendEffectChain(Sampler* pSampler, int iAudioOutputDevice, int iSendEffectChain) throw (Exception) {
2833        std::map<uint,AudioOutputDevice*> devices = pSampler->GetAudioOutputDevices();
2834        if (!devices.count(iAudioOutputDevice))
2835            throw Exception(
2836                "There is no audio output device with index " +
2837                ToString(iAudioOutputDevice) + "."
2838            );
2839        AudioOutputDevice* pDevice = devices[iAudioOutputDevice];
2840        EffectChain* pEffectChain = pDevice->SendEffectChainByID(iSendEffectChain);
2841        if(pEffectChain != NULL) return pEffectChain;
2842        throw Exception(
2843            "There is no send effect chain with ID " +
2844            ToString(iSendEffectChain) + " for audio output device " +
2845            ToString(iAudioOutputDevice) + "."
2846        );
2847    }
2848    
2849    String LSCPServer::GetSendEffectChainInfo(int iAudioOutputDevice, int iSendEffectChain) {
2850        dmsg(2,("LSCPServer: GetSendEffectChainInfo(%d,%d)\n", iAudioOutputDevice, iSendEffectChain));
2851        LSCPResultSet result;
2852        try {
2853            EffectChain* pEffectChain =
2854                _getSendEffectChain(pSampler, iAudioOutputDevice, iSendEffectChain);
2855            String sEffectSequence;
2856            for (int i = 0; i < pEffectChain->EffectCount(); i++) {
2857                if (i) sEffectSequence += ",";
2858                sEffectSequence += ToString(pEffectChain->GetEffect(i)->ID());
2859            }
2860            result.Add("EFFECT_COUNT", pEffectChain->EffectCount());
2861            result.Add("EFFECT_SEQUENCE", sEffectSequence);
2862        } catch (Exception e) {
2863            result.Error(e);
2864        }
2865        return result.Produce();
2866    }
2867    
2868    String LSCPServer::AppendSendEffectChainEffect(int iAudioOutputDevice, int iSendEffectChain, int iEffectInstance) {
2869        dmsg(2,("LSCPServer: AppendSendEffectChainEffect(%d,%d,%d)\n", iAudioOutputDevice, iSendEffectChain, iEffectInstance));
2870        LSCPResultSet result;
2871        try {
2872            EffectChain* pEffectChain =
2873                _getSendEffectChain(pSampler, iAudioOutputDevice, iSendEffectChain);
2874            Effect* pEffect = EffectFactory::GetEffectInstanceByID(iEffectInstance);
2875            if (!pEffect)
2876                throw Exception("There is no effect instance with ID " + ToString(iEffectInstance));
2877            pEffectChain->AppendEffect(pEffect);
2878            LSCPServer::SendLSCPNotify(LSCPEvent(LSCPEvent::event_send_fx_chain_info, iAudioOutputDevice, iSendEffectChain, pEffectChain->EffectCount()));
2879        } catch (Exception e) {
2880            result.Error(e);
2881        }
2882        return result.Produce();
2883    }
2884    
2885    String LSCPServer::InsertSendEffectChainEffect(int iAudioOutputDevice, int iSendEffectChain, int iEffectChainPosition, int iEffectInstance) {
2886        dmsg(2,("LSCPServer: InsertSendEffectChainEffect(%d,%d,%d,%d)\n", iAudioOutputDevice, iSendEffectChain, iEffectChainPosition, iEffectInstance));
2887        LSCPResultSet result;
2888        try {
2889            EffectChain* pEffectChain =
2890                _getSendEffectChain(pSampler, iAudioOutputDevice, iSendEffectChain);
2891            Effect* pEffect = EffectFactory::GetEffectInstanceByID(iEffectInstance);
2892            if (!pEffect)
2893                throw Exception("There is no effect instance with index " + ToString(iEffectInstance));
2894            pEffectChain->InsertEffect(pEffect, iEffectChainPosition);
2895            LSCPServer::SendLSCPNotify(LSCPEvent(LSCPEvent::event_send_fx_chain_info, iAudioOutputDevice, iSendEffectChain, pEffectChain->EffectCount()));
2896        } catch (Exception e) {
2897            result.Error(e);
2898        }
2899        return result.Produce();
2900    }
2901    
2902    String LSCPServer::RemoveSendEffectChainEffect(int iAudioOutputDevice, int iSendEffectChain, int iEffectChainPosition) {
2903        dmsg(2,("LSCPServer: RemoveSendEffectChainEffect(%d,%d,%d)\n", iAudioOutputDevice, iSendEffectChain, iEffectChainPosition));
2904        LSCPResultSet result;
2905        try {
2906            EffectChain* pEffectChain =
2907                _getSendEffectChain(pSampler, iAudioOutputDevice, iSendEffectChain);
2908    
2909            std::set<EngineChannel*> engineChannels = EngineChannelFactory::EngineChannelInstances();
2910            std::set<EngineChannel*>::iterator itEngineChannel = engineChannels.begin();
2911            std::set<EngineChannel*>::iterator itEnd           = engineChannels.end();
2912            for (; itEngineChannel != itEnd; ++itEngineChannel) {
2913                AudioOutputDevice* pDev = (*itEngineChannel)->GetAudioOutputDevice();
2914                if (pDev != NULL && pDev->deviceId() == iAudioOutputDevice) {
2915                    for (int i = 0; i < (*itEngineChannel)->GetFxSendCount(); i++) {
2916                        FxSend* fxs = (*itEngineChannel)->GetFxSend(i);
2917                        if(fxs != NULL && fxs->DestinationEffectChain() == iSendEffectChain && fxs->DestinationEffectChainPosition() == iEffectChainPosition) {
2918                            throw Exception("The effect instance is still in use by channel " + ToString((*itEngineChannel)->GetSamplerChannel()->Index()));
2919                        }
2920                    }
2921                }
2922            }
2923    
2924            pEffectChain->RemoveEffect(iEffectChainPosition);
2925            LSCPServer::SendLSCPNotify(LSCPEvent(LSCPEvent::event_send_fx_chain_info, iAudioOutputDevice, iSendEffectChain, pEffectChain->EffectCount()));
2926        } catch (Exception e) {
2927            result.Error(e);
2928        }
2929        return result.Produce();
2930    }
2931    
2932    String LSCPServer::EditSamplerChannelInstrument(uint uiSamplerChannel) {
2933        dmsg(2,("LSCPServer: EditSamplerChannelInstrument(SamplerChannel=%d)\n", uiSamplerChannel));
2934        LSCPResultSet result;
2935        try {
2936            EngineChannel* pEngineChannel = GetEngineChannel(uiSamplerChannel);
2937            if (pEngineChannel->InstrumentStatus() < 0) throw Exception("No instrument loaded to sampler channel");
2938            Engine* pEngine = pEngineChannel->GetEngine();
2939            InstrumentManager* pInstrumentManager = pEngine->GetInstrumentManager();
2940            if (!pInstrumentManager) throw Exception("Engine does not provide an instrument manager");
2941            InstrumentManager::instrument_id_t instrumentID;
2942            instrumentID.FileName = pEngineChannel->InstrumentFileName();
2943            instrumentID.Index    = pEngineChannel->InstrumentIndex();
2944            pInstrumentManager->LaunchInstrumentEditor(instrumentID);
2945        } catch (Exception e) {
2946            result.Error(e);
2947        }
2948        return result.Produce();
2949    }
2950    
2951    String LSCPServer::SendChannelMidiData(String MidiMsg, uint uiSamplerChannel, uint Arg1, uint Arg2) {
2952        dmsg(2,("LSCPServer: SendChannelMidiData(MidiMsg=%s,uiSamplerChannel=%d,Arg1=%d,Arg2=%d)\n", MidiMsg.c_str(), uiSamplerChannel, Arg1, Arg2));
2953        LSCPResultSet result;
2954        try {
2955            EngineChannel* pEngineChannel = GetEngineChannel(uiSamplerChannel);
2956    
2957            if (Arg1 > 127 || Arg2 > 127) {
2958                throw Exception("Invalid MIDI message");
2959            }
2960    
2961            VirtualMidiDevice* pMidiDevice = NULL;
2962            std::vector<EventHandler::midi_listener_entry>::iterator iter = eventHandler.channelMidiListeners.begin();
2963            for (; iter != eventHandler.channelMidiListeners.end(); ++iter) {
2964                if ((*iter).pEngineChannel == pEngineChannel) {
2965                    pMidiDevice = (*iter).pMidiListener;
2966                    break;
2967                }
2968            }
2969            
2970            if(pMidiDevice == NULL) throw Exception("Couldn't find virtual MIDI device");
2971    
2972            if (MidiMsg == "NOTE_ON") {
2973                pMidiDevice->SendNoteOnToDevice(Arg1, Arg2);
2974                bool b = pMidiDevice->SendNoteOnToSampler(Arg1, Arg2);
2975                if (!b) throw Exception("MIDI event failed: " + MidiMsg + " " + ToString(Arg1) + " " + ToString(Arg2));
2976            } else if (MidiMsg == "NOTE_OFF") {
2977                pMidiDevice->SendNoteOffToDevice(Arg1, Arg2);
2978                bool b = pMidiDevice->SendNoteOffToSampler(Arg1, Arg2);
2979                if (!b) throw Exception("MIDI event failed: " + MidiMsg + " " + ToString(Arg1) + " " + ToString(Arg2));
2980            } else if (MidiMsg == "CC") {
2981                pMidiDevice->SendCCToDevice(Arg1, Arg2);
2982                bool b = pMidiDevice->SendCCToSampler(Arg1, Arg2);
2983                if (!b) throw Exception("MIDI event failed: " + MidiMsg + " " + ToString(Arg1) + " " + ToString(Arg2));
2984            } else {
2985                throw Exception("Unknown MIDI message type: " + MidiMsg);
2986            }
2987        } catch (Exception e) {
2988            result.Error(e);
2989        }
2990        return result.Produce();
2991    }
2992    
2993  /**  /**
2994   * Will be called by the parser to reset a particular sampler channel.   * Will be called by the parser to reset a particular sampler channel.
2995   */   */
# Line 1517  String LSCPServer::ResetChannel(uint uiS Line 2997  String LSCPServer::ResetChannel(uint uiS
2997      dmsg(2,("LSCPServer: ResetChannel(SamplerChannel=%d)\n", uiSamplerChannel));      dmsg(2,("LSCPServer: ResetChannel(SamplerChannel=%d)\n", uiSamplerChannel));
2998      LSCPResultSet result;      LSCPResultSet result;
2999      try {      try {
3000          SamplerChannel* pSamplerChannel = pSampler->GetSamplerChannel(uiSamplerChannel);          EngineChannel* pEngineChannel = GetEngineChannel(uiSamplerChannel);
         if (!pSamplerChannel) throw LinuxSamplerException("Invalid sampler channel number " + ToString(uiSamplerChannel));  
         EngineChannel* pEngineChannel = pSamplerChannel->GetEngineChannel();  
         if (!pEngineChannel) throw LinuxSamplerException("No engine type assigned to sampler channel");  
3001          pEngineChannel->Reset();          pEngineChannel->Reset();
3002      }      }
3003      catch (LinuxSamplerException e) {      catch (Exception e) {
3004           result.Error(e);           result.Error(e);
3005      }      }
3006      return result.Produce();      return result.Produce();
# Line 1545  String LSCPServer::ResetSampler() { Line 3022  String LSCPServer::ResetSampler() {
3022   */   */
3023  String LSCPServer::GetServerInfo() {  String LSCPServer::GetServerInfo() {
3024      dmsg(2,("LSCPServer: GetServerInfo()\n"));      dmsg(2,("LSCPServer: GetServerInfo()\n"));
3025        const std::string description =
3026            _escapeLscpResponse("LinuxSampler - modular, streaming capable sampler");
3027      LSCPResultSet result;      LSCPResultSet result;
3028      result.Add("DESCRIPTION", "LinuxSampler - modular, streaming capable sampler");      result.Add("DESCRIPTION", description);
3029      result.Add("VERSION", VERSION);      result.Add("VERSION", VERSION);
3030      result.Add("PROTOCOL_VERSION", "1.0");      result.Add("PROTOCOL_VERSION", ToString(LSCP_RELEASE_MAJOR) + "." + ToString(LSCP_RELEASE_MINOR));
3031    #if HAVE_SQLITE3
3032        result.Add("INSTRUMENTS_DB_SUPPORT", "yes");
3033    #else
3034        result.Add("INSTRUMENTS_DB_SUPPORT", "no");
3035    #endif
3036    
3037        return result.Produce();
3038    }
3039    
3040    /**
3041     * Will be called by the parser to return the current number of all active streams.
3042     */
3043    String LSCPServer::GetTotalStreamCount() {
3044        dmsg(2,("LSCPServer: GetTotalStreamCount()\n"));
3045        LSCPResultSet result;
3046        result.Add(pSampler->GetDiskStreamCount());
3047      return result.Produce();      return result.Produce();
3048  }  }
3049    
# Line 1568  String LSCPServer::GetTotalVoiceCount() Line 3063  String LSCPServer::GetTotalVoiceCount()
3063  String LSCPServer::GetTotalVoiceCountMax() {  String LSCPServer::GetTotalVoiceCountMax() {
3064      dmsg(2,("LSCPServer: GetTotalVoiceCountMax()\n"));      dmsg(2,("LSCPServer: GetTotalVoiceCountMax()\n"));
3065      LSCPResultSet result;      LSCPResultSet result;
3066      result.Add(EngineFactory::EngineInstances().size() * CONFIG_MAX_VOICES);      result.Add(EngineFactory::EngineInstances().size() * pSampler->GetGlobalMaxVoices());
3067      return result.Produce();      return result.Produce();
3068  }  }
3069    
3070  /**  /**
3071     * Will be called by the parser to return the sampler global maximum
3072     * allowed number of voices.
3073     */
3074    String LSCPServer::GetGlobalMaxVoices() {
3075        dmsg(2,("LSCPServer: GetGlobalMaxVoices()\n"));
3076        LSCPResultSet result;
3077        result.Add(pSampler->GetGlobalMaxVoices());
3078        return result.Produce();
3079    }
3080    
3081    /**
3082     * Will be called by the parser to set the sampler global maximum number of
3083     * voices.
3084     */
3085    String LSCPServer::SetGlobalMaxVoices(int iVoices) {
3086        dmsg(2,("LSCPServer: SetGlobalMaxVoices(%d)\n", iVoices));
3087        LSCPResultSet result;
3088        try {
3089            pSampler->SetGlobalMaxVoices(iVoices);
3090            LSCPServer::SendLSCPNotify(
3091                LSCPEvent(LSCPEvent::event_global_info, "VOICES", pSampler->GetGlobalMaxVoices())
3092            );
3093        } catch (Exception e) {
3094            result.Error(e);
3095        }
3096        return result.Produce();
3097    }
3098    
3099    /**
3100     * Will be called by the parser to return the sampler global maximum
3101     * allowed number of disk streams.
3102     */
3103    String LSCPServer::GetGlobalMaxStreams() {
3104        dmsg(2,("LSCPServer: GetGlobalMaxStreams()\n"));
3105        LSCPResultSet result;
3106        result.Add(pSampler->GetGlobalMaxStreams());
3107        return result.Produce();
3108    }
3109    
3110    /**
3111     * Will be called by the parser to set the sampler global maximum number of
3112     * disk streams.
3113     */
3114    String LSCPServer::SetGlobalMaxStreams(int iStreams) {
3115        dmsg(2,("LSCPServer: SetGlobalMaxStreams(%d)\n", iStreams));
3116        LSCPResultSet result;
3117        try {
3118            pSampler->SetGlobalMaxStreams(iStreams);
3119            LSCPServer::SendLSCPNotify(
3120                LSCPEvent(LSCPEvent::event_global_info, "STREAMS", pSampler->GetGlobalMaxStreams())
3121            );
3122        } catch (Exception e) {
3123            result.Error(e);
3124        }
3125        return result.Produce();
3126    }
3127    
3128    String LSCPServer::GetGlobalVolume() {
3129        LSCPResultSet result;
3130        result.Add(ToString(GLOBAL_VOLUME)); // see common/global.cpp
3131        return result.Produce();
3132    }
3133    
3134    String LSCPServer::SetGlobalVolume(double dVolume) {
3135        LSCPResultSet result;
3136        try {
3137            if (dVolume < 0) throw Exception("Volume may not be negative");
3138            GLOBAL_VOLUME = dVolume; // see common/global_private.cpp
3139            LSCPServer::SendLSCPNotify(LSCPEvent(LSCPEvent::event_global_info, "VOLUME", GLOBAL_VOLUME));
3140        } catch (Exception e) {
3141            result.Error(e);
3142        }
3143        return result.Produce();
3144    }
3145    
3146    String LSCPServer::GetFileInstruments(String Filename) {
3147        dmsg(2,("LSCPServer: GetFileInstruments(String Filename=%s)\n",Filename.c_str()));
3148        LSCPResultSet result;
3149        try {
3150            VerifyFile(Filename);
3151        } catch (Exception e) {
3152            result.Error(e);
3153            return result.Produce();
3154        }
3155        // try to find a sampler engine that can handle the file
3156        bool bFound = false;
3157        std::vector<String> engineTypes = EngineFactory::AvailableEngineTypes();
3158        for (int i = 0; !bFound && i < engineTypes.size(); i++) {
3159            Engine* pEngine = NULL;
3160            try {
3161                pEngine = EngineFactory::Create(engineTypes[i]);
3162                if (!pEngine) throw Exception("Internal error: could not create '" + engineTypes[i] + "' engine");
3163                InstrumentManager* pManager = pEngine->GetInstrumentManager();
3164                if (pManager) {
3165                    std::vector<InstrumentManager::instrument_id_t> IDs =
3166                        pManager->GetInstrumentFileContent(Filename);
3167                    // return the amount of instruments in the file
3168                    result.Add(IDs.size());
3169                    // no more need to ask other engine types
3170                    bFound = true;
3171                } else dmsg(1,("Warning: engine '%s' does not provide an instrument manager\n", engineTypes[i].c_str()));
3172            } catch (Exception e) {
3173                // NOOP, as exception is thrown if engine doesn't support file
3174            }
3175            if (pEngine) EngineFactory::Destroy(pEngine);
3176        }
3177    
3178        if (!bFound) result.Error("Unknown file format");
3179        return result.Produce();
3180    }
3181    
3182    String LSCPServer::ListFileInstruments(String Filename) {
3183        dmsg(2,("LSCPServer: ListFileInstruments(String Filename=%s)\n",Filename.c_str()));
3184        LSCPResultSet result;
3185        try {
3186            VerifyFile(Filename);
3187        } catch (Exception e) {
3188            result.Error(e);
3189            return result.Produce();
3190        }
3191        // try to find a sampler engine that can handle the file
3192        bool bFound = false;
3193        std::vector<String> engineTypes = EngineFactory::AvailableEngineTypes();
3194        for (int i = 0; !bFound && i < engineTypes.size(); i++) {
3195            Engine* pEngine = NULL;
3196            try {
3197                pEngine = EngineFactory::Create(engineTypes[i]);
3198                if (!pEngine) throw Exception("Internal error: could not create '" + engineTypes[i] + "' engine");
3199                InstrumentManager* pManager = pEngine->GetInstrumentManager();
3200                if (pManager) {
3201                    std::vector<InstrumentManager::instrument_id_t> IDs =
3202                        pManager->GetInstrumentFileContent(Filename);
3203                    // return a list of IDs of the instruments in the file
3204                    String s;
3205                    for (int j = 0; j < IDs.size(); j++) {
3206                        if (s.size()) s += ",";
3207                        s += ToString(IDs[j].Index);
3208                    }
3209                    result.Add(s);
3210                    // no more need to ask other engine types
3211                    bFound = true;
3212                } else dmsg(1,("Warning: engine '%s' does not provide an instrument manager\n", engineTypes[i].c_str()));
3213            } catch (Exception e) {
3214                // NOOP, as exception is thrown if engine doesn't support file
3215            }
3216            if (pEngine) EngineFactory::Destroy(pEngine);
3217        }
3218    
3219        if (!bFound) result.Error("Unknown file format");
3220        return result.Produce();
3221    }
3222    
3223    String LSCPServer::GetFileInstrumentInfo(String Filename, uint InstrumentID) {
3224        dmsg(2,("LSCPServer: GetFileInstrumentInfo(String Filename=%s, InstrumentID=%d)\n",Filename.c_str(),InstrumentID));
3225        LSCPResultSet result;
3226        try {
3227            VerifyFile(Filename);
3228        } catch (Exception e) {
3229            result.Error(e);
3230            return result.Produce();
3231        }
3232        InstrumentManager::instrument_id_t id;
3233        id.FileName = Filename;
3234        id.Index    = InstrumentID;
3235        // try to find a sampler engine that can handle the file
3236        bool bFound = false;
3237        bool bFatalErr = false;
3238        std::vector<String> engineTypes = EngineFactory::AvailableEngineTypes();
3239        for (int i = 0; !bFound && !bFatalErr && i < engineTypes.size(); i++) {
3240            Engine* pEngine = NULL;
3241            try {
3242                pEngine = EngineFactory::Create(engineTypes[i]);
3243                if (!pEngine) throw Exception("Internal error: could not create '" + engineTypes[i] + "' engine");
3244                InstrumentManager* pManager = pEngine->GetInstrumentManager();
3245                if (pManager) {
3246                    // check if the instrument index is valid
3247                    // FIXME: this won't work if an engine only supports parts of the instrument file
3248                    std::vector<InstrumentManager::instrument_id_t> IDs =
3249                        pManager->GetInstrumentFileContent(Filename);
3250                    if (std::find(IDs.begin(), IDs.end(), id) == IDs.end()) {
3251                        std::stringstream ss;
3252                        ss << "Invalid instrument index " << InstrumentID << " for instrument file '" << Filename << "'";
3253                        bFatalErr = true;
3254                        throw Exception(ss.str());
3255                    }
3256                    // get the info of the requested instrument
3257                    InstrumentManager::instrument_info_t info =
3258                        pManager->GetInstrumentInfo(id);
3259                    // return detailed informations about the file
3260                    result.Add("NAME", info.InstrumentName);
3261                    result.Add("FORMAT_FAMILY", engineTypes[i]);
3262                    result.Add("FORMAT_VERSION", info.FormatVersion);
3263                    result.Add("PRODUCT", info.Product);
3264                    result.Add("ARTISTS", info.Artists);
3265    
3266                    std::stringstream ss;
3267                    bool b = false;
3268                    for (int i = 0; i < 128; i++) {
3269                        if (info.KeyBindings[i]) {
3270                            if (b) ss << ',';
3271                            ss << i; b = true;
3272                        }
3273                    }
3274                    result.Add("KEY_BINDINGS", ss.str());
3275    
3276                    b = false;
3277                    std::stringstream ss2;
3278                    for (int i = 0; i < 128; i++) {
3279                        if (info.KeySwitchBindings[i]) {
3280                            if (b) ss2 << ',';
3281                            ss2 << i; b = true;
3282                        }
3283                    }
3284                    result.Add("KEYSWITCH_BINDINGS", ss2.str());
3285                    // no more need to ask other engine types
3286                    bFound = true;
3287                } else dmsg(1,("Warning: engine '%s' does not provide an instrument manager\n", engineTypes[i].c_str()));
3288            } catch (Exception e) {
3289                // usually NOOP, as exception is thrown if engine doesn't support file
3290                if (bFatalErr) result.Error(e);
3291            }
3292            if (pEngine) EngineFactory::Destroy(pEngine);
3293        }
3294    
3295        if (!bFound && !bFatalErr) result.Error("Unknown file format");
3296        return result.Produce();
3297    }
3298    
3299    void LSCPServer::VerifyFile(String Filename) {
3300        #if WIN32
3301        WIN32_FIND_DATA win32FileAttributeData;
3302        BOOL res = GetFileAttributesEx( Filename.c_str(), GetFileExInfoStandard, &win32FileAttributeData );
3303        if (!res) {
3304            std::stringstream ss;
3305            ss << "File does not exist, GetFileAttributesEx failed `" << Filename << "`: Error " << GetLastError();
3306            throw Exception(ss.str());
3307        }
3308        if ( win32FileAttributeData.dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY ) {
3309            throw Exception("Directory is specified");
3310        }
3311        #else
3312        File f(Filename);
3313        if(!f.Exist()) throw Exception(f.GetErrorMsg());
3314        if (f.IsDirectory()) throw Exception("Directory is specified");
3315        #endif
3316    }
3317    
3318    /**
3319   * Will be called by the parser to subscribe a client (frontend) on the   * Will be called by the parser to subscribe a client (frontend) on the
3320   * server for receiving event messages.   * server for receiving event messages.
3321   */   */
# Line 1598  String LSCPServer::UnsubscribeNotificati Line 3341  String LSCPServer::UnsubscribeNotificati
3341      return result.Produce();      return result.Produce();
3342  }  }
3343    
3344  static int select_callback(void * lscpResultSet, int argc,  String LSCPServer::AddDbInstrumentDirectory(String Dir) {
3345                          char **argv, char **azColName)      dmsg(2,("LSCPServer: AddDbInstrumentDirectory(Dir=%s)\n", Dir.c_str()));
3346  {      LSCPResultSet result;
3347      LSCPResultSet* resultSet = (LSCPResultSet*) lscpResultSet;  #if HAVE_SQLITE3
3348      resultSet->Add(argc, argv);      try {
3349      return 0;          InstrumentsDb::GetInstrumentsDb()->AddDirectory(Dir);
3350        } catch (Exception e) {
3351             result.Error(e);
3352        }
3353    #else
3354        result.Error(String(DOESNT_HAVE_SQLITE3), 0);
3355    #endif
3356        return result.Produce();
3357  }  }
3358    
3359  String LSCPServer::QueryDatabase(String query) {  String LSCPServer::RemoveDbInstrumentDirectory(String Dir, bool Force) {
3360        dmsg(2,("LSCPServer: RemoveDbInstrumentDirectory(Dir=%s,Force=%d)\n", Dir.c_str(), Force));
3361      LSCPResultSet result;      LSCPResultSet result;
3362  #if HAVE_SQLITE3  #if HAVE_SQLITE3
3363      char* zErrMsg = NULL;      try {
3364      sqlite3 *db;          InstrumentsDb::GetInstrumentsDb()->RemoveDirectory(Dir, Force);
3365      String selectStr = "SELECT " + query;      } catch (Exception e) {
3366             result.Error(e);
3367        }
3368    #else
3369        result.Error(String(DOESNT_HAVE_SQLITE3), 0);
3370    #endif
3371        return result.Produce();
3372    }
3373    
3374      int rc = sqlite3_open("linuxsampler.db", &db);  String LSCPServer::GetDbInstrumentDirectoryCount(String Dir, bool Recursive) {
3375      if (rc == SQLITE_OK)      dmsg(2,("LSCPServer: GetDbInstrumentDirectoryCount(Dir=%s,Recursive=%d)\n", Dir.c_str(), Recursive));
3376      {      LSCPResultSet result;
3377              rc = sqlite3_exec(db, selectStr.c_str(), select_callback, &result, &zErrMsg);  #if HAVE_SQLITE3
3378        try {
3379            result.Add(InstrumentsDb::GetInstrumentsDb()->GetDirectoryCount(Dir, Recursive));
3380        } catch (Exception e) {
3381             result.Error(e);
3382      }      }
3383      if ( rc != SQLITE_OK )  #else
3384      {      result.Error(String(DOESNT_HAVE_SQLITE3), 0);
3385              result.Error(String(zErrMsg), rc);  #endif
3386        return result.Produce();
3387    }
3388    
3389    String LSCPServer::GetDbInstrumentDirectories(String Dir, bool Recursive) {
3390        dmsg(2,("LSCPServer: GetDbInstrumentDirectories(Dir=%s,Recursive=%d)\n", Dir.c_str(), Recursive));
3391        LSCPResultSet result;
3392    #if HAVE_SQLITE3
3393        try {
3394            String list;
3395            StringListPtr dirs = InstrumentsDb::GetInstrumentsDb()->GetDirectories(Dir, Recursive);
3396    
3397            for (int i = 0; i < dirs->size(); i++) {
3398                if (list != "") list += ",";
3399                list += "'" + InstrumentsDb::toEscapedPath(dirs->at(i)) + "'";
3400            }
3401    
3402            result.Add(list);
3403        } catch (Exception e) {
3404             result.Error(e);
3405        }
3406    #else
3407        result.Error(String(DOESNT_HAVE_SQLITE3), 0);
3408    #endif
3409        return result.Produce();
3410    }
3411    
3412    String LSCPServer::GetDbInstrumentDirectoryInfo(String Dir) {
3413        dmsg(2,("LSCPServer: GetDbInstrumentDirectoryInfo(Dir=%s)\n", Dir.c_str()));
3414        LSCPResultSet result;
3415    #if HAVE_SQLITE3
3416        try {
3417            DbDirectory info = InstrumentsDb::GetInstrumentsDb()->GetDirectoryInfo(Dir);
3418    
3419            result.Add("DESCRIPTION", _escapeLscpResponse(info.Description));
3420            result.Add("CREATED", info.Created);
3421            result.Add("MODIFIED", info.Modified);
3422        } catch (Exception e) {
3423             result.Error(e);
3424        }
3425    #else
3426        result.Error(String(DOESNT_HAVE_SQLITE3), 0);
3427    #endif
3428        return result.Produce();
3429    }
3430    
3431    String LSCPServer::SetDbInstrumentDirectoryName(String Dir, String Name) {
3432        dmsg(2,("LSCPServer: SetDbInstrumentDirectoryName(Dir=%s,Name=%s)\n", Dir.c_str(), Name.c_str()));
3433        LSCPResultSet result;
3434    #if HAVE_SQLITE3
3435        try {
3436            InstrumentsDb::GetInstrumentsDb()->RenameDirectory(Dir, Name);
3437        } catch (Exception e) {
3438             result.Error(e);
3439        }
3440    #else
3441        result.Error(String(DOESNT_HAVE_SQLITE3), 0);
3442    #endif
3443        return result.Produce();
3444    }
3445    
3446    String LSCPServer::MoveDbInstrumentDirectory(String Dir, String Dst) {
3447        dmsg(2,("LSCPServer: MoveDbInstrumentDirectory(Dir=%s,Dst=%s)\n", Dir.c_str(), Dst.c_str()));
3448        LSCPResultSet result;
3449    #if HAVE_SQLITE3
3450        try {
3451            InstrumentsDb::GetInstrumentsDb()->MoveDirectory(Dir, Dst);
3452        } catch (Exception e) {
3453             result.Error(e);
3454        }
3455    #else
3456        result.Error(String(DOESNT_HAVE_SQLITE3), 0);
3457    #endif
3458        return result.Produce();
3459    }
3460    
3461    String LSCPServer::CopyDbInstrumentDirectory(String Dir, String Dst) {
3462        dmsg(2,("LSCPServer: CopyDbInstrumentDirectory(Dir=%s,Dst=%s)\n", Dir.c_str(), Dst.c_str()));
3463        LSCPResultSet result;
3464    #if HAVE_SQLITE3
3465        try {
3466            InstrumentsDb::GetInstrumentsDb()->CopyDirectory(Dir, Dst);
3467        } catch (Exception e) {
3468             result.Error(e);
3469        }
3470    #else
3471        result.Error(String(DOESNT_HAVE_SQLITE3), 0);
3472    #endif
3473        return result.Produce();
3474    }
3475    
3476    String LSCPServer::SetDbInstrumentDirectoryDescription(String Dir, String Desc) {
3477        dmsg(2,("LSCPServer: SetDbInstrumentDirectoryDescription(Dir=%s,Desc=%s)\n", Dir.c_str(), Desc.c_str()));
3478        LSCPResultSet result;
3479    #if HAVE_SQLITE3
3480        try {
3481            InstrumentsDb::GetInstrumentsDb()->SetDirectoryDescription(Dir, Desc);
3482        } catch (Exception e) {
3483             result.Error(e);
3484        }
3485    #else
3486        result.Error(String(DOESNT_HAVE_SQLITE3), 0);
3487    #endif
3488        return result.Produce();
3489    }
3490    
3491    String LSCPServer::AddDbInstruments(String DbDir, String FilePath, int Index, bool bBackground) {
3492        dmsg(2,("LSCPServer: AddDbInstruments(DbDir=%s,FilePath=%s,Index=%d,bBackground=%d)\n", DbDir.c_str(), FilePath.c_str(), Index, bBackground));
3493        LSCPResultSet result;
3494    #if HAVE_SQLITE3
3495        try {
3496            int id;
3497            InstrumentsDb* db = InstrumentsDb::GetInstrumentsDb();
3498            id = db->AddInstruments(DbDir, FilePath, Index, bBackground);
3499            if (bBackground) result = id;
3500        } catch (Exception e) {
3501             result.Error(e);
3502        }
3503    #else
3504        result.Error(String(DOESNT_HAVE_SQLITE3), 0);
3505    #endif
3506        return result.Produce();
3507    }
3508    
3509    String LSCPServer::AddDbInstruments(String ScanMode, String DbDir, String FsDir, bool bBackground, bool insDir) {
3510        dmsg(2,("LSCPServer: AddDbInstruments(ScanMode=%s,DbDir=%s,FsDir=%s,bBackground=%d,insDir=%d)\n", ScanMode.c_str(), DbDir.c_str(), FsDir.c_str(), bBackground, insDir));
3511        LSCPResultSet result;
3512    #if HAVE_SQLITE3
3513        try {
3514            int id;
3515            InstrumentsDb* db = InstrumentsDb::GetInstrumentsDb();
3516            if (ScanMode.compare("RECURSIVE") == 0) {
3517                id = db->AddInstruments(RECURSIVE, DbDir, FsDir, bBackground, insDir);
3518            } else if (ScanMode.compare("NON_RECURSIVE") == 0) {
3519                id = db->AddInstruments(NON_RECURSIVE, DbDir, FsDir, bBackground, insDir);
3520            } else if (ScanMode.compare("FLAT") == 0) {
3521                id = db->AddInstruments(FLAT, DbDir, FsDir, bBackground, insDir);
3522            } else {
3523                throw Exception("Unknown scan mode: " + ScanMode);
3524            }
3525    
3526            if (bBackground) result = id;
3527        } catch (Exception e) {
3528             result.Error(e);
3529        }
3530    #else
3531        result.Error(String(DOESNT_HAVE_SQLITE3), 0);
3532    #endif
3533        return result.Produce();
3534    }
3535    
3536    String LSCPServer::RemoveDbInstrument(String Instr) {
3537        dmsg(2,("LSCPServer: RemoveDbInstrument(Instr=%s)\n", Instr.c_str()));
3538        LSCPResultSet result;
3539    #if HAVE_SQLITE3
3540        try {
3541            InstrumentsDb::GetInstrumentsDb()->RemoveInstrument(Instr);
3542        } catch (Exception e) {
3543             result.Error(e);
3544        }
3545    #else
3546        result.Error(String(DOESNT_HAVE_SQLITE3), 0);
3547    #endif
3548        return result.Produce();
3549    }
3550    
3551    String LSCPServer::GetDbInstrumentCount(String Dir, bool Recursive) {
3552        dmsg(2,("LSCPServer: GetDbInstrumentCount(Dir=%s,Recursive=%d)\n", Dir.c_str(), Recursive));
3553        LSCPResultSet result;
3554    #if HAVE_SQLITE3
3555        try {
3556            result.Add(InstrumentsDb::GetInstrumentsDb()->GetInstrumentCount(Dir, Recursive));
3557        } catch (Exception e) {
3558             result.Error(e);
3559        }
3560    #else
3561        result.Error(String(DOESNT_HAVE_SQLITE3), 0);
3562    #endif
3563        return result.Produce();
3564    }
3565    
3566    String LSCPServer::GetDbInstruments(String Dir, bool Recursive) {
3567        dmsg(2,("LSCPServer: GetDbInstruments(Dir=%s,Recursive=%d)\n", Dir.c_str(), Recursive));
3568        LSCPResultSet result;
3569    #if HAVE_SQLITE3
3570        try {
3571            String list;
3572            StringListPtr instrs = InstrumentsDb::GetInstrumentsDb()->GetInstruments(Dir, Recursive);
3573    
3574            for (int i = 0; i < instrs->size(); i++) {
3575                if (list != "") list += ",";
3576                list += "'" + InstrumentsDb::toEscapedPath(instrs->at(i)) + "'";
3577            }
3578    
3579            result.Add(list);
3580        } catch (Exception e) {
3581             result.Error(e);
3582        }
3583    #else
3584        result.Error(String(DOESNT_HAVE_SQLITE3), 0);
3585    #endif
3586        return result.Produce();
3587    }
3588    
3589    String LSCPServer::GetDbInstrumentInfo(String Instr) {
3590        dmsg(2,("LSCPServer: GetDbInstrumentInfo(Instr=%s)\n", Instr.c_str()));
3591        LSCPResultSet result;
3592    #if HAVE_SQLITE3
3593        try {
3594            DbInstrument info = InstrumentsDb::GetInstrumentsDb()->GetInstrumentInfo(Instr);
3595    
3596            result.Add("INSTRUMENT_FILE", info.InstrFile);
3597            result.Add("INSTRUMENT_NR", info.InstrNr);
3598            result.Add("FORMAT_FAMILY", info.FormatFamily);
3599            result.Add("FORMAT_VERSION", info.FormatVersion);
3600            result.Add("SIZE", (int)info.Size);
3601            result.Add("CREATED", info.Created);
3602            result.Add("MODIFIED", info.Modified);
3603            result.Add("DESCRIPTION", _escapeLscpResponse(info.Description));
3604            result.Add("IS_DRUM", info.IsDrum);
3605            result.Add("PRODUCT", _escapeLscpResponse(info.Product));
3606            result.Add("ARTISTS", _escapeLscpResponse(info.Artists));
3607            result.Add("KEYWORDS", _escapeLscpResponse(info.Keywords));
3608        } catch (Exception e) {
3609             result.Error(e);
3610        }
3611    #else
3612        result.Error(String(DOESNT_HAVE_SQLITE3), 0);
3613    #endif
3614        return result.Produce();
3615    }
3616    
3617    String LSCPServer::GetDbInstrumentsJobInfo(int JobId) {
3618        dmsg(2,("LSCPServer: GetDbInstrumentsJobInfo(JobId=%d)\n", JobId));
3619        LSCPResultSet result;
3620    #if HAVE_SQLITE3
3621        try {
3622            ScanJob job = InstrumentsDb::GetInstrumentsDb()->Jobs.GetJobById(JobId);
3623    
3624            result.Add("FILES_TOTAL", job.FilesTotal);
3625            result.Add("FILES_SCANNED", job.FilesScanned);
3626            result.Add("SCANNING", job.Scanning);
3627            result.Add("STATUS", job.Status);
3628        } catch (Exception e) {
3629             result.Error(e);
3630      }      }
     sqlite3_close(db);  
3631  #else  #else
3632      result.Error(String("SQLITE3 was not installed when linuxsampler was built. SELECT statement is not available."), 0);      result.Error(String(DOESNT_HAVE_SQLITE3), 0);
3633  #endif  #endif
3634      return result.Produce();      return result.Produce();
3635  }  }
3636    
3637    String LSCPServer::SetDbInstrumentName(String Instr, String Name) {
3638        dmsg(2,("LSCPServer: SetDbInstrumentName(Instr=%s,Name=%s)\n", Instr.c_str(), Name.c_str()));
3639        LSCPResultSet result;
3640    #if HAVE_SQLITE3
3641        try {
3642            InstrumentsDb::GetInstrumentsDb()->RenameInstrument(Instr, Name);
3643        } catch (Exception e) {
3644             result.Error(e);
3645        }
3646    #else
3647        result.Error(String(DOESNT_HAVE_SQLITE3), 0);
3648    #endif
3649        return result.Produce();
3650    }
3651    
3652    String LSCPServer::MoveDbInstrument(String Instr, String Dst) {
3653        dmsg(2,("LSCPServer: MoveDbInstrument(Instr=%s,Dst=%s)\n", Instr.c_str(), Dst.c_str()));
3654        LSCPResultSet result;
3655    #if HAVE_SQLITE3
3656        try {
3657            InstrumentsDb::GetInstrumentsDb()->MoveInstrument(Instr, Dst);
3658        } catch (Exception e) {
3659             result.Error(e);
3660        }
3661    #else
3662        result.Error(String(DOESNT_HAVE_SQLITE3), 0);
3663    #endif
3664        return result.Produce();
3665    }
3666    
3667    String LSCPServer::CopyDbInstrument(String Instr, String Dst) {
3668        dmsg(2,("LSCPServer: CopyDbInstrument(Instr=%s,Dst=%s)\n", Instr.c_str(), Dst.c_str()));
3669        LSCPResultSet result;
3670    #if HAVE_SQLITE3
3671        try {
3672            InstrumentsDb::GetInstrumentsDb()->CopyInstrument(Instr, Dst);
3673        } catch (Exception e) {
3674             result.Error(e);
3675        }
3676    #else
3677        result.Error(String(DOESNT_HAVE_SQLITE3), 0);
3678    #endif
3679        return result.Produce();
3680    }
3681    
3682    String LSCPServer::SetDbInstrumentDescription(String Instr, String Desc) {
3683        dmsg(2,("LSCPServer: SetDbInstrumentDescription(Instr=%s,Desc=%s)\n", Instr.c_str(), Desc.c_str()));
3684        LSCPResultSet result;
3685    #if HAVE_SQLITE3
3686        try {
3687            InstrumentsDb::GetInstrumentsDb()->SetInstrumentDescription(Instr, Desc);
3688        } catch (Exception e) {
3689             result.Error(e);
3690        }
3691    #else
3692        result.Error(String(DOESNT_HAVE_SQLITE3), 0);
3693    #endif
3694        return result.Produce();
3695    }
3696    
3697    String LSCPServer::SetDbInstrumentFilePath(String OldPath, String NewPath) {
3698        dmsg(2,("LSCPServer: SetDbInstrumentFilePath(OldPath=%s,NewPath=%s)\n", OldPath.c_str(), NewPath.c_str()));
3699        LSCPResultSet result;
3700    #if HAVE_SQLITE3
3701        try {
3702            InstrumentsDb::GetInstrumentsDb()->SetInstrumentFilePath(OldPath, NewPath);
3703        } catch (Exception e) {
3704             result.Error(e);
3705        }
3706    #else
3707        result.Error(String(DOESNT_HAVE_SQLITE3), 0);
3708    #endif
3709        return result.Produce();
3710    }
3711    
3712    String LSCPServer::FindLostDbInstrumentFiles() {
3713        dmsg(2,("LSCPServer: FindLostDbInstrumentFiles()\n"));
3714        LSCPResultSet result;
3715    #if HAVE_SQLITE3
3716        try {
3717            String list;
3718            StringListPtr pLostFiles = InstrumentsDb::GetInstrumentsDb()->FindLostInstrumentFiles();
3719    
3720            for (int i = 0; i < pLostFiles->size(); i++) {
3721                if (list != "") list += ",";
3722                list += "'" + pLostFiles->at(i) + "'";
3723            }
3724    
3725            result.Add(list);
3726        } catch (Exception e) {
3727             result.Error(e);
3728        }
3729    #else
3730        result.Error(String(DOESNT_HAVE_SQLITE3), 0);
3731    #endif
3732        return result.Produce();
3733    }
3734    
3735    String LSCPServer::FindDbInstrumentDirectories(String Dir, std::map<String,String> Parameters, bool Recursive) {
3736        dmsg(2,("LSCPServer: FindDbInstrumentDirectories(Dir=%s)\n", Dir.c_str()));
3737        LSCPResultSet result;
3738    #if HAVE_SQLITE3
3739        try {
3740            SearchQuery Query;
3741            std::map<String,String>::iterator iter;
3742            for (iter = Parameters.begin(); iter != Parameters.end(); iter++) {
3743                if (iter->first.compare("NAME") == 0) {
3744                    Query.Name = iter->second;
3745                } else if (iter->first.compare("CREATED") == 0) {
3746                    Query.SetCreated(iter->second);
3747                } else if (iter->first.compare("MODIFIED") == 0) {
3748                    Query.SetModified(iter->second);
3749                } else if (iter->first.compare("DESCRIPTION") == 0) {
3750                    Query.Description = iter->second;
3751                } else {
3752                    throw Exception("Unknown search criteria: " + iter->first);
3753                }
3754            }
3755    
3756            String list;
3757            StringListPtr pDirectories =
3758                InstrumentsDb::GetInstrumentsDb()->FindDirectories(Dir, &Query, Recursive);
3759    
3760            for (int i = 0; i < pDirectories->size(); i++) {
3761                if (list != "") list += ",";
3762                list += "'" + InstrumentsDb::toEscapedPath(pDirectories->at(i)) + "'";
3763            }
3764    
3765            result.Add(list);
3766        } catch (Exception e) {
3767             result.Error(e);
3768        }
3769    #else
3770        result.Error(String(DOESNT_HAVE_SQLITE3), 0);
3771    #endif
3772        return result.Produce();
3773    }
3774    
3775    String LSCPServer::FindDbInstruments(String Dir, std::map<String,String> Parameters, bool Recursive) {
3776        dmsg(2,("LSCPServer: FindDbInstruments(Dir=%s)\n", Dir.c_str()));
3777        LSCPResultSet result;
3778    #if HAVE_SQLITE3
3779        try {
3780            SearchQuery Query;
3781            std::map<String,String>::iterator iter;
3782            for (iter = Parameters.begin(); iter != Parameters.end(); iter++) {
3783                if (iter->first.compare("NAME") == 0) {
3784                    Query.Name = iter->second;
3785                } else if (iter->first.compare("FORMAT_FAMILIES") == 0) {
3786                    Query.SetFormatFamilies(iter->second);
3787                } else if (iter->first.compare("SIZE") == 0) {
3788                    Query.SetSize(iter->second);
3789                } else if (iter->first.compare("CREATED") == 0) {
3790                    Query.SetCreated(iter->second);
3791                } else if (iter->first.compare("MODIFIED") == 0) {
3792                    Query.SetModified(iter->second);
3793                } else if (iter->first.compare("DESCRIPTION") == 0) {
3794                    Query.Description = iter->second;
3795                } else if (iter->first.compare("IS_DRUM") == 0) {
3796                    if (!strcasecmp(iter->second.c_str(), "true")) {
3797                        Query.InstrType = SearchQuery::DRUM;
3798                    } else {
3799                        Query.InstrType = SearchQuery::CHROMATIC;
3800                    }
3801                } else if (iter->first.compare("PRODUCT") == 0) {
3802                     Query.Product = iter->second;
3803                } else if (iter->first.compare("ARTISTS") == 0) {
3804                     Query.Artists = iter->second;
3805                } else if (iter->first.compare("KEYWORDS") == 0) {
3806                     Query.Keywords = iter->second;
3807                } else {
3808                    throw Exception("Unknown search criteria: " + iter->first);
3809                }
3810            }
3811    
3812            String list;
3813            StringListPtr pInstruments =
3814                InstrumentsDb::GetInstrumentsDb()->FindInstruments(Dir, &Query, Recursive);
3815    
3816            for (int i = 0; i < pInstruments->size(); i++) {
3817                if (list != "") list += ",";
3818                list += "'" + InstrumentsDb::toEscapedPath(pInstruments->at(i)) + "'";
3819            }
3820    
3821            result.Add(list);
3822        } catch (Exception e) {
3823             result.Error(e);
3824        }
3825    #else
3826        result.Error(String(DOESNT_HAVE_SQLITE3), 0);
3827    #endif
3828        return result.Produce();
3829    }
3830    
3831    String LSCPServer::FormatInstrumentsDb() {
3832        dmsg(2,("LSCPServer: FormatInstrumentsDb()\n"));
3833        LSCPResultSet result;
3834    #if HAVE_SQLITE3
3835        try {
3836            InstrumentsDb::GetInstrumentsDb()->Format();
3837        } catch (Exception e) {
3838             result.Error(e);
3839        }
3840    #else
3841        result.Error(String(DOESNT_HAVE_SQLITE3), 0);
3842    #endif
3843        return result.Produce();
3844    }
3845    
3846    
3847  /**  /**
3848   * Will be called by the parser to enable or disable echo mode; if echo   * Will be called by the parser to enable or disable echo mode; if echo
3849   * mode is enabled, all commands from the client will (immediately) be   * mode is enabled, all commands from the client will (immediately) be
# Line 1640  String LSCPServer::SetEcho(yyparse_param Line 3855  String LSCPServer::SetEcho(yyparse_param
3855      try {      try {
3856          if      (boolean_value == 0) pSession->bVerbose = false;          if      (boolean_value == 0) pSession->bVerbose = false;
3857          else if (boolean_value == 1) pSession->bVerbose = true;          else if (boolean_value == 1) pSession->bVerbose = true;
3858          else throw LinuxSamplerException("Not a boolean value, must either be 0 or 1");          else throw Exception("Not a boolean value, must either be 0 or 1");
3859      }      }
3860      catch (LinuxSamplerException e) {      catch (Exception e) {
3861           result.Error(e);           result.Error(e);
3862      }      }
3863      return result.Produce();      return result.Produce();
3864  }  }
3865    
3866    }

Legend:
Removed from v.793  
changed lines
  Added in v.2375

  ViewVC Help
Powered by ViewVC