fix: do not leak IPC or context bridge promises (#23321)
This commit is contained in:
parent
088a8256f9
commit
075472d6ef
2 changed files with 57 additions and 21 deletions
|
@ -5,6 +5,7 @@
|
||||||
#include "shell/renderer/api/electron_api_context_bridge.h"
|
#include "shell/renderer/api/electron_api_context_bridge.h"
|
||||||
|
|
||||||
#include <map>
|
#include <map>
|
||||||
|
#include <memory>
|
||||||
#include <set>
|
#include <set>
|
||||||
#include <string>
|
#include <string>
|
||||||
#include <utility>
|
#include <utility>
|
||||||
|
@ -195,18 +196,32 @@ v8::MaybeLocal<v8::Value> PassValueToOtherContext(
|
||||||
v8::Context::Scope destination_scope(destination_context);
|
v8::Context::Scope destination_scope(destination_context);
|
||||||
{
|
{
|
||||||
auto source_promise = v8::Local<v8::Promise>::Cast(value);
|
auto source_promise = v8::Local<v8::Promise>::Cast(value);
|
||||||
auto* proxied_promise = new gin_helper::Promise<v8::Local<v8::Value>>(
|
// Make the promise a shared_ptr so that when the original promise is
|
||||||
destination_context->GetIsolate());
|
// freed the proxy promise is correctly freed as well instead of being
|
||||||
|
// left dangling
|
||||||
|
std::shared_ptr<gin_helper::Promise<v8::Local<v8::Value>>>
|
||||||
|
proxied_promise(new gin_helper::Promise<v8::Local<v8::Value>>(
|
||||||
|
destination_context->GetIsolate()));
|
||||||
v8::Local<v8::Promise> proxied_promise_handle =
|
v8::Local<v8::Promise> proxied_promise_handle =
|
||||||
proxied_promise->GetHandle();
|
proxied_promise->GetHandle();
|
||||||
|
|
||||||
|
v8::Global<v8::Context> global_then_source_context(
|
||||||
|
source_context->GetIsolate(), source_context);
|
||||||
|
v8::Global<v8::Context> global_then_destination_context(
|
||||||
|
destination_context->GetIsolate(), destination_context);
|
||||||
|
global_then_source_context.SetWeak();
|
||||||
|
global_then_destination_context.SetWeak();
|
||||||
auto then_cb = base::BindOnce(
|
auto then_cb = base::BindOnce(
|
||||||
[](gin_helper::Promise<v8::Local<v8::Value>>* proxied_promise,
|
[](std::shared_ptr<gin_helper::Promise<v8::Local<v8::Value>>>
|
||||||
|
proxied_promise,
|
||||||
v8::Isolate* isolate,
|
v8::Isolate* isolate,
|
||||||
v8::Global<v8::Context> global_source_context,
|
v8::Global<v8::Context> global_source_context,
|
||||||
v8::Global<v8::Context> global_destination_context,
|
v8::Global<v8::Context> global_destination_context,
|
||||||
context_bridge::RenderFrameFunctionStore* store,
|
context_bridge::RenderFrameFunctionStore* store,
|
||||||
v8::Local<v8::Value> result) {
|
v8::Local<v8::Value> result) {
|
||||||
|
if (global_source_context.IsEmpty() ||
|
||||||
|
global_destination_context.IsEmpty())
|
||||||
|
return;
|
||||||
context_bridge::ObjectCache object_cache;
|
context_bridge::ObjectCache object_cache;
|
||||||
auto val =
|
auto val =
|
||||||
PassValueToOtherContext(global_source_context.Get(isolate),
|
PassValueToOtherContext(global_source_context.Get(isolate),
|
||||||
|
@ -214,20 +229,28 @@ v8::MaybeLocal<v8::Value> PassValueToOtherContext(
|
||||||
result, store, &object_cache, false, 0);
|
result, store, &object_cache, false, 0);
|
||||||
if (!val.IsEmpty())
|
if (!val.IsEmpty())
|
||||||
proxied_promise->Resolve(val.ToLocalChecked());
|
proxied_promise->Resolve(val.ToLocalChecked());
|
||||||
delete proxied_promise;
|
|
||||||
},
|
},
|
||||||
proxied_promise, destination_context->GetIsolate(),
|
proxied_promise, destination_context->GetIsolate(),
|
||||||
v8::Global<v8::Context>(source_context->GetIsolate(), source_context),
|
std::move(global_then_source_context),
|
||||||
v8::Global<v8::Context>(destination_context->GetIsolate(),
|
std::move(global_then_destination_context), store);
|
||||||
destination_context),
|
|
||||||
store);
|
v8::Global<v8::Context> global_catch_source_context(
|
||||||
|
source_context->GetIsolate(), source_context);
|
||||||
|
v8::Global<v8::Context> global_catch_destination_context(
|
||||||
|
destination_context->GetIsolate(), destination_context);
|
||||||
|
global_catch_source_context.SetWeak();
|
||||||
|
global_catch_destination_context.SetWeak();
|
||||||
auto catch_cb = base::BindOnce(
|
auto catch_cb = base::BindOnce(
|
||||||
[](gin_helper::Promise<v8::Local<v8::Value>>* proxied_promise,
|
[](std::shared_ptr<gin_helper::Promise<v8::Local<v8::Value>>>
|
||||||
|
proxied_promise,
|
||||||
v8::Isolate* isolate,
|
v8::Isolate* isolate,
|
||||||
v8::Global<v8::Context> global_source_context,
|
v8::Global<v8::Context> global_source_context,
|
||||||
v8::Global<v8::Context> global_destination_context,
|
v8::Global<v8::Context> global_destination_context,
|
||||||
context_bridge::RenderFrameFunctionStore* store,
|
context_bridge::RenderFrameFunctionStore* store,
|
||||||
v8::Local<v8::Value> result) {
|
v8::Local<v8::Value> result) {
|
||||||
|
if (global_source_context.IsEmpty() ||
|
||||||
|
global_destination_context.IsEmpty())
|
||||||
|
return;
|
||||||
context_bridge::ObjectCache object_cache;
|
context_bridge::ObjectCache object_cache;
|
||||||
auto val =
|
auto val =
|
||||||
PassValueToOtherContext(global_source_context.Get(isolate),
|
PassValueToOtherContext(global_source_context.Get(isolate),
|
||||||
|
@ -235,13 +258,10 @@ v8::MaybeLocal<v8::Value> PassValueToOtherContext(
|
||||||
result, store, &object_cache, false, 0);
|
result, store, &object_cache, false, 0);
|
||||||
if (!val.IsEmpty())
|
if (!val.IsEmpty())
|
||||||
proxied_promise->Reject(val.ToLocalChecked());
|
proxied_promise->Reject(val.ToLocalChecked());
|
||||||
delete proxied_promise;
|
|
||||||
},
|
},
|
||||||
proxied_promise, destination_context->GetIsolate(),
|
proxied_promise, destination_context->GetIsolate(),
|
||||||
v8::Global<v8::Context>(source_context->GetIsolate(), source_context),
|
std::move(global_catch_source_context),
|
||||||
v8::Global<v8::Context>(destination_context->GetIsolate(),
|
std::move(global_catch_destination_context), store);
|
||||||
destination_context),
|
|
||||||
store);
|
|
||||||
|
|
||||||
ignore_result(source_promise->Then(
|
ignore_result(source_promise->Then(
|
||||||
source_context,
|
source_context,
|
||||||
|
|
|
@ -7,6 +7,7 @@
|
||||||
#include "base/task/post_task.h"
|
#include "base/task/post_task.h"
|
||||||
#include "base/values.h"
|
#include "base/values.h"
|
||||||
#include "content/public/renderer/render_frame.h"
|
#include "content/public/renderer/render_frame.h"
|
||||||
|
#include "content/public/renderer/render_frame_observer.h"
|
||||||
#include "gin/dictionary.h"
|
#include "gin/dictionary.h"
|
||||||
#include "gin/handle.h"
|
#include "gin/handle.h"
|
||||||
#include "gin/object_template_builder.h"
|
#include "gin/object_template_builder.h"
|
||||||
|
@ -36,7 +37,8 @@ RenderFrame* GetCurrentRenderFrame() {
|
||||||
return RenderFrame::FromWebFrame(frame);
|
return RenderFrame::FromWebFrame(frame);
|
||||||
}
|
}
|
||||||
|
|
||||||
class IPCRenderer : public gin::Wrappable<IPCRenderer> {
|
class IPCRenderer : public gin::Wrappable<IPCRenderer>,
|
||||||
|
public content::RenderFrameObserver {
|
||||||
public:
|
public:
|
||||||
static gin::WrapperInfo kWrapperInfo;
|
static gin::WrapperInfo kWrapperInfo;
|
||||||
|
|
||||||
|
@ -44,19 +46,32 @@ class IPCRenderer : public gin::Wrappable<IPCRenderer> {
|
||||||
return gin::CreateHandle(isolate, new IPCRenderer(isolate));
|
return gin::CreateHandle(isolate, new IPCRenderer(isolate));
|
||||||
}
|
}
|
||||||
|
|
||||||
explicit IPCRenderer(v8::Isolate* isolate) {
|
explicit IPCRenderer(v8::Isolate* isolate)
|
||||||
|
: content::RenderFrameObserver(GetCurrentRenderFrame()) {
|
||||||
RenderFrame* render_frame = GetCurrentRenderFrame();
|
RenderFrame* render_frame = GetCurrentRenderFrame();
|
||||||
DCHECK(render_frame);
|
DCHECK(render_frame);
|
||||||
|
weak_context_ =
|
||||||
|
v8::Global<v8::Context>(isolate, isolate->GetCurrentContext());
|
||||||
|
weak_context_.SetWeak();
|
||||||
|
|
||||||
render_frame->GetRemoteInterfaces()->GetInterface(
|
render_frame->GetRemoteInterfaces()->GetInterface(
|
||||||
mojo::MakeRequest(&electron_browser_ptr_));
|
mojo::MakeRequest(&electron_browser_ptr_));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void OnDestruct() override { electron_browser_ptr_.reset(); }
|
||||||
|
|
||||||
|
void WillReleaseScriptContext(v8::Local<v8::Context> context,
|
||||||
|
int32_t world_id) override {
|
||||||
|
if (weak_context_.IsEmpty() ||
|
||||||
|
weak_context_.Get(context->GetIsolate()) == context)
|
||||||
|
electron_browser_ptr_.reset();
|
||||||
|
}
|
||||||
|
|
||||||
// gin::Wrappable:
|
// gin::Wrappable:
|
||||||
gin::ObjectTemplateBuilder GetObjectTemplateBuilder(
|
gin::ObjectTemplateBuilder GetObjectTemplateBuilder(
|
||||||
v8::Isolate* isolate) override {
|
v8::Isolate* isolate) override {
|
||||||
return gin::Wrappable<IPCRenderer>::GetObjectTemplateBuilder(isolate)
|
return gin::Wrappable<IPCRenderer>::GetObjectTemplateBuilder(isolate)
|
||||||
.SetMethod("send", &IPCRenderer::Send)
|
.SetMethod("send", &IPCRenderer::SendMessage)
|
||||||
.SetMethod("sendSync", &IPCRenderer::SendSync)
|
.SetMethod("sendSync", &IPCRenderer::SendSync)
|
||||||
.SetMethod("sendTo", &IPCRenderer::SendTo)
|
.SetMethod("sendTo", &IPCRenderer::SendTo)
|
||||||
.SetMethod("sendToHost", &IPCRenderer::SendToHost)
|
.SetMethod("sendToHost", &IPCRenderer::SendToHost)
|
||||||
|
@ -67,7 +82,7 @@ class IPCRenderer : public gin::Wrappable<IPCRenderer> {
|
||||||
const char* GetTypeName() override { return "IPCRenderer"; }
|
const char* GetTypeName() override { return "IPCRenderer"; }
|
||||||
|
|
||||||
private:
|
private:
|
||||||
void Send(v8::Isolate* isolate,
|
void SendMessage(v8::Isolate* isolate,
|
||||||
bool internal,
|
bool internal,
|
||||||
const std::string& channel,
|
const std::string& channel,
|
||||||
v8::Local<v8::Value> arguments) {
|
v8::Local<v8::Value> arguments) {
|
||||||
|
@ -175,6 +190,7 @@ class IPCRenderer : public gin::Wrappable<IPCRenderer> {
|
||||||
return electron::DeserializeV8Value(isolate, result);
|
return electron::DeserializeV8Value(isolate, result);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
v8::Global<v8::Context> weak_context_;
|
||||||
electron::mojom::ElectronBrowserPtr electron_browser_ptr_;
|
electron::mojom::ElectronBrowserPtr electron_browser_ptr_;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue