Allow passing null pointers to IPlugView::setFrame

This commit is contained in:
Robbert van der Helm
2021-01-24 14:44:42 +01:00
parent d5e4424463
commit 0044bc6b60
4 changed files with 35 additions and 13 deletions
+7 -1
View File
@@ -629,7 +629,13 @@ bool Vst3Logger::log_request(bool is_host_vst,
const YaPlugView::SetFrame& request) { const YaPlugView::SetFrame& request) {
return log_request_base(is_host_vst, [&](auto& message) { return log_request_base(is_host_vst, [&](auto& message) {
message << request.owner_instance_id message << request.owner_instance_id
<< ": IPlugView::setFrame(frame = <IPlugFrame*>)"; << ": IPlugView::setFrame(frame = ";
if (request.plug_frame_args) {
message << "<IPlugFrame*>";
} else {
message << "<nullptr>";
}
message << ")";
}); });
} }
@@ -17,6 +17,7 @@
#pragma once #pragma once
#include <pluginterfaces/gui/iplugview.h> #include <pluginterfaces/gui/iplugview.h>
#include "bitsery/ext/std_optional.h"
#include "../../common.h" #include "../../common.h"
#include "../base.h" #include "../base.h"
@@ -291,12 +292,18 @@ class YaPlugView : public Steinberg::IPlugView {
native_size_t owner_instance_id; native_size_t owner_instance_id;
Vst3PlugFrameProxy::ConstructArgs plug_frame_args; /**
* Some hosts will pass a null pointer to explicitly destroy the
* `IPlugFrame` instance before freeing the plugin's `IPlugView`
* instance. This also happens in the SDK, so this seems like valid
* behaviour we should support.
*/
std::optional<Vst3PlugFrameProxy::ConstructArgs> plug_frame_args;
template <typename S> template <typename S>
void serialize(S& s) { void serialize(S& s) {
s.value8b(owner_instance_id); s.value8b(owner_instance_id);
s.object(plug_frame_args); s.ext(plug_frame_args, bitsery::ext::StdOptional{});
} }
}; };
@@ -200,7 +200,8 @@ tresult PLUGIN_API Vst3PlugViewProxyImpl::onFocus(TBool state) {
tresult PLUGIN_API tresult PLUGIN_API
Vst3PlugViewProxyImpl::setFrame(Steinberg::IPlugFrame* frame) { Vst3PlugViewProxyImpl::setFrame(Steinberg::IPlugFrame* frame) {
// TODO: Null pointers are valid here, should we pass them through? // Null pointers are valid here going from the reference implementations in
// the SDK
if (frame) { if (frame) {
// We'll store the pointer for when the plugin later makes a callback to // We'll store the pointer for when the plugin later makes a callback to
// this component handler // this component handler
@@ -229,9 +230,12 @@ Vst3PlugViewProxyImpl::setFrame(Steinberg::IPlugFrame* frame) {
.plug_frame_args = Vst3PlugFrameProxy::ConstructArgs( .plug_frame_args = Vst3PlugFrameProxy::ConstructArgs(
plug_frame, owner_instance_id())}); plug_frame, owner_instance_id())});
} else { } else {
bridge.logger.log( plug_frame.reset();
"WARNING: Null pointer passed to 'IPlugView::setFrame()'"); run_loop_tasks.reset();
return Steinberg::kInvalidArgument;
return send_mutually_recursive_message(
YaPlugView::SetFrame{.owner_instance_id = owner_instance_id(),
.plug_frame_args = std::nullopt});
} }
} }
+11 -6
View File
@@ -737,13 +737,18 @@ void Vst3Bridge::run() {
}, },
[&](YaPlugView::SetFrame& request) [&](YaPlugView::SetFrame& request)
-> YaPlugView::SetFrame::Response { -> YaPlugView::SetFrame::Response {
// We'll create a proxy object for the `IPlugFrame` object and // If the host passed a valid `IPlugFrame*`, then We'll create a
// pass that to the `setFrame()` function. The lifetime of this // proxy object for the `IPlugFrame` object and pass that to the
// object is tied to that of the actual `IPlugFrame` object // `setFrame()` function. The lifetime of this object is tied to
// we're passing this proxy to. // that of the actual `IPlugFrame` object we're passing this
// proxy to. IF the host passed a null pointer (which seems to
// be common when terminating plugins) we'll do the same thing
// here.
object_instances[request.owner_instance_id].plug_frame_proxy = object_instances[request.owner_instance_id].plug_frame_proxy =
Steinberg::owned(new Vst3PlugFrameProxyImpl( request.plug_frame_args
*this, std::move(request.plug_frame_args))); ? Steinberg::owned(new Vst3PlugFrameProxyImpl(
*this, std::move(*request.plug_frame_args)))
: nullptr;
// This likely doesn't have to be run from the GUI thread, but // This likely doesn't have to be run from the GUI thread, but
// since 80% of the `IPlugView` functions have to be we'll do it // since 80% of the `IPlugView` functions have to be we'll do it