From 0bc02f12cf64386ef0d1310c6875116bf4b9e45a Mon Sep 17 00:00:00 2001 From: Jean Pierre Cimalando Date: Mon, 20 Sep 2021 07:02:09 +0200 Subject: [PATCH 1/2] Fix attempt for X11 runloop problems --- plugins/vst/SfizzVstEditor.cpp | 22 ++-- plugins/vst/X11RunLoop.cpp | 210 +++++++++++++++++++++++---------- plugins/vst/X11RunLoop.h | 46 ++++---- 3 files changed, 185 insertions(+), 93 deletions(-) diff --git a/plugins/vst/SfizzVstEditor.cpp b/plugins/vst/SfizzVstEditor.cpp index 5bb967250..2e1506ca9 100644 --- a/plugins/vst/SfizzVstEditor.cpp +++ b/plugins/vst/SfizzVstEditor.cpp @@ -59,11 +59,8 @@ bool PLUGIN_API SfizzVstEditor::open(void* parent, const VSTGUI::PlatformType& p config = &x11config; #endif - Editor* editor = editor_.get(); - if (!editor) { - editor = new Editor(*this); - editor_.reset(editor); - } + Editor* editor = new Editor(*this); + editor_.reset(editor); if (!frame->open(parent, platformType, config)) { fprintf(stderr, "[sfizz] error opening frame\n"); @@ -122,12 +119,21 @@ void PLUGIN_API SfizzVstEditor::close() for (FObject* update : updates_) update->removeDependent(this); - if (editor_) + if (editor_) { editor_->close(); + editor_ = nullptr; + } + if (frame->getNbReference() != 1) frame->forget(); - else + else { frame->close(); +#if !defined(__APPLE__) && !defined(_WIN32) + // if vstgui is done using the runloop, destroy it + if (!RunLoop::get()) + _runLoop = nullptr; +#endif + } this->frame = nullptr; } @@ -158,8 +164,6 @@ CMessageResult SfizzVstEditor::notify(CBaseObject* sender, const char* message) // notifier of X11 events is working. If there is, remove this and // avoid polluting Linux hosts which implement the loop correctly. runLoop->processSomeEvents(); - - runLoop->cleanupDeadHandlers(); } } #endif diff --git a/plugins/vst/X11RunLoop.cpp b/plugins/vst/X11RunLoop.cpp index 990778468..dec080746 100644 --- a/plugins/vst/X11RunLoop.cpp +++ b/plugins/vst/X11RunLoop.cpp @@ -1,33 +1,36 @@ -// SPDX-License-Identifier: GPL-3.0 +// SPDX-License-Identifier: BSD-2-Clause + +// This code is part of the sfizz library and is licensed under a BSD 2-clause +// license. You should have receive a LICENSE.md file along with the code. +// If not, contact the sfizz maintainers at https://github.com/sfztools/sfizz #if !defined(__APPLE__) && !defined(_WIN32) #include "X11RunLoop.h" #include "vstgui/lib/platform/linux/x11platform.h" #include "base/source/fobject.h" +#include +#include namespace VSTGUI { -RunLoop::RunLoop(Steinberg::FUnknown* runLoop) - : runLoop(runLoop) -{ -} +struct RunLoop::Impl { + struct EventHandler; + struct TimerHandler; -RunLoop::~RunLoop() {} + using EventHandlers = std::vector>; + using TimerHandlers = std::vector>; -SharedPointer RunLoop::get() -{ - return X11::RunLoop::get().cast(); -} + EventHandlers eventHandlers; + TimerHandlers timerHandlers; + Steinberg::FUnknownPtr runLoop; +}; -struct RunLoop::EventHandler final : Steinberg::Linux::IEventHandler, public Steinberg::FObject { +//------------------------------------------------------------------------------ +struct RunLoop::Impl::EventHandler final : Steinberg::Linux::IEventHandler, public Steinberg::FObject { X11::IEventHandler* handler { nullptr }; bool alive { false }; - void PLUGIN_API onFDIsSet(Steinberg::Linux::FileDescriptor) override - { - if (alive && handler) - handler->onEvent(); - } + void PLUGIN_API onFDIsSet(Steinberg::Linux::FileDescriptor) override; DELEGATE_REFCOUNT(Steinberg::FObject) DEFINE_INTERFACES @@ -35,15 +38,11 @@ struct RunLoop::EventHandler final : Steinberg::Linux::IEventHandler, public Ste END_DEFINE_INTERFACES(Steinberg::FObject) }; -struct RunLoop::TimerHandler final : Steinberg::Linux::ITimerHandler, public Steinberg::FObject { +struct RunLoop::Impl::TimerHandler final : Steinberg::Linux::ITimerHandler, public Steinberg::FObject { X11::ITimerHandler* handler { nullptr }; bool alive { false }; - void PLUGIN_API onTimer() override - { - if (alive && handler) - handler->onTimer(); - } + void PLUGIN_API onTimer() override; DELEGATE_REFCOUNT(Steinberg::FObject) DEFINE_INTERFACES @@ -51,45 +50,126 @@ struct RunLoop::TimerHandler final : Steinberg::Linux::ITimerHandler, public Ste END_DEFINE_INTERFACES(Steinberg::FObject) }; +//------------------------------------------------------------------------------ +void PLUGIN_API RunLoop::Impl::EventHandler::onFDIsSet(Steinberg::Linux::FileDescriptor) +{ + SharedPointer runLoop = RunLoop::get(); + if (!runLoop) { + fprintf(stderr, "[x11] event has fired without active runloop\n"); + return; + } + + if (alive && handler) + handler->onEvent(); +} + +void PLUGIN_API RunLoop::Impl::TimerHandler::onTimer() +{ + SharedPointer runLoop = RunLoop::get(); + if (!runLoop) { + fprintf(stderr, "[x11] timer has fired without active runloop\n"); + return; + } + + if (alive && handler) + handler->onTimer(); +} + +//------------------------------------------------------------------------------ +RunLoop::RunLoop(Steinberg::FUnknown* runLoop) + : impl(new Impl) +{ + impl->runLoop = runLoop; +} + +RunLoop::~RunLoop() +{ + //dumpCurrentState(); + + if (0) { + // remove any leftover handlers + for (size_t i = 0; i < impl->eventHandlers.size(); ++i) { + const auto& eh = impl->eventHandlers[i]; + if (eh->alive && eh->handler) { + impl->runLoop->unregisterEventHandler(eh.get()); + } + } + for (size_t i = 0; i < impl->timerHandlers.size(); ++i) { + const auto& th = impl->timerHandlers[i]; + if (th->alive && th->handler) { + impl->runLoop->unregisterTimer(th.get()); + } + } + } +} + +SharedPointer RunLoop::get() +{ + return X11::RunLoop::get().cast(); +} void RunLoop::processSomeEvents() { - for (size_t i = 0; i < eventHandlers.size(); ++i) { - const auto& eh = eventHandlers[i]; + for (size_t i = 0; i < impl->eventHandlers.size(); ++i) { + const auto& eh = impl->eventHandlers[i]; if (eh->alive && eh->handler) { eh->handler->onEvent(); } } } -void RunLoop::cleanupDeadHandlers() +void RunLoop::dumpCurrentState() { - for (size_t i = 0; i < eventHandlers.size(); ++i) { - const auto& eh = eventHandlers[i]; - if (!eh->alive) { - runLoop->unregisterEventHandler(eh); - eventHandlers.erase(eventHandlers.begin() + i--); - } + fprintf(stderr, "=== X11 runloop ===\n"); + + fprintf(stderr, "\t" "Event slots:\n"); + for (size_t i = 0, n = impl->eventHandlers.size(); i < n; ++i) { + Impl::EventHandler *eh = impl->eventHandlers[i].get(); + fprintf(stderr, "\t\t" "(%lu) alive=%d handler=%p type=%s\n", i, eh->alive, eh->handler, (eh->alive && eh->handler) ? typeid(*eh->handler).name() : ""); } - for (size_t i = 0; i < timerHandlers.size(); ++i) { - const auto& th = timerHandlers[i]; - if (!th->alive) { - runLoop->unregisterTimer(th); - timerHandlers.erase(timerHandlers.begin() + i--); - } + + fprintf(stderr, "\t" "Timer slots:\n"); + for (size_t i = 0, n = impl->timerHandlers.size(); i < n; ++i) { + Impl::TimerHandler *th = impl->timerHandlers[i].get(); + fprintf(stderr, "\t\t" "(%lu) alive=%d handler=%p type=%s\n", i, th->alive, th->handler, (th->alive && th->handler) ? typeid(*th->handler).name() : ""); + } + + fprintf(stderr, "===/X11 runloop ===\n"); +} + +template +static void insertHandler(std::vector>& list, Steinberg::IPtr handler) +{ + size_t i = 0; + size_t n = list.size(); + while (i < n && list[i]->alive) + ++i; + if (i < n) + list[i] = handler; + else + list.emplace_back(handler); +} + +template +static size_t findHandler(const std::vector>& list, U* handler) +{ + for (size_t i = 0, n = list.size(); i < n; ++i) { + if (list[i]->alive && list[i]->handler == handler) + return i; } + return ~size_t(0); } bool RunLoop::registerEventHandler(int fd, X11::IEventHandler* handler) { - if (!runLoop) + if (!impl->runLoop) return false; - auto smtgHandler = Steinberg::owned(new EventHandler()); + auto smtgHandler = Steinberg::owned(new Impl::EventHandler); smtgHandler->handler = handler; smtgHandler->alive = true; - if (runLoop->registerEventHandler(smtgHandler, fd) == Steinberg::kResultTrue) { - eventHandlers.push_back(smtgHandler); + if (impl->runLoop->registerEventHandler(smtgHandler, fd) == Steinberg::kResultTrue) { + insertHandler(impl->eventHandlers, smtgHandler); return true; } return false; @@ -97,29 +177,31 @@ bool RunLoop::registerEventHandler(int fd, X11::IEventHandler* handler) bool RunLoop::unregisterEventHandler(X11::IEventHandler* handler) { - if (!runLoop) + if (!impl->runLoop) return false; - for (size_t i = 0; i < eventHandlers.size(); ++i) { - const auto& eh = eventHandlers[i]; - if (eh->alive && eh->handler == handler) { - eh->alive = false; - return true; - } - } - return false; + size_t index = findHandler(impl->eventHandlers, handler); + if (index == ~size_t(0)) + return false; + + Impl::EventHandler *eh = impl->eventHandlers[index].get(); + if (!impl->runLoop->unregisterEventHandler(eh)) + return false; + + eh->alive = false; + return true; } bool RunLoop::registerTimer(uint64_t interval, X11::ITimerHandler* handler) { - if (!runLoop) + if (!impl->runLoop) return false; - auto smtgHandler = Steinberg::owned(new TimerHandler()); + auto smtgHandler = Steinberg::owned(new Impl::TimerHandler); smtgHandler->handler = handler; smtgHandler->alive = true; - if (runLoop->registerTimer(smtgHandler, interval) == Steinberg::kResultTrue) { - timerHandlers.push_back(smtgHandler); + if (impl->runLoop->registerTimer(smtgHandler, interval) == Steinberg::kResultTrue) { + insertHandler(impl->timerHandlers, smtgHandler); return true; } return false; @@ -127,17 +209,19 @@ bool RunLoop::registerTimer(uint64_t interval, X11::ITimerHandler* handler) bool RunLoop::unregisterTimer(X11::ITimerHandler* handler) { - if (!runLoop) + if (!impl->runLoop) return false; - for (size_t i = 0; i < timerHandlers.size(); ++i) { - const auto& th = timerHandlers[i]; - if (th->alive && th->handler == handler) { - th->alive = false; - return true; - } - } - return false; + size_t index = findHandler(impl->timerHandlers, handler); + if (index == ~size_t(0)) + return false; + + Impl::TimerHandler *th = impl->timerHandlers[index].get(); + if (!impl->runLoop->unregisterTimer(th)) + return false; + + th->alive = false; + return true; } } // namespace VSTGUI diff --git a/plugins/vst/X11RunLoop.h b/plugins/vst/X11RunLoop.h index cb47b34c3..5755a06ba 100644 --- a/plugins/vst/X11RunLoop.h +++ b/plugins/vst/X11RunLoop.h @@ -1,16 +1,28 @@ -// SPDX-License-Identifier: GPL-3.0 +// SPDX-License-Identifier: BSD-2-Clause + +// This code is part of the sfizz library and is licensed under a BSD 2-clause +// license. You should have receive a LICENSE.md file along with the code. +// If not, contact the sfizz maintainers at https://github.com/sfztools/sfizz + /* - This is a modified version the X11 run loop from vst3editor.cpp. + This runloop connects to X11 VSTGUI, it connects VST3 and VSTGUI together. + The Windows and macOS runloops do not need this, the OS-provided + functionality is used instead. - This version is edited to add more safeguards to protect against host bugs. - It also permits to call event processing externally in case the host has a - defective X11 event loop notifier. + Previously, this was based on VSTGUI code provided by Steinberg. + This is replaced with a rewrite, because the original code has too many + issues. For example, it has no robustness in case handlers get added or + removed within the execution of the handler. + + This version allows to call event processing externally, in case the host + has a defective X11 event loop notifier. (some versions of Bitwig do) */ #pragma once #if !defined(__APPLE__) && !defined(_WIN32) #include "vstgui/lib/platform/linux/x11frame.h" #include "pluginterfaces/gui/iplugview.h" +#include namespace VSTGUI { @@ -22,24 +34,16 @@ class RunLoop final : public X11::IRunLoop, public AtomicReferenceCounted { static SharedPointer get(); void processSomeEvents(); - void cleanupDeadHandlers(); + void dumpCurrentState(); // X11::IRunLoop - bool registerEventHandler(int fd, X11::IEventHandler* handler); - bool unregisterEventHandler(X11::IEventHandler* handler); - bool registerTimer(uint64_t interval, X11::ITimerHandler* handler); - bool unregisterTimer(X11::ITimerHandler* handler); - -private: - struct EventHandler; - struct TimerHandler; - -private: - using EventHandlers = std::vector>; - using TimerHandlers = std::vector>; - EventHandlers eventHandlers; - TimerHandlers timerHandlers; - Steinberg::FUnknownPtr runLoop; + bool registerEventHandler(int fd, X11::IEventHandler* handler) override; + bool unregisterEventHandler(X11::IEventHandler* handler) override; + bool registerTimer(uint64_t interval, X11::ITimerHandler* handler) override; + bool unregisterTimer(X11::ITimerHandler* handler) override; + + struct Impl; + std::unique_ptr impl; }; } // namespace VSTGUI From b2368d6e80592d54a8e65c28a8257b94a9a1db8e Mon Sep 17 00:00:00 2001 From: Jean Pierre Cimalando Date: Mon, 20 Sep 2021 07:59:38 +0200 Subject: [PATCH 2/2] Add a couple of includes used in the file [ci skip] --- plugins/vst/X11RunLoop.cpp | 2 ++ 1 file changed, 2 insertions(+) diff --git a/plugins/vst/X11RunLoop.cpp b/plugins/vst/X11RunLoop.cpp index dec080746..04725c5a1 100644 --- a/plugins/vst/X11RunLoop.cpp +++ b/plugins/vst/X11RunLoop.cpp @@ -9,6 +9,8 @@ #include "vstgui/lib/platform/linux/x11platform.h" #include "base/source/fobject.h" #include +#include +#include #include namespace VSTGUI {