/[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 556 by schoenebeck, Sat May 21 01:10:12 2005 UTC revision 707 by iliev, Thu Jul 21 08:38:15 2005 UTC
# Line 33  Line 33 
33  #endif  #endif
34    
35  #include "../engines/EngineFactory.h"  #include "../engines/EngineFactory.h"
36    #include "../engines/EngineChannelFactory.h"
37  #include "../drivers/audio/AudioOutputDeviceFactory.h"  #include "../drivers/audio/AudioOutputDeviceFactory.h"
38  #include "../drivers/midi/MidiInputDeviceFactory.h"  #include "../drivers/midi/MidiInputDeviceFactory.h"
39    
# Line 60  Mutex LSCPServer::NotifyBufferMutex = Mu Line 61  Mutex LSCPServer::NotifyBufferMutex = Mu
61  Mutex LSCPServer::SubscriptionMutex = Mutex();  Mutex LSCPServer::SubscriptionMutex = Mutex();
62  Mutex LSCPServer::RTNotifyMutex = Mutex();  Mutex LSCPServer::RTNotifyMutex = Mutex();
63    
64  LSCPServer::LSCPServer(Sampler* pSampler) : Thread(true, false, 0, -4) {  LSCPServer::LSCPServer(Sampler* pSampler, long int addr, short int port) : Thread(true, false, 0, -4) {
65        SocketAddress.sin_family      = AF_INET;
66        SocketAddress.sin_addr.s_addr = addr;
67        SocketAddress.sin_port        = port;
68      this->pSampler = pSampler;      this->pSampler = pSampler;
69      LSCPEvent::RegisterEvent(LSCPEvent::event_channel_count, "CHANNEL_COUNT");      LSCPEvent::RegisterEvent(LSCPEvent::event_channel_count, "CHANNEL_COUNT");
70      LSCPEvent::RegisterEvent(LSCPEvent::event_voice_count, "VOICE_COUNT");      LSCPEvent::RegisterEvent(LSCPEvent::event_voice_count, "VOICE_COUNT");
# Line 97  int LSCPServer::Main() { Line 101  int LSCPServer::Main() {
101          exit(EXIT_FAILURE);          exit(EXIT_FAILURE);
102      }      }
103    
     SocketAddress.sin_family      = AF_INET;  
     SocketAddress.sin_port        = htons(LSCP_PORT);  
     SocketAddress.sin_addr.s_addr = htonl(INADDR_ANY);  
   
104      if (bind(hSocket, (sockaddr*) &SocketAddress, sizeof(sockaddr_in)) < 0) {      if (bind(hSocket, (sockaddr*) &SocketAddress, sizeof(sockaddr_in)) < 0) {
105          std::cerr << "LSCPServer: Could not bind server socket, retrying for " << ToString(LSCP_SERVER_BIND_TIMEOUT) << " seconds...";          std::cerr << "LSCPServer: Could not bind server socket, retrying for " << ToString(LSCP_SERVER_BIND_TIMEOUT) << " seconds...";
106          for (int trial = 0; true; trial++) { // retry for LSCP_SERVER_BIND_TIMEOUT seconds          for (int trial = 0; true; trial++) { // retry for LSCP_SERVER_BIND_TIMEOUT seconds
# Line 190  int LSCPServer::Main() { Line 190  int LSCPServer::Main() {
190                  }                  }
191          }          }
192    
193            // check if some engine channel's parameter / status changed, if so notify the respective LSCP event subscribers
194            {
195                std::set<EngineChannel*> engineChannels = EngineChannelFactory::EngineChannelInstances();
196                std::set<EngineChannel*>::iterator itEngineChannel = engineChannels.begin();
197                std::set<EngineChannel*>::iterator itEnd           = engineChannels.end();
198                for (; itEngineChannel != itEnd; ++itEngineChannel) {
199                    if ((*itEngineChannel)->StatusChanged()) {
200                        SendLSCPNotify(LSCPEvent(LSCPEvent::event_channel_info, (*itEngineChannel)->iSamplerChannelIndex));
201                    }
202                }
203            }
204    
205          //Now let's deliver late notifies (if any)          //Now let's deliver late notifies (if any)
206          NotifyBufferMutex.Lock();          NotifyBufferMutex.Lock();
207          for (std::map<int,String>::iterator iterNotify = bufferedNotifies.begin(); iterNotify != bufferedNotifies.end(); iterNotify++) {          for (std::map<int,String>::iterator iterNotify = bufferedNotifies.begin(); iterNotify != bufferedNotifies.end(); iterNotify++) {
208                  send(iterNotify->first, iterNotify->second.c_str(), iterNotify->second.size(), 0);                  send(iterNotify->first, iterNotify->second.c_str(), iterNotify->second.size(), MSG_NOSIGNAL);
209                  bufferedNotifies.erase(iterNotify);                  bufferedNotifies.erase(iterNotify);
210          }          }
211          NotifyBufferMutex.Unlock();          NotifyBufferMutex.Unlock();
# Line 243  void LSCPServer::SendLSCPNotify( LSCPEve Line 255  void LSCPServer::SendLSCPNotify( LSCPEve
255          while (true) {          while (true) {
256                  if (NotifyMutex.Trylock()) {                  if (NotifyMutex.Trylock()) {
257                          for(;iter != end; iter++)                          for(;iter != end; iter++)
258                                  send(*iter, notify.c_str(), notify.size(), 0);                                  send(*iter, notify.c_str(), notify.size(), MSG_NOSIGNAL);
259                          NotifyMutex.Unlock();                          NotifyMutex.Unlock();
260                          break;                          break;
261                  } else {                  } else {
# Line 352  void LSCPServer::AnswerClient(String Ret Line 364  void LSCPServer::AnswerClient(String Ret
364      dmsg(2,("LSCPServer::AnswerClient(ReturnMessage=%s)", ReturnMessage.c_str()));      dmsg(2,("LSCPServer::AnswerClient(ReturnMessage=%s)", ReturnMessage.c_str()));
365      if (currentSocket != -1) {      if (currentSocket != -1) {
366              NotifyMutex.Lock();              NotifyMutex.Lock();
367              send(currentSocket, ReturnMessage.c_str(), ReturnMessage.size(), 0);              send(currentSocket, ReturnMessage.c_str(), ReturnMessage.size(), MSG_NOSIGNAL);
368              NotifyMutex.Unlock();              NotifyMutex.Unlock();
369      }      }
370  }  }
# Line 485  String LSCPServer::LoadInstrument(String Line 497  String LSCPServer::LoadInstrument(String
497   * sampler channel.   * sampler channel.
498   */   */
499  String LSCPServer::SetEngineType(String EngineName, uint uiSamplerChannel) {  String LSCPServer::SetEngineType(String EngineName, uint uiSamplerChannel) {
500      dmsg(2,("LSCPServer: LoadEngine(EngineName=%s,SamplerChannel=%d)\n", EngineName.c_str(), uiSamplerChannel));      dmsg(2,("LSCPServer: SetEngineType(EngineName=%s,uiSamplerChannel=%d)\n", EngineName.c_str(), uiSamplerChannel));
501      LSCPResultSet result;      LSCPResultSet result;
502      try {      try {
503          SamplerChannel* pSamplerChannel = pSampler->GetSamplerChannel(uiSamplerChannel);          SamplerChannel* pSamplerChannel = pSampler->GetSamplerChannel(uiSamplerChannel);
504          if (!pSamplerChannel) throw LinuxSamplerException("Invalid sampler channel number " + ToString(uiSamplerChannel));          if (!pSamplerChannel) throw LinuxSamplerException("Invalid sampler channel number " + ToString(uiSamplerChannel));
505          LockRTNotify();          LockRTNotify();
506          pSamplerChannel->SetEngineType(EngineName);          pSamplerChannel->SetEngineType(EngineName);
507            if(HasSoloChannel()) pSamplerChannel->GetEngineChannel()->SetMute(-1);
508          UnlockRTNotify();          UnlockRTNotify();
509      }      }
510      catch (LinuxSamplerException e) {      catch (LinuxSamplerException e) {
# Line 578  String LSCPServer::GetEngineInfo(String Line 591  String LSCPServer::GetEngineInfo(String
591          Engine* pEngine = EngineFactory::Create(EngineName);          Engine* pEngine = EngineFactory::Create(EngineName);
592          result.Add("DESCRIPTION", pEngine->Description());          result.Add("DESCRIPTION", pEngine->Description());
593          result.Add("VERSION",     pEngine->Version());          result.Add("VERSION",     pEngine->Version());
594          delete pEngine;          EngineFactory::Destroy(pEngine);
595      }      }
596      catch (LinuxSamplerException e) {      catch (LinuxSamplerException e) {
597           result.Error(e);           result.Error(e);
# Line 607  String LSCPServer::GetChannelInfo(uint u Line 620  String LSCPServer::GetChannelInfo(uint u
620          int InstrumentStatus = -1;          int InstrumentStatus = -1;
621          int AudioOutputChannels = 0;          int AudioOutputChannels = 0;
622          String AudioRouting;          String AudioRouting;
623            int Mute = 0;
624            bool Solo = false;
625    
626          if (pEngineChannel) {          if (pEngineChannel) {
627              EngineName          = pEngineChannel->EngineName();              EngineName          = pEngineChannel->EngineName();
# Line 622  String LSCPServer::GetChannelInfo(uint u Line 637  String LSCPServer::GetChannelInfo(uint u
637                  if (AudioRouting != "") AudioRouting += ",";                  if (AudioRouting != "") AudioRouting += ",";
638                  AudioRouting += ToString(pEngineChannel->OutputChannel(chan));                  AudioRouting += ToString(pEngineChannel->OutputChannel(chan));
639              }              }
640                Mute = pEngineChannel->GetMute();
641                Solo = pEngineChannel->GetSolo();
642          }          }
643    
644          result.Add("ENGINE_NAME", EngineName);          result.Add("ENGINE_NAME", EngineName);
# Line 634  String LSCPServer::GetChannelInfo(uint u Line 651  String LSCPServer::GetChannelInfo(uint u
651    
652          result.Add("MIDI_INPUT_DEVICE", GetMidiInputDeviceIndex(pSamplerChannel->GetMidiInputDevice()));          result.Add("MIDI_INPUT_DEVICE", GetMidiInputDeviceIndex(pSamplerChannel->GetMidiInputDevice()));
653          result.Add("MIDI_INPUT_PORT", pSamplerChannel->GetMidiInputPort());          result.Add("MIDI_INPUT_PORT", pSamplerChannel->GetMidiInputPort());
654          if (pSamplerChannel->GetMidiInputChannel() == MidiInputPort::midi_chan_all) result.Add("MIDI_INPUT_CHANNEL", "ALL");          if (pSamplerChannel->GetMidiInputChannel() == midi_chan_all) result.Add("MIDI_INPUT_CHANNEL", "ALL");
655          else result.Add("MIDI_INPUT_CHANNEL", pSamplerChannel->GetMidiInputChannel());          else result.Add("MIDI_INPUT_CHANNEL", pSamplerChannel->GetMidiInputChannel());
656    
657          result.Add("INSTRUMENT_FILE", InstrumentFileName);          result.Add("INSTRUMENT_FILE", InstrumentFileName);
658          result.Add("INSTRUMENT_NR", InstrumentIndex);          result.Add("INSTRUMENT_NR", InstrumentIndex);
659          result.Add("INSTRUMENT_NAME", InstrumentName);          result.Add("INSTRUMENT_NAME", InstrumentName);
660          result.Add("INSTRUMENT_STATUS", InstrumentStatus);          result.Add("INSTRUMENT_STATUS", InstrumentStatus);
661            result.Add("MUTE", Mute == -1 ? "MUTED_BY_SOLO" : (Mute ? "true" : "false"));
662            result.Add("SOLO", Solo);
663      }      }
664      catch (LinuxSamplerException e) {      catch (LinuxSamplerException e) {
665           result.Error(e);           result.Error(e);
# Line 660  String LSCPServer::GetVoiceCount(uint ui Line 679  String LSCPServer::GetVoiceCount(uint ui
679          if (!pSamplerChannel) throw LinuxSamplerException("Invalid sampler channel number " + ToString(uiSamplerChannel));          if (!pSamplerChannel) throw LinuxSamplerException("Invalid sampler channel number " + ToString(uiSamplerChannel));
680          EngineChannel* pEngineChannel = pSamplerChannel->GetEngineChannel();          EngineChannel* pEngineChannel = pSamplerChannel->GetEngineChannel();
681          if (!pEngineChannel) throw LinuxSamplerException("No engine loaded on sampler channel");          if (!pEngineChannel) throw LinuxSamplerException("No engine loaded on sampler channel");
682            if (!pEngineChannel->GetEngine()) throw LinuxSamplerException("No audio output device connected to sampler channel");
683          result.Add(pEngineChannel->GetEngine()->VoiceCount());          result.Add(pEngineChannel->GetEngine()->VoiceCount());
684      }      }
685      catch (LinuxSamplerException e) {      catch (LinuxSamplerException e) {
# Line 680  String LSCPServer::GetStreamCount(uint u Line 700  String LSCPServer::GetStreamCount(uint u
700          if (!pSamplerChannel) throw LinuxSamplerException("Invalid sampler channel number " + ToString(uiSamplerChannel));          if (!pSamplerChannel) throw LinuxSamplerException("Invalid sampler channel number " + ToString(uiSamplerChannel));
701          EngineChannel* pEngineChannel = pSamplerChannel->GetEngineChannel();          EngineChannel* pEngineChannel = pSamplerChannel->GetEngineChannel();
702          if (!pEngineChannel) throw LinuxSamplerException("No engine type assigned to sampler channel");          if (!pEngineChannel) throw LinuxSamplerException("No engine type assigned to sampler channel");
703            if (!pEngineChannel->GetEngine()) throw LinuxSamplerException("No audio output device connected to sampler channel");
704          result.Add(pEngineChannel->GetEngine()->DiskStreamCount());          result.Add(pEngineChannel->GetEngine()->DiskStreamCount());
705      }      }
706      catch (LinuxSamplerException e) {      catch (LinuxSamplerException e) {
# Line 700  String LSCPServer::GetBufferFill(fill_re Line 721  String LSCPServer::GetBufferFill(fill_re
721          if (!pSamplerChannel) throw LinuxSamplerException("Invalid sampler channel number " + ToString(uiSamplerChannel));          if (!pSamplerChannel) throw LinuxSamplerException("Invalid sampler channel number " + ToString(uiSamplerChannel));
722          EngineChannel* pEngineChannel = pSamplerChannel->GetEngineChannel();          EngineChannel* pEngineChannel = pSamplerChannel->GetEngineChannel();
723          if (!pEngineChannel) throw LinuxSamplerException("No engine type assigned to sampler channel");          if (!pEngineChannel) throw LinuxSamplerException("No engine type assigned to sampler channel");
724            if (!pEngineChannel->GetEngine()) throw LinuxSamplerException("No audio output device connected to sampler channel");
725          if (!pEngineChannel->GetEngine()->DiskStreamSupported()) result.Add("NA");          if (!pEngineChannel->GetEngine()->DiskStreamSupported()) result.Add("NA");
726          else {          else {
727              switch (ResponseType) {              switch (ResponseType) {
# Line 1275  String LSCPServer::SetMIDIInputChannel(u Line 1297  String LSCPServer::SetMIDIInputChannel(u
1297      try {      try {
1298          SamplerChannel* pSamplerChannel = pSampler->GetSamplerChannel(uiSamplerChannel);          SamplerChannel* pSamplerChannel = pSampler->GetSamplerChannel(uiSamplerChannel);
1299          if (!pSamplerChannel) throw LinuxSamplerException("Invalid sampler channel number " + ToString(uiSamplerChannel));          if (!pSamplerChannel) throw LinuxSamplerException("Invalid sampler channel number " + ToString(uiSamplerChannel));
1300          pSamplerChannel->SetMidiInputChannel((MidiInputPort::midi_chan_t) MIDIChannel);          pSamplerChannel->SetMidiInputChannel((midi_chan_t) MIDIChannel);
1301      }      }
1302      catch (LinuxSamplerException e) {      catch (LinuxSamplerException e) {
1303           result.Error(e);           result.Error(e);
# Line 1352  String LSCPServer::SetMIDIInput(uint MID Line 1374  String LSCPServer::SetMIDIInput(uint MID
1374          std::map<uint, MidiInputDevice*> devices =  pSampler->GetMidiInputDevices();          std::map<uint, MidiInputDevice*> devices =  pSampler->GetMidiInputDevices();
1375          if (!devices.count(MIDIDeviceId)) throw LinuxSamplerException("There is no MIDI input device with index " + ToString(MIDIDeviceId));          if (!devices.count(MIDIDeviceId)) throw LinuxSamplerException("There is no MIDI input device with index " + ToString(MIDIDeviceId));
1376          MidiInputDevice* pDevice = devices[MIDIDeviceId];          MidiInputDevice* pDevice = devices[MIDIDeviceId];
1377          pSamplerChannel->SetMidiInput(pDevice, MIDIPort, (MidiInputPort::midi_chan_t) MIDIChannel);          pSamplerChannel->SetMidiInput(pDevice, MIDIPort, (midi_chan_t) MIDIChannel);
1378      }      }
1379      catch (LinuxSamplerException e) {      catch (LinuxSamplerException e) {
1380           result.Error(e);           result.Error(e);
# Line 1381  String LSCPServer::SetVolume(double dVol Line 1403  String LSCPServer::SetVolume(double dVol
1403  }  }
1404    
1405  /**  /**
1406     * Will be called by the parser to mute/unmute particular sampler channel.
1407     */
1408    String LSCPServer::SetChannelMute(bool bMute, uint uiSamplerChannel) {
1409        dmsg(2,("LSCPServer: SetChannelMute(bMute=%d,uiSamplerChannel=%d)\n",bMute,uiSamplerChannel));
1410        LSCPResultSet result;
1411        try {
1412            SamplerChannel* pSamplerChannel = pSampler->GetSamplerChannel(uiSamplerChannel);
1413            if (!pSamplerChannel) throw LinuxSamplerException("Invalid sampler channel number " + ToString(uiSamplerChannel));
1414    
1415            EngineChannel* pEngineChannel = pSamplerChannel->GetEngineChannel();
1416            if (!pEngineChannel) throw LinuxSamplerException("No engine type assigned to sampler channel");
1417    
1418            if(!bMute) pEngineChannel->SetMute((HasSoloChannel() && !pEngineChannel->GetSolo()) ? -1 : 0);
1419            else pEngineChannel->SetMute(1);
1420        } catch (LinuxSamplerException e) {
1421            result.Error(e);
1422        }
1423        return result.Produce();
1424    }
1425    
1426    /**
1427     * Will be called by the parser to solo particular sampler channel.
1428     */
1429    String LSCPServer::SetChannelSolo(bool bSolo, uint uiSamplerChannel) {
1430        dmsg(2,("LSCPServer: SetChannelSolo(bSolo=%d,uiSamplerChannel=%d)\n",bSolo,uiSamplerChannel));
1431        LSCPResultSet result;
1432        try {
1433            SamplerChannel* pSamplerChannel = pSampler->GetSamplerChannel(uiSamplerChannel);
1434            if (!pSamplerChannel) throw LinuxSamplerException("Invalid sampler channel number " + ToString(uiSamplerChannel));
1435    
1436            EngineChannel* pEngineChannel = pSamplerChannel->GetEngineChannel();
1437            if (!pEngineChannel) throw LinuxSamplerException("No engine type assigned to sampler channel");
1438    
1439            bool oldSolo = pEngineChannel->GetSolo();
1440            bool hadSoloChannel = HasSoloChannel();
1441            
1442            pEngineChannel->SetSolo(bSolo);
1443            
1444            if(!oldSolo && bSolo) {
1445                if(pEngineChannel->GetMute() == -1) pEngineChannel->SetMute(0);
1446                if(!hadSoloChannel) MuteNonSoloChannels();
1447            }
1448            
1449            if(oldSolo && !bSolo) {
1450                if(!HasSoloChannel()) UnmuteChannels();
1451                else if(!pEngineChannel->GetMute()) pEngineChannel->SetMute(-1);
1452            }
1453        } catch (LinuxSamplerException e) {
1454            result.Error(e);
1455        }
1456        return result.Produce();
1457    }
1458    
1459    /**
1460     * Determines whether there is at least one solo channel in the channel list.
1461     *
1462     * @returns true if there is at least one solo channel in the channel list,
1463     * false otherwise.
1464     */
1465    bool LSCPServer::HasSoloChannel() {
1466        std::map<uint,SamplerChannel*> channels = pSampler->GetSamplerChannels();
1467        std::map<uint,SamplerChannel*>::iterator iter = channels.begin();
1468        for (; iter != channels.end(); iter++) {
1469            EngineChannel* c = iter->second->GetEngineChannel();
1470            if(c && c->GetSolo()) return true;
1471        }
1472    
1473        return false;
1474    }
1475    
1476    /**
1477     * Mutes all unmuted non-solo channels. Notice that the channels are muted
1478     * with -1 which indicates that they are muted because of the presence
1479     * of a solo channel(s). Channels muted with -1 will be automatically unmuted
1480     * when there are no solo channels left.
1481     */
1482    void LSCPServer::MuteNonSoloChannels() {
1483        dmsg(2,("LSCPServer: MuteNonSoloChannels()\n"));
1484        std::map<uint,SamplerChannel*> channels = pSampler->GetSamplerChannels();
1485        std::map<uint,SamplerChannel*>::iterator iter = channels.begin();
1486        for (; iter != channels.end(); iter++) {
1487            EngineChannel* c = iter->second->GetEngineChannel();
1488            if(c && !c->GetSolo() && !c->GetMute()) c->SetMute(-1);
1489        }
1490    }
1491    
1492    /**
1493     * Unmutes all channels that are muted because of the presence
1494     * of a solo channel(s).
1495     */
1496    void  LSCPServer::UnmuteChannels() {
1497        dmsg(2,("LSCPServer: UnmuteChannels()\n"));
1498        std::map<uint,SamplerChannel*> channels = pSampler->GetSamplerChannels();
1499        std::map<uint,SamplerChannel*>::iterator iter = channels.begin();
1500        for (; iter != channels.end(); iter++) {
1501            EngineChannel* c = iter->second->GetEngineChannel();
1502            if(c && c->GetMute() == -1) c->SetMute(0);
1503        }
1504    }
1505    
1506    /**
1507   * Will be called by the parser to reset a particular sampler channel.   * Will be called by the parser to reset a particular sampler channel.
1508   */   */
1509  String LSCPServer::ResetChannel(uint uiSamplerChannel) {  String LSCPServer::ResetChannel(uint uiSamplerChannel) {
# Line 1391  String LSCPServer::ResetChannel(uint uiS Line 1514  String LSCPServer::ResetChannel(uint uiS
1514          if (!pSamplerChannel) throw LinuxSamplerException("Invalid sampler channel number " + ToString(uiSamplerChannel));          if (!pSamplerChannel) throw LinuxSamplerException("Invalid sampler channel number " + ToString(uiSamplerChannel));
1515          EngineChannel* pEngineChannel = pSamplerChannel->GetEngineChannel();          EngineChannel* pEngineChannel = pSamplerChannel->GetEngineChannel();
1516          if (!pEngineChannel) throw LinuxSamplerException("No engine type assigned to sampler channel");          if (!pEngineChannel) throw LinuxSamplerException("No engine type assigned to sampler channel");
1517          pEngineChannel->GetEngine()->Reset();          pEngineChannel->Reset();
1518      }      }
1519      catch (LinuxSamplerException e) {      catch (LinuxSamplerException e) {
1520           result.Error(e);           result.Error(e);
# Line 1409  String LSCPServer::ResetSampler() { Line 1532  String LSCPServer::ResetSampler() {
1532      return result.Produce();      return result.Produce();
1533  }  }
1534    
1535    /**
1536     * Will be called by the parser to return general informations about this
1537     * sampler.
1538     */
1539    String LSCPServer::GetServerInfo() {
1540        dmsg(2,("LSCPServer: GetServerInfo()\n"));
1541        LSCPResultSet result;
1542        result.Add("DESCRIPTION", "LinuxSampler - modular, streaming capable sampler");
1543        result.Add("VERSION", VERSION);
1544        result.Add("PROTOCOL_VERSION", "1.0");
1545        return result.Produce();
1546    }
1547    
1548  /**  /**
1549   * 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
1550   * server for receiving event messages.   * server for receiving event messages.

Legend:
Removed from v.556  
changed lines
  Added in v.707

  ViewVC Help
Powered by ViewVC