/[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 1135 by iliev, Thu Mar 29 09:40:45 2007 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 - 2007 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;
# Line 81  LSCPServer::LSCPServer(Sampler* pSampler Line 127  LSCPServer::LSCPServer(Sampler* pSampler
127      LSCPEvent::RegisterEvent(LSCPEvent::event_midi_instr_map_info, "MIDI_INSTRUMENT_MAP_INFO");      LSCPEvent::RegisterEvent(LSCPEvent::event_midi_instr_map_info, "MIDI_INSTRUMENT_MAP_INFO");
128      LSCPEvent::RegisterEvent(LSCPEvent::event_midi_instr_count, "MIDI_INSTRUMENT_COUNT");      LSCPEvent::RegisterEvent(LSCPEvent::event_midi_instr_count, "MIDI_INSTRUMENT_COUNT");
129      LSCPEvent::RegisterEvent(LSCPEvent::event_midi_instr_info, "MIDI_INSTRUMENT_INFO");      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");      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) {  void LSCPServer::EventHandler::ChannelCountChanged(int NewCount) {
170      LSCPServer::SendLSCPNotify(LSCPEvent(LSCPEvent::event_channel_count, NewCount));      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) {  void LSCPServer::EventHandler::AudioDeviceCountChanged(int NewCount) {
216      LSCPServer::SendLSCPNotify(LSCPEvent(LSCPEvent::event_audio_device_count, NewCount));      LSCPServer::SendLSCPNotify(LSCPEvent(LSCPEvent::event_audio_device_count, NewCount));
217  }  }
# Line 103  void LSCPServer::EventHandler::MidiDevic Line 220  void LSCPServer::EventHandler::MidiDevic
220      LSCPServer::SendLSCPNotify(LSCPEvent(LSCPEvent::event_midi_device_count, NewCount));      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) {  void LSCPServer::EventHandler::MidiInstrumentCountChanged(int MapId, int NewCount) {
272      LSCPServer::SendLSCPNotify(LSCPEvent(LSCPEvent::event_midi_instr_count, MapId, NewCount));      LSCPServer::SendLSCPNotify(LSCPEvent(LSCPEvent::event_midi_instr_count, MapId, NewCount));
273  }  }
# Line 139  void LSCPServer::EventHandler::TotalVoic Line 304  void LSCPServer::EventHandler::TotalVoic
304      LSCPServer::SendLSCPNotify(LSCPEvent(LSCPEvent::event_total_voice_count, NewCount));      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  /**  /**
365   * Blocks the calling thread until the LSCP Server is initialized and   * Blocks the calling thread until the LSCP Server is initialized and
# Line 155  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 168  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 180  int LSCPServer::Main() { Line 414  int LSCPServer::Main() {
414    
415      listen(hSocket, 1);      listen(hSocket, 1);
416      Initialized.Set(true);      Initialized.Set(true);
417        
418      // Registering event listeners      // Registering event listeners
419      pSampler->AddChannelCountListener(&eventHandler);      pSampler->AddChannelCountListener(&eventHandler);
420      pSampler->AddAudioDeviceCountListener(&eventHandler);      pSampler->AddAudioDeviceCountListener(&eventHandler);
# Line 188  int LSCPServer::Main() { Line 422  int LSCPServer::Main() {
422      pSampler->AddVoiceCountListener(&eventHandler);      pSampler->AddVoiceCountListener(&eventHandler);
423      pSampler->AddStreamCountListener(&eventHandler);      pSampler->AddStreamCountListener(&eventHandler);
424      pSampler->AddBufferFillListener(&eventHandler);      pSampler->AddBufferFillListener(&eventHandler);
425        pSampler->AddTotalStreamCountListener(&eventHandler);
426      pSampler->AddTotalVoiceCountListener(&eventHandler);      pSampler->AddTotalVoiceCountListener(&eventHandler);
427      pSampler->AddFxSendCountListener(&eventHandler);      pSampler->AddFxSendCountListener(&eventHandler);
428      MidiInstrumentMapper::AddMidiInstrumentCountListener(&eventHandler);      MidiInstrumentMapper::AddMidiInstrumentCountListener(&eventHandler);
429      MidiInstrumentMapper::AddMidiInstrumentInfoListener(&eventHandler);      MidiInstrumentMapper::AddMidiInstrumentInfoListener(&eventHandler);
430      MidiInstrumentMapper::AddMidiInstrumentMapCountListener(&eventHandler);      MidiInstrumentMapper::AddMidiInstrumentMapCountListener(&eventHandler);
431      MidiInstrumentMapper::AddMidiInstrumentMapInfoListener(&eventHandler);      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 205  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++) {                  for (int i = 0; i < (*itEngineChannel)->GetFxSendCount(); i++) {
460                      FxSend* fxs = (*itEngineChannel)->GetFxSend(i);                      FxSend* fxs = (*itEngineChannel)->GetFxSend(i);
461                      if(fxs != NULL && fxs->IsInfoChanged()) {                      if(fxs != NULL && fxs->IsInfoChanged()) {
462                          int chn = (*itEngineChannel)->iSamplerChannelIndex;                          int chn = (*itEngineChannel)->GetSamplerChannel()->Index();
463                          LSCPServer::SendLSCPNotify(LSCPEvent(LSCPEvent::event_fx_send_info, chn, fxs->Id()));                          LSCPServer::SendLSCPNotify(LSCPEvent(LSCPEvent::event_fx_send_info, chn, fxs->Id()));
464                          fxs->SetInfoChanged(false);                          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          }          }
519    
520          //Now let's deliver late notifies (if any)          //Now let's deliver late notifies (if any)
# Line 244  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 260  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 287  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 320  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 385  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 395  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 410  bool LSCPServer::GetLSCPCommand( std::ve Line 750  bool LSCPServer::GetLSCPCommand( std::ve
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 448  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 572  EngineChannel* LSCPServer::GetEngineChan Line 923  EngineChannel* LSCPServer::GetEngineChan
923      EngineChannel* pEngineChannel = pSamplerChannel->GetEngineChannel();      EngineChannel* pEngineChannel = pSamplerChannel->GetEngineChannel();
924      if (!pEngineChannel) throw Exception("There is no engine deployed on this sampler channel yet");      if (!pEngineChannel) throw Exception("There is no engine deployed on this sampler channel yet");
925    
926      return pEngineChannel;              return pEngineChannel;
927  }  }
928    
929  /**  /**
# Line 721  String LSCPServer::GetEngineInfo(String Line 1072  String LSCPServer::GetEngineInfo(String
1072      LockRTNotify();      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      }      }
# Line 794  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);
# Line 816  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);
         if (!pSamplerChannel) throw Exception("Invalid sampler channel number " + ToString(uiSamplerChannel));  
         EngineChannel* pEngineChannel = pSamplerChannel->GetEngineChannel();  
         if (!pEngineChannel) throw Exception("No engine loaded on sampler channel");  
1182          if (!pEngineChannel->GetEngine()) throw Exception("No audio output device connected to sampler channel");          if (!pEngineChannel->GetEngine()) throw Exception("No audio output device connected to sampler channel");
1183          result.Add(pEngineChannel->GetEngine()->VoiceCount());          result.Add(pEngineChannel->GetEngine()->VoiceCount());
1184      }      }
# Line 837  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);
         if (!pSamplerChannel) throw Exception("Invalid sampler channel number " + ToString(uiSamplerChannel));  
         EngineChannel* pEngineChannel = pSamplerChannel->GetEngineChannel();  
         if (!pEngineChannel) throw Exception("No engine type assigned to sampler channel");  
1200          if (!pEngineChannel->GetEngine()) throw Exception("No audio output device connected to sampler channel");          if (!pEngineChannel->GetEngine()) throw Exception("No audio output device connected to sampler channel");
1201          result.Add(pEngineChannel->GetEngine()->DiskStreamCount());          result.Add(pEngineChannel->GetEngine()->DiskStreamCount());
1202      }      }
# Line 858  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);
         if (!pSamplerChannel) throw Exception("Invalid sampler channel number " + ToString(uiSamplerChannel));  
         EngineChannel* pEngineChannel = pSamplerChannel->GetEngineChannel();  
         if (!pEngineChannel) throw Exception("No engine type assigned to sampler channel");  
1218          if (!pEngineChannel->GetEngine()) throw Exception("No audio output device connected to sampler channel");          if (!pEngineChannel->GetEngine()) throw Exception("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 {
# Line 949  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          }          }
# Line 973  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          }          }
# Line 1003  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 (Exception e) {      catch (Exception e) {
1364          result.Error(e);          result.Error(e);
# Line 1030  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 (Exception e) {      catch (Exception e) {
1392          result.Error(e);          result.Error(e);
# Line 1496  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)
# Line 1539  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 Exception("Invalid sampler channel number " + ToString(uiSamplerChannel));  
         EngineChannel* pEngineChannel = pSamplerChannel->GetEngineChannel();  
         if (!pEngineChannel) throw Exception("No engine type assigned to sampler channel");  
1899          pEngineChannel->Volume(dVolume);          pEngineChannel->Volume(dVolume);
1900      }      }
1901      catch (Exception e) {      catch (Exception e) {
# Line 1558  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 Exception("Invalid sampler channel number " + ToString(uiSamplerChannel));  
   
         EngineChannel* pEngineChannel = pSamplerChannel->GetEngineChannel();  
         if (!pEngineChannel) throw Exception("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);
# Line 1579  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 Exception("Invalid sampler channel number " + ToString(uiSamplerChannel));  
   
         EngineChannel* pEngineChannel = pSamplerChannel->GetEngineChannel();  
         if (!pEngineChannel) throw Exception("No engine type assigned to sampler channel");  
1932    
1933          bool oldSolo = pEngineChannel->GetSolo();          bool oldSolo = pEngineChannel->GetSolo();
1934          bool hadSoloChannel = HasSoloChannel();          bool hadSoloChannel = HasSoloChannel();
# Line 1703  String LSCPServer::GetMidiInstrumentMapp Line 2048  String LSCPServer::GetMidiInstrumentMapp
2048      dmsg(2,("LSCPServer: GetMidiInstrumentMappings()\n"));      dmsg(2,("LSCPServer: GetMidiInstrumentMappings()\n"));
2049      LSCPResultSet result;      LSCPResultSet result;
2050      try {      try {
2051          result.Add(MidiInstrumentMapper::Entries(MidiMapID).size());          result.Add(MidiInstrumentMapper::GetInstrumentCount(MidiMapID));
2052      } catch (Exception e) {      } catch (Exception e) {
2053          result.Error(e);          result.Error(e);
2054      }      }
# Line 1714  String LSCPServer::GetMidiInstrumentMapp Line 2059  String LSCPServer::GetMidiInstrumentMapp
2059  String LSCPServer::GetAllMidiInstrumentMappings() {  String LSCPServer::GetAllMidiInstrumentMappings() {
2060      dmsg(2,("LSCPServer: GetAllMidiInstrumentMappings()\n"));      dmsg(2,("LSCPServer: GetAllMidiInstrumentMappings()\n"));
2061      LSCPResultSet result;      LSCPResultSet result;
2062      std::vector<int> maps = MidiInstrumentMapper::Maps();      try {
2063      int totalMappings = 0;          result.Add(MidiInstrumentMapper::GetInstrumentCount());
2064      for (int i = 0; i < maps.size(); i++) {      } catch (Exception e) {
2065          try {          result.Error(e);
             totalMappings += MidiInstrumentMapper::Entries(maps[i]).size();  
         } catch (Exception e) { /*NOOP*/ }  
