forked from chromium/chromium
-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathgc_controller.cc
117 lines (96 loc) · 4.04 KB
/
gc_controller.cc
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
// Copyright 2014 The Chromium Authors
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "content/web_test/renderer/gc_controller.h"
#include <tuple>
#include "base/functional/bind.h"
#include "base/task/single_thread_task_runner.h"
#include "gin/arguments.h"
#include "gin/handle.h"
#include "gin/object_template_builder.h"
#include "third_party/blink/public/platform/scheduler/web_agent_group_scheduler.h"
#include "third_party/blink/public/web/web_local_frame.h"
#include "v8/include/v8.h"
namespace content {
gin::WrapperInfo GCController::kWrapperInfo = {gin::kEmbedderNativeGin};
// static
void GCController::Install(blink::WebLocalFrame* frame) {
v8::Isolate* isolate = frame->GetAgentGroupScheduler()->Isolate();
v8::HandleScope handle_scope(isolate);
v8::Local<v8::Context> context = frame->MainWorldScriptContext();
if (context.IsEmpty())
return;
v8::Context::Scope context_scope(context);
gin::Handle<GCController> controller =
gin::CreateHandle(isolate, new GCController(frame));
if (controller.IsEmpty())
return;
v8::Local<v8::Object> global = context->Global();
global
->Set(context, gin::StringToV8(isolate, "GCController"),
controller.ToV8())
.Check();
}
GCController::GCController(blink::WebLocalFrame* frame) : frame_(frame) {}
GCController::~GCController() = default;
gin::ObjectTemplateBuilder GCController::GetObjectTemplateBuilder(
v8::Isolate* isolate) {
return gin::Wrappable<GCController>::GetObjectTemplateBuilder(isolate)
.SetMethod("collect", &GCController::Collect)
.SetMethod("collectAll", &GCController::CollectAll)
.SetMethod("minorCollect", &GCController::MinorCollect)
.SetMethod("asyncCollectAll", &GCController::AsyncCollectAll);
}
void GCController::Collect(const gin::Arguments& args) {
args.isolate()->RequestGarbageCollectionForTesting(
v8::Isolate::kFullGarbageCollection);
}
void GCController::CollectAll(const gin::Arguments& args) {
for (int i = 0; i < kNumberOfGCsForFullCollection; i++) {
args.isolate()->RequestGarbageCollectionForTesting(
v8::Isolate::kFullGarbageCollection);
}
}
void GCController::AsyncCollectAll(const gin::Arguments& args) {
v8::HandleScope scope(args.isolate());
if (args.PeekNext().IsEmpty() || !args.PeekNext()->IsFunction()) {
args.ThrowTypeError(
"asyncCollectAll should be called with a callback argument being a "
"v8::Function.");
return;
}
v8::UniquePersistent<v8::Function> js_callback(
args.isolate(), v8::Local<v8::Function>::Cast(args.PeekNext()));
// Bind the js callback into a OnceClosure that will be run asynchronously in
// a fresh call stack.
base::OnceClosure run_async =
base::BindOnce(&GCController::AsyncCollectAllWithEmptyStack,
weak_ptr_factory_.GetWeakPtr(), std::move(js_callback));
frame_->GetTaskRunner(blink::TaskType::kInternalTest)
->PostTask(FROM_HERE, std::move(run_async));
}
void GCController::AsyncCollectAllWithEmptyStack(
v8::UniquePersistent<v8::Function> callback) {
v8::Isolate* const isolate = frame_->GetAgentGroupScheduler()->Isolate();
for (int i = 0; i < kNumberOfGCsForFullCollection; i++) {
isolate->RequestGarbageCollectionForTesting(
v8::Isolate::kFullGarbageCollection,
cppgc::EmbedderStackState::kNoHeapPointers);
}
v8::HandleScope scope(isolate);
v8::Local<v8::Function> func = callback.Get(isolate);
v8::Local<v8::Context> context = func->GetCreationContextChecked(isolate);
v8::Context::Scope context_scope(context);
v8::TryCatch try_catch(isolate);
v8::MicrotasksScope microtasks_scope(
isolate, context->GetMicrotaskQueue(),
v8::MicrotasksScope::kDoNotRunMicrotasks);
auto result = func->Call(context, context->Global(), 0, nullptr);
// Swallow potential exception.
std::ignore = result;
}
void GCController::MinorCollect(const gin::Arguments& args) {
args.isolate()->RequestGarbageCollectionForTesting(
v8::Isolate::kMinorGarbageCollection);
}
} // namespace content