| 
							- /*
 -  * Carla UI bridge code
 -  * Copyright (C) 2011-2012 Filipe Coelho <falktx@falktx.com>
 -  *
 -  * This program is free software; you can redistribute it and/or modify
 -  * it under the terms of the GNU General Public License as published by
 -  * the Free Software Foundation; either version 2 of the License, or
 -  * any later version.
 -  *
 -  * This program is distributed in the hope that it will be useful,
 -  * but WITHOUT ANY WARRANTY; without even the implied warranty of
 -  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 -  * GNU General Public License for more details.
 -  *
 -  * For a full copy of the GNU General Public License see the COPYING file
 -  */
 - 
 - #ifdef BRIDGE_VST
 - 
 - #include "carla_bridge_client.h"
 - #include "carla_vst.h"
 - #include "carla_midi.h"
 - 
 - #include <QtCore/QTimerEvent>
 - #include <QtGui/QDialog>
 - 
 - #ifdef Q_WS_X11
 - #include <QtGui/QX11Info>
 - #endif
 - 
 - CARLA_BRIDGE_START_NAMESPACE
 - 
 - // -------------------------------------------------------------------------
 - 
 - // fake values
 - uint32_t bufferSize = 512;
 - double   sampleRate = 44100.0;
 - 
 - class CarlaVstClient : public CarlaClient, QObject
 - {
 - public:
 -     CarlaVstClient(CarlaToolkit* const toolkit)
 -         : CarlaClient(toolkit),
 -           QObject(nullptr)
 -     {
 -         effect = nullptr;
 - 
 -         idleTimer = 0;
 - 
 -         // make client valid
 -         unique1 = unique2 = rand();
 -     }
 - 
 -     ~CarlaVstClient()
 -     {
 -         // make client invalid
 -         unique2 += 1;
 -     }
 - 
 -     // ---------------------------------------------------------------------
 -     // ui initialization
 - 
 -     bool init(const char* binary, const char*)
 -     {
 -         // -----------------------------------------------------------------
 -         // init
 - 
 -         CarlaClient::init(binary, nullptr);
 - 
 -         // -----------------------------------------------------------------
 -         // open DLL
 - 
 -         if (! libOpen(binary))
 -         {
 -             qWarning("%s", libError());
 -             return false;
 -         }
 - 
 -         // -----------------------------------------------------------------
 -         // get DLL main entry
 - 
 -         VST_Function vstFn = (VST_Function)libSymbol("VSTPluginMain");
 - 
 -         if (! vstFn)
 -             vstFn = (VST_Function)libSymbol("main");
 - 
 -         if (! vstFn)
 -             return false;
 - 
 -         // -----------------------------------------------------------------
 -         // initialize plugin
 - 
 -         effect = vstFn(hostCallback);
 - 
 -         if (! (effect && effect->magic == kEffectMagic))
 -             return false;
 - 
 -         // -----------------------------------------------------------------
 -         // initialize VST stuff
 - 
 - #ifdef VESTIGE_HEADER
 -         effect->ptr1 = this;
 - #else
 -         effect->resvd1 = (intptr_t)this;
 - #endif
 - 
 -         int32_t value = 0;
 - #ifdef Q_WS_X11
 -         value = (int64_t)QX11Info::display();
 - #endif
 - 
 -         effect->dispatcher(effect, effOpen, 0, 0, nullptr, 0.0f);
 - #if ! VST_FORCE_DEPRECATED
 -         effect->dispatcher(effect, effSetBlockSizeAndSampleRate, 0, bufferSize, nullptr, sampleRate);
 - #endif
 -         effect->dispatcher(effect, effSetSampleRate, 0, 0, nullptr, sampleRate);
 -         effect->dispatcher(effect, effSetBlockSize, 0, bufferSize, nullptr, 0.0f);
 -         effect->dispatcher(effect, effSetProcessPrecision, 0, kVstProcessPrecision32, nullptr, 0.0f);
 - 
 -         if (effect->dispatcher(effect, effEditOpen, 0, value, getContainerId(), 0.0f) != 1)
 -         {
 -             effect->dispatcher(effect, effClose, 0, 0, nullptr, 0.0f);
 -             return false;
 -         }
 - 
 -         // -----------------------------------------------------------------
 -         // initialize gui stuff
 - 
 -         ERect* vstRect = nullptr;
 -         effect->dispatcher(effect, effEditGetRect, 0, 0, &vstRect, 0.0f);
 - 
 -         if (vstRect)
 -         {
 -             int width  = vstRect->right - vstRect->left;
 -             int height = vstRect->bottom - vstRect->top;
 - 
 -             if (width > 0 && height > 0)
 -                 toolkitResize(width, height);
 -         }
 - 
 -         idleTimer = startTimer(50);
 - 
 -         return true;
 -     }
 - 
 -     void close()
 -     {
 -         CarlaClient::close();
 - 
 -         if (effect)
 -         {
 -             effect->dispatcher(effect, effEditClose, 0, 0, nullptr, 0.0f);
 -             effect->dispatcher(effect, effClose, 0, 0, nullptr, 0.0f);
 -         }
 -     }
 - 
 -     // ---------------------------------------------------------------------
 -     // ui management
 - 
 -     void* getWidget() const
 -     {
 -         return nullptr; // VST always uses reparent
 -     }
 - 
 -     bool isResizable() const
 -     {
 -         return false;
 -     }
 - 
 -     bool needsReparent() const
 -     {
 -         return true;
 -     }
 - 
 -     // ---------------------------------------------------------------------
 -     // processing
 - 
 -     void setParameter(const int32_t rindex, const double value)
 -     {
 -         if (effect)
 -             effect->setParameter(effect, rindex, value);
 -     }
 - 
 -     void setProgram(const uint32_t index)
 -     {
 -         if (effect)
 -             effect->dispatcher(effect, effSetProgram, 0, index, nullptr, 0.0f);
 -     }
 - 
 -     void setMidiProgram(const uint32_t, const uint32_t)
 -     {
 -     }
 - 
 -     void noteOn(const uint8_t, const uint8_t, const uint8_t)
 -     {
 -     }
 - 
 -     void noteOff(const uint8_t, const uint8_t)
 -     {
 -     }
 - 
 -     // ---------------------------------------------------------------------
 - 
 -     void handleAudioMasterAutomate(const uint32_t index, const float value)
 -     {
 -         effect->setParameter(effect, index, value);
 -         sendOscControl(index, value);
 -     }
 - 
 -     intptr_t handleAudioMasterGetCurrentProcessLevel()
 -     {
 -         return kVstProcessLevelUser;
 -     }
 - 
 -     intptr_t handleAudioMasterProcessEvents(const VstEvents* const vstEvents)
 -     {
 -         for (int32_t i=0; i < vstEvents->numEvents; i++)
 -         {
 -             if (! vstEvents->events[i])
 -                 break;
 - 
 -             const VstMidiEvent* const vstMidiEvent = (const VstMidiEvent*)vstEvents->events[i];
 - 
 -             if (vstMidiEvent->type != kVstMidiType)
 -             {
 -                 uint8_t status = vstMidiEvent->midiData[0];
 - 
 -                 // Fix bad note-off
 -                 if (MIDI_IS_STATUS_NOTE_ON(status) && vstMidiEvent->midiData[2] == 0)
 -                     status -= 0x10;
 - 
 -                 uint8_t midiBuf[4] = { 0, status, (uint8_t)vstMidiEvent->midiData[1], (uint8_t)vstMidiEvent->midiData[2] };
 -                 sendOscMidi(midiBuf);
 -             }
 -         }
 - 
 -         return 1;
 -     }
 - 
 -     intptr_t handleAdioMasterSizeWindow(int32_t width, int32_t height)
 -     {
 -         toolkitResize(width, height);
 - 
 -         return 1;
 -     }
 - 
 -     void handleAudioMasterUpdateDisplay()
 -     {
 -         sendOscConfigure("reloadprograms", "");
 -     }
 - 
 -     // ---------------------------------------------------------------------
 - 
 -     static intptr_t hostCanDo(const char* const feature)
 -     {
 -         qDebug("CarlaVstClient::hostCanDo(\"%s\")", feature);
 - 
 -         if (strcmp(feature, "supplyIdle") == 0)
 -             return 1;
 -         if (strcmp(feature, "sendVstEvents") == 0)
 -             return 1;
 -         if (strcmp(feature, "sendVstMidiEvent") == 0)
 -             return 1;
 -         if (strcmp(feature, "sendVstMidiEventFlagIsRealtime") == 0)
 -             return -1;
 -         if (strcmp(feature, "sendVstTimeInfo") == 0)
 -             return 1;
 -         if (strcmp(feature, "receiveVstEvents") == 0)
 -             return 1;
 -         if (strcmp(feature, "receiveVstMidiEvent") == 0)
 -             return 1;
 -         if (strcmp(feature, "receiveVstTimeInfo") == 0)
 -             return -1;
 -         if (strcmp(feature, "reportConnectionChanges") == 0)
 -             return -1;
 -         if (strcmp(feature, "acceptIOChanges") == 0)
 -             return 1;
 -         if (strcmp(feature, "sizeWindow") == 0)
 -             return 1;
 -         if (strcmp(feature, "offline") == 0)
 -             return -1;
 -         if (strcmp(feature, "openFileSelector") == 0)
 -             return -1;
 -         if (strcmp(feature, "closeFileSelector") == 0)
 -             return -1;
 -         if (strcmp(feature, "startStopProcess") == 0)
 -             return 1;
 -         if (strcmp(feature, "supportShell") == 0)
 -             return -1;
 -         if (strcmp(feature, "shellCategory") == 0)
 -             return -1;
 - 
 -         // unimplemented
 -         qWarning("CarlaVstClient::hostCanDo(\"%s\") - unknown feature", feature);
 -         return 0;
 -     }
 - 
 -     static intptr_t VSTCALLBACK hostCallback(AEffect* const effect, const int32_t opcode, const int32_t index, const intptr_t value, void* const ptr, const float opt)
 -     {
 - #if DEBUG
 -         qDebug("CarlaVstClient::hostCallback(%p, %s, %i, " P_INTPTR ", %p, %f", effect, vstMasterOpcode2str(opcode), index, value, ptr, opt);
 - #endif
 - 
 -         // Check if 'resvd1' points to this client
 -         CarlaVstClient* self = nullptr;
 - 
 - #ifdef VESTIGE_HEADER
 -         if (effect && effect->ptr1)
 -         {
 -             self = (CarlaVstClient*)effect->ptr1;
 - #else
 -         if (effect && effect->resvd1)
 -         {
 -             self = (CarlaVstClient*)effect->resvd1;
 - #endif
 -             if (self->unique1 != self->unique2)
 -                 self = nullptr;
 -         }
 - 
 -         intptr_t ret = 0;
 - 
 -         switch (opcode)
 -         {
 -         case audioMasterAutomate:
 -             if (self)
 -                 self->handleAudioMasterAutomate(index, opt);
 -             break;
 - 
 -         case audioMasterVersion:
 -             ret = kVstVersion;
 -             break;
 - 
 -         case audioMasterCurrentId:
 -             // TODO
 -             break;
 - 
 -         case audioMasterIdle:
 -             //if (effect)
 -             //    effect->dispatcher(effect, effEditIdle, 0, 0, nullptr, 0.0f);
 -             break;
 - 
 -         case audioMasterGetTime:
 -             static VstTimeInfo_R timeInfo;
 -             memset(&timeInfo, 0, sizeof(VstTimeInfo_R));
 -             timeInfo.sampleRate = sampleRate;
 - 
 -             // Tempo
 -             timeInfo.tempo  = 120.0;
 -             timeInfo.flags |= kVstTempoValid;
 - 
 -             // Time Signature
 -             timeInfo.timeSigNumerator   = 4;
 -             timeInfo.timeSigDenominator = 4;
 -             timeInfo.flags |= kVstTimeSigValid;
 - 
 -             ret = (intptr_t)&timeInfo;
 -             break;
 - 
 -         case audioMasterProcessEvents:
 -             if (self && ptr)
 -                 ret = self->handleAudioMasterProcessEvents((const VstEvents*)ptr);
 -             break;
 - 
 - #if ! VST_FORCE_DEPRECATED
 -         case audioMasterTempoAt:
 -             // Deprecated in VST SDK 2.4
 -             ret = 120 * 10000;
 -             break;
 - #endif
 - 
 -         case audioMasterSizeWindow:
 -             if (self && index > 0 && value > 0)
 -                 ret = self->handleAdioMasterSizeWindow(index, value);
 -             break;
 - 
 -         case audioMasterGetSampleRate:
 -             ret = sampleRate;
 -             break;
 - 
 -         case audioMasterGetBlockSize:
 -             ret = bufferSize;
 -             break;
 - 
 -         case audioMasterGetCurrentProcessLevel:
 -             ret = kVstProcessLevelUser;
 -             break;
 - 
 -         case audioMasterGetAutomationState:
 -             ret = kVstAutomationReadWrite;
 -             break;
 - 
 -         case audioMasterGetVendorString:
 -             if (ptr)
 -                 strcpy((char*)ptr, "Cadence");
 -             break;
 - 
 -         case audioMasterGetProductString:
 -             if (ptr)
 -                 strcpy((char*)ptr, "Carla-Bridge");
 -             break;
 - 
 -         case audioMasterGetVendorVersion:
 -             ret = 0x050; // 0.5.0
 -             break;
 - 
 -         case audioMasterCanDo:
 -             if (ptr)
 -                 ret = hostCanDo((const char*)ptr);
 -             break;
 - 
 -         case audioMasterGetLanguage:
 -             ret = kVstLangEnglish;
 -             break;
 - 
 -         case audioMasterUpdateDisplay:
 -             if (self)
 -                 self->handleAudioMasterUpdateDisplay();
 -             break;
 - 
 -         default:
 - #ifdef DEBUG
 -             qDebug("CarlaVstClient::hostCallback(%p, %s, %i, " P_INTPTR ", %p, %f", effect, vstMasterOpcode2str(opcode), index, value, ptr, opt);
 - #endif
 -             break;
 -         }
 - 
 -         return ret;
 -     }
 - 
 - protected:
 -     void timerEvent(QTimerEvent* const event)
 -     {
 -         if (event->timerId() == idleTimer && effect)
 -         {
 -             effect->dispatcher(effect, effIdle, 0, 0, nullptr, 0.0f);
 -             effect->dispatcher(effect, effEditIdle, 0, 0, nullptr, 0.0f);
 -         }
 - 
 -         QObject::timerEvent(event);
 -     }
 - 
 - private:
 -     int unique1;
 -     AEffect* effect;
 -     int idleTimer;
 -     int unique2;
 - };
 - 
 - CARLA_BRIDGE_END_NAMESPACE
 - 
 - int main(int argc, char* argv[])
 - {
 -     using namespace CarlaBridge;
 - 
 -     if (argc != 4)
 -     {
 -         qWarning("usage: %s <osc-url|\"null\"> <binary> <ui-title>", argv[0]);
 -         return 1;
 -     }
 - 
 -     const char* oscUrl  = argv[1];
 -     const char* binary  = argv[2];
 -     const char* uiTitle = argv[3];
 - 
 -     const bool useOsc = strcmp(oscUrl, "null");
 - 
 -     // try to get sampleRate value
 -     const char* const sampleRateStr = getenv("CARLA_SAMPLE_RATE");
 - 
 -     if (sampleRateStr)
 -         sampleRate = atof(sampleRateStr);
 - 
 -     // Init toolkit
 -     CarlaToolkit* const toolkit = CarlaToolkit::createNew(uiTitle);
 -     toolkit->init();
 - 
 -     // Init VST-UI
 -     CarlaVstClient client(toolkit);
 - 
 -     // Init OSC
 -     if (useOsc && ! client.oscInit(oscUrl))
 -     {
 -         toolkit->quit();
 -         delete toolkit;
 -         return -1;
 -     }
 - 
 -     // Load UI
 -     int ret;
 - 
 -     if (client.init(binary, nullptr))
 -     {
 -         toolkit->exec(&client, !useOsc);
 -         ret = 0;
 -     }
 -     else
 -     {
 -         qCritical("Failed to load VST UI");
 -         ret = 1;
 -     }
 - 
 -     // Close OSC
 -     if (useOsc)
 -     {
 -         client.oscClose();
 -     }
 - 
 -     // Close VST-UI
 -     client.close();
 - 
 -     // Close toolkit
 -     toolkit->quit();
 -     delete toolkit;
 - 
 -     return ret;
 - }
 - 
 - #endif
 
 
  |