2066      }      }
     result.Add(totalMappings);  
2067      return result.Produce();      return result.Produce();
2068  }  }
2069    
# Line 1729  String LSCPServer::GetMidiInstrumentMapp Line 2071  String LSCPServer::GetMidiInstrumentMapp
2071      dmsg(2,("LSCPServer: GetMidiIstrumentMapping()\n"));      dmsg(2,("LSCPServer: GetMidiIstrumentMapping()\n"));
2072      LSCPResultSet result;      LSCPResultSet result;
2073      try {      try {
2074          midi_prog_index_t idx;          MidiInstrumentMapper::entry_t entry = MidiInstrumentMapper::GetEntry(MidiMapID, MidiBank, MidiProg);
2075          idx.midi_bank_msb = (MidiBank >> 7) & 0x7f;          // convert the filename into the correct encoding as defined for LSCP
2076          idx.midi_bank_lsb = MidiBank & 0x7f;          // (especially in terms of special characters -> escape sequences)
2077          idx.midi_prog     = MidiProg;  #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          std::map<midi_prog_index_t,MidiInstrumentMapper::entry_t> mappings = MidiInstrumentMapper::Entries(MidiMapID);          result.Add("NAME", _escapeLscpResponse(entry.Name));
2085          std::map<midi_prog_index_t,MidiInstrumentMapper::entry_t>::iterator iter = mappings.find(idx);          result.Add("ENGINE_NAME", entry.EngineName);
2086          if (iter == mappings.end()) result.Error("there is no map entry with that index");          result.Add("INSTRUMENT_FILE", instrumentFileName);
2087          else { // found          result.Add("INSTRUMENT_NR", (int) entry.InstrumentIndex);
2088              result.Add("NAME", iter->second.Name);          String instrumentName;
2089              result.Add("ENGINE_NAME", iter->second.EngineName);          Engine* pEngine = EngineFactory::Create(entry.EngineName);
2090              result.Add("INSTRUMENT_FILE", iter->second.InstrumentFile);          if (pEngine) {
2091              result.Add("INSTRUMENT_NR", (int) iter->second.InstrumentIndex);              if (pEngine->GetInstrumentManager()) {
2092              String instrumentName;                  InstrumentManager::instrument_id_t instrID;
2093              Engine* pEngine = EngineFactory::Create(iter->second.EngineName);                  instrID.FileName = entry.InstrumentFile;
2094              if (pEngine) {                  instrID.Index    = entry.InstrumentIndex;
2095                  if (pEngine->GetInstrumentManager()) {                  instrumentName = pEngine->GetInstrumentManager()->GetInstrumentName(instrID);
                     InstrumentManager::instrument_id_t instrID;  
                     instrID.FileName = iter->second.InstrumentFile;  
                     instrID.Index    = iter->second.InstrumentIndex;  
                     instrumentName = pEngine->GetInstrumentManager()->GetInstrumentName(instrID);  
                 }  
                 EngineFactory::Destroy(pEngine);  
2096              }              }
2097              result.Add("INSTRUMENT_NAME", instrumentName);              EngineFactory::Destroy(pEngine);
             switch (iter->second.LoadMode) {  
                 case MidiInstrumentMapper::ON_DEMAND:  
                     result.Add("LOAD_MODE", "ON_DEMAND");  
                     break;  
                 case MidiInstrumentMapper::ON_DEMAND_HOLD:  
                     result.Add("LOAD_MODE", "ON_DEMAND_HOLD");  
                     break;  
                 case MidiInstrumentMapper::PERSISTENT:  
                     result.Add("LOAD_MODE", "PERSISTENT");  
                     break;  
                 default:  
                     throw Exception("entry reflects invalid LOAD_MODE, consider this as a bug!");  
             }  
             result.Add("VOLUME", iter->second.Volume);  
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) {      } catch (Exception e) {
2115          result.Error(e);          result.Error(e);
2116      }      }
# Line 1908  String LSCPServer::GetMidiInstrumentMap( Line 2250  String LSCPServer::GetMidiInstrumentMap(
2250      dmsg(2,("LSCPServer: GetMidiInstrumentMap()\n"));      dmsg(2,("LSCPServer: GetMidiInstrumentMap()\n"));
2251      LSCPResultSet result;      LSCPResultSet result;
2252      try {      try {
2253          result.Add("NAME", MidiInstrumentMapper::MapName(MidiMapID));          result.Add("NAME", _escapeLscpResponse(MidiInstrumentMapper::MapName(MidiMapID)));
2254          result.Add("DEFAULT", MidiInstrumentMapper::GetDefaultMap() == MidiMapID);          result.Add("DEFAULT", MidiInstrumentMapper::GetDefaultMap() == MidiMapID);
2255      } catch (Exception e) {      } catch (Exception e) {
2256          result.Error(e);          result.Error(e);
# Line 1939  String LSCPServer::SetChannelMap(uint ui Line 2281  String LSCPServer::SetChannelMap(uint ui
2281      dmsg(2,("LSCPServer: SetChannelMap()\n"));      dmsg(2,("LSCPServer: SetChannelMap()\n"));
2282      LSCPResultSet result;      LSCPResultSet result;
2283      try {      try {
2284          SamplerChannel* pSamplerChannel = pSampler->GetSamplerChannel(uiSamplerChannel);          EngineChannel* pEngineChannel = GetEngineChannel(uiSamplerChannel);
         if (!pSamplerChannel) throw Exception("Invalid sampler channel number " + ToString(uiSamplerChannel));  
   
         EngineChannel* pEngineChannel = pSamplerChannel->GetEngineChannel();  
         if (!pEngineChannel) throw Exception("There is no engine deployed on this sampler channel yet");  
2285    
2286          if      (MidiMapID == -1) pEngineChannel->SetMidiInstrumentMapToNone();          if      (MidiMapID == -1) pEngineChannel->SetMidiInstrumentMapToNone();
2287          else if (MidiMapID == -2) pEngineChannel->SetMidiInstrumentMapToDefault();          else if (MidiMapID == -2) pEngineChannel->SetMidiInstrumentMapToDefault();
# Line 1959  String LSCPServer::CreateFxSend(uint uiS Line 2297  String LSCPServer::CreateFxSend(uint uiS
2297      LSCPResultSet result;      LSCPResultSet result;
2298      try {      try {
2299          EngineChannel* pEngineChannel = GetEngineChannel(uiSamplerChannel);          EngineChannel* pEngineChannel = GetEngineChannel(uiSamplerChannel);
2300            
2301          FxSend* pFxSend = pEngineChannel->AddFxSend(MidiCtrl, Name);          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)");          if (!pFxSend) throw Exception("Could not add FxSend, don't ask, I don't know why (probably a bug)");
2303    
# Line 2043  String LSCPServer::GetFxSendInfo(uint ui Line 2381  String LSCPServer::GetFxSendInfo(uint ui
2381      try {      try {
2382          EngineChannel* pEngineChannel = GetEngineChannel(uiSamplerChannel);          EngineChannel* pEngineChannel = GetEngineChannel(uiSamplerChannel);
2383          FxSend* pFxSend = GetFxSend(uiSamplerChannel, FxSendID);          FxSend* pFxSend = GetFxSend(uiSamplerChannel, FxSendID);
2384            
2385          // gather audio routing informations          // gather audio routing informations
2386          String AudioRouting;          String AudioRouting;
2387          for (int chan = 0; chan < pEngineChannel->Channels(); chan++) {          for (int chan = 0; chan < pEngineChannel->Channels(); chan++) {
# Line 2051  String LSCPServer::GetFxSendInfo(uint ui Line 2389  String LSCPServer::GetFxSendInfo(uint ui
2389              AudioRouting += ToString(pFxSend->DestinationChannel(chan));              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          // success
2398          result.Add("NAME", pFxSend->Name());          result.Add("NAME", _escapeLscpResponse(pFxSend->Name()));
2399          result.Add("MIDI_CONTROLLER", pFxSend->MidiController());          result.Add("MIDI_CONTROLLER", pFxSend->MidiController());
2400          result.Add("LEVEL", ToString(pFxSend->Level()));          result.Add("LEVEL", ToString(pFxSend->Level()));
2401          result.Add("AUDIO_OUTPUT_ROUTING", AudioRouting);          result.Add("AUDIO_OUTPUT_ROUTING", AudioRouting);
2402            result.Add("EFFECT", sEffectRouting);
2403      } catch (Exception e) {      } catch (Exception e) {
2404          result.Error(e);          result.Error(e);
2405      }      }
# Line 2118  String LSCPServer::SetFxSendLevel(uint u Line 2462  String LSCPServer::SetFxSendLevel(uint u
2462      return result.Produce();      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 2125  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 Exception("Invalid sampler channel number " + ToString(uiSamplerChannel));  
         EngineChannel* pEngineChannel = pSamplerChannel->GetEngineChannel();  
         if (!pEngineChannel) throw Exception("No engine type assigned to sampler channel");  
3001          pEngineChannel->Reset();          pEngineChannel->Reset();
3002      }      }
3003      catch (Exception e) {      catch (Exception e) {
# Line 2153  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", ToString(LSCP_RELEASE_MAJOR) + "." + ToString(LSCP_RELEASE_MINOR));      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 2176  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();
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();      return result.Produce();
3126  }  }
3127    
# Line 2190  String LSCPServer::SetGlobalVolume(doubl Line 3135  String LSCPServer::SetGlobalVolume(doubl
3135      LSCPResultSet result;      LSCPResultSet result;
3136      try {      try {
3137          if (dVolume < 0) throw Exception("Volume may not be negative");          if (dVolume < 0) throw Exception("Volume may not be negative");
3138          GLOBAL_VOLUME = dVolume; // see common/global.cpp          GLOBAL_VOLUME = dVolume; // see common/global_private.cpp
3139          LSCPServer::SendLSCPNotify(LSCPEvent(LSCPEvent::event_global_info, "VOLUME", GLOBAL_VOLUME));          LSCPServer::SendLSCPNotify(LSCPEvent(LSCPEvent::event_global_info, "VOLUME", GLOBAL_VOLUME));
3140      } catch (Exception e) {      } catch (Exception e) {
3141          result.Error(e);          result.Error(e);
# Line 2198  String LSCPServer::SetGlobalVolume(doubl Line 3143  String LSCPServer::SetGlobalVolume(doubl
3143      return result.Produce();      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.
# Line 2224  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::RemoveDbInstrumentDirectory(String Dir, bool Force) {
3360        dmsg(2,("LSCPServer: RemoveDbInstrumentDirectory(Dir=%s,Force=%d)\n", Dir.c_str(), Force));
3361        LSCPResultSet result;
3362    #if HAVE_SQLITE3
3363        try {
3364            InstrumentsDb::GetInstrumentsDb()->RemoveDirectory(Dir, Force);
3365        } 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    String LSCPServer::GetDbInstrumentDirectoryCount(String Dir, bool Recursive) {
3375        dmsg(2,("LSCPServer: GetDbInstrumentDirectoryCount(Dir=%s,Recursive=%d)\n", Dir.c_str(), Recursive));
3376        LSCPResultSet result;
3377    #if HAVE_SQLITE3
3378        try {
3379            result.Add(InstrumentsDb::GetInstrumentsDb()->GetDirectoryCount(Dir, Recursive));
3380        } catch (Exception e) {
3381             result.Error(e);
3382        }
3383    #else
3384        result.Error(String(DOESNT_HAVE_SQLITE3), 0);
3385    #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::QueryDatabase(String query) {  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;      LSCPResultSet result;
3512  #if HAVE_SQLITE3  #if HAVE_SQLITE3
3513      char* zErrMsg = NULL;      try {
3514      sqlite3 *db;          int id;
3515      String selectStr = "SELECT " + query;          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      int rc = sqlite3_open("linuxsampler.db", &db);  String LSCPServer::GetDbInstrumentCount(String Dir, bool Recursive) {
3552      if (rc == SQLITE_OK)      dmsg(2,("LSCPServer: GetDbInstrumentCount(Dir=%s,Recursive=%d)\n", Dir.c_str(), Recursive));
3553      {      LSCPResultSet result;
3554              rc = sqlite3_exec(db, selectStr.c_str(), select_callback, &result, &zErrMsg);  #if HAVE_SQLITE3
3555        try {
3556            result.Add(InstrumentsDb::GetInstrumentsDb()->GetInstrumentCount(Dir, Recursive));
3557        } catch (Exception e) {
3558             result.Error(e);
3559      }      }
3560      if ( rc != SQLITE_OK )  #else
3561      {      result.Error(String(DOESNT_HAVE_SQLITE3), 0);
3562              result.Error(String(zErrMsg), rc);  #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      }      }
     sqlite3_close(db);  
3583  #else  #else
3584      result.Error(String("SQLITE3 was not installed when linuxsampler was built. SELECT statement is not available."), 0);      result.Error(String(DOESNT_HAVE_SQLITE3), 0);
3585  #endif  #endif
3586      return result.Produce();      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        }
3631    #else
3632        result.Error(String(DOESNT_HAVE_SQLITE3), 0);
3633    #endif
3634        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 2273  String LSCPServer::SetEcho(yyparse_param Line 3862  String LSCPServer::SetEcho(yyparse_param
3862      }      }
3863      return result.Produce();      return result.Produce();
3864  }  }
3865    
3866    }

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

  ViewVC Help
Powered by ViewVC