/* ============================================================================== This file is part of the JUCE library. Copyright (c) 2022 - Raw Material Software Limited JUCE is an open source library subject to commercial or open-source licensing. By using JUCE, you agree to the terms of both the JUCE 7 End-User License Agreement and JUCE Privacy Policy. End User License Agreement: www.juce.com/juce-7-licence Privacy Policy: www.juce.com/juce-privacy-policy Or: You may also use this code under the terms of the GPL v3 (see www.gnu.org/licenses). JUCE IS PROVIDED "AS IS" WITHOUT ANY WARRANTY, AND ALL WARRANTIES, WHETHER EXPRESSED OR IMPLIED, INCLUDING MERCHANTABILITY AND FITNESS FOR PURPOSE, ARE DISCLAIMED. ============================================================================== */ #if JucePlugin_Build_LV2 && (! (JUCE_ANDROID || JUCE_IOS)) #ifndef _SCL_SECURE_NO_WARNINGS #define _SCL_SECURE_NO_WARNINGS #endif #ifndef _CRT_SECURE_NO_WARNINGS #define _CRT_SECURE_NO_WARNINGS #endif #define JUCE_CORE_INCLUDE_NATIVE_HEADERS 1 #define JUCE_CORE_INCLUDE_OBJC_HELPERS 1 #define JUCE_GUI_BASICS_INCLUDE_XHEADERS 1 #include #include #include #include #include #include "JuceLV2Defines.h" #include #include #define JUCE_TURTLE_RECALL_URI "https://lv2-extensions.juce.com/turtle_recall" #ifndef JucePlugin_LV2URI #error "You need to define the JucePlugin_LV2URI value! If you're using the Projucer/CMake, the definition will be written into JuceLV2Defines.h automatically." #endif namespace juce { namespace lv2_client { constexpr auto uriSeparator = ":"; const auto JucePluginLV2UriUi = String (JucePlugin_LV2URI) + uriSeparator + "UI"; const auto JucePluginLV2UriState = String (JucePlugin_LV2URI) + uriSeparator + "StateString"; const auto JucePluginLV2UriProgram = String (JucePlugin_LV2URI) + uriSeparator + "Program"; static const LV2_Feature* findMatchingFeature (const LV2_Feature* const* features, const char* uri) { for (auto feature = features; *feature != nullptr; ++feature) if (std::strcmp ((*feature)->URI, uri) == 0) return *feature; return nullptr; } static bool hasFeature (const LV2_Feature* const* features, const char* uri) { return findMatchingFeature (features, uri) != nullptr; } template Data findMatchingFeatureData (const LV2_Feature* const* features, const char* uri) { if (const auto* feature = findMatchingFeature (features, uri)) return static_cast (feature->data); return {}; } static const LV2_Options_Option* findMatchingOption (const LV2_Options_Option* options, LV2_URID urid) { for (auto option = options; option->value != nullptr; ++option) if (option->key == urid) return option; return nullptr; } class ParameterStorage : private AudioProcessorListener { public: ParameterStorage (AudioProcessor& proc, LV2_URID_Map map) : processor (proc), mapFeature (map), legacyParameters (proc, false) { processor.addListener (this); } ~ParameterStorage() override { processor.removeListener (this); } /* This is the string that will be used to uniquely identify the parameter. This string will be written into the plugin's manifest as an IRI, so it must be syntactically valid. We escape this string rather than writing the user-defined parameter ID directly to avoid writing a malformed manifest in the case that user IDs contain spaces or other reserved characters. This should allow users to keep the same param IDs for all plugin formats. */ static String getIri (const AudioProcessorParameter& param) { const auto urlSanitised = URL::addEscapeChars (LegacyAudioParameter::getParamID (¶m, false), true); const auto ttlSanitised = lv2_shared::sanitiseStringAsTtlName (urlSanitised); // If this is hit, the parameter ID could not be represented directly in the plugin ttl. // We'll replace offending characters with '_'. jassert (urlSanitised == ttlSanitised); return ttlSanitised; } void setValueFromHost (LV2_URID urid, float value) noexcept { const auto it = uridToIndexMap.find (urid); if (it == uridToIndexMap.end()) { // No such parameter. jassertfalse; return; } if (auto* param = legacyParameters.getParamForIndex ((int) it->second)) { const auto scaledValue = [&] { if (auto* rangedParam = dynamic_cast (param)) return rangedParam->convertTo0to1 (value); return value; }(); if (scaledValue != param->getValue()) { ScopedValueSetter scope (ignoreCallbacks, true); param->setValueNotifyingHost (scaledValue); } } } struct Options { bool parameterValue, gestureBegin, gestureEnd; }; static constexpr auto newClientValue = 1 << 0, gestureBegan = 1 << 1, gestureEnded = 1 << 2; template void forEachChangedParameter (Callback&& callback) { stateCache.ifSet ([this, &callback] (size_t parameterIndex, float, uint32_t bits) { const Options options { (bits & newClientValue) != 0, (bits & gestureBegan) != 0, (bits & gestureEnded) != 0 }; callback (*legacyParameters.getParamForIndex ((int) parameterIndex), indexToUridMap[parameterIndex], options); }); } private: void audioProcessorParameterChanged (AudioProcessor*, int parameterIndex, float value) override { if (! ignoreCallbacks) stateCache.setValueAndBits ((size_t) parameterIndex, value, newClientValue); } void audioProcessorParameterChangeGestureBegin (AudioProcessor*, int parameterIndex) override { if (! ignoreCallbacks) stateCache.setBits ((size_t) parameterIndex, gestureBegan); } void audioProcessorParameterChangeGestureEnd (AudioProcessor*, int parameterIndex) override { if (! ignoreCallbacks) stateCache.setBits ((size_t) parameterIndex, gestureEnded); } void audioProcessorChanged (AudioProcessor*, const ChangeDetails&) override {} AudioProcessor& processor; const LV2_URID_Map mapFeature; const LegacyAudioParametersWrapper legacyParameters; const std::vector indexToUridMap = [&] { std::vector result; for (auto* param : legacyParameters) { jassert ((size_t) param->getParameterIndex() == result.size()); const auto uri = JucePlugin_LV2URI + String (uriSeparator) + getIri (*param); const auto urid = mapFeature.map (mapFeature.handle, uri.toRawUTF8()); result.push_back (urid); } // If this is hit, some parameters have duplicate IDs. // This may be because the IDs resolve to the same string when removing characters that // are invalid in a TTL name. jassert (std::set (result.begin(), result.end()).size() == result.size()); return result; }(); const std::map uridToIndexMap = [&] { std::map result; size_t index = 0; for (const auto& urid : indexToUridMap) result.emplace (urid, index++); return result; }(); FlaggedFloatCache<3> stateCache { (size_t) legacyParameters.getNumParameters() }; bool ignoreCallbacks = false; JUCE_LEAK_DETECTOR (ParameterStorage) }; enum class PortKind { seqInput, seqOutput, latencyOutput, freeWheelingInput, enabledInput }; struct PortIndices { PortIndices (int numInputsIn, int numOutputsIn) : numInputs (numInputsIn), numOutputs (numOutputsIn) {} int getPortIndexForAudioInput (int audioIndex) const noexcept { return audioIndex; } int getPortIndexForAudioOutput (int audioIndex) const noexcept { return audioIndex + numInputs; } int getPortIndexFor (PortKind p) const noexcept { return getMaxAudioPortIndex() + (int) p; } // Audio ports are numbered from 0 to numInputs + numOutputs int getMaxAudioPortIndex() const noexcept { return numInputs + numOutputs; } int numInputs, numOutputs; }; //============================================================================== class PlayHead : public AudioPlayHead { public: PlayHead (LV2_URID_Map mapFeatureIn, double sampleRateIn) : parser (mapFeatureIn), sampleRate (sampleRateIn) { } void invalidate() { info = nullopt; } void readNewInfo (const LV2_Atom_Event* event) { if (event->body.type != mLV2_ATOM__Object && event->body.type != mLV2_ATOM__Blank) return; const auto* object = reinterpret_cast (&event->body); if (object->body.otype != mLV2_TIME__Position) return; const LV2_Atom* atomFrame = nullptr; const LV2_Atom* atomSpeed = nullptr; const LV2_Atom* atomBar = nullptr; const LV2_Atom* atomBeat = nullptr; const LV2_Atom* atomBeatUnit = nullptr; const LV2_Atom* atomBeatsPerBar = nullptr; const LV2_Atom* atomBeatsPerMinute = nullptr; LV2_Atom_Object_Query query[] { { mLV2_TIME__frame, &atomFrame }, { mLV2_TIME__speed, &atomSpeed }, { mLV2_TIME__bar, &atomBar }, { mLV2_TIME__beat, &atomBeat }, { mLV2_TIME__beatUnit, &atomBeatUnit }, { mLV2_TIME__beatsPerBar, &atomBeatsPerBar }, { mLV2_TIME__beatsPerMinute, &atomBeatsPerMinute }, LV2_ATOM_OBJECT_QUERY_END }; lv2_atom_object_query (object, query); info.emplace(); // Carla always seems to give us an integral 'beat' even though I'd expect // it to be a floating-point value const auto numerator = parser.parseNumericAtom (atomBeatsPerBar); const auto denominator = parser.parseNumericAtom (atomBeatUnit); if (numerator.hasValue() && denominator.hasValue()) info->setTimeSignature (TimeSignature { (int) *numerator, (int) *denominator }); info->setBpm (parser.parseNumericAtom (atomBeatsPerMinute)); info->setPpqPosition (parser.parseNumericAtom (atomBeat)); info->setIsPlaying (parser.parseNumericAtom (atomSpeed).orFallback (0.0f) != 0.0f); info->setBarCount (parser.parseNumericAtom (atomBar)); if (const auto parsed = parser.parseNumericAtom (atomFrame)) { info->setTimeInSamples (*parsed); info->setTimeInSeconds ((double) *parsed / sampleRate); } } Optional getPosition() const override { return info; } private: lv2_shared::NumericAtomParser parser; Optional info; double sampleRate; #define X(str) const LV2_URID m##str = parser.map (str); X (LV2_ATOM__Blank) X (LV2_ATOM__Object) X (LV2_TIME__Position) X (LV2_TIME__beat) X (LV2_TIME__beatUnit) X (LV2_TIME__beatsPerBar) X (LV2_TIME__beatsPerMinute) X (LV2_TIME__frame) X (LV2_TIME__speed) X (LV2_TIME__bar) #undef X JUCE_LEAK_DETECTOR (PlayHead) }; //============================================================================== class Ports { public: Ports (LV2_URID_Map map, int numInputsIn, int numOutputsIn) : forge (map), indices (numInputsIn, numOutputsIn), mLV2_ATOM__Sequence (map.map (map.handle, LV2_ATOM__Sequence)) { audioBuffers.resize (static_cast (numInputsIn + numOutputsIn), nullptr); } void connect (int port, void* data) { // The following is not UB _if_ data really points to an object with the expected type. if (port == indices.getPortIndexFor (PortKind::seqInput)) { inputData = static_cast (data); } else if (port == indices.getPortIndexFor (PortKind::seqOutput)) { outputData = static_cast (data); } else if (port == indices.getPortIndexFor (PortKind::latencyOutput)) { latency = static_cast (data); } else if (port == indices.getPortIndexFor (PortKind::freeWheelingInput)) { freeWheeling = static_cast (data); } else if (port == indices.getPortIndexFor (PortKind::enabledInput)) { enabled = static_cast (data); } else if (isPositiveAndBelow (port, indices.getMaxAudioPortIndex())) { audioBuffers[(size_t) port] = static_cast (data); } else { // This port was not declared! jassertfalse; } } template void forEachInputEvent (Callback&& callback) { if (inputData != nullptr && inputData->atom.type == mLV2_ATOM__Sequence) for (const auto* event : lv2_shared::SequenceIterator { lv2_shared::SequenceWithSize { inputData } }) callback (event); } void prepareToWrite() { // Note: Carla seems to have a bug (verified with the eg-fifths plugin) where // the output buffer size is incorrect on alternate calls. forge.setBuffer (reinterpret_cast (outputData), outputData->atom.size); } void writeLatency (int value) { if (latency != nullptr) *latency = (float) value; } const float* getBufferForAudioInput (int index) const noexcept { return audioBuffers[(size_t) indices.getPortIndexForAudioInput (index)]; } float* getBufferForAudioOutput (int index) const noexcept { return audioBuffers[(size_t) indices.getPortIndexForAudioOutput (index)]; } bool isFreeWheeling() const noexcept { if (freeWheeling != nullptr) return *freeWheeling > 0.5f; return false; } bool isEnabled() const noexcept { if (enabled != nullptr) return *enabled > 0.5f; return true; } lv2_shared::AtomForge forge; PortIndices indices; private: static constexpr auto numParamPorts = 3; const LV2_Atom_Sequence* inputData = nullptr; LV2_Atom_Sequence* outputData = nullptr; float* latency = nullptr; float* freeWheeling = nullptr; float* enabled = nullptr; std::vector audioBuffers; const LV2_URID mLV2_ATOM__Sequence; JUCE_LEAK_DETECTOR (Ports) }; class LV2PluginInstance : private AudioProcessorListener { public: LV2PluginInstance (double sampleRate, int64_t maxBlockSize, const char*, LV2_URID_Map mapFeatureIn) : mapFeature (mapFeatureIn), playHead (mapFeature, sampleRate) { processor->addListener (this); processor->setPlayHead (&playHead); prepare (sampleRate, (int) maxBlockSize); } void connect (uint32_t port, void* data) { ports.connect ((int) port, data); } void activate() {} template static void iterateAudioBuffer (AudioBuffer& ab, UnaryFunction fn) { float* const* sampleData = ab.getArrayOfWritePointers(); for (int c = ab.getNumChannels(); --c >= 0;) for (int s = ab.getNumSamples(); --s >= 0;) fn (sampleData[c][s]); } static int countNaNs (AudioBuffer& ab) noexcept { int count = 0; iterateAudioBuffer (ab, [&count] (float s) { if (std::isnan (s)) ++count; }); return count; } void run (uint32_t numSteps) { // If this is hit, the host is trying to process more samples than it told us to prepare jassert (static_cast (numSteps) <= processor->getBlockSize()); midi.clear(); playHead.invalidate(); audio.setSize (audio.getNumChannels(), static_cast (numSteps), true, false, true); ports.forEachInputEvent ([&] (const LV2_Atom_Event* event) { struct Callback { explicit Callback (LV2PluginInstance& s) : self (s) {} void setParameter (LV2_URID property, float value) const noexcept { self.parameters.setValueFromHost (property, value); } // The host probably shouldn't send us 'touched' messages. void gesture (LV2_URID, bool) const noexcept {} LV2PluginInstance& self; }; patchSetHelper.processPatchSet (event, Callback { *this }); playHead.readNewInfo (event); if (event->body.type == mLV2_MIDI__MidiEvent) midi.addEvent (event + 1, static_cast (event->body.size), static_cast (event->time.frames)); }); processor->setNonRealtime (ports.isFreeWheeling()); for (auto i = 0, end = processor->getTotalNumInputChannels(); i < end; ++i) audio.copyFrom (i, 0, ports.getBufferForAudioInput (i), audio.getNumSamples()); jassert (countNaNs (audio) == 0); { const ScopedLock lock { processor->getCallbackLock() }; if (processor->isSuspended()) { for (auto i = 0, end = processor->getTotalNumOutputChannels(); i < end; ++i) { const auto ptr = ports.getBufferForAudioOutput (i); std::fill (ptr, ptr + numSteps, 0.0f); } } else { const auto isEnabled = ports.isEnabled(); if (auto* param = processor->getBypassParameter()) { param->setValueNotifyingHost (isEnabled ? 0.0f : 1.0f); processor->processBlock (audio, midi); } else if (isEnabled) { processor->processBlock (audio, midi); } else { processor->processBlockBypassed (audio, midi); } } } for (auto i = 0, end = processor->getTotalNumOutputChannels(); i < end; ++i) { const auto src = audio.getReadPointer (i); const auto dst = ports.getBufferForAudioOutput (i); if (dst != nullptr) std::copy (src, src + numSteps, dst); } ports.prepareToWrite(); auto* forge = ports.forge.get(); lv2_shared::SequenceFrame sequence { forge, (uint32_t) 0 }; parameters.forEachChangedParameter ([&] (const AudioProcessorParameter& param, LV2_URID paramUrid, const ParameterStorage::Options& options) { const auto sendTouched = [&] (bool state) { // TODO Implement begin/end change gesture support once it's supported by LV2 ignoreUnused (state); }; if (options.gestureBegin) sendTouched (true); if (options.parameterValue) { lv2_atom_forge_frame_time (forge, 0); lv2_shared::ObjectFrame object { forge, (uint32_t) 0, patchSetHelper.mLV2_PATCH__Set }; lv2_atom_forge_key (forge, patchSetHelper.mLV2_PATCH__property); lv2_atom_forge_urid (forge, paramUrid); lv2_atom_forge_key (forge, patchSetHelper.mLV2_PATCH__value); lv2_atom_forge_float (forge, [&] { if (auto* rangedParam = dynamic_cast (¶m)) return rangedParam->convertFrom0to1 (rangedParam->getValue()); return param.getValue(); }()); } if (options.gestureEnd) sendTouched (false); }); if (shouldSendStateChange.exchange (false)) { lv2_atom_forge_frame_time (forge, 0); lv2_shared::ObjectFrame { forge, (uint32_t) 0, mLV2_STATE__StateChanged }; } for (const auto meta : midi) { const auto bytes = static_cast (meta.numBytes); lv2_atom_forge_frame_time (forge, meta.samplePosition); lv2_atom_forge_atom (forge, bytes, mLV2_MIDI__MidiEvent); lv2_atom_forge_write (forge, meta.data, bytes); } ports.writeLatency (processor->getLatencySamples()); } void deactivate() {} LV2_State_Status store (LV2_State_Store_Function storeFn, LV2_State_Handle handle, uint32_t, const LV2_Feature* const*) { MemoryBlock block; processor->getStateInformation (block); const auto text = block.toBase64Encoding(); storeFn (handle, mJucePluginLV2UriState, text.toRawUTF8(), text.getNumBytesAsUTF8() + 1, mLV2_ATOM__String, LV2_STATE_IS_POD | LV2_STATE_IS_PORTABLE); return LV2_STATE_SUCCESS; } LV2_State_Status retrieve (LV2_State_Retrieve_Function retrieveFn, LV2_State_Handle handle, uint32_t, const LV2_Feature* const*) { size_t size = 0; uint32_t type = 0; uint32_t dataFlags = 0; // Try retrieving a port index (if this is a 'program' preset). const auto* programData = retrieveFn (handle, mJucePluginLV2UriProgram, &size, &type, &dataFlags); if (programData != nullptr && type == mLV2_ATOM__Int && size == sizeof (int32_t)) { const auto programIndex = readUnaligned (programData); processor->setCurrentProgram (programIndex); return LV2_STATE_SUCCESS; } // This doesn't seem to be a 'program' preset, try setting the full state from a string instead. const auto* data = retrieveFn (handle, mJucePluginLV2UriState, &size, &type, &dataFlags); if (data == nullptr) return LV2_STATE_ERR_NO_PROPERTY; if (type != mLV2_ATOM__String) return LV2_STATE_ERR_BAD_TYPE; String text (static_cast (data), (size_t) size); MemoryBlock block; block.fromBase64Encoding (text); processor->setStateInformation (block.getData(), (int) block.getSize()); return LV2_STATE_SUCCESS; } std::unique_ptr createEditor() { return std::unique_ptr (processor->createEditorIfNeeded()); } void editorBeingDeleted (AudioProcessorEditor* editor) { processor->editorBeingDeleted (editor); } static std::unique_ptr createProcessorInstance() { auto result = createPluginFilterOfType (AudioProcessor::wrapperType_LV2); #if defined (JucePlugin_PreferredChannelConfigurations) constexpr short channelConfigurations[][2] { JucePlugin_PreferredChannelConfigurations }; static_assert (numElementsInArray (channelConfigurations) > 0, "JucePlugin_PreferredChannelConfigurations must contain at least one entry"); static_assert (channelConfigurations[0][0] > 0 || channelConfigurations[0][1] > 0, "JucePlugin_PreferredChannelConfigurations must have at least one input or output channel"); result->setPlayConfigDetails (channelConfigurations[0][0], channelConfigurations[0][1], 44100.0, 1024); const auto desiredChannels = std::make_tuple (channelConfigurations[0][0], channelConfigurations[0][1]); const auto actualChannels = std::make_tuple (result->getTotalNumInputChannels(), result->getTotalNumOutputChannels()); if (desiredChannels != actualChannels) Logger::outputDebugString ("Failed to apply requested channel configuration!"); #else result->enableAllBuses(); #endif return result; } private: void audioProcessorParameterChanged (AudioProcessor*, int, float) override {} void audioProcessorChanged (AudioProcessor*, const ChangeDetails& details) override { // Only check for non-parameter state here because: // - Latency is automatically written every block. // - There's no way for an LV2 plugin to report an internal program change. // - Parameter info is hard-coded in the plugin's turtle description. if (details.nonParameterStateChanged) shouldSendStateChange = true; } void prepare (double sampleRate, int maxBlockSize) { jassert (processor != nullptr); processor->setRateAndBufferSizeDetails (sampleRate, maxBlockSize); processor->prepareToPlay (sampleRate, maxBlockSize); const auto numChannels = jmax (processor->getTotalNumInputChannels(), processor->getTotalNumOutputChannels()); midi.ensureSize (8192); audio.setSize (numChannels, maxBlockSize); audio.clear(); } LV2_URID map (StringRef uri) const { return mapFeature.map (mapFeature.handle, uri); } ScopedJuceInitialiser_GUI scopedJuceInitialiser; #if JUCE_LINUX || JUCE_BSD SharedResourcePointer messageThread; #endif std::unique_ptr processor = createProcessorInstance(); const LV2_URID_Map mapFeature; ParameterStorage parameters { *processor, mapFeature }; Ports ports { mapFeature, processor->getTotalNumInputChannels(), processor->getTotalNumOutputChannels() }; lv2_shared::PatchSetHelper patchSetHelper { mapFeature, JucePlugin_LV2URI }; PlayHead playHead; MidiBuffer midi; AudioBuffer audio; std::atomic shouldSendStateChange { false }; #define X(str) const LV2_URID m##str = map (str); X (JucePluginLV2UriProgram) X (JucePluginLV2UriState) X (LV2_ATOM__Int) X (LV2_ATOM__String) X (LV2_BUF_SIZE__maxBlockLength) X (LV2_BUF_SIZE__sequenceSize) X (LV2_MIDI__MidiEvent) X (LV2_PATCH__Set) X (LV2_STATE__StateChanged) #undef X JUCE_LEAK_DETECTOR (LV2PluginInstance) }; //============================================================================== struct RecallFeature { int (*doRecall) (const char*) = [] (const char* libraryPath) -> int { const ScopedJuceInitialiser_GUI scope; const auto processor = LV2PluginInstance::createProcessorInstance(); const String pathString { CharPointer_UTF8 { libraryPath } }; const auto absolutePath = File::isAbsolutePath (pathString) ? File (pathString) : File::getCurrentWorkingDirectory().getChildFile (pathString); const auto writers = { writeManifestTtl, writeDspTtl, writeUiTtl }; const auto wroteSuccessfully = [&processor, &absolutePath] (auto* fn) { const auto result = fn (*processor, absolutePath); if (! result.wasOk()) std::cerr << result.getErrorMessage() << '\n'; return result.wasOk(); }; return std::all_of (writers.begin(), writers.end(), wroteSuccessfully) ? 0 : 1; }; private: static String getPresetUri (int index) { return JucePlugin_LV2URI + String (uriSeparator) + "preset" + String (index + 1); } static std::ofstream openStream (const File& libraryPath, StringRef name) { return std::ofstream { libraryPath.getSiblingFile (name + ".ttl").getFullPathName().toRawUTF8() }; } static Result writeManifestTtl (AudioProcessor& proc, const File& libraryPath) { auto os = openStream (libraryPath, "manifest"); os << "@prefix lv2: .\n" "@prefix rdfs: .\n" "@prefix pset: .\n" "@prefix state: .\n" "@prefix ui: .\n" "@prefix xsd: .\n" "\n" "<" JucePlugin_LV2URI ">\n" "\ta lv2:Plugin ;\n" "\tlv2:binary <" << URL::addEscapeChars (libraryPath.getFileName(), false) << "> ;\n" "\trdfs:seeAlso .\n"; if (proc.hasEditor()) { #if JUCE_MAC #define JUCE_LV2_UI_KIND "CocoaUI" #elif JUCE_WINDOWS #define JUCE_LV2_UI_KIND "WindowsUI" #elif JUCE_LINUX || JUCE_BSD #define JUCE_LV2_UI_KIND "X11UI" #else #error "LV2 UI is unsupported on this platform" #endif os << "\n" "<" << JucePluginLV2UriUi << ">\n" "\ta ui:" JUCE_LV2_UI_KIND " ;\n" "\tlv2:binary <" << URL::addEscapeChars (libraryPath.getFileName(), false) << "> ;\n" "\trdfs:seeAlso .\n" "\n"; } for (auto i = 0, end = proc.getNumPrograms(); i < end; ++i) { os << "<" << getPresetUri (i) << ">\n" "\ta pset:Preset ;\n" "\tlv2:appliesTo <" JucePlugin_LV2URI "> ;\n" "\trdfs:label \"" << proc.getProgramName (i) << "\" ;\n" "\tstate:state [ <" << JucePluginLV2UriProgram << "> \"" << i << "\"^^xsd:int ; ] .\n" "\n"; } return Result::ok(); } static std::vector findAllSubgroupsDepthFirst (const AudioProcessorParameterGroup& group, std::vector foundSoFar = {}) { foundSoFar.push_back (&group); for (auto* node : group) { if (auto* subgroup = node->getGroup()) foundSoFar = findAllSubgroupsDepthFirst (*subgroup, std::move (foundSoFar)); } return foundSoFar; } using GroupSymbolMap = std::map; static String getFlattenedGroupSymbol (const AudioProcessorParameterGroup& group, String symbol = "") { if (auto* parent = group.getParent()) return getFlattenedGroupSymbol (*parent, group.getID() + (symbol.isEmpty() ? "" : group.getSeparator() + symbol)); return symbol; } static String getSymbolForGroup (const AudioProcessorParameterGroup& group) { const String allowedCharacters ("abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ_0123456789"); const auto base = getFlattenedGroupSymbol (group); if (base.isEmpty()) return {}; String copy; for (const auto character : base) copy << String::charToString (allowedCharacters.containsChar (character) ? character : (juce_wchar) '_'); return "paramgroup_" + copy; } static GroupSymbolMap getGroupsAndSymbols (const std::vector& groups) { std::set symbols; GroupSymbolMap result; for (const auto& group : groups) { const auto symbol = [&] { const auto idealSymbol = getSymbolForGroup (*group); if (symbols.find (idealSymbol) == symbols.cend()) return idealSymbol; for (auto i = 2; i < std::numeric_limits::max(); ++i) { const auto toTest = idealSymbol + "_" + String (i); if (symbols.find (toTest) == symbols.cend()) return toTest; } jassertfalse; return String(); }(); symbols.insert (symbol); result.emplace (group, symbol); } return result; } template static void visitAllParameters (const GroupSymbolMap& groups, Fn&& fn) { for (const auto& group : groups) for (const auto* node : *group.first) if (auto* param = node->getParameter()) fn (group.second, *param); } static Result writeDspTtl (AudioProcessor& proc, const File& libraryPath) { auto os = openStream (libraryPath, "dsp"); os << "@prefix atom: .\n" "@prefix bufs: .\n" "@prefix doap: .\n" "@prefix foaf: .\n" "@prefix lv2: .\n" "@prefix midi: .\n" "@prefix opts: .\n" "@prefix param: .\n" "@prefix patch: .\n" "@prefix pg: .\n" "@prefix plug: <" JucePlugin_LV2URI << uriSeparator << "> .\n" "@prefix pprop: .\n" "@prefix rdfs: .\n" "@prefix rdf: .\n" "@prefix rsz: .\n" "@prefix state: .\n" "@prefix time: .\n" "@prefix ui: .\n" "@prefix units: .\n" "@prefix urid: .\n" "@prefix xsd: .\n" "\n"; LegacyAudioParametersWrapper legacyParameters (proc, false); const auto allGroups = findAllSubgroupsDepthFirst (legacyParameters.getGroup()); const auto groupsAndSymbols = getGroupsAndSymbols (allGroups); const auto parameterVisitor = [&] (const String& symbol, const AudioProcessorParameter& param) { os << "plug:" << ParameterStorage::getIri (param) << "\n" "\ta lv2:Parameter ;\n" "\trdfs:label \"" << param.getName (1024) << "\" ;\n"; if (symbol.isNotEmpty()) os << "\tpg:group plug:" << symbol << " ;\n"; os << "\trdfs:range atom:Float ;\n"; if (auto* rangedParam = dynamic_cast (¶m)) { os << "\tlv2:default " << rangedParam->convertFrom0to1 (rangedParam->getDefaultValue()) << " ;\n" "\tlv2:minimum " << rangedParam->getNormalisableRange().start << " ;\n" "\tlv2:maximum " << rangedParam->getNormalisableRange().end; } else { os << "\tlv2:default " << param.getDefaultValue() << " ;\n" "\tlv2:minimum 0.0 ;\n" "\tlv2:maximum 1.0"; } // Avoid writing out loads of scale points for parameters with lots of steps constexpr auto stepLimit = 1000; const auto numSteps = param.getNumSteps(); if (param.isDiscrete() && 2 <= numSteps && numSteps < stepLimit) { os << "\t ;\n" "\tlv2:portProperty lv2:enumeration " << (param.isBoolean() ? ", lv2:toggled " : "") << ";\n" "\tlv2:scalePoint "; const auto maxIndex = numSteps - 1; for (int i = 0; i < numSteps; ++i) { const auto value = (float) i / (float) maxIndex; const auto text = param.getText (value, 1024); os << (i != 0 ? ", " : "") << "[\n" "\t\trdfs:label \"" << text << "\" ;\n" "\t\trdf:value " << value << " ;\n" "\t]"; } } os << " .\n\n"; }; visitAllParameters (groupsAndSymbols, parameterVisitor); for (const auto& groupInfo : groupsAndSymbols) { if (groupInfo.second.isEmpty()) continue; os << "plug:" << groupInfo.second << "\n" "\ta pg:Group ;\n"; if (auto* parent = groupInfo.first->getParent()) { if (parent->getParent() != nullptr) { const auto it = groupsAndSymbols.find (parent); if (it != groupsAndSymbols.cend()) os << "\tpg:subGroupOf plug:" << it->second << " ;\n"; } } os << "\tlv2:symbol \"" << groupInfo.second << "\" ;\n" "\tlv2:name \"" << groupInfo.first->getName() << "\" .\n\n"; } const auto getBaseBusName = [] (bool isInput) { return isInput ? "input_group_" : "output_group_"; }; for (const auto isInput : { true, false }) { const auto baseBusName = getBaseBusName (isInput); const auto groupKind = isInput ? "InputGroup" : "OutputGroup"; const auto busCount = proc.getBusCount (isInput); for (auto i = 0; i < busCount; ++i) { if (const auto* bus = proc.getBus (isInput, i)) { os << "plug:" << baseBusName << (i + 1) << "\n" "\ta pg:" << groupKind << " ;\n" "\tlv2:name \"" << bus->getName() << "\" ;\n" "\tlv2:symbol \"" << baseBusName << (i + 1) << "\" .\n\n"; } } } os << "<" JucePlugin_LV2URI ">\n"; if (proc.hasEditor()) os << "\tui:ui <" << JucePluginLV2UriUi << "> ;\n"; const auto versionParts = StringArray::fromTokens (JucePlugin_VersionString, ".", ""); const auto getVersionOrZero = [&] (int indexFromBack) { const auto str = versionParts[versionParts.size() - indexFromBack]; return str.isEmpty() ? 0 : str.getIntValue(); }; const auto minorVersion = getVersionOrZero (2); const auto microVersion = getVersionOrZero (1); os << "\ta " #if JucePlugin_IsSynth "lv2:InstrumentPlugin" #else "lv2:Plugin" #endif " ;\n" "\tdoap:name \"" JucePlugin_Name "\" ;\n" "\tdoap:description \"" JucePlugin_Desc "\" ;\n" "\tlv2:minorVersion " << minorVersion << " ;\n" "\tlv2:microVersion " << microVersion << " ;\n" "\tdoap:maintainer [\n" "\t\ta foaf:Person ;\n" "\t\tfoaf:name \"" JucePlugin_Manufacturer "\" ;\n" "\t\tfoaf:homepage <" JucePlugin_ManufacturerWebsite "> ;\n" "\t\tfoaf:mbox <" JucePlugin_ManufacturerEmail "> ;\n" "\t] ;\n" "\tdoap:release [\n" "\t\ta doap:Version ;\n" "\t\tdoap:revision \"" JucePlugin_VersionString "\" ;\n" "\t] ;\n" "\tlv2:optionalFeature\n" "\t\tlv2:hardRTCapable ;\n" "\tlv2:extensionData\n" "\t\tstate:interface ;\n" "\tlv2:requiredFeature\n" "\t\turid:map ,\n" "\t\topts:options ,\n" "\t\tbufs:boundedBlockLength ;\n"; for (const auto isInput : { true, false }) { const auto kind = isInput ? "mainInput" : "mainOutput"; if (proc.getBusCount (isInput) > 0) os << "\tpg:" << kind << " plug:" << getBaseBusName (isInput) << "1 ;\n"; } if (legacyParameters.size() != 0) { for (const auto header : { "writable", "readable" }) { os << "\tpatch:" << header; bool isFirst = true; for (const auto* param : legacyParameters) { os << (isFirst ? "" : " ,") << "\n\t\tplug:" << ParameterStorage::getIri (*param); isFirst = false; } os << " ;\n"; } } os << "\tlv2:port [\n"; const PortIndices indices (proc.getTotalNumInputChannels(), proc.getTotalNumOutputChannels()); const auto designationMap = [&] { std::map result; for (const auto& pair : lv2_shared::channelDesignationMap) result.emplace (pair.second, pair.first); return result; }(); // TODO add support for specific audio group kinds for (const auto isInput : { true, false }) { const auto baseBusName = getBaseBusName (isInput); const auto portKind = isInput ? "InputPort" : "OutputPort"; const auto portName = isInput ? "Audio In " : "Audio Out "; const auto portSymbol = isInput ? "audio_in_" : "audio_out_"; const auto busCount = proc.getBusCount (isInput); auto channelCounter = 0; for (auto busIndex = 0; busIndex < busCount; ++busIndex) { if (const auto* bus = proc.getBus (isInput, busIndex)) { const auto channelCount = bus->getNumberOfChannels(); const auto optionalBus = ! bus->isEnabledByDefault(); for (auto channelIndex = 0; channelIndex < channelCount; ++channelIndex, ++channelCounter) { const auto portIndex = isInput ? indices.getPortIndexForAudioInput (channelCounter) : indices.getPortIndexForAudioOutput (channelCounter); os << "\t\ta lv2:" << portKind << " , lv2:AudioPort ;\n" "\t\tlv2:index " << portIndex << " ;\n" "\t\tlv2:symbol \"" << portSymbol << (channelCounter + 1) << "\" ;\n" "\t\tlv2:name \"" << portName << (channelCounter + 1) << "\" ;\n" "\t\tpg:group plug:" << baseBusName << (busIndex + 1) << " ;\n"; if (optionalBus) os << "\t\tlv2:portProperty lv2:connectionOptional ;\n"; const auto designation = bus->getCurrentLayout().getTypeOfChannel (channelIndex); const auto it = designationMap.find (designation); if (it != designationMap.end()) os << "\t\tlv2:designation <" << it->second << "> ;\n"; os << "\t] , [\n"; } } } } // In the event that the plugin decides to send all of its parameters in one go, // we should ensure that the output buffer is large enough to accommodate, with some // extra room for the sequence header, MIDI messages etc.. const auto patchSetSizeBytes = 72; const auto additionalSize = 8192; const auto atomPortMinSize = proc.getParameters().size() * patchSetSizeBytes + additionalSize; os << "\t\ta lv2:InputPort , atom:AtomPort ;\n" "\t\trsz:minimumSize " << atomPortMinSize << " ;\n" "\t\tatom:bufferType atom:Sequence ;\n" "\t\tatom:supports\n"; #if ! JucePlugin_IsSynth && ! JucePlugin_IsMidiEffect if (proc.acceptsMidi()) #endif os << "\t\t\tmidi:MidiEvent ,\n"; os << "\t\t\tpatch:Message ,\n" "\t\t\ttime:Position ;\n" "\t\tlv2:designation lv2:control ;\n" "\t\tlv2:index " << indices.getPortIndexFor (PortKind::seqInput) << " ;\n" "\t\tlv2:symbol \"in\" ;\n" "\t\tlv2:name \"In\" ;\n" "\t] , [\n" "\t\ta lv2:OutputPort , atom:AtomPort ;\n" "\t\trsz:minimumSize " << atomPortMinSize << " ;\n" "\t\tatom:bufferType atom:Sequence ;\n" "\t\tatom:supports\n"; #if ! JucePlugin_IsMidiEffect if (proc.producesMidi()) #endif os << "\t\t\tmidi:MidiEvent ,\n"; os << "\t\t\tpatch:Message ;\n" "\t\tlv2:designation lv2:control ;\n" "\t\tlv2:index " << indices.getPortIndexFor (PortKind::seqOutput) << " ;\n" "\t\tlv2:symbol \"out\" ;\n" "\t\tlv2:name \"Out\" ;\n" "\t] , [\n" "\t\ta lv2:OutputPort , lv2:ControlPort ;\n" "\t\tlv2:designation lv2:latency ;\n" "\t\tlv2:symbol \"latency\" ;\n" "\t\tlv2:name \"Latency\" ;\n" "\t\tlv2:index " << indices.getPortIndexFor (PortKind::latencyOutput) << " ;\n" "\t\tlv2:portProperty lv2:reportsLatency , lv2:integer , lv2:connectionOptional , pprop:notOnGUI ;\n" "\t\tunits:unit units:frame ;\n" "\t] , [\n" "\t\ta lv2:InputPort , lv2:ControlPort ;\n" "\t\tlv2:designation lv2:freeWheeling ;\n" "\t\tlv2:symbol \"freeWheeling\" ;\n" "\t\tlv2:name \"Free Wheeling\" ;\n" "\t\tlv2:default 0.0 ;\n" "\t\tlv2:minimum 0.0 ;\n" "\t\tlv2:maximum 1.0 ;\n" "\t\tlv2:index " << indices.getPortIndexFor (PortKind::freeWheelingInput) << " ;\n" "\t\tlv2:portProperty lv2:toggled , lv2:connectionOptional , pprop:notOnGUI ;\n" "\t] , [\n" "\t\ta lv2:InputPort , lv2:ControlPort ;\n" "\t\tlv2:designation lv2:enabled ;\n" "\t\tlv2:symbol \"enabled\" ;\n" "\t\tlv2:name \"Enabled\" ;\n" "\t\tlv2:default 1.0 ;\n" "\t\tlv2:minimum 0.0 ;\n" "\t\tlv2:maximum 1.0 ;\n" "\t\tlv2:index " << indices.getPortIndexFor (PortKind::enabledInput) << " ;\n" "\t\tlv2:portProperty lv2:toggled , lv2:connectionOptional , pprop:notOnGUI ;\n" "\t] ;\n" "\topts:supportedOption\n" "\t\tbufs:maxBlockLength .\n"; return Result::ok(); } static Result writeUiTtl (AudioProcessor& proc, const File& libraryPath) { if (! proc.hasEditor()) return Result::ok(); auto os = openStream (libraryPath, "ui"); const auto editorInstance = rawToUniquePtr (proc.createEditor()); const auto resizeFeatureString = editorInstance->isResizable() ? "ui:resize" : "ui:noUserResize"; os << "@prefix lv2: .\n" "@prefix opts: .\n" "@prefix param: .\n" "@prefix ui: .\n" "@prefix urid: .\n" "\n" "<" << JucePluginLV2UriUi << ">\n" "\tlv2:extensionData\n" #if JUCE_LINUX || JUCE_BSD "\t\tui:idleInterface ,\n" #endif "\t\topts:interface ,\n" "\t\tui:noUserResize ,\n" // resize and noUserResize are always present in the extension data array "\t\tui:resize ;\n" "\n" "\tlv2:requiredFeature\n" #if JUCE_LINUX || JUCE_BSD "\t\tui:idleInterface ,\n" #endif "\t\turid:map ,\n" "\t\tui:parent ,\n" "\t\t ;\n" "\n" "\tlv2:optionalFeature\n" "\t\t" << resizeFeatureString << " ,\n" "\t\topts:interface ,\n" "\t\topts:options ;\n\n" "\topts:supportedOption\n" "\t\tui:scaleFactor ,\n" "\t\tparam:sampleRate .\n"; return Result::ok(); } JUCE_LEAK_DETECTOR (RecallFeature) }; //============================================================================== LV2_SYMBOL_EXPORT const LV2_Descriptor* lv2_descriptor (uint32_t index) { if (index != 0) return nullptr; static const LV2_Descriptor descriptor { JucePlugin_LV2URI, // TODO some constexpr check that this is a valid URI in terms of RFC 3986 [] (const LV2_Descriptor*, double sampleRate, const char* pathToBundle, const LV2_Feature* const* features) -> LV2_Handle { const auto* mapFeature = findMatchingFeatureData (features, LV2_URID__map); if (mapFeature == nullptr) { // The host doesn't provide the 'urid map' feature jassertfalse; return nullptr; } const auto boundedBlockLength = hasFeature (features, LV2_BUF_SIZE__boundedBlockLength); if (! boundedBlockLength) { // The host doesn't provide the 'bounded block length' feature jassertfalse; return nullptr; } const auto* options = findMatchingFeatureData (features, LV2_OPTIONS__options); if (options == nullptr) { // The host doesn't provide the 'options' feature jassertfalse; return nullptr; } const lv2_shared::NumericAtomParser parser { *mapFeature }; const auto blockLengthUrid = mapFeature->map (mapFeature->handle, LV2_BUF_SIZE__maxBlockLength); const auto blockSize = parser.parseNumericOption (findMatchingOption (options, blockLengthUrid)); if (! blockSize.hasValue()) { // The host doesn't specify a maximum block size jassertfalse; return nullptr; } return new LV2PluginInstance { sampleRate, *blockSize, pathToBundle, *mapFeature }; }, [] (LV2_Handle instance, uint32_t port, void* data) { static_cast (instance)->connect (port, data); }, [] (LV2_Handle instance) { static_cast (instance)->activate(); }, [] (LV2_Handle instance, uint32_t sampleCount) { static_cast (instance)->run (sampleCount); }, [] (LV2_Handle instance) { static_cast (instance)->deactivate(); }, [] (LV2_Handle instance) { JUCE_AUTORELEASEPOOL { delete static_cast (instance); } }, [] (const char* uri) -> const void* { const auto uriMatches = [&] (const LV2_Feature& f) { return std::strcmp (f.URI, uri) == 0; }; static RecallFeature recallFeature; static LV2_State_Interface stateInterface { [] (LV2_Handle instance, LV2_State_Store_Function store, LV2_State_Handle handle, uint32_t flags, const LV2_Feature* const* features) -> LV2_State_Status { return static_cast (instance)->store (store, handle, flags, features); }, [] (LV2_Handle instance, LV2_State_Retrieve_Function retrieve, LV2_State_Handle handle, uint32_t flags, const LV2_Feature* const* features) -> LV2_State_Status { return static_cast (instance)->retrieve (retrieve, handle, flags, features); } }; static const LV2_Feature features[] { { JUCE_TURTLE_RECALL_URI, &recallFeature }, { LV2_STATE__interface, &stateInterface } }; const auto it = std::find_if (std::begin (features), std::end (features), uriMatches); return it != std::end (features) ? it->data : nullptr; } }; return &descriptor; } static Optional findScaleFactor (const LV2_URID_Map* symap, const LV2_Options_Option* options) { if (options == nullptr || symap == nullptr) return {}; const lv2_shared::NumericAtomParser parser { *symap }; const auto scaleFactorUrid = symap->map (symap->handle, LV2_UI__scaleFactor); const auto* scaleFactorOption = findMatchingOption (options, scaleFactorUrid); return parser.parseNumericOption (scaleFactorOption); } class LV2UIInstance : private Component, private ComponentListener { public: LV2UIInstance (const char*, const char*, LV2UI_Write_Function writeFunctionIn, LV2UI_Controller controllerIn, LV2UI_Widget* widget, LV2PluginInstance* pluginIn, LV2UI_Widget parentIn, const LV2_URID_Map* symapIn, const LV2UI_Resize* resizeFeatureIn, Optional scaleFactorIn) : writeFunction (writeFunctionIn), controller (controllerIn), plugin (pluginIn), parent (parentIn), symap (symapIn), resizeFeature (resizeFeatureIn), scaleFactor (scaleFactorIn), editor (plugin->createEditor()) { jassert (plugin != nullptr); jassert (parent != nullptr); jassert (editor != nullptr); if (editor == nullptr) return; const auto bounds = getSizeToContainChild(); setSize (bounds.getWidth(), bounds.getHeight()); addAndMakeVisible (*editor); setBroughtToFrontOnMouseClick (true); setOpaque (true); setVisible (false); removeFromDesktop(); addToDesktop (0, parent); editor->addComponentListener (this); *widget = getWindowHandle(); setVisible (true); editor->setScaleFactor (getScaleFactor()); requestResize(); } ~LV2UIInstance() override { plugin->editorBeingDeleted (editor.get()); } // This is called by the host when a parameter changes. // We don't care, our UI will listen to the processor directly. void portEvent (uint32_t, uint32_t, uint32_t, const void*) {} // Called when the host requests a resize int resize (int width, int height) { const ScopedValueSetter scope (hostRequestedResize, true); setSize (width, height); return 0; } // Called by the host to give us an opportunity to process UI events void idleCallback() { #if JUCE_LINUX || JUCE_BSD messageThread->processPendingEvents(); #endif } void resized() override { const ScopedValueSetter scope (hostRequestedResize, true); if (editor != nullptr) { const auto localArea = editor->getLocalArea (this, getLocalBounds()); editor->setBoundsConstrained ({ localArea.getWidth(), localArea.getHeight() }); } } void paint (Graphics& g) override { g.fillAll (Colours::black); } uint32_t getOptions (LV2_Options_Option* options) { const auto scaleFactorUrid = symap->map (symap->handle, LV2_UI__scaleFactor); const auto floatUrid = symap->map (symap->handle, LV2_ATOM__Float);; for (auto* opt = options; opt->key != 0; ++opt) { if (opt->context != LV2_OPTIONS_INSTANCE || opt->subject != 0 || opt->key != scaleFactorUrid) continue; if (scaleFactor.hasValue()) { opt->type = floatUrid; opt->size = sizeof (float); opt->value = &(*scaleFactor); } } return LV2_OPTIONS_SUCCESS; } uint32_t setOptions (const LV2_Options_Option* options) { const auto scaleFactorUrid = symap->map (symap->handle, LV2_UI__scaleFactor); const auto floatUrid = symap->map (symap->handle, LV2_ATOM__Float);; for (auto* opt = options; opt->key != 0; ++opt) { if (opt->context != LV2_OPTIONS_INSTANCE || opt->subject != 0 || opt->key != scaleFactorUrid || opt->type != floatUrid || opt->size != sizeof (float)) { continue; } scaleFactor = *static_cast (opt->value); updateScale(); } return LV2_OPTIONS_SUCCESS; } private: void updateScale() { editor->setScaleFactor (getScaleFactor()); requestResize(); } Rectangle getSizeToContainChild() const { if (editor != nullptr) return getLocalArea (editor.get(), editor->getLocalBounds()); return {}; } float getScaleFactor() const noexcept { return scaleFactor.hasValue() ? *scaleFactor : 1.0f; } void componentMovedOrResized (Component&, bool, bool wasResized) override { if (! hostRequestedResize && wasResized) requestResize(); } void write (uint32_t portIndex, uint32_t bufferSize, uint32_t portProtocol, const void* data) { writeFunction (controller, portIndex, bufferSize, portProtocol, data); } void requestResize() { if (editor == nullptr) return; const auto bounds = getSizeToContainChild(); if (resizeFeature == nullptr) return; if (auto* fn = resizeFeature->ui_resize) fn (resizeFeature->handle, bounds.getWidth(), bounds.getHeight()); setSize (bounds.getWidth(), bounds.getHeight()); repaint(); } #if JUCE_LINUX || JUCE_BSD SharedResourcePointer messageThread; #endif LV2UI_Write_Function writeFunction; LV2UI_Controller controller; LV2PluginInstance* plugin; LV2UI_Widget parent; const LV2_URID_Map* symap = nullptr; const LV2UI_Resize* resizeFeature = nullptr; Optional scaleFactor; std::unique_ptr editor; bool hostRequestedResize = false; JUCE_LEAK_DETECTOR (LV2UIInstance) }; LV2_SYMBOL_EXPORT const LV2UI_Descriptor* lv2ui_descriptor (uint32_t index) { if (index != 0) return nullptr; static const LV2UI_Descriptor descriptor { JucePluginLV2UriUi.toRawUTF8(), // TODO some constexpr check that this is a valid URI in terms of RFC 3986 [] (const LV2UI_Descriptor*, const char* pluginUri, const char* bundlePath, LV2UI_Write_Function writeFunction, LV2UI_Controller controller, LV2UI_Widget* widget, const LV2_Feature* const* features) -> LV2UI_Handle { #if JUCE_LINUX || JUCE_BSD SharedResourcePointer messageThread; #endif auto* plugin = findMatchingFeatureData (features, LV2_INSTANCE_ACCESS_URI); if (plugin == nullptr) { // No instance access jassertfalse; return nullptr; } auto* parent = findMatchingFeatureData (features, LV2_UI__parent); if (parent == nullptr) { // No parent access jassertfalse; return nullptr; } auto* resizeFeature = findMatchingFeatureData (features, LV2_UI__resize); const auto* symap = findMatchingFeatureData (features, LV2_URID__map); const auto scaleFactor = findScaleFactor (symap, findMatchingFeatureData (features, LV2_OPTIONS__options)); return new LV2UIInstance { pluginUri, bundlePath, writeFunction, controller, widget, plugin, parent, symap, resizeFeature, scaleFactor }; }, [] (LV2UI_Handle ui) { #if JUCE_LINUX || JUCE_BSD SharedResourcePointer messageThread; #endif JUCE_AUTORELEASEPOOL { delete static_cast (ui); } }, [] (LV2UI_Handle ui, uint32_t portIndex, uint32_t bufferSize, uint32_t format, const void* buffer) { JUCE_ASSERT_MESSAGE_THREAD static_cast (ui)->portEvent (portIndex, bufferSize, format, buffer); }, [] (const char* uri) -> const void* { const auto uriMatches = [&] (const LV2_Feature& f) { return std::strcmp (f.URI, uri) == 0; }; static LV2UI_Resize resize { nullptr, [] (LV2UI_Feature_Handle handle, int width, int height) -> int { JUCE_ASSERT_MESSAGE_THREAD return static_cast (handle)->resize (width, height); } }; static LV2UI_Idle_Interface idle { [] (LV2UI_Handle handle) { static_cast (handle)->idleCallback(); return 0; } }; static LV2_Options_Interface options { [] (LV2_Handle handle, LV2_Options_Option* optionsIn) { return static_cast (handle)->getOptions (optionsIn); }, [] (LV2_Handle handle, const LV2_Options_Option* optionsIn) { return static_cast (handle)->setOptions (optionsIn); } }; // We'll always define noUserResize and idle in the extension data array, but we'll // only declare them in the ui.ttl if the UI is actually non-resizable, or requires // idle callbacks. // Well-behaved hosts should check the ttl before trying to search the // extension-data array. static const LV2_Feature features[] { { LV2_UI__resize, &resize }, { LV2_UI__noUserResize, nullptr }, { LV2_UI__idleInterface, &idle }, { LV2_OPTIONS__interface, &options } }; const auto it = std::find_if (std::begin (features), std::end (features), uriMatches); return it != std::end (features) ? it->data : nullptr; } }; return &descriptor; } } } #endif