• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 // Copyright 2012 the V8 project authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
4 
5 #include "src/api/api.h"
6 
7 #include <algorithm>  // For min
8 #include <cmath>      // For isnan.
9 #include <limits>
10 #include <sstream>
11 #include <string>
12 #include <utility>  // For move
13 #include <vector>
14 
15 #include "include/v8-callbacks.h"
16 #include "include/v8-cppgc.h"
17 #include "include/v8-date.h"
18 #include "include/v8-embedder-state-scope.h"
19 #include "include/v8-extension.h"
20 #include "include/v8-fast-api-calls.h"
21 #include "include/v8-function.h"
22 #include "include/v8-json.h"
23 #include "include/v8-locker.h"
24 #include "include/v8-primitive-object.h"
25 #include "include/v8-profiler.h"
26 #include "include/v8-unwinder-state.h"
27 #include "include/v8-util.h"
28 #include "include/v8-wasm.h"
29 #include "src/api/api-inl.h"
30 #include "src/api/api-natives.h"
31 #include "src/base/functional.h"
32 #include "src/base/logging.h"
33 #include "src/base/platform/platform.h"
34 #include "src/base/platform/time.h"
35 #include "src/base/safe_conversions.h"
36 #include "src/base/utils/random-number-generator.h"
37 #include "src/baseline/baseline-batch-compiler.h"
38 #include "src/builtins/accessors.h"
39 #include "src/builtins/builtins-utils.h"
40 #include "src/codegen/compiler.h"
41 #include "src/codegen/cpu-features.h"
42 #include "src/codegen/script-details.h"
43 #include "src/common/assert-scope.h"
44 #include "src/common/globals.h"
45 #include "src/compiler-dispatcher/lazy-compile-dispatcher.h"
46 #include "src/date/date.h"
47 #include "src/objects/primitive-heap-object.h"
48 #if V8_ENABLE_WEBASSEMBLY
49 #include "src/debug/debug-wasm-objects.h"
50 #endif  // V8_ENABLE_WEBASSEMBLY
51 #include "src/debug/liveedit.h"
52 #include "src/deoptimizer/deoptimizer.h"
53 #include "src/execution/embedder-state.h"
54 #include "src/execution/execution.h"
55 #include "src/execution/frames-inl.h"
56 #include "src/execution/isolate-inl.h"
57 #include "src/execution/messages.h"
58 #include "src/execution/microtask-queue.h"
59 #include "src/execution/simulator.h"
60 #include "src/execution/v8threads.h"
61 #include "src/execution/vm-state-inl.h"
62 #include "src/handles/global-handles.h"
63 #include "src/handles/persistent-handles.h"
64 #include "src/heap/embedder-tracing.h"
65 #include "src/heap/heap-inl.h"
66 #include "src/heap/heap-write-barrier.h"
67 #include "src/heap/safepoint.h"
68 #include "src/init/bootstrapper.h"
69 #include "src/init/icu_util.h"
70 #include "src/init/startup-data-util.h"
71 #include "src/init/v8.h"
72 #include "src/json/json-parser.h"
73 #include "src/json/json-stringifier.h"
74 #include "src/logging/counters-scopes.h"
75 #include "src/logging/metrics.h"
76 #include "src/logging/runtime-call-stats-scope.h"
77 #include "src/logging/tracing-flags.h"
78 #include "src/numbers/conversions-inl.h"
79 #include "src/objects/api-callbacks.h"
80 #include "src/objects/contexts.h"
81 #include "src/objects/embedder-data-array-inl.h"
82 #include "src/objects/embedder-data-slot-inl.h"
83 #include "src/objects/hash-table-inl.h"
84 #include "src/objects/heap-object.h"
85 #include "src/objects/js-array-buffer-inl.h"
86 #include "src/objects/js-array-inl.h"
87 #include "src/objects/js-collection-inl.h"
88 #include "src/objects/js-promise-inl.h"
89 #include "src/objects/js-regexp-inl.h"
90 #include "src/objects/js-weak-refs-inl.h"
91 #include "src/objects/module-inl.h"
92 #include "src/objects/objects-inl.h"
93 #include "src/objects/oddball.h"
94 #include "src/objects/ordered-hash-table-inl.h"
95 #include "src/objects/property-descriptor.h"
96 #include "src/objects/property-details.h"
97 #include "src/objects/property.h"
98 #include "src/objects/prototype.h"
99 #include "src/objects/shared-function-info.h"
100 #include "src/objects/slots.h"
101 #include "src/objects/smi.h"
102 #include "src/objects/synthetic-module-inl.h"
103 #include "src/objects/templates.h"
104 #include "src/objects/value-serializer.h"
105 #include "src/parsing/parse-info.h"
106 #include "src/parsing/parser.h"
107 #include "src/parsing/pending-compilation-error-handler.h"
108 #include "src/parsing/scanner-character-streams.h"
109 #include "src/profiler/cpu-profiler.h"
110 #include "src/profiler/heap-profiler.h"
111 #include "src/profiler/heap-snapshot-generator-inl.h"
112 #include "src/profiler/profile-generator-inl.h"
113 #include "src/profiler/tick-sample.h"
114 #include "src/regexp/regexp-utils.h"
115 #include "src/runtime/runtime.h"
116 #include "src/sandbox/external-pointer.h"
117 #include "src/sandbox/sandbox.h"
118 #include "src/snapshot/code-serializer.h"
119 #include "src/snapshot/embedded/embedded-data.h"
120 #include "src/snapshot/snapshot.h"
121 #include "src/snapshot/startup-serializer.h"  // For SerializedHandleChecker.
122 #include "src/strings/char-predicates-inl.h"
123 #include "src/strings/string-hasher.h"
124 #include "src/strings/unicode-inl.h"
125 #include "src/tracing/trace-event.h"
126 #include "src/utils/detachable-vector.h"
127 #include "src/utils/version.h"
128 #include "src/web-snapshot/web-snapshot.h"
129 
130 #if V8_ENABLE_WEBASSEMBLY
131 #include "src/trap-handler/trap-handler.h"
132 #include "src/wasm/streaming-decoder.h"
133 #include "src/wasm/value-type.h"
134 #include "src/wasm/wasm-engine.h"
135 #include "src/wasm/wasm-js.h"
136 #include "src/wasm/wasm-objects-inl.h"
137 #include "src/wasm/wasm-result.h"
138 #include "src/wasm/wasm-serialization.h"
139 #endif  // V8_ENABLE_WEBASSEMBLY
140 
141 #if V8_OS_LINUX || V8_OS_DARWIN || V8_OS_FREEBSD
142 #include <signal.h>
143 #include "include/v8-wasm-trap-handler-posix.h"
144 #include "src/trap-handler/handler-inside-posix.h"
145 #endif
146 
147 #if V8_OS_WIN
148 #include <windows.h>
149 
150 // This has to come after windows.h.
151 #include <versionhelpers.h>
152 
153 #include "include/v8-wasm-trap-handler-win.h"
154 #include "src/trap-handler/handler-inside-win.h"
155 #if defined(V8_OS_WIN64)
156 #include "src/base/platform/wrappers.h"
157 #include "src/diagnostics/unwinding-info-win64.h"
158 #endif  // V8_OS_WIN64
159 #endif  // V8_OS_WIN
160 
161 // Has to be the last include (doesn't have include guards):
162 #include "src/api/api-macros.h"
163 
164 #define TRACE_BS(...)                                     \
165   do {                                                    \
166     if (i::FLAG_trace_backing_store) PrintF(__VA_ARGS__); \
167   } while (false)
168 
169 namespace v8 {
170 
171 static OOMErrorCallback g_oom_error_callback = nullptr;
172 
GetScriptOriginForScript(i::Isolate * isolate,i::Handle<i::Script> script)173 static ScriptOrigin GetScriptOriginForScript(i::Isolate* isolate,
174                                              i::Handle<i::Script> script) {
175   i::Handle<i::Object> scriptName(script->GetNameOrSourceURL(), isolate);
176   i::Handle<i::Object> source_map_url(script->source_mapping_url(), isolate);
177   i::Handle<i::Object> host_defined_options(script->host_defined_options(),
178                                             isolate);
179   ScriptOriginOptions options(script->origin_options());
180   bool is_wasm = false;
181 #if V8_ENABLE_WEBASSEMBLY
182   is_wasm = script->type() == i::Script::TYPE_WASM;
183 #endif  // V8_ENABLE_WEBASSEMBLY
184   v8::ScriptOrigin origin(
185       reinterpret_cast<v8::Isolate*>(isolate), Utils::ToLocal(scriptName),
186       script->line_offset(), script->column_offset(),
187       options.IsSharedCrossOrigin(), script->id(),
188       Utils::ToLocal(source_map_url), options.IsOpaque(), is_wasm,
189       options.IsModule(), Utils::ToLocal(host_defined_options));
190   return origin;
191 }
192 
HostDefinedOptions() const193 Local<PrimitiveArray> ScriptOrigin::HostDefinedOptions() const {
194   // TODO(cbruni, chromium:1244145): remove once migrated to the context.
195   Utils::ApiCheck(!host_defined_options_->IsFixedArray(),
196                   "ScriptOrigin::HostDefinedOptions",
197                   "HostDefinedOptions is not a PrimitiveArray, please use "
198                   "ScriptOrigin::GetHostDefinedOptions()");
199   i::Handle<i::FixedArray> options =
200       Utils::OpenHandle(*host_defined_options_.As<FixedArray>());
201   return Utils::PrimitiveArrayToLocal(options);
202 }
203 
204 // --- E x c e p t i o n   B e h a v i o r ---
205 
FatalProcessOutOfMemory(i::Isolate * isolate,const char * location)206 void i::FatalProcessOutOfMemory(i::Isolate* isolate, const char* location) {
207   i::V8::FatalProcessOutOfMemory(isolate, location, false);
208 }
209 
210 // When V8 cannot allocate memory FatalProcessOutOfMemory is called. The default
211 // OOM error handler is called and execution is stopped.
FatalProcessOutOfMemory(i::Isolate * isolate,const char * location,bool is_heap_oom)212 void i::V8::FatalProcessOutOfMemory(i::Isolate* isolate, const char* location,
213                                     bool is_heap_oom) {
214   char last_few_messages[Heap::kTraceRingBufferSize + 1];
215   char js_stacktrace[Heap::kStacktraceBufferSize + 1];
216   i::HeapStats heap_stats;
217 
218   if (isolate == nullptr) {
219     isolate = Isolate::TryGetCurrent();
220   }
221 
222   if (isolate == nullptr) {
223     // If the Isolate is not available for the current thread we cannot retrieve
224     // memory information from the Isolate. Write easy-to-recognize values on
225     // the stack.
226     memset(last_few_messages, 0x0BADC0DE, Heap::kTraceRingBufferSize + 1);
227     memset(js_stacktrace, 0x0BADC0DE, Heap::kStacktraceBufferSize + 1);
228     memset(&heap_stats, 0xBADC0DE, sizeof(heap_stats));
229     // Give the embedder a chance to handle the condition. If it doesn't,
230     // just crash.
231     if (g_oom_error_callback) g_oom_error_callback(location, is_heap_oom);
232     FATAL("Fatal process out of memory: %s", location);
233     UNREACHABLE();
234   }
235 
236   memset(last_few_messages, 0, Heap::kTraceRingBufferSize + 1);
237   memset(js_stacktrace, 0, Heap::kStacktraceBufferSize + 1);
238 
239   intptr_t start_marker;
240   heap_stats.start_marker = &start_marker;
241   size_t ro_space_size;
242   heap_stats.ro_space_size = &ro_space_size;
243   size_t ro_space_capacity;
244   heap_stats.ro_space_capacity = &ro_space_capacity;
245   size_t new_space_size;
246   heap_stats.new_space_size = &new_space_size;
247   size_t new_space_capacity;
248   heap_stats.new_space_capacity = &new_space_capacity;
249   size_t old_space_size;
250   heap_stats.old_space_size = &old_space_size;
251   size_t old_space_capacity;
252   heap_stats.old_space_capacity = &old_space_capacity;
253   size_t code_space_size;
254   heap_stats.code_space_size = &code_space_size;
255   size_t code_space_capacity;
256   heap_stats.code_space_capacity = &code_space_capacity;
257   size_t map_space_size;
258   heap_stats.map_space_size = &map_space_size;
259   size_t map_space_capacity;
260   heap_stats.map_space_capacity = &map_space_capacity;
261   size_t lo_space_size;
262   heap_stats.lo_space_size = &lo_space_size;
263   size_t code_lo_space_size;
264   heap_stats.code_lo_space_size = &code_lo_space_size;
265   size_t global_handle_count;
266   heap_stats.global_handle_count = &global_handle_count;
267   size_t weak_global_handle_count;
268   heap_stats.weak_global_handle_count = &weak_global_handle_count;
269   size_t pending_global_handle_count;
270   heap_stats.pending_global_handle_count = &pending_global_handle_count;
271   size_t near_death_global_handle_count;
272   heap_stats.near_death_global_handle_count = &near_death_global_handle_count;
273   size_t free_global_handle_count;
274   heap_stats.free_global_handle_count = &free_global_handle_count;
275   size_t memory_allocator_size;
276   heap_stats.memory_allocator_size = &memory_allocator_size;
277   size_t memory_allocator_capacity;
278   heap_stats.memory_allocator_capacity = &memory_allocator_capacity;
279   size_t malloced_memory;
280   heap_stats.malloced_memory = &malloced_memory;
281   size_t malloced_peak_memory;
282   heap_stats.malloced_peak_memory = &malloced_peak_memory;
283   size_t objects_per_type[LAST_TYPE + 1] = {0};
284   heap_stats.objects_per_type = objects_per_type;
285   size_t size_per_type[LAST_TYPE + 1] = {0};
286   heap_stats.size_per_type = size_per_type;
287   int os_error;
288   heap_stats.os_error = &os_error;
289   heap_stats.last_few_messages = last_few_messages;
290   heap_stats.js_stacktrace = js_stacktrace;
291   intptr_t end_marker;
292   heap_stats.end_marker = &end_marker;
293   if (isolate->heap()->HasBeenSetUp()) {
294     // BUG(1718): Don't use the take_snapshot since we don't support
295     // HeapObjectIterator here without doing a special GC.
296     isolate->heap()->RecordStats(&heap_stats, false);
297     if (!FLAG_correctness_fuzzer_suppressions) {
298       char* first_newline = strchr(last_few_messages, '\n');
299       if (first_newline == nullptr || first_newline[1] == '\0')
300         first_newline = last_few_messages;
301       base::OS::PrintError("\n<--- Last few GCs --->\n%s\n", first_newline);
302       base::OS::PrintError("\n<--- JS stacktrace --->\n%s\n", js_stacktrace);
303     }
304   }
305   Utils::ReportOOMFailure(isolate, location, is_heap_oom);
306   if (g_oom_error_callback) g_oom_error_callback(location, is_heap_oom);
307   // If the fatal error handler returns, we stop execution.
308   FATAL("API fatal error handler returned after process out of memory");
309 }
310 
ReportApiFailure(const char * location,const char * message)311 void Utils::ReportApiFailure(const char* location, const char* message) {
312   i::Isolate* isolate = i::Isolate::TryGetCurrent();
313   FatalErrorCallback callback = nullptr;
314   if (isolate != nullptr) {
315     callback = isolate->exception_behavior();
316   }
317   if (callback == nullptr) {
318     base::OS::PrintError("\n#\n# Fatal error in %s\n# %s\n#\n\n", location,
319                          message);
320     base::OS::Abort();
321   } else {
322     callback(location, message);
323   }
324   isolate->SignalFatalError();
325 }
326 
ReportOOMFailure(i::Isolate * isolate,const char * location,bool is_heap_oom)327 void Utils::ReportOOMFailure(i::Isolate* isolate, const char* location,
328                              bool is_heap_oom) {
329   OOMErrorCallback oom_callback = isolate->oom_behavior();
330   if (oom_callback == nullptr) {
331     // TODO(wfh): Remove this fallback once Blink is setting OOM handler. See
332     // crbug.com/614440.
333     FatalErrorCallback fatal_callback = isolate->exception_behavior();
334     if (fatal_callback == nullptr) {
335       base::OS::PrintError("\n#\n# Fatal %s OOM in %s\n#\n\n",
336                            is_heap_oom ? "javascript" : "process", location);
337 #ifdef V8_FUZZILLI
338       exit(0);
339 #else
340       base::OS::Abort();
341 #endif  // V8_FUZZILLI
342     } else {
343       fatal_callback(location,
344                      is_heap_oom
345                          ? "Allocation failed - JavaScript heap out of memory"
346                          : "Allocation failed - process out of memory");
347     }
348   } else {
349     oom_callback(location, is_heap_oom);
350   }
351   isolate->SignalFatalError();
352 }
353 
SetSnapshotDataBlob(StartupData * snapshot_blob)354 void V8::SetSnapshotDataBlob(StartupData* snapshot_blob) {
355   i::V8::SetSnapshotBlob(snapshot_blob);
356 }
357 
358 namespace {
359 
360 #ifdef V8_SANDBOXED_POINTERS
361 // ArrayBufferAllocator to use when sandboxed pointers are used in which case
362 // all ArrayBuffer backing stores need to be allocated inside the sandbox.
363 // Note, the current implementation is extremely inefficient as it uses the
364 // BoundedPageAllocator. In the future, we'll need a proper allocator
365 // implementation.
366 class ArrayBufferAllocator : public v8::ArrayBuffer::Allocator {
367  public:
ArrayBufferAllocator()368   ArrayBufferAllocator() { CHECK(page_allocator_); }
369 
Allocate(size_t length)370   void* Allocate(size_t length) override {
371     return page_allocator_->AllocatePages(nullptr, RoundUp(length, page_size_),
372                                           page_size_,
373                                           PageAllocator::kReadWrite);
374   }
375 
AllocateUninitialized(size_t length)376   void* AllocateUninitialized(size_t length) override {
377     return Allocate(length);
378   }
379 
Free(void * data,size_t length)380   void Free(void* data, size_t length) override {
381     page_allocator_->FreePages(data, RoundUp(length, page_size_));
382   }
383 
384  private:
385   PageAllocator* page_allocator_ = internal::GetArrayBufferPageAllocator();
386   const size_t page_size_ = page_allocator_->AllocatePageSize();
387 };
388 
389 #else
390 
391 class ArrayBufferAllocator : public v8::ArrayBuffer::Allocator {
392  public:
393   void* Allocate(size_t length) override {
394 #if V8_OS_AIX && _LINUX_SOURCE_COMPAT
395     // Work around for GCC bug on AIX
396     // See: https://gcc.gnu.org/bugzilla/show_bug.cgi?id=79839
397     void* data = __linux_calloc(length, 1);
398 #else
399     void* data = base::Calloc(length, 1);
400 #endif
401     return data;
402   }
403 
404   void* AllocateUninitialized(size_t length) override {
405 #if V8_OS_AIX && _LINUX_SOURCE_COMPAT
406     // Work around for GCC bug on AIX
407     // See: https://gcc.gnu.org/bugzilla/show_bug.cgi?id=79839
408     void* data = __linux_malloc(length);
409 #else
410     void* data = base::Malloc(length);
411 #endif
412     return data;
413   }
414 
415   void Free(void* data, size_t) override { base::Free(data); }
416 
417   void* Reallocate(void* data, size_t old_length, size_t new_length) override {
418 #if V8_OS_AIX && _LINUX_SOURCE_COMPAT
419     // Work around for GCC bug on AIX
420     // See: https://gcc.gnu.org/bugzilla/show_bug.cgi?id=79839
421     void* new_data = __linux_realloc(data, new_length);
422 #else
423     void* new_data = base::Realloc(data, new_length);
424 #endif
425     if (new_length > old_length) {
426       memset(reinterpret_cast<uint8_t*>(new_data) + old_length, 0,
427              new_length - old_length);
428     }
429     return new_data;
430   }
431 };
432 #endif  // V8_SANDBOXED_POINTERS
433 
434 struct SnapshotCreatorData {
SnapshotCreatorDatav8::__anon1d4f464a0111::SnapshotCreatorData435   explicit SnapshotCreatorData(Isolate* isolate)
436       : isolate_(isolate),
437         default_context_(),
438         contexts_(isolate),
439         created_(false) {}
440 
castv8::__anon1d4f464a0111::SnapshotCreatorData441   static SnapshotCreatorData* cast(void* data) {
442     return reinterpret_cast<SnapshotCreatorData*>(data);
443   }
444 
445   ArrayBufferAllocator allocator_;
446   Isolate* isolate_;
447   Persistent<Context> default_context_;
448   SerializeInternalFieldsCallback default_embedder_fields_serializer_;
449   PersistentValueVector<Context> contexts_;
450   std::vector<SerializeInternalFieldsCallback> embedder_fields_serializers_;
451   bool created_;
452 };
453 
454 }  // namespace
455 
SnapshotCreator(Isolate * isolate,const intptr_t * external_references,StartupData * existing_snapshot)456 SnapshotCreator::SnapshotCreator(Isolate* isolate,
457                                  const intptr_t* external_references,
458                                  StartupData* existing_snapshot) {
459   SnapshotCreatorData* data = new SnapshotCreatorData(isolate);
460   i::Isolate* internal_isolate = reinterpret_cast<i::Isolate*>(isolate);
461   internal_isolate->set_array_buffer_allocator(&data->allocator_);
462   internal_isolate->set_api_external_references(external_references);
463   internal_isolate->enable_serializer();
464   isolate->Enter();
465   const StartupData* blob = existing_snapshot
466                                 ? existing_snapshot
467                                 : i::Snapshot::DefaultSnapshotBlob();
468   if (blob && blob->raw_size > 0) {
469     internal_isolate->set_snapshot_blob(blob);
470     i::Snapshot::Initialize(internal_isolate);
471   } else {
472     internal_isolate->InitWithoutSnapshot();
473   }
474   data_ = data;
475   // Disable batch compilation during snapshot creation.
476   internal_isolate->baseline_batch_compiler()->set_enabled(false);
477 }
478 
SnapshotCreator(const intptr_t * external_references,StartupData * existing_snapshot)479 SnapshotCreator::SnapshotCreator(const intptr_t* external_references,
480                                  StartupData* existing_snapshot)
481     : SnapshotCreator(Isolate::Allocate(), external_references,
482                       existing_snapshot) {}
483 
~SnapshotCreator()484 SnapshotCreator::~SnapshotCreator() {
485   SnapshotCreatorData* data = SnapshotCreatorData::cast(data_);
486   Isolate* isolate = data->isolate_;
487   isolate->Exit();
488   isolate->Dispose();
489   delete data;
490 }
491 
GetIsolate()492 Isolate* SnapshotCreator::GetIsolate() {
493   return SnapshotCreatorData::cast(data_)->isolate_;
494 }
495 
SetDefaultContext(Local<Context> context,SerializeInternalFieldsCallback callback)496 void SnapshotCreator::SetDefaultContext(
497     Local<Context> context, SerializeInternalFieldsCallback callback) {
498   DCHECK(!context.IsEmpty());
499   SnapshotCreatorData* data = SnapshotCreatorData::cast(data_);
500   DCHECK(!data->created_);
501   DCHECK(data->default_context_.IsEmpty());
502   Isolate* isolate = data->isolate_;
503   CHECK_EQ(isolate, context->GetIsolate());
504   data->default_context_.Reset(isolate, context);
505   data->default_embedder_fields_serializer_ = callback;
506 }
507 
AddContext(Local<Context> context,SerializeInternalFieldsCallback callback)508 size_t SnapshotCreator::AddContext(Local<Context> context,
509                                    SerializeInternalFieldsCallback callback) {
510   DCHECK(!context.IsEmpty());
511   SnapshotCreatorData* data = SnapshotCreatorData::cast(data_);
512   DCHECK(!data->created_);
513   Isolate* isolate = data->isolate_;
514   CHECK_EQ(isolate, context->GetIsolate());
515   size_t index = data->contexts_.Size();
516   data->contexts_.Append(context);
517   data->embedder_fields_serializers_.push_back(callback);
518   return index;
519 }
520 
AddData(i::Address object)521 size_t SnapshotCreator::AddData(i::Address object) {
522   DCHECK_NE(object, i::kNullAddress);
523   SnapshotCreatorData* data = SnapshotCreatorData::cast(data_);
524   DCHECK(!data->created_);
525   i::Isolate* isolate = reinterpret_cast<i::Isolate*>(data->isolate_);
526   i::HandleScope scope(isolate);
527   i::Handle<i::Object> obj(i::Object(object), isolate);
528   i::Handle<i::ArrayList> list;
529   if (!isolate->heap()->serialized_objects().IsArrayList()) {
530     list = i::ArrayList::New(isolate, 1);
531   } else {
532     list = i::Handle<i::ArrayList>(
533         i::ArrayList::cast(isolate->heap()->serialized_objects()), isolate);
534   }
535   size_t index = static_cast<size_t>(list->Length());
536   list = i::ArrayList::Add(isolate, list, obj);
537   isolate->heap()->SetSerializedObjects(*list);
538   return index;
539 }
540 
AddData(Local<Context> context,i::Address object)541 size_t SnapshotCreator::AddData(Local<Context> context, i::Address object) {
542   DCHECK_NE(object, i::kNullAddress);
543   DCHECK(!SnapshotCreatorData::cast(data_)->created_);
544   i::Handle<i::Context> ctx = Utils::OpenHandle(*context);
545   i::Isolate* isolate = ctx->GetIsolate();
546   i::HandleScope scope(isolate);
547   i::Handle<i::Object> obj(i::Object(object), isolate);
548   i::Handle<i::ArrayList> list;
549   if (!ctx->serialized_objects().IsArrayList()) {
550     list = i::ArrayList::New(isolate, 1);
551   } else {
552     list = i::Handle<i::ArrayList>(
553         i::ArrayList::cast(ctx->serialized_objects()), isolate);
554   }
555   size_t index = static_cast<size_t>(list->Length());
556   list = i::ArrayList::Add(isolate, list, obj);
557   ctx->set_serialized_objects(*list);
558   return index;
559 }
560 
561 namespace {
ConvertSerializedObjectsToFixedArray(Local<Context> context)562 void ConvertSerializedObjectsToFixedArray(Local<Context> context) {
563   i::Handle<i::Context> ctx = Utils::OpenHandle(*context);
564   i::Isolate* isolate = ctx->GetIsolate();
565   if (!ctx->serialized_objects().IsArrayList()) {
566     ctx->set_serialized_objects(i::ReadOnlyRoots(isolate).empty_fixed_array());
567   } else {
568     i::Handle<i::ArrayList> list(i::ArrayList::cast(ctx->serialized_objects()),
569                                  isolate);
570     i::Handle<i::FixedArray> elements = i::ArrayList::Elements(isolate, list);
571     ctx->set_serialized_objects(*elements);
572   }
573 }
574 
ConvertSerializedObjectsToFixedArray(i::Isolate * isolate)575 void ConvertSerializedObjectsToFixedArray(i::Isolate* isolate) {
576   if (!isolate->heap()->serialized_objects().IsArrayList()) {
577     isolate->heap()->SetSerializedObjects(
578         i::ReadOnlyRoots(isolate).empty_fixed_array());
579   } else {
580     i::Handle<i::ArrayList> list(
581         i::ArrayList::cast(isolate->heap()->serialized_objects()), isolate);
582     i::Handle<i::FixedArray> elements = i::ArrayList::Elements(isolate, list);
583     isolate->heap()->SetSerializedObjects(*elements);
584   }
585 }
586 }  // anonymous namespace
587 
CreateBlob(SnapshotCreator::FunctionCodeHandling function_code_handling)588 StartupData SnapshotCreator::CreateBlob(
589     SnapshotCreator::FunctionCodeHandling function_code_handling) {
590   SnapshotCreatorData* data = SnapshotCreatorData::cast(data_);
591   i::Isolate* isolate = reinterpret_cast<i::Isolate*>(data->isolate_);
592   Utils::ApiCheck(!data->created_, "v8::SnapshotCreator::CreateBlob",
593                   "CreateBlob() cannot be called more than once on the same "
594                   "SnapshotCreator.");
595   Utils::ApiCheck(
596       !data->default_context_.IsEmpty(), "v8::SnapshotCreator::CreateBlob",
597       "CreateBlob() cannot be called before the default context is set.");
598 
599   const int num_additional_contexts = static_cast<int>(data->contexts_.Size());
600   const int num_contexts = num_additional_contexts + 1;  // The default context.
601 
602   // Create and store lists of embedder-provided data needed during
603   // serialization.
604   {
605     i::HandleScope scope(isolate);
606     // Convert list of context-independent data to FixedArray.
607     ConvertSerializedObjectsToFixedArray(isolate);
608 
609     // Convert lists of context-dependent data to FixedArray.
610     ConvertSerializedObjectsToFixedArray(
611         data->default_context_.Get(data->isolate_));
612     for (int i = 0; i < num_additional_contexts; i++) {
613       ConvertSerializedObjectsToFixedArray(data->contexts_.Get(i));
614     }
615 
616     // We need to store the global proxy size upfront in case we need the
617     // bootstrapper to create a global proxy before we deserialize the context.
618     i::Handle<i::FixedArray> global_proxy_sizes =
619         isolate->factory()->NewFixedArray(num_additional_contexts,
620                                           i::AllocationType::kOld);
621     for (int i = 0; i < num_additional_contexts; i++) {
622       i::Handle<i::Context> context =
623           v8::Utils::OpenHandle(*data->contexts_.Get(i));
624       global_proxy_sizes->set(i,
625                               i::Smi::FromInt(context->global_proxy().Size()));
626     }
627     isolate->heap()->SetSerializedGlobalProxySizes(*global_proxy_sizes);
628   }
629 
630   // We might rehash strings and re-sort descriptors. Clear the lookup cache.
631   isolate->descriptor_lookup_cache()->Clear();
632 
633   // If we don't do this then we end up with a stray root pointing at the
634   // context even after we have disposed of the context.
635   isolate->heap()->CollectAllAvailableGarbage(
636       i::GarbageCollectionReason::kSnapshotCreator);
637   {
638     i::HandleScope scope(isolate);
639     isolate->heap()->CompactWeakArrayLists();
640   }
641 
642   i::Snapshot::ClearReconstructableDataForSerialization(
643       isolate, function_code_handling == FunctionCodeHandling::kClear);
644 
645   i::GlobalSafepointScope global_safepoint(isolate);
646   i::DisallowGarbageCollection no_gc_from_here_on;
647 
648   // Create a vector with all contexts and clear associated Persistent fields.
649   // Note these contexts may be dead after calling Clear(), but will not be
650   // collected until serialization completes and the DisallowGarbageCollection
651   // scope above goes out of scope.
652   std::vector<i::Context> contexts;
653   contexts.reserve(num_contexts);
654   {
655     i::HandleScope scope(isolate);
656     contexts.push_back(
657         *v8::Utils::OpenHandle(*data->default_context_.Get(data->isolate_)));
658     data->default_context_.Reset();
659     for (int i = 0; i < num_additional_contexts; i++) {
660       i::Handle<i::Context> context =
661           v8::Utils::OpenHandle(*data->contexts_.Get(i));
662       contexts.push_back(*context);
663     }
664     data->contexts_.Clear();
665   }
666 
667   // Check that values referenced by global/eternal handles are accounted for.
668   i::SerializedHandleChecker handle_checker(isolate, &contexts);
669   CHECK(handle_checker.CheckGlobalAndEternalHandles());
670 
671   // Create a vector with all embedder fields serializers.
672   std::vector<SerializeInternalFieldsCallback> embedder_fields_serializers;
673   embedder_fields_serializers.reserve(num_contexts);
674   embedder_fields_serializers.push_back(
675       data->default_embedder_fields_serializer_);
676   for (int i = 0; i < num_additional_contexts; i++) {
677     embedder_fields_serializers.push_back(
678         data->embedder_fields_serializers_[i]);
679   }
680 
681   data->created_ = true;
682   return i::Snapshot::Create(isolate, &contexts, embedder_fields_serializers,
683                              global_safepoint, no_gc_from_here_on);
684 }
685 
CanBeRehashed() const686 bool StartupData::CanBeRehashed() const {
687   DCHECK(i::Snapshot::VerifyChecksum(this));
688   return i::Snapshot::ExtractRehashability(this);
689 }
690 
IsValid() const691 bool StartupData::IsValid() const { return i::Snapshot::VersionIsValid(this); }
692 
SetDcheckErrorHandler(DcheckErrorCallback that)693 void V8::SetDcheckErrorHandler(DcheckErrorCallback that) {
694   v8::base::SetDcheckFunction(that);
695 }
696 
SetFlagsFromString(const char * str)697 void V8::SetFlagsFromString(const char* str) {
698   SetFlagsFromString(str, strlen(str));
699 }
700 
SetFlagsFromString(const char * str,size_t length)701 void V8::SetFlagsFromString(const char* str, size_t length) {
702   i::FlagList::SetFlagsFromString(str, length);
703   i::FlagList::EnforceFlagImplications();
704 }
705 
SetFlagsFromCommandLine(int * argc,char ** argv,bool remove_flags)706 void V8::SetFlagsFromCommandLine(int* argc, char** argv, bool remove_flags) {
707   using HelpOptions = i::FlagList::HelpOptions;
708   i::FlagList::SetFlagsFromCommandLine(argc, argv, remove_flags,
709                                        HelpOptions(HelpOptions::kDontExit));
710 }
711 
712 RegisteredExtension* RegisteredExtension::first_extension_ = nullptr;
713 
RegisteredExtension(std::unique_ptr<Extension> extension)714 RegisteredExtension::RegisteredExtension(std::unique_ptr<Extension> extension)
715     : extension_(std::move(extension)) {}
716 
717 // static
Register(std::unique_ptr<Extension> extension)718 void RegisteredExtension::Register(std::unique_ptr<Extension> extension) {
719   RegisteredExtension* new_extension =
720       new RegisteredExtension(std::move(extension));
721   new_extension->next_ = first_extension_;
722   first_extension_ = new_extension;
723 }
724 
725 // static
UnregisterAll()726 void RegisteredExtension::UnregisterAll() {
727   RegisteredExtension* re = first_extension_;
728   while (re != nullptr) {
729     RegisteredExtension* next = re->next();
730     delete re;
731     re = next;
732   }
733   first_extension_ = nullptr;
734 }
735 
736 namespace {
737 class ExtensionResource : public String::ExternalOneByteStringResource {
738  public:
ExtensionResource()739   ExtensionResource() : data_(nullptr), length_(0) {}
ExtensionResource(const char * data,size_t length)740   ExtensionResource(const char* data, size_t length)
741       : data_(data), length_(length) {}
data() const742   const char* data() const override { return data_; }
length() const743   size_t length() const override { return length_; }
Dispose()744   void Dispose() override {}
745 
746  private:
747   const char* data_;
748   size_t length_;
749 };
750 }  // anonymous namespace
751 
RegisterExtension(std::unique_ptr<Extension> extension)752 void RegisterExtension(std::unique_ptr<Extension> extension) {
753   RegisteredExtension::Register(std::move(extension));
754 }
755 
Extension(const char * name,const char * source,int dep_count,const char ** deps,int source_length)756 Extension::Extension(const char* name, const char* source, int dep_count,
757                      const char** deps, int source_length)
758     : name_(name),
759       source_length_(source_length >= 0
760                          ? source_length
761                          : (source ? static_cast<int>(strlen(source)) : 0)),
762       dep_count_(dep_count),
763       deps_(deps),
764       auto_enable_(false) {
765   source_ = new ExtensionResource(source, source_length_);
766   CHECK(source != nullptr || source_length_ == 0);
767 }
768 
ConfigureDefaultsFromHeapSize(size_t initial_heap_size_in_bytes,size_t maximum_heap_size_in_bytes)769 void ResourceConstraints::ConfigureDefaultsFromHeapSize(
770     size_t initial_heap_size_in_bytes, size_t maximum_heap_size_in_bytes) {
771   CHECK_LE(initial_heap_size_in_bytes, maximum_heap_size_in_bytes);
772   if (maximum_heap_size_in_bytes == 0) {
773     return;
774   }
775   size_t young_generation, old_generation;
776   i::Heap::GenerationSizesFromHeapSize(maximum_heap_size_in_bytes,
777                                        &young_generation, &old_generation);
778   set_max_young_generation_size_in_bytes(
779       std::max(young_generation, i::Heap::MinYoungGenerationSize()));
780   set_max_old_generation_size_in_bytes(
781       std::max(old_generation, i::Heap::MinOldGenerationSize()));
782   if (initial_heap_size_in_bytes > 0) {
783     i::Heap::GenerationSizesFromHeapSize(initial_heap_size_in_bytes,
784                                          &young_generation, &old_generation);
785     // We do not set lower bounds for the initial sizes.
786     set_initial_young_generation_size_in_bytes(young_generation);
787     set_initial_old_generation_size_in_bytes(old_generation);
788   }
789   if (i::kPlatformRequiresCodeRange) {
790     set_code_range_size_in_bytes(
791         std::min(i::kMaximalCodeRangeSize, maximum_heap_size_in_bytes));
792   }
793 }
794 
ConfigureDefaults(uint64_t physical_memory,uint64_t virtual_memory_limit)795 void ResourceConstraints::ConfigureDefaults(uint64_t physical_memory,
796                                             uint64_t virtual_memory_limit) {
797   size_t heap_size = i::Heap::HeapSizeFromPhysicalMemory(physical_memory);
798   size_t young_generation, old_generation;
799   i::Heap::GenerationSizesFromHeapSize(heap_size, &young_generation,
800                                        &old_generation);
801   set_max_young_generation_size_in_bytes(young_generation);
802   set_max_old_generation_size_in_bytes(old_generation);
803 
804   if (virtual_memory_limit > 0 && i::kPlatformRequiresCodeRange) {
805     set_code_range_size_in_bytes(
806         std::min(i::kMaximalCodeRangeSize,
807                  static_cast<size_t>(virtual_memory_limit / 8)));
808   }
809 }
810 
811 namespace internal {
812 
GlobalizeTracedReference(i::Isolate * isolate,i::Address * obj,internal::Address * slot,GlobalHandleStoreMode store_mode)813 i::Address* GlobalizeTracedReference(i::Isolate* isolate, i::Address* obj,
814                                      internal::Address* slot,
815                                      GlobalHandleStoreMode store_mode) {
816   API_RCS_SCOPE(isolate, TracedGlobal, New);
817 #ifdef DEBUG
818   Utils::ApiCheck((slot != nullptr), "v8::GlobalizeTracedReference",
819                   "the address slot must be not null");
820 #endif
821   i::Handle<i::Object> result =
822       isolate->global_handles()->CreateTraced(*obj, slot, store_mode);
823 #ifdef VERIFY_HEAP
824   if (i::FLAG_verify_heap) {
825     i::Object(*obj).ObjectVerify(isolate);
826   }
827 #endif  // VERIFY_HEAP
828   return result.location();
829 }
830 
MoveTracedReference(internal::Address ** from,internal::Address ** to)831 void MoveTracedReference(internal::Address** from, internal::Address** to) {
832   GlobalHandles::MoveTracedReference(from, to);
833 }
834 
CopyTracedReference(const internal::Address * const * from,internal::Address ** to)835 void CopyTracedReference(const internal::Address* const* from,
836                          internal::Address** to) {
837   GlobalHandles::CopyTracedReference(from, to);
838 }
839 
DisposeTracedReference(internal::Address * location)840 void DisposeTracedReference(internal::Address* location) {
841   GlobalHandles::DestroyTracedReference(location);
842 }
843 
844 }  // namespace internal
845 
846 namespace api_internal {
847 
GlobalizeReference(i::Isolate * isolate,i::Address * obj)848 i::Address* GlobalizeReference(i::Isolate* isolate, i::Address* obj) {
849   API_RCS_SCOPE(isolate, Persistent, New);
850   i::Handle<i::Object> result = isolate->global_handles()->Create(*obj);
851 #ifdef VERIFY_HEAP
852   if (i::FLAG_verify_heap) {
853     i::Object(*obj).ObjectVerify(isolate);
854   }
855 #endif  // VERIFY_HEAP
856   return result.location();
857 }
858 
CopyGlobalReference(i::Address * from)859 i::Address* CopyGlobalReference(i::Address* from) {
860   i::Handle<i::Object> result = i::GlobalHandles::CopyGlobal(from);
861   return result.location();
862 }
863 
MoveGlobalReference(internal::Address ** from,internal::Address ** to)864 void MoveGlobalReference(internal::Address** from, internal::Address** to) {
865   i::GlobalHandles::MoveGlobal(from, to);
866 }
867 
MakeWeak(i::Address * location,void * parameter,WeakCallbackInfo<void>::Callback weak_callback,WeakCallbackType type)868 void MakeWeak(i::Address* location, void* parameter,
869               WeakCallbackInfo<void>::Callback weak_callback,
870               WeakCallbackType type) {
871   i::GlobalHandles::MakeWeak(location, parameter, weak_callback, type);
872 }
873 
MakeWeak(i::Address ** location_addr)874 void MakeWeak(i::Address** location_addr) {
875   i::GlobalHandles::MakeWeak(location_addr);
876 }
877 
ClearWeak(i::Address * location)878 void* ClearWeak(i::Address* location) {
879   return i::GlobalHandles::ClearWeakness(location);
880 }
881 
AnnotateStrongRetainer(i::Address * location,const char * label)882 void AnnotateStrongRetainer(i::Address* location, const char* label) {
883   i::GlobalHandles::AnnotateStrongRetainer(location, label);
884 }
885 
DisposeGlobal(i::Address * location)886 void DisposeGlobal(i::Address* location) {
887   i::GlobalHandles::Destroy(location);
888 }
889 
Eternalize(Isolate * v8_isolate,Value * value)890 Value* Eternalize(Isolate* v8_isolate, Value* value) {
891   i::Isolate* isolate = reinterpret_cast<i::Isolate*>(v8_isolate);
892   i::Object object = *Utils::OpenHandle(value);
893   int index = -1;
894   isolate->eternal_handles()->Create(isolate, object, &index);
895   return reinterpret_cast<Value*>(
896       isolate->eternal_handles()->Get(index).location());
897 }
898 
FromJustIsNothing()899 void FromJustIsNothing() {
900   Utils::ApiCheck(false, "v8::FromJust", "Maybe value is Nothing.");
901 }
902 
ToLocalEmpty()903 void ToLocalEmpty() {
904   Utils::ApiCheck(false, "v8::ToLocalChecked", "Empty MaybeLocal.");
905 }
906 
InternalFieldOutOfBounds(int index)907 void InternalFieldOutOfBounds(int index) {
908   Utils::ApiCheck(0 <= index && index < kInternalFieldsInWeakCallback,
909                   "WeakCallbackInfo::GetInternalField",
910                   "Internal field out of bounds.");
911 }
912 
913 }  // namespace api_internal
914 
915 // --- H a n d l e s ---
916 
HandleScope(Isolate * isolate)917 HandleScope::HandleScope(Isolate* isolate) { Initialize(isolate); }
918 
Initialize(Isolate * isolate)919 void HandleScope::Initialize(Isolate* isolate) {
920   i::Isolate* internal_isolate = reinterpret_cast<i::Isolate*>(isolate);
921   // We do not want to check the correct usage of the Locker class all over the
922   // place, so we do it only here: Without a HandleScope, an embedder can do
923   // almost nothing, so it is enough to check in this central place.
924   // We make an exception if the serializer is enabled, which means that the
925   // Isolate is exclusively used to create a snapshot.
926   Utils::ApiCheck(
927       !internal_isolate->was_locker_ever_used() ||
928           internal_isolate->thread_manager()->IsLockedByCurrentThread() ||
929           internal_isolate->serializer_enabled(),
930       "HandleScope::HandleScope",
931       "Entering the V8 API without proper locking in place");
932   i::HandleScopeData* current = internal_isolate->handle_scope_data();
933   isolate_ = internal_isolate;
934   prev_next_ = current->next;
935   prev_limit_ = current->limit;
936   current->level++;
937 }
938 
~HandleScope()939 HandleScope::~HandleScope() {
940   i::HandleScope::CloseScope(isolate_, prev_next_, prev_limit_);
941 }
942 
operator new(size_t)943 void* HandleScope::operator new(size_t) { base::OS::Abort(); }
operator new[](size_t)944 void* HandleScope::operator new[](size_t) { base::OS::Abort(); }
operator delete(void *,size_t)945 void HandleScope::operator delete(void*, size_t) { base::OS::Abort(); }
operator delete[](void *,size_t)946 void HandleScope::operator delete[](void*, size_t) { base::OS::Abort(); }
947 
NumberOfHandles(Isolate * isolate)948 int HandleScope::NumberOfHandles(Isolate* isolate) {
949   return i::HandleScope::NumberOfHandles(
950       reinterpret_cast<i::Isolate*>(isolate));
951 }
952 
CreateHandle(i::Isolate * isolate,i::Address value)953 i::Address* HandleScope::CreateHandle(i::Isolate* isolate, i::Address value) {
954   return i::HandleScope::CreateHandle(isolate, value);
955 }
956 
EscapableHandleScope(Isolate * v8_isolate)957 EscapableHandleScope::EscapableHandleScope(Isolate* v8_isolate) {
958   i::Isolate* isolate = reinterpret_cast<i::Isolate*>(v8_isolate);
959   escape_slot_ =
960       CreateHandle(isolate, i::ReadOnlyRoots(isolate).the_hole_value().ptr());
961   Initialize(v8_isolate);
962 }
963 
Escape(i::Address * escape_value)964 i::Address* EscapableHandleScope::Escape(i::Address* escape_value) {
965   i::Heap* heap = reinterpret_cast<i::Isolate*>(GetIsolate())->heap();
966   Utils::ApiCheck(i::Object(*escape_slot_).IsTheHole(heap->isolate()),
967                   "EscapableHandleScope::Escape", "Escape value set twice");
968   if (escape_value == nullptr) {
969     *escape_slot_ = i::ReadOnlyRoots(heap).undefined_value().ptr();
970     return nullptr;
971   }
972   *escape_slot_ = *escape_value;
973   return escape_slot_;
974 }
975 
operator new(size_t)976 void* EscapableHandleScope::operator new(size_t) { base::OS::Abort(); }
operator new[](size_t)977 void* EscapableHandleScope::operator new[](size_t) { base::OS::Abort(); }
operator delete(void *,size_t)978 void EscapableHandleScope::operator delete(void*, size_t) { base::OS::Abort(); }
operator delete[](void *,size_t)979 void EscapableHandleScope::operator delete[](void*, size_t) {
980   base::OS::Abort();
981 }
982 
SealHandleScope(Isolate * isolate)983 SealHandleScope::SealHandleScope(Isolate* isolate)
984     : isolate_(reinterpret_cast<i::Isolate*>(isolate)) {
985   i::HandleScopeData* current = isolate_->handle_scope_data();
986   prev_limit_ = current->limit;
987   current->limit = current->next;
988   prev_sealed_level_ = current->sealed_level;
989   current->sealed_level = current->level;
990 }
991 
~SealHandleScope()992 SealHandleScope::~SealHandleScope() {
993   i::HandleScopeData* current = isolate_->handle_scope_data();
994   DCHECK_EQ(current->next, current->limit);
995   current->limit = prev_limit_;
996   DCHECK_EQ(current->level, current->sealed_level);
997   current->sealed_level = prev_sealed_level_;
998 }
999 
operator new(size_t)1000 void* SealHandleScope::operator new(size_t) { base::OS::Abort(); }
operator new[](size_t)1001 void* SealHandleScope::operator new[](size_t) { base::OS::Abort(); }
operator delete(void *,size_t)1002 void SealHandleScope::operator delete(void*, size_t) { base::OS::Abort(); }
operator delete[](void *,size_t)1003 void SealHandleScope::operator delete[](void*, size_t) { base::OS::Abort(); }
1004 
IsModule() const1005 bool Data::IsModule() const { return Utils::OpenHandle(this)->IsModule(); }
IsFixedArray() const1006 bool Data::IsFixedArray() const {
1007   return Utils::OpenHandle(this)->IsFixedArray();
1008 }
1009 
IsValue() const1010 bool Data::IsValue() const {
1011   i::DisallowGarbageCollection no_gc;
1012   i::Object self = *Utils::OpenHandle(this);
1013   if (self.IsSmi()) return true;
1014   i::HeapObject heap_object = i::HeapObject::cast(self);
1015   DCHECK(!heap_object.IsTheHole());
1016   if (heap_object.IsSymbol()) {
1017     return !i::Symbol::cast(heap_object).is_private();
1018   }
1019   return heap_object.IsPrimitiveHeapObject() || heap_object.IsJSReceiver();
1020 }
1021 
IsPrivate() const1022 bool Data::IsPrivate() const {
1023   return Utils::OpenHandle(this)->IsPrivateSymbol();
1024 }
1025 
IsObjectTemplate() const1026 bool Data::IsObjectTemplate() const {
1027   return Utils::OpenHandle(this)->IsObjectTemplateInfo();
1028 }
1029 
IsFunctionTemplate() const1030 bool Data::IsFunctionTemplate() const {
1031   return Utils::OpenHandle(this)->IsFunctionTemplateInfo();
1032 }
1033 
IsContext() const1034 bool Data::IsContext() const { return Utils::OpenHandle(this)->IsContext(); }
1035 
Enter()1036 void Context::Enter() {
1037   i::Handle<i::Context> env = Utils::OpenHandle(this);
1038   i::Isolate* isolate = env->GetIsolate();
1039   ENTER_V8_NO_SCRIPT_NO_EXCEPTION(isolate);
1040   i::HandleScopeImplementer* impl = isolate->handle_scope_implementer();
1041   impl->EnterContext(*env);
1042   impl->SaveContext(isolate->context());
1043   isolate->set_context(*env);
1044 }
1045 
Exit()1046 void Context::Exit() {
1047   i::Handle<i::Context> env = Utils::OpenHandle(this);
1048   i::Isolate* isolate = env->GetIsolate();
1049   ENTER_V8_NO_SCRIPT_NO_EXCEPTION(isolate);
1050   i::HandleScopeImplementer* impl = isolate->handle_scope_implementer();
1051   if (!Utils::ApiCheck(impl->LastEnteredContextWas(*env), "v8::Context::Exit()",
1052                        "Cannot exit non-entered context")) {
1053     return;
1054   }
1055   impl->LeaveContext();
1056   isolate->set_context(impl->RestoreContext());
1057 }
1058 
BackupIncumbentScope(Local<Context> backup_incumbent_context)1059 Context::BackupIncumbentScope::BackupIncumbentScope(
1060     Local<Context> backup_incumbent_context)
1061     : backup_incumbent_context_(backup_incumbent_context) {
1062   DCHECK(!backup_incumbent_context_.IsEmpty());
1063 
1064   i::Handle<i::Context> env = Utils::OpenHandle(*backup_incumbent_context_);
1065   i::Isolate* isolate = env->GetIsolate();
1066 
1067   js_stack_comparable_address_ =
1068       i::SimulatorStack::RegisterJSStackComparableAddress(isolate);
1069 
1070   prev_ = isolate->top_backup_incumbent_scope();
1071   isolate->set_top_backup_incumbent_scope(this);
1072 }
1073 
~BackupIncumbentScope()1074 Context::BackupIncumbentScope::~BackupIncumbentScope() {
1075   i::Handle<i::Context> env = Utils::OpenHandle(*backup_incumbent_context_);
1076   i::Isolate* isolate = env->GetIsolate();
1077 
1078   i::SimulatorStack::UnregisterJSStackComparableAddress(isolate);
1079 
1080   isolate->set_top_backup_incumbent_scope(prev_);
1081 }
1082 
1083 STATIC_ASSERT(i::Internals::kEmbedderDataSlotSize == i::kEmbedderDataSlotSize);
1084 
EmbedderDataFor(Context * context,int index,bool can_grow,const char * location)1085 static i::Handle<i::EmbedderDataArray> EmbedderDataFor(Context* context,
1086                                                        int index, bool can_grow,
1087                                                        const char* location) {
1088   i::Handle<i::Context> env = Utils::OpenHandle(context);
1089   i::Isolate* isolate = env->GetIsolate();
1090   ASSERT_NO_SCRIPT_NO_EXCEPTION(isolate);
1091   bool ok = Utils::ApiCheck(env->IsNativeContext(), location,
1092                             "Not a native context") &&
1093             Utils::ApiCheck(index >= 0, location, "Negative index");
1094   if (!ok) return i::Handle<i::EmbedderDataArray>();
1095   // TODO(ishell): remove cast once embedder_data slot has a proper type.
1096   i::Handle<i::EmbedderDataArray> data(
1097       i::EmbedderDataArray::cast(env->embedder_data()), isolate);
1098   if (index < data->length()) return data;
1099   if (!Utils::ApiCheck(can_grow && index < i::EmbedderDataArray::kMaxLength,
1100                        location, "Index too large")) {
1101     return i::Handle<i::EmbedderDataArray>();
1102   }
1103   data = i::EmbedderDataArray::EnsureCapacity(isolate, data, index);
1104   env->set_embedder_data(*data);
1105   return data;
1106 }
1107 
GetNumberOfEmbedderDataFields()1108 uint32_t Context::GetNumberOfEmbedderDataFields() {
1109   i::Handle<i::Context> context = Utils::OpenHandle(this);
1110   ASSERT_NO_SCRIPT_NO_EXCEPTION(context->GetIsolate());
1111   Utils::ApiCheck(context->IsNativeContext(),
1112                   "Context::GetNumberOfEmbedderDataFields",
1113                   "Not a native context");
1114   // TODO(ishell): remove cast once embedder_data slot has a proper type.
1115   return static_cast<uint32_t>(
1116       i::EmbedderDataArray::cast(context->embedder_data()).length());
1117 }
1118 
SlowGetEmbedderData(int index)1119 v8::Local<v8::Value> Context::SlowGetEmbedderData(int index) {
1120   const char* location = "v8::Context::GetEmbedderData()";
1121   i::Handle<i::EmbedderDataArray> data =
1122       EmbedderDataFor(this, index, false, location);
1123   if (data.is_null()) return Local<Value>();
1124   i::Isolate* isolate = Utils::OpenHandle(this)->GetIsolate();
1125   i::Handle<i::Object> result(i::EmbedderDataSlot(*data, index).load_tagged(),
1126                               isolate);
1127   return Utils::ToLocal(result);
1128 }
1129 
SetEmbedderData(int index,v8::Local<Value> value)1130 void Context::SetEmbedderData(int index, v8::Local<Value> value) {
1131   const char* location = "v8::Context::SetEmbedderData()";
1132   i::Handle<i::EmbedderDataArray> data =
1133       EmbedderDataFor(this, index, true, location);
1134   if (data.is_null()) return;
1135   i::Handle<i::Object> val = Utils::OpenHandle(*value);
1136   i::EmbedderDataSlot::store_tagged(*data, index, *val);
1137   DCHECK_EQ(*Utils::OpenHandle(*value),
1138             *Utils::OpenHandle(*GetEmbedderData(index)));
1139 }
1140 
SlowGetAlignedPointerFromEmbedderData(int index)1141 void* Context::SlowGetAlignedPointerFromEmbedderData(int index) {
1142   const char* location = "v8::Context::GetAlignedPointerFromEmbedderData()";
1143   i::Isolate* isolate = Utils::OpenHandle(this)->GetIsolate();
1144   i::HandleScope handle_scope(isolate);
1145   i::Handle<i::EmbedderDataArray> data =
1146       EmbedderDataFor(this, index, false, location);
1147   if (data.is_null()) return nullptr;
1148   void* result;
1149   Utils::ApiCheck(
1150       i::EmbedderDataSlot(*data, index).ToAlignedPointer(isolate, &result),
1151       location, "Pointer is not aligned");
1152   return result;
1153 }
1154 
SetAlignedPointerInEmbedderData(int index,void * value)1155 void Context::SetAlignedPointerInEmbedderData(int index, void* value) {
1156   const char* location = "v8::Context::SetAlignedPointerInEmbedderData()";
1157   i::Isolate* isolate = Utils::OpenHandle(this)->GetIsolate();
1158   i::Handle<i::EmbedderDataArray> data =
1159       EmbedderDataFor(this, index, true, location);
1160   bool ok =
1161       i::EmbedderDataSlot(*data, index).store_aligned_pointer(isolate, value);
1162   Utils::ApiCheck(ok, location, "Pointer is not aligned");
1163   DCHECK_EQ(value, GetAlignedPointerFromEmbedderData(index));
1164 }
1165 
1166 // --- T e m p l a t e ---
1167 
InitializeTemplate(i::TemplateInfo that,int type,bool do_not_cache)1168 static void InitializeTemplate(i::TemplateInfo that, int type,
1169                                bool do_not_cache) {
1170   that.set_number_of_properties(0);
1171   that.set_tag(type);
1172   int serial_number =
1173       do_not_cache ? i::TemplateInfo::kDoNotCache : i::TemplateInfo::kUncached;
1174   that.set_serial_number(serial_number);
1175 }
1176 
Set(v8::Local<Name> name,v8::Local<Data> value,v8::PropertyAttribute attribute)1177 void Template::Set(v8::Local<Name> name, v8::Local<Data> value,
1178                    v8::PropertyAttribute attribute) {
1179   auto templ = Utils::OpenHandle(this);
1180   i::Isolate* isolate = templ->GetIsolate();
1181   ENTER_V8_NO_SCRIPT_NO_EXCEPTION(isolate);
1182   i::HandleScope scope(isolate);
1183   auto value_obj = Utils::OpenHandle(*value);
1184 
1185   Utils::ApiCheck(!value_obj->IsJSReceiver() || value_obj->IsTemplateInfo(),
1186                   "v8::Template::Set",
1187                   "Invalid value, must be a primitive or a Template");
1188 
1189   // The template cache only performs shallow clones, if we set an
1190   // ObjectTemplate as a property value then we can not cache the receiver
1191   // template.
1192   if (value_obj->IsObjectTemplateInfo()) {
1193     templ->set_serial_number(i::TemplateInfo::kDoNotCache);
1194   }
1195 
1196   i::ApiNatives::AddDataProperty(isolate, templ, Utils::OpenHandle(*name),
1197                                  value_obj,
1198                                  static_cast<i::PropertyAttributes>(attribute));
1199 }
1200 
SetPrivate(v8::Local<Private> name,v8::Local<Data> value,v8::PropertyAttribute attribute)1201 void Template::SetPrivate(v8::Local<Private> name, v8::Local<Data> value,
1202                           v8::PropertyAttribute attribute) {
1203   Set(Utils::ToLocal(Utils::OpenHandle(reinterpret_cast<Name*>(*name))), value,
1204       attribute);
1205 }
1206 
SetAccessorProperty(v8::Local<v8::Name> name,v8::Local<FunctionTemplate> getter,v8::Local<FunctionTemplate> setter,v8::PropertyAttribute attribute,v8::AccessControl access_control)1207 void Template::SetAccessorProperty(v8::Local<v8::Name> name,
1208                                    v8::Local<FunctionTemplate> getter,
1209                                    v8::Local<FunctionTemplate> setter,
1210                                    v8::PropertyAttribute attribute,
1211                                    v8::AccessControl access_control) {
1212   // TODO(verwaest): Remove |access_control|.
1213   DCHECK_EQ(v8::DEFAULT, access_control);
1214   auto templ = Utils::OpenHandle(this);
1215   auto isolate = templ->GetIsolate();
1216   ENTER_V8_NO_SCRIPT_NO_EXCEPTION(isolate);
1217   DCHECK(!name.IsEmpty());
1218   DCHECK(!getter.IsEmpty() || !setter.IsEmpty());
1219   i::HandleScope scope(isolate);
1220   i::ApiNatives::AddAccessorProperty(
1221       isolate, templ, Utils::OpenHandle(*name),
1222       Utils::OpenHandle(*getter, true), Utils::OpenHandle(*setter, true),
1223       static_cast<i::PropertyAttributes>(attribute));
1224 }
1225 
1226 // --- F u n c t i o n   T e m p l a t e ---
InitializeFunctionTemplate(i::FunctionTemplateInfo info,bool do_not_cache)1227 static void InitializeFunctionTemplate(i::FunctionTemplateInfo info,
1228                                        bool do_not_cache) {
1229   InitializeTemplate(info, Consts::FUNCTION_TEMPLATE, do_not_cache);
1230   info.set_flag(0);
1231 }
1232 
1233 static Local<ObjectTemplate> ObjectTemplateNew(
1234     i::Isolate* isolate, v8::Local<FunctionTemplate> constructor,
1235     bool do_not_cache);
1236 
PrototypeTemplate()1237 Local<ObjectTemplate> FunctionTemplate::PrototypeTemplate() {
1238   auto self = Utils::OpenHandle(this);
1239   i::Isolate* i_isolate = self->GetIsolate();
1240   ENTER_V8_NO_SCRIPT_NO_EXCEPTION(i_isolate);
1241   i::Handle<i::HeapObject> result(self->GetPrototypeTemplate(), i_isolate);
1242   if (result->IsUndefined(i_isolate)) {
1243     // Do not cache prototype objects.
1244     result = Utils::OpenHandle(
1245         *ObjectTemplateNew(i_isolate, Local<FunctionTemplate>(), true));
1246     i::FunctionTemplateInfo::SetPrototypeTemplate(i_isolate, self, result);
1247   }
1248   return ToApiHandle<ObjectTemplate>(result);
1249 }
1250 
SetPrototypeProviderTemplate(Local<FunctionTemplate> prototype_provider)1251 void FunctionTemplate::SetPrototypeProviderTemplate(
1252     Local<FunctionTemplate> prototype_provider) {
1253   auto self = Utils::OpenHandle(this);
1254   i::Isolate* i_isolate = self->GetIsolate();
1255   ENTER_V8_NO_SCRIPT_NO_EXCEPTION(i_isolate);
1256   i::Handle<i::FunctionTemplateInfo> result =
1257       Utils::OpenHandle(*prototype_provider);
1258   Utils::ApiCheck(self->GetPrototypeTemplate().IsUndefined(i_isolate),
1259                   "v8::FunctionTemplate::SetPrototypeProviderTemplate",
1260                   "Protoype must be undefined");
1261   Utils::ApiCheck(self->GetParentTemplate().IsUndefined(i_isolate),
1262                   "v8::FunctionTemplate::SetPrototypeProviderTemplate",
1263                   "Prototype provider must be empty");
1264   i::FunctionTemplateInfo::SetPrototypeProviderTemplate(i_isolate, self,
1265                                                         result);
1266 }
1267 
EnsureNotPublished(i::Handle<i::FunctionTemplateInfo> info,const char * func)1268 static void EnsureNotPublished(i::Handle<i::FunctionTemplateInfo> info,
1269                                const char* func) {
1270   DCHECK_IMPLIES(info->instantiated(), info->published());
1271   Utils::ApiCheck(!info->published(), func,
1272                   "FunctionTemplate already instantiated");
1273 }
1274 
Inherit(v8::Local<FunctionTemplate> value)1275 void FunctionTemplate::Inherit(v8::Local<FunctionTemplate> value) {
1276   auto info = Utils::OpenHandle(this);
1277   EnsureNotPublished(info, "v8::FunctionTemplate::Inherit");
1278   i::Isolate* i_isolate = info->GetIsolate();
1279   ENTER_V8_NO_SCRIPT_NO_EXCEPTION(i_isolate);
1280   Utils::ApiCheck(info->GetPrototypeProviderTemplate().IsUndefined(i_isolate),
1281                   "v8::FunctionTemplate::Inherit",
1282                   "Protoype provider must be empty");
1283   i::FunctionTemplateInfo::SetParentTemplate(i_isolate, info,
1284                                              Utils::OpenHandle(*value));
1285 }
1286 
FunctionTemplateNew(i::Isolate * isolate,FunctionCallback callback,v8::Local<Value> data,v8::Local<Signature> signature,int length,ConstructorBehavior behavior,bool do_not_cache,v8::Local<Private> cached_property_name=v8::Local<Private> (),SideEffectType side_effect_type=SideEffectType::kHasSideEffect,const MemorySpan<const CFunction> & c_function_overloads={},uint8_t instance_type=0,uint8_t allowed_receiver_instance_type_range_start=0,uint8_t allowed_receiver_instance_type_range_end=0)1287 static Local<FunctionTemplate> FunctionTemplateNew(
1288     i::Isolate* isolate, FunctionCallback callback, v8::Local<Value> data,
1289     v8::Local<Signature> signature, int length, ConstructorBehavior behavior,
1290     bool do_not_cache,
1291     v8::Local<Private> cached_property_name = v8::Local<Private>(),
1292     SideEffectType side_effect_type = SideEffectType::kHasSideEffect,
1293     const MemorySpan<const CFunction>& c_function_overloads = {},
1294     uint8_t instance_type = 0,
1295     uint8_t allowed_receiver_instance_type_range_start = 0,
1296     uint8_t allowed_receiver_instance_type_range_end = 0) {
1297   i::Handle<i::Struct> struct_obj = isolate->factory()->NewStruct(
1298       i::FUNCTION_TEMPLATE_INFO_TYPE, i::AllocationType::kOld);
1299   i::Handle<i::FunctionTemplateInfo> obj =
1300       i::Handle<i::FunctionTemplateInfo>::cast(struct_obj);
1301   {
1302     // Disallow GC until all fields of obj have acceptable types.
1303     i::DisallowGarbageCollection no_gc;
1304     i::FunctionTemplateInfo raw = *obj;
1305     InitializeFunctionTemplate(raw, do_not_cache);
1306     raw.set_length(length);
1307     raw.set_undetectable(false);
1308     raw.set_needs_access_check(false);
1309     raw.set_accept_any_receiver(true);
1310     if (!signature.IsEmpty()) {
1311       raw.set_signature(*Utils::OpenHandle(*signature));
1312     }
1313     raw.set_cached_property_name(
1314         cached_property_name.IsEmpty()
1315             ? i::ReadOnlyRoots(isolate).the_hole_value()
1316             : *Utils::OpenHandle(*cached_property_name));
1317     if (behavior == ConstructorBehavior::kThrow) raw.set_remove_prototype(true);
1318     raw.SetInstanceType(instance_type);
1319     raw.set_allowed_receiver_instance_type_range_start(
1320         allowed_receiver_instance_type_range_start);
1321     raw.set_allowed_receiver_instance_type_range_end(
1322         allowed_receiver_instance_type_range_end);
1323   }
1324   if (callback != nullptr) {
1325     Utils::ToLocal(obj)->SetCallHandler(callback, data, side_effect_type,
1326                                         c_function_overloads);
1327   }
1328   return Utils::ToLocal(obj);
1329 }
1330 
New(Isolate * isolate,FunctionCallback callback,v8::Local<Value> data,v8::Local<Signature> signature,int length,ConstructorBehavior behavior,SideEffectType side_effect_type,const CFunction * c_function,uint16_t instance_type,uint16_t allowed_receiver_instance_type_range_start,uint16_t allowed_receiver_instance_type_range_end)1331 Local<FunctionTemplate> FunctionTemplate::New(
1332     Isolate* isolate, FunctionCallback callback, v8::Local<Value> data,
1333     v8::Local<Signature> signature, int length, ConstructorBehavior behavior,
1334     SideEffectType side_effect_type, const CFunction* c_function,
1335     uint16_t instance_type, uint16_t allowed_receiver_instance_type_range_start,
1336     uint16_t allowed_receiver_instance_type_range_end) {
1337   i::Isolate* i_isolate = reinterpret_cast<i::Isolate*>(isolate);
1338   // Changes to the environment cannot be captured in the snapshot. Expect no
1339   // function templates when the isolate is created for serialization.
1340   API_RCS_SCOPE(i_isolate, FunctionTemplate, New);
1341 
1342   if (!Utils::ApiCheck(
1343           !c_function || behavior == ConstructorBehavior::kThrow,
1344           "FunctionTemplate::New",
1345           "Fast API calls are not supported for constructor functions.")) {
1346     return Local<FunctionTemplate>();
1347   }
1348 
1349   ENTER_V8_NO_SCRIPT_NO_EXCEPTION(i_isolate);
1350   return FunctionTemplateNew(
1351       i_isolate, callback, data, signature, length, behavior, false,
1352       Local<Private>(), side_effect_type,
1353       c_function ? MemorySpan<const CFunction>{c_function, 1}
1354                  : MemorySpan<const CFunction>{},
1355       instance_type, allowed_receiver_instance_type_range_start,
1356       allowed_receiver_instance_type_range_end);
1357 }
1358 
NewWithCFunctionOverloads(Isolate * isolate,FunctionCallback callback,v8::Local<Value> data,v8::Local<Signature> signature,int length,ConstructorBehavior behavior,SideEffectType side_effect_type,const MemorySpan<const CFunction> & c_function_overloads)1359 Local<FunctionTemplate> FunctionTemplate::NewWithCFunctionOverloads(
1360     Isolate* isolate, FunctionCallback callback, v8::Local<Value> data,
1361     v8::Local<Signature> signature, int length, ConstructorBehavior behavior,
1362     SideEffectType side_effect_type,
1363     const MemorySpan<const CFunction>& c_function_overloads) {
1364   i::Isolate* i_isolate = reinterpret_cast<i::Isolate*>(isolate);
1365   API_RCS_SCOPE(i_isolate, FunctionTemplate, New);
1366 
1367   if (!Utils::ApiCheck(
1368           c_function_overloads.size() == 0 ||
1369               behavior == ConstructorBehavior::kThrow,
1370           "FunctionTemplate::NewWithCFunctionOverloads",
1371           "Fast API calls are not supported for constructor functions.")) {
1372     return Local<FunctionTemplate>();
1373   }
1374 
1375   ENTER_V8_NO_SCRIPT_NO_EXCEPTION(i_isolate);
1376   return FunctionTemplateNew(i_isolate, callback, data, signature, length,
1377                              behavior, false, Local<Private>(),
1378                              side_effect_type, c_function_overloads);
1379 }
1380 
NewWithCache(Isolate * isolate,FunctionCallback callback,Local<Private> cache_property,Local<Value> data,Local<Signature> signature,int length,SideEffectType side_effect_type)1381 Local<FunctionTemplate> FunctionTemplate::NewWithCache(
1382     Isolate* isolate, FunctionCallback callback, Local<Private> cache_property,
1383     Local<Value> data, Local<Signature> signature, int length,
1384     SideEffectType side_effect_type) {
1385   i::Isolate* i_isolate = reinterpret_cast<i::Isolate*>(isolate);
1386   API_RCS_SCOPE(i_isolate, FunctionTemplate, NewWithCache);
1387   ENTER_V8_NO_SCRIPT_NO_EXCEPTION(i_isolate);
1388   return FunctionTemplateNew(i_isolate, callback, data, signature, length,
1389                              ConstructorBehavior::kAllow, false, cache_property,
1390                              side_effect_type);
1391 }
1392 
New(Isolate * isolate,Local<FunctionTemplate> receiver)1393 Local<Signature> Signature::New(Isolate* isolate,
1394                                 Local<FunctionTemplate> receiver) {
1395   return Utils::SignatureToLocal(Utils::OpenHandle(*receiver));
1396 }
1397 
New(Isolate * isolate,Local<FunctionTemplate> receiver)1398 Local<AccessorSignature> AccessorSignature::New(
1399     Isolate* isolate, Local<FunctionTemplate> receiver) {
1400   return Utils::AccessorSignatureToLocal(Utils::OpenHandle(*receiver));
1401 }
1402 
1403 #define SET_FIELD_WRAPPED(isolate, obj, setter, cdata)        \
1404   do {                                                        \
1405     i::Handle<i::Object> foreign = FromCData(isolate, cdata); \
1406     (obj)->setter(*foreign);                                  \
1407   } while (false)
1408 
SetCallHandler(FunctionCallback callback,v8::Local<Value> data,SideEffectType side_effect_type,const MemorySpan<const CFunction> & c_function_overloads)1409 void FunctionTemplate::SetCallHandler(
1410     FunctionCallback callback, v8::Local<Value> data,
1411     SideEffectType side_effect_type,
1412     const MemorySpan<const CFunction>& c_function_overloads) {
1413   auto info = Utils::OpenHandle(this);
1414   EnsureNotPublished(info, "v8::FunctionTemplate::SetCallHandler");
1415   i::Isolate* isolate = info->GetIsolate();
1416   ENTER_V8_NO_SCRIPT_NO_EXCEPTION(isolate);
1417   i::HandleScope scope(isolate);
1418   i::Handle<i::CallHandlerInfo> obj = isolate->factory()->NewCallHandlerInfo(
1419       side_effect_type == SideEffectType::kHasNoSideEffect);
1420   SET_FIELD_WRAPPED(isolate, obj, set_callback, callback);
1421   SET_FIELD_WRAPPED(isolate, obj, set_js_callback, obj->redirected_callback());
1422   if (data.IsEmpty()) {
1423     data = v8::Undefined(reinterpret_cast<v8::Isolate*>(isolate));
1424   }
1425   obj->set_data(*Utils::OpenHandle(*data));
1426   if (c_function_overloads.size() > 0) {
1427     // Stores the data for a sequence of CFunction overloads into a single
1428     // FixedArray, as [address_0, signature_0, ... address_n-1, signature_n-1].
1429     i::Handle<i::FixedArray> function_overloads =
1430         isolate->factory()->NewFixedArray(static_cast<int>(
1431             c_function_overloads.size() *
1432             i::FunctionTemplateInfo::kFunctionOverloadEntrySize));
1433     int function_count = static_cast<int>(c_function_overloads.size());
1434     for (int i = 0; i < function_count; i++) {
1435       const CFunction& c_function = c_function_overloads.data()[i];
1436       i::Handle<i::Object> address =
1437           FromCData(isolate, c_function.GetAddress());
1438       function_overloads->set(
1439           i::FunctionTemplateInfo::kFunctionOverloadEntrySize * i, *address);
1440       i::Handle<i::Object> signature =
1441           FromCData(isolate, c_function.GetTypeInfo());
1442       function_overloads->set(
1443           i::FunctionTemplateInfo::kFunctionOverloadEntrySize * i + 1,
1444           *signature);
1445     }
1446     i::FunctionTemplateInfo::SetCFunctionOverloads(isolate, info,
1447                                                    function_overloads);
1448   }
1449   info->set_call_code(*obj, kReleaseStore);
1450 }
1451 
1452 namespace {
1453 
1454 template <typename Getter, typename Setter>
MakeAccessorInfo(i::Isolate * isolate,v8::Local<Name> name,Getter getter,Setter setter,v8::Local<Value> data,v8::AccessControl settings,v8::Local<AccessorSignature> signature,bool is_special_data_property,bool replace_on_access)1455 i::Handle<i::AccessorInfo> MakeAccessorInfo(
1456     i::Isolate* isolate, v8::Local<Name> name, Getter getter, Setter setter,
1457     v8::Local<Value> data, v8::AccessControl settings,
1458     v8::Local<AccessorSignature> signature, bool is_special_data_property,
1459     bool replace_on_access) {
1460   i::Handle<i::AccessorInfo> obj = isolate->factory()->NewAccessorInfo();
1461   SET_FIELD_WRAPPED(isolate, obj, set_getter, getter);
1462   DCHECK_IMPLIES(replace_on_access,
1463                  is_special_data_property && setter == nullptr);
1464   if (is_special_data_property && setter == nullptr) {
1465     setter = reinterpret_cast<Setter>(&i::Accessors::ReconfigureToDataProperty);
1466   }
1467   SET_FIELD_WRAPPED(isolate, obj, set_setter, setter);
1468   i::Address redirected = obj->redirected_getter();
1469   if (redirected != i::kNullAddress) {
1470     SET_FIELD_WRAPPED(isolate, obj, set_js_getter, redirected);
1471   }
1472 
1473   i::Handle<i::Name> accessor_name = Utils::OpenHandle(*name);
1474   if (!accessor_name->IsUniqueName()) {
1475     accessor_name = isolate->factory()->InternalizeString(
1476         i::Handle<i::String>::cast(accessor_name));
1477   }
1478   i::DisallowGarbageCollection no_gc;
1479   i::AccessorInfo raw_obj = *obj;
1480   if (data.IsEmpty()) {
1481     raw_obj.set_data(i::ReadOnlyRoots(isolate).undefined_value());
1482   } else {
1483     raw_obj.set_data(*Utils::OpenHandle(*data));
1484   }
1485   raw_obj.set_name(*accessor_name);
1486   raw_obj.set_is_special_data_property(is_special_data_property);
1487   raw_obj.set_replace_on_access(replace_on_access);
1488   if (settings & ALL_CAN_READ) raw_obj.set_all_can_read(true);
1489   if (settings & ALL_CAN_WRITE) raw_obj.set_all_can_write(true);
1490   raw_obj.set_initial_property_attributes(i::NONE);
1491   if (!signature.IsEmpty()) {
1492     raw_obj.set_expected_receiver_type(*Utils::OpenHandle(*signature));
1493   }
1494   return obj;
1495 }
1496 
1497 }  // namespace
1498 
InstanceTemplate()1499 Local<ObjectTemplate> FunctionTemplate::InstanceTemplate() {
1500   i::Handle<i::FunctionTemplateInfo> handle = Utils::OpenHandle(this, true);
1501   if (!Utils::ApiCheck(!handle.is_null(),
1502                        "v8::FunctionTemplate::InstanceTemplate()",
1503                        "Reading from empty handle")) {
1504     return Local<ObjectTemplate>();
1505   }
1506   i::Isolate* isolate = handle->GetIsolate();
1507   ENTER_V8_NO_SCRIPT_NO_EXCEPTION(isolate);
1508   if (handle->GetInstanceTemplate().IsUndefined(isolate)) {
1509     Local<ObjectTemplate> templ =
1510         ObjectTemplate::New(isolate, ToApiHandle<FunctionTemplate>(handle));
1511     i::FunctionTemplateInfo::SetInstanceTemplate(isolate, handle,
1512                                                  Utils::OpenHandle(*templ));
1513   }
1514   i::Handle<i::ObjectTemplateInfo> result(
1515       i::ObjectTemplateInfo::cast(handle->GetInstanceTemplate()), isolate);
1516   return Utils::ToLocal(result);
1517 }
1518 
SetLength(int length)1519 void FunctionTemplate::SetLength(int length) {
1520   auto info = Utils::OpenHandle(this);
1521   EnsureNotPublished(info, "v8::FunctionTemplate::SetLength");
1522   auto isolate = info->GetIsolate();
1523   ENTER_V8_NO_SCRIPT_NO_EXCEPTION(isolate);
1524   info->set_length(length);
1525 }
1526 
SetClassName(Local<String> name)1527 void FunctionTemplate::SetClassName(Local<String> name) {
1528   auto info = Utils::OpenHandle(this);
1529   EnsureNotPublished(info, "v8::FunctionTemplate::SetClassName");
1530   auto isolate = info->GetIsolate();
1531   ENTER_V8_NO_SCRIPT_NO_EXCEPTION(isolate);
1532   info->set_class_name(*Utils::OpenHandle(*name));
1533 }
1534 
SetAcceptAnyReceiver(bool value)1535 void FunctionTemplate::SetAcceptAnyReceiver(bool value) {
1536   auto info = Utils::OpenHandle(this);
1537   EnsureNotPublished(info, "v8::FunctionTemplate::SetAcceptAnyReceiver");
1538   auto isolate = info->GetIsolate();
1539   ENTER_V8_NO_SCRIPT_NO_EXCEPTION(isolate);
1540   info->set_accept_any_receiver(value);
1541 }
1542 
ReadOnlyPrototype()1543 void FunctionTemplate::ReadOnlyPrototype() {
1544   auto info = Utils::OpenHandle(this);
1545   EnsureNotPublished(info, "v8::FunctionTemplate::ReadOnlyPrototype");
1546   auto isolate = info->GetIsolate();
1547   ENTER_V8_NO_SCRIPT_NO_EXCEPTION(isolate);
1548   info->set_read_only_prototype(true);
1549 }
1550 
RemovePrototype()1551 void FunctionTemplate::RemovePrototype() {
1552   auto info = Utils::OpenHandle(this);
1553   EnsureNotPublished(info, "v8::FunctionTemplate::RemovePrototype");
1554   auto isolate = info->GetIsolate();
1555   ENTER_V8_NO_SCRIPT_NO_EXCEPTION(isolate);
1556   info->set_remove_prototype(true);
1557 }
1558 
1559 // --- O b j e c t T e m p l a t e ---
1560 
New(Isolate * isolate,v8::Local<FunctionTemplate> constructor)1561 Local<ObjectTemplate> ObjectTemplate::New(
1562     Isolate* isolate, v8::Local<FunctionTemplate> constructor) {
1563   return New(reinterpret_cast<i::Isolate*>(isolate), constructor);
1564 }
1565 
ObjectTemplateNew(i::Isolate * isolate,v8::Local<FunctionTemplate> constructor,bool do_not_cache)1566 static Local<ObjectTemplate> ObjectTemplateNew(
1567     i::Isolate* isolate, v8::Local<FunctionTemplate> constructor,
1568     bool do_not_cache) {
1569   API_RCS_SCOPE(isolate, ObjectTemplate, New);
1570   ENTER_V8_NO_SCRIPT_NO_EXCEPTION(isolate);
1571   i::Handle<i::Struct> struct_obj = isolate->factory()->NewStruct(
1572       i::OBJECT_TEMPLATE_INFO_TYPE, i::AllocationType::kOld);
1573   i::Handle<i::ObjectTemplateInfo> obj =
1574       i::Handle<i::ObjectTemplateInfo>::cast(struct_obj);
1575   {
1576     // Disallow GC until all fields of obj have acceptable types.
1577     i::DisallowGarbageCollection no_gc;
1578     i::ObjectTemplateInfo raw = *obj;
1579     InitializeTemplate(raw, Consts::OBJECT_TEMPLATE, do_not_cache);
1580     raw.set_data(0);
1581     if (!constructor.IsEmpty()) {
1582       raw.set_constructor(*Utils::OpenHandle(*constructor));
1583     }
1584   }
1585   return Utils::ToLocal(obj);
1586 }
1587 
New(i::Isolate * isolate,v8::Local<FunctionTemplate> constructor)1588 Local<ObjectTemplate> ObjectTemplate::New(
1589     i::Isolate* isolate, v8::Local<FunctionTemplate> constructor) {
1590   return ObjectTemplateNew(isolate, constructor, false);
1591 }
1592 
1593 // Ensure that the object template has a constructor.  If no
1594 // constructor is available we create one.
EnsureConstructor(i::Isolate * isolate,ObjectTemplate * object_template)1595 static i::Handle<i::FunctionTemplateInfo> EnsureConstructor(
1596     i::Isolate* isolate, ObjectTemplate* object_template) {
1597   i::Object obj = Utils::OpenHandle(object_template)->constructor();
1598   if (!obj.IsUndefined(isolate)) {
1599     i::FunctionTemplateInfo info = i::FunctionTemplateInfo::cast(obj);
1600     return i::Handle<i::FunctionTemplateInfo>(info, isolate);
1601   }
1602   Local<FunctionTemplate> templ =
1603       FunctionTemplate::New(reinterpret_cast<Isolate*>(isolate));
1604   i::Handle<i::FunctionTemplateInfo> constructor = Utils::OpenHandle(*templ);
1605   i::FunctionTemplateInfo::SetInstanceTemplate(
1606       isolate, constructor, Utils::OpenHandle(object_template));
1607   Utils::OpenHandle(object_template)->set_constructor(*constructor);
1608   return constructor;
1609 }
1610 
1611 template <typename Getter, typename Setter, typename Data, typename Template>
TemplateSetAccessor(Template * template_obj,v8::Local<Name> name,Getter getter,Setter setter,Data data,AccessControl settings,PropertyAttribute attribute,v8::Local<AccessorSignature> signature,bool is_special_data_property,bool replace_on_access,SideEffectType getter_side_effect_type,SideEffectType setter_side_effect_type)1612 static void TemplateSetAccessor(
1613     Template* template_obj, v8::Local<Name> name, Getter getter, Setter setter,
1614     Data data, AccessControl settings, PropertyAttribute attribute,
1615     v8::Local<AccessorSignature> signature, bool is_special_data_property,
1616     bool replace_on_access, SideEffectType getter_side_effect_type,
1617     SideEffectType setter_side_effect_type) {
1618   auto info = Utils::OpenHandle(template_obj);
1619   auto isolate = info->GetIsolate();
1620   ENTER_V8_NO_SCRIPT_NO_EXCEPTION(isolate);
1621   i::HandleScope scope(isolate);
1622   i::Handle<i::AccessorInfo> accessor_info =
1623       MakeAccessorInfo(isolate, name, getter, setter, data, settings, signature,
1624                        is_special_data_property, replace_on_access);
1625   {
1626     i::DisallowGarbageCollection no_gc;
1627     i::AccessorInfo raw = *accessor_info;
1628     raw.set_initial_property_attributes(
1629         static_cast<i::PropertyAttributes>(attribute));
1630     raw.set_getter_side_effect_type(getter_side_effect_type);
1631     raw.set_setter_side_effect_type(setter_side_effect_type);
1632   }
1633   i::ApiNatives::AddNativeDataProperty(isolate, info, accessor_info);
1634 }
1635 
SetNativeDataProperty(v8::Local<String> name,AccessorGetterCallback getter,AccessorSetterCallback setter,v8::Local<Value> data,PropertyAttribute attribute,AccessControl settings,SideEffectType getter_side_effect_type,SideEffectType setter_side_effect_type)1636 void Template::SetNativeDataProperty(v8::Local<String> name,
1637                                      AccessorGetterCallback getter,
1638                                      AccessorSetterCallback setter,
1639                                      v8::Local<Value> data,
1640                                      PropertyAttribute attribute,
1641                                      AccessControl settings,
1642                                      SideEffectType getter_side_effect_type,
1643                                      SideEffectType setter_side_effect_type) {
1644   TemplateSetAccessor(this, name, getter, setter, data, settings, attribute,
1645                       Local<AccessorSignature>(), true, false,
1646                       getter_side_effect_type, setter_side_effect_type);
1647 }
1648 
SetNativeDataProperty(v8::Local<String> name,AccessorGetterCallback getter,AccessorSetterCallback setter,v8::Local<Value> data,PropertyAttribute attribute,v8::Local<AccessorSignature> signature,AccessControl settings,SideEffectType getter_side_effect_type,SideEffectType setter_side_effect_type)1649 void Template::SetNativeDataProperty(
1650     v8::Local<String> name, AccessorGetterCallback getter,
1651     AccessorSetterCallback setter, v8::Local<Value> data,
1652     PropertyAttribute attribute, v8::Local<AccessorSignature> signature,
1653     AccessControl settings, SideEffectType getter_side_effect_type,
1654     SideEffectType setter_side_effect_type) {
1655   TemplateSetAccessor(this, name, getter, setter, data, settings, attribute,
1656                       signature, true, false, getter_side_effect_type,
1657                       setter_side_effect_type);
1658 }
1659 
SetNativeDataProperty(v8::Local<Name> name,AccessorNameGetterCallback getter,AccessorNameSetterCallback setter,v8::Local<Value> data,PropertyAttribute attribute,AccessControl settings,SideEffectType getter_side_effect_type,SideEffectType setter_side_effect_type)1660 void Template::SetNativeDataProperty(v8::Local<Name> name,
1661                                      AccessorNameGetterCallback getter,
1662                                      AccessorNameSetterCallback setter,
1663                                      v8::Local<Value> data,
1664                                      PropertyAttribute attribute,
1665                                      AccessControl settings,
1666                                      SideEffectType getter_side_effect_type,
1667                                      SideEffectType setter_side_effect_type) {
1668   TemplateSetAccessor(this, name, getter, setter, data, settings, attribute,
1669                       Local<AccessorSignature>(), true, false,
1670                       getter_side_effect_type, setter_side_effect_type);
1671 }
1672 
SetNativeDataProperty(v8::Local<Name> name,AccessorNameGetterCallback getter,AccessorNameSetterCallback setter,v8::Local<Value> data,PropertyAttribute attribute,v8::Local<AccessorSignature> signature,AccessControl settings,SideEffectType getter_side_effect_type,SideEffectType setter_side_effect_type)1673 void Template::SetNativeDataProperty(
1674     v8::Local<Name> name, AccessorNameGetterCallback getter,
1675     AccessorNameSetterCallback setter, v8::Local<Value> data,
1676     PropertyAttribute attribute, v8::Local<AccessorSignature> signature,
1677     AccessControl settings, SideEffectType getter_side_effect_type,
1678     SideEffectType setter_side_effect_type) {
1679   TemplateSetAccessor(this, name, getter, setter, data, settings, attribute,
1680                       signature, true, false, getter_side_effect_type,
1681                       setter_side_effect_type);
1682 }
1683 
SetLazyDataProperty(v8::Local<Name> name,AccessorNameGetterCallback getter,v8::Local<Value> data,PropertyAttribute attribute,SideEffectType getter_side_effect_type,SideEffectType setter_side_effect_type)1684 void Template::SetLazyDataProperty(v8::Local<Name> name,
1685                                    AccessorNameGetterCallback getter,
1686                                    v8::Local<Value> data,
1687                                    PropertyAttribute attribute,
1688                                    SideEffectType getter_side_effect_type,
1689                                    SideEffectType setter_side_effect_type) {
1690   TemplateSetAccessor(this, name, getter,
1691                       static_cast<AccessorNameSetterCallback>(nullptr), data,
1692                       DEFAULT, attribute, Local<AccessorSignature>(), true,
1693                       true, getter_side_effect_type, setter_side_effect_type);
1694 }
1695 
SetIntrinsicDataProperty(Local<Name> name,Intrinsic intrinsic,PropertyAttribute attribute)1696 void Template::SetIntrinsicDataProperty(Local<Name> name, Intrinsic intrinsic,
1697                                         PropertyAttribute attribute) {
1698   auto templ = Utils::OpenHandle(this);
1699   i::Isolate* isolate = templ->GetIsolate();
1700   ENTER_V8_NO_SCRIPT_NO_EXCEPTION(isolate);
1701   i::HandleScope scope(isolate);
1702   i::ApiNatives::AddDataProperty(isolate, templ, Utils::OpenHandle(*name),
1703                                  intrinsic,
1704                                  static_cast<i::PropertyAttributes>(attribute));
1705 }
1706 
SetAccessor(v8::Local<String> name,AccessorGetterCallback getter,AccessorSetterCallback setter,v8::Local<Value> data,AccessControl settings,PropertyAttribute attribute,SideEffectType getter_side_effect_type,SideEffectType setter_side_effect_type)1707 void ObjectTemplate::SetAccessor(v8::Local<String> name,
1708                                  AccessorGetterCallback getter,
1709                                  AccessorSetterCallback setter,
1710                                  v8::Local<Value> data, AccessControl settings,
1711                                  PropertyAttribute attribute,
1712                                  SideEffectType getter_side_effect_type,
1713                                  SideEffectType setter_side_effect_type) {
1714   TemplateSetAccessor(this, name, getter, setter, data, settings, attribute,
1715                       Local<AccessorSignature>(),
1716                       i::FLAG_disable_old_api_accessors, false,
1717                       getter_side_effect_type, setter_side_effect_type);
1718 }
1719 
SetAccessor(v8::Local<Name> name,AccessorNameGetterCallback getter,AccessorNameSetterCallback setter,v8::Local<Value> data,AccessControl settings,PropertyAttribute attribute,SideEffectType getter_side_effect_type,SideEffectType setter_side_effect_type)1720 void ObjectTemplate::SetAccessor(v8::Local<Name> name,
1721                                  AccessorNameGetterCallback getter,
1722                                  AccessorNameSetterCallback setter,
1723                                  v8::Local<Value> data, AccessControl settings,
1724                                  PropertyAttribute attribute,
1725                                  SideEffectType getter_side_effect_type,
1726                                  SideEffectType setter_side_effect_type) {
1727   TemplateSetAccessor(this, name, getter, setter, data, settings, attribute,
1728                       Local<AccessorSignature>(),
1729                       i::FLAG_disable_old_api_accessors, false,
1730                       getter_side_effect_type, setter_side_effect_type);
1731 }
1732 
SetAccessor(v8::Local<String> name,AccessorGetterCallback getter,AccessorSetterCallback setter,v8::Local<Value> data,AccessControl settings,PropertyAttribute attribute,v8::Local<AccessorSignature> signature,SideEffectType getter_side_effect_type,SideEffectType setter_side_effect_type)1733 void ObjectTemplate::SetAccessor(v8::Local<String> name,
1734                                  AccessorGetterCallback getter,
1735                                  AccessorSetterCallback setter,
1736                                  v8::Local<Value> data, AccessControl settings,
1737                                  PropertyAttribute attribute,
1738                                  v8::Local<AccessorSignature> signature,
1739                                  SideEffectType getter_side_effect_type,
1740                                  SideEffectType setter_side_effect_type) {
1741   TemplateSetAccessor(this, name, getter, setter, data, settings, attribute,
1742                       signature, i::FLAG_disable_old_api_accessors, false,
1743                       getter_side_effect_type, setter_side_effect_type);
1744 }
1745 
SetAccessor(v8::Local<Name> name,AccessorNameGetterCallback getter,AccessorNameSetterCallback setter,v8::Local<Value> data,AccessControl settings,PropertyAttribute attribute,v8::Local<AccessorSignature> signature,SideEffectType getter_side_effect_type,SideEffectType setter_side_effect_type)1746 void ObjectTemplate::SetAccessor(v8::Local<Name> name,
1747                                  AccessorNameGetterCallback getter,
1748                                  AccessorNameSetterCallback setter,
1749                                  v8::Local<Value> data, AccessControl settings,
1750                                  PropertyAttribute attribute,
1751                                  v8::Local<AccessorSignature> signature,
1752                                  SideEffectType getter_side_effect_type,
1753                                  SideEffectType setter_side_effect_type) {
1754   TemplateSetAccessor(this, name, getter, setter, data, settings, attribute,
1755                       signature, i::FLAG_disable_old_api_accessors, false,
1756                       getter_side_effect_type, setter_side_effect_type);
1757 }
1758 
1759 template <typename Getter, typename Setter, typename Query, typename Descriptor,
1760           typename Deleter, typename Enumerator, typename Definer>
CreateInterceptorInfo(i::Isolate * isolate,Getter getter,Setter setter,Query query,Descriptor descriptor,Deleter remover,Enumerator enumerator,Definer definer,Local<Value> data,PropertyHandlerFlags flags)1761 static i::Handle<i::InterceptorInfo> CreateInterceptorInfo(
1762     i::Isolate* isolate, Getter getter, Setter setter, Query query,
1763     Descriptor descriptor, Deleter remover, Enumerator enumerator,
1764     Definer definer, Local<Value> data, PropertyHandlerFlags flags) {
1765   auto obj = i::Handle<i::InterceptorInfo>::cast(isolate->factory()->NewStruct(
1766       i::INTERCEPTOR_INFO_TYPE, i::AllocationType::kOld));
1767   obj->set_flags(0);
1768 
1769   if (getter != nullptr) SET_FIELD_WRAPPED(isolate, obj, set_getter, getter);
1770   if (setter != nullptr) SET_FIELD_WRAPPED(isolate, obj, set_setter, setter);
1771   if (query != nullptr) SET_FIELD_WRAPPED(isolate, obj, set_query, query);
1772   if (descriptor != nullptr)
1773     SET_FIELD_WRAPPED(isolate, obj, set_descriptor, descriptor);
1774   if (remover != nullptr) SET_FIELD_WRAPPED(isolate, obj, set_deleter, remover);
1775   if (enumerator != nullptr)
1776     SET_FIELD_WRAPPED(isolate, obj, set_enumerator, enumerator);
1777   if (definer != nullptr) SET_FIELD_WRAPPED(isolate, obj, set_definer, definer);
1778   obj->set_can_intercept_symbols(
1779       !(static_cast<int>(flags) &
1780         static_cast<int>(PropertyHandlerFlags::kOnlyInterceptStrings)));
1781   obj->set_all_can_read(static_cast<int>(flags) &
1782                         static_cast<int>(PropertyHandlerFlags::kAllCanRead));
1783   obj->set_non_masking(static_cast<int>(flags) &
1784                        static_cast<int>(PropertyHandlerFlags::kNonMasking));
1785   obj->set_has_no_side_effect(
1786       static_cast<int>(flags) &
1787       static_cast<int>(PropertyHandlerFlags::kHasNoSideEffect));
1788 
1789   if (data.IsEmpty()) {
1790     data = v8::Undefined(reinterpret_cast<v8::Isolate*>(isolate));
1791   }
1792   obj->set_data(*Utils::OpenHandle(*data));
1793   return obj;
1794 }
1795 
1796 template <typename Getter, typename Setter, typename Query, typename Descriptor,
1797           typename Deleter, typename Enumerator, typename Definer>
CreateNamedInterceptorInfo(i::Isolate * isolate,Getter getter,Setter setter,Query query,Descriptor descriptor,Deleter remover,Enumerator enumerator,Definer definer,Local<Value> data,PropertyHandlerFlags flags)1798 static i::Handle<i::InterceptorInfo> CreateNamedInterceptorInfo(
1799     i::Isolate* isolate, Getter getter, Setter setter, Query query,
1800     Descriptor descriptor, Deleter remover, Enumerator enumerator,
1801     Definer definer, Local<Value> data, PropertyHandlerFlags flags) {
1802   auto interceptor =
1803       CreateInterceptorInfo(isolate, getter, setter, query, descriptor, remover,
1804                             enumerator, definer, data, flags);
1805   interceptor->set_is_named(true);
1806   return interceptor;
1807 }
1808 
1809 template <typename Getter, typename Setter, typename Query, typename Descriptor,
1810           typename Deleter, typename Enumerator, typename Definer>
CreateIndexedInterceptorInfo(i::Isolate * isolate,Getter getter,Setter setter,Query query,Descriptor descriptor,Deleter remover,Enumerator enumerator,Definer definer,Local<Value> data,PropertyHandlerFlags flags)1811 static i::Handle<i::InterceptorInfo> CreateIndexedInterceptorInfo(
1812     i::Isolate* isolate, Getter getter, Setter setter, Query query,
1813     Descriptor descriptor, Deleter remover, Enumerator enumerator,
1814     Definer definer, Local<Value> data, PropertyHandlerFlags flags) {
1815   auto interceptor =
1816       CreateInterceptorInfo(isolate, getter, setter, query, descriptor, remover,
1817                             enumerator, definer, data, flags);
1818   interceptor->set_is_named(false);
1819   return interceptor;
1820 }
1821 
1822 template <typename Getter, typename Setter, typename Query, typename Descriptor,
1823           typename Deleter, typename Enumerator, typename Definer>
ObjectTemplateSetNamedPropertyHandler(ObjectTemplate * templ,Getter getter,Setter setter,Query query,Descriptor descriptor,Deleter remover,Enumerator enumerator,Definer definer,Local<Value> data,PropertyHandlerFlags flags)1824 static void ObjectTemplateSetNamedPropertyHandler(
1825     ObjectTemplate* templ, Getter getter, Setter setter, Query query,
1826     Descriptor descriptor, Deleter remover, Enumerator enumerator,
1827     Definer definer, Local<Value> data, PropertyHandlerFlags flags) {
1828   i::Isolate* isolate = Utils::OpenHandle(templ)->GetIsolate();
1829   ENTER_V8_NO_SCRIPT_NO_EXCEPTION(isolate);
1830   i::HandleScope scope(isolate);
1831   auto cons = EnsureConstructor(isolate, templ);
1832   EnsureNotPublished(cons, "ObjectTemplateSetNamedPropertyHandler");
1833   auto obj =
1834       CreateNamedInterceptorInfo(isolate, getter, setter, query, descriptor,
1835                                  remover, enumerator, definer, data, flags);
1836   i::FunctionTemplateInfo::SetNamedPropertyHandler(isolate, cons, obj);
1837 }
1838 
SetHandler(const NamedPropertyHandlerConfiguration & config)1839 void ObjectTemplate::SetHandler(
1840     const NamedPropertyHandlerConfiguration& config) {
1841   ObjectTemplateSetNamedPropertyHandler(
1842       this, config.getter, config.setter, config.query, config.descriptor,
1843       config.deleter, config.enumerator, config.definer, config.data,
1844       config.flags);
1845 }
1846 
MarkAsUndetectable()1847 void ObjectTemplate::MarkAsUndetectable() {
1848   i::Isolate* isolate = Utils::OpenHandle(this)->GetIsolate();
1849   ENTER_V8_NO_SCRIPT_NO_EXCEPTION(isolate);
1850   i::HandleScope scope(isolate);
1851   auto cons = EnsureConstructor(isolate, this);
1852   EnsureNotPublished(cons, "v8::ObjectTemplate::MarkAsUndetectable");
1853   cons->set_undetectable(true);
1854 }
1855 
SetAccessCheckCallback(AccessCheckCallback callback,Local<Value> data)1856 void ObjectTemplate::SetAccessCheckCallback(AccessCheckCallback callback,
1857                                             Local<Value> data) {
1858   i::Isolate* isolate = Utils::OpenHandle(this)->GetIsolate();
1859   ENTER_V8_NO_SCRIPT_NO_EXCEPTION(isolate);
1860   i::HandleScope scope(isolate);
1861   auto cons = EnsureConstructor(isolate, this);
1862   EnsureNotPublished(cons, "v8::ObjectTemplate::SetAccessCheckCallback");
1863 
1864   i::Handle<i::Struct> struct_info = isolate->factory()->NewStruct(
1865       i::ACCESS_CHECK_INFO_TYPE, i::AllocationType::kOld);
1866   i::Handle<i::AccessCheckInfo> info =
1867       i::Handle<i::AccessCheckInfo>::cast(struct_info);
1868 
1869   SET_FIELD_WRAPPED(isolate, info, set_callback, callback);
1870   info->set_named_interceptor(i::Object());
1871   info->set_indexed_interceptor(i::Object());
1872 
1873   if (data.IsEmpty()) {
1874     data = v8::Undefined(reinterpret_cast<v8::Isolate*>(isolate));
1875   }
1876   info->set_data(*Utils::OpenHandle(*data));
1877 
1878   i::FunctionTemplateInfo::SetAccessCheckInfo(isolate, cons, info);
1879   cons->set_needs_access_check(true);
1880 }
1881 
SetAccessCheckCallbackAndHandler(AccessCheckCallback callback,const NamedPropertyHandlerConfiguration & named_handler,const IndexedPropertyHandlerConfiguration & indexed_handler,Local<Value> data)1882 void ObjectTemplate::SetAccessCheckCallbackAndHandler(
1883     AccessCheckCallback callback,
1884     const NamedPropertyHandlerConfiguration& named_handler,
1885     const IndexedPropertyHandlerConfiguration& indexed_handler,
1886     Local<Value> data) {
1887   i::Isolate* isolate = Utils::OpenHandle(this)->GetIsolate();
1888   ENTER_V8_NO_SCRIPT_NO_EXCEPTION(isolate);
1889   i::HandleScope scope(isolate);
1890   auto cons = EnsureConstructor(isolate, this);
1891   EnsureNotPublished(cons,
1892                      "v8::ObjectTemplate::SetAccessCheckCallbackWithHandler");
1893 
1894   i::Handle<i::Struct> struct_info = isolate->factory()->NewStruct(
1895       i::ACCESS_CHECK_INFO_TYPE, i::AllocationType::kOld);
1896   i::Handle<i::AccessCheckInfo> info =
1897       i::Handle<i::AccessCheckInfo>::cast(struct_info);
1898 
1899   SET_FIELD_WRAPPED(isolate, info, set_callback, callback);
1900   auto named_interceptor = CreateNamedInterceptorInfo(
1901       isolate, named_handler.getter, named_handler.setter, named_handler.query,
1902       named_handler.descriptor, named_handler.deleter, named_handler.enumerator,
1903       named_handler.definer, named_handler.data, named_handler.flags);
1904   info->set_named_interceptor(*named_interceptor);
1905   auto indexed_interceptor = CreateIndexedInterceptorInfo(
1906       isolate, indexed_handler.getter, indexed_handler.setter,
1907       indexed_handler.query, indexed_handler.descriptor,
1908       indexed_handler.deleter, indexed_handler.enumerator,
1909       indexed_handler.definer, indexed_handler.data, indexed_handler.flags);
1910   info->set_indexed_interceptor(*indexed_interceptor);
1911 
1912   if (data.IsEmpty()) {
1913     data = v8::Undefined(reinterpret_cast<v8::Isolate*>(isolate));
1914   }
1915   info->set_data(*Utils::OpenHandle(*data));
1916 
1917   i::FunctionTemplateInfo::SetAccessCheckInfo(isolate, cons, info);
1918   cons->set_needs_access_check(true);
1919 }
1920 
SetHandler(const IndexedPropertyHandlerConfiguration & config)1921 void ObjectTemplate::SetHandler(
1922     const IndexedPropertyHandlerConfiguration& config) {
1923   i::Isolate* isolate = Utils::OpenHandle(this)->GetIsolate();
1924   ENTER_V8_NO_SCRIPT_NO_EXCEPTION(isolate);
1925   i::HandleScope scope(isolate);
1926   auto cons = EnsureConstructor(isolate, this);
1927   EnsureNotPublished(cons, "v8::ObjectTemplate::SetHandler");
1928   auto obj = CreateIndexedInterceptorInfo(
1929       isolate, config.getter, config.setter, config.query, config.descriptor,
1930       config.deleter, config.enumerator, config.definer, config.data,
1931       config.flags);
1932   i::FunctionTemplateInfo::SetIndexedPropertyHandler(isolate, cons, obj);
1933 }
1934 
SetCallAsFunctionHandler(FunctionCallback callback,Local<Value> data)1935 void ObjectTemplate::SetCallAsFunctionHandler(FunctionCallback callback,
1936                                               Local<Value> data) {
1937   i::Isolate* isolate = Utils::OpenHandle(this)->GetIsolate();
1938   ENTER_V8_NO_SCRIPT_NO_EXCEPTION(isolate);
1939   i::HandleScope scope(isolate);
1940   auto cons = EnsureConstructor(isolate, this);
1941   EnsureNotPublished(cons, "v8::ObjectTemplate::SetCallAsFunctionHandler");
1942   i::Handle<i::CallHandlerInfo> obj = isolate->factory()->NewCallHandlerInfo();
1943   SET_FIELD_WRAPPED(isolate, obj, set_callback, callback);
1944   SET_FIELD_WRAPPED(isolate, obj, set_js_callback, obj->redirected_callback());
1945   if (data.IsEmpty()) {
1946     data = v8::Undefined(reinterpret_cast<v8::Isolate*>(isolate));
1947   }
1948   obj->set_data(*Utils::OpenHandle(*data));
1949   i::FunctionTemplateInfo::SetInstanceCallHandler(isolate, cons, obj);
1950 }
1951 
InternalFieldCount() const1952 int ObjectTemplate::InternalFieldCount() const {
1953   return Utils::OpenHandle(this)->embedder_field_count();
1954 }
1955 
SetInternalFieldCount(int value)1956 void ObjectTemplate::SetInternalFieldCount(int value) {
1957   i::Isolate* isolate = Utils::OpenHandle(this)->GetIsolate();
1958   if (!Utils::ApiCheck(i::Smi::IsValid(value),
1959                        "v8::ObjectTemplate::SetInternalFieldCount()",
1960                        "Invalid embedder field count")) {
1961     return;
1962   }
1963   ENTER_V8_NO_SCRIPT_NO_EXCEPTION(isolate);
1964   if (value > 0) {
1965     // The embedder field count is set by the constructor function's
1966     // construct code, so we ensure that there is a constructor
1967     // function to do the setting.
1968     EnsureConstructor(isolate, this);
1969   }
1970   Utils::OpenHandle(this)->set_embedder_field_count(value);
1971 }
1972 
IsImmutableProto() const1973 bool ObjectTemplate::IsImmutableProto() const {
1974   return Utils::OpenHandle(this)->immutable_proto();
1975 }
1976 
SetImmutableProto()1977 void ObjectTemplate::SetImmutableProto() {
1978   auto self = Utils::OpenHandle(this);
1979   i::Isolate* isolate = self->GetIsolate();
1980   ENTER_V8_NO_SCRIPT_NO_EXCEPTION(isolate);
1981   self->set_immutable_proto(true);
1982 }
1983 
IsCodeLike() const1984 bool ObjectTemplate::IsCodeLike() const {
1985   return Utils::OpenHandle(this)->code_like();
1986 }
1987 
SetCodeLike()1988 void ObjectTemplate::SetCodeLike() {
1989   auto self = Utils::OpenHandle(this);
1990   i::Isolate* isolate = self->GetIsolate();
1991   ENTER_V8_NO_SCRIPT_NO_EXCEPTION(isolate);
1992   self->set_code_like(true);
1993 }
1994 
1995 // --- S c r i p t s ---
1996 
1997 // Internally, UnboundScript is a SharedFunctionInfo, and Script is a
1998 // JSFunction.
1999 
CachedData(const uint8_t * data_,int length_,BufferPolicy buffer_policy_)2000 ScriptCompiler::CachedData::CachedData(const uint8_t* data_, int length_,
2001                                        BufferPolicy buffer_policy_)
2002     : data(data_),
2003       length(length_),
2004       rejected(false),
2005       buffer_policy(buffer_policy_) {}
2006 
~CachedData()2007 ScriptCompiler::CachedData::~CachedData() {
2008   if (buffer_policy == BufferOwned) {
2009     delete[] data;
2010   }
2011 }
2012 
StreamedSource(std::unique_ptr<ExternalSourceStream> stream,Encoding encoding)2013 ScriptCompiler::StreamedSource::StreamedSource(
2014     std::unique_ptr<ExternalSourceStream> stream, Encoding encoding)
2015     : impl_(new i::ScriptStreamingData(std::move(stream), encoding)) {}
2016 
2017 ScriptCompiler::StreamedSource::~StreamedSource() = default;
2018 
BindToCurrentContext()2019 Local<Script> UnboundScript::BindToCurrentContext() {
2020   auto function_info =
2021       i::Handle<i::SharedFunctionInfo>::cast(Utils::OpenHandle(this));
2022   i::Isolate* isolate = function_info->GetIsolate();
2023   ENTER_V8_NO_SCRIPT_NO_EXCEPTION(isolate);
2024   i::Handle<i::JSFunction> function =
2025       i::Factory::JSFunctionBuilder{isolate, function_info,
2026                                     isolate->native_context()}
2027           .Build();
2028   return ToApiHandle<Script>(function);
2029 }
2030 
GetId() const2031 int UnboundScript::GetId() const {
2032   auto function_info = i::SharedFunctionInfo::cast(*Utils::OpenHandle(this));
2033   API_RCS_SCOPE(function_info.GetIsolate(), UnboundScript, GetId);
2034   return i::Script::cast(function_info.script()).id();
2035 }
2036 
GetLineNumber(int code_pos)2037 int UnboundScript::GetLineNumber(int code_pos) {
2038   i::Handle<i::SharedFunctionInfo> obj =
2039       i::Handle<i::SharedFunctionInfo>::cast(Utils::OpenHandle(this));
2040   i::Isolate* isolate = obj->GetIsolate();
2041   ENTER_V8_NO_SCRIPT_NO_EXCEPTION(isolate);
2042   API_RCS_SCOPE(isolate, UnboundScript, GetLineNumber);
2043   if (obj->script().IsScript()) {
2044     i::Handle<i::Script> script(i::Script::cast(obj->script()), isolate);
2045     return i::Script::GetLineNumber(script, code_pos);
2046   } else {
2047     return -1;
2048   }
2049 }
2050 
GetScriptName()2051 Local<Value> UnboundScript::GetScriptName() {
2052   i::Handle<i::SharedFunctionInfo> obj =
2053       i::Handle<i::SharedFunctionInfo>::cast(Utils::OpenHandle(this));
2054   i::Isolate* isolate = obj->GetIsolate();
2055   ENTER_V8_NO_SCRIPT_NO_EXCEPTION(isolate);
2056   API_RCS_SCOPE(isolate, UnboundScript, GetName);
2057   if (obj->script().IsScript()) {
2058     i::Object name = i::Script::cast(obj->script()).name();
2059     return Utils::ToLocal(i::Handle<i::Object>(name, isolate));
2060   } else {
2061     return Local<String>();
2062   }
2063 }
2064 
GetSourceURL()2065 Local<Value> UnboundScript::GetSourceURL() {
2066   i::Handle<i::SharedFunctionInfo> obj =
2067       i::Handle<i::SharedFunctionInfo>::cast(Utils::OpenHandle(this));
2068   i::Isolate* isolate = obj->GetIsolate();
2069   ENTER_V8_NO_SCRIPT_NO_EXCEPTION(isolate);
2070   API_RCS_SCOPE(isolate, UnboundScript, GetSourceURL);
2071   if (obj->script().IsScript()) {
2072     i::Object url = i::Script::cast(obj->script()).source_url();
2073     return Utils::ToLocal(i::Handle<i::Object>(url, isolate));
2074   } else {
2075     return Local<String>();
2076   }
2077 }
2078 
GetSourceMappingURL()2079 Local<Value> UnboundScript::GetSourceMappingURL() {
2080   i::Handle<i::SharedFunctionInfo> obj =
2081       i::Handle<i::SharedFunctionInfo>::cast(Utils::OpenHandle(this));
2082   i::Isolate* isolate = obj->GetIsolate();
2083   API_RCS_SCOPE(isolate, UnboundScript, GetSourceMappingURL);
2084   ENTER_V8_NO_SCRIPT_NO_EXCEPTION(isolate);
2085   if (obj->script().IsScript()) {
2086     i::Object url = i::Script::cast(obj->script()).source_mapping_url();
2087     return Utils::ToLocal(i::Handle<i::Object>(url, isolate));
2088   } else {
2089     return Local<String>();
2090   }
2091 }
2092 
Run(Local<Context> context)2093 MaybeLocal<Value> Script::Run(Local<Context> context) {
2094   return Run(context, Local<Data>());
2095 }
2096 
Run(Local<Context> context,Local<Data> host_defined_options)2097 MaybeLocal<Value> Script::Run(Local<Context> context,
2098                               Local<Data> host_defined_options) {
2099   auto v8_isolate = context->GetIsolate();
2100   auto isolate = reinterpret_cast<i::Isolate*>(v8_isolate);
2101   TRACE_EVENT_CALL_STATS_SCOPED(isolate, "v8", "V8.Execute");
2102   ENTER_V8(isolate, context, Script, Run, MaybeLocal<Value>(),
2103            InternalEscapableScope);
2104   i::TimerEventScope<i::TimerEventExecute> timer_scope(isolate);
2105   i::NestedTimedHistogramScope execute_timer(isolate->counters()->execute(),
2106                                              isolate);
2107   i::AggregatingHistogramTimerScope histogram_timer(
2108       isolate->counters()->compile_lazy());
2109   auto fun = i::Handle<i::JSFunction>::cast(Utils::OpenHandle(this));
2110 
2111   // TODO(crbug.com/1193459): remove once ablation study is completed
2112   base::ElapsedTimer timer;
2113   base::TimeDelta delta;
2114   if (i::FLAG_script_delay > 0) {
2115     delta = v8::base::TimeDelta::FromMillisecondsD(i::FLAG_script_delay);
2116   }
2117   if (i::FLAG_script_delay_once > 0 && !isolate->did_run_script_delay()) {
2118     delta = v8::base::TimeDelta::FromMillisecondsD(i::FLAG_script_delay_once);
2119     isolate->set_did_run_script_delay(true);
2120   }
2121   if (i::FLAG_script_delay_fraction > 0.0) {
2122     timer.Start();
2123   } else if (delta.InMicroseconds() > 0) {
2124     timer.Start();
2125     while (timer.Elapsed() < delta) {
2126       // Busy wait.
2127     }
2128   }
2129 
2130   if (V8_UNLIKELY(i::FLAG_experimental_web_snapshots)) {
2131     i::Handle<i::HeapObject> maybe_script =
2132         handle(fun->shared().script(), isolate);
2133     if (maybe_script->IsScript() &&
2134         i::Script::cast(*maybe_script).type() == i::Script::TYPE_WEB_SNAPSHOT) {
2135       i::WebSnapshotDeserializer deserializer(
2136           reinterpret_cast<i::Isolate*>(v8_isolate),
2137           i::Handle<i::Script>::cast(maybe_script));
2138       deserializer.Deserialize();
2139       RETURN_ON_FAILED_EXECUTION(Value);
2140       Local<Value> result = v8::Undefined(v8_isolate);
2141       RETURN_ESCAPED(result);
2142     }
2143   }
2144 
2145   i::Handle<i::Object> receiver = isolate->global_proxy();
2146   // TODO(cbruni, chromium:1244145): Remove once migrated to the context.
2147   i::Handle<i::Object> options(
2148       i::Script::cast(fun->shared().script()).host_defined_options(), isolate);
2149   Local<Value> result;
2150   has_pending_exception = !ToLocal<Value>(
2151       i::Execution::CallScript(isolate, fun, receiver, options), &result);
2152 
2153   if (i::FLAG_script_delay_fraction > 0.0) {
2154     delta = v8::base::TimeDelta::FromMillisecondsD(
2155         timer.Elapsed().InMillisecondsF() * i::FLAG_script_delay_fraction);
2156     timer.Restart();
2157     while (timer.Elapsed() < delta) {
2158       // Busy wait.
2159     }
2160   }
2161 
2162   RETURN_ON_FAILED_EXECUTION(Value);
2163   RETURN_ESCAPED(result);
2164 }
2165 
GetResourceName()2166 Local<Value> ScriptOrModule::GetResourceName() {
2167   i::Handle<i::ScriptOrModule> obj = Utils::OpenHandle(this);
2168   i::Isolate* isolate = i::GetIsolateFromWritableObject(*obj);
2169   ENTER_V8_NO_SCRIPT_NO_EXCEPTION(isolate);
2170   i::Handle<i::Object> val(obj->resource_name(), isolate);
2171   return ToApiHandle<Value>(val);
2172 }
2173 
GetHostDefinedOptions()2174 Local<PrimitiveArray> ScriptOrModule::GetHostDefinedOptions() {
2175   return HostDefinedOptions().As<PrimitiveArray>();
2176 }
2177 
HostDefinedOptions()2178 Local<Data> ScriptOrModule::HostDefinedOptions() {
2179   i::Handle<i::ScriptOrModule> obj = Utils::OpenHandle(this);
2180   i::Isolate* isolate = i::GetIsolateFromWritableObject(*obj);
2181   ENTER_V8_NO_SCRIPT_NO_EXCEPTION(isolate);
2182   i::Handle<i::Object> val(obj->host_defined_options(), isolate);
2183   return ToApiHandle<Data>(val);
2184 }
2185 
GetUnboundScript()2186 Local<UnboundScript> Script::GetUnboundScript() {
2187   i::DisallowGarbageCollection no_gc;
2188   i::Handle<i::JSFunction> obj = Utils::OpenHandle(this);
2189   i::SharedFunctionInfo sfi = (*obj).shared();
2190   i::Isolate* isolate = sfi.GetIsolate();
2191   return ToApiHandle<UnboundScript>(i::handle(sfi, isolate));
2192 }
2193 
GetResourceName()2194 Local<Value> Script::GetResourceName() {
2195   i::DisallowGarbageCollection no_gc;
2196   i::Handle<i::JSFunction> func = Utils::OpenHandle(this);
2197   i::SharedFunctionInfo sfi = (*func).shared();
2198   i::Isolate* isolate = func->GetIsolate();
2199   CHECK(sfi.script().IsScript());
2200   return ToApiHandle<Value>(
2201       i::handle(i::Script::cast(sfi.script()).name(), isolate));
2202 }
2203 
2204 // static
New(Isolate * v8_isolate,int length)2205 Local<PrimitiveArray> PrimitiveArray::New(Isolate* v8_isolate, int length) {
2206   i::Isolate* isolate = reinterpret_cast<i::Isolate*>(v8_isolate);
2207   ENTER_V8_NO_SCRIPT_NO_EXCEPTION(isolate);
2208   Utils::ApiCheck(length >= 0, "v8::PrimitiveArray::New",
2209                   "length must be equal or greater than zero");
2210   i::Handle<i::FixedArray> array = isolate->factory()->NewFixedArray(length);
2211   return ToApiHandle<PrimitiveArray>(array);
2212 }
2213 
Length() const2214 int PrimitiveArray::Length() const {
2215   i::Handle<i::FixedArray> array = Utils::OpenHandle(this);
2216   return array->length();
2217 }
2218 
Set(Isolate * v8_isolate,int index,Local<Primitive> item)2219 void PrimitiveArray::Set(Isolate* v8_isolate, int index,
2220                          Local<Primitive> item) {
2221   i::Isolate* isolate = reinterpret_cast<i::Isolate*>(v8_isolate);
2222   i::Handle<i::FixedArray> array = Utils::OpenHandle(this);
2223   ENTER_V8_NO_SCRIPT_NO_EXCEPTION(isolate);
2224   Utils::ApiCheck(index >= 0 && index < array->length(),
2225                   "v8::PrimitiveArray::Set",
2226                   "index must be greater than or equal to 0 and less than the "
2227                   "array length");
2228   i::Handle<i::Object> i_item = Utils::OpenHandle(*item);
2229   array->set(index, *i_item);
2230 }
2231 
Get(Isolate * v8_isolate,int index)2232 Local<Primitive> PrimitiveArray::Get(Isolate* v8_isolate, int index) {
2233   i::Isolate* isolate = reinterpret_cast<i::Isolate*>(v8_isolate);
2234   i::Handle<i::FixedArray> array = Utils::OpenHandle(this);
2235   ENTER_V8_NO_SCRIPT_NO_EXCEPTION(isolate);
2236   Utils::ApiCheck(index >= 0 && index < array->length(),
2237                   "v8::PrimitiveArray::Get",
2238                   "index must be greater than or equal to 0 and less than the "
2239                   "array length");
2240   i::Handle<i::Object> i_item(array->get(index), isolate);
2241   return ToApiHandle<Primitive>(i_item);
2242 }
2243 
CheckCast(v8::Data * that)2244 void v8::PrimitiveArray::CheckCast(v8::Data* that) {
2245   i::Handle<i::Object> obj = Utils::OpenHandle(that);
2246   Utils::ApiCheck(
2247       obj->IsFixedArray(), "v8::PrimitiveArray::Cast",
2248       "Value is not a PrimitiveArray. This is a temporary issue, v8::Data and "
2249       "v8::PrimitiveArray will not be compatible in the future.");
2250 }
2251 
Length() const2252 int FixedArray::Length() const {
2253   i::Handle<i::FixedArray> self = Utils::OpenHandle(this);
2254   return self->length();
2255 }
2256 
Get(Local<Context> context,int i) const2257 Local<Data> FixedArray::Get(Local<Context> context, int i) const {
2258   i::Handle<i::FixedArray> self = Utils::OpenHandle(this);
2259   auto isolate = reinterpret_cast<i::Isolate*>(context->GetIsolate());
2260   CHECK_LT(i, self->length());
2261   i::Handle<i::Object> entry(self->get(i), isolate);
2262   return ToApiHandle<Data>(entry);
2263 }
2264 
GetSpecifier() const2265 Local<String> ModuleRequest::GetSpecifier() const {
2266   i::Handle<i::ModuleRequest> self = Utils::OpenHandle(this);
2267   i::Isolate* isolate = self->GetIsolate();
2268   return ToApiHandle<String>(i::handle(self->specifier(), isolate));
2269 }
2270 
GetSourceOffset() const2271 int ModuleRequest::GetSourceOffset() const {
2272   return Utils::OpenHandle(this)->position();
2273 }
2274 
GetImportAssertions() const2275 Local<FixedArray> ModuleRequest::GetImportAssertions() const {
2276   i::Handle<i::ModuleRequest> self = Utils::OpenHandle(this);
2277   i::Isolate* isolate = self->GetIsolate();
2278   return ToApiHandle<FixedArray>(i::handle(self->import_assertions(), isolate));
2279 }
2280 
GetStatus() const2281 Module::Status Module::GetStatus() const {
2282   i::Handle<i::Module> self = Utils::OpenHandle(this);
2283   switch (self->status()) {
2284     case i::Module::kUnlinked:
2285     case i::Module::kPreLinking:
2286       return kUninstantiated;
2287     case i::Module::kLinking:
2288       return kInstantiating;
2289     case i::Module::kLinked:
2290       return kInstantiated;
2291     case i::Module::kEvaluating:
2292     case i::Module::kEvaluatingAsync:
2293       return kEvaluating;
2294     case i::Module::kEvaluated:
2295       return kEvaluated;
2296     case i::Module::kErrored:
2297       return kErrored;
2298   }
2299   UNREACHABLE();
2300 }
2301 
GetException() const2302 Local<Value> Module::GetException() const {
2303   Utils::ApiCheck(GetStatus() == kErrored, "v8::Module::GetException",
2304                   "Module status must be kErrored");
2305   i::Handle<i::Module> self = Utils::OpenHandle(this);
2306   i::Isolate* isolate = self->GetIsolate();
2307   ENTER_V8_NO_SCRIPT_NO_EXCEPTION(isolate);
2308   return ToApiHandle<Value>(i::handle(self->GetException(), isolate));
2309 }
2310 
GetModuleRequests() const2311 Local<FixedArray> Module::GetModuleRequests() const {
2312   i::Handle<i::Module> self = Utils::OpenHandle(this);
2313   if (self->IsSyntheticModule()) {
2314     // Synthetic modules are leaf nodes in the module graph. They have no
2315     // ModuleRequests.
2316     return ToApiHandle<FixedArray>(
2317         self->GetReadOnlyRoots().empty_fixed_array_handle());
2318   } else {
2319     i::Isolate* isolate = self->GetIsolate();
2320     i::Handle<i::FixedArray> module_requests(
2321         i::Handle<i::SourceTextModule>::cast(self)->info().module_requests(),
2322         isolate);
2323     return ToApiHandle<FixedArray>(module_requests);
2324   }
2325 }
2326 
SourceOffsetToLocation(int offset) const2327 Location Module::SourceOffsetToLocation(int offset) const {
2328   i::Handle<i::Module> self = Utils::OpenHandle(this);
2329   i::Isolate* isolate = self->GetIsolate();
2330   ENTER_V8_NO_SCRIPT_NO_EXCEPTION(isolate);
2331   i::HandleScope scope(isolate);
2332   Utils::ApiCheck(
2333       self->IsSourceTextModule(), "v8::Module::SourceOffsetToLocation",
2334       "v8::Module::SourceOffsetToLocation must be used on an SourceTextModule");
2335   i::Handle<i::Script> script(
2336       i::Handle<i::SourceTextModule>::cast(self)->GetScript(), isolate);
2337   i::Script::PositionInfo info;
2338   i::Script::GetPositionInfo(script, offset, &info, i::Script::WITH_OFFSET);
2339   return v8::Location(info.line, info.column);
2340 }
2341 
GetModuleNamespace()2342 Local<Value> Module::GetModuleNamespace() {
2343   Utils::ApiCheck(
2344       GetStatus() >= kInstantiated, "v8::Module::GetModuleNamespace",
2345       "v8::Module::GetModuleNamespace must be used on an instantiated module");
2346   i::Handle<i::Module> self = Utils::OpenHandle(this);
2347   auto isolate = self->GetIsolate();
2348   ASSERT_NO_SCRIPT_NO_EXCEPTION(isolate);
2349   i::Handle<i::JSModuleNamespace> module_namespace =
2350       i::Module::GetModuleNamespace(isolate, self);
2351   return ToApiHandle<Value>(module_namespace);
2352 }
2353 
GetUnboundModuleScript()2354 Local<UnboundModuleScript> Module::GetUnboundModuleScript() {
2355   i::Handle<i::Module> self = Utils::OpenHandle(this);
2356   Utils::ApiCheck(
2357       self->IsSourceTextModule(), "v8::Module::GetUnboundModuleScript",
2358       "v8::Module::GetUnboundModuleScript must be used on an SourceTextModule");
2359   auto isolate = self->GetIsolate();
2360   ASSERT_NO_SCRIPT_NO_EXCEPTION(isolate);
2361   return ToApiHandle<UnboundModuleScript>(i::handle(
2362       i::Handle<i::SourceTextModule>::cast(self)->GetSharedFunctionInfo(),
2363       isolate));
2364 }
2365 
ScriptId() const2366 int Module::ScriptId() const {
2367   i::Module self = *Utils::OpenHandle(this);
2368   Utils::ApiCheck(self.IsSourceTextModule(), "v8::Module::ScriptId",
2369                   "v8::Module::ScriptId must be used on an SourceTextModule");
2370   ASSERT_NO_SCRIPT_NO_EXCEPTION(self.GetIsolate());
2371   return i::SourceTextModule::cast(self).GetScript().id();
2372 }
2373 
IsGraphAsync() const2374 bool Module::IsGraphAsync() const {
2375   Utils::ApiCheck(
2376       GetStatus() >= kInstantiated, "v8::Module::IsGraphAsync",
2377       "v8::Module::IsGraphAsync must be used on an instantiated module");
2378   i::Module self = *Utils::OpenHandle(this);
2379   auto isolate = self.GetIsolate();
2380   ASSERT_NO_SCRIPT_NO_EXCEPTION(isolate);
2381   return self.IsGraphAsync(isolate);
2382 }
2383 
IsSourceTextModule() const2384 bool Module::IsSourceTextModule() const {
2385   return Utils::OpenHandle(this)->IsSourceTextModule();
2386 }
2387 
IsSyntheticModule() const2388 bool Module::IsSyntheticModule() const {
2389   return Utils::OpenHandle(this)->IsSyntheticModule();
2390 }
2391 
GetIdentityHash() const2392 int Module::GetIdentityHash() const { return Utils::OpenHandle(this)->hash(); }
2393 
InstantiateModule(Local<Context> context,Module::ResolveModuleCallback callback)2394 Maybe<bool> Module::InstantiateModule(Local<Context> context,
2395                                       Module::ResolveModuleCallback callback) {
2396   auto isolate = reinterpret_cast<i::Isolate*>(context->GetIsolate());
2397   ENTER_V8(isolate, context, Module, InstantiateModule, Nothing<bool>(),
2398            i::HandleScope);
2399   has_pending_exception = !i::Module::Instantiate(
2400       isolate, Utils::OpenHandle(this), context, callback, nullptr);
2401   RETURN_ON_FAILED_EXECUTION_PRIMITIVE(bool);
2402   return Just(true);
2403 }
2404 
Evaluate(Local<Context> context)2405 MaybeLocal<Value> Module::Evaluate(Local<Context> context) {
2406   auto isolate = reinterpret_cast<i::Isolate*>(context->GetIsolate());
2407   TRACE_EVENT_CALL_STATS_SCOPED(isolate, "v8", "V8.Execute");
2408   ENTER_V8(isolate, context, Module, Evaluate, MaybeLocal<Value>(),
2409            InternalEscapableScope);
2410   i::TimerEventScope<i::TimerEventExecute> timer_scope(isolate);
2411   i::NestedTimedHistogramScope execute_timer(isolate->counters()->execute(),
2412                                              isolate);
2413   i::AggregatingHistogramTimerScope timer(isolate->counters()->compile_lazy());
2414 
2415   i::Handle<i::Module> self = Utils::OpenHandle(this);
2416   Utils::ApiCheck(self->status() >= i::Module::kLinked, "Module::Evaluate",
2417                   "Expected instantiated module");
2418 
2419   Local<Value> result;
2420   has_pending_exception = !ToLocal(i::Module::Evaluate(isolate, self), &result);
2421   RETURN_ON_FAILED_EXECUTION(Value);
2422   RETURN_ESCAPED(result);
2423 }
2424 
CreateSyntheticModule(Isolate * isolate,Local<String> module_name,const std::vector<Local<v8::String>> & export_names,v8::Module::SyntheticModuleEvaluationSteps evaluation_steps)2425 Local<Module> Module::CreateSyntheticModule(
2426     Isolate* isolate, Local<String> module_name,
2427     const std::vector<Local<v8::String>>& export_names,
2428     v8::Module::SyntheticModuleEvaluationSteps evaluation_steps) {
2429   auto i_isolate = reinterpret_cast<i::Isolate*>(isolate);
2430   ENTER_V8_NO_SCRIPT_NO_EXCEPTION(i_isolate);
2431   i::Handle<i::String> i_module_name = Utils::OpenHandle(*module_name);
2432   i::Handle<i::FixedArray> i_export_names = i_isolate->factory()->NewFixedArray(
2433       static_cast<int>(export_names.size()));
2434   for (int i = 0; i < i_export_names->length(); ++i) {
2435     i::Handle<i::String> str = i_isolate->factory()->InternalizeString(
2436         Utils::OpenHandle(*export_names[i]));
2437     i_export_names->set(i, *str);
2438   }
2439   return v8::Utils::ToLocal(
2440       i::Handle<i::Module>(i_isolate->factory()->NewSyntheticModule(
2441           i_module_name, i_export_names, evaluation_steps)));
2442 }
2443 
SetSyntheticModuleExport(Isolate * isolate,Local<String> export_name,Local<v8::Value> export_value)2444 Maybe<bool> Module::SetSyntheticModuleExport(Isolate* isolate,
2445                                              Local<String> export_name,
2446                                              Local<v8::Value> export_value) {
2447   auto i_isolate = reinterpret_cast<i::Isolate*>(isolate);
2448   i::Handle<i::String> i_export_name = Utils::OpenHandle(*export_name);
2449   i::Handle<i::Object> i_export_value = Utils::OpenHandle(*export_value);
2450   i::Handle<i::Module> self = Utils::OpenHandle(this);
2451   Utils::ApiCheck(self->IsSyntheticModule(),
2452                   "v8::Module::SyntheticModuleSetExport",
2453                   "v8::Module::SyntheticModuleSetExport must only be called on "
2454                   "a SyntheticModule");
2455   ENTER_V8_NO_SCRIPT(i_isolate, isolate->GetCurrentContext(), Module,
2456                      SetSyntheticModuleExport, Nothing<bool>(), i::HandleScope);
2457   has_pending_exception =
2458       i::SyntheticModule::SetExport(i_isolate,
2459                                     i::Handle<i::SyntheticModule>::cast(self),
2460                                     i_export_name, i_export_value)
2461           .IsNothing();
2462   RETURN_ON_FAILED_EXECUTION_PRIMITIVE(bool);
2463   return Just(true);
2464 }
2465 
2466 namespace {
2467 
GetScriptDetails(i::Isolate * isolate,Local<Value> resource_name,int resource_line_offset,int resource_column_offset,Local<Value> source_map_url,Local<Data> host_defined_options,ScriptOriginOptions origin_options)2468 i::ScriptDetails GetScriptDetails(
2469     i::Isolate* isolate, Local<Value> resource_name, int resource_line_offset,
2470     int resource_column_offset, Local<Value> source_map_url,
2471     Local<Data> host_defined_options, ScriptOriginOptions origin_options) {
2472   i::ScriptDetails script_details(Utils::OpenHandle(*(resource_name), true),
2473                                   origin_options);
2474   script_details.line_offset = resource_line_offset;
2475   script_details.column_offset = resource_column_offset;
2476   script_details.host_defined_options =
2477       host_defined_options.IsEmpty()
2478           ? isolate->factory()->empty_fixed_array()
2479           : Utils::OpenHandle(*(host_defined_options));
2480   if (!source_map_url.IsEmpty()) {
2481     script_details.source_map_url = Utils::OpenHandle(*(source_map_url));
2482   }
2483   return script_details;
2484 }
2485 
2486 }  // namespace
2487 
CompileUnboundInternal(Isolate * v8_isolate,Source * source,CompileOptions options,NoCacheReason no_cache_reason)2488 MaybeLocal<UnboundScript> ScriptCompiler::CompileUnboundInternal(
2489     Isolate* v8_isolate, Source* source, CompileOptions options,
2490     NoCacheReason no_cache_reason) {
2491   auto isolate = reinterpret_cast<i::Isolate*>(v8_isolate);
2492   TRACE_EVENT_CALL_STATS_SCOPED(isolate, "v8", "V8.ScriptCompiler");
2493   ENTER_V8_NO_SCRIPT(isolate, v8_isolate->GetCurrentContext(), ScriptCompiler,
2494                      CompileUnbound, MaybeLocal<UnboundScript>(),
2495                      InternalEscapableScope);
2496 
2497   i::Handle<i::String> str = Utils::OpenHandle(*(source->source_string));
2498 
2499   i::Handle<i::SharedFunctionInfo> result;
2500   TRACE_EVENT0(TRACE_DISABLED_BY_DEFAULT("v8.compile"), "V8.CompileScript");
2501   i::ScriptDetails script_details = GetScriptDetails(
2502       isolate, source->resource_name, source->resource_line_offset,
2503       source->resource_column_offset, source->source_map_url,
2504       source->host_defined_options, source->resource_options);
2505 
2506   i::MaybeHandle<i::SharedFunctionInfo> maybe_function_info;
2507   if (options == kConsumeCodeCache) {
2508     if (source->consume_cache_task) {
2509       // Take ownership of the internal deserialization task and clear it off
2510       // the consume task on the source.
2511       DCHECK_NOT_NULL(source->consume_cache_task->impl_);
2512       std::unique_ptr<i::BackgroundDeserializeTask> deserialize_task =
2513           std::move(source->consume_cache_task->impl_);
2514       maybe_function_info =
2515           i::Compiler::GetSharedFunctionInfoForScriptWithDeserializeTask(
2516               isolate, str, script_details, deserialize_task.get(), options,
2517               no_cache_reason, i::NOT_NATIVES_CODE);
2518       source->cached_data->rejected = deserialize_task->rejected();
2519     } else {
2520       DCHECK(source->cached_data);
2521       // AlignedCachedData takes care of pointer-aligning the data.
2522       auto cached_data = std::make_unique<i::AlignedCachedData>(
2523           source->cached_data->data, source->cached_data->length);
2524       maybe_function_info =
2525           i::Compiler::GetSharedFunctionInfoForScriptWithCachedData(
2526               isolate, str, script_details, cached_data.get(), options,
2527               no_cache_reason, i::NOT_NATIVES_CODE);
2528       source->cached_data->rejected = cached_data->rejected();
2529     }
2530   } else {
2531     // Compile without any cache.
2532     maybe_function_info = i::Compiler::GetSharedFunctionInfoForScript(
2533         isolate, str, script_details, options, no_cache_reason,
2534         i::NOT_NATIVES_CODE);
2535   }
2536 
2537   has_pending_exception = !maybe_function_info.ToHandle(&result);
2538   RETURN_ON_FAILED_EXECUTION(UnboundScript);
2539   RETURN_ESCAPED(ToApiHandle<UnboundScript>(result));
2540 }
2541 
CompileUnboundScript(Isolate * v8_isolate,Source * source,CompileOptions options,NoCacheReason no_cache_reason)2542 MaybeLocal<UnboundScript> ScriptCompiler::CompileUnboundScript(
2543     Isolate* v8_isolate, Source* source, CompileOptions options,
2544     NoCacheReason no_cache_reason) {
2545   Utils::ApiCheck(
2546       !source->GetResourceOptions().IsModule(),
2547       "v8::ScriptCompiler::CompileUnboundScript",
2548       "v8::ScriptCompiler::CompileModule must be used to compile modules");
2549   return CompileUnboundInternal(v8_isolate, source, options, no_cache_reason);
2550 }
2551 
Compile(Local<Context> context,Source * source,CompileOptions options,NoCacheReason no_cache_reason)2552 MaybeLocal<Script> ScriptCompiler::Compile(Local<Context> context,
2553                                            Source* source,
2554                                            CompileOptions options,
2555                                            NoCacheReason no_cache_reason) {
2556   Utils::ApiCheck(
2557       !source->GetResourceOptions().IsModule(), "v8::ScriptCompiler::Compile",
2558       "v8::ScriptCompiler::CompileModule must be used to compile modules");
2559   auto isolate = context->GetIsolate();
2560   MaybeLocal<UnboundScript> maybe =
2561       CompileUnboundInternal(isolate, source, options, no_cache_reason);
2562   Local<UnboundScript> result;
2563   if (!maybe.ToLocal(&result)) return MaybeLocal<Script>();
2564   v8::Context::Scope scope(context);
2565   return result->BindToCurrentContext();
2566 }
2567 
CompileModule(Isolate * isolate,Source * source,CompileOptions options,NoCacheReason no_cache_reason)2568 MaybeLocal<Module> ScriptCompiler::CompileModule(
2569     Isolate* isolate, Source* source, CompileOptions options,
2570     NoCacheReason no_cache_reason) {
2571   Utils::ApiCheck(options == kNoCompileOptions || options == kConsumeCodeCache,
2572                   "v8::ScriptCompiler::CompileModule",
2573                   "Invalid CompileOptions");
2574   Utils::ApiCheck(source->GetResourceOptions().IsModule(),
2575                   "v8::ScriptCompiler::CompileModule",
2576                   "Invalid ScriptOrigin: is_module must be true");
2577   MaybeLocal<UnboundScript> maybe =
2578       CompileUnboundInternal(isolate, source, options, no_cache_reason);
2579   Local<UnboundScript> unbound;
2580   if (!maybe.ToLocal(&unbound)) return MaybeLocal<Module>();
2581   i::Handle<i::SharedFunctionInfo> shared = Utils::OpenHandle(*unbound);
2582   i::Isolate* i_isolate = reinterpret_cast<i::Isolate*>(isolate);
2583   return ToApiHandle<Module>(i_isolate->factory()->NewSourceTextModule(shared));
2584 }
2585 
2586 // static
CompileFunction(Local<Context> context,Source * source,size_t arguments_count,Local<String> arguments[],size_t context_extension_count,Local<Object> context_extensions[],CompileOptions options,NoCacheReason no_cache_reason)2587 V8_WARN_UNUSED_RESULT MaybeLocal<Function> ScriptCompiler::CompileFunction(
2588     Local<Context> context, Source* source, size_t arguments_count,
2589     Local<String> arguments[], size_t context_extension_count,
2590     Local<Object> context_extensions[], CompileOptions options,
2591     NoCacheReason no_cache_reason) {
2592   return CompileFunctionInternal(context, source, arguments_count, arguments,
2593                                  context_extension_count, context_extensions,
2594                                  options, no_cache_reason, nullptr);
2595 }
2596 
2597 // static
CompileFunctionInContext(Local<Context> context,Source * source,size_t arguments_count,Local<String> arguments[],size_t context_extension_count,Local<Object> context_extensions[],CompileOptions options,NoCacheReason no_cache_reason,Local<ScriptOrModule> * script_or_module_out)2598 MaybeLocal<Function> ScriptCompiler::CompileFunctionInContext(
2599     Local<Context> context, Source* source, size_t arguments_count,
2600     Local<String> arguments[], size_t context_extension_count,
2601     Local<Object> context_extensions[], CompileOptions options,
2602     NoCacheReason no_cache_reason,
2603     Local<ScriptOrModule>* script_or_module_out) {
2604   return CompileFunctionInternal(
2605       context, source, arguments_count, arguments, context_extension_count,
2606       context_extensions, options, no_cache_reason, script_or_module_out);
2607 }
2608 
CompileFunctionInternal(Local<Context> v8_context,Source * source,size_t arguments_count,Local<String> arguments[],size_t context_extension_count,Local<Object> context_extensions[],CompileOptions options,NoCacheReason no_cache_reason,Local<ScriptOrModule> * script_or_module_out)2609 MaybeLocal<Function> ScriptCompiler::CompileFunctionInternal(
2610     Local<Context> v8_context, Source* source, size_t arguments_count,
2611     Local<String> arguments[], size_t context_extension_count,
2612     Local<Object> context_extensions[], CompileOptions options,
2613     NoCacheReason no_cache_reason,
2614     Local<ScriptOrModule>* script_or_module_out) {
2615   Local<Function> result;
2616 
2617   {
2618     PREPARE_FOR_EXECUTION(v8_context, ScriptCompiler, CompileFunction,
2619                           Function);
2620     TRACE_EVENT_CALL_STATS_SCOPED(isolate, "v8", "V8.ScriptCompiler");
2621 
2622     DCHECK(options == CompileOptions::kConsumeCodeCache ||
2623            options == CompileOptions::kEagerCompile ||
2624            options == CompileOptions::kNoCompileOptions);
2625 
2626     i::Handle<i::Context> context = Utils::OpenHandle(*v8_context);
2627 
2628     DCHECK(context->IsNativeContext());
2629 
2630     i::Handle<i::FixedArray> arguments_list =
2631         isolate->factory()->NewFixedArray(static_cast<int>(arguments_count));
2632     for (int i = 0; i < static_cast<int>(arguments_count); i++) {
2633       i::Handle<i::String> argument = Utils::OpenHandle(*arguments[i]);
2634       if (!i::String::IsIdentifier(isolate, argument)) return Local<Function>();
2635       arguments_list->set(i, *argument);
2636     }
2637 
2638     for (size_t i = 0; i < context_extension_count; ++i) {
2639       i::Handle<i::JSReceiver> extension =
2640           Utils::OpenHandle(*context_extensions[i]);
2641       if (!extension->IsJSObject()) return Local<Function>();
2642       context = isolate->factory()->NewWithContext(
2643           context,
2644           i::ScopeInfo::CreateForWithScope(
2645               isolate,
2646               context->IsNativeContext()
2647                   ? i::Handle<i::ScopeInfo>::null()
2648                   : i::Handle<i::ScopeInfo>(context->scope_info(), isolate)),
2649           extension);
2650     }
2651 
2652     i::ScriptDetails script_details = GetScriptDetails(
2653         isolate, source->resource_name, source->resource_line_offset,
2654         source->resource_column_offset, source->source_map_url,
2655         source->host_defined_options, source->resource_options);
2656 
2657     std::unique_ptr<i::AlignedCachedData> cached_data;
2658     if (options == kConsumeCodeCache) {
2659       DCHECK(source->cached_data);
2660       // ScriptData takes care of pointer-aligning the data.
2661       cached_data.reset(new i::AlignedCachedData(source->cached_data->data,
2662                                                  source->cached_data->length));
2663     }
2664 
2665     i::Handle<i::JSFunction> scoped_result;
2666     has_pending_exception =
2667         !i::Compiler::GetWrappedFunction(
2668              Utils::OpenHandle(*source->source_string), arguments_list, context,
2669              script_details, cached_data.get(), options, no_cache_reason)
2670              .ToHandle(&scoped_result);
2671     if (options == kConsumeCodeCache) {
2672       source->cached_data->rejected = cached_data->rejected();
2673     }
2674     RETURN_ON_FAILED_EXECUTION(Function);
2675     result = handle_scope.Escape(Utils::CallableToLocal(scoped_result));
2676   }
2677   // TODO(cbruni): remove script_or_module_out paramater
2678   if (script_or_module_out != nullptr) {
2679     i::Handle<i::JSFunction> function =
2680         i::Handle<i::JSFunction>::cast(Utils::OpenHandle(*result));
2681     i::Isolate* isolate = function->GetIsolate();
2682     i::Handle<i::SharedFunctionInfo> shared(function->shared(), isolate);
2683     i::Handle<i::Script> script(i::Script::cast(shared->script()), isolate);
2684     // TODO(cbruni, v8:12302): Avoid creating tempory ScriptOrModule objects.
2685     auto script_or_module = i::Handle<i::ScriptOrModule>::cast(
2686         isolate->factory()->NewStruct(i::SCRIPT_OR_MODULE_TYPE));
2687     script_or_module->set_resource_name(script->name());
2688     script_or_module->set_host_defined_options(script->host_defined_options());
2689 #ifdef V8_SCRIPTORMODULE_LEGACY_LIFETIME
2690     i::Handle<i::ArrayList> list =
2691         i::handle(script->script_or_modules(), isolate);
2692     list = i::ArrayList::Add(isolate, list, script_or_module);
2693     script->set_script_or_modules(*list);
2694 #endif  // V8_SCRIPTORMODULE_LEGACY_LIFETIME
2695     *script_or_module_out = v8::Utils::ToLocal(script_or_module);
2696   }
2697   return result;
2698 }
2699 
Run()2700 void ScriptCompiler::ScriptStreamingTask::Run() { data_->task->Run(); }
2701 
StartStreaming(Isolate * v8_isolate,StreamedSource * source,v8::ScriptType type)2702 ScriptCompiler::ScriptStreamingTask* ScriptCompiler::StartStreaming(
2703     Isolate* v8_isolate, StreamedSource* source, v8::ScriptType type) {
2704   if (!i::FLAG_script_streaming) return nullptr;
2705   i::Isolate* isolate = reinterpret_cast<i::Isolate*>(v8_isolate);
2706   ASSERT_NO_SCRIPT_NO_EXCEPTION(isolate);
2707   i::ScriptStreamingData* data = source->impl();
2708   std::unique_ptr<i::BackgroundCompileTask> task =
2709       std::make_unique<i::BackgroundCompileTask>(data, isolate, type);
2710   data->task = std::move(task);
2711   return new ScriptCompiler::ScriptStreamingTask(data);
2712 }
2713 
ConsumeCodeCacheTask(std::unique_ptr<i::BackgroundDeserializeTask> impl)2714 ScriptCompiler::ConsumeCodeCacheTask::ConsumeCodeCacheTask(
2715     std::unique_ptr<i::BackgroundDeserializeTask> impl)
2716     : impl_(std::move(impl)) {}
2717 
2718 ScriptCompiler::ConsumeCodeCacheTask::~ConsumeCodeCacheTask() = default;
2719 
Run()2720 void ScriptCompiler::ConsumeCodeCacheTask::Run() { impl_->Run(); }
2721 
StartConsumingCodeCache(Isolate * v8_isolate,std::unique_ptr<CachedData> cached_data)2722 ScriptCompiler::ConsumeCodeCacheTask* ScriptCompiler::StartConsumingCodeCache(
2723     Isolate* v8_isolate, std::unique_ptr<CachedData> cached_data) {
2724   if (!i::FLAG_concurrent_cache_deserialization) return nullptr;
2725   i::Isolate* isolate = reinterpret_cast<i::Isolate*>(v8_isolate);
2726   ASSERT_NO_SCRIPT_NO_EXCEPTION(isolate);
2727   return new ScriptCompiler::ConsumeCodeCacheTask(
2728       std::make_unique<i::BackgroundDeserializeTask>(isolate,
2729                                                      std::move(cached_data)));
2730 }
2731 
2732 namespace {
CompileStreamedSource(i::Isolate * isolate,ScriptCompiler::StreamedSource * v8_source,Local<String> full_source_string,const ScriptOrigin & origin)2733 i::MaybeHandle<i::SharedFunctionInfo> CompileStreamedSource(
2734     i::Isolate* isolate, ScriptCompiler::StreamedSource* v8_source,
2735     Local<String> full_source_string, const ScriptOrigin& origin) {
2736   i::Handle<i::String> str = Utils::OpenHandle(*(full_source_string));
2737   i::ScriptDetails script_details =
2738       GetScriptDetails(isolate, origin.ResourceName(), origin.LineOffset(),
2739                        origin.ColumnOffset(), origin.SourceMapUrl(),
2740                        origin.GetHostDefinedOptions(), origin.Options());
2741   i::ScriptStreamingData* data = v8_source->impl();
2742   return i::Compiler::GetSharedFunctionInfoForStreamedScript(
2743       isolate, str, script_details, data);
2744 }
2745 
2746 }  // namespace
2747 
Compile(Local<Context> context,StreamedSource * v8_source,Local<String> full_source_string,const ScriptOrigin & origin)2748 MaybeLocal<Script> ScriptCompiler::Compile(Local<Context> context,
2749                                            StreamedSource* v8_source,
2750                                            Local<String> full_source_string,
2751                                            const ScriptOrigin& origin) {
2752   PREPARE_FOR_EXECUTION(context, ScriptCompiler, Compile, Script);
2753   TRACE_EVENT_CALL_STATS_SCOPED(isolate, "v8", "V8.ScriptCompiler");
2754   TRACE_EVENT0(TRACE_DISABLED_BY_DEFAULT("v8.compile"),
2755                "V8.CompileStreamedScript");
2756   i::Handle<i::SharedFunctionInfo> sfi;
2757   i::MaybeHandle<i::SharedFunctionInfo> maybe_sfi =
2758       CompileStreamedSource(isolate, v8_source, full_source_string, origin);
2759   has_pending_exception = !maybe_sfi.ToHandle(&sfi);
2760   if (has_pending_exception) isolate->ReportPendingMessages();
2761   RETURN_ON_FAILED_EXECUTION(Script);
2762   Local<UnboundScript> generic = ToApiHandle<UnboundScript>(sfi);
2763   if (generic.IsEmpty()) return Local<Script>();
2764   Local<Script> bound = generic->BindToCurrentContext();
2765   if (bound.IsEmpty()) return Local<Script>();
2766   RETURN_ESCAPED(bound);
2767 }
2768 
CompileModule(Local<Context> context,StreamedSource * v8_source,Local<String> full_source_string,const ScriptOrigin & origin)2769 MaybeLocal<Module> ScriptCompiler::CompileModule(
2770     Local<Context> context, StreamedSource* v8_source,
2771     Local<String> full_source_string, const ScriptOrigin& origin) {
2772   PREPARE_FOR_EXECUTION(context, ScriptCompiler, Compile, Module);
2773   TRACE_EVENT_CALL_STATS_SCOPED(isolate, "v8", "V8.ScriptCompiler");
2774   TRACE_EVENT0(TRACE_DISABLED_BY_DEFAULT("v8.compile"),
2775                "V8.CompileStreamedModule");
2776   i::Handle<i::SharedFunctionInfo> sfi;
2777   i::MaybeHandle<i::SharedFunctionInfo> maybe_sfi =
2778       CompileStreamedSource(isolate, v8_source, full_source_string, origin);
2779   has_pending_exception = !maybe_sfi.ToHandle(&sfi);
2780   if (has_pending_exception) isolate->ReportPendingMessages();
2781   RETURN_ON_FAILED_EXECUTION(Module);
2782   i::Isolate* i_isolate = reinterpret_cast<i::Isolate*>(isolate);
2783   RETURN_ESCAPED(
2784       ToApiHandle<Module>(i_isolate->factory()->NewSourceTextModule(sfi)));
2785 }
2786 
CachedDataVersionTag()2787 uint32_t ScriptCompiler::CachedDataVersionTag() {
2788   return static_cast<uint32_t>(base::hash_combine(
2789       internal::Version::Hash(), internal::FlagList::Hash(),
2790       static_cast<uint32_t>(internal::CpuFeatures::SupportedFeatures())));
2791 }
2792 
CreateCodeCache(Local<UnboundScript> unbound_script)2793 ScriptCompiler::CachedData* ScriptCompiler::CreateCodeCache(
2794     Local<UnboundScript> unbound_script) {
2795   i::Handle<i::SharedFunctionInfo> shared =
2796       i::Handle<i::SharedFunctionInfo>::cast(
2797           Utils::OpenHandle(*unbound_script));
2798   ASSERT_NO_SCRIPT_NO_EXCEPTION(shared->GetIsolate());
2799   DCHECK(shared->is_toplevel());
2800   return i::CodeSerializer::Serialize(shared);
2801 }
2802 
2803 // static
CreateCodeCache(Local<UnboundModuleScript> unbound_module_script)2804 ScriptCompiler::CachedData* ScriptCompiler::CreateCodeCache(
2805     Local<UnboundModuleScript> unbound_module_script) {
2806   i::Handle<i::SharedFunctionInfo> shared =
2807       i::Handle<i::SharedFunctionInfo>::cast(
2808           Utils::OpenHandle(*unbound_module_script));
2809   ASSERT_NO_SCRIPT_NO_EXCEPTION(shared->GetIsolate());
2810   DCHECK(shared->is_toplevel());
2811   return i::CodeSerializer::Serialize(shared);
2812 }
2813 
CreateCodeCacheForFunction(Local<Function> function)2814 ScriptCompiler::CachedData* ScriptCompiler::CreateCodeCacheForFunction(
2815     Local<Function> function) {
2816   auto js_function =
2817       i::Handle<i::JSFunction>::cast(Utils::OpenHandle(*function));
2818   i::Handle<i::SharedFunctionInfo> shared(js_function->shared(),
2819                                           js_function->GetIsolate());
2820   ASSERT_NO_SCRIPT_NO_EXCEPTION(shared->GetIsolate());
2821   Utils::ApiCheck(shared->is_wrapped(),
2822                   "v8::ScriptCompiler::CreateCodeCacheForFunction",
2823                   "Expected SharedFunctionInfo with wrapped source code.");
2824   return i::CodeSerializer::Serialize(shared);
2825 }
2826 
Compile(Local<Context> context,Local<String> source,ScriptOrigin * origin)2827 MaybeLocal<Script> Script::Compile(Local<Context> context, Local<String> source,
2828                                    ScriptOrigin* origin) {
2829   if (origin) {
2830     ScriptCompiler::Source script_source(source, *origin);
2831     return ScriptCompiler::Compile(context, &script_source);
2832   }
2833   ScriptCompiler::Source script_source(source);
2834   return ScriptCompiler::Compile(context, &script_source);
2835 }
2836 
2837 // --- E x c e p t i o n s ---
2838 
TryCatch(v8::Isolate * isolate)2839 v8::TryCatch::TryCatch(v8::Isolate* isolate)
2840     : isolate_(reinterpret_cast<i::Isolate*>(isolate)),
2841       next_(isolate_->try_catch_handler()),
2842       is_verbose_(false),
2843       can_continue_(true),
2844       capture_message_(true),
2845       rethrow_(false),
2846       has_terminated_(false) {
2847   ResetInternal();
2848   // Special handling for simulators which have a separate JS stack.
2849   js_stack_comparable_address_ = static_cast<internal::Address>(
2850       i::SimulatorStack::RegisterJSStackComparableAddress(isolate_));
2851   isolate_->RegisterTryCatchHandler(this);
2852 }
2853 
~TryCatch()2854 v8::TryCatch::~TryCatch() {
2855   if (rethrow_) {
2856     v8::Isolate* isolate = reinterpret_cast<Isolate*>(isolate_);
2857     v8::HandleScope scope(isolate);
2858     v8::Local<v8::Value> exc = v8::Local<v8::Value>::New(isolate, Exception());
2859     if (HasCaught() && capture_message_) {
2860       // If an exception was caught and rethrow_ is indicated, the saved
2861       // message, script, and location need to be restored to Isolate TLS
2862       // for reuse.  capture_message_ needs to be disabled so that Throw()
2863       // does not create a new message.
2864       isolate_->thread_local_top()->rethrowing_message_ = true;
2865       isolate_->RestorePendingMessageFromTryCatch(this);
2866     }
2867     isolate_->UnregisterTryCatchHandler(this);
2868     i::SimulatorStack::UnregisterJSStackComparableAddress(isolate_);
2869     reinterpret_cast<Isolate*>(isolate_)->ThrowException(exc);
2870     DCHECK(!isolate_->thread_local_top()->rethrowing_message_);
2871   } else {
2872     if (HasCaught() && isolate_->has_scheduled_exception()) {
2873       // If an exception was caught but is still scheduled because no API call
2874       // promoted it, then it is canceled to prevent it from being propagated.
2875       // Note that this will not cancel termination exceptions.
2876       isolate_->CancelScheduledExceptionFromTryCatch(this);
2877     }
2878     isolate_->UnregisterTryCatchHandler(this);
2879     i::SimulatorStack::UnregisterJSStackComparableAddress(isolate_);
2880   }
2881 }
2882 
operator new(size_t)2883 void* v8::TryCatch::operator new(size_t) { base::OS::Abort(); }
operator new[](size_t)2884 void* v8::TryCatch::operator new[](size_t) { base::OS::Abort(); }
operator delete(void *,size_t)2885 void v8::TryCatch::operator delete(void*, size_t) { base::OS::Abort(); }
operator delete[](void *,size_t)2886 void v8::TryCatch::operator delete[](void*, size_t) { base::OS::Abort(); }
2887 
HasCaught() const2888 bool v8::TryCatch::HasCaught() const {
2889   return !i::Object(reinterpret_cast<i::Address>(exception_))
2890               .IsTheHole(isolate_);
2891 }
2892 
CanContinue() const2893 bool v8::TryCatch::CanContinue() const { return can_continue_; }
2894 
HasTerminated() const2895 bool v8::TryCatch::HasTerminated() const { return has_terminated_; }
2896 
ReThrow()2897 v8::Local<v8::Value> v8::TryCatch::ReThrow() {
2898   if (!HasCaught()) return v8::Local<v8::Value>();
2899   rethrow_ = true;
2900   return v8::Undefined(reinterpret_cast<v8::Isolate*>(isolate_));
2901 }
2902 
Exception() const2903 v8::Local<Value> v8::TryCatch::Exception() const {
2904   if (HasCaught()) {
2905     // Check for out of memory exception.
2906     i::Object exception(reinterpret_cast<i::Address>(exception_));
2907     return v8::Utils::ToLocal(i::Handle<i::Object>(exception, isolate_));
2908   } else {
2909     return v8::Local<Value>();
2910   }
2911 }
2912 
StackTrace(Local<Context> context,Local<Value> exception)2913 MaybeLocal<Value> v8::TryCatch::StackTrace(Local<Context> context,
2914                                            Local<Value> exception) {
2915   i::Handle<i::Object> i_exception = Utils::OpenHandle(*exception);
2916   if (!i_exception->IsJSObject()) return v8::Local<Value>();
2917   PREPARE_FOR_EXECUTION(context, TryCatch, StackTrace, Value);
2918   auto obj = i::Handle<i::JSObject>::cast(i_exception);
2919   i::Handle<i::String> name = isolate->factory()->stack_string();
2920   Maybe<bool> maybe = i::JSReceiver::HasProperty(isolate, obj, name);
2921   has_pending_exception = maybe.IsNothing();
2922   RETURN_ON_FAILED_EXECUTION(Value);
2923   if (!maybe.FromJust()) return v8::Local<Value>();
2924   Local<Value> result;
2925   has_pending_exception =
2926       !ToLocal<Value>(i::JSReceiver::GetProperty(isolate, obj, name), &result);
2927   RETURN_ON_FAILED_EXECUTION(Value);
2928   RETURN_ESCAPED(result);
2929 }
2930 
StackTrace(Local<Context> context) const2931 MaybeLocal<Value> v8::TryCatch::StackTrace(Local<Context> context) const {
2932   if (!HasCaught()) return v8::Local<Value>();
2933   return StackTrace(context, Exception());
2934 }
2935 
Message() const2936 v8::Local<v8::Message> v8::TryCatch::Message() const {
2937   i::Object message(reinterpret_cast<i::Address>(message_obj_));
2938   DCHECK(message.IsJSMessageObject() || message.IsTheHole(isolate_));
2939   if (HasCaught() && !message.IsTheHole(isolate_)) {
2940     return v8::Utils::MessageToLocal(i::Handle<i::Object>(message, isolate_));
2941   } else {
2942     return v8::Local<v8::Message>();
2943   }
2944 }
2945 
Reset()2946 void v8::TryCatch::Reset() {
2947   if (!rethrow_ && HasCaught() && isolate_->has_scheduled_exception()) {
2948     // If an exception was caught but is still scheduled because no API call
2949     // promoted it, then it is canceled to prevent it from being propagated.
2950     // Note that this will not cancel termination exceptions.
2951     isolate_->CancelScheduledExceptionFromTryCatch(this);
2952   }
2953   ResetInternal();
2954 }
2955 
ResetInternal()2956 void v8::TryCatch::ResetInternal() {
2957   i::Object the_hole = i::ReadOnlyRoots(isolate_).the_hole_value();
2958   exception_ = reinterpret_cast<void*>(the_hole.ptr());
2959   message_obj_ = reinterpret_cast<void*>(the_hole.ptr());
2960 }
2961 
SetVerbose(bool value)2962 void v8::TryCatch::SetVerbose(bool value) { is_verbose_ = value; }
2963 
IsVerbose() const2964 bool v8::TryCatch::IsVerbose() const { return is_verbose_; }
2965 
SetCaptureMessage(bool value)2966 void v8::TryCatch::SetCaptureMessage(bool value) { capture_message_ = value; }
2967 
2968 // --- M e s s a g e ---
2969 
Get() const2970 Local<String> Message::Get() const {
2971   auto self = Utils::OpenHandle(this);
2972   i::Isolate* isolate = self->GetIsolate();
2973   ENTER_V8_NO_SCRIPT_NO_EXCEPTION(isolate);
2974   EscapableHandleScope scope(reinterpret_cast<Isolate*>(isolate));
2975   i::Handle<i::String> raw_result =
2976       i::MessageHandler::GetMessage(isolate, self);
2977   Local<String> result = Utils::ToLocal(raw_result);
2978   return scope.Escape(result);
2979 }
2980 
GetIsolate() const2981 v8::Isolate* Message::GetIsolate() const {
2982   i::Isolate* isolate = Utils::OpenHandle(this)->GetIsolate();
2983   return reinterpret_cast<Isolate*>(isolate);
2984 }
2985 
GetScriptOrigin() const2986 ScriptOrigin Message::GetScriptOrigin() const {
2987   auto self = Utils::OpenHandle(this);
2988   i::Isolate* isolate = self->GetIsolate();
2989   ENTER_V8_NO_SCRIPT_NO_EXCEPTION(isolate);
2990   i::Handle<i::Script> script(self->script(), isolate);
2991   return GetScriptOriginForScript(isolate, script);
2992 }
2993 
VerifyHostDefinedOptions() const2994 void ScriptOrigin::VerifyHostDefinedOptions() const {
2995   // TODO(cbruni, chromium:1244145): Remove checks once we allow arbitrary
2996   // host-defined options.
2997   USE(isolate_);
2998   if (host_defined_options_.IsEmpty()) return;
2999   Utils::ApiCheck(host_defined_options_->IsFixedArray(), "ScriptOrigin()",
3000                   "Host-defined options has to be a PrimitiveArray");
3001   i::Handle<i::FixedArray> options =
3002       Utils::OpenHandle(*host_defined_options_.As<FixedArray>());
3003   for (int i = 0; i < options->length(); i++) {
3004     Utils::ApiCheck(options->get(i).IsPrimitive(), "ScriptOrigin()",
3005                     "PrimitiveArray can only contain primtive values");
3006   }
3007 }
3008 
GetScriptResourceName() const3009 v8::Local<Value> Message::GetScriptResourceName() const {
3010   ASSERT_NO_SCRIPT_NO_EXCEPTION(Utils::OpenHandle(this)->GetIsolate());
3011   return GetScriptOrigin().ResourceName();
3012 }
3013 
GetStackTrace() const3014 v8::Local<v8::StackTrace> Message::GetStackTrace() const {
3015   auto self = Utils::OpenHandle(this);
3016   i::Isolate* isolate = self->GetIsolate();
3017   ENTER_V8_NO_SCRIPT_NO_EXCEPTION(isolate);
3018   EscapableHandleScope scope(reinterpret_cast<Isolate*>(isolate));
3019   i::Handle<i::Object> stackFramesObj(self->stack_frames(), isolate);
3020   if (!stackFramesObj->IsFixedArray()) return v8::Local<v8::StackTrace>();
3021   auto stackTrace = i::Handle<i::FixedArray>::cast(stackFramesObj);
3022   return scope.Escape(Utils::StackTraceToLocal(stackTrace));
3023 }
3024 
GetLineNumber(Local<Context> context) const3025 Maybe<int> Message::GetLineNumber(Local<Context> context) const {
3026   auto self = Utils::OpenHandle(this);
3027   i::Isolate* isolate = self->GetIsolate();
3028   ENTER_V8_NO_SCRIPT_NO_EXCEPTION(isolate);
3029   EscapableHandleScope handle_scope(reinterpret_cast<Isolate*>(isolate));
3030   i::JSMessageObject::EnsureSourcePositionsAvailable(isolate, self);
3031   return Just(self->GetLineNumber());
3032 }
3033 
GetStartPosition() const3034 int Message::GetStartPosition() const {
3035   auto self = Utils::OpenHandle(this);
3036   i::Isolate* isolate = self->GetIsolate();
3037   ENTER_V8_NO_SCRIPT_NO_EXCEPTION(isolate);
3038   EscapableHandleScope handle_scope(reinterpret_cast<Isolate*>(isolate));
3039   i::JSMessageObject::EnsureSourcePositionsAvailable(isolate, self);
3040   return self->GetStartPosition();
3041 }
3042 
GetEndPosition() const3043 int Message::GetEndPosition() const {
3044   auto self = Utils::OpenHandle(this);
3045   i::Isolate* isolate = self->GetIsolate();
3046   ENTER_V8_NO_SCRIPT_NO_EXCEPTION(isolate);
3047   EscapableHandleScope handle_scope(reinterpret_cast<Isolate*>(isolate));
3048   i::JSMessageObject::EnsureSourcePositionsAvailable(isolate, self);
3049   return self->GetEndPosition();
3050 }
3051 
ErrorLevel() const3052 int Message::ErrorLevel() const {
3053   auto self = Utils::OpenHandle(this);
3054   ASSERT_NO_SCRIPT_NO_EXCEPTION(self->GetIsolate());
3055   return self->error_level();
3056 }
3057 
GetStartColumn() const3058 int Message::GetStartColumn() const {
3059   auto self = Utils::OpenHandle(this);
3060   i::Isolate* isolate = self->GetIsolate();
3061   ENTER_V8_NO_SCRIPT_NO_EXCEPTION(isolate);
3062   EscapableHandleScope handle_scope(reinterpret_cast<Isolate*>(isolate));
3063   i::JSMessageObject::EnsureSourcePositionsAvailable(isolate, self);
3064   return self->GetColumnNumber();
3065 }
3066 
GetWasmFunctionIndex() const3067 int Message::GetWasmFunctionIndex() const {
3068 #if V8_ENABLE_WEBASSEMBLY
3069   auto self = Utils::OpenHandle(this);
3070   i::Isolate* isolate = self->GetIsolate();
3071   ENTER_V8_NO_SCRIPT_NO_EXCEPTION(isolate);
3072   EscapableHandleScope handle_scope(reinterpret_cast<Isolate*>(isolate));
3073   i::JSMessageObject::EnsureSourcePositionsAvailable(isolate, self);
3074   int start_position = self->GetColumnNumber();
3075   if (start_position == -1) return Message::kNoWasmFunctionIndexInfo;
3076 
3077   i::Handle<i::Script> script(self->script(), isolate);
3078 
3079   if (script->type() != i::Script::TYPE_WASM) {
3080     return Message::kNoWasmFunctionIndexInfo;
3081   }
3082 
3083   auto debug_script = ToApiHandle<debug::Script>(script);
3084   return Local<debug::WasmScript>::Cast(debug_script)
3085       ->GetContainingFunction(start_position);
3086 #else
3087   return Message::kNoWasmFunctionIndexInfo;
3088 #endif  // V8_ENABLE_WEBASSEMBLY
3089 }
3090 
GetStartColumn(Local<Context> context) const3091 Maybe<int> Message::GetStartColumn(Local<Context> context) const {
3092   return Just(GetStartColumn());
3093 }
3094 
GetEndColumn() const3095 int Message::GetEndColumn() const {
3096   auto self = Utils::OpenHandle(this);
3097   i::Isolate* isolate = self->GetIsolate();
3098   ENTER_V8_NO_SCRIPT_NO_EXCEPTION(isolate);
3099   EscapableHandleScope handle_scope(reinterpret_cast<Isolate*>(isolate));
3100   i::JSMessageObject::EnsureSourcePositionsAvailable(isolate, self);
3101   const int column_number = self->GetColumnNumber();
3102   if (column_number == -1) return -1;
3103   const int start = self->GetStartPosition();
3104   const int end = self->GetEndPosition();
3105   return column_number + (end - start);
3106 }
3107 
GetEndColumn(Local<Context> context) const3108 Maybe<int> Message::GetEndColumn(Local<Context> context) const {
3109   return Just(GetEndColumn());
3110 }
3111 
IsSharedCrossOrigin() const3112 bool Message::IsSharedCrossOrigin() const {
3113   auto self = Utils::OpenHandle(this);
3114   i::Isolate* isolate = self->GetIsolate();
3115   ENTER_V8_NO_SCRIPT_NO_EXCEPTION(isolate);
3116   return self->script().origin_options().IsSharedCrossOrigin();
3117 }
3118 
IsOpaque() const3119 bool Message::IsOpaque() const {
3120   auto self = Utils::OpenHandle(this);
3121   i::Isolate* isolate = self->GetIsolate();
3122   ENTER_V8_NO_SCRIPT_NO_EXCEPTION(isolate);
3123   return self->script().origin_options().IsOpaque();
3124 }
3125 
GetSource(Local<Context> context) const3126 MaybeLocal<String> Message::GetSource(Local<Context> context) const {
3127   auto self = Utils::OpenHandle(this);
3128   i::Isolate* isolate = self->GetIsolate();
3129   ENTER_V8_NO_SCRIPT_NO_EXCEPTION(isolate);
3130   EscapableHandleScope handle_scope(reinterpret_cast<Isolate*>(isolate));
3131   i::Handle<i::String> source(self->GetSource(), isolate);
3132   RETURN_ESCAPED(Utils::ToLocal(source));
3133 }
3134 
GetSourceLine(Local<Context> context) const3135 MaybeLocal<String> Message::GetSourceLine(Local<Context> context) const {
3136   auto self = Utils::OpenHandle(this);
3137   i::Isolate* isolate = self->GetIsolate();
3138   ENTER_V8_NO_SCRIPT_NO_EXCEPTION(isolate);
3139   EscapableHandleScope handle_scope(reinterpret_cast<Isolate*>(isolate));
3140   i::JSMessageObject::EnsureSourcePositionsAvailable(isolate, self);
3141   RETURN_ESCAPED(Utils::ToLocal(self->GetSourceLine()));
3142 }
3143 
PrintCurrentStackTrace(Isolate * isolate,std::ostream & out)3144 void Message::PrintCurrentStackTrace(Isolate* isolate, std::ostream& out) {
3145   i::Isolate* i_isolate = reinterpret_cast<i::Isolate*>(isolate);
3146   ENTER_V8_NO_SCRIPT_NO_EXCEPTION(i_isolate);
3147   i_isolate->PrintCurrentStackTrace(out);
3148 }
3149 
3150 // --- S t a c k T r a c e ---
3151 
GetFrame(Isolate * v8_isolate,uint32_t index) const3152 Local<StackFrame> StackTrace::GetFrame(Isolate* v8_isolate,
3153                                        uint32_t index) const {
3154   i::Isolate* isolate = reinterpret_cast<i::Isolate*>(v8_isolate);
3155   i::Handle<i::StackFrameInfo> info(
3156       i::StackFrameInfo::cast(Utils::OpenHandle(this)->get(index)), isolate);
3157   return Utils::StackFrameToLocal(info);
3158 }
3159 
GetFrameCount() const3160 int StackTrace::GetFrameCount() const {
3161   return Utils::OpenHandle(this)->length();
3162 }
3163 
CurrentStackTrace(Isolate * isolate,int frame_limit,StackTraceOptions options)3164 Local<StackTrace> StackTrace::CurrentStackTrace(Isolate* isolate,
3165                                                 int frame_limit,
3166                                                 StackTraceOptions options) {
3167   i::Isolate* i_isolate = reinterpret_cast<i::Isolate*>(isolate);
3168   ENTER_V8_NO_SCRIPT_NO_EXCEPTION(i_isolate);
3169   i::Handle<i::FixedArray> stackTrace =
3170       i_isolate->CaptureDetailedStackTrace(frame_limit, options);
3171   return Utils::StackTraceToLocal(stackTrace);
3172 }
3173 
CurrentScriptNameOrSourceURL(Isolate * v8_isolate)3174 Local<String> StackTrace::CurrentScriptNameOrSourceURL(Isolate* v8_isolate) {
3175   i::Isolate* isolate = reinterpret_cast<i::Isolate*>(v8_isolate);
3176   ENTER_V8_NO_SCRIPT_NO_EXCEPTION(isolate);
3177   i::Handle<i::String> name_or_source_url =
3178       isolate->CurrentScriptNameOrSourceURL();
3179   return Utils::ToLocal(name_or_source_url);
3180 }
3181 
3182 // --- S t a c k F r a m e ---
3183 
GetLocation() const3184 Location StackFrame::GetLocation() const {
3185   i::Handle<i::StackFrameInfo> self = Utils::OpenHandle(this);
3186   i::Isolate* isolate = self->GetIsolate();
3187   i::Handle<i::Script> script(self->script(), isolate);
3188   i::Script::PositionInfo info;
3189   CHECK(i::Script::GetPositionInfo(script,
3190                                    i::StackFrameInfo::GetSourcePosition(self),
3191                                    &info, i::Script::WITH_OFFSET));
3192   if (script->HasSourceURLComment()) {
3193     info.line -= script->line_offset();
3194     if (info.line == 0) {
3195       info.column -= script->column_offset();
3196     }
3197   }
3198   return {info.line, info.column};
3199 }
3200 
GetScriptId() const3201 int StackFrame::GetScriptId() const {
3202   return Utils::OpenHandle(this)->script().id();
3203 }
3204 
GetScriptName() const3205 Local<String> StackFrame::GetScriptName() const {
3206   i::Handle<i::StackFrameInfo> self = Utils::OpenHandle(this);
3207   i::Isolate* isolate = self->GetIsolate();
3208   i::Handle<i::Object> name(self->script().name(), isolate);
3209   if (!name->IsString()) return {};
3210   return Utils::ToLocal(i::Handle<i::String>::cast(name));
3211 }
3212 
GetScriptNameOrSourceURL() const3213 Local<String> StackFrame::GetScriptNameOrSourceURL() const {
3214   i::Handle<i::StackFrameInfo> self = Utils::OpenHandle(this);
3215   i::Isolate* isolate = self->GetIsolate();
3216   i::Handle<i::Object> name_or_source_url(self->script().GetNameOrSourceURL(),
3217                                           isolate);
3218   if (!name_or_source_url->IsString()) return {};
3219   return Utils::ToLocal(i::Handle<i::String>::cast(name_or_source_url));
3220 }
3221 
GetScriptSource() const3222 Local<String> StackFrame::GetScriptSource() const {
3223   i::Handle<i::StackFrameInfo> self = Utils::OpenHandle(this);
3224   i::Isolate* isolate = self->GetIsolate();
3225   if (!self->script().HasValidSource()) return {};
3226   i::Handle<i::PrimitiveHeapObject> source(self->script().source(), isolate);
3227   if (!source->IsString()) return {};
3228   return Utils::ToLocal(i::Handle<i::String>::cast(source));
3229 }
3230 
GetScriptSourceMappingURL() const3231 Local<String> StackFrame::GetScriptSourceMappingURL() const {
3232   i::Handle<i::StackFrameInfo> self = Utils::OpenHandle(this);
3233   i::Isolate* isolate = self->GetIsolate();
3234   i::Handle<i::Object> source_mapping_url(self->script().source_mapping_url(),
3235                                           isolate);
3236   if (!source_mapping_url->IsString()) return {};
3237   return Utils::ToLocal(i::Handle<i::String>::cast(source_mapping_url));
3238 }
3239 
GetFunctionName() const3240 Local<String> StackFrame::GetFunctionName() const {
3241   i::Handle<i::StackFrameInfo> self = Utils::OpenHandle(this);
3242   i::Isolate* isolate = self->GetIsolate();
3243   i::Handle<i::String> name(self->function_name(), isolate);
3244   if (name->length() == 0) return {};
3245   return Utils::ToLocal(name);
3246 }
3247 
IsEval() const3248 bool StackFrame::IsEval() const {
3249   i::Handle<i::StackFrameInfo> self = Utils::OpenHandle(this);
3250   return self->script().compilation_type() == i::Script::COMPILATION_TYPE_EVAL;
3251 }
3252 
IsConstructor() const3253 bool StackFrame::IsConstructor() const {
3254   return Utils::OpenHandle(this)->is_constructor();
3255 }
3256 
IsWasm() const3257 bool StackFrame::IsWasm() const { return !IsUserJavaScript(); }
3258 
IsUserJavaScript() const3259 bool StackFrame::IsUserJavaScript() const {
3260   return Utils::OpenHandle(this)->script().IsUserJavaScript();
3261 }
3262 
3263 // --- J S O N ---
3264 
Parse(Local<Context> context,Local<String> json_string)3265 MaybeLocal<Value> JSON::Parse(Local<Context> context,
3266                               Local<String> json_string) {
3267   PREPARE_FOR_EXECUTION(context, JSON, Parse, Value);
3268   i::Handle<i::String> string = Utils::OpenHandle(*json_string);
3269   i::Handle<i::String> source = i::String::Flatten(isolate, string);
3270   i::Handle<i::Object> undefined = isolate->factory()->undefined_value();
3271   auto maybe = source->IsOneByteRepresentation()
3272                    ? i::JsonParser<uint8_t>::Parse(isolate, source, undefined)
3273                    : i::JsonParser<uint16_t>::Parse(isolate, source, undefined);
3274   Local<Value> result;
3275   has_pending_exception = !ToLocal<Value>(maybe, &result);
3276   RETURN_ON_FAILED_EXECUTION(Value);
3277   RETURN_ESCAPED(result);
3278 }
3279 
Stringify(Local<Context> context,Local<Value> json_object,Local<String> gap)3280 MaybeLocal<String> JSON::Stringify(Local<Context> context,
3281                                    Local<Value> json_object,
3282                                    Local<String> gap) {
3283   PREPARE_FOR_EXECUTION(context, JSON, Stringify, String);
3284   i::Handle<i::Object> object = Utils::OpenHandle(*json_object);
3285   i::Handle<i::Object> replacer = isolate->factory()->undefined_value();
3286   i::Handle<i::String> gap_string = gap.IsEmpty()
3287                                         ? isolate->factory()->empty_string()
3288                                         : Utils::OpenHandle(*gap);
3289   i::Handle<i::Object> maybe;
3290   has_pending_exception =
3291       !i::JsonStringify(isolate, object, replacer, gap_string).ToHandle(&maybe);
3292   RETURN_ON_FAILED_EXECUTION(String);
3293   Local<String> result;
3294   has_pending_exception =
3295       !ToLocal<String>(i::Object::ToString(isolate, maybe), &result);
3296   RETURN_ON_FAILED_EXECUTION(String);
3297   RETURN_ESCAPED(result);
3298 }
3299 
3300 // --- V a l u e   S e r i a l i z a t i o n ---
3301 
WriteHostObject(Isolate * v8_isolate,Local<Object> object)3302 Maybe<bool> ValueSerializer::Delegate::WriteHostObject(Isolate* v8_isolate,
3303                                                        Local<Object> object) {
3304   i::Isolate* isolate = reinterpret_cast<i::Isolate*>(v8_isolate);
3305   isolate->ScheduleThrow(*isolate->factory()->NewError(
3306       isolate->error_function(), i::MessageTemplate::kDataCloneError,
3307       Utils::OpenHandle(*object)));
3308   return Nothing<bool>();
3309 }
3310 
GetSharedArrayBufferId(Isolate * v8_isolate,Local<SharedArrayBuffer> shared_array_buffer)3311 Maybe<uint32_t> ValueSerializer::Delegate::GetSharedArrayBufferId(
3312     Isolate* v8_isolate, Local<SharedArrayBuffer> shared_array_buffer) {
3313   i::Isolate* isolate = reinterpret_cast<i::Isolate*>(v8_isolate);
3314   isolate->ScheduleThrow(*isolate->factory()->NewError(
3315       isolate->error_function(), i::MessageTemplate::kDataCloneError,
3316       Utils::OpenHandle(*shared_array_buffer)));
3317   return Nothing<uint32_t>();
3318 }
3319 
GetWasmModuleTransferId(Isolate * v8_isolate,Local<WasmModuleObject> module)3320 Maybe<uint32_t> ValueSerializer::Delegate::GetWasmModuleTransferId(
3321     Isolate* v8_isolate, Local<WasmModuleObject> module) {
3322   return Nothing<uint32_t>();
3323 }
3324 
SupportsSharedValues() const3325 bool ValueSerializer::Delegate::SupportsSharedValues() const { return false; }
3326 
GetSharedValueId(Isolate * v8_isolate,Local<Value> shared_value)3327 Maybe<uint32_t> ValueSerializer::Delegate::GetSharedValueId(
3328     Isolate* v8_isolate, Local<Value> shared_value) {
3329   i::Isolate* isolate = reinterpret_cast<i::Isolate*>(v8_isolate);
3330   isolate->ScheduleThrow(*isolate->factory()->NewError(
3331       isolate->error_function(), i::MessageTemplate::kDataCloneError,
3332       Utils::OpenHandle(*shared_value)));
3333   return Nothing<uint32_t>();
3334 }
3335 
ReallocateBufferMemory(void * old_buffer,size_t size,size_t * actual_size)3336 void* ValueSerializer::Delegate::ReallocateBufferMemory(void* old_buffer,
3337                                                         size_t size,
3338                                                         size_t* actual_size) {
3339   *actual_size = size;
3340   return base::Realloc(old_buffer, size);
3341 }
3342 
FreeBufferMemory(void * buffer)3343 void ValueSerializer::Delegate::FreeBufferMemory(void* buffer) {
3344   return base::Free(buffer);
3345 }
3346 
3347 struct ValueSerializer::PrivateData {
PrivateDatav8::ValueSerializer::PrivateData3348   explicit PrivateData(i::Isolate* i, ValueSerializer::Delegate* delegate)
3349       : isolate(i), serializer(i, delegate) {}
3350   i::Isolate* isolate;
3351   i::ValueSerializer serializer;
3352 };
3353 
ValueSerializer(Isolate * isolate)3354 ValueSerializer::ValueSerializer(Isolate* isolate)
3355     : ValueSerializer(isolate, nullptr) {}
3356 
ValueSerializer(Isolate * isolate,Delegate * delegate)3357 ValueSerializer::ValueSerializer(Isolate* isolate, Delegate* delegate)
3358     : private_(
3359           new PrivateData(reinterpret_cast<i::Isolate*>(isolate), delegate)) {}
3360 
~ValueSerializer()3361 ValueSerializer::~ValueSerializer() { delete private_; }
3362 
WriteHeader()3363 void ValueSerializer::WriteHeader() { private_->serializer.WriteHeader(); }
3364 
SetTreatArrayBufferViewsAsHostObjects(bool mode)3365 void ValueSerializer::SetTreatArrayBufferViewsAsHostObjects(bool mode) {
3366   private_->serializer.SetTreatArrayBufferViewsAsHostObjects(mode);
3367 }
3368 
WriteValue(Local<Context> context,Local<Value> value)3369 Maybe<bool> ValueSerializer::WriteValue(Local<Context> context,
3370                                         Local<Value> value) {
3371   auto isolate = reinterpret_cast<i::Isolate*>(context->GetIsolate());
3372   ENTER_V8(isolate, context, ValueSerializer, WriteValue, Nothing<bool>(),
3373            i::HandleScope);
3374   i::Handle<i::Object> object = Utils::OpenHandle(*value);
3375   Maybe<bool> result = private_->serializer.WriteObject(object);
3376   has_pending_exception = result.IsNothing();
3377   RETURN_ON_FAILED_EXECUTION_PRIMITIVE(bool);
3378   return result;
3379 }
3380 
Release()3381 std::pair<uint8_t*, size_t> ValueSerializer::Release() {
3382   return private_->serializer.Release();
3383 }
3384 
TransferArrayBuffer(uint32_t transfer_id,Local<ArrayBuffer> array_buffer)3385 void ValueSerializer::TransferArrayBuffer(uint32_t transfer_id,
3386                                           Local<ArrayBuffer> array_buffer) {
3387   private_->serializer.TransferArrayBuffer(transfer_id,
3388                                            Utils::OpenHandle(*array_buffer));
3389 }
3390 
WriteUint32(uint32_t value)3391 void ValueSerializer::WriteUint32(uint32_t value) {
3392   private_->serializer.WriteUint32(value);
3393 }
3394 
WriteUint64(uint64_t value)3395 void ValueSerializer::WriteUint64(uint64_t value) {
3396   private_->serializer.WriteUint64(value);
3397 }
3398 
WriteDouble(double value)3399 void ValueSerializer::WriteDouble(double value) {
3400   private_->serializer.WriteDouble(value);
3401 }
3402 
WriteRawBytes(const void * source,size_t length)3403 void ValueSerializer::WriteRawBytes(const void* source, size_t length) {
3404   private_->serializer.WriteRawBytes(source, length);
3405 }
3406 
ReadHostObject(Isolate * v8_isolate)3407 MaybeLocal<Object> ValueDeserializer::Delegate::ReadHostObject(
3408     Isolate* v8_isolate) {
3409   i::Isolate* isolate = reinterpret_cast<i::Isolate*>(v8_isolate);
3410   isolate->ScheduleThrow(*isolate->factory()->NewError(
3411       isolate->error_function(),
3412       i::MessageTemplate::kDataCloneDeserializationError));
3413   return MaybeLocal<Object>();
3414 }
3415 
GetWasmModuleFromId(Isolate * v8_isolate,uint32_t id)3416 MaybeLocal<WasmModuleObject> ValueDeserializer::Delegate::GetWasmModuleFromId(
3417     Isolate* v8_isolate, uint32_t id) {
3418   i::Isolate* isolate = reinterpret_cast<i::Isolate*>(v8_isolate);
3419   isolate->ScheduleThrow(*isolate->factory()->NewError(
3420       isolate->error_function(),
3421       i::MessageTemplate::kDataCloneDeserializationError));
3422   return MaybeLocal<WasmModuleObject>();
3423 }
3424 
SupportsSharedValues() const3425 bool ValueDeserializer::Delegate::SupportsSharedValues() const { return false; }
3426 
GetSharedValueFromId(Isolate * v8_isolate,uint32_t shared_value_id)3427 MaybeLocal<Value> ValueDeserializer::Delegate::GetSharedValueFromId(
3428     Isolate* v8_isolate, uint32_t shared_value_id) {
3429   i::Isolate* isolate = reinterpret_cast<i::Isolate*>(v8_isolate);
3430   isolate->ScheduleThrow(*isolate->factory()->NewError(
3431       isolate->error_function(),
3432       i::MessageTemplate::kDataCloneDeserializationError));
3433   return MaybeLocal<Value>();
3434 }
3435 
3436 MaybeLocal<SharedArrayBuffer>
GetSharedArrayBufferFromId(Isolate * v8_isolate,uint32_t id)3437 ValueDeserializer::Delegate::GetSharedArrayBufferFromId(Isolate* v8_isolate,
3438                                                         uint32_t id) {
3439   i::Isolate* isolate = reinterpret_cast<i::Isolate*>(v8_isolate);
3440   isolate->ScheduleThrow(*isolate->factory()->NewError(
3441       isolate->error_function(),
3442       i::MessageTemplate::kDataCloneDeserializationError));
3443   return MaybeLocal<SharedArrayBuffer>();
3444 }
3445 
3446 struct ValueDeserializer::PrivateData {
PrivateDatav8::ValueDeserializer::PrivateData3447   PrivateData(i::Isolate* i, base::Vector<const uint8_t> data,
3448               Delegate* delegate)
3449       : isolate(i), deserializer(i, data, delegate) {}
3450   i::Isolate* isolate;
3451   i::ValueDeserializer deserializer;
3452   bool supports_legacy_wire_format = false;
3453 };
3454 
ValueDeserializer(Isolate * isolate,const uint8_t * data,size_t size)3455 ValueDeserializer::ValueDeserializer(Isolate* isolate, const uint8_t* data,
3456                                      size_t size)
3457     : ValueDeserializer(isolate, data, size, nullptr) {}
3458 
ValueDeserializer(Isolate * isolate,const uint8_t * data,size_t size,Delegate * delegate)3459 ValueDeserializer::ValueDeserializer(Isolate* isolate, const uint8_t* data,
3460                                      size_t size, Delegate* delegate) {
3461   private_ = new PrivateData(reinterpret_cast<i::Isolate*>(isolate),
3462                              base::Vector<const uint8_t>(data, size), delegate);
3463 }
3464 
~ValueDeserializer()3465 ValueDeserializer::~ValueDeserializer() { delete private_; }
3466 
ReadHeader(Local<Context> context)3467 Maybe<bool> ValueDeserializer::ReadHeader(Local<Context> context) {
3468   auto isolate = reinterpret_cast<i::Isolate*>(context->GetIsolate());
3469   ENTER_V8_NO_SCRIPT(isolate, context, ValueDeserializer, ReadHeader,
3470                      Nothing<bool>(), i::HandleScope);
3471 
3472   bool read_header = false;
3473   has_pending_exception = !private_->deserializer.ReadHeader().To(&read_header);
3474   RETURN_ON_FAILED_EXECUTION_PRIMITIVE(bool);
3475   DCHECK(read_header);
3476 
3477   static const uint32_t kMinimumNonLegacyVersion = 13;
3478   if (GetWireFormatVersion() < kMinimumNonLegacyVersion &&
3479       !private_->supports_legacy_wire_format) {
3480     isolate->Throw(*isolate->factory()->NewError(
3481         i::MessageTemplate::kDataCloneDeserializationVersionError));
3482     has_pending_exception = true;
3483     RETURN_ON_FAILED_EXECUTION_PRIMITIVE(bool);
3484   }
3485 
3486   return Just(true);
3487 }
3488 
SetSupportsLegacyWireFormat(bool supports_legacy_wire_format)3489 void ValueDeserializer::SetSupportsLegacyWireFormat(
3490     bool supports_legacy_wire_format) {
3491   private_->supports_legacy_wire_format = supports_legacy_wire_format;
3492 }
3493 
GetWireFormatVersion() const3494 uint32_t ValueDeserializer::GetWireFormatVersion() const {
3495   return private_->deserializer.GetWireFormatVersion();
3496 }
3497 
ReadValue(Local<Context> context)3498 MaybeLocal<Value> ValueDeserializer::ReadValue(Local<Context> context) {
3499   PREPARE_FOR_EXECUTION(context, ValueDeserializer, ReadValue, Value);
3500   i::MaybeHandle<i::Object> result;
3501   if (GetWireFormatVersion() > 0) {
3502     result = private_->deserializer.ReadObjectWrapper();
3503   } else {
3504     result =
3505         private_->deserializer.ReadObjectUsingEntireBufferForLegacyFormat();
3506   }
3507   Local<Value> value;
3508   has_pending_exception = !ToLocal(result, &value);
3509   RETURN_ON_FAILED_EXECUTION(Value);
3510   RETURN_ESCAPED(value);
3511 }
3512 
TransferArrayBuffer(uint32_t transfer_id,Local<ArrayBuffer> array_buffer)3513 void ValueDeserializer::TransferArrayBuffer(uint32_t transfer_id,
3514                                             Local<ArrayBuffer> array_buffer) {
3515   private_->deserializer.TransferArrayBuffer(transfer_id,
3516                                              Utils::OpenHandle(*array_buffer));
3517 }
3518 
TransferSharedArrayBuffer(uint32_t transfer_id,Local<SharedArrayBuffer> shared_array_buffer)3519 void ValueDeserializer::TransferSharedArrayBuffer(
3520     uint32_t transfer_id, Local<SharedArrayBuffer> shared_array_buffer) {
3521   private_->deserializer.TransferArrayBuffer(
3522       transfer_id, Utils::OpenHandle(*shared_array_buffer));
3523 }
3524 
ReadUint32(uint32_t * value)3525 bool ValueDeserializer::ReadUint32(uint32_t* value) {
3526   return private_->deserializer.ReadUint32(value);
3527 }
3528 
ReadUint64(uint64_t * value)3529 bool ValueDeserializer::ReadUint64(uint64_t* value) {
3530   return private_->deserializer.ReadUint64(value);
3531 }
3532 
ReadDouble(double * value)3533 bool ValueDeserializer::ReadDouble(double* value) {
3534   return private_->deserializer.ReadDouble(value);
3535 }
3536 
ReadRawBytes(size_t length,const void ** data)3537 bool ValueDeserializer::ReadRawBytes(size_t length, const void** data) {
3538   return private_->deserializer.ReadRawBytes(length, data);
3539 }
3540 
3541 // --- D a t a ---
3542 
FullIsUndefined() const3543 bool Value::FullIsUndefined() const {
3544   i::Handle<i::Object> object = Utils::OpenHandle(this);
3545   bool result = object->IsUndefined();
3546   DCHECK_EQ(result, QuickIsUndefined());
3547   return result;
3548 }
3549 
FullIsNull() const3550 bool Value::FullIsNull() const {
3551   i::Handle<i::Object> object = Utils::OpenHandle(this);
3552   bool result = object->IsNull();
3553   DCHECK_EQ(result, QuickIsNull());
3554   return result;
3555 }
3556 
IsTrue() const3557 bool Value::IsTrue() const {
3558   i::Object object = *Utils::OpenHandle(this);
3559   if (object.IsSmi()) return false;
3560   return object.IsTrue();
3561 }
3562 
IsFalse() const3563 bool Value::IsFalse() const {
3564   i::Object object = *Utils::OpenHandle(this);
3565   if (object.IsSmi()) return false;
3566   return object.IsFalse();
3567 }
3568 
IsFunction() const3569 bool Value::IsFunction() const { return Utils::OpenHandle(this)->IsCallable(); }
3570 
IsName() const3571 bool Value::IsName() const { return Utils::OpenHandle(this)->IsName(); }
3572 
FullIsString() const3573 bool Value::FullIsString() const {
3574   bool result = Utils::OpenHandle(this)->IsString();
3575   DCHECK_EQ(result, QuickIsString());
3576   return result;
3577 }
3578 
IsSymbol() const3579 bool Value::IsSymbol() const {
3580   return Utils::OpenHandle(this)->IsPublicSymbol();
3581 }
3582 
IsArray() const3583 bool Value::IsArray() const { return Utils::OpenHandle(this)->IsJSArray(); }
3584 
IsArrayBuffer() const3585 bool Value::IsArrayBuffer() const {
3586   i::Object obj = *Utils::OpenHandle(this);
3587   if (!obj.IsJSArrayBuffer()) return false;
3588   return !i::JSArrayBuffer::cast(obj).is_shared();
3589 }
3590 
IsArrayBufferView() const3591 bool Value::IsArrayBufferView() const {
3592   return Utils::OpenHandle(this)->IsJSArrayBufferView();
3593 }
3594 
IsTypedArray() const3595 bool Value::IsTypedArray() const {
3596   return Utils::OpenHandle(this)->IsJSTypedArray();
3597 }
3598 
3599 #define VALUE_IS_TYPED_ARRAY(Type, typeName, TYPE, ctype)                   \
3600   bool Value::Is##Type##Array() const {                                     \
3601     i::Handle<i::Object> obj = Utils::OpenHandle(this);                     \
3602     return obj->IsJSTypedArray() &&                                         \
3603            i::JSTypedArray::cast(*obj).type() == i::kExternal##Type##Array; \
3604   }
3605 
TYPED_ARRAYS(VALUE_IS_TYPED_ARRAY)3606 TYPED_ARRAYS(VALUE_IS_TYPED_ARRAY)
3607 
3608 #undef VALUE_IS_TYPED_ARRAY
3609 
3610 bool Value::IsDataView() const {
3611   return Utils::OpenHandle(this)->IsJSDataView();
3612 }
3613 
IsSharedArrayBuffer() const3614 bool Value::IsSharedArrayBuffer() const {
3615   i::Object obj = *Utils::OpenHandle(this);
3616   if (!obj.IsJSArrayBuffer()) return false;
3617   return i::JSArrayBuffer::cast(obj).is_shared();
3618 }
3619 
IsObject() const3620 bool Value::IsObject() const { return Utils::OpenHandle(this)->IsJSReceiver(); }
3621 
IsNumber() const3622 bool Value::IsNumber() const { return Utils::OpenHandle(this)->IsNumber(); }
3623 
IsBigInt() const3624 bool Value::IsBigInt() const { return Utils::OpenHandle(this)->IsBigInt(); }
3625 
IsProxy() const3626 bool Value::IsProxy() const { return Utils::OpenHandle(this)->IsJSProxy(); }
3627 
3628 #define VALUE_IS_SPECIFIC_TYPE(Type, Check)             \
3629   bool Value::Is##Type() const {                        \
3630     i::Handle<i::Object> obj = Utils::OpenHandle(this); \
3631     return obj->Is##Check();                            \
3632   }
3633 
VALUE_IS_SPECIFIC_TYPE(ArgumentsObject,JSArgumentsObject)3634 VALUE_IS_SPECIFIC_TYPE(ArgumentsObject, JSArgumentsObject)
3635 VALUE_IS_SPECIFIC_TYPE(BigIntObject, BigIntWrapper)
3636 VALUE_IS_SPECIFIC_TYPE(BooleanObject, BooleanWrapper)
3637 VALUE_IS_SPECIFIC_TYPE(NumberObject, NumberWrapper)
3638 VALUE_IS_SPECIFIC_TYPE(StringObject, StringWrapper)
3639 VALUE_IS_SPECIFIC_TYPE(SymbolObject, SymbolWrapper)
3640 VALUE_IS_SPECIFIC_TYPE(Date, JSDate)
3641 VALUE_IS_SPECIFIC_TYPE(Map, JSMap)
3642 VALUE_IS_SPECIFIC_TYPE(Set, JSSet)
3643 #if V8_ENABLE_WEBASSEMBLY
3644 VALUE_IS_SPECIFIC_TYPE(WasmMemoryObject, WasmMemoryObject)
3645 VALUE_IS_SPECIFIC_TYPE(WasmModuleObject, WasmModuleObject)
3646 #else
3647 bool Value::IsWasmMemoryObject() const { return false; }
3648 bool Value::IsWasmModuleObject() const { return false; }
3649 #endif  // V8_ENABLE_WEBASSEMBLY
3650 VALUE_IS_SPECIFIC_TYPE(WeakMap, JSWeakMap)
3651 VALUE_IS_SPECIFIC_TYPE(WeakSet, JSWeakSet)
3652 
3653 #undef VALUE_IS_SPECIFIC_TYPE
3654 
3655 bool Value::IsBoolean() const { return Utils::OpenHandle(this)->IsBoolean(); }
3656 
IsExternal() const3657 bool Value::IsExternal() const {
3658   i::Object obj = *Utils::OpenHandle(this);
3659   return obj.IsJSExternalObject();
3660 }
3661 
IsInt32() const3662 bool Value::IsInt32() const {
3663   i::Object obj = *Utils::OpenHandle(this);
3664   if (obj.IsSmi()) return true;
3665   if (obj.IsNumber()) {
3666     return i::IsInt32Double(obj.Number());
3667   }
3668   return false;
3669 }
3670 
IsUint32() const3671 bool Value::IsUint32() const {
3672   i::Handle<i::Object> obj = Utils::OpenHandle(this);
3673   if (obj->IsSmi()) return i::Smi::ToInt(*obj) >= 0;
3674   if (obj->IsNumber()) {
3675     double value = obj->Number();
3676     return !i::IsMinusZero(value) && value >= 0 && value <= i::kMaxUInt32 &&
3677            value == i::FastUI2D(i::FastD2UI(value));
3678   }
3679   return false;
3680 }
3681 
IsNativeError() const3682 bool Value::IsNativeError() const {
3683   return Utils::OpenHandle(this)->IsJSError();
3684 }
3685 
IsRegExp() const3686 bool Value::IsRegExp() const {
3687   i::Handle<i::Object> obj = Utils::OpenHandle(this);
3688   return obj->IsJSRegExp();
3689 }
3690 
IsAsyncFunction() const3691 bool Value::IsAsyncFunction() const {
3692   i::Object obj = *Utils::OpenHandle(this);
3693   if (!obj.IsJSFunction()) return false;
3694   i::JSFunction func = i::JSFunction::cast(obj);
3695   return i::IsAsyncFunction(func.shared().kind());
3696 }
3697 
IsGeneratorFunction() const3698 bool Value::IsGeneratorFunction() const {
3699   i::Object obj = *Utils::OpenHandle(this);
3700   if (!obj.IsJSFunction()) return false;
3701   i::JSFunction func = i::JSFunction::cast(obj);
3702   ASSERT_NO_SCRIPT_NO_EXCEPTION(func.GetIsolate());
3703   return i::IsGeneratorFunction(func.shared().kind());
3704 }
3705 
IsGeneratorObject() const3706 bool Value::IsGeneratorObject() const {
3707   return Utils::OpenHandle(this)->IsJSGeneratorObject();
3708 }
3709 
IsMapIterator() const3710 bool Value::IsMapIterator() const {
3711   return Utils::OpenHandle(this)->IsJSMapIterator();
3712 }
3713 
IsSetIterator() const3714 bool Value::IsSetIterator() const {
3715   return Utils::OpenHandle(this)->IsJSSetIterator();
3716 }
3717 
IsPromise() const3718 bool Value::IsPromise() const { return Utils::OpenHandle(this)->IsJSPromise(); }
3719 
IsModuleNamespaceObject() const3720 bool Value::IsModuleNamespaceObject() const {
3721   return Utils::OpenHandle(this)->IsJSModuleNamespace();
3722 }
3723 
ToString(Local<Context> context) const3724 MaybeLocal<String> Value::ToString(Local<Context> context) const {
3725   auto obj = Utils::OpenHandle(this);
3726   if (obj->IsString()) return ToApiHandle<String>(obj);
3727   PREPARE_FOR_EXECUTION(context, Object, ToString, String);
3728   Local<String> result;
3729   has_pending_exception =
3730       !ToLocal<String>(i::Object::ToString(isolate, obj), &result);
3731   RETURN_ON_FAILED_EXECUTION(String);
3732   RETURN_ESCAPED(result);
3733 }
3734 
ToDetailString(Local<Context> context) const3735 MaybeLocal<String> Value::ToDetailString(Local<Context> context) const {
3736   i::Handle<i::Object> obj = Utils::OpenHandle(this);
3737   if (obj->IsString()) return ToApiHandle<String>(obj);
3738   PREPARE_FOR_EXECUTION(context, Object, ToDetailString, String);
3739   Local<String> result =
3740       Utils::ToLocal(i::Object::NoSideEffectsToString(isolate, obj));
3741   RETURN_ON_FAILED_EXECUTION(String);
3742   RETURN_ESCAPED(result);
3743 }
3744 
ToObject(Local<Context> context) const3745 MaybeLocal<Object> Value::ToObject(Local<Context> context) const {
3746   auto obj = Utils::OpenHandle(this);
3747   if (obj->IsJSReceiver()) return ToApiHandle<Object>(obj);
3748   PREPARE_FOR_EXECUTION(context, Object, ToObject, Object);
3749   Local<Object> result;
3750   has_pending_exception =
3751       !ToLocal<Object>(i::Object::ToObject(isolate, obj), &result);
3752   RETURN_ON_FAILED_EXECUTION(Object);
3753   RETURN_ESCAPED(result);
3754 }
3755 
ToBigInt(Local<Context> context) const3756 MaybeLocal<BigInt> Value::ToBigInt(Local<Context> context) const {
3757   i::Handle<i::Object> obj = Utils::OpenHandle(this);
3758   if (obj->IsBigInt()) return ToApiHandle<BigInt>(obj);
3759   PREPARE_FOR_EXECUTION(context, Object, ToBigInt, BigInt);
3760   Local<BigInt> result;
3761   has_pending_exception =
3762       !ToLocal<BigInt>(i::BigInt::FromObject(isolate, obj), &result);
3763   RETURN_ON_FAILED_EXECUTION(BigInt);
3764   RETURN_ESCAPED(result);
3765 }
3766 
BooleanValue(Isolate * v8_isolate) const3767 bool Value::BooleanValue(Isolate* v8_isolate) const {
3768   return Utils::OpenHandle(this)->BooleanValue(
3769       reinterpret_cast<i::Isolate*>(v8_isolate));
3770 }
3771 
ToBoolean(Isolate * v8_isolate) const3772 Local<Boolean> Value::ToBoolean(Isolate* v8_isolate) const {
3773   auto isolate = reinterpret_cast<i::Isolate*>(v8_isolate);
3774   ASSERT_NO_SCRIPT_NO_EXCEPTION(isolate);
3775   return ToApiHandle<Boolean>(
3776       isolate->factory()->ToBoolean(BooleanValue(v8_isolate)));
3777 }
3778 
ToNumber(Local<Context> context) const3779 MaybeLocal<Number> Value::ToNumber(Local<Context> context) const {
3780   auto obj = Utils::OpenHandle(this);
3781   if (obj->IsNumber()) return ToApiHandle<Number>(obj);
3782   PREPARE_FOR_EXECUTION(context, Object, ToNumber, Number);
3783   Local<Number> result;
3784   has_pending_exception =
3785       !ToLocal<Number>(i::Object::ToNumber(isolate, obj), &result);
3786   RETURN_ON_FAILED_EXECUTION(Number);
3787   RETURN_ESCAPED(result);
3788 }
3789 
ToInteger(Local<Context> context) const3790 MaybeLocal<Integer> Value::ToInteger(Local<Context> context) const {
3791   auto obj = Utils::OpenHandle(this);
3792   if (obj->IsSmi()) return ToApiHandle<Integer>(obj);
3793   PREPARE_FOR_EXECUTION(context, Object, ToInteger, Integer);
3794   Local<Integer> result;
3795   has_pending_exception =
3796       !ToLocal<Integer>(i::Object::ToInteger(isolate, obj), &result);
3797   RETURN_ON_FAILED_EXECUTION(Integer);
3798   RETURN_ESCAPED(result);
3799 }
3800 
ToInt32(Local<Context> context) const3801 MaybeLocal<Int32> Value::ToInt32(Local<Context> context) const {
3802   auto obj = Utils::OpenHandle(this);
3803   if (obj->IsSmi()) return ToApiHandle<Int32>(obj);
3804   Local<Int32> result;
3805   PREPARE_FOR_EXECUTION(context, Object, ToInt32, Int32);
3806   has_pending_exception =
3807       !ToLocal<Int32>(i::Object::ToInt32(isolate, obj), &result);
3808   RETURN_ON_FAILED_EXECUTION(Int32);
3809   RETURN_ESCAPED(result);
3810 }
3811 
ToUint32(Local<Context> context) const3812 MaybeLocal<Uint32> Value::ToUint32(Local<Context> context) const {
3813   auto obj = Utils::OpenHandle(this);
3814   if (obj->IsSmi()) return ToApiHandle<Uint32>(obj);
3815   Local<Uint32> result;
3816   PREPARE_FOR_EXECUTION(context, Object, ToUint32, Uint32);
3817   has_pending_exception =
3818       !ToLocal<Uint32>(i::Object::ToUint32(isolate, obj), &result);
3819   RETURN_ON_FAILED_EXECUTION(Uint32);
3820   RETURN_ESCAPED(result);
3821 }
3822 
DecodeExternalPointerImpl(const i::Isolate * isolate,i::ExternalPointer_t encoded_pointer,ExternalPointerTag tag)3823 i::Address i::DecodeExternalPointerImpl(const i::Isolate* isolate,
3824                                         i::ExternalPointer_t encoded_pointer,
3825                                         ExternalPointerTag tag) {
3826   return i::DecodeExternalPointer(isolate, encoded_pointer, tag);
3827 }
3828 
IsolateFromNeverReadOnlySpaceObject(i::Address obj)3829 i::Isolate* i::IsolateFromNeverReadOnlySpaceObject(i::Address obj) {
3830   return i::GetIsolateFromWritableObject(i::HeapObject::cast(i::Object(obj)));
3831 }
3832 
ShouldThrowOnError(i::Isolate * isolate)3833 bool i::ShouldThrowOnError(i::Isolate* isolate) {
3834   return i::GetShouldThrow(isolate, Nothing<i::ShouldThrow>()) ==
3835          i::ShouldThrow::kThrowOnError;
3836 }
3837 
CanHaveInternalField(int instance_type)3838 bool i::CanHaveInternalField(int instance_type) {
3839   return instance_type == i::Internals::kJSObjectType ||
3840          instance_type == i::Internals::kJSSpecialApiObjectType ||
3841          v8::internal::InstanceTypeChecker::IsJSApiObject(
3842              static_cast<v8::internal::InstanceType>(instance_type));
3843 }
3844 
CheckInitializedImpl(v8::Isolate * external_isolate)3845 void i::Internals::CheckInitializedImpl(v8::Isolate* external_isolate) {
3846   i::Isolate* isolate = reinterpret_cast<i::Isolate*>(external_isolate);
3847   Utils::ApiCheck(isolate != nullptr && !isolate->IsDead(),
3848                   "v8::internal::Internals::CheckInitialized",
3849                   "Isolate is not initialized or V8 has died");
3850 }
3851 
CheckCast(Data * that)3852 void v8::Value::CheckCast(Data* that) {
3853   Utils::ApiCheck(that->IsValue(), "v8::Value::Cast", "Data is not a Value");
3854 }
3855 
CheckCast(v8::Value * that)3856 void External::CheckCast(v8::Value* that) {
3857   Utils::ApiCheck(that->IsExternal(), "v8::External::Cast",
3858                   "Value is not an External");
3859 }
3860 
CheckCast(Value * that)3861 void v8::Object::CheckCast(Value* that) {
3862   i::Handle<i::Object> obj = Utils::OpenHandle(that);
3863   Utils::ApiCheck(obj->IsJSReceiver(), "v8::Object::Cast",
3864                   "Value is not an Object");
3865 }
3866 
CheckCast(Value * that)3867 void v8::Function::CheckCast(Value* that) {
3868   i::Handle<i::Object> obj = Utils::OpenHandle(that);
3869   Utils::ApiCheck(obj->IsCallable(), "v8::Function::Cast",
3870                   "Value is not a Function");
3871 }
3872 
CheckCast(v8::Data * that)3873 void v8::Boolean::CheckCast(v8::Data* that) {
3874   i::Handle<i::Object> obj = Utils::OpenHandle(that);
3875   Utils::ApiCheck(obj->IsBoolean(), "v8::Boolean::Cast",
3876                   "Value is not a Boolean");
3877 }
3878 
CheckCast(v8::Data * that)3879 void v8::Name::CheckCast(v8::Data* that) {
3880   i::Handle<i::Object> obj = Utils::OpenHandle(that);
3881   Utils::ApiCheck(obj->IsName(), "v8::Name::Cast", "Value is not a Name");
3882 }
3883 
CheckCast(v8::Data * that)3884 void v8::String::CheckCast(v8::Data* that) {
3885   i::Handle<i::Object> obj = Utils::OpenHandle(that);
3886   Utils::ApiCheck(obj->IsString(), "v8::String::Cast", "Value is not a String");
3887 }
3888 
CheckCast(v8::Data * that)3889 void v8::Symbol::CheckCast(v8::Data* that) {
3890   i::Handle<i::Object> obj = Utils::OpenHandle(that);
3891   Utils::ApiCheck(obj->IsSymbol(), "v8::Symbol::Cast", "Value is not a Symbol");
3892 }
3893 
CheckCast(v8::Data * that)3894 void v8::Private::CheckCast(v8::Data* that) {
3895   i::Handle<i::Object> obj = Utils::OpenHandle(that);
3896   Utils::ApiCheck(
3897       obj->IsSymbol() && i::Handle<i::Symbol>::cast(obj)->is_private(),
3898       "v8::Private::Cast", "Value is not a Private");
3899 }
3900 
CheckCast(v8::Data * that)3901 void v8::FixedArray::CheckCast(v8::Data* that) {
3902   i::Handle<i::Object> obj = Utils::OpenHandle(that);
3903   Utils::ApiCheck(obj->IsFixedArray(), "v8::FixedArray::Cast",
3904                   "Value is not a FixedArray");
3905 }
3906 
CheckCast(v8::Data * that)3907 void v8::ModuleRequest::CheckCast(v8::Data* that) {
3908   i::Handle<i::Object> obj = Utils::OpenHandle(that);
3909   Utils::ApiCheck(obj->IsModuleRequest(), "v8::ModuleRequest::Cast",
3910                   "Value is not a ModuleRequest");
3911 }
3912 
CheckCast(v8::Data * that)3913 void v8::Module::CheckCast(v8::Data* that) {
3914   i::Handle<i::Object> obj = Utils::OpenHandle(that);
3915   Utils::ApiCheck(obj->IsModule(), "v8::Module::Cast", "Value is not a Module");
3916 }
3917 
CheckCast(v8::Data * that)3918 void v8::Number::CheckCast(v8::Data* that) {
3919   i::Handle<i::Object> obj = Utils::OpenHandle(that);
3920   Utils::ApiCheck(obj->IsNumber(), "v8::Number::Cast()",
3921                   "Value is not a Number");
3922 }
3923 
CheckCast(v8::Data * that)3924 void v8::Integer::CheckCast(v8::Data* that) {
3925   i::Handle<i::Object> obj = Utils::OpenHandle(that);
3926   Utils::ApiCheck(obj->IsNumber(), "v8::Integer::Cast",
3927                   "Value is not an Integer");
3928 }
3929 
CheckCast(v8::Data * that)3930 void v8::Int32::CheckCast(v8::Data* that) {
3931   Utils::ApiCheck(Value::Cast(that)->IsInt32(), "v8::Int32::Cast",
3932                   "Value is not a 32-bit signed integer");
3933 }
3934 
CheckCast(v8::Data * that)3935 void v8::Uint32::CheckCast(v8::Data* that) {
3936   Utils::ApiCheck(Value::Cast(that)->IsUint32(), "v8::Uint32::Cast",
3937                   "Value is not a 32-bit unsigned integer");
3938 }
3939 
CheckCast(v8::Data * that)3940 void v8::BigInt::CheckCast(v8::Data* that) {
3941   Utils::ApiCheck(Value::Cast(that)->IsBigInt(), "v8::BigInt::Cast",
3942                   "Value is not a BigInt");
3943 }
3944 
CheckCast(v8::Data * that)3945 void v8::Context::CheckCast(v8::Data* that) {
3946   i::Handle<i::Object> obj = Utils::OpenHandle(that);
3947   Utils::ApiCheck(obj->IsContext(), "v8::Context::Cast",
3948                   "Value is not a Context");
3949 }
3950 
CheckCast(Value * that)3951 void v8::Array::CheckCast(Value* that) {
3952   i::Handle<i::Object> obj = Utils::OpenHandle(that);
3953   Utils::ApiCheck(obj->IsJSArray(), "v8::Array::Cast", "Value is not an Array");
3954 }
3955 
CheckCast(Value * that)3956 void v8::Map::CheckCast(Value* that) {
3957   i::Handle<i::Object> obj = Utils::OpenHandle(that);
3958   Utils::ApiCheck(obj->IsJSMap(), "v8::Map::Cast", "Value is not a Map");
3959 }
3960 
CheckCast(Value * that)3961 void v8::Set::CheckCast(Value* that) {
3962   i::Handle<i::Object> obj = Utils::OpenHandle(that);
3963   Utils::ApiCheck(obj->IsJSSet(), "v8_Set_Cast", "Value is not a Set");
3964 }
3965 
CheckCast(Value * that)3966 void v8::Promise::CheckCast(Value* that) {
3967   Utils::ApiCheck(that->IsPromise(), "v8::Promise::Cast",
3968                   "Value is not a Promise");
3969 }
3970 
CheckCast(Value * that)3971 void v8::Promise::Resolver::CheckCast(Value* that) {
3972   Utils::ApiCheck(that->IsPromise(), "v8::Promise::Resolver::Cast",
3973                   "Value is not a Promise::Resolver");
3974 }
3975 
CheckCast(Value * that)3976 void v8::Proxy::CheckCast(Value* that) {
3977   Utils::ApiCheck(that->IsProxy(), "v8::Proxy::Cast", "Value is not a Proxy");
3978 }
3979 
CheckCast(Value * that)3980 void v8::WasmMemoryObject::CheckCast(Value* that) {
3981   Utils::ApiCheck(that->IsWasmMemoryObject(), "v8::WasmMemoryObject::Cast",
3982                   "Value is not a WasmMemoryObject");
3983 }
3984 
CheckCast(Value * that)3985 void v8::WasmModuleObject::CheckCast(Value* that) {
3986   Utils::ApiCheck(that->IsWasmModuleObject(), "v8::WasmModuleObject::Cast",
3987                   "Value is not a WasmModuleObject");
3988 }
3989 
~BackingStore()3990 v8::BackingStore::~BackingStore() {
3991   auto i_this = reinterpret_cast<const i::BackingStore*>(this);
3992   i_this->~BackingStore();  // manually call internal destructor
3993 }
3994 
Data() const3995 void* v8::BackingStore::Data() const {
3996   return reinterpret_cast<const i::BackingStore*>(this)->buffer_start();
3997 }
3998 
ByteLength() const3999 size_t v8::BackingStore::ByteLength() const {
4000   return reinterpret_cast<const i::BackingStore*>(this)->byte_length();
4001 }
4002 
IsShared() const4003 bool v8::BackingStore::IsShared() const {
4004   return reinterpret_cast<const i::BackingStore*>(this)->is_shared();
4005 }
4006 
4007 // static
Reallocate(v8::Isolate * isolate,std::unique_ptr<v8::BackingStore> backing_store,size_t byte_length)4008 std::unique_ptr<v8::BackingStore> v8::BackingStore::Reallocate(
4009     v8::Isolate* isolate, std::unique_ptr<v8::BackingStore> backing_store,
4010     size_t byte_length) {
4011   i::Isolate* i_isolate = reinterpret_cast<i::Isolate*>(isolate);
4012   API_RCS_SCOPE(i_isolate, ArrayBuffer, BackingStore_Reallocate);
4013   Utils::ApiCheck(byte_length <= i::JSArrayBuffer::kMaxByteLength,
4014                   "v8::BackingStore::Reallocate", "byte_lenght is too large");
4015   ENTER_V8_NO_SCRIPT_NO_EXCEPTION(i_isolate);
4016   i::BackingStore* i_backing_store =
4017       reinterpret_cast<i::BackingStore*>(backing_store.get());
4018   if (!i_backing_store->Reallocate(i_isolate, byte_length)) {
4019     i::FatalProcessOutOfMemory(i_isolate, "v8::BackingStore::Reallocate");
4020   }
4021   return backing_store;
4022 }
4023 
4024 // static
EmptyDeleter(void * data,size_t length,void * deleter_data)4025 void v8::BackingStore::EmptyDeleter(void* data, size_t length,
4026                                     void* deleter_data) {
4027   DCHECK_NULL(deleter_data);
4028 }
4029 
GetBackingStore()4030 std::shared_ptr<v8::BackingStore> v8::ArrayBuffer::GetBackingStore() {
4031   i::Handle<i::JSArrayBuffer> self = Utils::OpenHandle(this);
4032   std::shared_ptr<i::BackingStore> backing_store = self->GetBackingStore();
4033   if (!backing_store) {
4034     backing_store =
4035         i::BackingStore::EmptyBackingStore(i::SharedFlag::kNotShared);
4036   }
4037   std::shared_ptr<i::BackingStoreBase> bs_base = backing_store;
4038   return std::static_pointer_cast<v8::BackingStore>(bs_base);
4039 }
4040 
Data() const4041 void* v8::ArrayBuffer::Data() const {
4042   i::Handle<i::JSArrayBuffer> self = Utils::OpenHandle(this);
4043   return self->backing_store();
4044 }
4045 
GetBackingStore()4046 std::shared_ptr<v8::BackingStore> v8::SharedArrayBuffer::GetBackingStore() {
4047   i::Handle<i::JSArrayBuffer> self = Utils::OpenHandle(this);
4048   std::shared_ptr<i::BackingStore> backing_store = self->GetBackingStore();
4049   if (!backing_store) {
4050     backing_store = i::BackingStore::EmptyBackingStore(i::SharedFlag::kShared);
4051   }
4052   std::shared_ptr<i::BackingStoreBase> bs_base = backing_store;
4053   return std::static_pointer_cast<v8::BackingStore>(bs_base);
4054 }
4055 
Data() const4056 void* v8::SharedArrayBuffer::Data() const {
4057   i::Handle<i::JSArrayBuffer> self = Utils::OpenHandle(this);
4058   return self->backing_store();
4059 }
4060 
CheckCast(Value * that)4061 void v8::ArrayBuffer::CheckCast(Value* that) {
4062   i::Handle<i::Object> obj = Utils::OpenHandle(that);
4063   Utils::ApiCheck(
4064       obj->IsJSArrayBuffer() && !i::JSArrayBuffer::cast(*obj).is_shared(),
4065       "v8::ArrayBuffer::Cast()", "Value is not an ArrayBuffer");
4066 }
4067 
CheckCast(Value * that)4068 void v8::ArrayBufferView::CheckCast(Value* that) {
4069   i::Handle<i::Object> obj = Utils::OpenHandle(that);
4070   Utils::ApiCheck(obj->IsJSArrayBufferView(), "v8::ArrayBufferView::Cast()",
4071                   "Value is not an ArrayBufferView");
4072 }
4073 
4074 constexpr size_t v8::TypedArray::kMaxLength;
4075 
CheckCast(Value * that)4076 void v8::TypedArray::CheckCast(Value* that) {
4077   i::Handle<i::Object> obj = Utils::OpenHandle(that);
4078   Utils::ApiCheck(obj->IsJSTypedArray(), "v8::TypedArray::Cast()",
4079                   "Value is not a TypedArray");
4080 }
4081 
4082 #define CHECK_TYPED_ARRAY_CAST(Type, typeName, TYPE, ctype)                  \
4083   void v8::Type##Array::CheckCast(Value* that) {                             \
4084     i::Handle<i::Object> obj = Utils::OpenHandle(that);                      \
4085     Utils::ApiCheck(                                                         \
4086         obj->IsJSTypedArray() &&                                             \
4087             i::JSTypedArray::cast(*obj).type() == i::kExternal##Type##Array, \
4088         "v8::" #Type "Array::Cast()", "Value is not a " #Type "Array");      \
4089   }
4090 
TYPED_ARRAYS(CHECK_TYPED_ARRAY_CAST)4091 TYPED_ARRAYS(CHECK_TYPED_ARRAY_CAST)
4092 
4093 #undef CHECK_TYPED_ARRAY_CAST
4094 
4095 void v8::DataView::CheckCast(Value* that) {
4096   i::Handle<i::Object> obj = Utils::OpenHandle(that);
4097   Utils::ApiCheck(obj->IsJSDataView(), "v8::DataView::Cast()",
4098                   "Value is not a DataView");
4099 }
4100 
CheckCast(Value * that)4101 void v8::SharedArrayBuffer::CheckCast(Value* that) {
4102   i::Handle<i::Object> obj = Utils::OpenHandle(that);
4103   Utils::ApiCheck(
4104       obj->IsJSArrayBuffer() && i::JSArrayBuffer::cast(*obj).is_shared(),
4105       "v8::SharedArrayBuffer::Cast()", "Value is not a SharedArrayBuffer");
4106 }
4107 
CheckCast(v8::Value * that)4108 void v8::Date::CheckCast(v8::Value* that) {
4109   i::Handle<i::Object> obj = Utils::OpenHandle(that);
4110   Utils::ApiCheck(obj->IsJSDate(), "v8::Date::Cast()", "Value is not a Date");
4111 }
4112 
CheckCast(v8::Value * that)4113 void v8::StringObject::CheckCast(v8::Value* that) {
4114   i::Handle<i::Object> obj = Utils::OpenHandle(that);
4115   Utils::ApiCheck(obj->IsStringWrapper(), "v8::StringObject::Cast()",
4116                   "Value is not a StringObject");
4117 }
4118 
CheckCast(v8::Value * that)4119 void v8::SymbolObject::CheckCast(v8::Value* that) {
4120   i::Handle<i::Object> obj = Utils::OpenHandle(that);
4121   Utils::ApiCheck(obj->IsSymbolWrapper(), "v8::SymbolObject::Cast()",
4122                   "Value is not a SymbolObject");
4123 }
4124 
CheckCast(v8::Value * that)4125 void v8::NumberObject::CheckCast(v8::Value* that) {
4126   i::Handle<i::Object> obj = Utils::OpenHandle(that);
4127   Utils::ApiCheck(obj->IsNumberWrapper(), "v8::NumberObject::Cast()",
4128                   "Value is not a NumberObject");
4129 }
4130 
CheckCast(v8::Value * that)4131 void v8::BigIntObject::CheckCast(v8::Value* that) {
4132   i::Handle<i::Object> obj = Utils::OpenHandle(that);
4133   Utils::ApiCheck(obj->IsBigIntWrapper(), "v8::BigIntObject::Cast()",
4134                   "Value is not a BigIntObject");
4135 }
4136 
CheckCast(v8::Value * that)4137 void v8::BooleanObject::CheckCast(v8::Value* that) {
4138   i::Handle<i::Object> obj = Utils::OpenHandle(that);
4139   Utils::ApiCheck(obj->IsBooleanWrapper(), "v8::BooleanObject::Cast()",
4140                   "Value is not a BooleanObject");
4141 }
4142 
CheckCast(v8::Value * that)4143 void v8::RegExp::CheckCast(v8::Value* that) {
4144   i::Handle<i::Object> obj = Utils::OpenHandle(that);
4145   Utils::ApiCheck(obj->IsJSRegExp(), "v8::RegExp::Cast()",
4146                   "Value is not a RegExp");
4147 }
4148 
NumberValue(Local<Context> context) const4149 Maybe<double> Value::NumberValue(Local<Context> context) const {
4150   auto obj = Utils::OpenHandle(this);
4151   if (obj->IsNumber()) return Just(obj->Number());
4152   auto isolate = reinterpret_cast<i::Isolate*>(context->GetIsolate());
4153   ENTER_V8(isolate, context, Value, NumberValue, Nothing<double>(),
4154            i::HandleScope);
4155   i::Handle<i::Object> num;
4156   has_pending_exception = !i::Object::ToNumber(isolate, obj).ToHandle(&num);
4157   RETURN_ON_FAILED_EXECUTION_PRIMITIVE(double);
4158   return Just(num->Number());
4159 }
4160 
IntegerValue(Local<Context> context) const4161 Maybe<int64_t> Value::IntegerValue(Local<Context> context) const {
4162   auto obj = Utils::OpenHandle(this);
4163   if (obj->IsNumber()) {
4164     return Just(NumberToInt64(*obj));
4165   }
4166   auto isolate = reinterpret_cast<i::Isolate*>(context->GetIsolate());
4167   ENTER_V8(isolate, context, Value, IntegerValue, Nothing<int64_t>(),
4168            i::HandleScope);
4169   i::Handle<i::Object> num;
4170   has_pending_exception = !i::Object::ToInteger(isolate, obj).ToHandle(&num);
4171   RETURN_ON_FAILED_EXECUTION_PRIMITIVE(int64_t);
4172   return Just(NumberToInt64(*num));
4173 }
4174 
Int32Value(Local<Context> context) const4175 Maybe<int32_t> Value::Int32Value(Local<Context> context) const {
4176   auto obj = Utils::OpenHandle(this);
4177   if (obj->IsNumber()) return Just(NumberToInt32(*obj));
4178   auto isolate = reinterpret_cast<i::Isolate*>(context->GetIsolate());
4179   ENTER_V8(isolate, context, Value, Int32Value, Nothing<int32_t>(),
4180            i::HandleScope);
4181   i::Handle<i::Object> num;
4182   has_pending_exception = !i::Object::ToInt32(isolate, obj).ToHandle(&num);
4183   RETURN_ON_FAILED_EXECUTION_PRIMITIVE(int32_t);
4184   return Just(num->IsSmi() ? i::Smi::ToInt(*num)
4185                            : static_cast<int32_t>(num->Number()));
4186 }
4187 
Uint32Value(Local<Context> context) const4188 Maybe<uint32_t> Value::Uint32Value(Local<Context> context) const {
4189   auto obj = Utils::OpenHandle(this);
4190   if (obj->IsNumber()) return Just(NumberToUint32(*obj));
4191   auto isolate = reinterpret_cast<i::Isolate*>(context->GetIsolate());
4192   ENTER_V8(isolate, context, Value, Uint32Value, Nothing<uint32_t>(),
4193            i::HandleScope);
4194   i::Handle<i::Object> num;
4195   has_pending_exception = !i::Object::ToUint32(isolate, obj).ToHandle(&num);
4196   RETURN_ON_FAILED_EXECUTION_PRIMITIVE(uint32_t);
4197   return Just(num->IsSmi() ? static_cast<uint32_t>(i::Smi::ToInt(*num))
4198                            : static_cast<uint32_t>(num->Number()));
4199 }
4200 
ToArrayIndex(Local<Context> context) const4201 MaybeLocal<Uint32> Value::ToArrayIndex(Local<Context> context) const {
4202   auto self = Utils::OpenHandle(this);
4203   if (self->IsSmi()) {
4204     if (i::Smi::ToInt(*self) >= 0) return Utils::Uint32ToLocal(self);
4205     return Local<Uint32>();
4206   }
4207   PREPARE_FOR_EXECUTION(context, Object, ToArrayIndex, Uint32);
4208   i::Handle<i::Object> string_obj;
4209   has_pending_exception =
4210       !i::Object::ToString(isolate, self).ToHandle(&string_obj);
4211   RETURN_ON_FAILED_EXECUTION(Uint32);
4212   i::Handle<i::String> str = i::Handle<i::String>::cast(string_obj);
4213   uint32_t index;
4214   if (str->AsArrayIndex(&index)) {
4215     i::Handle<i::Object> value;
4216     if (index <= static_cast<uint32_t>(i::Smi::kMaxValue)) {
4217       value = i::Handle<i::Object>(i::Smi::FromInt(index), isolate);
4218     } else {
4219       value = isolate->factory()->NewNumber(index);
4220     }
4221     RETURN_ESCAPED(Utils::Uint32ToLocal(value));
4222   }
4223   return Local<Uint32>();
4224 }
4225 
Equals(Local<Context> context,Local<Value> that) const4226 Maybe<bool> Value::Equals(Local<Context> context, Local<Value> that) const {
4227   i::Isolate* isolate = Utils::OpenHandle(*context)->GetIsolate();
4228   ENTER_V8(isolate, context, Value, Equals, Nothing<bool>(), i::HandleScope);
4229   auto self = Utils::OpenHandle(this);
4230   auto other = Utils::OpenHandle(*that);
4231   Maybe<bool> result = i::Object::Equals(isolate, self, other);
4232   has_pending_exception = result.IsNothing();
4233   RETURN_ON_FAILED_EXECUTION_PRIMITIVE(bool);
4234   return result;
4235 }
4236 
StrictEquals(Local<Value> that) const4237 bool Value::StrictEquals(Local<Value> that) const {
4238   auto self = Utils::OpenHandle(this);
4239   auto other = Utils::OpenHandle(*that);
4240   return self->StrictEquals(*other);
4241 }
4242 
SameValue(Local<Value> that) const4243 bool Value::SameValue(Local<Value> that) const {
4244   auto self = Utils::OpenHandle(this);
4245   auto other = Utils::OpenHandle(*that);
4246   return self->SameValue(*other);
4247 }
4248 
TypeOf(v8::Isolate * external_isolate)4249 Local<String> Value::TypeOf(v8::Isolate* external_isolate) {
4250   i::Isolate* isolate = reinterpret_cast<i::Isolate*>(external_isolate);
4251   ENTER_V8_NO_SCRIPT_NO_EXCEPTION(isolate);
4252   API_RCS_SCOPE(isolate, Value, TypeOf);
4253   return Utils::ToLocal(i::Object::TypeOf(isolate, Utils::OpenHandle(this)));
4254 }
4255 
InstanceOf(v8::Local<v8::Context> context,v8::Local<v8::Object> object)4256 Maybe<bool> Value::InstanceOf(v8::Local<v8::Context> context,
4257                               v8::Local<v8::Object> object) {
4258   auto isolate = reinterpret_cast<i::Isolate*>(context->GetIsolate());
4259   ENTER_V8(isolate, context, Value, InstanceOf, Nothing<bool>(),
4260            i::HandleScope);
4261   auto left = Utils::OpenHandle(this);
4262   auto right = Utils::OpenHandle(*object);
4263   i::Handle<i::Object> result;
4264   has_pending_exception =
4265       !i::Object::InstanceOf(isolate, left, right).ToHandle(&result);
4266   RETURN_ON_FAILED_EXECUTION_PRIMITIVE(bool);
4267   return Just(result->IsTrue(isolate));
4268 }
4269 
Set(v8::Local<v8::Context> context,v8::Local<Value> key,v8::Local<Value> value)4270 Maybe<bool> v8::Object::Set(v8::Local<v8::Context> context,
4271                             v8::Local<Value> key, v8::Local<Value> value) {
4272   auto isolate = reinterpret_cast<i::Isolate*>(context->GetIsolate());
4273   ENTER_V8(isolate, context, Object, Set, Nothing<bool>(), i::HandleScope);
4274   auto self = Utils::OpenHandle(this);
4275   auto key_obj = Utils::OpenHandle(*key);
4276   auto value_obj = Utils::OpenHandle(*value);
4277   has_pending_exception =
4278       i::Runtime::SetObjectProperty(isolate, self, key_obj, value_obj,
4279                                     i::StoreOrigin::kMaybeKeyed,
4280                                     Just(i::ShouldThrow::kDontThrow))
4281           .is_null();
4282   RETURN_ON_FAILED_EXECUTION_PRIMITIVE(bool);
4283   return Just(true);
4284 }
4285 
Set(v8::Local<v8::Context> context,uint32_t index,v8::Local<Value> value)4286 Maybe<bool> v8::Object::Set(v8::Local<v8::Context> context, uint32_t index,
4287                             v8::Local<Value> value) {
4288   auto isolate = reinterpret_cast<i::Isolate*>(context->GetIsolate());
4289   ENTER_V8(isolate, context, Object, Set, Nothing<bool>(), i::HandleScope);
4290   auto self = Utils::OpenHandle(this);
4291   auto value_obj = Utils::OpenHandle(*value);
4292   has_pending_exception = i::Object::SetElement(isolate, self, index, value_obj,
4293                                                 i::ShouldThrow::kDontThrow)
4294                               .is_null();
4295   RETURN_ON_FAILED_EXECUTION_PRIMITIVE(bool);
4296   return Just(true);
4297 }
4298 
CreateDataProperty(v8::Local<v8::Context> context,v8::Local<Name> key,v8::Local<Value> value)4299 Maybe<bool> v8::Object::CreateDataProperty(v8::Local<v8::Context> context,
4300                                            v8::Local<Name> key,
4301                                            v8::Local<Value> value) {
4302   auto isolate = reinterpret_cast<i::Isolate*>(context->GetIsolate());
4303   i::Handle<i::JSReceiver> self = Utils::OpenHandle(this);
4304   i::Handle<i::Name> key_obj = Utils::OpenHandle(*key);
4305   i::Handle<i::Object> value_obj = Utils::OpenHandle(*value);
4306 
4307   i::PropertyKey lookup_key(isolate, key_obj);
4308   i::LookupIterator it(isolate, self, lookup_key, i::LookupIterator::OWN);
4309   if (self->IsJSProxy()) {
4310     ENTER_V8(isolate, context, Object, CreateDataProperty, Nothing<bool>(),
4311              i::HandleScope);
4312     Maybe<bool> result =
4313         i::JSReceiver::CreateDataProperty(&it, value_obj, Just(i::kDontThrow));
4314     has_pending_exception = result.IsNothing();
4315     RETURN_ON_FAILED_EXECUTION_PRIMITIVE(bool);
4316     return result;
4317   } else {
4318     ENTER_V8_NO_SCRIPT(isolate, context, Object, CreateDataProperty,
4319                        Nothing<bool>(), i::HandleScope);
4320     Maybe<bool> result =
4321         i::JSObject::CreateDataProperty(&it, value_obj, Just(i::kDontThrow));
4322     has_pending_exception = result.IsNothing();
4323     RETURN_ON_FAILED_EXECUTION_PRIMITIVE(bool);
4324     return result;
4325   }
4326 }
4327 
CreateDataProperty(v8::Local<v8::Context> context,uint32_t index,v8::Local<Value> value)4328 Maybe<bool> v8::Object::CreateDataProperty(v8::Local<v8::Context> context,
4329                                            uint32_t index,
4330                                            v8::Local<Value> value) {
4331   auto isolate = reinterpret_cast<i::Isolate*>(context->GetIsolate());
4332   i::Handle<i::JSReceiver> self = Utils::OpenHandle(this);
4333   i::Handle<i::Object> value_obj = Utils::OpenHandle(*value);
4334 
4335   i::LookupIterator it(isolate, self, index, self, i::LookupIterator::OWN);
4336   if (self->IsJSProxy()) {
4337     ENTER_V8(isolate, context, Object, CreateDataProperty, Nothing<bool>(),
4338              i::HandleScope);
4339     Maybe<bool> result =
4340         i::JSReceiver::CreateDataProperty(&it, value_obj, Just(i::kDontThrow));
4341     has_pending_exception = result.IsNothing();
4342     RETURN_ON_FAILED_EXECUTION_PRIMITIVE(bool);
4343     return result;
4344   } else {
4345     ENTER_V8_NO_SCRIPT(isolate, context, Object, CreateDataProperty,
4346                        Nothing<bool>(), i::HandleScope);
4347     Maybe<bool> result =
4348         i::JSObject::CreateDataProperty(&it, value_obj, Just(i::kDontThrow));
4349     has_pending_exception = result.IsNothing();
4350     RETURN_ON_FAILED_EXECUTION_PRIMITIVE(bool);
4351     return result;
4352   }
4353 }
4354 
4355 struct v8::PropertyDescriptor::PrivateData {
PrivateDatav8::v8::PropertyDescriptor::PrivateData4356   PrivateData() : desc() {}
4357   i::PropertyDescriptor desc;
4358 };
4359 
PropertyDescriptor()4360 v8::PropertyDescriptor::PropertyDescriptor() : private_(new PrivateData()) {}
4361 
4362 // DataDescriptor
PropertyDescriptor(v8::Local<v8::Value> value)4363 v8::PropertyDescriptor::PropertyDescriptor(v8::Local<v8::Value> value)
4364     : private_(new PrivateData()) {
4365   private_->desc.set_value(Utils::OpenHandle(*value, true));
4366 }
4367 
4368 // DataDescriptor with writable field
PropertyDescriptor(v8::Local<v8::Value> value,bool writable)4369 v8::PropertyDescriptor::PropertyDescriptor(v8::Local<v8::Value> value,
4370                                            bool writable)
4371     : private_(new PrivateData()) {
4372   private_->desc.set_value(Utils::OpenHandle(*value, true));
4373   private_->desc.set_writable(writable);
4374 }
4375 
4376 // AccessorDescriptor
PropertyDescriptor(v8::Local<v8::Value> get,v8::Local<v8::Value> set)4377 v8::PropertyDescriptor::PropertyDescriptor(v8::Local<v8::Value> get,
4378                                            v8::Local<v8::Value> set)
4379     : private_(new PrivateData()) {
4380   DCHECK(get.IsEmpty() || get->IsUndefined() || get->IsFunction());
4381   DCHECK(set.IsEmpty() || set->IsUndefined() || set->IsFunction());
4382   private_->desc.set_get(Utils::OpenHandle(*get, true));
4383   private_->desc.set_set(Utils::OpenHandle(*set, true));
4384 }
4385 
~PropertyDescriptor()4386 v8::PropertyDescriptor::~PropertyDescriptor() { delete private_; }
4387 
value() const4388 v8::Local<Value> v8::PropertyDescriptor::value() const {
4389   DCHECK(private_->desc.has_value());
4390   return Utils::ToLocal(private_->desc.value());
4391 }
4392 
get() const4393 v8::Local<Value> v8::PropertyDescriptor::get() const {
4394   DCHECK(private_->desc.has_get());
4395   return Utils::ToLocal(private_->desc.get());
4396 }
4397 
set() const4398 v8::Local<Value> v8::PropertyDescriptor::set() const {
4399   DCHECK(private_->desc.has_set());
4400   return Utils::ToLocal(private_->desc.set());
4401 }
4402 
has_value() const4403 bool v8::PropertyDescriptor::has_value() const {
4404   return private_->desc.has_value();
4405 }
has_get() const4406 bool v8::PropertyDescriptor::has_get() const {
4407   return private_->desc.has_get();
4408 }
has_set() const4409 bool v8::PropertyDescriptor::has_set() const {
4410   return private_->desc.has_set();
4411 }
4412 
writable() const4413 bool v8::PropertyDescriptor::writable() const {
4414   DCHECK(private_->desc.has_writable());
4415   return private_->desc.writable();
4416 }
4417 
has_writable() const4418 bool v8::PropertyDescriptor::has_writable() const {
4419   return private_->desc.has_writable();
4420 }
4421 
set_enumerable(bool enumerable)4422 void v8::PropertyDescriptor::set_enumerable(bool enumerable) {
4423   private_->desc.set_enumerable(enumerable);
4424 }
4425 
enumerable() const4426 bool v8::PropertyDescriptor::enumerable() const {
4427   DCHECK(private_->desc.has_enumerable());
4428   return private_->desc.enumerable();
4429 }
4430 
has_enumerable() const4431 bool v8::PropertyDescriptor::has_enumerable() const {
4432   return private_->desc.has_enumerable();
4433 }
4434 
set_configurable(bool configurable)4435 void v8::PropertyDescriptor::set_configurable(bool configurable) {
4436   private_->desc.set_configurable(configurable);
4437 }
4438 
configurable() const4439 bool v8::PropertyDescriptor::configurable() const {
4440   DCHECK(private_->desc.has_configurable());
4441   return private_->desc.configurable();
4442 }
4443 
has_configurable() const4444 bool v8::PropertyDescriptor::has_configurable() const {
4445   return private_->desc.has_configurable();
4446 }
4447 
DefineOwnProperty(v8::Local<v8::Context> context,v8::Local<Name> key,v8::Local<Value> value,v8::PropertyAttribute attributes)4448 Maybe<bool> v8::Object::DefineOwnProperty(v8::Local<v8::Context> context,
4449                                           v8::Local<Name> key,
4450                                           v8::Local<Value> value,
4451                                           v8::PropertyAttribute attributes) {
4452   auto isolate = reinterpret_cast<i::Isolate*>(context->GetIsolate());
4453   i::Handle<i::JSReceiver> self = Utils::OpenHandle(this);
4454   i::Handle<i::Name> key_obj = Utils::OpenHandle(*key);
4455   i::Handle<i::Object> value_obj = Utils::OpenHandle(*value);
4456 
4457   i::PropertyDescriptor desc;
4458   desc.set_writable(!(attributes & v8::ReadOnly));
4459   desc.set_enumerable(!(attributes & v8::DontEnum));
4460   desc.set_configurable(!(attributes & v8::DontDelete));
4461   desc.set_value(value_obj);
4462 
4463   if (self->IsJSProxy()) {
4464     ENTER_V8(isolate, context, Object, DefineOwnProperty, Nothing<bool>(),
4465              i::HandleScope);
4466     Maybe<bool> success = i::JSReceiver::DefineOwnProperty(
4467         isolate, self, key_obj, &desc, Just(i::kDontThrow));
4468     // Even though we said kDontThrow, there might be accessors that do throw.
4469     RETURN_ON_FAILED_EXECUTION_PRIMITIVE(bool);
4470     return success;
4471   } else {
4472     // If it's not a JSProxy, i::JSReceiver::DefineOwnProperty should never run
4473     // a script.
4474     ENTER_V8_NO_SCRIPT(isolate, context, Object, DefineOwnProperty,
4475                        Nothing<bool>(), i::HandleScope);
4476     Maybe<bool> success = i::JSReceiver::DefineOwnProperty(
4477         isolate, self, key_obj, &desc, Just(i::kDontThrow));
4478     RETURN_ON_FAILED_EXECUTION_PRIMITIVE(bool);
4479     return success;
4480   }
4481 }
4482 
DefineProperty(v8::Local<v8::Context> context,v8::Local<Name> key,PropertyDescriptor & descriptor)4483 Maybe<bool> v8::Object::DefineProperty(v8::Local<v8::Context> context,
4484                                        v8::Local<Name> key,
4485                                        PropertyDescriptor& descriptor) {
4486   auto isolate = reinterpret_cast<i::Isolate*>(context->GetIsolate());
4487   ENTER_V8(isolate, context, Object, DefineOwnProperty, Nothing<bool>(),
4488            i::HandleScope);
4489   i::Handle<i::JSReceiver> self = Utils::OpenHandle(this);
4490   i::Handle<i::Name> key_obj = Utils::OpenHandle(*key);
4491 
4492   Maybe<bool> success = i::JSReceiver::DefineOwnProperty(
4493       isolate, self, key_obj, &descriptor.get_private()->desc,
4494       Just(i::kDontThrow));
4495   RETURN_ON_FAILED_EXECUTION_PRIMITIVE(bool);
4496   return success;
4497 }
4498 
SetPrivate(Local<Context> context,Local<Private> key,Local<Value> value)4499 Maybe<bool> v8::Object::SetPrivate(Local<Context> context, Local<Private> key,
4500                                    Local<Value> value) {
4501   auto isolate = reinterpret_cast<i::Isolate*>(context->GetIsolate());
4502   ENTER_V8_NO_SCRIPT(isolate, context, Object, SetPrivate, Nothing<bool>(),
4503                      i::HandleScope);
4504   auto self = Utils::OpenHandle(this);
4505   auto key_obj = Utils::OpenHandle(reinterpret_cast<Name*>(*key));
4506   auto value_obj = Utils::OpenHandle(*value);
4507   if (self->IsJSProxy()) {
4508     i::PropertyDescriptor desc;
4509     desc.set_writable(true);
4510     desc.set_enumerable(false);
4511     desc.set_configurable(true);
4512     desc.set_value(value_obj);
4513     return i::JSProxy::SetPrivateSymbol(
4514         isolate, i::Handle<i::JSProxy>::cast(self),
4515         i::Handle<i::Symbol>::cast(key_obj), &desc, Just(i::kDontThrow));
4516   }
4517   auto js_object = i::Handle<i::JSObject>::cast(self);
4518   i::LookupIterator it(isolate, js_object, key_obj, js_object);
4519   has_pending_exception = i::JSObject::DefineOwnPropertyIgnoreAttributes(
4520                               &it, value_obj, i::DONT_ENUM)
4521                               .is_null();
4522   RETURN_ON_FAILED_EXECUTION_PRIMITIVE(bool);
4523   return Just(true);
4524 }
4525 
Get(Local<v8::Context> context,Local<Value> key)4526 MaybeLocal<Value> v8::Object::Get(Local<v8::Context> context,
4527                                   Local<Value> key) {
4528   PREPARE_FOR_EXECUTION(context, Object, Get, Value);
4529   auto self = Utils::OpenHandle(this);
4530   auto key_obj = Utils::OpenHandle(*key);
4531   i::Handle<i::Object> result;
4532   has_pending_exception =
4533       !i::Runtime::GetObjectProperty(isolate, self, key_obj).ToHandle(&result);
4534   RETURN_ON_FAILED_EXECUTION(Value);
4535   RETURN_ESCAPED(Utils::ToLocal(result));
4536 }
4537 
Get(Local<Context> context,uint32_t index)4538 MaybeLocal<Value> v8::Object::Get(Local<Context> context, uint32_t index) {
4539   PREPARE_FOR_EXECUTION(context, Object, Get, Value);
4540   auto self = Utils::OpenHandle(this);
4541   i::Handle<i::Object> result;
4542   has_pending_exception =
4543       !i::JSReceiver::GetElement(isolate, self, index).ToHandle(&result);
4544   RETURN_ON_FAILED_EXECUTION(Value);
4545   RETURN_ESCAPED(Utils::ToLocal(result));
4546 }
4547 
GetPrivate(Local<Context> context,Local<Private> key)4548 MaybeLocal<Value> v8::Object::GetPrivate(Local<Context> context,
4549                                          Local<Private> key) {
4550   return Get(context, Local<Value>(reinterpret_cast<Value*>(*key)));
4551 }
4552 
GetPropertyAttributes(Local<Context> context,Local<Value> key)4553 Maybe<PropertyAttribute> v8::Object::GetPropertyAttributes(
4554     Local<Context> context, Local<Value> key) {
4555   auto isolate = reinterpret_cast<i::Isolate*>(context->GetIsolate());
4556   ENTER_V8(isolate, context, Object, GetPropertyAttributes,
4557            Nothing<PropertyAttribute>(), i::HandleScope);
4558   auto self = Utils::OpenHandle(this);
4559   auto key_obj = Utils::OpenHandle(*key);
4560   if (!key_obj->IsName()) {
4561     has_pending_exception =
4562         !i::Object::ToString(isolate, key_obj).ToHandle(&key_obj);
4563     RETURN_ON_FAILED_EXECUTION_PRIMITIVE(PropertyAttribute);
4564   }
4565   auto key_name = i::Handle<i::Name>::cast(key_obj);
4566   auto result = i::JSReceiver::GetPropertyAttributes(self, key_name);
4567   has_pending_exception = result.IsNothing();
4568   RETURN_ON_FAILED_EXECUTION_PRIMITIVE(PropertyAttribute);
4569   if (result.FromJust() == i::ABSENT) {
4570     return Just(static_cast<PropertyAttribute>(i::NONE));
4571   }
4572   return Just(static_cast<PropertyAttribute>(result.FromJust()));
4573 }
4574 
GetOwnPropertyDescriptor(Local<Context> context,Local<Name> key)4575 MaybeLocal<Value> v8::Object::GetOwnPropertyDescriptor(Local<Context> context,
4576                                                        Local<Name> key) {
4577   PREPARE_FOR_EXECUTION(context, Object, GetOwnPropertyDescriptor, Value);
4578   i::Handle<i::JSReceiver> obj = Utils::OpenHandle(this);
4579   i::Handle<i::Name> key_name = Utils::OpenHandle(*key);
4580 
4581   i::PropertyDescriptor desc;
4582   Maybe<bool> found =
4583       i::JSReceiver::GetOwnPropertyDescriptor(isolate, obj, key_name, &desc);
4584   has_pending_exception = found.IsNothing();
4585   RETURN_ON_FAILED_EXECUTION(Value);
4586   if (!found.FromJust()) {
4587     return v8::Undefined(reinterpret_cast<v8::Isolate*>(isolate));
4588   }
4589   RETURN_ESCAPED(Utils::ToLocal(desc.ToObject(isolate)));
4590 }
4591 
GetPrototype()4592 Local<Value> v8::Object::GetPrototype() {
4593   auto self = Utils::OpenHandle(this);
4594   auto isolate = self->GetIsolate();
4595   i::PrototypeIterator iter(isolate, self);
4596   return Utils::ToLocal(i::PrototypeIterator::GetCurrent(iter));
4597 }
4598 
SetPrototype(Local<Context> context,Local<Value> value)4599 Maybe<bool> v8::Object::SetPrototype(Local<Context> context,
4600                                      Local<Value> value) {
4601   auto isolate = reinterpret_cast<i::Isolate*>(context->GetIsolate());
4602   auto self = Utils::OpenHandle(this);
4603   auto value_obj = Utils::OpenHandle(*value);
4604   if (self->IsJSProxy()) {
4605     ENTER_V8(isolate, context, Object, SetPrototype, Nothing<bool>(),
4606              i::HandleScope);
4607     // We do not allow exceptions thrown while setting the prototype
4608     // to propagate outside.
4609     TryCatch try_catch(reinterpret_cast<v8::Isolate*>(isolate));
4610     auto result =
4611         i::JSProxy::SetPrototype(isolate, i::Handle<i::JSProxy>::cast(self),
4612                                  value_obj, false, i::kThrowOnError);
4613     has_pending_exception = result.IsNothing();
4614     RETURN_ON_FAILED_EXECUTION_PRIMITIVE(bool);
4615   } else {
4616     ENTER_V8_NO_SCRIPT_NO_EXCEPTION(isolate);
4617     auto result =
4618         i::JSObject::SetPrototype(isolate, i::Handle<i::JSObject>::cast(self),
4619                                   value_obj, false, i::kThrowOnError);
4620     if (result.IsNothing()) {
4621       isolate->clear_pending_exception();
4622       return Nothing<bool>();
4623     }
4624   }
4625   return Just(true);
4626 }
4627 
FindInstanceInPrototypeChain(v8::Local<FunctionTemplate> tmpl)4628 Local<Object> v8::Object::FindInstanceInPrototypeChain(
4629     v8::Local<FunctionTemplate> tmpl) {
4630   auto self = Utils::OpenHandle(this);
4631   auto isolate = self->GetIsolate();
4632   i::PrototypeIterator iter(isolate, *self, i::kStartAtReceiver);
4633   auto tmpl_info = *Utils::OpenHandle(*tmpl);
4634   while (!tmpl_info.IsTemplateFor(iter.GetCurrent<i::JSObject>())) {
4635     iter.Advance();
4636     if (iter.IsAtEnd()) return Local<Object>();
4637     if (!iter.GetCurrent().IsJSObject()) return Local<Object>();
4638   }
4639   // IsTemplateFor() ensures that iter.GetCurrent() can't be a Proxy here.
4640   return Utils::ToLocal(i::handle(iter.GetCurrent<i::JSObject>(), isolate));
4641 }
4642 
GetPropertyNames(Local<Context> context)4643 MaybeLocal<Array> v8::Object::GetPropertyNames(Local<Context> context) {
4644   return GetPropertyNames(
4645       context, v8::KeyCollectionMode::kIncludePrototypes,
4646       static_cast<v8::PropertyFilter>(ONLY_ENUMERABLE | SKIP_SYMBOLS),
4647       v8::IndexFilter::kIncludeIndices);
4648 }
4649 
GetPropertyNames(Local<Context> context,KeyCollectionMode mode,PropertyFilter property_filter,IndexFilter index_filter,KeyConversionMode key_conversion)4650 MaybeLocal<Array> v8::Object::GetPropertyNames(
4651     Local<Context> context, KeyCollectionMode mode,
4652     PropertyFilter property_filter, IndexFilter index_filter,
4653     KeyConversionMode key_conversion) {
4654   PREPARE_FOR_EXECUTION(context, Object, GetPropertyNames, Array);
4655   auto self = Utils::OpenHandle(this);
4656   i::Handle<i::FixedArray> value;
4657   i::KeyAccumulator accumulator(
4658       isolate, static_cast<i::KeyCollectionMode>(mode),
4659       static_cast<i::PropertyFilter>(property_filter));
4660   accumulator.set_skip_indices(index_filter == IndexFilter::kSkipIndices);
4661   has_pending_exception = accumulator.CollectKeys(self, self).IsNothing();
4662   RETURN_ON_FAILED_EXECUTION(Array);
4663   value =
4664       accumulator.GetKeys(static_cast<i::GetKeysConversion>(key_conversion));
4665   DCHECK(self->map().EnumLength() == i::kInvalidEnumCacheSentinel ||
4666          self->map().EnumLength() == 0 ||
4667          self->map().instance_descriptors(isolate).enum_cache().keys() !=
4668              *value);
4669   auto result = isolate->factory()->NewJSArrayWithElements(value);
4670   RETURN_ESCAPED(Utils::ToLocal(result));
4671 }
4672 
GetOwnPropertyNames(Local<Context> context)4673 MaybeLocal<Array> v8::Object::GetOwnPropertyNames(Local<Context> context) {
4674   return GetOwnPropertyNames(
4675       context, static_cast<v8::PropertyFilter>(ONLY_ENUMERABLE | SKIP_SYMBOLS));
4676 }
4677 
GetOwnPropertyNames(Local<Context> context,PropertyFilter filter,KeyConversionMode key_conversion)4678 MaybeLocal<Array> v8::Object::GetOwnPropertyNames(
4679     Local<Context> context, PropertyFilter filter,
4680     KeyConversionMode key_conversion) {
4681   return GetPropertyNames(context, KeyCollectionMode::kOwnOnly, filter,
4682                           v8::IndexFilter::kIncludeIndices, key_conversion);
4683 }
4684 
ObjectProtoToString(Local<Context> context)4685 MaybeLocal<String> v8::Object::ObjectProtoToString(Local<Context> context) {
4686   PREPARE_FOR_EXECUTION(context, Object, ObjectProtoToString, String);
4687   auto self = Utils::OpenHandle(this);
4688   Local<Value> result;
4689   has_pending_exception = !ToLocal<Value>(
4690       i::Execution::CallBuiltin(isolate, isolate->object_to_string(), self, 0,
4691                                 nullptr),
4692       &result);
4693   RETURN_ON_FAILED_EXECUTION(String);
4694   RETURN_ESCAPED(Local<String>::Cast(result));
4695 }
4696 
GetConstructorName()4697 Local<String> v8::Object::GetConstructorName() {
4698   auto self = Utils::OpenHandle(this);
4699   // TODO(v8:12547): Support shared objects.
4700   DCHECK(!self->InSharedHeap());
4701   i::Handle<i::String> name =
4702       i::JSReceiver::GetConstructorName(self->GetIsolate(), self);
4703   return Utils::ToLocal(name);
4704 }
4705 
SetIntegrityLevel(Local<Context> context,IntegrityLevel level)4706 Maybe<bool> v8::Object::SetIntegrityLevel(Local<Context> context,
4707                                           IntegrityLevel level) {
4708   auto isolate = reinterpret_cast<i::Isolate*>(context->GetIsolate());
4709   ENTER_V8(isolate, context, Object, SetIntegrityLevel, Nothing<bool>(),
4710            i::HandleScope);
4711   auto self = Utils::OpenHandle(this);
4712   i::JSReceiver::IntegrityLevel i_level =
4713       level == IntegrityLevel::kFrozen ? i::FROZEN : i::SEALED;
4714   Maybe<bool> result =
4715       i::JSReceiver::SetIntegrityLevel(self, i_level, i::kThrowOnError);
4716   has_pending_exception = result.IsNothing();
4717   RETURN_ON_FAILED_EXECUTION_PRIMITIVE(bool);
4718   return result;
4719 }
4720 
Delete(Local<Context> context,Local<Value> key)4721 Maybe<bool> v8::Object::Delete(Local<Context> context, Local<Value> key) {
4722   auto isolate = reinterpret_cast<i::Isolate*>(context->GetIsolate());
4723   auto self = Utils::OpenHandle(this);
4724   auto key_obj = Utils::OpenHandle(*key);
4725   if (self->IsJSProxy()) {
4726     ENTER_V8(isolate, context, Object, Delete, Nothing<bool>(), i::HandleScope);
4727     Maybe<bool> result = i::Runtime::DeleteObjectProperty(
4728         isolate, self, key_obj, i::LanguageMode::kSloppy);
4729     has_pending_exception = result.IsNothing();
4730     RETURN_ON_FAILED_EXECUTION_PRIMITIVE(bool);
4731     return result;
4732   } else {
4733     // If it's not a JSProxy, i::Runtime::DeleteObjectProperty should never run
4734     // a script.
4735     ENTER_V8_NO_SCRIPT(isolate, context, Object, Delete, Nothing<bool>(),
4736                        i::HandleScope);
4737     Maybe<bool> result = i::Runtime::DeleteObjectProperty(
4738         isolate, self, key_obj, i::LanguageMode::kSloppy);
4739     has_pending_exception = result.IsNothing();
4740     RETURN_ON_FAILED_EXECUTION_PRIMITIVE(bool);
4741     return result;
4742   }
4743 }
4744 
DeletePrivate(Local<Context> context,Local<Private> key)4745 Maybe<bool> v8::Object::DeletePrivate(Local<Context> context,
4746                                       Local<Private> key) {
4747   auto isolate = reinterpret_cast<i::Isolate*>(context->GetIsolate());
4748   // In case of private symbols, i::Runtime::DeleteObjectProperty does not run
4749   // any author script.
4750   ENTER_V8_NO_SCRIPT(isolate, context, Object, Delete, Nothing<bool>(),
4751                      i::HandleScope);
4752   auto self = Utils::OpenHandle(this);
4753   auto key_obj = Utils::OpenHandle(*key);
4754   Maybe<bool> result = i::Runtime::DeleteObjectProperty(
4755       isolate, self, key_obj, i::LanguageMode::kSloppy);
4756   has_pending_exception = result.IsNothing();
4757   RETURN_ON_FAILED_EXECUTION_PRIMITIVE(bool);
4758   return result;
4759 }
4760 
Has(Local<Context> context,Local<Value> key)4761 Maybe<bool> v8::Object::Has(Local<Context> context, Local<Value> key) {
4762   auto isolate = reinterpret_cast<i::Isolate*>(context->GetIsolate());
4763   ENTER_V8(isolate, context, Object, Has, Nothing<bool>(), i::HandleScope);
4764   auto self = Utils::OpenHandle(this);
4765   auto key_obj = Utils::OpenHandle(*key);
4766   Maybe<bool> maybe = Nothing<bool>();
4767   // Check if the given key is an array index.
4768   uint32_t index = 0;
4769   if (key_obj->ToArrayIndex(&index)) {
4770     maybe = i::JSReceiver::HasElement(isolate, self, index);
4771   } else {
4772     // Convert the key to a name - possibly by calling back into JavaScript.
4773     i::Handle<i::Name> name;
4774     if (i::Object::ToName(isolate, key_obj).ToHandle(&name)) {
4775       maybe = i::JSReceiver::HasProperty(isolate, self, name);
4776     }
4777   }
4778   has_pending_exception = maybe.IsNothing();
4779   RETURN_ON_FAILED_EXECUTION_PRIMITIVE(bool);
4780   return maybe;
4781 }
4782 
HasPrivate(Local<Context> context,Local<Private> key)4783 Maybe<bool> v8::Object::HasPrivate(Local<Context> context, Local<Private> key) {
4784   return HasOwnProperty(context, Local<Name>(reinterpret_cast<Name*>(*key)));
4785 }
4786 
Delete(Local<Context> context,uint32_t index)4787 Maybe<bool> v8::Object::Delete(Local<Context> context, uint32_t index) {
4788   auto isolate = reinterpret_cast<i::Isolate*>(context->GetIsolate());
4789   ENTER_V8(isolate, context, Object, Delete, Nothing<bool>(), i::HandleScope);
4790   auto self = Utils::OpenHandle(this);
4791   Maybe<bool> result = i::JSReceiver::DeleteElement(self, index);
4792   has_pending_exception = result.IsNothing();
4793   RETURN_ON_FAILED_EXECUTION_PRIMITIVE(bool);
4794   return result;
4795 }
4796 
Has(Local<Context> context,uint32_t index)4797 Maybe<bool> v8::Object::Has(Local<Context> context, uint32_t index) {
4798   auto isolate = reinterpret_cast<i::Isolate*>(context->GetIsolate());
4799   ENTER_V8(isolate, context, Object, Has, Nothing<bool>(), i::HandleScope);
4800   auto self = Utils::OpenHandle(this);
4801   auto maybe = i::JSReceiver::HasElement(isolate, self, index);
4802   has_pending_exception = maybe.IsNothing();
4803   RETURN_ON_FAILED_EXECUTION_PRIMITIVE(bool);
4804   return maybe;
4805 }
4806 
4807 template <typename Getter, typename Setter, typename Data>
ObjectSetAccessor(Local<Context> context,Object * self,Local<Name> name,Getter getter,Setter setter,Data data,AccessControl settings,PropertyAttribute attributes,bool is_special_data_property,bool replace_on_access,SideEffectType getter_side_effect_type,SideEffectType setter_side_effect_type)4808 static Maybe<bool> ObjectSetAccessor(
4809     Local<Context> context, Object* self, Local<Name> name, Getter getter,
4810     Setter setter, Data data, AccessControl settings,
4811     PropertyAttribute attributes, bool is_special_data_property,
4812     bool replace_on_access, SideEffectType getter_side_effect_type,
4813     SideEffectType setter_side_effect_type) {
4814   auto isolate = reinterpret_cast<i::Isolate*>(context->GetIsolate());
4815   ENTER_V8_NO_SCRIPT(isolate, context, Object, SetAccessor, Nothing<bool>(),
4816                      i::HandleScope);
4817   if (!Utils::OpenHandle(self)->IsJSObject()) return Just(false);
4818   i::Handle<i::JSObject> obj =
4819       i::Handle<i::JSObject>::cast(Utils::OpenHandle(self));
4820   v8::Local<AccessorSignature> signature;
4821   i::Handle<i::AccessorInfo> info =
4822       MakeAccessorInfo(isolate, name, getter, setter, data, settings, signature,
4823                        is_special_data_property, replace_on_access);
4824   info->set_getter_side_effect_type(getter_side_effect_type);
4825   info->set_setter_side_effect_type(setter_side_effect_type);
4826   if (info.is_null()) return Nothing<bool>();
4827   bool fast = obj->HasFastProperties();
4828   i::Handle<i::Object> result;
4829 
4830   i::Handle<i::Name> accessor_name(info->name(), isolate);
4831   i::PropertyAttributes attrs = static_cast<i::PropertyAttributes>(attributes);
4832   has_pending_exception =
4833       !i::JSObject::SetAccessor(obj, accessor_name, info, attrs)
4834            .ToHandle(&result);
4835   RETURN_ON_FAILED_EXECUTION_PRIMITIVE(bool);
4836   if (result->IsUndefined(isolate)) return Just(false);
4837   if (fast) {
4838     i::JSObject::MigrateSlowToFast(obj, 0, "APISetAccessor");
4839   }
4840   return Just(true);
4841 }
4842 
SetAccessor(Local<Context> context,Local<Name> name,AccessorNameGetterCallback getter,AccessorNameSetterCallback setter,MaybeLocal<Value> data,AccessControl settings,PropertyAttribute attribute,SideEffectType getter_side_effect_type,SideEffectType setter_side_effect_type)4843 Maybe<bool> Object::SetAccessor(Local<Context> context, Local<Name> name,
4844                                 AccessorNameGetterCallback getter,
4845                                 AccessorNameSetterCallback setter,
4846                                 MaybeLocal<Value> data, AccessControl settings,
4847                                 PropertyAttribute attribute,
4848                                 SideEffectType getter_side_effect_type,
4849                                 SideEffectType setter_side_effect_type) {
4850   return ObjectSetAccessor(context, this, name, getter, setter,
4851                            data.FromMaybe(Local<Value>()), settings, attribute,
4852                            i::FLAG_disable_old_api_accessors, false,
4853                            getter_side_effect_type, setter_side_effect_type);
4854 }
4855 
SetAccessorProperty(Local<Name> name,Local<Function> getter,Local<Function> setter,PropertyAttribute attribute,AccessControl settings)4856 void Object::SetAccessorProperty(Local<Name> name, Local<Function> getter,
4857                                  Local<Function> setter,
4858                                  PropertyAttribute attribute,
4859                                  AccessControl settings) {
4860   // TODO(verwaest): Remove |settings|.
4861   DCHECK_EQ(v8::DEFAULT, settings);
4862   auto self = Utils::OpenHandle(this);
4863   i::Isolate* isolate = self->GetIsolate();
4864   ENTER_V8_NO_SCRIPT_NO_EXCEPTION(isolate);
4865   i::HandleScope scope(isolate);
4866   if (!self->IsJSObject()) return;
4867   i::Handle<i::Object> getter_i = v8::Utils::OpenHandle(*getter);
4868   i::Handle<i::Object> setter_i = v8::Utils::OpenHandle(*setter, true);
4869   if (setter_i.is_null()) setter_i = isolate->factory()->null_value();
4870   i::JSObject::DefineAccessor(i::Handle<i::JSObject>::cast(self),
4871                               v8::Utils::OpenHandle(*name), getter_i, setter_i,
4872                               static_cast<i::PropertyAttributes>(attribute));
4873 }
4874 
SetNativeDataProperty(v8::Local<v8::Context> context,v8::Local<Name> name,AccessorNameGetterCallback getter,AccessorNameSetterCallback setter,v8::Local<Value> data,PropertyAttribute attributes,SideEffectType getter_side_effect_type,SideEffectType setter_side_effect_type)4875 Maybe<bool> Object::SetNativeDataProperty(
4876     v8::Local<v8::Context> context, v8::Local<Name> name,
4877     AccessorNameGetterCallback getter, AccessorNameSetterCallback setter,
4878     v8::Local<Value> data, PropertyAttribute attributes,
4879     SideEffectType getter_side_effect_type,
4880     SideEffectType setter_side_effect_type) {
4881   return ObjectSetAccessor(context, this, name, getter, setter, data, DEFAULT,
4882                            attributes, true, false, getter_side_effect_type,
4883                            setter_side_effect_type);
4884 }
4885 
SetLazyDataProperty(v8::Local<v8::Context> context,v8::Local<Name> name,AccessorNameGetterCallback getter,v8::Local<Value> data,PropertyAttribute attributes,SideEffectType getter_side_effect_type,SideEffectType setter_side_effect_type)4886 Maybe<bool> Object::SetLazyDataProperty(
4887     v8::Local<v8::Context> context, v8::Local<Name> name,
4888     AccessorNameGetterCallback getter, v8::Local<Value> data,
4889     PropertyAttribute attributes, SideEffectType getter_side_effect_type,
4890     SideEffectType setter_side_effect_type) {
4891   return ObjectSetAccessor(context, this, name, getter,
4892                            static_cast<AccessorNameSetterCallback>(nullptr),
4893                            data, DEFAULT, attributes, true, true,
4894                            getter_side_effect_type, setter_side_effect_type);
4895 }
4896 
HasOwnProperty(Local<Context> context,Local<Name> key)4897 Maybe<bool> v8::Object::HasOwnProperty(Local<Context> context,
4898                                        Local<Name> key) {
4899   auto isolate = reinterpret_cast<i::Isolate*>(context->GetIsolate());
4900   ENTER_V8(isolate, context, Object, HasOwnProperty, Nothing<bool>(),
4901            i::HandleScope);
4902   auto self = Utils::OpenHandle(this);
4903   auto key_val = Utils::OpenHandle(*key);
4904   auto result = i::JSReceiver::HasOwnProperty(isolate, self, key_val);
4905   has_pending_exception = result.IsNothing();
4906   RETURN_ON_FAILED_EXECUTION_PRIMITIVE(bool);
4907   return result;
4908 }
4909 
HasOwnProperty(Local<Context> context,uint32_t index)4910 Maybe<bool> v8::Object::HasOwnProperty(Local<Context> context, uint32_t index) {
4911   auto isolate = reinterpret_cast<i::Isolate*>(context->GetIsolate());
4912   ENTER_V8(isolate, context, Object, HasOwnProperty, Nothing<bool>(),
4913            i::HandleScope);
4914   auto self = Utils::OpenHandle(this);
4915   auto result = i::JSReceiver::HasOwnProperty(isolate, self, index);
4916   has_pending_exception = result.IsNothing();
4917   RETURN_ON_FAILED_EXECUTION_PRIMITIVE(bool);
4918   return result;
4919 }
4920 
HasRealNamedProperty(Local<Context> context,Local<Name> key)4921 Maybe<bool> v8::Object::HasRealNamedProperty(Local<Context> context,
4922                                              Local<Name> key) {
4923   auto isolate = reinterpret_cast<i::Isolate*>(context->GetIsolate());
4924   ENTER_V8_NO_SCRIPT(isolate, context, Object, HasRealNamedProperty,
4925                      Nothing<bool>(), i::HandleScope);
4926   auto self = Utils::OpenHandle(this);
4927   if (!self->IsJSObject()) return Just(false);
4928   auto key_val = Utils::OpenHandle(*key);
4929   auto result = i::JSObject::HasRealNamedProperty(
4930       isolate, i::Handle<i::JSObject>::cast(self), key_val);
4931   has_pending_exception = result.IsNothing();
4932   RETURN_ON_FAILED_EXECUTION_PRIMITIVE(bool);
4933   return result;
4934 }
4935 
HasRealIndexedProperty(Local<Context> context,uint32_t index)4936 Maybe<bool> v8::Object::HasRealIndexedProperty(Local<Context> context,
4937                                                uint32_t index) {
4938   auto isolate = reinterpret_cast<i::Isolate*>(context->GetIsolate());
4939   ENTER_V8_NO_SCRIPT(isolate, context, Object, HasRealIndexedProperty,
4940                      Nothing<bool>(), i::HandleScope);
4941   auto self = Utils::OpenHandle(this);
4942   if (!self->IsJSObject()) return Just(false);
4943   auto result = i::JSObject::HasRealElementProperty(
4944       isolate, i::Handle<i::JSObject>::cast(self), index);
4945   has_pending_exception = result.IsNothing();
4946   RETURN_ON_FAILED_EXECUTION_PRIMITIVE(bool);
4947   return result;
4948 }
4949 
HasRealNamedCallbackProperty(Local<Context> context,Local<Name> key)4950 Maybe<bool> v8::Object::HasRealNamedCallbackProperty(Local<Context> context,
4951                                                      Local<Name> key) {
4952   auto isolate = reinterpret_cast<i::Isolate*>(context->GetIsolate());
4953   ENTER_V8_NO_SCRIPT(isolate, context, Object, HasRealNamedCallbackProperty,
4954                      Nothing<bool>(), i::HandleScope);
4955   auto self = Utils::OpenHandle(this);
4956   if (!self->IsJSObject()) return Just(false);
4957   auto key_val = Utils::OpenHandle(*key);
4958   auto result = i::JSObject::HasRealNamedCallbackProperty(
4959       isolate, i::Handle<i::JSObject>::cast(self), key_val);
4960   has_pending_exception = result.IsNothing();
4961   RETURN_ON_FAILED_EXECUTION_PRIMITIVE(bool);
4962   return result;
4963 }
4964 
HasNamedLookupInterceptor() const4965 bool v8::Object::HasNamedLookupInterceptor() const {
4966   auto self = *Utils::OpenHandle(this);
4967   if (self.IsJSObject()) return false;
4968   return i::JSObject::cast(self).HasNamedInterceptor();
4969 }
4970 
HasIndexedLookupInterceptor() const4971 bool v8::Object::HasIndexedLookupInterceptor() const {
4972   auto self = *Utils::OpenHandle(this);
4973   if (self.IsJSObject()) return false;
4974   return i::JSObject::cast(self).HasIndexedInterceptor();
4975 }
4976 
GetRealNamedPropertyInPrototypeChain(Local<Context> context,Local<Name> key)4977 MaybeLocal<Value> v8::Object::GetRealNamedPropertyInPrototypeChain(
4978     Local<Context> context, Local<Name> key) {
4979   PREPARE_FOR_EXECUTION(context, Object, GetRealNamedPropertyInPrototypeChain,
4980                         Value);
4981   i::Handle<i::JSReceiver> self = Utils::OpenHandle(this);
4982   if (!self->IsJSObject()) return MaybeLocal<Value>();
4983   i::Handle<i::Name> key_obj = Utils::OpenHandle(*key);
4984   i::PrototypeIterator iter(isolate, self);
4985   if (iter.IsAtEnd()) return MaybeLocal<Value>();
4986   i::Handle<i::JSReceiver> proto =
4987       i::PrototypeIterator::GetCurrent<i::JSReceiver>(iter);
4988   i::PropertyKey lookup_key(isolate, key_obj);
4989   i::LookupIterator it(isolate, self, lookup_key, proto,
4990                        i::LookupIterator::PROTOTYPE_CHAIN_SKIP_INTERCEPTOR);
4991   Local<Value> result;
4992   has_pending_exception = !ToLocal<Value>(i::Object::GetProperty(&it), &result);
4993   RETURN_ON_FAILED_EXECUTION(Value);
4994   if (!it.IsFound()) return MaybeLocal<Value>();
4995   RETURN_ESCAPED(result);
4996 }
4997 
4998 Maybe<PropertyAttribute>
GetRealNamedPropertyAttributesInPrototypeChain(Local<Context> context,Local<Name> key)4999 v8::Object::GetRealNamedPropertyAttributesInPrototypeChain(
5000     Local<Context> context, Local<Name> key) {
5001   auto isolate = reinterpret_cast<i::Isolate*>(context->GetIsolate());
5002   ENTER_V8(isolate, context, Object,
5003            GetRealNamedPropertyAttributesInPrototypeChain,
5004            Nothing<PropertyAttribute>(), i::HandleScope);
5005   i::Handle<i::JSReceiver> self = Utils::OpenHandle(this);
5006   if (!self->IsJSObject()) return Nothing<PropertyAttribute>();
5007   i::Handle<i::Name> key_obj = Utils::OpenHandle(*key);
5008   i::PrototypeIterator iter(isolate, self);
5009   if (iter.IsAtEnd()) return Nothing<PropertyAttribute>();
5010   i::Handle<i::JSReceiver> proto =
5011       i::PrototypeIterator::GetCurrent<i::JSReceiver>(iter);
5012   i::PropertyKey lookup_key(isolate, key_obj);
5013   i::LookupIterator it(isolate, self, lookup_key, proto,
5014                        i::LookupIterator::PROTOTYPE_CHAIN_SKIP_INTERCEPTOR);
5015   Maybe<i::PropertyAttributes> result =
5016       i::JSReceiver::GetPropertyAttributes(&it);
5017   has_pending_exception = result.IsNothing();
5018   RETURN_ON_FAILED_EXECUTION_PRIMITIVE(PropertyAttribute);
5019   if (!it.IsFound()) return Nothing<PropertyAttribute>();
5020   if (result.FromJust() == i::ABSENT) return Just(None);
5021   return Just(static_cast<PropertyAttribute>(result.FromJust()));
5022 }
5023 
GetRealNamedProperty(Local<Context> context,Local<Name> key)5024 MaybeLocal<Value> v8::Object::GetRealNamedProperty(Local<Context> context,
5025                                                    Local<Name> key) {
5026   PREPARE_FOR_EXECUTION(context, Object, GetRealNamedProperty, Value);
5027   i::Handle<i::JSReceiver> self = Utils::OpenHandle(this);
5028   i::Handle<i::Name> key_obj = Utils::OpenHandle(*key);
5029   i::PropertyKey lookup_key(isolate, key_obj);
5030   i::LookupIterator it(isolate, self, lookup_key, self,
5031                        i::LookupIterator::PROTOTYPE_CHAIN_SKIP_INTERCEPTOR);
5032   Local<Value> result;
5033   has_pending_exception = !ToLocal<Value>(i::Object::GetProperty(&it), &result);
5034   RETURN_ON_FAILED_EXECUTION(Value);
5035   if (!it.IsFound()) return MaybeLocal<Value>();
5036   RETURN_ESCAPED(result);
5037 }
5038 
GetRealNamedPropertyAttributes(Local<Context> context,Local<Name> key)5039 Maybe<PropertyAttribute> v8::Object::GetRealNamedPropertyAttributes(
5040     Local<Context> context, Local<Name> key) {
5041   auto isolate = reinterpret_cast<i::Isolate*>(context->GetIsolate());
5042   ENTER_V8(isolate, context, Object, GetRealNamedPropertyAttributes,
5043            Nothing<PropertyAttribute>(), i::HandleScope);
5044   i::Handle<i::JSReceiver> self = Utils::OpenHandle(this);
5045   i::Handle<i::Name> key_obj = Utils::OpenHandle(*key);
5046   i::PropertyKey lookup_key(isolate, key_obj);
5047   i::LookupIterator it(isolate, self, lookup_key, self,
5048                        i::LookupIterator::PROTOTYPE_CHAIN_SKIP_INTERCEPTOR);
5049   auto result = i::JSReceiver::GetPropertyAttributes(&it);
5050   has_pending_exception = result.IsNothing();
5051   RETURN_ON_FAILED_EXECUTION_PRIMITIVE(PropertyAttribute);
5052   if (!it.IsFound()) return Nothing<PropertyAttribute>();
5053   if (result.FromJust() == i::ABSENT) {
5054     return Just(static_cast<PropertyAttribute>(i::NONE));
5055   }
5056   return Just<PropertyAttribute>(
5057       static_cast<PropertyAttribute>(result.FromJust()));
5058 }
5059 
Clone()5060 Local<v8::Object> v8::Object::Clone() {
5061   auto self = i::Handle<i::JSObject>::cast(Utils::OpenHandle(this));
5062   auto isolate = self->GetIsolate();
5063   ENTER_V8_NO_SCRIPT_NO_EXCEPTION(isolate);
5064   i::Handle<i::JSObject> result = isolate->factory()->CopyJSObject(self);
5065   return Utils::ToLocal(result);
5066 }
5067 
5068 namespace {
CreationContextImpl(i::Handle<i::JSReceiver> self)5069 Local<v8::Context> CreationContextImpl(i::Handle<i::JSReceiver> self) {
5070   i::Handle<i::Context> context;
5071   if (self->GetCreationContext().ToHandle(&context)) {
5072     return Utils::ToLocal(context);
5073   }
5074 
5075   return Local<v8::Context>();
5076 }
5077 }  // namespace
5078 
CreationContext()5079 Local<v8::Context> v8::Object::CreationContext() {
5080   auto self = Utils::OpenHandle(this);
5081   return CreationContextImpl(self);
5082 }
5083 
CreationContext(const PersistentBase<Object> & object)5084 Local<v8::Context> v8::Object::CreationContext(
5085     const PersistentBase<Object>& object) {
5086   auto self = Utils::OpenHandle(object.val_);
5087   return CreationContextImpl(self);
5088 }
5089 
GetCreationContext()5090 MaybeLocal<v8::Context> v8::Object::GetCreationContext() {
5091   auto self = Utils::OpenHandle(this);
5092   i::Handle<i::Context> context;
5093   if (self->GetCreationContext().ToHandle(&context)) {
5094     return Utils::ToLocal(context);
5095   }
5096   return MaybeLocal<v8::Context>();
5097 }
5098 
GetCreationContextChecked()5099 Local<v8::Context> v8::Object::GetCreationContextChecked() {
5100   Local<Context> context;
5101   Utils::ApiCheck(GetCreationContext().ToLocal(&context),
5102                   "v8::Object::GetCreationContextChecked",
5103                   "No creation context available");
5104   return context;
5105 }
5106 
GetIdentityHash()5107 int v8::Object::GetIdentityHash() {
5108   i::DisallowGarbageCollection no_gc;
5109   auto self = Utils::OpenHandle(this);
5110   auto isolate = self->GetIsolate();
5111   ASSERT_NO_SCRIPT_NO_EXCEPTION(isolate);
5112   i::HandleScope scope(isolate);
5113   return self->GetOrCreateIdentityHash(isolate).value();
5114 }
5115 
IsCallable() const5116 bool v8::Object::IsCallable() const {
5117   auto self = Utils::OpenHandle(this);
5118   return self->IsCallable();
5119 }
5120 
IsConstructor() const5121 bool v8::Object::IsConstructor() const {
5122   auto self = Utils::OpenHandle(this);
5123   return self->IsConstructor();
5124 }
5125 
IsApiWrapper() const5126 bool v8::Object::IsApiWrapper() const {
5127   auto self = i::Handle<i::JSObject>::cast(Utils::OpenHandle(this));
5128   // Objects with embedder fields can wrap API objects.
5129   return self->MayHaveEmbedderFields();
5130 }
5131 
IsUndetectable() const5132 bool v8::Object::IsUndetectable() const {
5133   auto self = i::Handle<i::JSObject>::cast(Utils::OpenHandle(this));
5134   return self->IsUndetectable();
5135 }
5136 
CallAsFunction(Local<Context> context,Local<Value> recv,int argc,Local<Value> argv[])5137 MaybeLocal<Value> Object::CallAsFunction(Local<Context> context,
5138                                          Local<Value> recv, int argc,
5139                                          Local<Value> argv[]) {
5140   auto isolate = reinterpret_cast<i::Isolate*>(context->GetIsolate());
5141   TRACE_EVENT_CALL_STATS_SCOPED(isolate, "v8", "V8.Execute");
5142   ENTER_V8(isolate, context, Object, CallAsFunction, MaybeLocal<Value>(),
5143            InternalEscapableScope);
5144   i::TimerEventScope<i::TimerEventExecute> timer_scope(isolate);
5145   i::NestedTimedHistogramScope execute_timer(isolate->counters()->execute(),
5146                                              isolate);
5147   auto self = Utils::OpenHandle(this);
5148   auto recv_obj = Utils::OpenHandle(*recv);
5149   STATIC_ASSERT(sizeof(v8::Local<v8::Value>) == sizeof(i::Handle<i::Object>));
5150   i::Handle<i::Object>* args = reinterpret_cast<i::Handle<i::Object>*>(argv);
5151   Local<Value> result;
5152   has_pending_exception = !ToLocal<Value>(
5153       i::Execution::Call(isolate, self, recv_obj, argc, args), &result);
5154   RETURN_ON_FAILED_EXECUTION(Value);
5155   RETURN_ESCAPED(result);
5156 }
5157 
CallAsConstructor(Local<Context> context,int argc,Local<Value> argv[])5158 MaybeLocal<Value> Object::CallAsConstructor(Local<Context> context, int argc,
5159                                             Local<Value> argv[]) {
5160   auto isolate = reinterpret_cast<i::Isolate*>(context->GetIsolate());
5161   TRACE_EVENT_CALL_STATS_SCOPED(isolate, "v8", "V8.Execute");
5162   ENTER_V8(isolate, context, Object, CallAsConstructor, MaybeLocal<Value>(),
5163            InternalEscapableScope);
5164   i::TimerEventScope<i::TimerEventExecute> timer_scope(isolate);
5165   i::NestedTimedHistogramScope execute_timer(isolate->counters()->execute(),
5166                                              isolate);
5167   auto self = Utils::OpenHandle(this);
5168   STATIC_ASSERT(sizeof(v8::Local<v8::Value>) == sizeof(i::Handle<i::Object>));
5169   i::Handle<i::Object>* args = reinterpret_cast<i::Handle<i::Object>*>(argv);
5170   Local<Value> result;
5171   has_pending_exception = !ToLocal<Value>(
5172       i::Execution::New(isolate, self, self, argc, args), &result);
5173   RETURN_ON_FAILED_EXECUTION(Value);
5174   RETURN_ESCAPED(result);
5175 }
5176 
New(Local<Context> context,FunctionCallback callback,Local<Value> data,int length,ConstructorBehavior behavior,SideEffectType side_effect_type)5177 MaybeLocal<Function> Function::New(Local<Context> context,
5178                                    FunctionCallback callback, Local<Value> data,
5179                                    int length, ConstructorBehavior behavior,
5180                                    SideEffectType side_effect_type) {
5181   i::Isolate* isolate = Utils::OpenHandle(*context)->GetIsolate();
5182   API_RCS_SCOPE(isolate, Function, New);
5183   ENTER_V8_NO_SCRIPT_NO_EXCEPTION(isolate);
5184   auto templ =
5185       FunctionTemplateNew(isolate, callback, data, Local<Signature>(), length,
5186                           behavior, true, Local<Private>(), side_effect_type);
5187   return templ->GetFunction(context);
5188 }
5189 
NewInstance(Local<Context> context,int argc,v8::Local<v8::Value> argv[]) const5190 MaybeLocal<Object> Function::NewInstance(Local<Context> context, int argc,
5191                                          v8::Local<v8::Value> argv[]) const {
5192   return NewInstanceWithSideEffectType(context, argc, argv,
5193                                        SideEffectType::kHasSideEffect);
5194 }
5195 
NewInstanceWithSideEffectType(Local<Context> context,int argc,v8::Local<v8::Value> argv[],SideEffectType side_effect_type) const5196 MaybeLocal<Object> Function::NewInstanceWithSideEffectType(
5197     Local<Context> context, int argc, v8::Local<v8::Value> argv[],
5198     SideEffectType side_effect_type) const {
5199   auto isolate = reinterpret_cast<i::Isolate*>(context->GetIsolate());
5200   TRACE_EVENT_CALL_STATS_SCOPED(isolate, "v8", "V8.Execute");
5201   ENTER_V8(isolate, context, Function, NewInstance, MaybeLocal<Object>(),
5202            InternalEscapableScope);
5203   i::TimerEventScope<i::TimerEventExecute> timer_scope(isolate);
5204   i::NestedTimedHistogramScope execute_timer(isolate->counters()->execute(),
5205                                              isolate);
5206   auto self = Utils::OpenHandle(this);
5207   STATIC_ASSERT(sizeof(v8::Local<v8::Value>) == sizeof(i::Handle<i::Object>));
5208   bool should_set_has_no_side_effect =
5209       side_effect_type == SideEffectType::kHasNoSideEffect &&
5210       isolate->debug_execution_mode() == i::DebugInfo::kSideEffects;
5211   if (should_set_has_no_side_effect) {
5212     CHECK(self->IsJSFunction() &&
5213           i::JSFunction::cast(*self).shared().IsApiFunction());
5214     i::Object obj =
5215         i::JSFunction::cast(*self).shared().get_api_func_data().call_code(
5216             kAcquireLoad);
5217     if (obj.IsCallHandlerInfo()) {
5218       i::CallHandlerInfo handler_info = i::CallHandlerInfo::cast(obj);
5219       if (!handler_info.IsSideEffectFreeCallHandlerInfo()) {
5220         handler_info.SetNextCallHasNoSideEffect();
5221       }
5222     }
5223   }
5224   i::Handle<i::Object>* args = reinterpret_cast<i::Handle<i::Object>*>(argv);
5225   Local<Object> result;
5226   has_pending_exception = !ToLocal<Object>(
5227       i::Execution::New(isolate, self, self, argc, args), &result);
5228   if (should_set_has_no_side_effect) {
5229     i::Object obj =
5230         i::JSFunction::cast(*self).shared().get_api_func_data().call_code(
5231             kAcquireLoad);
5232     if (obj.IsCallHandlerInfo()) {
5233       i::CallHandlerInfo handler_info = i::CallHandlerInfo::cast(obj);
5234       if (has_pending_exception) {
5235         // Restore the map if an exception prevented restoration.
5236         handler_info.NextCallHasNoSideEffect();
5237       } else {
5238         DCHECK(handler_info.IsSideEffectCallHandlerInfo() ||
5239                handler_info.IsSideEffectFreeCallHandlerInfo());
5240       }
5241     }
5242   }
5243   RETURN_ON_FAILED_EXECUTION(Object);
5244   RETURN_ESCAPED(result);
5245 }
5246 
Call(Local<Context> context,v8::Local<v8::Value> recv,int argc,v8::Local<v8::Value> argv[])5247 MaybeLocal<v8::Value> Function::Call(Local<Context> context,
5248                                      v8::Local<v8::Value> recv, int argc,
5249                                      v8::Local<v8::Value> argv[]) {
5250   auto isolate = reinterpret_cast<i::Isolate*>(context->GetIsolate());
5251   TRACE_EVENT_CALL_STATS_SCOPED(isolate, "v8", "V8.Execute");
5252   ENTER_V8(isolate, context, Function, Call, MaybeLocal<Value>(),
5253            InternalEscapableScope);
5254   i::TimerEventScope<i::TimerEventExecute> timer_scope(isolate);
5255   i::NestedTimedHistogramScope execute_timer(isolate->counters()->execute(),
5256                                              isolate);
5257   auto self = Utils::OpenHandle(this);
5258   Utils::ApiCheck(!self.is_null(), "v8::Function::Call",
5259                   "Function to be called is a null pointer");
5260   i::Handle<i::Object> recv_obj = Utils::OpenHandle(*recv);
5261   STATIC_ASSERT(sizeof(v8::Local<v8::Value>) == sizeof(i::Handle<i::Object>));
5262   i::Handle<i::Object>* args = reinterpret_cast<i::Handle<i::Object>*>(argv);
5263   Local<Value> result;
5264   has_pending_exception = !ToLocal<Value>(
5265       i::Execution::Call(isolate, self, recv_obj, argc, args), &result);
5266   RETURN_ON_FAILED_EXECUTION(Value);
5267   RETURN_ESCAPED(result);
5268 }
5269 
SetName(v8::Local<v8::String> name)5270 void Function::SetName(v8::Local<v8::String> name) {
5271   auto self = Utils::OpenHandle(this);
5272   if (!self->IsJSFunction()) return;
5273   auto func = i::Handle<i::JSFunction>::cast(self);
5274   ASSERT_NO_SCRIPT_NO_EXCEPTION(func->GetIsolate());
5275   func->shared().SetName(*Utils::OpenHandle(*name));
5276 }
5277 
GetName() const5278 Local<Value> Function::GetName() const {
5279   auto self = Utils::OpenHandle(this);
5280   i::Isolate* isolate = self->GetIsolate();
5281   if (self->IsJSBoundFunction()) {
5282     auto func = i::Handle<i::JSBoundFunction>::cast(self);
5283     i::Handle<i::Object> name;
5284     ASSIGN_RETURN_ON_EXCEPTION_VALUE(isolate, name,
5285                                      i::JSBoundFunction::GetName(isolate, func),
5286                                      Local<Value>());
5287     return Utils::ToLocal(name);
5288   }
5289   if (self->IsJSFunction()) {
5290     auto func = i::Handle<i::JSFunction>::cast(self);
5291     return Utils::ToLocal(handle(func->shared().Name(), isolate));
5292   }
5293   return ToApiHandle<Primitive>(isolate->factory()->undefined_value());
5294 }
5295 
GetInferredName() const5296 Local<Value> Function::GetInferredName() const {
5297   auto self = Utils::OpenHandle(this);
5298   if (!self->IsJSFunction()) {
5299     return ToApiHandle<Primitive>(
5300         self->GetIsolate()->factory()->undefined_value());
5301   }
5302   auto func = i::Handle<i::JSFunction>::cast(self);
5303   return Utils::ToLocal(
5304       i::Handle<i::Object>(func->shared().inferred_name(), func->GetIsolate()));
5305 }
5306 
GetDebugName() const5307 Local<Value> Function::GetDebugName() const {
5308   auto self = Utils::OpenHandle(this);
5309   if (!self->IsJSFunction()) {
5310     return ToApiHandle<Primitive>(
5311         self->GetIsolate()->factory()->undefined_value());
5312   }
5313   auto func = i::Handle<i::JSFunction>::cast(self);
5314   i::Handle<i::String> name = i::JSFunction::GetDebugName(func);
5315   return Utils::ToLocal(i::Handle<i::Object>(*name, self->GetIsolate()));
5316 }
5317 
GetScriptOrigin() const5318 ScriptOrigin Function::GetScriptOrigin() const {
5319   auto self = Utils::OpenHandle(this);
5320   auto isolate = reinterpret_cast<v8::Isolate*>(self->GetIsolate());
5321   if (!self->IsJSFunction()) return v8::ScriptOrigin(isolate, Local<Value>());
5322   auto func = i::Handle<i::JSFunction>::cast(self);
5323   if (func->shared().script().IsScript()) {
5324     i::Handle<i::Script> script(i::Script::cast(func->shared().script()),
5325                                 func->GetIsolate());
5326     return GetScriptOriginForScript(func->GetIsolate(), script);
5327   }
5328   return v8::ScriptOrigin(isolate, Local<Value>());
5329 }
5330 
5331 const int Function::kLineOffsetNotFound = -1;
5332 
GetScriptLineNumber() const5333 int Function::GetScriptLineNumber() const {
5334   auto self = Utils::OpenHandle(this);
5335   if (!self->IsJSFunction()) {
5336     return kLineOffsetNotFound;
5337   }
5338   auto func = i::Handle<i::JSFunction>::cast(self);
5339   if (func->shared().script().IsScript()) {
5340     i::Handle<i::Script> script(i::Script::cast(func->shared().script()),
5341                                 func->GetIsolate());
5342     return i::Script::GetLineNumber(script, func->shared().StartPosition());
5343   }
5344   return kLineOffsetNotFound;
5345 }
5346 
GetScriptColumnNumber() const5347 int Function::GetScriptColumnNumber() const {
5348   auto self = Utils::OpenHandle(this);
5349   if (!self->IsJSFunction()) {
5350     return kLineOffsetNotFound;
5351   }
5352   auto func = i::Handle<i::JSFunction>::cast(self);
5353   if (func->shared().script().IsScript()) {
5354     i::Handle<i::Script> script(i::Script::cast(func->shared().script()),
5355                                 func->GetIsolate());
5356     return i::Script::GetColumnNumber(script, func->shared().StartPosition());
5357   }
5358   return kLineOffsetNotFound;
5359 }
5360 
GetUnboundScript() const5361 MaybeLocal<UnboundScript> Function::GetUnboundScript() const {
5362   i::Handle<i::Object> self = Utils::OpenHandle(this);
5363   if (!self->IsJSFunction()) return MaybeLocal<UnboundScript>();
5364   i::SharedFunctionInfo sfi = i::JSFunction::cast(*self).shared();
5365   i::Isolate* isolate = sfi.GetIsolate();
5366   return ToApiHandle<UnboundScript>(i::handle(sfi, isolate));
5367 }
5368 
ScriptId() const5369 int Function::ScriptId() const {
5370   i::JSReceiver self = *Utils::OpenHandle(this);
5371   if (!self.IsJSFunction()) return v8::UnboundScript::kNoScriptId;
5372   auto func = i::JSFunction::cast(self);
5373   if (!func.shared().script().IsScript()) return v8::UnboundScript::kNoScriptId;
5374   return i::Script::cast(func.shared().script()).id();
5375 }
5376 
GetBoundFunction() const5377 Local<v8::Value> Function::GetBoundFunction() const {
5378   auto self = Utils::OpenHandle(this);
5379   if (self->IsJSBoundFunction()) {
5380     auto bound_function = i::Handle<i::JSBoundFunction>::cast(self);
5381     auto bound_target_function = i::handle(
5382         bound_function->bound_target_function(), bound_function->GetIsolate());
5383     return Utils::CallableToLocal(bound_target_function);
5384   }
5385   return v8::Undefined(reinterpret_cast<v8::Isolate*>(self->GetIsolate()));
5386 }
5387 
FunctionProtoToString(Local<Context> context)5388 MaybeLocal<String> v8::Function::FunctionProtoToString(Local<Context> context) {
5389   PREPARE_FOR_EXECUTION(context, Function, FunctionProtoToString, String);
5390   auto self = Utils::OpenHandle(this);
5391   Local<Value> result;
5392   has_pending_exception = !ToLocal<Value>(
5393       i::Execution::CallBuiltin(isolate, isolate->function_to_string(), self, 0,
5394                                 nullptr),
5395       &result);
5396   RETURN_ON_FAILED_EXECUTION(String);
5397   RETURN_ESCAPED(Local<String>::Cast(result));
5398 }
5399 
GetIdentityHash()5400 int Name::GetIdentityHash() {
5401   auto self = Utils::OpenHandle(this);
5402   return static_cast<int>(self->EnsureHash());
5403 }
5404 
Length() const5405 int String::Length() const {
5406   i::Handle<i::String> str = Utils::OpenHandle(this);
5407   return str->length();
5408 }
5409 
IsOneByte() const5410 bool String::IsOneByte() const {
5411   i::Handle<i::String> str = Utils::OpenHandle(this);
5412   return str->IsOneByteRepresentation();
5413 }
5414 
5415 // Helpers for ContainsOnlyOneByteHelper
5416 template <size_t size>
5417 struct OneByteMask;
5418 template <>
5419 struct OneByteMask<4> {
5420   static const uint32_t value = 0xFF00FF00;
5421 };
5422 template <>
5423 struct OneByteMask<8> {
5424   static const uint64_t value = 0xFF00'FF00'FF00'FF00;
5425 };
5426 static const uintptr_t kOneByteMask = OneByteMask<sizeof(uintptr_t)>::value;
5427 static const uintptr_t kAlignmentMask = sizeof(uintptr_t) - 1;
Unaligned(const uint16_t * chars)5428 static inline bool Unaligned(const uint16_t* chars) {
5429   return reinterpret_cast<const uintptr_t>(chars) & kAlignmentMask;
5430 }
5431 
Align(const uint16_t * chars)5432 static inline const uint16_t* Align(const uint16_t* chars) {
5433   return reinterpret_cast<uint16_t*>(reinterpret_cast<uintptr_t>(chars) &
5434                                      ~kAlignmentMask);
5435 }
5436 
5437 class ContainsOnlyOneByteHelper {
5438  public:
ContainsOnlyOneByteHelper()5439   ContainsOnlyOneByteHelper() : is_one_byte_(true) {}
5440   ContainsOnlyOneByteHelper(const ContainsOnlyOneByteHelper&) = delete;
5441   ContainsOnlyOneByteHelper& operator=(const ContainsOnlyOneByteHelper&) =
5442       delete;
Check(i::String string)5443   bool Check(i::String string) {
5444     i::ConsString cons_string = i::String::VisitFlat(this, string, 0);
5445     if (cons_string.is_null()) return is_one_byte_;
5446     return CheckCons(cons_string);
5447   }
VisitOneByteString(const uint8_t * chars,int length)5448   void VisitOneByteString(const uint8_t* chars, int length) {
5449     // Nothing to do.
5450   }
VisitTwoByteString(const uint16_t * chars,int length)5451   void VisitTwoByteString(const uint16_t* chars, int length) {
5452     // Accumulated bits.
5453     uintptr_t acc = 0;
5454     // Align to uintptr_t.
5455     const uint16_t* end = chars + length;
5456     while (Unaligned(chars) && chars != end) {
5457       acc |= *chars++;
5458     }
5459     // Read word aligned in blocks,
5460     // checking the return value at the end of each block.
5461     const uint16_t* aligned_end = Align(end);
5462     const int increment = sizeof(uintptr_t) / sizeof(uint16_t);
5463     const int inner_loops = 16;
5464     while (chars + inner_loops * increment < aligned_end) {
5465       for (int i = 0; i < inner_loops; i++) {
5466         acc |= *reinterpret_cast<const uintptr_t*>(chars);
5467         chars += increment;
5468       }
5469       // Check for early return.
5470       if ((acc & kOneByteMask) != 0) {
5471         is_one_byte_ = false;
5472         return;
5473       }
5474     }
5475     // Read the rest.
5476     while (chars != end) {
5477       acc |= *chars++;
5478     }
5479     // Check result.
5480     if ((acc & kOneByteMask) != 0) is_one_byte_ = false;
5481   }
5482 
5483  private:
CheckCons(i::ConsString cons_string)5484   bool CheckCons(i::ConsString cons_string) {
5485     while (true) {
5486       // Check left side if flat.
5487       i::String left = cons_string.first();
5488       i::ConsString left_as_cons = i::String::VisitFlat(this, left, 0);
5489       if (!is_one_byte_) return false;
5490       // Check right side if flat.
5491       i::String right = cons_string.second();
5492       i::ConsString right_as_cons = i::String::VisitFlat(this, right, 0);
5493       if (!is_one_byte_) return false;
5494       // Standard recurse/iterate trick.
5495       if (!left_as_cons.is_null() && !right_as_cons.is_null()) {
5496         if (left.length() < right.length()) {
5497           CheckCons(left_as_cons);
5498           cons_string = right_as_cons;
5499         } else {
5500           CheckCons(right_as_cons);
5501           cons_string = left_as_cons;
5502         }
5503         // Check fast return.
5504         if (!is_one_byte_) return false;
5505         continue;
5506       }
5507       // Descend left in place.
5508       if (!left_as_cons.is_null()) {
5509         cons_string = left_as_cons;
5510         continue;
5511       }
5512       // Descend right in place.
5513       if (!right_as_cons.is_null()) {
5514         cons_string = right_as_cons;
5515         continue;
5516       }
5517       // Terminate.
5518       break;
5519     }
5520     return is_one_byte_;
5521   }
5522   bool is_one_byte_;
5523 };
5524 
ContainsOnlyOneByte() const5525 bool String::ContainsOnlyOneByte() const {
5526   i::Handle<i::String> str = Utils::OpenHandle(this);
5527   if (str->IsOneByteRepresentation()) return true;
5528   ContainsOnlyOneByteHelper helper;
5529   return helper.Check(*str);
5530 }
5531 
Utf8Length(Isolate * isolate) const5532 int String::Utf8Length(Isolate* isolate) const {
5533   i::Handle<i::String> str = Utils::OpenHandle(this);
5534   str = i::String::Flatten(reinterpret_cast<i::Isolate*>(isolate), str);
5535   int length = str->length();
5536   if (length == 0) return 0;
5537   i::DisallowGarbageCollection no_gc;
5538   i::String::FlatContent flat = str->GetFlatContent(no_gc);
5539   DCHECK(flat.IsFlat());
5540   int utf8_length = 0;
5541   if (flat.IsOneByte()) {
5542     for (uint8_t c : flat.ToOneByteVector()) {
5543       utf8_length += c >> 7;
5544     }
5545     utf8_length += length;
5546   } else {
5547     int last_character = unibrow::Utf16::kNoPreviousCharacter;
5548     for (uint16_t c : flat.ToUC16Vector()) {
5549       utf8_length += unibrow::Utf8::Length(c, last_character);
5550       last_character = c;
5551     }
5552   }
5553   return utf8_length;
5554 }
5555 
5556 namespace {
5557 // Writes the flat content of a string to a buffer. This is done in two phases.
5558 // The first phase calculates a pessimistic estimate (writable_length) on how
5559 // many code units can be safely written without exceeding the buffer capacity
5560 // and without leaving at a lone surrogate. The estimated number of code units
5561 // is then written out in one go, and the reported byte usage is used to
5562 // correct the estimate. This is repeated until the estimate becomes <= 0 or
5563 // all code units have been written out. The second phase writes out code
5564 // units until the buffer capacity is reached, would be exceeded by the next
5565 // unit, or all code units have been written out.
5566 template <typename Char>
WriteUtf8Impl(base::Vector<const Char> string,char * write_start,int write_capacity,int options,int * utf16_chars_read_out)5567 static int WriteUtf8Impl(base::Vector<const Char> string, char* write_start,
5568                          int write_capacity, int options,
5569                          int* utf16_chars_read_out) {
5570   bool write_null = !(options & v8::String::NO_NULL_TERMINATION);
5571   bool replace_invalid_utf8 = (options & v8::String::REPLACE_INVALID_UTF8);
5572   char* current_write = write_start;
5573   const Char* read_start = string.begin();
5574   int read_index = 0;
5575   int read_length = string.length();
5576   int prev_char = unibrow::Utf16::kNoPreviousCharacter;
5577   // Do a fast loop where there is no exit capacity check.
5578   // Need enough space to write everything but one character.
5579   STATIC_ASSERT(unibrow::Utf16::kMaxExtraUtf8BytesForOneUtf16CodeUnit == 3);
5580   static const int kMaxSizePerChar = sizeof(Char) == 1 ? 2 : 3;
5581   while (read_index < read_length) {
5582     int up_to = read_length;
5583     if (write_capacity != -1) {
5584       int remaining_capacity =
5585           write_capacity - static_cast<int>(current_write - write_start);
5586       int writable_length =
5587           (remaining_capacity - kMaxSizePerChar) / kMaxSizePerChar;
5588       // Need to drop into slow loop.
5589       if (writable_length <= 0) break;
5590       up_to = std::min(up_to, read_index + writable_length);
5591     }
5592     // Write the characters to the stream.
5593     if (sizeof(Char) == 1) {
5594       // Simply memcpy if we only have ASCII characters.
5595       uint8_t char_mask = 0;
5596       for (int i = read_index; i < up_to; i++) char_mask |= read_start[i];
5597       if ((char_mask & 0x80) == 0) {
5598         int copy_length = up_to - read_index;
5599         memcpy(current_write, read_start + read_index, copy_length);
5600         current_write += copy_length;
5601         read_index = up_to;
5602       } else {
5603         for (; read_index < up_to; read_index++) {
5604           current_write += unibrow::Utf8::EncodeOneByte(
5605               current_write, static_cast<uint8_t>(read_start[read_index]));
5606           DCHECK(write_capacity == -1 ||
5607                  (current_write - write_start) <= write_capacity);
5608         }
5609       }
5610     } else {
5611       for (; read_index < up_to; read_index++) {
5612         uint16_t character = read_start[read_index];
5613         current_write += unibrow::Utf8::Encode(current_write, character,
5614                                                prev_char, replace_invalid_utf8);
5615         prev_char = character;
5616         DCHECK(write_capacity == -1 ||
5617                (current_write - write_start) <= write_capacity);
5618       }
5619     }
5620   }
5621   if (read_index < read_length) {
5622     DCHECK_NE(-1, write_capacity);
5623     // Aborted due to limited capacity. Check capacity on each iteration.
5624     int remaining_capacity =
5625         write_capacity - static_cast<int>(current_write - write_start);
5626     DCHECK_GE(remaining_capacity, 0);
5627     for (; read_index < read_length && remaining_capacity > 0; read_index++) {
5628       uint32_t character = read_start[read_index];
5629       int written = 0;
5630       // We can't use a local buffer here because Encode needs to modify
5631       // previous characters in the stream.  We know, however, that
5632       // exactly one character will be advanced.
5633       if (unibrow::Utf16::IsSurrogatePair(prev_char, character)) {
5634         written = unibrow::Utf8::Encode(current_write, character, prev_char,
5635                                         replace_invalid_utf8);
5636         DCHECK_EQ(written, 1);
5637       } else {
5638         // Use a scratch buffer to check the required characters.
5639         char temp_buffer[unibrow::Utf8::kMaxEncodedSize];
5640         // Encoding a surrogate pair to Utf8 always takes 4 bytes.
5641         static const int kSurrogatePairEncodedSize =
5642             static_cast<int>(unibrow::Utf8::kMaxEncodedSize);
5643         // For REPLACE_INVALID_UTF8, catch the case where we cut off in the
5644         // middle of a surrogate pair. Abort before encoding the pair instead.
5645         if (replace_invalid_utf8 &&
5646             remaining_capacity < kSurrogatePairEncodedSize &&
5647             unibrow::Utf16::IsLeadSurrogate(character) &&
5648             read_index + 1 < read_length &&
5649             unibrow::Utf16::IsTrailSurrogate(read_start[read_index + 1])) {
5650           write_null = false;
5651           break;
5652         }
5653         // Can't encode using prev_char as gcc has array bounds issues.
5654         written = unibrow::Utf8::Encode(temp_buffer, character,
5655                                         unibrow::Utf16::kNoPreviousCharacter,
5656                                         replace_invalid_utf8);
5657         if (written > remaining_capacity) {
5658           // Won't fit. Abort and do not null-terminate the result.
5659           write_null = false;
5660           break;
5661         }
5662         // Copy over the character from temp_buffer.
5663         for (int i = 0; i < written; i++) current_write[i] = temp_buffer[i];
5664       }
5665 
5666       current_write += written;
5667       remaining_capacity -= written;
5668       prev_char = character;
5669     }
5670   }
5671 
5672   // Write out number of utf16 characters written to the stream.
5673   if (utf16_chars_read_out != nullptr) *utf16_chars_read_out = read_index;
5674 
5675   // Only null-terminate if there's space.
5676   if (write_null && (write_capacity == -1 ||
5677                      (current_write - write_start) < write_capacity)) {
5678     *current_write++ = '\0';
5679   }
5680   return static_cast<int>(current_write - write_start);
5681 }
5682 }  // anonymous namespace
5683 
WriteUtf8(Isolate * v8_isolate,char * buffer,int capacity,int * nchars_ref,int options) const5684 int String::WriteUtf8(Isolate* v8_isolate, char* buffer, int capacity,
5685                       int* nchars_ref, int options) const {
5686   i::Handle<i::String> str = Utils::OpenHandle(this);
5687   i::Isolate* isolate = reinterpret_cast<i::Isolate*>(v8_isolate);
5688   API_RCS_SCOPE(isolate, String, WriteUtf8);
5689   ENTER_V8_NO_SCRIPT_NO_EXCEPTION(isolate);
5690   str = i::String::Flatten(isolate, str);
5691   i::DisallowGarbageCollection no_gc;
5692   i::String::FlatContent content = str->GetFlatContent(no_gc);
5693   if (content.IsOneByte()) {
5694     return WriteUtf8Impl<uint8_t>(content.ToOneByteVector(), buffer, capacity,
5695                                   options, nchars_ref);
5696   } else {
5697     return WriteUtf8Impl<uint16_t>(content.ToUC16Vector(), buffer, capacity,
5698                                    options, nchars_ref);
5699   }
5700 }
5701 
5702 template <typename CharType>
WriteHelper(i::Isolate * isolate,const String * string,CharType * buffer,int start,int length,int options)5703 static inline int WriteHelper(i::Isolate* isolate, const String* string,
5704                               CharType* buffer, int start, int length,
5705                               int options) {
5706   API_RCS_SCOPE(isolate, String, Write);
5707   ENTER_V8_NO_SCRIPT_NO_EXCEPTION(isolate);
5708   DCHECK(start >= 0 && length >= -1);
5709   i::Handle<i::String> str = Utils::OpenHandle(string);
5710   str = i::String::Flatten(isolate, str);
5711   int end = start + length;
5712   if ((length == -1) || (length > str->length() - start)) end = str->length();
5713   if (end < 0) return 0;
5714   int write_length = end - start;
5715   if (start < end) i::String::WriteToFlat(*str, buffer, start, write_length);
5716   if (!(options & String::NO_NULL_TERMINATION) &&
5717       (length == -1 || write_length < length)) {
5718     buffer[write_length] = '\0';
5719   }
5720   return write_length;
5721 }
5722 
WriteOneByte(Isolate * isolate,uint8_t * buffer,int start,int length,int options) const5723 int String::WriteOneByte(Isolate* isolate, uint8_t* buffer, int start,
5724                          int length, int options) const {
5725   return WriteHelper(reinterpret_cast<i::Isolate*>(isolate), this, buffer,
5726                      start, length, options);
5727 }
5728 
Write(Isolate * isolate,uint16_t * buffer,int start,int length,int options) const5729 int String::Write(Isolate* isolate, uint16_t* buffer, int start, int length,
5730                   int options) const {
5731   return WriteHelper(reinterpret_cast<i::Isolate*>(isolate), this, buffer,
5732                      start, length, options);
5733 }
5734 
IsExternal() const5735 bool v8::String::IsExternal() const {
5736   i::Handle<i::String> str = Utils::OpenHandle(this);
5737   return i::StringShape(*str).IsExternal();
5738 }
5739 
IsExternalTwoByte() const5740 bool v8::String::IsExternalTwoByte() const {
5741   i::Handle<i::String> str = Utils::OpenHandle(this);
5742   return i::StringShape(*str).IsExternalTwoByte();
5743 }
5744 
IsExternalOneByte() const5745 bool v8::String::IsExternalOneByte() const {
5746   i::Handle<i::String> str = Utils::OpenHandle(this);
5747   return i::StringShape(*str).IsExternalOneByte();
5748 }
5749 
VerifyExternalStringResource(v8::String::ExternalStringResource * value) const5750 void v8::String::VerifyExternalStringResource(
5751     v8::String::ExternalStringResource* value) const {
5752   i::DisallowGarbageCollection no_gc;
5753   i::String str = *Utils::OpenHandle(this);
5754   const v8::String::ExternalStringResource* expected;
5755 
5756   if (str.IsThinString()) {
5757     str = i::ThinString::cast(str).actual();
5758   }
5759 
5760   if (i::StringShape(str).IsExternalTwoByte()) {
5761     const void* resource = i::ExternalTwoByteString::cast(str).resource();
5762     expected = reinterpret_cast<const ExternalStringResource*>(resource);
5763   } else {
5764     expected = nullptr;
5765   }
5766   CHECK_EQ(expected, value);
5767 }
5768 
VerifyExternalStringResourceBase(v8::String::ExternalStringResourceBase * value,Encoding encoding) const5769 void v8::String::VerifyExternalStringResourceBase(
5770     v8::String::ExternalStringResourceBase* value, Encoding encoding) const {
5771   i::DisallowGarbageCollection no_gc;
5772   i::String str = *Utils::OpenHandle(this);
5773   const v8::String::ExternalStringResourceBase* expected;
5774   Encoding expectedEncoding;
5775 
5776   if (str.IsThinString()) {
5777     str = i::ThinString::cast(str).actual();
5778   }
5779 
5780   if (i::StringShape(str).IsExternalOneByte()) {
5781     const void* resource = i::ExternalOneByteString::cast(str).resource();
5782     expected = reinterpret_cast<const ExternalStringResourceBase*>(resource);
5783     expectedEncoding = ONE_BYTE_ENCODING;
5784   } else if (i::StringShape(str).IsExternalTwoByte()) {
5785     const void* resource = i::ExternalTwoByteString::cast(str).resource();
5786     expected = reinterpret_cast<const ExternalStringResourceBase*>(resource);
5787     expectedEncoding = TWO_BYTE_ENCODING;
5788   } else {
5789     expected = nullptr;
5790     expectedEncoding =
5791         str.IsOneByteRepresentation() ? ONE_BYTE_ENCODING : TWO_BYTE_ENCODING;
5792   }
5793   CHECK_EQ(expected, value);
5794   CHECK_EQ(expectedEncoding, encoding);
5795 }
5796 
GetExternalStringResourceSlow() const5797 String::ExternalStringResource* String::GetExternalStringResourceSlow() const {
5798   i::DisallowGarbageCollection no_gc;
5799   using I = internal::Internals;
5800   i::String str = *Utils::OpenHandle(this);
5801 
5802   if (str.IsThinString()) {
5803     str = i::ThinString::cast(str).actual();
5804   }
5805 
5806   if (i::StringShape(str).IsExternalTwoByte()) {
5807     internal::Isolate* isolate = I::GetIsolateForSandbox(str.ptr());
5808     internal::Address value = I::ReadExternalPointerField(
5809         isolate, str.ptr(), I::kStringResourceOffset,
5810         internal::kExternalStringResourceTag);
5811     return reinterpret_cast<String::ExternalStringResource*>(value);
5812   }
5813   return nullptr;
5814 }
5815 
UpdateDataCache()5816 void String::ExternalStringResource::UpdateDataCache() {
5817   DCHECK(IsCacheable());
5818   cached_data_ = data();
5819 }
5820 
CheckCachedDataInvariants() const5821 void String::ExternalStringResource::CheckCachedDataInvariants() const {
5822   DCHECK(IsCacheable() && cached_data_ != nullptr);
5823 }
5824 
UpdateDataCache()5825 void String::ExternalOneByteStringResource::UpdateDataCache() {
5826   DCHECK(IsCacheable());
5827   cached_data_ = data();
5828 }
5829 
CheckCachedDataInvariants() const5830 void String::ExternalOneByteStringResource::CheckCachedDataInvariants() const {
5831   DCHECK(IsCacheable() && cached_data_ != nullptr);
5832 }
5833 
GetExternalStringResourceBaseSlow(String::Encoding * encoding_out) const5834 String::ExternalStringResourceBase* String::GetExternalStringResourceBaseSlow(
5835     String::Encoding* encoding_out) const {
5836   i::DisallowGarbageCollection no_gc;
5837   using I = internal::Internals;
5838   ExternalStringResourceBase* resource = nullptr;
5839   i::String str = *Utils::OpenHandle(this);
5840 
5841   if (str.IsThinString()) {
5842     str = i::ThinString::cast(str).actual();
5843   }
5844 
5845   internal::Address string = str.ptr();
5846   int type =
5847       I::GetInstanceType(string) & I::kStringRepresentationAndEncodingMask;
5848   *encoding_out = static_cast<Encoding>(type & I::kStringEncodingMask);
5849   if (i::StringShape(str).IsExternalOneByte() ||
5850       i::StringShape(str).IsExternalTwoByte()) {
5851     internal::Isolate* isolate = I::GetIsolateForSandbox(string);
5852     internal::Address value =
5853         I::ReadExternalPointerField(isolate, string, I::kStringResourceOffset,
5854                                     internal::kExternalStringResourceTag);
5855     resource = reinterpret_cast<ExternalStringResourceBase*>(value);
5856   }
5857   return resource;
5858 }
5859 
5860 const v8::String::ExternalOneByteStringResource*
GetExternalOneByteStringResource() const5861 v8::String::GetExternalOneByteStringResource() const {
5862   i::DisallowGarbageCollection no_gc;
5863   i::String str = *Utils::OpenHandle(this);
5864   if (i::StringShape(str).IsExternalOneByte()) {
5865     return i::ExternalOneByteString::cast(str).resource();
5866   } else if (str.IsThinString()) {
5867     str = i::ThinString::cast(str).actual();
5868     if (i::StringShape(str).IsExternalOneByte()) {
5869       return i::ExternalOneByteString::cast(str).resource();
5870     }
5871   }
5872   return nullptr;
5873 }
5874 
Description(Isolate * isolate) const5875 Local<Value> Symbol::Description(Isolate* isolate) const {
5876   i::Handle<i::Symbol> sym = Utils::OpenHandle(this);
5877   i::Handle<i::Object> description(sym->description(),
5878                                    reinterpret_cast<i::Isolate*>(isolate));
5879   return Utils::ToLocal(description);
5880 }
5881 
Name() const5882 Local<Value> Private::Name() const {
5883   const Symbol* sym = reinterpret_cast<const Symbol*>(this);
5884   i::Handle<i::Symbol> i_sym = Utils::OpenHandle(sym);
5885   // v8::Private symbols are created by API and are therefore writable, so we
5886   // can always recover an Isolate.
5887   i::Isolate* isolate = i::GetIsolateFromWritableObject(*i_sym);
5888   return sym->Description(reinterpret_cast<Isolate*>(isolate));
5889 }
5890 
Value() const5891 double Number::Value() const {
5892   i::Handle<i::Object> obj = Utils::OpenHandle(this);
5893   return obj->Number();
5894 }
5895 
Value() const5896 bool Boolean::Value() const {
5897   i::Handle<i::Object> obj = Utils::OpenHandle(this);
5898   return obj->IsTrue();
5899 }
5900 
Value() const5901 int64_t Integer::Value() const {
5902   i::Object obj = *Utils::OpenHandle(this);
5903   if (obj.IsSmi()) {
5904     return i::Smi::ToInt(obj);
5905   } else {
5906     return static_cast<int64_t>(obj.Number());
5907   }
5908 }
5909 
Value() const5910 int32_t Int32::Value() const {
5911   i::Object obj = *Utils::OpenHandle(this);
5912   if (obj.IsSmi()) {
5913     return i::Smi::ToInt(obj);
5914   } else {
5915     return static_cast<int32_t>(obj.Number());
5916   }
5917 }
5918 
Value() const5919 uint32_t Uint32::Value() const {
5920   i::Object obj = *Utils::OpenHandle(this);
5921   if (obj.IsSmi()) {
5922     return i::Smi::ToInt(obj);
5923   } else {
5924     return static_cast<uint32_t>(obj.Number());
5925   }
5926 }
5927 
InternalFieldCount() const5928 int v8::Object::InternalFieldCount() const {
5929   i::JSReceiver self = *Utils::OpenHandle(this);
5930   if (!self.IsJSObject()) return 0;
5931   return i::JSObject::cast(self).GetEmbedderFieldCount();
5932 }
5933 
InternalFieldOK(i::Handle<i::JSReceiver> obj,int index,const char * location)5934 static bool InternalFieldOK(i::Handle<i::JSReceiver> obj, int index,
5935                             const char* location) {
5936   return Utils::ApiCheck(
5937       obj->IsJSObject() &&
5938           (index < i::Handle<i::JSObject>::cast(obj)->GetEmbedderFieldCount()),
5939       location, "Internal field out of bounds");
5940 }
5941 
SlowGetInternalField(int index)5942 Local<Value> v8::Object::SlowGetInternalField(int index) {
5943   i::Handle<i::JSReceiver> obj = Utils::OpenHandle(this);
5944   const char* location = "v8::Object::GetInternalField()";
5945   if (!InternalFieldOK(obj, index, location)) return Local<Value>();
5946   i::Handle<i::Object> value(i::JSObject::cast(*obj).GetEmbedderField(index),
5947                              obj->GetIsolate());
5948   return Utils::ToLocal(value);
5949 }
5950 
5951 template<typename T>
SetInternalFieldImpl(v8::Object * receiver,int index,v8::Local<T> value)5952 void SetInternalFieldImpl(v8::Object* receiver, int index, v8::Local<T> value) {
5953   i::Handle<i::JSReceiver> obj = Utils::OpenHandle(receiver);
5954   const char* location = "v8::Object::SetInternalField()";
5955   if (!InternalFieldOK(obj, index, location)) return;
5956   i::Handle<i::Object> val = Utils::OpenHandle(*value);
5957   i::Handle<i::JSObject>::cast(obj)->SetEmbedderField(index, *val);
5958 }
5959 
SetInternalField(int index,v8::Local<Value> value)5960 void v8::Object::SetInternalField(int index, v8::Local<Value> value) {
5961   SetInternalFieldImpl(this, index, value);
5962 }
5963 
5964 /**
5965  * These are Node.js-specific extentions used to avoid breaking changes in
5966  * Node.js v20.x.
5967  */
SetInternalFieldForNodeCore(int index,v8::Local<Module> value)5968 void v8::Object::SetInternalFieldForNodeCore(int index,
5969                                              v8::Local<Module> value) {
5970   SetInternalFieldImpl(this, index, value);
5971 }
5972 
SetInternalFieldForNodeCore(int index,v8::Local<UnboundScript> value)5973 void v8::Object::SetInternalFieldForNodeCore(int index,
5974                                              v8::Local<UnboundScript> value) {
5975   SetInternalFieldImpl(this, index, value);
5976 }
5977 
SlowGetAlignedPointerFromInternalField(int index)5978 void* v8::Object::SlowGetAlignedPointerFromInternalField(int index) {
5979   i::Handle<i::JSReceiver> obj = Utils::OpenHandle(this);
5980   const char* location = "v8::Object::GetAlignedPointerFromInternalField()";
5981   if (!InternalFieldOK(obj, index, location)) return nullptr;
5982   void* result;
5983   Utils::ApiCheck(i::EmbedderDataSlot(i::JSObject::cast(*obj), index)
5984                       .ToAlignedPointer(obj->GetIsolate(), &result),
5985                   location, "Unaligned pointer");
5986   return result;
5987 }
5988 
SetAlignedPointerInInternalField(int index,void * value)5989 void v8::Object::SetAlignedPointerInInternalField(int index, void* value) {
5990   i::Handle<i::JSReceiver> obj = Utils::OpenHandle(this);
5991   const char* location = "v8::Object::SetAlignedPointerInInternalField()";
5992   if (!InternalFieldOK(obj, index, location)) return;
5993 
5994   i::DisallowGarbageCollection no_gc;
5995 
5996   // There's no need to invalidate slots as embedder fields are always
5997   // tagged.
5998   obj->GetHeap()->NotifyObjectLayoutChange(*obj, no_gc,
5999                                            i::InvalidateRecordedSlots::kNo);
6000 
6001   Utils::ApiCheck(i::EmbedderDataSlot(i::JSObject::cast(*obj), index)
6002                       .store_aligned_pointer(obj->GetIsolate(), value),
6003                   location, "Unaligned pointer");
6004   DCHECK_EQ(value, GetAlignedPointerFromInternalField(index));
6005   internal::WriteBarrier::MarkingFromInternalFields(i::JSObject::cast(*obj));
6006 
6007 #ifdef VERIFY_HEAP
6008   obj->GetHeap()->VerifyObjectLayoutChange(*obj, obj->map());
6009 #endif  // VERIFY_HEAP
6010 }
6011 
SetAlignedPointerInInternalFields(int argc,int indices[],void * values[])6012 void v8::Object::SetAlignedPointerInInternalFields(int argc, int indices[],
6013                                                    void* values[]) {
6014   i::Handle<i::JSReceiver> obj = Utils::OpenHandle(this);
6015 
6016   i::DisallowGarbageCollection no_gc;
6017   // There's no need to invalidate slots as embedder fields are always
6018   // tagged.
6019   obj->GetHeap()->NotifyObjectLayoutChange(*obj, no_gc,
6020                                            i::InvalidateRecordedSlots::kNo);
6021 
6022   const char* location = "v8::Object::SetAlignedPointerInInternalFields()";
6023   i::JSObject js_obj = i::JSObject::cast(*obj);
6024   int nof_embedder_fields = js_obj.GetEmbedderFieldCount();
6025   for (int i = 0; i < argc; i++) {
6026     int index = indices[i];
6027     if (!Utils::ApiCheck(index < nof_embedder_fields, location,
6028                          "Internal field out of bounds")) {
6029       return;
6030     }
6031     void* value = values[i];
6032     Utils::ApiCheck(i::EmbedderDataSlot(js_obj, index)
6033                         .store_aligned_pointer(obj->GetIsolate(), value),
6034                     location, "Unaligned pointer");
6035     DCHECK_EQ(value, GetAlignedPointerFromInternalField(index));
6036   }
6037   internal::WriteBarrier::MarkingFromInternalFields(js_obj);
6038 
6039 #ifdef VERIFY_HEAP
6040   obj->GetHeap()->VerifyObjectLayoutChange(*obj, obj->map());
6041 #endif  // VERIFY_HEAP
6042 }
6043 
6044 // --- E n v i r o n m e n t ---
6045 
InitializePlatform(Platform * platform)6046 void v8::V8::InitializePlatform(Platform* platform) {
6047   i::V8::InitializePlatform(platform);
6048 }
6049 
6050 #ifdef V8_SANDBOX
InitializeSandbox()6051 bool v8::V8::InitializeSandbox() { return i::V8::InitializeSandbox(); }
6052 #endif
6053 
DisposePlatform()6054 void v8::V8::DisposePlatform() { i::V8::DisposePlatform(); }
6055 
Initialize(const int build_config)6056 bool v8::V8::Initialize(const int build_config) {
6057   const bool kEmbedderPointerCompression =
6058       (build_config & kPointerCompression) != 0;
6059   if (kEmbedderPointerCompression != COMPRESS_POINTERS_BOOL) {
6060     FATAL(
6061         "Embedder-vs-V8 build configuration mismatch. On embedder side "
6062         "pointer compression is %s while on V8 side it's %s.",
6063         kEmbedderPointerCompression ? "ENABLED" : "DISABLED",
6064         COMPRESS_POINTERS_BOOL ? "ENABLED" : "DISABLED");
6065   }
6066 
6067   const int kEmbedderSmiValueSize = (build_config & k31BitSmis) ? 31 : 32;
6068   if (kEmbedderSmiValueSize != internal::kSmiValueSize) {
6069     FATAL(
6070         "Embedder-vs-V8 build configuration mismatch. On embedder side "
6071         "Smi value size is %d while on V8 side it's %d.",
6072         kEmbedderSmiValueSize, internal::kSmiValueSize);
6073   }
6074 
6075   const bool kEmbedderSandboxedExternalPointers =
6076       (build_config & kSandboxedExternalPointers) != 0;
6077   if (kEmbedderSandboxedExternalPointers !=
6078       V8_SANDBOXED_EXTERNAL_POINTERS_BOOL) {
6079     FATAL(
6080         "Embedder-vs-V8 build configuration mismatch. On embedder side "
6081         "sandboxed external pointers is %s while on V8 side it's %s.",
6082         kEmbedderSandboxedExternalPointers ? "ENABLED" : "DISABLED",
6083         V8_SANDBOXED_EXTERNAL_POINTERS_BOOL ? "ENABLED" : "DISABLED");
6084   }
6085 
6086   const bool kEmbedderSandbox = (build_config & kSandbox) != 0;
6087   if (kEmbedderSandbox != V8_SANDBOX_BOOL) {
6088     FATAL(
6089         "Embedder-vs-V8 build configuration mismatch. On embedder side "
6090         "sandbox is %s while on V8 side it's %s.",
6091         kEmbedderSandbox ? "ENABLED" : "DISABLED",
6092         V8_SANDBOX_BOOL ? "ENABLED" : "DISABLED");
6093   }
6094 
6095   i::V8::Initialize();
6096   return true;
6097 }
6098 
6099 #if V8_OS_LINUX || V8_OS_DARWIN
TryHandleWebAssemblyTrapPosix(int sig_code,siginfo_t * info,void * context)6100 bool TryHandleWebAssemblyTrapPosix(int sig_code, siginfo_t* info,
6101                                    void* context) {
6102 #if V8_ENABLE_WEBASSEMBLY && V8_TRAP_HANDLER_SUPPORTED
6103   return i::trap_handler::TryHandleSignal(sig_code, info, context);
6104 #else
6105   return false;
6106 #endif
6107 }
6108 #endif
6109 
6110 #if V8_OS_WIN
TryHandleWebAssemblyTrapWindows(EXCEPTION_POINTERS * exception)6111 bool TryHandleWebAssemblyTrapWindows(EXCEPTION_POINTERS* exception) {
6112 #if V8_ENABLE_WEBASSEMBLY && V8_TRAP_HANDLER_SUPPORTED
6113   return i::trap_handler::TryHandleWasmTrap(exception);
6114 #else
6115   return false;
6116 #endif
6117 }
6118 #endif
6119 
EnableWebAssemblyTrapHandler(bool use_v8_signal_handler)6120 bool V8::EnableWebAssemblyTrapHandler(bool use_v8_signal_handler) {
6121 #if V8_ENABLE_WEBASSEMBLY
6122   return v8::internal::trap_handler::EnableTrapHandler(use_v8_signal_handler);
6123 #else
6124   return false;
6125 #endif
6126 }
6127 
6128 #if defined(V8_OS_WIN)
SetUnhandledExceptionCallback(UnhandledExceptionCallback unhandled_exception_callback)6129 void V8::SetUnhandledExceptionCallback(
6130     UnhandledExceptionCallback unhandled_exception_callback) {
6131 #if defined(V8_OS_WIN64)
6132   v8::internal::win64_unwindinfo::SetUnhandledExceptionCallback(
6133       unhandled_exception_callback);
6134 #else
6135   // Not implemented, port needed.
6136 #endif  // V8_OS_WIN64
6137 }
6138 #endif  // V8_OS_WIN
6139 
SetFatalMemoryErrorCallback(v8::OOMErrorCallback oom_error_callback)6140 void v8::V8::SetFatalMemoryErrorCallback(
6141     v8::OOMErrorCallback oom_error_callback) {
6142   g_oom_error_callback = oom_error_callback;
6143 }
6144 
SetEntropySource(EntropySource entropy_source)6145 void v8::V8::SetEntropySource(EntropySource entropy_source) {
6146   base::RandomNumberGenerator::SetEntropySource(entropy_source);
6147 }
6148 
SetReturnAddressLocationResolver(ReturnAddressLocationResolver return_address_resolver)6149 void v8::V8::SetReturnAddressLocationResolver(
6150     ReturnAddressLocationResolver return_address_resolver) {
6151   i::StackFrame::SetReturnAddressLocationResolver(return_address_resolver);
6152 }
6153 
Dispose()6154 bool v8::V8::Dispose() {
6155   i::V8::Dispose();
6156   return true;
6157 }
6158 
SharedMemoryStatistics()6159 SharedMemoryStatistics::SharedMemoryStatistics()
6160     : read_only_space_size_(0),
6161       read_only_space_used_size_(0),
6162       read_only_space_physical_size_(0) {}
6163 
HeapStatistics()6164 HeapStatistics::HeapStatistics()
6165     : total_heap_size_(0),
6166       total_heap_size_executable_(0),
6167       total_physical_size_(0),
6168       total_available_size_(0),
6169       used_heap_size_(0),
6170       heap_size_limit_(0),
6171       malloced_memory_(0),
6172       external_memory_(0),
6173       peak_malloced_memory_(0),
6174       does_zap_garbage_(false),
6175       number_of_native_contexts_(0),
6176       number_of_detached_contexts_(0) {}
6177 
HeapSpaceStatistics()6178 HeapSpaceStatistics::HeapSpaceStatistics()
6179     : space_name_(nullptr),
6180       space_size_(0),
6181       space_used_size_(0),
6182       space_available_size_(0),
6183       physical_space_size_(0) {}
6184 
HeapObjectStatistics()6185 HeapObjectStatistics::HeapObjectStatistics()
6186     : object_type_(nullptr),
6187       object_sub_type_(nullptr),
6188       object_count_(0),
6189       object_size_(0) {}
6190 
HeapCodeStatistics()6191 HeapCodeStatistics::HeapCodeStatistics()
6192     : code_and_metadata_size_(0),
6193       bytecode_and_metadata_size_(0),
6194       external_script_source_size_(0),
6195       cpu_profiler_metadata_size_(0) {}
6196 
InitializeICU(const char * icu_data_file)6197 bool v8::V8::InitializeICU(const char* icu_data_file) {
6198   return i::InitializeICU(icu_data_file);
6199 }
6200 
InitializeICUDefaultLocation(const char * exec_path,const char * icu_data_file)6201 bool v8::V8::InitializeICUDefaultLocation(const char* exec_path,
6202                                           const char* icu_data_file) {
6203   return i::InitializeICUDefaultLocation(exec_path, icu_data_file);
6204 }
6205 
InitializeExternalStartupData(const char * directory_path)6206 void v8::V8::InitializeExternalStartupData(const char* directory_path) {
6207   i::InitializeExternalStartupData(directory_path);
6208 }
6209 
6210 // static
InitializeExternalStartupDataFromFile(const char * snapshot_blob)6211 void v8::V8::InitializeExternalStartupDataFromFile(const char* snapshot_blob) {
6212   i::InitializeExternalStartupDataFromFile(snapshot_blob);
6213 }
6214 
GetVersion()6215 const char* v8::V8::GetVersion() { return i::Version::GetVersion(); }
6216 
6217 #ifdef V8_SANDBOX
GetSandboxAddressSpace()6218 VirtualAddressSpace* v8::V8::GetSandboxAddressSpace() {
6219   Utils::ApiCheck(i::GetProcessWideSandbox()->is_initialized(),
6220                   "v8::V8::GetSandboxAddressSpace",
6221                   "The sandbox must be initialized first.");
6222   return i::GetProcessWideSandbox()->address_space();
6223 }
6224 
GetVirtualMemoryCagePageAllocator()6225 PageAllocator* v8::V8::GetVirtualMemoryCagePageAllocator() {
6226   Utils::ApiCheck(i::GetProcessWideSandbox()->is_initialized(),
6227                   "v8::V8::GetVirtualMemoryCagePageAllocator",
6228                   "The sandbox must be initialized first.");
6229   return i::GetProcessWideSandbox()->page_allocator();
6230 }
6231 
GetSandboxSizeInBytes()6232 size_t v8::V8::GetSandboxSizeInBytes() {
6233   if (!i::GetProcessWideSandbox()->is_initialized()) {
6234     return 0;
6235   } else {
6236     return i::GetProcessWideSandbox()->size();
6237   }
6238 }
6239 
IsSandboxConfiguredSecurely()6240 bool v8::V8::IsSandboxConfiguredSecurely() {
6241   Utils::ApiCheck(i::GetProcessWideSandbox()->is_initialized(),
6242                   "v8::V8::IsSandoxConfiguredSecurely",
6243                   "The sandbox must be initialized first.");
6244   // TODO(saelo) For now, we only treat a partially reserved sandbox as
6245   // insecure. Once we use sandboxed pointers, which assume that the sandbox
6246   // has a fixed size, we'll also treat sandboxes with a smaller size as
6247   // insecure because these pointers can then access memory outside of them.
6248   return !i::GetProcessWideSandbox()->is_partially_reserved();
6249 }
6250 #endif
6251 
GetSharedMemoryStatistics(SharedMemoryStatistics * statistics)6252 void V8::GetSharedMemoryStatistics(SharedMemoryStatistics* statistics) {
6253   i::ReadOnlyHeap::PopulateReadOnlySpaceStatistics(statistics);
6254 }
6255 
6256 template <typename ObjectType>
6257 struct InvokeBootstrapper;
6258 
6259 template <>
6260 struct InvokeBootstrapper<i::Context> {
Invokev8::InvokeBootstrapper6261   i::Handle<i::Context> Invoke(
6262       i::Isolate* isolate, i::MaybeHandle<i::JSGlobalProxy> maybe_global_proxy,
6263       v8::Local<v8::ObjectTemplate> global_proxy_template,
6264       v8::ExtensionConfiguration* extensions, size_t context_snapshot_index,
6265       v8::DeserializeInternalFieldsCallback embedder_fields_deserializer,
6266       v8::MicrotaskQueue* microtask_queue) {
6267     return isolate->bootstrapper()->CreateEnvironment(
6268         maybe_global_proxy, global_proxy_template, extensions,
6269         context_snapshot_index, embedder_fields_deserializer, microtask_queue);
6270   }
6271 };
6272 
6273 template <>
6274 struct InvokeBootstrapper<i::JSGlobalProxy> {
Invokev8::InvokeBootstrapper6275   i::Handle<i::JSGlobalProxy> Invoke(
6276       i::Isolate* isolate, i::MaybeHandle<i::JSGlobalProxy> maybe_global_proxy,
6277       v8::Local<v8::ObjectTemplate> global_proxy_template,
6278       v8::ExtensionConfiguration* extensions, size_t context_snapshot_index,
6279       v8::DeserializeInternalFieldsCallback embedder_fields_deserializer,
6280       v8::MicrotaskQueue* microtask_queue) {
6281     USE(extensions);
6282     USE(context_snapshot_index);
6283     return isolate->bootstrapper()->NewRemoteContext(maybe_global_proxy,
6284                                                      global_proxy_template);
6285   }
6286 };
6287 
6288 template <typename ObjectType>
CreateEnvironment(i::Isolate * isolate,v8::ExtensionConfiguration * extensions,v8::MaybeLocal<ObjectTemplate> maybe_global_template,v8::MaybeLocal<Value> maybe_global_proxy,size_t context_snapshot_index,v8::DeserializeInternalFieldsCallback embedder_fields_deserializer,v8::MicrotaskQueue * microtask_queue)6289 static i::Handle<ObjectType> CreateEnvironment(
6290     i::Isolate* isolate, v8::ExtensionConfiguration* extensions,
6291     v8::MaybeLocal<ObjectTemplate> maybe_global_template,
6292     v8::MaybeLocal<Value> maybe_global_proxy, size_t context_snapshot_index,
6293     v8::DeserializeInternalFieldsCallback embedder_fields_deserializer,
6294     v8::MicrotaskQueue* microtask_queue) {
6295   i::Handle<ObjectType> result;
6296 
6297   {
6298     ENTER_V8_FOR_NEW_CONTEXT(isolate);
6299     v8::Local<ObjectTemplate> proxy_template;
6300     i::Handle<i::FunctionTemplateInfo> proxy_constructor;
6301     i::Handle<i::FunctionTemplateInfo> global_constructor;
6302     i::Handle<i::HeapObject> named_interceptor(
6303         isolate->factory()->undefined_value());
6304     i::Handle<i::HeapObject> indexed_interceptor(
6305         isolate->factory()->undefined_value());
6306 
6307     if (!maybe_global_template.IsEmpty()) {
6308       v8::Local<v8::ObjectTemplate> global_template =
6309           maybe_global_template.ToLocalChecked();
6310       // Make sure that the global_template has a constructor.
6311       global_constructor = EnsureConstructor(isolate, *global_template);
6312 
6313       // Create a fresh template for the global proxy object.
6314       proxy_template =
6315           ObjectTemplate::New(reinterpret_cast<v8::Isolate*>(isolate));
6316       proxy_constructor = EnsureConstructor(isolate, *proxy_template);
6317 
6318       // Set the global template to be the prototype template of
6319       // global proxy template.
6320       i::FunctionTemplateInfo::SetPrototypeTemplate(
6321           isolate, proxy_constructor, Utils::OpenHandle(*global_template));
6322 
6323       proxy_template->SetInternalFieldCount(
6324           global_template->InternalFieldCount());
6325 
6326       // Migrate security handlers from global_template to
6327       // proxy_template.  Temporarily removing access check
6328       // information from the global template.
6329       if (!global_constructor->GetAccessCheckInfo().IsUndefined(isolate)) {
6330         i::FunctionTemplateInfo::SetAccessCheckInfo(
6331             isolate, proxy_constructor,
6332             i::handle(global_constructor->GetAccessCheckInfo(), isolate));
6333         proxy_constructor->set_needs_access_check(
6334             global_constructor->needs_access_check());
6335         global_constructor->set_needs_access_check(false);
6336         i::FunctionTemplateInfo::SetAccessCheckInfo(
6337             isolate, global_constructor,
6338             i::ReadOnlyRoots(isolate).undefined_value_handle());
6339       }
6340 
6341       // Same for other interceptors. If the global constructor has
6342       // interceptors, we need to replace them temporarily with noop
6343       // interceptors, so the map is correctly marked as having interceptors,
6344       // but we don't invoke any.
6345       if (!global_constructor->GetNamedPropertyHandler().IsUndefined(isolate)) {
6346         named_interceptor =
6347             handle(global_constructor->GetNamedPropertyHandler(), isolate);
6348         i::FunctionTemplateInfo::SetNamedPropertyHandler(
6349             isolate, global_constructor,
6350             i::ReadOnlyRoots(isolate).noop_interceptor_info_handle());
6351       }
6352       if (!global_constructor->GetIndexedPropertyHandler().IsUndefined(
6353               isolate)) {
6354         indexed_interceptor =
6355             handle(global_constructor->GetIndexedPropertyHandler(), isolate);
6356         i::FunctionTemplateInfo::SetIndexedPropertyHandler(
6357             isolate, global_constructor,
6358             i::ReadOnlyRoots(isolate).noop_interceptor_info_handle());
6359       }
6360     }
6361 
6362     i::MaybeHandle<i::JSGlobalProxy> maybe_proxy;
6363     if (!maybe_global_proxy.IsEmpty()) {
6364       maybe_proxy = i::Handle<i::JSGlobalProxy>::cast(
6365           Utils::OpenHandle(*maybe_global_proxy.ToLocalChecked()));
6366     }
6367     // Create the environment.
6368     InvokeBootstrapper<ObjectType> invoke;
6369     result = invoke.Invoke(isolate, maybe_proxy, proxy_template, extensions,
6370                            context_snapshot_index, embedder_fields_deserializer,
6371                            microtask_queue);
6372 
6373     // Restore the access check info and interceptors on the global template.
6374     if (!maybe_global_template.IsEmpty()) {
6375       DCHECK(!global_constructor.is_null());
6376       DCHECK(!proxy_constructor.is_null());
6377       i::FunctionTemplateInfo::SetAccessCheckInfo(
6378           isolate, global_constructor,
6379           i::handle(proxy_constructor->GetAccessCheckInfo(), isolate));
6380       global_constructor->set_needs_access_check(
6381           proxy_constructor->needs_access_check());
6382       i::FunctionTemplateInfo::SetNamedPropertyHandler(
6383           isolate, global_constructor, named_interceptor);
6384       i::FunctionTemplateInfo::SetIndexedPropertyHandler(
6385           isolate, global_constructor, indexed_interceptor);
6386     }
6387   }
6388   // Leave V8.
6389 
6390   return result;
6391 }
6392 
NewContext(v8::Isolate * external_isolate,v8::ExtensionConfiguration * extensions,v8::MaybeLocal<ObjectTemplate> global_template,v8::MaybeLocal<Value> global_object,size_t context_snapshot_index,v8::DeserializeInternalFieldsCallback embedder_fields_deserializer,v8::MicrotaskQueue * microtask_queue)6393 Local<Context> NewContext(
6394     v8::Isolate* external_isolate, v8::ExtensionConfiguration* extensions,
6395     v8::MaybeLocal<ObjectTemplate> global_template,
6396     v8::MaybeLocal<Value> global_object, size_t context_snapshot_index,
6397     v8::DeserializeInternalFieldsCallback embedder_fields_deserializer,
6398     v8::MicrotaskQueue* microtask_queue) {
6399   i::Isolate* isolate = reinterpret_cast<i::Isolate*>(external_isolate);
6400   // TODO(jkummerow): This is for crbug.com/713699. Remove it if it doesn't
6401   // fail.
6402   // Sanity-check that the isolate is initialized and usable.
6403   CHECK(isolate->builtins()->code(i::Builtin::kIllegal).IsCodeT());
6404 
6405   TRACE_EVENT_CALL_STATS_SCOPED(isolate, "v8", "V8.NewContext");
6406   API_RCS_SCOPE(isolate, Context, New);
6407   i::HandleScope scope(isolate);
6408   ExtensionConfiguration no_extensions;
6409   if (extensions == nullptr) extensions = &no_extensions;
6410   i::Handle<i::Context> env = CreateEnvironment<i::Context>(
6411       isolate, extensions, global_template, global_object,
6412       context_snapshot_index, embedder_fields_deserializer, microtask_queue);
6413   if (env.is_null()) {
6414     if (isolate->has_pending_exception()) isolate->clear_pending_exception();
6415     return Local<Context>();
6416   }
6417   return Utils::ToLocal(scope.CloseAndEscape(env));
6418 }
6419 
New(v8::Isolate * external_isolate,v8::ExtensionConfiguration * extensions,v8::MaybeLocal<ObjectTemplate> global_template,v8::MaybeLocal<Value> global_object,DeserializeInternalFieldsCallback internal_fields_deserializer,v8::MicrotaskQueue * microtask_queue)6420 Local<Context> v8::Context::New(
6421     v8::Isolate* external_isolate, v8::ExtensionConfiguration* extensions,
6422     v8::MaybeLocal<ObjectTemplate> global_template,
6423     v8::MaybeLocal<Value> global_object,
6424     DeserializeInternalFieldsCallback internal_fields_deserializer,
6425     v8::MicrotaskQueue* microtask_queue) {
6426   return NewContext(external_isolate, extensions, global_template,
6427                     global_object, 0, internal_fields_deserializer,
6428                     microtask_queue);
6429 }
6430 
FromSnapshot(v8::Isolate * external_isolate,size_t context_snapshot_index,v8::DeserializeInternalFieldsCallback embedder_fields_deserializer,v8::ExtensionConfiguration * extensions,MaybeLocal<Value> global_object,v8::MicrotaskQueue * microtask_queue)6431 MaybeLocal<Context> v8::Context::FromSnapshot(
6432     v8::Isolate* external_isolate, size_t context_snapshot_index,
6433     v8::DeserializeInternalFieldsCallback embedder_fields_deserializer,
6434     v8::ExtensionConfiguration* extensions, MaybeLocal<Value> global_object,
6435     v8::MicrotaskQueue* microtask_queue) {
6436   size_t index_including_default_context = context_snapshot_index + 1;
6437   if (!i::Snapshot::HasContextSnapshot(
6438           reinterpret_cast<i::Isolate*>(external_isolate),
6439           index_including_default_context)) {
6440     return MaybeLocal<Context>();
6441   }
6442   return NewContext(external_isolate, extensions, MaybeLocal<ObjectTemplate>(),
6443                     global_object, index_including_default_context,
6444                     embedder_fields_deserializer, microtask_queue);
6445 }
6446 
NewRemoteContext(v8::Isolate * external_isolate,v8::Local<ObjectTemplate> global_template,v8::MaybeLocal<v8::Value> global_object)6447 MaybeLocal<Object> v8::Context::NewRemoteContext(
6448     v8::Isolate* external_isolate, v8::Local<ObjectTemplate> global_template,
6449     v8::MaybeLocal<v8::Value> global_object) {
6450   i::Isolate* isolate = reinterpret_cast<i::Isolate*>(external_isolate);
6451   API_RCS_SCOPE(isolate, Context, NewRemoteContext);
6452   i::HandleScope scope(isolate);
6453   i::Handle<i::FunctionTemplateInfo> global_constructor =
6454       EnsureConstructor(isolate, *global_template);
6455   Utils::ApiCheck(global_constructor->needs_access_check(),
6456                   "v8::Context::NewRemoteContext",
6457                   "Global template needs to have access checks enabled.");
6458   i::Handle<i::AccessCheckInfo> access_check_info = i::handle(
6459       i::AccessCheckInfo::cast(global_constructor->GetAccessCheckInfo()),
6460       isolate);
6461   Utils::ApiCheck(access_check_info->named_interceptor() != i::Object(),
6462                   "v8::Context::NewRemoteContext",
6463                   "Global template needs to have access check handlers.");
6464   i::Handle<i::JSObject> global_proxy = CreateEnvironment<i::JSGlobalProxy>(
6465       isolate, nullptr, global_template, global_object, 0,
6466       DeserializeInternalFieldsCallback(), nullptr);
6467   if (global_proxy.is_null()) {
6468     if (isolate->has_pending_exception()) isolate->clear_pending_exception();
6469     return MaybeLocal<Object>();
6470   }
6471   return Utils::ToLocal(scope.CloseAndEscape(global_proxy));
6472 }
6473 
SetSecurityToken(Local<Value> token)6474 void v8::Context::SetSecurityToken(Local<Value> token) {
6475   i::Handle<i::Context> env = Utils::OpenHandle(this);
6476   i::Handle<i::Object> token_handle = Utils::OpenHandle(*token);
6477   env->set_security_token(*token_handle);
6478 }
6479 
UseDefaultSecurityToken()6480 void v8::Context::UseDefaultSecurityToken() {
6481   i::Handle<i::Context> env = Utils::OpenHandle(this);
6482   env->set_security_token(env->global_object());
6483 }
6484 
GetSecurityToken()6485 Local<Value> v8::Context::GetSecurityToken() {
6486   i::Handle<i::Context> env = Utils::OpenHandle(this);
6487   i::Isolate* isolate = env->GetIsolate();
6488   i::Object security_token = env->security_token();
6489   i::Handle<i::Object> token_handle(security_token, isolate);
6490   return Utils::ToLocal(token_handle);
6491 }
6492 
GetIsolate()6493 v8::Isolate* Context::GetIsolate() {
6494   i::Handle<i::Context> env = Utils::OpenHandle(this);
6495   return reinterpret_cast<Isolate*>(env->GetIsolate());
6496 }
6497 
GetMicrotaskQueue()6498 v8::MicrotaskQueue* Context::GetMicrotaskQueue() {
6499   i::Handle<i::Context> env = Utils::OpenHandle(this);
6500   Utils::ApiCheck(env->IsNativeContext(), "v8::Context::GetMicrotaskQueue",
6501                   "Must be calld on a native context");
6502   return i::Handle<i::NativeContext>::cast(env)->microtask_queue();
6503 }
6504 
Global()6505 v8::Local<v8::Object> Context::Global() {
6506   i::Handle<i::Context> context = Utils::OpenHandle(this);
6507   i::Isolate* isolate = context->GetIsolate();
6508   i::Handle<i::Object> global(context->global_proxy(), isolate);
6509   // TODO(chromium:324812): This should always return the global proxy
6510   // but can't presently as calls to GetProtoype will return the wrong result.
6511   if (i::Handle<i::JSGlobalProxy>::cast(global)->IsDetachedFrom(
6512           context->global_object())) {
6513     global = i::Handle<i::Object>(context->global_object(), isolate);
6514   }
6515   return Utils::ToLocal(i::Handle<i::JSObject>::cast(global));
6516 }
6517 
DetachGlobal()6518 void Context::DetachGlobal() {
6519   i::Handle<i::Context> context = Utils::OpenHandle(this);
6520   i::Isolate* isolate = context->GetIsolate();
6521   ENTER_V8_NO_SCRIPT_NO_EXCEPTION(isolate);
6522   isolate->DetachGlobal(context);
6523 }
6524 
GetExtrasBindingObject()6525 Local<v8::Object> Context::GetExtrasBindingObject() {
6526   i::Handle<i::Context> context = Utils::OpenHandle(this);
6527   i::Isolate* isolate = context->GetIsolate();
6528   i::Handle<i::JSObject> binding(context->extras_binding_object(), isolate);
6529   return Utils::ToLocal(binding);
6530 }
6531 
AllowCodeGenerationFromStrings(bool allow)6532 void Context::AllowCodeGenerationFromStrings(bool allow) {
6533   i::Handle<i::Context> context = Utils::OpenHandle(this);
6534   i::Isolate* isolate = context->GetIsolate();
6535   ENTER_V8_NO_SCRIPT_NO_EXCEPTION(isolate);
6536   context->set_allow_code_gen_from_strings(
6537       allow ? i::ReadOnlyRoots(isolate).true_value()
6538             : i::ReadOnlyRoots(isolate).false_value());
6539 }
6540 
IsCodeGenerationFromStringsAllowed() const6541 bool Context::IsCodeGenerationFromStringsAllowed() const {
6542   i::Context context = *Utils::OpenHandle(this);
6543   return !context.allow_code_gen_from_strings().IsFalse(context.GetIsolate());
6544 }
6545 
SetErrorMessageForCodeGenerationFromStrings(Local<String> error)6546 void Context::SetErrorMessageForCodeGenerationFromStrings(Local<String> error) {
6547   i::Handle<i::Context> context = Utils::OpenHandle(this);
6548   i::Handle<i::String> error_handle = Utils::OpenHandle(*error);
6549   context->set_error_message_for_code_gen_from_strings(*error_handle);
6550 }
6551 
SetAbortScriptExecution(Context::AbortScriptExecutionCallback callback)6552 void Context::SetAbortScriptExecution(
6553     Context::AbortScriptExecutionCallback callback) {
6554   i::Handle<i::Context> context = Utils::OpenHandle(this);
6555   i::Isolate* isolate = context->GetIsolate();
6556   if (callback == nullptr) {
6557     context->set_script_execution_callback(
6558         i::ReadOnlyRoots(isolate).undefined_value());
6559   } else {
6560     SET_FIELD_WRAPPED(isolate, context, set_script_execution_callback,
6561                       callback);
6562   }
6563 }
6564 
GetContinuationPreservedEmbedderData() const6565 Local<Value> Context::GetContinuationPreservedEmbedderData() const {
6566   i::Handle<i::Context> context = Utils::OpenHandle(this);
6567   i::Isolate* isolate = context->GetIsolate();
6568   i::Handle<i::Object> data(
6569       context->native_context().continuation_preserved_embedder_data(),
6570       isolate);
6571   return ToApiHandle<Object>(data);
6572 }
6573 
SetContinuationPreservedEmbedderData(Local<Value> data)6574 void Context::SetContinuationPreservedEmbedderData(Local<Value> data) {
6575   i::Handle<i::Context> context = Utils::OpenHandle(this);
6576   i::Isolate* isolate = context->GetIsolate();
6577   if (data.IsEmpty())
6578     data = v8::Undefined(reinterpret_cast<v8::Isolate*>(isolate));
6579   context->native_context().set_continuation_preserved_embedder_data(
6580       *i::Handle<i::HeapObject>::cast(Utils::OpenHandle(*data)));
6581 }
6582 
SetPromiseHooks(Local<Function> init_hook,Local<Function> before_hook,Local<Function> after_hook,Local<Function> resolve_hook)6583 void v8::Context::SetPromiseHooks(Local<Function> init_hook,
6584                                   Local<Function> before_hook,
6585                                   Local<Function> after_hook,
6586                                   Local<Function> resolve_hook) {
6587 #ifdef V8_ENABLE_JAVASCRIPT_PROMISE_HOOKS
6588   i::Handle<i::Context> context = Utils::OpenHandle(this);
6589   i::Isolate* isolate = context->GetIsolate();
6590 
6591   i::Handle<i::Object> init = isolate->factory()->undefined_value();
6592   i::Handle<i::Object> before = isolate->factory()->undefined_value();
6593   i::Handle<i::Object> after = isolate->factory()->undefined_value();
6594   i::Handle<i::Object> resolve = isolate->factory()->undefined_value();
6595 
6596   bool has_hook = false;
6597 
6598   if (!init_hook.IsEmpty()) {
6599     init = Utils::OpenHandle(*init_hook);
6600     has_hook = true;
6601   }
6602   if (!before_hook.IsEmpty()) {
6603     before = Utils::OpenHandle(*before_hook);
6604     has_hook = true;
6605   }
6606   if (!after_hook.IsEmpty()) {
6607     after = Utils::OpenHandle(*after_hook);
6608     has_hook = true;
6609   }
6610   if (!resolve_hook.IsEmpty()) {
6611     resolve = Utils::OpenHandle(*resolve_hook);
6612     has_hook = true;
6613   }
6614 
6615   isolate->SetHasContextPromiseHooks(has_hook);
6616 
6617   context->native_context().set_promise_hook_init_function(*init);
6618   context->native_context().set_promise_hook_before_function(*before);
6619   context->native_context().set_promise_hook_after_function(*after);
6620   context->native_context().set_promise_hook_resolve_function(*resolve);
6621 #else   // V8_ENABLE_JAVASCRIPT_PROMISE_HOOKS
6622   Utils::ApiCheck(false, "v8::Context::SetPromiseHook",
6623                   "V8 was compiled without JavaScript Promise hooks");
6624 #endif  // V8_ENABLE_JAVASCRIPT_PROMISE_HOOKS
6625 }
6626 
GetContext(Isolate * isolate,metrics::Recorder::ContextId id)6627 MaybeLocal<Context> metrics::Recorder::GetContext(
6628     Isolate* isolate, metrics::Recorder::ContextId id) {
6629   i::Isolate* i_isolate = reinterpret_cast<i::Isolate*>(isolate);
6630   return i_isolate->GetContextFromRecorderContextId(id);
6631 }
6632 
GetContextId(Local<Context> context)6633 metrics::Recorder::ContextId metrics::Recorder::GetContextId(
6634     Local<Context> context) {
6635   i::Handle<i::Context> i_context = Utils::OpenHandle(*context);
6636   i::Isolate* isolate = i_context->GetIsolate();
6637   return isolate->GetOrRegisterRecorderContextId(
6638       handle(i_context->native_context(), isolate));
6639 }
6640 
Get(v8::Isolate * isolate)6641 metrics::LongTaskStats metrics::LongTaskStats::Get(v8::Isolate* isolate) {
6642   i::Isolate* i_isolate = reinterpret_cast<i::Isolate*>(isolate);
6643   return *i_isolate->GetCurrentLongTaskStats();
6644 }
6645 
6646 namespace {
GetSerializedDataFromFixedArray(i::Isolate * isolate,i::FixedArray list,size_t index)6647 i::Address* GetSerializedDataFromFixedArray(i::Isolate* isolate,
6648                                             i::FixedArray list, size_t index) {
6649   if (index < static_cast<size_t>(list.length())) {
6650     int int_index = static_cast<int>(index);
6651     i::Object object = list.get(int_index);
6652     if (!object.IsTheHole(isolate)) {
6653       list.set_the_hole(isolate, int_index);
6654       // Shrink the list so that the last element is not the hole (unless it's
6655       // the first element, because we don't want to end up with a non-canonical
6656       // empty FixedArray).
6657       int last = list.length() - 1;
6658       while (last >= 0 && list.is_the_hole(isolate, last)) last--;
6659       if (last != -1) list.Shrink(isolate, last + 1);
6660       return i::Handle<i::Object>(object, isolate).location();
6661     }
6662   }
6663   return nullptr;
6664 }
6665 }  // anonymous namespace
6666 
GetDataFromSnapshotOnce(size_t index)6667 i::Address* Context::GetDataFromSnapshotOnce(size_t index) {
6668   auto context = Utils::OpenHandle(this);
6669   i::Isolate* i_isolate = context->GetIsolate();
6670   i::FixedArray list = context->serialized_objects();
6671   return GetSerializedDataFromFixedArray(i_isolate, list, index);
6672 }
6673 
NewInstance(Local<Context> context)6674 MaybeLocal<v8::Object> ObjectTemplate::NewInstance(Local<Context> context) {
6675   PREPARE_FOR_EXECUTION(context, ObjectTemplate, NewInstance, Object);
6676   auto self = Utils::OpenHandle(this);
6677   Local<Object> result;
6678   has_pending_exception = !ToLocal<Object>(
6679       i::ApiNatives::InstantiateObject(isolate, self), &result);
6680   RETURN_ON_FAILED_EXECUTION(Object);
6681   RETURN_ESCAPED(result);
6682 }
6683 
CheckCast(Data * that)6684 void v8::ObjectTemplate::CheckCast(Data* that) {
6685   i::Handle<i::Object> obj = Utils::OpenHandle(that);
6686   Utils::ApiCheck(obj->IsObjectTemplateInfo(), "v8::ObjectTemplate::Cast",
6687                   "Value is not an ObjectTemplate");
6688 }
6689 
CheckCast(Data * that)6690 void v8::FunctionTemplate::CheckCast(Data* that) {
6691   i::Handle<i::Object> obj = Utils::OpenHandle(that);
6692   Utils::ApiCheck(obj->IsFunctionTemplateInfo(), "v8::FunctionTemplate::Cast",
6693                   "Value is not a FunctionTemplate");
6694 }
6695 
CheckCast(Data * that)6696 void v8::Signature::CheckCast(Data* that) {
6697   i::Handle<i::Object> obj = Utils::OpenHandle(that);
6698   Utils::ApiCheck(obj->IsFunctionTemplateInfo(), "v8::Signature::Cast",
6699                   "Value is not a Signature");
6700 }
6701 
CheckCast(Data * that)6702 void v8::AccessorSignature::CheckCast(Data* that) {
6703   i::Handle<i::Object> obj = Utils::OpenHandle(that);
6704   Utils::ApiCheck(obj->IsFunctionTemplateInfo(), "v8::AccessorSignature::Cast",
6705                   "Value is not an AccessorSignature");
6706 }
6707 
GetFunction(Local<Context> context)6708 MaybeLocal<v8::Function> FunctionTemplate::GetFunction(Local<Context> context) {
6709   PREPARE_FOR_EXECUTION(context, FunctionTemplate, GetFunction, Function);
6710   auto self = Utils::OpenHandle(this);
6711   Local<Function> result;
6712   has_pending_exception =
6713       !ToLocal<Function>(i::ApiNatives::InstantiateFunction(self), &result);
6714   RETURN_ON_FAILED_EXECUTION(Function);
6715   RETURN_ESCAPED(result);
6716 }
6717 
NewRemoteInstance()6718 MaybeLocal<v8::Object> FunctionTemplate::NewRemoteInstance() {
6719   auto self = Utils::OpenHandle(this);
6720   i::Isolate* isolate = self->GetIsolate();
6721   API_RCS_SCOPE(isolate, FunctionTemplate, NewRemoteInstance);
6722   i::HandleScope scope(isolate);
6723   i::Handle<i::FunctionTemplateInfo> constructor =
6724       EnsureConstructor(isolate, *InstanceTemplate());
6725   Utils::ApiCheck(constructor->needs_access_check(),
6726                   "v8::FunctionTemplate::NewRemoteInstance",
6727                   "InstanceTemplate needs to have access checks enabled.");
6728   i::Handle<i::AccessCheckInfo> access_check_info = i::handle(
6729       i::AccessCheckInfo::cast(constructor->GetAccessCheckInfo()), isolate);
6730   Utils::ApiCheck(access_check_info->named_interceptor() != i::Object(),
6731                   "v8::FunctionTemplate::NewRemoteInstance",
6732                   "InstanceTemplate needs to have access check handlers.");
6733   i::Handle<i::JSObject> object;
6734   if (!i::ApiNatives::InstantiateRemoteObject(
6735            Utils::OpenHandle(*InstanceTemplate()))
6736            .ToHandle(&object)) {
6737     if (isolate->has_pending_exception()) {
6738       isolate->OptionalRescheduleException(true);
6739     }
6740     return MaybeLocal<Object>();
6741   }
6742   return Utils::ToLocal(scope.CloseAndEscape(object));
6743 }
6744 
HasInstance(v8::Local<v8::Value> value)6745 bool FunctionTemplate::HasInstance(v8::Local<v8::Value> value) {
6746   auto self = Utils::OpenHandle(this);
6747   auto obj = Utils::OpenHandle(*value);
6748   if (obj->IsJSObject() && self->IsTemplateFor(i::JSObject::cast(*obj))) {
6749     return true;
6750   }
6751   if (obj->IsJSGlobalProxy()) {
6752     // If it's a global proxy, then test with the global object. Note that the
6753     // inner global object may not necessarily be a JSGlobalObject.
6754     i::PrototypeIterator iter(self->GetIsolate(),
6755                               i::JSObject::cast(*obj).map());
6756     // The global proxy should always have a prototype, as it is a bug to call
6757     // this on a detached JSGlobalProxy.
6758     DCHECK(!iter.IsAtEnd());
6759     return self->IsTemplateFor(iter.GetCurrent<i::JSObject>());
6760   }
6761   return false;
6762 }
6763 
IsLeafTemplateForApiObject(v8::Local<v8::Value> value) const6764 bool FunctionTemplate::IsLeafTemplateForApiObject(
6765     v8::Local<v8::Value> value) const {
6766   i::DisallowGarbageCollection no_gc;
6767 
6768   i::Object object = *Utils::OpenHandle(*value);
6769 
6770   auto self = Utils::OpenHandle(this);
6771   return self->IsLeafTemplateForApiObject(object);
6772 }
6773 
New(Isolate * isolate,void * value)6774 Local<External> v8::External::New(Isolate* isolate, void* value) {
6775   STATIC_ASSERT(sizeof(value) == sizeof(i::Address));
6776   // Nullptr is not allowed here because serialization/deserialization of
6777   // nullptr external api references is not possible as nullptr is used as an
6778   // external_references table terminator, see v8::SnapshotCreator()
6779   // constructors.
6780   DCHECK_NOT_NULL(value);
6781   i::Isolate* i_isolate = reinterpret_cast<i::Isolate*>(isolate);
6782   API_RCS_SCOPE(i_isolate, External, New);
6783   ENTER_V8_NO_SCRIPT_NO_EXCEPTION(i_isolate);
6784   i::Handle<i::JSObject> external = i_isolate->factory()->NewExternal(value);
6785   return Utils::ExternalToLocal(external);
6786 }
6787 
Value() const6788 void* External::Value() const {
6789   auto self = Utils::OpenHandle(this);
6790   return i::JSExternalObject::cast(*self).value();
6791 }
6792 
6793 // anonymous namespace for string creation helper functions
6794 namespace {
6795 
StringLength(const char * string)6796 inline int StringLength(const char* string) {
6797   size_t len = strlen(string);
6798   CHECK_GE(i::kMaxInt, len);
6799   return static_cast<int>(len);
6800 }
6801 
StringLength(const uint8_t * string)6802 inline int StringLength(const uint8_t* string) {
6803   return StringLength(reinterpret_cast<const char*>(string));
6804 }
6805 
StringLength(const uint16_t * string)6806 inline int StringLength(const uint16_t* string) {
6807   size_t length = 0;
6808   while (string[length] != '\0') length++;
6809   CHECK_GE(i::kMaxInt, length);
6810   return static_cast<int>(length);
6811 }
6812 
6813 V8_WARN_UNUSED_RESULT
NewString(i::Factory * factory,NewStringType type,base::Vector<const char> string)6814 inline i::MaybeHandle<i::String> NewString(i::Factory* factory,
6815                                            NewStringType type,
6816                                            base::Vector<const char> string) {
6817   if (type == NewStringType::kInternalized) {
6818     return factory->InternalizeUtf8String(string);
6819   }
6820   return factory->NewStringFromUtf8(string);
6821 }
6822 
6823 V8_WARN_UNUSED_RESULT
NewString(i::Factory * factory,NewStringType type,base::Vector<const uint8_t> string)6824 inline i::MaybeHandle<i::String> NewString(i::Factory* factory,
6825                                            NewStringType type,
6826                                            base::Vector<const uint8_t> string) {
6827   if (type == NewStringType::kInternalized) {
6828     return factory->InternalizeString(string);
6829   }
6830   return factory->NewStringFromOneByte(string);
6831 }
6832 
6833 V8_WARN_UNUSED_RESULT
NewString(i::Factory * factory,NewStringType type,base::Vector<const uint16_t> string)6834 inline i::MaybeHandle<i::String> NewString(
6835     i::Factory* factory, NewStringType type,
6836     base::Vector<const uint16_t> string) {
6837   if (type == NewStringType::kInternalized) {
6838     return factory->InternalizeString(string);
6839   }
6840   return factory->NewStringFromTwoByte(string);
6841 }
6842 
6843 STATIC_ASSERT(v8::String::kMaxLength == i::String::kMaxLength);
6844 
6845 }  // anonymous namespace
6846 
6847 // TODO(dcarney): throw a context free exception.
6848 #define NEW_STRING(isolate, class_name, function_name, Char, data, type,   \
6849                    length)                                                 \
6850   MaybeLocal<String> result;                                               \
6851   if (length == 0) {                                                       \
6852     result = String::Empty(isolate);                                       \
6853   } else if (length > i::String::kMaxLength) {                             \
6854     result = MaybeLocal<String>();                                         \
6855   } else {                                                                 \
6856     i::Isolate* i_isolate = reinterpret_cast<internal::Isolate*>(isolate); \
6857     ENTER_V8_NO_SCRIPT_NO_EXCEPTION(i_isolate);                            \
6858     API_RCS_SCOPE(i_isolate, class_name, function_name);                   \
6859     if (length < 0) length = StringLength(data);                           \
6860     i::Handle<i::String> handle_result =                                   \
6861         NewString(i_isolate->factory(), type,                              \
6862                   base::Vector<const Char>(data, length))                  \
6863             .ToHandleChecked();                                            \
6864     result = Utils::ToLocal(handle_result);                                \
6865   }
6866 
NewFromUtf8Literal(Isolate * isolate,const char * literal,NewStringType type,int length)6867 Local<String> String::NewFromUtf8Literal(Isolate* isolate, const char* literal,
6868                                          NewStringType type, int length) {
6869   DCHECK_LE(length, i::String::kMaxLength);
6870   i::Isolate* i_isolate = reinterpret_cast<internal::Isolate*>(isolate);
6871   ENTER_V8_NO_SCRIPT_NO_EXCEPTION(i_isolate);
6872   API_RCS_SCOPE(i_isolate, String, NewFromUtf8Literal);
6873   i::Handle<i::String> handle_result =
6874       NewString(i_isolate->factory(), type,
6875                 base::Vector<const char>(literal, length))
6876           .ToHandleChecked();
6877   return Utils::ToLocal(handle_result);
6878 }
6879 
NewFromUtf8(Isolate * isolate,const char * data,NewStringType type,int length)6880 MaybeLocal<String> String::NewFromUtf8(Isolate* isolate, const char* data,
6881                                        NewStringType type, int length) {
6882   NEW_STRING(isolate, String, NewFromUtf8, char, data, type, length);
6883   return result;
6884 }
6885 
NewFromOneByte(Isolate * isolate,const uint8_t * data,NewStringType type,int length)6886 MaybeLocal<String> String::NewFromOneByte(Isolate* isolate, const uint8_t* data,
6887                                           NewStringType type, int length) {
6888   NEW_STRING(isolate, String, NewFromOneByte, uint8_t, data, type, length);
6889   return result;
6890 }
6891 
NewFromTwoByte(Isolate * isolate,const uint16_t * data,NewStringType type,int length)6892 MaybeLocal<String> String::NewFromTwoByte(Isolate* isolate,
6893                                           const uint16_t* data,
6894                                           NewStringType type, int length) {
6895   NEW_STRING(isolate, String, NewFromTwoByte, uint16_t, data, type, length);
6896   return result;
6897 }
6898 
Concat(Isolate * v8_isolate,Local<String> left,Local<String> right)6899 Local<String> v8::String::Concat(Isolate* v8_isolate, Local<String> left,
6900                                  Local<String> right) {
6901   i::Isolate* isolate = reinterpret_cast<i::Isolate*>(v8_isolate);
6902   i::Handle<i::String> left_string = Utils::OpenHandle(*left);
6903   ENTER_V8_NO_SCRIPT_NO_EXCEPTION(isolate);
6904   API_RCS_SCOPE(isolate, String, Concat);
6905   i::Handle<i::String> right_string = Utils::OpenHandle(*right);
6906   // If we are steering towards a range error, do not wait for the error to be
6907   // thrown, and return the null handle instead.
6908   if (left_string->length() + right_string->length() > i::String::kMaxLength) {
6909     return Local<String>();
6910   }
6911   i::Handle<i::String> result = isolate->factory()
6912                                     ->NewConsString(left_string, right_string)
6913                                     .ToHandleChecked();
6914   return Utils::ToLocal(result);
6915 }
6916 
NewExternalTwoByte(Isolate * isolate,v8::String::ExternalStringResource * resource)6917 MaybeLocal<String> v8::String::NewExternalTwoByte(
6918     Isolate* isolate, v8::String::ExternalStringResource* resource) {
6919   CHECK(resource && resource->data());
6920   // TODO(dcarney): throw a context free exception.
6921   if (resource->length() > static_cast<size_t>(i::String::kMaxLength)) {
6922     return MaybeLocal<String>();
6923   }
6924   i::Isolate* i_isolate = reinterpret_cast<i::Isolate*>(isolate);
6925   ENTER_V8_NO_SCRIPT_NO_EXCEPTION(i_isolate);
6926   API_RCS_SCOPE(i_isolate, String, NewExternalTwoByte);
6927   if (resource->length() > 0) {
6928     i::Handle<i::String> string = i_isolate->factory()
6929                                       ->NewExternalStringFromTwoByte(resource)
6930                                       .ToHandleChecked();
6931     return Utils::ToLocal(string);
6932   } else {
6933     // The resource isn't going to be used, free it immediately.
6934     resource->Dispose();
6935     return Utils::ToLocal(i_isolate->factory()->empty_string());
6936   }
6937 }
6938 
NewExternalOneByte(Isolate * isolate,v8::String::ExternalOneByteStringResource * resource)6939 MaybeLocal<String> v8::String::NewExternalOneByte(
6940     Isolate* isolate, v8::String::ExternalOneByteStringResource* resource) {
6941   CHECK_NOT_NULL(resource);
6942   // TODO(dcarney): throw a context free exception.
6943   if (resource->length() > static_cast<size_t>(i::String::kMaxLength)) {
6944     return MaybeLocal<String>();
6945   }
6946   i::Isolate* i_isolate = reinterpret_cast<i::Isolate*>(isolate);
6947   ENTER_V8_NO_SCRIPT_NO_EXCEPTION(i_isolate);
6948   API_RCS_SCOPE(i_isolate, String, NewExternalOneByte);
6949   if (resource->length() == 0) {
6950     // The resource isn't going to be used, free it immediately.
6951     resource->Dispose();
6952     return Utils::ToLocal(i_isolate->factory()->empty_string());
6953   }
6954   CHECK_NOT_NULL(resource->data());
6955   i::Handle<i::String> string = i_isolate->factory()
6956                                     ->NewExternalStringFromOneByte(resource)
6957                                     .ToHandleChecked();
6958   return Utils::ToLocal(string);
6959 }
6960 
MakeExternal(v8::String::ExternalStringResource * resource)6961 bool v8::String::MakeExternal(v8::String::ExternalStringResource* resource) {
6962   i::DisallowGarbageCollection no_gc;
6963 
6964   i::String obj = *Utils::OpenHandle(this);
6965 
6966   if (obj.IsThinString()) {
6967     obj = i::ThinString::cast(obj).actual();
6968   }
6969 
6970   if (!obj.SupportsExternalization()) {
6971     return false;
6972   }
6973 
6974   // It is safe to call GetIsolateFromWritableHeapObject because
6975   // SupportsExternalization already checked that the object is writable.
6976   i::Isolate* isolate = i::GetIsolateFromWritableObject(obj);
6977   ENTER_V8_NO_SCRIPT_NO_EXCEPTION(isolate);
6978 
6979   CHECK(resource && resource->data());
6980 
6981   bool result = obj.MakeExternal(resource);
6982   DCHECK(result);
6983   DCHECK(obj.IsExternalString());
6984   return result;
6985 }
6986 
MakeExternal(v8::String::ExternalOneByteStringResource * resource)6987 bool v8::String::MakeExternal(
6988     v8::String::ExternalOneByteStringResource* resource) {
6989   i::DisallowGarbageCollection no_gc;
6990 
6991   i::String obj = *Utils::OpenHandle(this);
6992 
6993   if (obj.IsThinString()) {
6994     obj = i::ThinString::cast(obj).actual();
6995   }
6996 
6997   if (!obj.SupportsExternalization()) {
6998     return false;
6999   }
7000 
7001   // It is safe to call GetIsolateFromWritableHeapObject because
7002   // SupportsExternalization already checked that the object is writable.
7003   i::Isolate* isolate = i::GetIsolateFromWritableObject(obj);
7004   ENTER_V8_NO_SCRIPT_NO_EXCEPTION(isolate);
7005 
7006   CHECK(resource && resource->data());
7007 
7008   bool result = obj.MakeExternal(resource);
7009   DCHECK_IMPLIES(result, obj.IsExternalString());
7010   return result;
7011 }
7012 
CanMakeExternal() const7013 bool v8::String::CanMakeExternal() const {
7014   i::String obj = *Utils::OpenHandle(this);
7015 
7016   if (obj.IsThinString()) {
7017     obj = i::ThinString::cast(obj).actual();
7018   }
7019 
7020   if (!obj.SupportsExternalization()) {
7021     return false;
7022   }
7023 
7024   // Only old space strings should be externalized.
7025   return !i::Heap::InYoungGeneration(obj);
7026 }
7027 
StringEquals(Local<String> that) const7028 bool v8::String::StringEquals(Local<String> that) const {
7029   auto self = Utils::OpenHandle(this);
7030   auto other = Utils::OpenHandle(*that);
7031   return self->Equals(*other);
7032 }
7033 
GetIsolate()7034 Isolate* v8::Object::GetIsolate() {
7035   i::Isolate* i_isolate = Utils::OpenHandle(this)->GetIsolate();
7036   return reinterpret_cast<Isolate*>(i_isolate);
7037 }
7038 
New(Isolate * isolate)7039 Local<v8::Object> v8::Object::New(Isolate* isolate) {
7040   i::Isolate* i_isolate = reinterpret_cast<i::Isolate*>(isolate);
7041   API_RCS_SCOPE(i_isolate, Object, New);
7042   ENTER_V8_NO_SCRIPT_NO_EXCEPTION(i_isolate);
7043   i::Handle<i::JSObject> obj =
7044       i_isolate->factory()->NewJSObject(i_isolate->object_function());
7045   return Utils::ToLocal(obj);
7046 }
7047 
7048 namespace {
7049 
7050 // TODO(v8:7569): This is a workaround for the Handle vs MaybeHandle difference
7051 // in the return types of the different Add functions:
7052 // OrderedNameDictionary::Add returns MaybeHandle, NameDictionary::Add returns
7053 // Handle.
7054 template <typename T>
ToHandle(i::Handle<T> h)7055 i::Handle<T> ToHandle(i::Handle<T> h) {
7056   return h;
7057 }
7058 template <typename T>
ToHandle(i::MaybeHandle<T> h)7059 i::Handle<T> ToHandle(i::MaybeHandle<T> h) {
7060   return h.ToHandleChecked();
7061 }
7062 
7063 template <typename Dictionary>
AddPropertiesAndElementsToObject(i::Isolate * i_isolate,i::Handle<Dictionary> & properties,i::Handle<i::FixedArrayBase> & elements,Local<Name> * names,Local<Value> * values,size_t length)7064 void AddPropertiesAndElementsToObject(i::Isolate* i_isolate,
7065                                       i::Handle<Dictionary>& properties,
7066                                       i::Handle<i::FixedArrayBase>& elements,
7067                                       Local<Name>* names, Local<Value>* values,
7068                                       size_t length) {
7069   for (size_t i = 0; i < length; ++i) {
7070     i::Handle<i::Name> name = Utils::OpenHandle(*names[i]);
7071     i::Handle<i::Object> value = Utils::OpenHandle(*values[i]);
7072 
7073     // See if the {name} is a valid array index, in which case we need to
7074     // add the {name}/{value} pair to the {elements}, otherwise they end
7075     // up in the {properties} backing store.
7076     uint32_t index;
7077     if (name->AsArrayIndex(&index)) {
7078       // If this is the first element, allocate a proper
7079       // dictionary elements backing store for {elements}.
7080       if (!elements->IsNumberDictionary()) {
7081         elements =
7082             i::NumberDictionary::New(i_isolate, static_cast<int>(length));
7083       }
7084       elements = i::NumberDictionary::Set(
7085           i_isolate, i::Handle<i::NumberDictionary>::cast(elements), index,
7086           value);
7087     } else {
7088       // Internalize the {name} first.
7089       name = i_isolate->factory()->InternalizeName(name);
7090       i::InternalIndex const entry = properties->FindEntry(i_isolate, name);
7091       if (entry.is_not_found()) {
7092         // Add the {name}/{value} pair as a new entry.
7093         properties = ToHandle(Dictionary::Add(
7094             i_isolate, properties, name, value, i::PropertyDetails::Empty()));
7095       } else {
7096         // Overwrite the {entry} with the {value}.
7097         properties->ValueAtPut(entry, *value);
7098       }
7099     }
7100   }
7101 }
7102 
7103 }  // namespace
7104 
New(Isolate * isolate,Local<Value> prototype_or_null,Local<Name> * names,Local<Value> * values,size_t length)7105 Local<v8::Object> v8::Object::New(Isolate* isolate,
7106                                   Local<Value> prototype_or_null,
7107                                   Local<Name>* names, Local<Value>* values,
7108                                   size_t length) {
7109   i::Isolate* i_isolate = reinterpret_cast<i::Isolate*>(isolate);
7110   i::Handle<i::Object> proto = Utils::OpenHandle(*prototype_or_null);
7111   if (!Utils::ApiCheck(proto->IsNull() || proto->IsJSReceiver(),
7112                        "v8::Object::New", "prototype must be null or object")) {
7113     return Local<v8::Object>();
7114   }
7115   API_RCS_SCOPE(i_isolate, Object, New);
7116   ENTER_V8_NO_SCRIPT_NO_EXCEPTION(i_isolate);
7117 
7118   i::Handle<i::FixedArrayBase> elements =
7119       i_isolate->factory()->empty_fixed_array();
7120 
7121   // We assume that this API is mostly used to create objects with named
7122   // properties, and so we default to creating a properties backing store
7123   // large enough to hold all of them, while we start with no elements
7124   // (see http://bit.ly/v8-fast-object-create-cpp for the motivation).
7125   if (V8_ENABLE_SWISS_NAME_DICTIONARY_BOOL) {
7126     i::Handle<i::SwissNameDictionary> properties =
7127         i_isolate->factory()->NewSwissNameDictionary(static_cast<int>(length));
7128     AddPropertiesAndElementsToObject(i_isolate, properties, elements, names,
7129                                      values, length);
7130     i::Handle<i::JSObject> obj =
7131         i_isolate->factory()->NewSlowJSObjectWithPropertiesAndElements(
7132             i::Handle<i::HeapObject>::cast(proto), properties, elements);
7133     return Utils::ToLocal(obj);
7134   } else {
7135     i::Handle<i::NameDictionary> properties =
7136         i::NameDictionary::New(i_isolate, static_cast<int>(length));
7137     AddPropertiesAndElementsToObject(i_isolate, properties, elements, names,
7138                                      values, length);
7139     i::Handle<i::JSObject> obj =
7140         i_isolate->factory()->NewSlowJSObjectWithPropertiesAndElements(
7141             i::Handle<i::HeapObject>::cast(proto), properties, elements);
7142     return Utils::ToLocal(obj);
7143   }
7144 }
7145 
New(Isolate * isolate,double value)7146 Local<v8::Value> v8::NumberObject::New(Isolate* isolate, double value) {
7147   i::Isolate* i_isolate = reinterpret_cast<i::Isolate*>(isolate);
7148   API_RCS_SCOPE(i_isolate, NumberObject, New);
7149   ENTER_V8_NO_SCRIPT_NO_EXCEPTION(i_isolate);
7150   i::Handle<i::Object> number = i_isolate->factory()->NewNumber(value);
7151   i::Handle<i::Object> obj =
7152       i::Object::ToObject(i_isolate, number).ToHandleChecked();
7153   return Utils::ToLocal(obj);
7154 }
7155 
ValueOf() const7156 double v8::NumberObject::ValueOf() const {
7157   i::Handle<i::Object> obj = Utils::OpenHandle(this);
7158   i::Handle<i::JSPrimitiveWrapper> js_primitive_wrapper =
7159       i::Handle<i::JSPrimitiveWrapper>::cast(obj);
7160   API_RCS_SCOPE(js_primitive_wrapper->GetIsolate(), NumberObject, NumberValue);
7161   return js_primitive_wrapper->value().Number();
7162 }
7163 
New(Isolate * isolate,int64_t value)7164 Local<v8::Value> v8::BigIntObject::New(Isolate* isolate, int64_t value) {
7165   i::Isolate* i_isolate = reinterpret_cast<i::Isolate*>(isolate);
7166   API_RCS_SCOPE(i_isolate, BigIntObject, New);
7167   ENTER_V8_NO_SCRIPT_NO_EXCEPTION(i_isolate);
7168   i::Handle<i::Object> bigint = i::BigInt::FromInt64(i_isolate, value);
7169   i::Handle<i::Object> obj =
7170       i::Object::ToObject(i_isolate, bigint).ToHandleChecked();
7171   return Utils::ToLocal(obj);
7172 }
7173 
ValueOf() const7174 Local<v8::BigInt> v8::BigIntObject::ValueOf() const {
7175   i::Handle<i::Object> obj = Utils::OpenHandle(this);
7176   i::Handle<i::JSPrimitiveWrapper> js_primitive_wrapper =
7177       i::Handle<i::JSPrimitiveWrapper>::cast(obj);
7178   i::Isolate* isolate = js_primitive_wrapper->GetIsolate();
7179   API_RCS_SCOPE(isolate, BigIntObject, BigIntValue);
7180   return Utils::ToLocal(i::Handle<i::BigInt>(
7181       i::BigInt::cast(js_primitive_wrapper->value()), isolate));
7182 }
7183 
New(Isolate * isolate,bool value)7184 Local<v8::Value> v8::BooleanObject::New(Isolate* isolate, bool value) {
7185   i::Isolate* i_isolate = reinterpret_cast<i::Isolate*>(isolate);
7186   API_RCS_SCOPE(i_isolate, BooleanObject, New);
7187   ENTER_V8_NO_SCRIPT_NO_EXCEPTION(i_isolate);
7188   i::Handle<i::Object> boolean(value
7189                                    ? i::ReadOnlyRoots(i_isolate).true_value()
7190                                    : i::ReadOnlyRoots(i_isolate).false_value(),
7191                                i_isolate);
7192   i::Handle<i::Object> obj =
7193       i::Object::ToObject(i_isolate, boolean).ToHandleChecked();
7194   return Utils::ToLocal(obj);
7195 }
7196 
ValueOf() const7197 bool v8::BooleanObject::ValueOf() const {
7198   i::Object obj = *Utils::OpenHandle(this);
7199   i::JSPrimitiveWrapper js_primitive_wrapper = i::JSPrimitiveWrapper::cast(obj);
7200   i::Isolate* isolate = js_primitive_wrapper.GetIsolate();
7201   API_RCS_SCOPE(isolate, BooleanObject, BooleanValue);
7202   return js_primitive_wrapper.value().IsTrue(isolate);
7203 }
7204 
New(Isolate * v8_isolate,Local<String> value)7205 Local<v8::Value> v8::StringObject::New(Isolate* v8_isolate,
7206                                        Local<String> value) {
7207   i::Handle<i::String> string = Utils::OpenHandle(*value);
7208   i::Isolate* isolate = reinterpret_cast<i::Isolate*>(v8_isolate);
7209   API_RCS_SCOPE(isolate, StringObject, New);
7210   ENTER_V8_NO_SCRIPT_NO_EXCEPTION(isolate);
7211   i::Handle<i::Object> obj =
7212       i::Object::ToObject(isolate, string).ToHandleChecked();
7213   return Utils::ToLocal(obj);
7214 }
7215 
ValueOf() const7216 Local<v8::String> v8::StringObject::ValueOf() const {
7217   i::Handle<i::Object> obj = Utils::OpenHandle(this);
7218   i::Handle<i::JSPrimitiveWrapper> js_primitive_wrapper =
7219       i::Handle<i::JSPrimitiveWrapper>::cast(obj);
7220   i::Isolate* isolate = js_primitive_wrapper->GetIsolate();
7221   API_RCS_SCOPE(isolate, StringObject, StringValue);
7222   return Utils::ToLocal(i::Handle<i::String>(
7223       i::String::cast(js_primitive_wrapper->value()), isolate));
7224 }
7225 
New(Isolate * isolate,Local<Symbol> value)7226 Local<v8::Value> v8::SymbolObject::New(Isolate* isolate, Local<Symbol> value) {
7227   i::Isolate* i_isolate = reinterpret_cast<i::Isolate*>(isolate);
7228   API_RCS_SCOPE(i_isolate, SymbolObject, New);
7229   ENTER_V8_NO_SCRIPT_NO_EXCEPTION(i_isolate);
7230   i::Handle<i::Object> obj =
7231       i::Object::ToObject(i_isolate, Utils::OpenHandle(*value))
7232           .ToHandleChecked();
7233   return Utils::ToLocal(obj);
7234 }
7235 
ValueOf() const7236 Local<v8::Symbol> v8::SymbolObject::ValueOf() const {
7237   i::Handle<i::Object> obj = Utils::OpenHandle(this);
7238   i::Handle<i::JSPrimitiveWrapper> js_primitive_wrapper =
7239       i::Handle<i::JSPrimitiveWrapper>::cast(obj);
7240   i::Isolate* isolate = js_primitive_wrapper->GetIsolate();
7241   API_RCS_SCOPE(isolate, SymbolObject, SymbolValue);
7242   return Utils::ToLocal(i::Handle<i::Symbol>(
7243       i::Symbol::cast(js_primitive_wrapper->value()), isolate));
7244 }
7245 
New(Local<Context> context,double time)7246 MaybeLocal<v8::Value> v8::Date::New(Local<Context> context, double time) {
7247   if (std::isnan(time)) {
7248     // Introduce only canonical NaN value into the VM, to avoid signaling NaNs.
7249     time = std::numeric_limits<double>::quiet_NaN();
7250   }
7251   PREPARE_FOR_EXECUTION(context, Date, New, Value);
7252   Local<Value> result;
7253   has_pending_exception = !ToLocal<Value>(
7254       i::JSDate::New(isolate->date_function(), isolate->date_function(), time),
7255       &result);
7256   RETURN_ON_FAILED_EXECUTION(Value);
7257   RETURN_ESCAPED(result);
7258 }
7259 
ValueOf() const7260 double v8::Date::ValueOf() const {
7261   i::Handle<i::Object> obj = Utils::OpenHandle(this);
7262   i::Handle<i::JSDate> jsdate = i::Handle<i::JSDate>::cast(obj);
7263   API_RCS_SCOPE(jsdate->GetIsolate(), Date, NumberValue);
7264   return jsdate->value().Number();
7265 }
7266 
7267 // Assert that the static TimeZoneDetection cast in
7268 // DateTimeConfigurationChangeNotification is valid.
7269 #define TIME_ZONE_DETECTION_ASSERT_EQ(value)                     \
7270   STATIC_ASSERT(                                                 \
7271       static_cast<int>(v8::Isolate::TimeZoneDetection::value) == \
7272       static_cast<int>(base::TimezoneCache::TimeZoneDetection::value));
7273 TIME_ZONE_DETECTION_ASSERT_EQ(kSkip)
TIME_ZONE_DETECTION_ASSERT_EQ(kRedetect)7274 TIME_ZONE_DETECTION_ASSERT_EQ(kRedetect)
7275 #undef TIME_ZONE_DETECTION_ASSERT_EQ
7276 
7277 MaybeLocal<v8::RegExp> v8::RegExp::New(Local<Context> context,
7278                                        Local<String> pattern, Flags flags) {
7279   PREPARE_FOR_EXECUTION(context, RegExp, New, RegExp);
7280   Local<v8::RegExp> result;
7281   has_pending_exception =
7282       !ToLocal<RegExp>(i::JSRegExp::New(isolate, Utils::OpenHandle(*pattern),
7283                                         static_cast<i::JSRegExp::Flags>(flags)),
7284                        &result);
7285   RETURN_ON_FAILED_EXECUTION(RegExp);
7286   RETURN_ESCAPED(result);
7287 }
7288 
NewWithBacktrackLimit(Local<Context> context,Local<String> pattern,Flags flags,uint32_t backtrack_limit)7289 MaybeLocal<v8::RegExp> v8::RegExp::NewWithBacktrackLimit(
7290     Local<Context> context, Local<String> pattern, Flags flags,
7291     uint32_t backtrack_limit) {
7292   Utils::ApiCheck(i::Smi::IsValid(backtrack_limit),
7293                   "v8::RegExp::NewWithBacktrackLimit",
7294                   "backtrack_limit is too large or too small.");
7295   Utils::ApiCheck(backtrack_limit != i::JSRegExp::kNoBacktrackLimit,
7296                   "v8::RegExp::NewWithBacktrackLimit",
7297                   "Must set backtrack_limit");
7298   PREPARE_FOR_EXECUTION(context, RegExp, New, RegExp);
7299   Local<v8::RegExp> result;
7300   has_pending_exception = !ToLocal<RegExp>(
7301       i::JSRegExp::New(isolate, Utils::OpenHandle(*pattern),
7302                        static_cast<i::JSRegExp::Flags>(flags), backtrack_limit),
7303       &result);
7304   RETURN_ON_FAILED_EXECUTION(RegExp);
7305   RETURN_ESCAPED(result);
7306 }
7307 
GetSource() const7308 Local<v8::String> v8::RegExp::GetSource() const {
7309   i::Handle<i::JSRegExp> obj = Utils::OpenHandle(this);
7310   return Utils::ToLocal(
7311       i::Handle<i::String>(obj->EscapedPattern(), obj->GetIsolate()));
7312 }
7313 
7314 // Assert that the static flags cast in GetFlags is valid.
7315 #define REGEXP_FLAG_ASSERT_EQ(flag)                   \
7316   STATIC_ASSERT(static_cast<int>(v8::RegExp::flag) == \
7317                 static_cast<int>(i::JSRegExp::flag))
7318 REGEXP_FLAG_ASSERT_EQ(kNone);
7319 REGEXP_FLAG_ASSERT_EQ(kGlobal);
7320 REGEXP_FLAG_ASSERT_EQ(kIgnoreCase);
7321 REGEXP_FLAG_ASSERT_EQ(kMultiline);
7322 REGEXP_FLAG_ASSERT_EQ(kSticky);
7323 REGEXP_FLAG_ASSERT_EQ(kUnicode);
7324 REGEXP_FLAG_ASSERT_EQ(kHasIndices);
7325 REGEXP_FLAG_ASSERT_EQ(kLinear);
7326 #undef REGEXP_FLAG_ASSERT_EQ
7327 
GetFlags() const7328 v8::RegExp::Flags v8::RegExp::GetFlags() const {
7329   i::Handle<i::JSRegExp> obj = Utils::OpenHandle(this);
7330   return RegExp::Flags(static_cast<int>(obj->flags()));
7331 }
7332 
Exec(Local<Context> context,Local<v8::String> subject)7333 MaybeLocal<v8::Object> v8::RegExp::Exec(Local<Context> context,
7334                                         Local<v8::String> subject) {
7335   PREPARE_FOR_EXECUTION(context, RegExp, Exec, Object);
7336 
7337   i::Handle<i::JSRegExp> regexp = Utils::OpenHandle(this);
7338   i::Handle<i::String> subject_string = Utils::OpenHandle(*subject);
7339 
7340   // TODO(jgruber): RegExpUtils::RegExpExec was not written with efficiency in
7341   // mind. It fetches the 'exec' property and then calls it through JSEntry.
7342   // Unfortunately, this is currently the only full implementation of
7343   // RegExp.prototype.exec available in C++.
7344   Local<v8::Object> result;
7345   has_pending_exception = !ToLocal<Object>(
7346       i::RegExpUtils::RegExpExec(isolate, regexp, subject_string,
7347                                  isolate->factory()->undefined_value()),
7348       &result);
7349 
7350   RETURN_ON_FAILED_EXECUTION(Object);
7351   RETURN_ESCAPED(result);
7352 }
7353 
New(Isolate * isolate,int length)7354 Local<v8::Array> v8::Array::New(Isolate* isolate, int length) {
7355   i::Isolate* i_isolate = reinterpret_cast<i::Isolate*>(isolate);
7356   API_RCS_SCOPE(i_isolate, Array, New);
7357   ENTER_V8_NO_SCRIPT_NO_EXCEPTION(i_isolate);
7358   int real_length = length > 0 ? length : 0;
7359   i::Handle<i::JSArray> obj = i_isolate->factory()->NewJSArray(real_length);
7360   i::Handle<i::Object> length_obj =
7361       i_isolate->factory()->NewNumberFromInt(real_length);
7362   obj->set_length(*length_obj);
7363   return Utils::ToLocal(obj);
7364 }
7365 
New(Isolate * isolate,Local<Value> * elements,size_t length)7366 Local<v8::Array> v8::Array::New(Isolate* isolate, Local<Value>* elements,
7367                                 size_t length) {
7368   i::Isolate* i_isolate = reinterpret_cast<i::Isolate*>(isolate);
7369   i::Factory* factory = i_isolate->factory();
7370   API_RCS_SCOPE(i_isolate, Array, New);
7371   ENTER_V8_NO_SCRIPT_NO_EXCEPTION(i_isolate);
7372   int len = static_cast<int>(length);
7373 
7374   i::Handle<i::FixedArray> result = factory->NewFixedArray(len);
7375   for (int i = 0; i < len; i++) {
7376     i::Handle<i::Object> element = Utils::OpenHandle(*elements[i]);
7377     result->set(i, *element);
7378   }
7379 
7380   return Utils::ToLocal(
7381       factory->NewJSArrayWithElements(result, i::PACKED_ELEMENTS, len));
7382 }
7383 
Length() const7384 uint32_t v8::Array::Length() const {
7385   i::Handle<i::JSArray> obj = Utils::OpenHandle(this);
7386   i::Object length = obj->length();
7387   if (length.IsSmi()) {
7388     return i::Smi::ToInt(length);
7389   } else {
7390     return static_cast<uint32_t>(length.Number());
7391   }
7392 }
7393 
New(Isolate * isolate)7394 Local<v8::Map> v8::Map::New(Isolate* isolate) {
7395   i::Isolate* i_isolate = reinterpret_cast<i::Isolate*>(isolate);
7396   API_RCS_SCOPE(i_isolate, Map, New);
7397   ENTER_V8_NO_SCRIPT_NO_EXCEPTION(i_isolate);
7398   i::Handle<i::JSMap> obj = i_isolate->factory()->NewJSMap();
7399   return Utils::ToLocal(obj);
7400 }
7401 
Size() const7402 size_t v8::Map::Size() const {
7403   i::Handle<i::JSMap> obj = Utils::OpenHandle(this);
7404   return i::OrderedHashMap::cast(obj->table()).NumberOfElements();
7405 }
7406 
Clear()7407 void Map::Clear() {
7408   auto self = Utils::OpenHandle(this);
7409   i::Isolate* isolate = self->GetIsolate();
7410   API_RCS_SCOPE(isolate, Map, Clear);
7411   ENTER_V8_NO_SCRIPT_NO_EXCEPTION(isolate);
7412   i::JSMap::Clear(isolate, self);
7413 }
7414 
Get(Local<Context> context,Local<Value> key)7415 MaybeLocal<Value> Map::Get(Local<Context> context, Local<Value> key) {
7416   PREPARE_FOR_EXECUTION(context, Map, Get, Value);
7417   auto self = Utils::OpenHandle(this);
7418   Local<Value> result;
7419   i::Handle<i::Object> argv[] = {Utils::OpenHandle(*key)};
7420   has_pending_exception =
7421       !ToLocal<Value>(i::Execution::CallBuiltin(isolate, isolate->map_get(),
7422                                                 self, arraysize(argv), argv),
7423                       &result);
7424   RETURN_ON_FAILED_EXECUTION(Value);
7425   RETURN_ESCAPED(result);
7426 }
7427 
Set(Local<Context> context,Local<Value> key,Local<Value> value)7428 MaybeLocal<Map> Map::Set(Local<Context> context, Local<Value> key,
7429                          Local<Value> value) {
7430   PREPARE_FOR_EXECUTION(context, Map, Set, Map);
7431   auto self = Utils::OpenHandle(this);
7432   i::Handle<i::Object> result;
7433   i::Handle<i::Object> argv[] = {Utils::OpenHandle(*key),
7434                                  Utils::OpenHandle(*value)};
7435   has_pending_exception =
7436       !i::Execution::CallBuiltin(isolate, isolate->map_set(), self,
7437                                  arraysize(argv), argv)
7438            .ToHandle(&result);
7439   RETURN_ON_FAILED_EXECUTION(Map);
7440   RETURN_ESCAPED(Local<Map>::Cast(Utils::ToLocal(result)));
7441 }
7442 
Has(Local<Context> context,Local<Value> key)7443 Maybe<bool> Map::Has(Local<Context> context, Local<Value> key) {
7444   auto isolate = reinterpret_cast<i::Isolate*>(context->GetIsolate());
7445   ENTER_V8(isolate, context, Map, Has, Nothing<bool>(), i::HandleScope);
7446   auto self = Utils::OpenHandle(this);
7447   i::Handle<i::Object> result;
7448   i::Handle<i::Object> argv[] = {Utils::OpenHandle(*key)};
7449   has_pending_exception =
7450       !i::Execution::CallBuiltin(isolate, isolate->map_has(), self,
7451                                  arraysize(argv), argv)
7452            .ToHandle(&result);
7453   RETURN_ON_FAILED_EXECUTION_PRIMITIVE(bool);
7454   return Just(result->IsTrue(isolate));
7455 }
7456 
Delete(Local<Context> context,Local<Value> key)7457 Maybe<bool> Map::Delete(Local<Context> context, Local<Value> key) {
7458   auto isolate = reinterpret_cast<i::Isolate*>(context->GetIsolate());
7459   ENTER_V8(isolate, context, Map, Delete, Nothing<bool>(), i::HandleScope);
7460   auto self = Utils::OpenHandle(this);
7461   i::Handle<i::Object> result;
7462   i::Handle<i::Object> argv[] = {Utils::OpenHandle(*key)};
7463   has_pending_exception =
7464       !i::Execution::CallBuiltin(isolate, isolate->map_delete(), self,
7465                                  arraysize(argv), argv)
7466            .ToHandle(&result);
7467   RETURN_ON_FAILED_EXECUTION_PRIMITIVE(bool);
7468   return Just(result->IsTrue(isolate));
7469 }
7470 
7471 namespace {
7472 
7473 enum class MapAsArrayKind {
7474   kEntries = i::JS_MAP_KEY_VALUE_ITERATOR_TYPE,
7475   kKeys = i::JS_MAP_KEY_ITERATOR_TYPE,
7476   kValues = i::JS_MAP_VALUE_ITERATOR_TYPE
7477 };
7478 
7479 enum class SetAsArrayKind {
7480   kEntries = i::JS_SET_KEY_VALUE_ITERATOR_TYPE,
7481   kValues = i::JS_SET_VALUE_ITERATOR_TYPE
7482 };
7483 
MapAsArray(i::Isolate * isolate,i::Object table_obj,int offset,MapAsArrayKind kind)7484 i::Handle<i::JSArray> MapAsArray(i::Isolate* isolate, i::Object table_obj,
7485                                  int offset, MapAsArrayKind kind) {
7486   i::Factory* factory = isolate->factory();
7487   i::Handle<i::OrderedHashMap> table(i::OrderedHashMap::cast(table_obj),
7488                                      isolate);
7489   const bool collect_keys =
7490       kind == MapAsArrayKind::kEntries || kind == MapAsArrayKind::kKeys;
7491   const bool collect_values =
7492       kind == MapAsArrayKind::kEntries || kind == MapAsArrayKind::kValues;
7493   int capacity = table->UsedCapacity();
7494   int max_length =
7495       (capacity - offset) * ((collect_keys && collect_values) ? 2 : 1);
7496   i::Handle<i::FixedArray> result = factory->NewFixedArray(max_length);
7497   int result_index = 0;
7498   {
7499     i::DisallowGarbageCollection no_gc;
7500     i::Oddball the_hole = i::ReadOnlyRoots(isolate).the_hole_value();
7501     for (int i = offset; i < capacity; ++i) {
7502       i::InternalIndex entry(i);
7503       i::Object key = table->KeyAt(entry);
7504       if (key == the_hole) continue;
7505       if (collect_keys) result->set(result_index++, key);
7506       if (collect_values) result->set(result_index++, table->ValueAt(entry));
7507     }
7508   }
7509   DCHECK_GE(max_length, result_index);
7510   if (result_index == 0) return factory->NewJSArray(0);
7511   result->Shrink(isolate, result_index);
7512   return factory->NewJSArrayWithElements(result, i::PACKED_ELEMENTS,
7513                                          result_index);
7514 }
7515 
7516 }  // namespace
7517 
AsArray() const7518 Local<Array> Map::AsArray() const {
7519   i::Handle<i::JSMap> obj = Utils::OpenHandle(this);
7520   i::Isolate* isolate = obj->GetIsolate();
7521   API_RCS_SCOPE(isolate, Map, AsArray);
7522   ENTER_V8_NO_SCRIPT_NO_EXCEPTION(isolate);
7523   return Utils::ToLocal(
7524       MapAsArray(isolate, obj->table(), 0, MapAsArrayKind::kEntries));
7525 }
7526 
New(Isolate * isolate)7527 Local<v8::Set> v8::Set::New(Isolate* isolate) {
7528   i::Isolate* i_isolate = reinterpret_cast<i::Isolate*>(isolate);
7529   API_RCS_SCOPE(i_isolate, Set, New);
7530   ENTER_V8_NO_SCRIPT_NO_EXCEPTION(i_isolate);
7531   i::Handle<i::JSSet> obj = i_isolate->factory()->NewJSSet();
7532   return Utils::ToLocal(obj);
7533 }
7534 
Size() const7535 size_t v8::Set::Size() const {
7536   i::Handle<i::JSSet> obj = Utils::OpenHandle(this);
7537   return i::OrderedHashSet::cast(obj->table()).NumberOfElements();
7538 }
7539 
Clear()7540 void Set::Clear() {
7541   auto self = Utils::OpenHandle(this);
7542   i::Isolate* isolate = self->GetIsolate();
7543   API_RCS_SCOPE(isolate, Set, Clear);
7544   ENTER_V8_NO_SCRIPT_NO_EXCEPTION(isolate);
7545   i::JSSet::Clear(isolate, self);
7546 }
7547 
Add(Local<Context> context,Local<Value> key)7548 MaybeLocal<Set> Set::Add(Local<Context> context, Local<Value> key) {
7549   PREPARE_FOR_EXECUTION(context, Set, Add, Set);
7550   auto self = Utils::OpenHandle(this);
7551   i::Handle<i::Object> result;
7552   i::Handle<i::Object> argv[] = {Utils::OpenHandle(*key)};
7553   has_pending_exception =
7554       !i::Execution::CallBuiltin(isolate, isolate->set_add(), self,
7555                                  arraysize(argv), argv)
7556            .ToHandle(&result);
7557   RETURN_ON_FAILED_EXECUTION(Set);
7558   RETURN_ESCAPED(Local<Set>::Cast(Utils::ToLocal(result)));
7559 }
7560 
Has(Local<Context> context,Local<Value> key)7561 Maybe<bool> Set::Has(Local<Context> context, Local<Value> key) {
7562   auto isolate = reinterpret_cast<i::Isolate*>(context->GetIsolate());
7563   ENTER_V8(isolate, context, Set, Has, Nothing<bool>(), i::HandleScope);
7564   auto self = Utils::OpenHandle(this);
7565   i::Handle<i::Object> result;
7566   i::Handle<i::Object> argv[] = {Utils::OpenHandle(*key)};
7567   has_pending_exception =
7568       !i::Execution::CallBuiltin(isolate, isolate->set_has(), self,
7569                                  arraysize(argv), argv)
7570            .ToHandle(&result);
7571   RETURN_ON_FAILED_EXECUTION_PRIMITIVE(bool);
7572   return Just(result->IsTrue(isolate));
7573 }
7574 
Delete(Local<Context> context,Local<Value> key)7575 Maybe<bool> Set::Delete(Local<Context> context, Local<Value> key) {
7576   auto isolate = reinterpret_cast<i::Isolate*>(context->GetIsolate());
7577   ENTER_V8(isolate, context, Set, Delete, Nothing<bool>(), i::HandleScope);
7578   auto self = Utils::OpenHandle(this);
7579   i::Handle<i::Object> result;
7580   i::Handle<i::Object> argv[] = {Utils::OpenHandle(*key)};
7581   has_pending_exception =
7582       !i::Execution::CallBuiltin(isolate, isolate->set_delete(), self,
7583                                  arraysize(argv), argv)
7584            .ToHandle(&result);
7585   RETURN_ON_FAILED_EXECUTION_PRIMITIVE(bool);
7586   return Just(result->IsTrue(isolate));
7587 }
7588 
7589 namespace {
SetAsArray(i::Isolate * isolate,i::Object table_obj,int offset,SetAsArrayKind kind)7590 i::Handle<i::JSArray> SetAsArray(i::Isolate* isolate, i::Object table_obj,
7591                                  int offset, SetAsArrayKind kind) {
7592   i::Factory* factory = isolate->factory();
7593   i::Handle<i::OrderedHashSet> table(i::OrderedHashSet::cast(table_obj),
7594                                      isolate);
7595   // Elements skipped by |offset| may already be deleted.
7596   int capacity = table->UsedCapacity();
7597   const bool collect_key_values = kind == SetAsArrayKind::kEntries;
7598   int max_length = (capacity - offset) * (collect_key_values ? 2 : 1);
7599   if (max_length == 0) return factory->NewJSArray(0);
7600   i::Handle<i::FixedArray> result = factory->NewFixedArray(max_length);
7601   int result_index = 0;
7602   {
7603     i::DisallowGarbageCollection no_gc;
7604     i::Oddball the_hole = i::ReadOnlyRoots(isolate).the_hole_value();
7605     for (int i = offset; i < capacity; ++i) {
7606       i::InternalIndex entry(i);
7607       i::Object key = table->KeyAt(entry);
7608       if (key == the_hole) continue;
7609       result->set(result_index++, key);
7610       if (collect_key_values) result->set(result_index++, key);
7611     }
7612   }
7613   DCHECK_GE(max_length, result_index);
7614   if (result_index == 0) return factory->NewJSArray(0);
7615   result->Shrink(isolate, result_index);
7616   return factory->NewJSArrayWithElements(result, i::PACKED_ELEMENTS,
7617                                          result_index);
7618 }
7619 }  // namespace
7620 
AsArray() const7621 Local<Array> Set::AsArray() const {
7622   i::Handle<i::JSSet> obj = Utils::OpenHandle(this);
7623   i::Isolate* isolate = obj->GetIsolate();
7624   API_RCS_SCOPE(isolate, Set, AsArray);
7625   ENTER_V8_NO_SCRIPT_NO_EXCEPTION(isolate);
7626   return Utils::ToLocal(
7627       SetAsArray(isolate, obj->table(), 0, SetAsArrayKind::kValues));
7628 }
7629 
New(Local<Context> context)7630 MaybeLocal<Promise::Resolver> Promise::Resolver::New(Local<Context> context) {
7631   PREPARE_FOR_EXECUTION(context, Promise_Resolver, New, Resolver);
7632   Local<Promise::Resolver> result;
7633   has_pending_exception =
7634       !ToLocal<Promise::Resolver>(isolate->factory()->NewJSPromise(), &result);
7635   RETURN_ON_FAILED_EXECUTION(Promise::Resolver);
7636   RETURN_ESCAPED(result);
7637 }
7638 
GetPromise()7639 Local<Promise> Promise::Resolver::GetPromise() {
7640   i::Handle<i::JSReceiver> promise = Utils::OpenHandle(this);
7641   return Local<Promise>::Cast(Utils::ToLocal(promise));
7642 }
7643 
Resolve(Local<Context> context,Local<Value> value)7644 Maybe<bool> Promise::Resolver::Resolve(Local<Context> context,
7645                                        Local<Value> value) {
7646   auto isolate = reinterpret_cast<i::Isolate*>(context->GetIsolate());
7647   ENTER_V8(isolate, context, Promise_Resolver, Resolve, Nothing<bool>(),
7648            i::HandleScope);
7649   auto self = Utils::OpenHandle(this);
7650   auto promise = i::Handle<i::JSPromise>::cast(self);
7651 
7652   if (promise->status() != Promise::kPending) {
7653     return Just(true);
7654   }
7655 
7656   has_pending_exception =
7657       i::JSPromise::Resolve(promise, Utils::OpenHandle(*value)).is_null();
7658   RETURN_ON_FAILED_EXECUTION_PRIMITIVE(bool);
7659   return Just(true);
7660 }
7661 
Reject(Local<Context> context,Local<Value> value)7662 Maybe<bool> Promise::Resolver::Reject(Local<Context> context,
7663                                       Local<Value> value) {
7664   auto isolate = reinterpret_cast<i::Isolate*>(context->GetIsolate());
7665   ENTER_V8(isolate, context, Promise_Resolver, Reject, Nothing<bool>(),
7666            i::HandleScope);
7667   auto self = Utils::OpenHandle(this);
7668   auto promise = i::Handle<i::JSPromise>::cast(self);
7669 
7670   if (promise->status() != Promise::kPending) {
7671     return Just(true);
7672   }
7673 
7674   has_pending_exception =
7675       i::JSPromise::Reject(promise, Utils::OpenHandle(*value)).is_null();
7676   RETURN_ON_FAILED_EXECUTION_PRIMITIVE(bool);
7677   return Just(true);
7678 }
7679 
Catch(Local<Context> context,Local<Function> handler)7680 MaybeLocal<Promise> Promise::Catch(Local<Context> context,
7681                                    Local<Function> handler) {
7682   PREPARE_FOR_EXECUTION(context, Promise, Catch, Promise);
7683   auto self = Utils::OpenHandle(this);
7684   i::Handle<i::Object> argv[] = {isolate->factory()->undefined_value(),
7685                                  Utils::OpenHandle(*handler)};
7686   i::Handle<i::Object> result;
7687   // Do not call the built-in Promise.prototype.catch!
7688   // v8::Promise should not call out to a monkeypatched Promise.prototype.then
7689   // as the implementation of Promise.prototype.catch does.
7690   has_pending_exception =
7691       !i::Execution::CallBuiltin(isolate, isolate->promise_then(), self,
7692                                  arraysize(argv), argv)
7693            .ToHandle(&result);
7694   RETURN_ON_FAILED_EXECUTION(Promise);
7695   RETURN_ESCAPED(Local<Promise>::Cast(Utils::ToLocal(result)));
7696 }
7697 
Then(Local<Context> context,Local<Function> handler)7698 MaybeLocal<Promise> Promise::Then(Local<Context> context,
7699                                   Local<Function> handler) {
7700   PREPARE_FOR_EXECUTION(context, Promise, Then, Promise);
7701   auto self = Utils::OpenHandle(this);
7702   i::Handle<i::Object> argv[] = {Utils::OpenHandle(*handler)};
7703   i::Handle<i::Object> result;
7704   has_pending_exception =
7705       !i::Execution::CallBuiltin(isolate, isolate->promise_then(), self,
7706                                  arraysize(argv), argv)
7707            .ToHandle(&result);
7708   RETURN_ON_FAILED_EXECUTION(Promise);
7709   RETURN_ESCAPED(Local<Promise>::Cast(Utils::ToLocal(result)));
7710 }
7711 
Then(Local<Context> context,Local<Function> on_fulfilled,Local<Function> on_rejected)7712 MaybeLocal<Promise> Promise::Then(Local<Context> context,
7713                                   Local<Function> on_fulfilled,
7714                                   Local<Function> on_rejected) {
7715   PREPARE_FOR_EXECUTION(context, Promise, Then, Promise);
7716   auto self = Utils::OpenHandle(this);
7717   i::Handle<i::Object> argv[] = {Utils::OpenHandle(*on_fulfilled),
7718                                  Utils::OpenHandle(*on_rejected)};
7719   i::Handle<i::Object> result;
7720   has_pending_exception =
7721       !i::Execution::CallBuiltin(isolate, isolate->promise_then(), self,
7722                                  arraysize(argv), argv)
7723            .ToHandle(&result);
7724   RETURN_ON_FAILED_EXECUTION(Promise);
7725   RETURN_ESCAPED(Local<Promise>::Cast(Utils::ToLocal(result)));
7726 }
7727 
HasHandler() const7728 bool Promise::HasHandler() const {
7729   i::JSReceiver promise = *Utils::OpenHandle(this);
7730   i::Isolate* isolate = promise.GetIsolate();
7731   API_RCS_SCOPE(isolate, Promise, HasRejectHandler);
7732   ENTER_V8_NO_SCRIPT_NO_EXCEPTION(isolate);
7733   if (!promise.IsJSPromise()) return false;
7734   return i::JSPromise::cast(promise).has_handler();
7735 }
7736 
Result()7737 Local<Value> Promise::Result() {
7738   i::Handle<i::JSReceiver> promise = Utils::OpenHandle(this);
7739   i::Isolate* isolate = promise->GetIsolate();
7740   API_RCS_SCOPE(isolate, Promise, Result);
7741   i::Handle<i::JSPromise> js_promise = i::Handle<i::JSPromise>::cast(promise);
7742   Utils::ApiCheck(js_promise->status() != kPending, "v8_Promise_Result",
7743                   "Promise is still pending");
7744   i::Handle<i::Object> result(js_promise->result(), isolate);
7745   return Utils::ToLocal(result);
7746 }
7747 
State()7748 Promise::PromiseState Promise::State() {
7749   i::Handle<i::JSReceiver> promise = Utils::OpenHandle(this);
7750   API_RCS_SCOPE(promise->GetIsolate(), Promise, Status);
7751   i::Handle<i::JSPromise> js_promise = i::Handle<i::JSPromise>::cast(promise);
7752   return static_cast<PromiseState>(js_promise->status());
7753 }
7754 
MarkAsHandled()7755 void Promise::MarkAsHandled() {
7756   i::Handle<i::JSPromise> js_promise = Utils::OpenHandle(this);
7757   js_promise->set_has_handler(true);
7758 }
7759 
MarkAsSilent()7760 void Promise::MarkAsSilent() {
7761   i::Handle<i::JSPromise> js_promise = Utils::OpenHandle(this);
7762   js_promise->set_is_silent(true);
7763 }
7764 
GetTarget()7765 Local<Value> Proxy::GetTarget() {
7766   i::Handle<i::JSProxy> self = Utils::OpenHandle(this);
7767   i::Handle<i::Object> target(self->target(), self->GetIsolate());
7768   return Utils::ToLocal(target);
7769 }
7770 
GetHandler()7771 Local<Value> Proxy::GetHandler() {
7772   i::Handle<i::JSProxy> self = Utils::OpenHandle(this);
7773   i::Handle<i::Object> handler(self->handler(), self->GetIsolate());
7774   return Utils::ToLocal(handler);
7775 }
7776 
IsRevoked() const7777 bool Proxy::IsRevoked() const {
7778   i::Handle<i::JSProxy> self = Utils::OpenHandle(this);
7779   return self->IsRevoked();
7780 }
7781 
Revoke()7782 void Proxy::Revoke() {
7783   i::Handle<i::JSProxy> self = Utils::OpenHandle(this);
7784   i::JSProxy::Revoke(self);
7785 }
7786 
New(Local<Context> context,Local<Object> local_target,Local<Object> local_handler)7787 MaybeLocal<Proxy> Proxy::New(Local<Context> context, Local<Object> local_target,
7788                              Local<Object> local_handler) {
7789   PREPARE_FOR_EXECUTION(context, Proxy, New, Proxy);
7790   i::Handle<i::JSReceiver> target = Utils::OpenHandle(*local_target);
7791   i::Handle<i::JSReceiver> handler = Utils::OpenHandle(*local_handler);
7792   Local<Proxy> result;
7793   has_pending_exception =
7794       !ToLocal<Proxy>(i::JSProxy::New(isolate, target, handler), &result);
7795   RETURN_ON_FAILED_EXECUTION(Proxy);
7796   RETURN_ESCAPED(result);
7797 }
7798 
CompiledWasmModule(std::shared_ptr<internal::wasm::NativeModule> native_module,const char * source_url,size_t url_length)7799 CompiledWasmModule::CompiledWasmModule(
7800     std::shared_ptr<internal::wasm::NativeModule> native_module,
7801     const char* source_url, size_t url_length)
7802     : native_module_(std::move(native_module)),
7803       source_url_(source_url, url_length) {
7804   CHECK_NOT_NULL(native_module_);
7805 }
7806 
Serialize()7807 OwnedBuffer CompiledWasmModule::Serialize() {
7808 #if V8_ENABLE_WEBASSEMBLY
7809   TRACE_EVENT0("v8.wasm", "wasm.SerializeModule");
7810   i::wasm::WasmSerializer wasm_serializer(native_module_.get());
7811   size_t buffer_size = wasm_serializer.GetSerializedNativeModuleSize();
7812   std::unique_ptr<uint8_t[]> buffer(new uint8_t[buffer_size]);
7813   if (!wasm_serializer.SerializeNativeModule({buffer.get(), buffer_size}))
7814     return {};
7815   return {std::move(buffer), buffer_size};
7816 #else
7817   UNREACHABLE();
7818 #endif  // V8_ENABLE_WEBASSEMBLY
7819 }
7820 
GetWireBytesRef()7821 MemorySpan<const uint8_t> CompiledWasmModule::GetWireBytesRef() {
7822 #if V8_ENABLE_WEBASSEMBLY
7823   base::Vector<const uint8_t> bytes_vec = native_module_->wire_bytes();
7824   return {bytes_vec.begin(), bytes_vec.size()};
7825 #else
7826   UNREACHABLE();
7827 #endif  // V8_ENABLE_WEBASSEMBLY
7828 }
7829 
Buffer()7830 Local<ArrayBuffer> v8::WasmMemoryObject::Buffer() {
7831 #if V8_ENABLE_WEBASSEMBLY
7832   i::Handle<i::WasmMemoryObject> obj = Utils::OpenHandle(this);
7833   i::Handle<i::JSArrayBuffer> buffer(obj->array_buffer(), obj->GetIsolate());
7834   return Utils::ToLocal(buffer);
7835 #else
7836   UNREACHABLE();
7837 #endif  // V8_ENABLE_WEBASSEMBLY
7838 }
7839 
GetCompiledModule()7840 CompiledWasmModule WasmModuleObject::GetCompiledModule() {
7841 #if V8_ENABLE_WEBASSEMBLY
7842   auto obj = i::Handle<i::WasmModuleObject>::cast(Utils::OpenHandle(this));
7843   auto url =
7844       i::handle(i::String::cast(obj->script().name()), obj->GetIsolate());
7845   int length;
7846   std::unique_ptr<char[]> cstring =
7847       url->ToCString(i::DISALLOW_NULLS, i::FAST_STRING_TRAVERSAL, &length);
7848   return CompiledWasmModule(std::move(obj->shared_native_module()),
7849                             cstring.get(), length);
7850 #else
7851   UNREACHABLE();
7852 #endif  // V8_ENABLE_WEBASSEMBLY
7853 }
7854 
FromCompiledModule(Isolate * isolate,const CompiledWasmModule & compiled_module)7855 MaybeLocal<WasmModuleObject> WasmModuleObject::FromCompiledModule(
7856     Isolate* isolate, const CompiledWasmModule& compiled_module) {
7857 #if V8_ENABLE_WEBASSEMBLY
7858   i::Isolate* i_isolate = reinterpret_cast<i::Isolate*>(isolate);
7859   i::Handle<i::WasmModuleObject> module_object =
7860       i::wasm::GetWasmEngine()->ImportNativeModule(
7861           i_isolate, compiled_module.native_module_,
7862           base::VectorOf(compiled_module.source_url()));
7863   return Local<WasmModuleObject>::Cast(
7864       Utils::ToLocal(i::Handle<i::JSObject>::cast(module_object)));
7865 #else
7866   UNREACHABLE();
7867 #endif  // V8_ENABLE_WEBASSEMBLY
7868 }
7869 
Compile(Isolate * isolate,MemorySpan<const uint8_t> wire_bytes)7870 MaybeLocal<WasmModuleObject> WasmModuleObject::Compile(
7871     Isolate* isolate, MemorySpan<const uint8_t> wire_bytes) {
7872 #if V8_ENABLE_WEBASSEMBLY
7873   const uint8_t* start = wire_bytes.data();
7874   size_t length = wire_bytes.size();
7875   i::Isolate* i_isolate = reinterpret_cast<i::Isolate*>(isolate);
7876   if (!i::wasm::IsWasmCodegenAllowed(i_isolate, i_isolate->native_context())) {
7877     return MaybeLocal<WasmModuleObject>();
7878   }
7879   i::MaybeHandle<i::JSObject> maybe_compiled;
7880   {
7881     i::wasm::ErrorThrower thrower(i_isolate, "WasmModuleObject::Compile()");
7882     auto enabled_features = i::wasm::WasmFeatures::FromIsolate(i_isolate);
7883     maybe_compiled = i::wasm::GetWasmEngine()->SyncCompile(
7884         i_isolate, enabled_features, &thrower,
7885         i::wasm::ModuleWireBytes(start, start + length));
7886   }
7887   CHECK_EQ(maybe_compiled.is_null(), i_isolate->has_pending_exception());
7888   if (maybe_compiled.is_null()) {
7889     i_isolate->OptionalRescheduleException(false);
7890     return MaybeLocal<WasmModuleObject>();
7891   }
7892   return Local<WasmModuleObject>::Cast(
7893       Utils::ToLocal(maybe_compiled.ToHandleChecked()));
7894 #else
7895   Utils::ApiCheck(false, "WasmModuleObject::Compile",
7896                   "WebAssembly support is not enabled.");
7897   UNREACHABLE();
7898 #endif  // V8_ENABLE_WEBASSEMBLY
7899 }
7900 
WasmModuleObjectBuilderStreaming(Isolate * isolate)7901 WasmModuleObjectBuilderStreaming::WasmModuleObjectBuilderStreaming(
7902     Isolate* isolate) {
7903   USE(isolate_);
7904 }
7905 
GetPromise()7906 Local<Promise> WasmModuleObjectBuilderStreaming::GetPromise() { return {}; }
7907 
OnBytesReceived(const uint8_t * bytes,size_t size)7908 void WasmModuleObjectBuilderStreaming::OnBytesReceived(const uint8_t* bytes,
7909                                                        size_t size) {}
7910 
Finish()7911 void WasmModuleObjectBuilderStreaming::Finish() {}
7912 
Abort(MaybeLocal<Value> exception)7913 void WasmModuleObjectBuilderStreaming::Abort(MaybeLocal<Value> exception) {}
7914 
Reallocate(void * data,size_t old_length,size_t new_length)7915 void* v8::ArrayBuffer::Allocator::Reallocate(void* data, size_t old_length,
7916                                              size_t new_length) {
7917   if (old_length == new_length) return data;
7918   uint8_t* new_data =
7919       reinterpret_cast<uint8_t*>(AllocateUninitialized(new_length));
7920   if (new_data == nullptr) return nullptr;
7921   size_t bytes_to_copy = std::min(old_length, new_length);
7922   memcpy(new_data, data, bytes_to_copy);
7923   if (new_length > bytes_to_copy) {
7924     memset(new_data + bytes_to_copy, 0, new_length - bytes_to_copy);
7925   }
7926   Free(data, old_length);
7927   return new_data;
7928 }
7929 
7930 // static
NewDefaultAllocator()7931 v8::ArrayBuffer::Allocator* v8::ArrayBuffer::Allocator::NewDefaultAllocator() {
7932   return new ArrayBufferAllocator();
7933 }
7934 
IsDetachable() const7935 bool v8::ArrayBuffer::IsDetachable() const {
7936   return Utils::OpenHandle(this)->is_detachable();
7937 }
7938 
WasDetached() const7939 bool v8::ArrayBuffer::WasDetached() const {
7940   return Utils::OpenHandle(this)->was_detached();
7941 }
7942 
7943 namespace {
ToInternal(std::shared_ptr<i::BackingStoreBase> backing_store)7944 std::shared_ptr<i::BackingStore> ToInternal(
7945     std::shared_ptr<i::BackingStoreBase> backing_store) {
7946   return std::static_pointer_cast<i::BackingStore>(backing_store);
7947 }
7948 }  // namespace
7949 
Detach()7950 void v8::ArrayBuffer::Detach() {
7951   i::Handle<i::JSArrayBuffer> obj = Utils::OpenHandle(this);
7952   i::Isolate* isolate = obj->GetIsolate();
7953   Utils::ApiCheck(obj->is_detachable(), "v8::ArrayBuffer::Detach",
7954                   "Only detachable ArrayBuffers can be detached");
7955   API_RCS_SCOPE(isolate, ArrayBuffer, Detach);
7956   ENTER_V8_NO_SCRIPT_NO_EXCEPTION(isolate);
7957   obj->Detach();
7958 }
7959 
ByteLength() const7960 size_t v8::ArrayBuffer::ByteLength() const {
7961   i::Handle<i::JSArrayBuffer> obj = Utils::OpenHandle(this);
7962   return obj->byte_length();
7963 }
7964 
New(Isolate * isolate,size_t byte_length)7965 Local<ArrayBuffer> v8::ArrayBuffer::New(Isolate* isolate, size_t byte_length) {
7966   i::Isolate* i_isolate = reinterpret_cast<i::Isolate*>(isolate);
7967   API_RCS_SCOPE(i_isolate, ArrayBuffer, New);
7968   ENTER_V8_NO_SCRIPT_NO_EXCEPTION(i_isolate);
7969   i::MaybeHandle<i::JSArrayBuffer> result =
7970       i_isolate->factory()->NewJSArrayBufferAndBackingStore(
7971           byte_length, i::InitializedFlag::kZeroInitialized);
7972 
7973   i::Handle<i::JSArrayBuffer> array_buffer;
7974   if (!result.ToHandle(&array_buffer)) {
7975     // TODO(jbroman): It may be useful in the future to provide a MaybeLocal
7976     // version that throws an exception or otherwise does not crash.
7977     i::FatalProcessOutOfMemory(i_isolate, "v8::ArrayBuffer::New");
7978   }
7979 
7980   return Utils::ToLocal(array_buffer);
7981 }
7982 
New(Isolate * isolate,std::shared_ptr<BackingStore> backing_store)7983 Local<ArrayBuffer> v8::ArrayBuffer::New(
7984     Isolate* isolate, std::shared_ptr<BackingStore> backing_store) {
7985   CHECK_IMPLIES(backing_store->ByteLength() != 0,
7986                 backing_store->Data() != nullptr);
7987   i::Isolate* i_isolate = reinterpret_cast<i::Isolate*>(isolate);
7988   API_RCS_SCOPE(i_isolate, ArrayBuffer, New);
7989   ENTER_V8_NO_SCRIPT_NO_EXCEPTION(i_isolate);
7990   std::shared_ptr<i::BackingStore> i_backing_store(
7991       ToInternal(std::move(backing_store)));
7992   Utils::ApiCheck(
7993       !i_backing_store->is_shared(), "v8_ArrayBuffer_New",
7994       "Cannot construct ArrayBuffer with a BackingStore of SharedArrayBuffer");
7995   i::Handle<i::JSArrayBuffer> obj =
7996       i_isolate->factory()->NewJSArrayBuffer(std::move(i_backing_store));
7997   return Utils::ToLocal(obj);
7998 }
7999 
NewBackingStore(Isolate * isolate,size_t byte_length)8000 std::unique_ptr<v8::BackingStore> v8::ArrayBuffer::NewBackingStore(
8001     Isolate* isolate, size_t byte_length) {
8002   i::Isolate* i_isolate = reinterpret_cast<i::Isolate*>(isolate);
8003   API_RCS_SCOPE(i_isolate, ArrayBuffer, NewBackingStore);
8004   CHECK_LE(byte_length, i::JSArrayBuffer::kMaxByteLength);
8005   ENTER_V8_NO_SCRIPT_NO_EXCEPTION(i_isolate);
8006   std::unique_ptr<i::BackingStoreBase> backing_store =
8007       i::BackingStore::Allocate(i_isolate, byte_length,
8008                                 i::SharedFlag::kNotShared,
8009                                 i::InitializedFlag::kZeroInitialized);
8010   if (!backing_store) {
8011     i::FatalProcessOutOfMemory(i_isolate, "v8::ArrayBuffer::NewBackingStore");
8012   }
8013   return std::unique_ptr<v8::BackingStore>(
8014       static_cast<v8::BackingStore*>(backing_store.release()));
8015 }
8016 
NewBackingStore(void * data,size_t byte_length,v8::BackingStore::DeleterCallback deleter,void * deleter_data)8017 std::unique_ptr<v8::BackingStore> v8::ArrayBuffer::NewBackingStore(
8018     void* data, size_t byte_length, v8::BackingStore::DeleterCallback deleter,
8019     void* deleter_data) {
8020   CHECK_LE(byte_length, i::JSArrayBuffer::kMaxByteLength);
8021   std::unique_ptr<i::BackingStoreBase> backing_store =
8022       i::BackingStore::WrapAllocation(data, byte_length, deleter, deleter_data,
8023                                       i::SharedFlag::kNotShared);
8024   return std::unique_ptr<v8::BackingStore>(
8025       static_cast<v8::BackingStore*>(backing_store.release()));
8026 }
8027 
Buffer()8028 Local<ArrayBuffer> v8::ArrayBufferView::Buffer() {
8029   i::Handle<i::JSArrayBufferView> obj = Utils::OpenHandle(this);
8030   i::Handle<i::JSArrayBuffer> buffer;
8031   if (obj->IsJSDataView()) {
8032     i::Handle<i::JSDataView> data_view(i::JSDataView::cast(*obj),
8033                                        obj->GetIsolate());
8034     DCHECK(data_view->buffer().IsJSArrayBuffer());
8035     buffer = i::handle(i::JSArrayBuffer::cast(data_view->buffer()),
8036                        data_view->GetIsolate());
8037   } else {
8038     DCHECK(obj->IsJSTypedArray());
8039     buffer = i::JSTypedArray::cast(*obj).GetBuffer();
8040   }
8041   return Utils::ToLocal(buffer);
8042 }
8043 
CopyContents(void * dest,size_t byte_length)8044 size_t v8::ArrayBufferView::CopyContents(void* dest, size_t byte_length) {
8045   i::Handle<i::JSArrayBufferView> self = Utils::OpenHandle(this);
8046   size_t bytes_to_copy = std::min(byte_length, self->byte_length());
8047   if (bytes_to_copy) {
8048     i::DisallowGarbageCollection no_gc;
8049     i::Isolate* isolate = self->GetIsolate();
8050     const char* source;
8051     if (self->IsJSTypedArray()) {
8052       i::Handle<i::JSTypedArray> array(i::JSTypedArray::cast(*self), isolate);
8053       source = reinterpret_cast<char*>(array->DataPtr());
8054     } else {
8055       DCHECK(self->IsJSDataView());
8056       i::Handle<i::JSDataView> data_view(i::JSDataView::cast(*self), isolate);
8057       source = reinterpret_cast<char*>(data_view->data_pointer());
8058     }
8059     memcpy(dest, source, bytes_to_copy);
8060   }
8061   return bytes_to_copy;
8062 }
8063 
HasBuffer() const8064 bool v8::ArrayBufferView::HasBuffer() const {
8065   i::Handle<i::JSArrayBufferView> self = Utils::OpenHandle(this);
8066   if (!self->IsJSTypedArray()) return true;
8067   auto typed_array = i::Handle<i::JSTypedArray>::cast(self);
8068   return !typed_array->is_on_heap();
8069 }
8070 
ByteOffset()8071 size_t v8::ArrayBufferView::ByteOffset() {
8072   i::Handle<i::JSArrayBufferView> obj = Utils::OpenHandle(this);
8073   return obj->WasDetached() ? 0 : obj->byte_offset();
8074 }
8075 
ByteLength()8076 size_t v8::ArrayBufferView::ByteLength() {
8077   i::Handle<i::JSArrayBufferView> obj = Utils::OpenHandle(this);
8078   return obj->WasDetached() ? 0 : obj->byte_length();
8079 }
8080 
Length()8081 size_t v8::TypedArray::Length() {
8082   i::Handle<i::JSTypedArray> obj = Utils::OpenHandle(this);
8083   return obj->WasDetached() ? 0 : obj->length();
8084 }
8085 
8086 static_assert(
8087     v8::TypedArray::kMaxLength == i::JSTypedArray::kMaxLength,
8088     "v8::TypedArray::kMaxLength must match i::JSTypedArray::kMaxLength");
8089 
8090 #define TYPED_ARRAY_NEW(Type, type, TYPE, ctype)                           \
8091   Local<Type##Array> Type##Array::New(Local<ArrayBuffer> array_buffer,     \
8092                                       size_t byte_offset, size_t length) { \
8093     i::Isolate* isolate = Utils::OpenHandle(*array_buffer)->GetIsolate();  \
8094     API_RCS_SCOPE(isolate, Type##Array, New);                              \
8095     ENTER_V8_NO_SCRIPT_NO_EXCEPTION(isolate);                              \
8096     if (!Utils::ApiCheck(length <= kMaxLength,                             \
8097                          "v8::" #Type                                      \
8098                          "Array::New(Local<ArrayBuffer>, size_t, size_t)", \
8099                          "length exceeds max allowed value")) {            \
8100       return Local<Type##Array>();                                         \
8101     }                                                                      \
8102     i::Handle<i::JSArrayBuffer> buffer = Utils::OpenHandle(*array_buffer); \
8103     i::Handle<i::JSTypedArray> obj = isolate->factory()->NewJSTypedArray(  \
8104         i::kExternal##Type##Array, buffer, byte_offset, length);           \
8105     return Utils::ToLocal##Type##Array(obj);                               \
8106   }                                                                        \
8107   Local<Type##Array> Type##Array::New(                                     \
8108       Local<SharedArrayBuffer> shared_array_buffer, size_t byte_offset,    \
8109       size_t length) {                                                     \
8110     CHECK(i::FLAG_harmony_sharedarraybuffer);                              \
8111     i::Isolate* isolate =                                                  \
8112         Utils::OpenHandle(*shared_array_buffer)->GetIsolate();             \
8113     API_RCS_SCOPE(isolate, Type##Array, New);                              \
8114     ENTER_V8_NO_SCRIPT_NO_EXCEPTION(isolate);                              \
8115     if (!Utils::ApiCheck(                                                  \
8116             length <= kMaxLength,                                          \
8117             "v8::" #Type                                                   \
8118             "Array::New(Local<SharedArrayBuffer>, size_t, size_t)",        \
8119             "length exceeds max allowed value")) {                         \
8120       return Local<Type##Array>();                                         \
8121     }                                                                      \
8122     i::Handle<i::JSArrayBuffer> buffer =                                   \
8123         Utils::OpenHandle(*shared_array_buffer);                           \
8124     i::Handle<i::JSTypedArray> obj = isolate->factory()->NewJSTypedArray(  \
8125         i::kExternal##Type##Array, buffer, byte_offset, length);           \
8126     return Utils::ToLocal##Type##Array(obj);                               \
8127   }
8128 
TYPED_ARRAYS(TYPED_ARRAY_NEW)8129 TYPED_ARRAYS(TYPED_ARRAY_NEW)
8130 #undef TYPED_ARRAY_NEW
8131 
8132 Local<DataView> DataView::New(Local<ArrayBuffer> array_buffer,
8133                               size_t byte_offset, size_t byte_length) {
8134   i::Handle<i::JSArrayBuffer> buffer = Utils::OpenHandle(*array_buffer);
8135   i::Isolate* isolate = buffer->GetIsolate();
8136   API_RCS_SCOPE(isolate, DataView, New);
8137   ENTER_V8_NO_SCRIPT_NO_EXCEPTION(isolate);
8138   i::Handle<i::JSDataView> obj =
8139       isolate->factory()->NewJSDataView(buffer, byte_offset, byte_length);
8140   return Utils::ToLocal(obj);
8141 }
8142 
New(Local<SharedArrayBuffer> shared_array_buffer,size_t byte_offset,size_t byte_length)8143 Local<DataView> DataView::New(Local<SharedArrayBuffer> shared_array_buffer,
8144                               size_t byte_offset, size_t byte_length) {
8145   CHECK(i::FLAG_harmony_sharedarraybuffer);
8146   i::Handle<i::JSArrayBuffer> buffer = Utils::OpenHandle(*shared_array_buffer);
8147   i::Isolate* isolate = buffer->GetIsolate();
8148   API_RCS_SCOPE(isolate, DataView, New);
8149   ENTER_V8_NO_SCRIPT_NO_EXCEPTION(isolate);
8150   i::Handle<i::JSDataView> obj =
8151       isolate->factory()->NewJSDataView(buffer, byte_offset, byte_length);
8152   return Utils::ToLocal(obj);
8153 }
8154 
ByteLength() const8155 size_t v8::SharedArrayBuffer::ByteLength() const {
8156   i::Handle<i::JSArrayBuffer> obj = Utils::OpenHandle(this);
8157   return obj->byte_length();
8158 }
8159 
New(Isolate * isolate,size_t byte_length)8160 Local<SharedArrayBuffer> v8::SharedArrayBuffer::New(Isolate* isolate,
8161                                                     size_t byte_length) {
8162   CHECK(i::FLAG_harmony_sharedarraybuffer);
8163   i::Isolate* i_isolate = reinterpret_cast<i::Isolate*>(isolate);
8164   API_RCS_SCOPE(i_isolate, SharedArrayBuffer, New);
8165   ENTER_V8_NO_SCRIPT_NO_EXCEPTION(i_isolate);
8166 
8167   std::unique_ptr<i::BackingStore> backing_store =
8168       i::BackingStore::Allocate(i_isolate, byte_length, i::SharedFlag::kShared,
8169                                 i::InitializedFlag::kZeroInitialized);
8170 
8171   if (!backing_store) {
8172     // TODO(jbroman): It may be useful in the future to provide a MaybeLocal
8173     // version that throws an exception or otherwise does not crash.
8174     i::FatalProcessOutOfMemory(i_isolate, "v8::SharedArrayBuffer::New");
8175   }
8176 
8177   i::Handle<i::JSArrayBuffer> obj =
8178       i_isolate->factory()->NewJSSharedArrayBuffer(std::move(backing_store));
8179   return Utils::ToLocalShared(obj);
8180 }
8181 
New(Isolate * isolate,std::shared_ptr<BackingStore> backing_store)8182 Local<SharedArrayBuffer> v8::SharedArrayBuffer::New(
8183     Isolate* isolate, std::shared_ptr<BackingStore> backing_store) {
8184   CHECK(i::FLAG_harmony_sharedarraybuffer);
8185   CHECK_IMPLIES(backing_store->ByteLength() != 0,
8186                 backing_store->Data() != nullptr);
8187   i::Isolate* i_isolate = reinterpret_cast<i::Isolate*>(isolate);
8188   API_RCS_SCOPE(i_isolate, SharedArrayBuffer, New);
8189   ENTER_V8_NO_SCRIPT_NO_EXCEPTION(i_isolate);
8190   std::shared_ptr<i::BackingStore> i_backing_store(ToInternal(backing_store));
8191   Utils::ApiCheck(
8192       i_backing_store->is_shared(), "v8_SharedArrayBuffer_New",
8193       "Cannot construct SharedArrayBuffer with BackingStore of ArrayBuffer");
8194   i::Handle<i::JSArrayBuffer> obj =
8195       i_isolate->factory()->NewJSSharedArrayBuffer(std::move(i_backing_store));
8196   return Utils::ToLocalShared(obj);
8197 }
8198 
NewBackingStore(Isolate * isolate,size_t byte_length)8199 std::unique_ptr<v8::BackingStore> v8::SharedArrayBuffer::NewBackingStore(
8200     Isolate* isolate, size_t byte_length) {
8201   i::Isolate* i_isolate = reinterpret_cast<i::Isolate*>(isolate);
8202   API_RCS_SCOPE(i_isolate, SharedArrayBuffer, NewBackingStore);
8203   CHECK_LE(byte_length, i::JSArrayBuffer::kMaxByteLength);
8204   ENTER_V8_NO_SCRIPT_NO_EXCEPTION(i_isolate);
8205   std::unique_ptr<i::BackingStoreBase> backing_store =
8206       i::BackingStore::Allocate(i_isolate, byte_length, i::SharedFlag::kShared,
8207                                 i::InitializedFlag::kZeroInitialized);
8208   if (!backing_store) {
8209     i::FatalProcessOutOfMemory(i_isolate,
8210                                "v8::SharedArrayBuffer::NewBackingStore");
8211   }
8212   return std::unique_ptr<v8::BackingStore>(
8213       static_cast<v8::BackingStore*>(backing_store.release()));
8214 }
8215 
NewBackingStore(void * data,size_t byte_length,v8::BackingStore::DeleterCallback deleter,void * deleter_data)8216 std::unique_ptr<v8::BackingStore> v8::SharedArrayBuffer::NewBackingStore(
8217     void* data, size_t byte_length, v8::BackingStore::DeleterCallback deleter,
8218     void* deleter_data) {
8219   CHECK_LE(byte_length, i::JSArrayBuffer::kMaxByteLength);
8220   std::unique_ptr<i::BackingStoreBase> backing_store =
8221       i::BackingStore::WrapAllocation(data, byte_length, deleter, deleter_data,
8222                                       i::SharedFlag::kShared);
8223   return std::unique_ptr<v8::BackingStore>(
8224       static_cast<v8::BackingStore*>(backing_store.release()));
8225 }
8226 
New(Isolate * isolate,Local<String> name)8227 Local<Symbol> v8::Symbol::New(Isolate* isolate, Local<String> name) {
8228   i::Isolate* i_isolate = reinterpret_cast<i::Isolate*>(isolate);
8229   API_RCS_SCOPE(i_isolate, Symbol, New);
8230   ENTER_V8_NO_SCRIPT_NO_EXCEPTION(i_isolate);
8231   i::Handle<i::Symbol> result = i_isolate->factory()->NewSymbol();
8232   if (!name.IsEmpty()) result->set_description(*Utils::OpenHandle(*name));
8233   return Utils::ToLocal(result);
8234 }
8235 
For(Isolate * isolate,Local<String> name)8236 Local<Symbol> v8::Symbol::For(Isolate* isolate, Local<String> name) {
8237   i::Isolate* i_isolate = reinterpret_cast<i::Isolate*>(isolate);
8238   i::Handle<i::String> i_name = Utils::OpenHandle(*name);
8239   return Utils::ToLocal(
8240       i_isolate->SymbolFor(i::RootIndex::kPublicSymbolTable, i_name, false));
8241 }
8242 
ForApi(Isolate * isolate,Local<String> name)8243 Local<Symbol> v8::Symbol::ForApi(Isolate* isolate, Local<String> name) {
8244   i::Isolate* i_isolate = reinterpret_cast<i::Isolate*>(isolate);
8245   i::Handle<i::String> i_name = Utils::OpenHandle(*name);
8246   return Utils::ToLocal(
8247       i_isolate->SymbolFor(i::RootIndex::kApiSymbolTable, i_name, false));
8248 }
8249 
8250 #define WELL_KNOWN_SYMBOLS(V)                 \
8251   V(AsyncIterator, async_iterator)            \
8252   V(HasInstance, has_instance)                \
8253   V(IsConcatSpreadable, is_concat_spreadable) \
8254   V(Iterator, iterator)                       \
8255   V(Match, match)                             \
8256   V(Replace, replace)                         \
8257   V(Search, search)                           \
8258   V(Split, split)                             \
8259   V(ToPrimitive, to_primitive)                \
8260   V(ToStringTag, to_string_tag)               \
8261   V(Unscopables, unscopables)
8262 
8263 #define SYMBOL_GETTER(Name, name)                                   \
8264   Local<Symbol> v8::Symbol::Get##Name(Isolate* isolate) {           \
8265     i::Isolate* i_isolate = reinterpret_cast<i::Isolate*>(isolate); \
8266     return Utils::ToLocal(i_isolate->factory()->name##_symbol());   \
8267   }
8268 
WELL_KNOWN_SYMBOLS(SYMBOL_GETTER)8269 WELL_KNOWN_SYMBOLS(SYMBOL_GETTER)
8270 
8271 #undef SYMBOL_GETTER
8272 #undef WELL_KNOWN_SYMBOLS
8273 
8274 Local<Private> v8::Private::New(Isolate* isolate, Local<String> name) {
8275   i::Isolate* i_isolate = reinterpret_cast<i::Isolate*>(isolate);
8276   API_RCS_SCOPE(i_isolate, Private, New);
8277   ENTER_V8_NO_SCRIPT_NO_EXCEPTION(i_isolate);
8278   i::Handle<i::Symbol> symbol = i_isolate->factory()->NewPrivateSymbol();
8279   if (!name.IsEmpty()) symbol->set_description(*Utils::OpenHandle(*name));
8280   Local<Symbol> result = Utils::ToLocal(symbol);
8281   return v8::Local<Private>(reinterpret_cast<Private*>(*result));
8282 }
8283 
ForApi(Isolate * isolate,Local<String> name)8284 Local<Private> v8::Private::ForApi(Isolate* isolate, Local<String> name) {
8285   i::Isolate* i_isolate = reinterpret_cast<i::Isolate*>(isolate);
8286   i::Handle<i::String> i_name = Utils::OpenHandle(*name);
8287   Local<Symbol> result = Utils::ToLocal(
8288       i_isolate->SymbolFor(i::RootIndex::kApiPrivateSymbolTable, i_name, true));
8289   return v8::Local<Private>(reinterpret_cast<Private*>(*result));
8290 }
8291 
New(Isolate * isolate,double value)8292 Local<Number> v8::Number::New(Isolate* isolate, double value) {
8293   i::Isolate* internal_isolate = reinterpret_cast<i::Isolate*>(isolate);
8294   if (std::isnan(value)) {
8295     // Introduce only canonical NaN value into the VM, to avoid signaling NaNs.
8296     value = std::numeric_limits<double>::quiet_NaN();
8297   }
8298   ENTER_V8_NO_SCRIPT_NO_EXCEPTION(internal_isolate);
8299   i::Handle<i::Object> result = internal_isolate->factory()->NewNumber(value);
8300   return Utils::NumberToLocal(result);
8301 }
8302 
New(Isolate * isolate,int32_t value)8303 Local<Integer> v8::Integer::New(Isolate* isolate, int32_t value) {
8304   i::Isolate* internal_isolate = reinterpret_cast<i::Isolate*>(isolate);
8305   if (i::Smi::IsValid(value)) {
8306     return Utils::IntegerToLocal(
8307         i::Handle<i::Object>(i::Smi::FromInt(value), internal_isolate));
8308   }
8309   ENTER_V8_NO_SCRIPT_NO_EXCEPTION(internal_isolate);
8310   i::Handle<i::Object> result = internal_isolate->factory()->NewNumber(value);
8311   return Utils::IntegerToLocal(result);
8312 }
8313 
NewFromUnsigned(Isolate * isolate,uint32_t value)8314 Local<Integer> v8::Integer::NewFromUnsigned(Isolate* isolate, uint32_t value) {
8315   i::Isolate* internal_isolate = reinterpret_cast<i::Isolate*>(isolate);
8316   bool fits_into_int32_t = (value & (1 << 31)) == 0;
8317   if (fits_into_int32_t) {
8318     return Integer::New(isolate, static_cast<int32_t>(value));
8319   }
8320   ENTER_V8_NO_SCRIPT_NO_EXCEPTION(internal_isolate);
8321   i::Handle<i::Object> result = internal_isolate->factory()->NewNumber(value);
8322   return Utils::IntegerToLocal(result);
8323 }
8324 
New(Isolate * isolate,int64_t value)8325 Local<BigInt> v8::BigInt::New(Isolate* isolate, int64_t value) {
8326   i::Isolate* internal_isolate = reinterpret_cast<i::Isolate*>(isolate);
8327   ENTER_V8_NO_SCRIPT_NO_EXCEPTION(internal_isolate);
8328   i::Handle<i::BigInt> result = i::BigInt::FromInt64(internal_isolate, value);
8329   return Utils::ToLocal(result);
8330 }
8331 
NewFromUnsigned(Isolate * isolate,uint64_t value)8332 Local<BigInt> v8::BigInt::NewFromUnsigned(Isolate* isolate, uint64_t value) {
8333   i::Isolate* internal_isolate = reinterpret_cast<i::Isolate*>(isolate);
8334   ENTER_V8_NO_SCRIPT_NO_EXCEPTION(internal_isolate);
8335   i::Handle<i::BigInt> result = i::BigInt::FromUint64(internal_isolate, value);
8336   return Utils::ToLocal(result);
8337 }
8338 
NewFromWords(Local<Context> context,int sign_bit,int word_count,const uint64_t * words)8339 MaybeLocal<BigInt> v8::BigInt::NewFromWords(Local<Context> context,
8340                                             int sign_bit, int word_count,
8341                                             const uint64_t* words) {
8342   i::Isolate* isolate = reinterpret_cast<i::Isolate*>(context->GetIsolate());
8343   ENTER_V8_NO_SCRIPT(isolate, context, BigInt, NewFromWords,
8344                      MaybeLocal<BigInt>(), InternalEscapableScope);
8345   i::MaybeHandle<i::BigInt> result =
8346       i::BigInt::FromWords64(isolate, sign_bit, word_count, words);
8347   has_pending_exception = result.is_null();
8348   RETURN_ON_FAILED_EXECUTION(BigInt);
8349   RETURN_ESCAPED(Utils::ToLocal(result.ToHandleChecked()));
8350 }
8351 
Uint64Value(bool * lossless) const8352 uint64_t v8::BigInt::Uint64Value(bool* lossless) const {
8353   i::Handle<i::BigInt> handle = Utils::OpenHandle(this);
8354   return handle->AsUint64(lossless);
8355 }
8356 
Int64Value(bool * lossless) const8357 int64_t v8::BigInt::Int64Value(bool* lossless) const {
8358   i::Handle<i::BigInt> handle = Utils::OpenHandle(this);
8359   return handle->AsInt64(lossless);
8360 }
8361 
WordCount() const8362 int BigInt::WordCount() const {
8363   i::Handle<i::BigInt> handle = Utils::OpenHandle(this);
8364   return handle->Words64Count();
8365 }
8366 
ToWordsArray(int * sign_bit,int * word_count,uint64_t * words) const8367 void BigInt::ToWordsArray(int* sign_bit, int* word_count,
8368                           uint64_t* words) const {
8369   i::Handle<i::BigInt> handle = Utils::OpenHandle(this);
8370   return handle->ToWordsArray64(sign_bit, word_count, words);
8371 }
8372 
ReportExternalAllocationLimitReached()8373 void Isolate::ReportExternalAllocationLimitReached() {
8374   i::Heap* heap = reinterpret_cast<i::Isolate*>(this)->heap();
8375   if (heap->gc_state() != i::Heap::NOT_IN_GC) return;
8376   heap->ReportExternalMemoryPressure();
8377 }
8378 
GetHeapProfiler()8379 HeapProfiler* Isolate::GetHeapProfiler() {
8380   i::HeapProfiler* heap_profiler =
8381       reinterpret_cast<i::Isolate*>(this)->heap_profiler();
8382   return reinterpret_cast<HeapProfiler*>(heap_profiler);
8383 }
8384 
SetIdle(bool is_idle)8385 void Isolate::SetIdle(bool is_idle) {
8386   i::Isolate* isolate = reinterpret_cast<i::Isolate*>(this);
8387   isolate->SetIdle(is_idle);
8388 }
8389 
GetArrayBufferAllocator()8390 ArrayBuffer::Allocator* Isolate::GetArrayBufferAllocator() {
8391   i::Isolate* isolate = reinterpret_cast<i::Isolate*>(this);
8392   return isolate->array_buffer_allocator();
8393 }
8394 
InContext()8395 bool Isolate::InContext() {
8396   i::Isolate* isolate = reinterpret_cast<i::Isolate*>(this);
8397   return !isolate->context().is_null();
8398 }
8399 
ClearKeptObjects()8400 void Isolate::ClearKeptObjects() {
8401   i::Isolate* isolate = reinterpret_cast<i::Isolate*>(this);
8402   isolate->ClearKeptObjects();
8403 }
8404 
GetCurrentContext()8405 v8::Local<v8::Context> Isolate::GetCurrentContext() {
8406   i::Isolate* isolate = reinterpret_cast<i::Isolate*>(this);
8407   i::Context context = isolate->context();
8408   if (context.is_null()) return Local<Context>();
8409   i::Context native_context = context.native_context();
8410   if (native_context.is_null()) return Local<Context>();
8411   return Utils::ToLocal(i::Handle<i::Context>(native_context, isolate));
8412 }
8413 
GetEnteredOrMicrotaskContext()8414 v8::Local<v8::Context> Isolate::GetEnteredOrMicrotaskContext() {
8415   i::Isolate* isolate = reinterpret_cast<i::Isolate*>(this);
8416   i::Handle<i::Object> last =
8417       isolate->handle_scope_implementer()->LastEnteredOrMicrotaskContext();
8418   if (last.is_null()) return Local<Context>();
8419   DCHECK(last->IsNativeContext());
8420   return Utils::ToLocal(i::Handle<i::Context>::cast(last));
8421 }
8422 
GetIncumbentContext()8423 v8::Local<v8::Context> Isolate::GetIncumbentContext() {
8424   i::Isolate* isolate = reinterpret_cast<i::Isolate*>(this);
8425   i::Handle<i::Context> context = isolate->GetIncumbentContext();
8426   return Utils::ToLocal(context);
8427 }
8428 
ThrowError(v8::Local<v8::String> message)8429 v8::Local<Value> Isolate::ThrowError(v8::Local<v8::String> message) {
8430   return ThrowException(v8::Exception::Error(message));
8431 }
8432 
ThrowException(v8::Local<v8::Value> value)8433 v8::Local<Value> Isolate::ThrowException(v8::Local<v8::Value> value) {
8434   i::Isolate* isolate = reinterpret_cast<i::Isolate*>(this);
8435   ENTER_V8_DO_NOT_USE(isolate);
8436   // If we're passed an empty handle, we throw an undefined exception
8437   // to deal more gracefully with out of memory situations.
8438   if (value.IsEmpty()) {
8439     isolate->ScheduleThrow(i::ReadOnlyRoots(isolate).undefined_value());
8440   } else {
8441     isolate->ScheduleThrow(*Utils::OpenHandle(*value));
8442   }
8443   return v8::Undefined(reinterpret_cast<v8::Isolate*>(isolate));
8444 }
8445 
AddGCPrologueCallback(GCCallbackWithData callback,void * data,GCType gc_type)8446 void Isolate::AddGCPrologueCallback(GCCallbackWithData callback, void* data,
8447                                     GCType gc_type) {
8448   i::Isolate* isolate = reinterpret_cast<i::Isolate*>(this);
8449   isolate->heap()->AddGCPrologueCallback(callback, gc_type, data);
8450 }
8451 
RemoveGCPrologueCallback(GCCallbackWithData callback,void * data)8452 void Isolate::RemoveGCPrologueCallback(GCCallbackWithData callback,
8453                                        void* data) {
8454   i::Isolate* isolate = reinterpret_cast<i::Isolate*>(this);
8455   isolate->heap()->RemoveGCPrologueCallback(callback, data);
8456 }
8457 
AddGCEpilogueCallback(GCCallbackWithData callback,void * data,GCType gc_type)8458 void Isolate::AddGCEpilogueCallback(GCCallbackWithData callback, void* data,
8459                                     GCType gc_type) {
8460   i::Isolate* isolate = reinterpret_cast<i::Isolate*>(this);
8461   isolate->heap()->AddGCEpilogueCallback(callback, gc_type, data);
8462 }
8463 
RemoveGCEpilogueCallback(GCCallbackWithData callback,void * data)8464 void Isolate::RemoveGCEpilogueCallback(GCCallbackWithData callback,
8465                                        void* data) {
8466   i::Isolate* isolate = reinterpret_cast<i::Isolate*>(this);
8467   isolate->heap()->RemoveGCEpilogueCallback(callback, data);
8468 }
8469 
CallGCCallbackWithoutData(Isolate * isolate,GCType type,GCCallbackFlags flags,void * data)8470 static void CallGCCallbackWithoutData(Isolate* isolate, GCType type,
8471                                       GCCallbackFlags flags, void* data) {
8472   reinterpret_cast<Isolate::GCCallback>(data)(isolate, type, flags);
8473 }
8474 
AddGCPrologueCallback(GCCallback callback,GCType gc_type)8475 void Isolate::AddGCPrologueCallback(GCCallback callback, GCType gc_type) {
8476   void* data = reinterpret_cast<void*>(callback);
8477   AddGCPrologueCallback(CallGCCallbackWithoutData, data, gc_type);
8478 }
8479 
RemoveGCPrologueCallback(GCCallback callback)8480 void Isolate::RemoveGCPrologueCallback(GCCallback callback) {
8481   void* data = reinterpret_cast<void*>(callback);
8482   RemoveGCPrologueCallback(CallGCCallbackWithoutData, data);
8483 }
8484 
AddGCEpilogueCallback(GCCallback callback,GCType gc_type)8485 void Isolate::AddGCEpilogueCallback(GCCallback callback, GCType gc_type) {
8486   void* data = reinterpret_cast<void*>(callback);
8487   AddGCEpilogueCallback(CallGCCallbackWithoutData, data, gc_type);
8488 }
8489 
RemoveGCEpilogueCallback(GCCallback callback)8490 void Isolate::RemoveGCEpilogueCallback(GCCallback callback) {
8491   void* data = reinterpret_cast<void*>(callback);
8492   RemoveGCEpilogueCallback(CallGCCallbackWithoutData, data);
8493 }
8494 
SetEmbedderHeapTracer(EmbedderHeapTracer * tracer)8495 void Isolate::SetEmbedderHeapTracer(EmbedderHeapTracer* tracer) {
8496   i::Isolate* isolate = reinterpret_cast<i::Isolate*>(this);
8497   CHECK_NULL(isolate->heap()->cpp_heap());
8498   isolate->heap()->SetEmbedderHeapTracer(tracer);
8499 }
8500 
GetEmbedderHeapTracer()8501 EmbedderHeapTracer* Isolate::GetEmbedderHeapTracer() {
8502   i::Isolate* isolate = reinterpret_cast<i::Isolate*>(this);
8503   return isolate->heap()->GetEmbedderHeapTracer();
8504 }
8505 
SetEmbedderRootsHandler(EmbedderRootsHandler * handler)8506 void Isolate::SetEmbedderRootsHandler(EmbedderRootsHandler* handler) {
8507   i::Isolate* isolate = reinterpret_cast<i::Isolate*>(this);
8508   isolate->heap()->SetEmbedderRootsHandler(handler);
8509 }
8510 
AttachCppHeap(CppHeap * cpp_heap)8511 void Isolate::AttachCppHeap(CppHeap* cpp_heap) {
8512   i::Isolate* isolate = reinterpret_cast<i::Isolate*>(this);
8513   CHECK_NULL(GetEmbedderHeapTracer());
8514   isolate->heap()->AttachCppHeap(cpp_heap);
8515 }
8516 
DetachCppHeap()8517 void Isolate::DetachCppHeap() {
8518   i::Isolate* isolate = reinterpret_cast<i::Isolate*>(this);
8519   isolate->heap()->DetachCppHeap();
8520 }
8521 
GetCppHeap() const8522 CppHeap* Isolate::GetCppHeap() const {
8523   const i::Isolate* isolate = reinterpret_cast<const i::Isolate*>(this);
8524   return isolate->heap()->cpp_heap();
8525 }
8526 
SetGetExternallyAllocatedMemoryInBytesCallback(GetExternallyAllocatedMemoryInBytesCallback callback)8527 void Isolate::SetGetExternallyAllocatedMemoryInBytesCallback(
8528     GetExternallyAllocatedMemoryInBytesCallback callback) {
8529   i::Isolate* isolate = reinterpret_cast<i::Isolate*>(this);
8530   isolate->heap()->SetGetExternallyAllocatedMemoryInBytesCallback(callback);
8531 }
8532 
TerminateExecution()8533 void Isolate::TerminateExecution() {
8534   i::Isolate* isolate = reinterpret_cast<i::Isolate*>(this);
8535   isolate->stack_guard()->RequestTerminateExecution();
8536 }
8537 
IsExecutionTerminating()8538 bool Isolate::IsExecutionTerminating() {
8539   i::Isolate* isolate = reinterpret_cast<i::Isolate*>(this);
8540   return IsExecutionTerminatingCheck(isolate);
8541 }
8542 
CancelTerminateExecution()8543 void Isolate::CancelTerminateExecution() {
8544   i::Isolate* isolate = reinterpret_cast<i::Isolate*>(this);
8545   isolate->stack_guard()->ClearTerminateExecution();
8546   isolate->CancelTerminateExecution();
8547 }
8548 
RequestInterrupt(InterruptCallback callback,void * data)8549 void Isolate::RequestInterrupt(InterruptCallback callback, void* data) {
8550   i::Isolate* isolate = reinterpret_cast<i::Isolate*>(this);
8551   isolate->RequestInterrupt(callback, data);
8552 }
8553 
HasPendingBackgroundTasks()8554 bool Isolate::HasPendingBackgroundTasks() {
8555 #if V8_ENABLE_WEBASSEMBLY
8556   i::Isolate* isolate = reinterpret_cast<i::Isolate*>(this);
8557   return i::wasm::GetWasmEngine()->HasRunningCompileJob(isolate);
8558 #else
8559   return false;
8560 #endif  // V8_ENABLE_WEBASSEMBLY
8561 }
8562 
RequestGarbageCollectionForTesting(GarbageCollectionType type)8563 void Isolate::RequestGarbageCollectionForTesting(GarbageCollectionType type) {
8564   Utils::ApiCheck(i::FLAG_expose_gc,
8565                   "v8::Isolate::RequestGarbageCollectionForTesting",
8566                   "Must use --expose-gc");
8567   if (type == kMinorGarbageCollection) {
8568     reinterpret_cast<i::Isolate*>(this)->heap()->CollectGarbage(
8569         i::NEW_SPACE, i::GarbageCollectionReason::kTesting,
8570         kGCCallbackFlagForced);
8571   } else {
8572     DCHECK_EQ(kFullGarbageCollection, type);
8573     reinterpret_cast<i::Isolate*>(this)->heap()->PreciseCollectAllGarbage(
8574         i::Heap::kNoGCFlags, i::GarbageCollectionReason::kTesting,
8575         kGCCallbackFlagForced);
8576   }
8577 }
8578 
RequestGarbageCollectionForTesting(GarbageCollectionType type,EmbedderHeapTracer::EmbedderStackState stack_state)8579 void Isolate::RequestGarbageCollectionForTesting(
8580     GarbageCollectionType type,
8581     EmbedderHeapTracer::EmbedderStackState stack_state) {
8582   base::Optional<i::EmbedderStackStateScope> stack_scope;
8583   if (type == kFullGarbageCollection) {
8584     stack_scope.emplace(reinterpret_cast<i::Isolate*>(this)->heap(),
8585                         i::EmbedderStackStateScope::kExplicitInvocation,
8586                         stack_state);
8587   }
8588   RequestGarbageCollectionForTesting(type);
8589 }
8590 
GetCurrent()8591 Isolate* Isolate::GetCurrent() {
8592   i::Isolate* isolate = i::Isolate::Current();
8593   return reinterpret_cast<Isolate*>(isolate);
8594 }
8595 
TryGetCurrent()8596 Isolate* Isolate::TryGetCurrent() {
8597   i::Isolate* isolate = i::Isolate::TryGetCurrent();
8598   return reinterpret_cast<Isolate*>(isolate);
8599 }
8600 
IsCurrent() const8601 bool Isolate::IsCurrent() const {
8602   return reinterpret_cast<const i::Isolate*>(this)->IsCurrent();
8603 }
8604 
8605 // static
Allocate()8606 Isolate* Isolate::Allocate() {
8607   return reinterpret_cast<Isolate*>(i::Isolate::New());
8608 }
8609 
8610 Isolate::CreateParams::CreateParams() = default;
8611 
8612 Isolate::CreateParams::~CreateParams() = default;
8613 
8614 // static
8615 // This is separate so that tests can provide a different |isolate|.
Initialize(Isolate * isolate,const v8::Isolate::CreateParams & params)8616 void Isolate::Initialize(Isolate* isolate,
8617                          const v8::Isolate::CreateParams& params) {
8618   i::Isolate* i_isolate = reinterpret_cast<i::Isolate*>(isolate);
8619   TRACE_EVENT_CALL_STATS_SCOPED(i_isolate, "v8", "V8.IsolateInitialize");
8620   if (auto allocator = params.array_buffer_allocator_shared) {
8621     CHECK(params.array_buffer_allocator == nullptr ||
8622           params.array_buffer_allocator == allocator.get());
8623     i_isolate->set_array_buffer_allocator(allocator.get());
8624     i_isolate->set_array_buffer_allocator_shared(std::move(allocator));
8625   } else {
8626     CHECK_NOT_NULL(params.array_buffer_allocator);
8627     i_isolate->set_array_buffer_allocator(params.array_buffer_allocator);
8628   }
8629   if (params.snapshot_blob != nullptr) {
8630     i_isolate->set_snapshot_blob(params.snapshot_blob);
8631   } else {
8632     i_isolate->set_snapshot_blob(i::Snapshot::DefaultSnapshotBlob());
8633   }
8634 
8635   if (params.fatal_error_callback) {
8636     isolate->SetFatalErrorHandler(params.fatal_error_callback);
8637   }
8638 
8639   if (params.oom_error_callback) {
8640     isolate->SetOOMErrorHandler(params.oom_error_callback);
8641   }
8642 
8643   if (params.counter_lookup_callback) {
8644     isolate->SetCounterFunction(params.counter_lookup_callback);
8645   }
8646 
8647   if (params.create_histogram_callback) {
8648     isolate->SetCreateHistogramFunction(params.create_histogram_callback);
8649   }
8650 
8651   if (params.add_histogram_sample_callback) {
8652     isolate->SetAddHistogramSampleFunction(
8653         params.add_histogram_sample_callback);
8654   }
8655 
8656   i_isolate->set_api_external_references(params.external_references);
8657   i_isolate->set_allow_atomics_wait(params.allow_atomics_wait);
8658 
8659   i_isolate->heap()->ConfigureHeap(params.constraints);
8660   if (params.constraints.stack_limit() != nullptr) {
8661     uintptr_t limit =
8662         reinterpret_cast<uintptr_t>(params.constraints.stack_limit());
8663     i_isolate->stack_guard()->SetStackLimit(limit);
8664   }
8665 
8666   if (params.experimental_attach_to_shared_isolate != nullptr) {
8667     i_isolate->set_shared_isolate(reinterpret_cast<i::Isolate*>(
8668         params.experimental_attach_to_shared_isolate));
8669   }
8670 
8671   // TODO(v8:2487): Once we got rid of Isolate::Current(), we can remove this.
8672   Isolate::Scope isolate_scope(isolate);
8673   if (i_isolate->snapshot_blob() == nullptr) {
8674     FATAL(
8675         "V8 snapshot blob was not set during initialization. This can mean "
8676         "that the snapshot blob file is corrupted or missing.");
8677   }
8678   if (!i::Snapshot::Initialize(i_isolate)) {
8679     // If snapshot data was provided and we failed to deserialize it must
8680     // have been corrupted.
8681     FATAL(
8682         "Failed to deserialize the V8 snapshot blob. This can mean that the "
8683         "snapshot blob file is corrupted or missing.");
8684   }
8685 
8686   {
8687     // Set up code event handlers. Needs to be after i::Snapshot::Initialize
8688     // because that is where we add the isolate to WasmEngine.
8689     auto code_event_handler = params.code_event_handler;
8690     if (code_event_handler) {
8691       isolate->SetJitCodeEventHandler(kJitCodeEventEnumExisting,
8692                                       code_event_handler);
8693     }
8694   }
8695 
8696   i_isolate->set_only_terminate_in_safe_scope(
8697       params.only_terminate_in_safe_scope);
8698   i_isolate->set_embedder_wrapper_type_index(
8699       params.embedder_wrapper_type_index);
8700   i_isolate->set_embedder_wrapper_object_index(
8701       params.embedder_wrapper_object_index);
8702 
8703   if (!i::V8::GetCurrentPlatform()
8704            ->GetForegroundTaskRunner(isolate)
8705            ->NonNestableTasksEnabled()) {
8706     FATAL(
8707         "The current platform's foreground task runner does not have "
8708         "non-nestable tasks enabled. The embedder must provide one.");
8709   }
8710 }
8711 
New(const Isolate::CreateParams & params)8712 Isolate* Isolate::New(const Isolate::CreateParams& params) {
8713   Isolate* isolate = Allocate();
8714   Initialize(isolate, params);
8715   return isolate;
8716 }
8717 
Dispose()8718 void Isolate::Dispose() {
8719   i::Isolate* isolate = reinterpret_cast<i::Isolate*>(this);
8720   if (!Utils::ApiCheck(!isolate->IsInUse(), "v8::Isolate::Dispose()",
8721                        "Disposing the isolate that is entered by a thread.")) {
8722     return;
8723   }
8724   i::Isolate::Delete(isolate);
8725 }
8726 
DumpAndResetStats()8727 void Isolate::DumpAndResetStats() {
8728   i::Isolate* isolate = reinterpret_cast<i::Isolate*>(this);
8729   isolate->DumpAndResetStats();
8730 }
8731 
DiscardThreadSpecificMetadata()8732 void Isolate::DiscardThreadSpecificMetadata() {
8733   i::Isolate* isolate = reinterpret_cast<i::Isolate*>(this);
8734   isolate->DiscardPerThreadDataForThisThread();
8735 }
8736 
Enter()8737 void Isolate::Enter() {
8738   i::Isolate* isolate = reinterpret_cast<i::Isolate*>(this);
8739   isolate->Enter();
8740 }
8741 
Exit()8742 void Isolate::Exit() {
8743   i::Isolate* isolate = reinterpret_cast<i::Isolate*>(this);
8744   isolate->Exit();
8745 }
8746 
SetAbortOnUncaughtExceptionCallback(AbortOnUncaughtExceptionCallback callback)8747 void Isolate::SetAbortOnUncaughtExceptionCallback(
8748     AbortOnUncaughtExceptionCallback callback) {
8749   i::Isolate* isolate = reinterpret_cast<i::Isolate*>(this);
8750   isolate->SetAbortOnUncaughtExceptionCallback(callback);
8751 }
8752 
SetHostImportModuleDynamicallyCallback(HostImportModuleDynamicallyWithImportAssertionsCallback callback)8753 void Isolate::SetHostImportModuleDynamicallyCallback(
8754     HostImportModuleDynamicallyWithImportAssertionsCallback callback) {
8755   i::Isolate* isolate = reinterpret_cast<i::Isolate*>(this);
8756   isolate->SetHostImportModuleDynamicallyCallback(callback);
8757 }
8758 
SetHostImportModuleDynamicallyCallback(HostImportModuleDynamicallyCallback callback)8759 void Isolate::SetHostImportModuleDynamicallyCallback(
8760     HostImportModuleDynamicallyCallback callback) {
8761   i::Isolate* isolate = reinterpret_cast<i::Isolate*>(this);
8762   isolate->SetHostImportModuleDynamicallyCallback(callback);
8763 }
8764 
SetHostInitializeImportMetaObjectCallback(HostInitializeImportMetaObjectCallback callback)8765 void Isolate::SetHostInitializeImportMetaObjectCallback(
8766     HostInitializeImportMetaObjectCallback callback) {
8767   i::Isolate* isolate = reinterpret_cast<i::Isolate*>(this);
8768   isolate->SetHostInitializeImportMetaObjectCallback(callback);
8769 }
8770 
SetHostCreateShadowRealmContextCallback(HostCreateShadowRealmContextCallback callback)8771 void Isolate::SetHostCreateShadowRealmContextCallback(
8772     HostCreateShadowRealmContextCallback callback) {
8773   i::Isolate* isolate = reinterpret_cast<i::Isolate*>(this);
8774   isolate->SetHostCreateShadowRealmContextCallback(callback);
8775 }
8776 
SetPrepareStackTraceCallback(PrepareStackTraceCallback callback)8777 void Isolate::SetPrepareStackTraceCallback(PrepareStackTraceCallback callback) {
8778   i::Isolate* isolate = reinterpret_cast<i::Isolate*>(this);
8779   isolate->SetPrepareStackTraceCallback(callback);
8780 }
8781 
DisallowJavascriptExecutionScope(Isolate * isolate,Isolate::DisallowJavascriptExecutionScope::OnFailure on_failure)8782 Isolate::DisallowJavascriptExecutionScope::DisallowJavascriptExecutionScope(
8783     Isolate* isolate,
8784     Isolate::DisallowJavascriptExecutionScope::OnFailure on_failure)
8785     : on_failure_(on_failure), isolate_(isolate) {
8786   i::Isolate* i_isolate = reinterpret_cast<i::Isolate*>(isolate);
8787   switch (on_failure_) {
8788     case CRASH_ON_FAILURE:
8789       i::DisallowJavascriptExecution::Open(i_isolate,
8790                                            &was_execution_allowed_assert_);
8791       break;
8792     case THROW_ON_FAILURE:
8793       i::ThrowOnJavascriptExecution::Open(i_isolate,
8794                                           &was_execution_allowed_throws_);
8795       break;
8796     case DUMP_ON_FAILURE:
8797       i::DumpOnJavascriptExecution::Open(i_isolate,
8798                                          &was_execution_allowed_dump_);
8799       break;
8800     default:
8801       UNREACHABLE();
8802   }
8803 }
8804 
~DisallowJavascriptExecutionScope()8805 Isolate::DisallowJavascriptExecutionScope::~DisallowJavascriptExecutionScope() {
8806   i::Isolate* i_isolate = reinterpret_cast<i::Isolate*>(isolate_);
8807   switch (on_failure_) {
8808     case CRASH_ON_FAILURE:
8809       i::DisallowJavascriptExecution::Close(i_isolate,
8810                                             was_execution_allowed_assert_);
8811       break;
8812     case THROW_ON_FAILURE:
8813       i::ThrowOnJavascriptExecution::Close(i_isolate,
8814                                            was_execution_allowed_throws_);
8815       break;
8816     case DUMP_ON_FAILURE:
8817       i::DumpOnJavascriptExecution::Close(i_isolate,
8818                                           was_execution_allowed_dump_);
8819       break;
8820     default:
8821       UNREACHABLE();
8822   }
8823 }
8824 
AllowJavascriptExecutionScope(Isolate * isolate)8825 Isolate::AllowJavascriptExecutionScope::AllowJavascriptExecutionScope(
8826     Isolate* isolate)
8827     : isolate_(isolate) {
8828   i::Isolate* i_isolate = reinterpret_cast<i::Isolate*>(isolate);
8829   i::AllowJavascriptExecution::Open(i_isolate, &was_execution_allowed_assert_);
8830   i::NoThrowOnJavascriptExecution::Open(i_isolate,
8831                                         &was_execution_allowed_throws_);
8832   i::NoDumpOnJavascriptExecution::Open(i_isolate, &was_execution_allowed_dump_);
8833 }
8834 
~AllowJavascriptExecutionScope()8835 Isolate::AllowJavascriptExecutionScope::~AllowJavascriptExecutionScope() {
8836   i::Isolate* i_isolate = reinterpret_cast<i::Isolate*>(isolate_);
8837   i::AllowJavascriptExecution::Close(i_isolate, was_execution_allowed_assert_);
8838   i::NoThrowOnJavascriptExecution::Close(i_isolate,
8839                                          was_execution_allowed_throws_);
8840   i::NoDumpOnJavascriptExecution::Close(i_isolate, was_execution_allowed_dump_);
8841 }
8842 
SuppressMicrotaskExecutionScope(Isolate * isolate,MicrotaskQueue * microtask_queue)8843 Isolate::SuppressMicrotaskExecutionScope::SuppressMicrotaskExecutionScope(
8844     Isolate* isolate, MicrotaskQueue* microtask_queue)
8845     : isolate_(reinterpret_cast<i::Isolate*>(isolate)),
8846       microtask_queue_(microtask_queue
8847                            ? static_cast<i::MicrotaskQueue*>(microtask_queue)
8848                            : isolate_->default_microtask_queue()) {
8849   isolate_->thread_local_top()->IncrementCallDepth(this);
8850   microtask_queue_->IncrementMicrotasksSuppressions();
8851 }
8852 
~SuppressMicrotaskExecutionScope()8853 Isolate::SuppressMicrotaskExecutionScope::~SuppressMicrotaskExecutionScope() {
8854   microtask_queue_->DecrementMicrotasksSuppressions();
8855   isolate_->thread_local_top()->DecrementCallDepth(this);
8856 }
8857 
SafeForTerminationScope(v8::Isolate * isolate)8858 Isolate::SafeForTerminationScope::SafeForTerminationScope(v8::Isolate* isolate)
8859     : isolate_(reinterpret_cast<i::Isolate*>(isolate)),
8860       prev_value_(isolate_->next_v8_call_is_safe_for_termination()) {
8861   isolate_->set_next_v8_call_is_safe_for_termination(true);
8862 }
8863 
~SafeForTerminationScope()8864 Isolate::SafeForTerminationScope::~SafeForTerminationScope() {
8865   isolate_->set_next_v8_call_is_safe_for_termination(prev_value_);
8866 }
8867 
GetDataFromSnapshotOnce(size_t index)8868 i::Address* Isolate::GetDataFromSnapshotOnce(size_t index) {
8869   i::Isolate* i_isolate = reinterpret_cast<i::Isolate*>(this);
8870   i::FixedArray list = i_isolate->heap()->serialized_objects();
8871   return GetSerializedDataFromFixedArray(i_isolate, list, index);
8872 }
8873 
GetHeapStatistics(HeapStatistics * heap_statistics)8874 void Isolate::GetHeapStatistics(HeapStatistics* heap_statistics) {
8875   i::Isolate* isolate = reinterpret_cast<i::Isolate*>(this);
8876   i::Heap* heap = isolate->heap();
8877 
8878   // The order of acquiring memory statistics is important here. We query in
8879   // this order because of concurrent allocation: 1) used memory 2) comitted
8880   // physical memory 3) committed memory. Therefore the condition used <=
8881   // committed physical <= committed should hold.
8882   heap_statistics->used_global_handles_size_ = heap->UsedGlobalHandlesSize();
8883   heap_statistics->total_global_handles_size_ = heap->TotalGlobalHandlesSize();
8884   DCHECK_LE(heap_statistics->used_global_handles_size_,
8885             heap_statistics->total_global_handles_size_);
8886 
8887   heap_statistics->used_heap_size_ = heap->SizeOfObjects();
8888   heap_statistics->total_physical_size_ = heap->CommittedPhysicalMemory();
8889   heap_statistics->total_heap_size_ = heap->CommittedMemory();
8890 
8891   heap_statistics->total_available_size_ = heap->Available();
8892 
8893   if (!i::ReadOnlyHeap::IsReadOnlySpaceShared()) {
8894     i::ReadOnlySpace* ro_space = heap->read_only_space();
8895     heap_statistics->used_heap_size_ += ro_space->Size();
8896     heap_statistics->total_physical_size_ +=
8897         ro_space->CommittedPhysicalMemory();
8898     heap_statistics->total_heap_size_ += ro_space->CommittedMemory();
8899   }
8900 
8901   // TODO(dinfuehr): Right now used <= committed physical does not hold. Fix
8902   // this and add DCHECK.
8903   DCHECK_LE(heap_statistics->used_heap_size_,
8904             heap_statistics->total_heap_size_);
8905 
8906   heap_statistics->total_heap_size_executable_ =
8907       heap->CommittedMemoryExecutable();
8908   heap_statistics->heap_size_limit_ = heap->MaxReserved();
8909   // TODO(7424): There is no public API for the {WasmEngine} yet. Once such an
8910   // API becomes available we should report the malloced memory separately. For
8911   // now we just add the values, thereby over-approximating the peak slightly.
8912   heap_statistics->malloced_memory_ =
8913       isolate->allocator()->GetCurrentMemoryUsage() +
8914       isolate->string_table()->GetCurrentMemoryUsage();
8915   // On 32-bit systems backing_store_bytes() might overflow size_t temporarily
8916   // due to concurrent array buffer sweeping.
8917   heap_statistics->external_memory_ =
8918       isolate->heap()->backing_store_bytes() < SIZE_MAX
8919           ? static_cast<size_t>(isolate->heap()->backing_store_bytes())
8920           : SIZE_MAX;
8921   heap_statistics->peak_malloced_memory_ =
8922       isolate->allocator()->GetMaxMemoryUsage();
8923   heap_statistics->number_of_native_contexts_ = heap->NumberOfNativeContexts();
8924   heap_statistics->number_of_detached_contexts_ =
8925       heap->NumberOfDetachedContexts();
8926   heap_statistics->does_zap_garbage_ = heap->ShouldZapGarbage();
8927 
8928 #if V8_ENABLE_WEBASSEMBLY
8929   heap_statistics->malloced_memory_ +=
8930       i::wasm::GetWasmEngine()->allocator()->GetCurrentMemoryUsage();
8931   heap_statistics->peak_malloced_memory_ +=
8932       i::wasm::GetWasmEngine()->allocator()->GetMaxMemoryUsage();
8933 #endif  // V8_ENABLE_WEBASSEMBLY
8934 }
8935 
NumberOfHeapSpaces()8936 size_t Isolate::NumberOfHeapSpaces() {
8937   return i::LAST_SPACE - i::FIRST_SPACE + 1;
8938 }
8939 
GetHeapSpaceStatistics(HeapSpaceStatistics * space_statistics,size_t index)8940 bool Isolate::GetHeapSpaceStatistics(HeapSpaceStatistics* space_statistics,
8941                                      size_t index) {
8942   if (!space_statistics) return false;
8943   if (!i::Heap::IsValidAllocationSpace(static_cast<i::AllocationSpace>(index)))
8944     return false;
8945 
8946   i::Isolate* isolate = reinterpret_cast<i::Isolate*>(this);
8947   i::Heap* heap = isolate->heap();
8948 
8949   i::AllocationSpace allocation_space = static_cast<i::AllocationSpace>(index);
8950   space_statistics->space_name_ = i::BaseSpace::GetSpaceName(allocation_space);
8951 
8952   if (allocation_space == i::RO_SPACE) {
8953     if (i::ReadOnlyHeap::IsReadOnlySpaceShared()) {
8954       // RO_SPACE memory is accounted for elsewhere when ReadOnlyHeap is shared.
8955       space_statistics->space_size_ = 0;
8956       space_statistics->space_used_size_ = 0;
8957       space_statistics->space_available_size_ = 0;
8958       space_statistics->physical_space_size_ = 0;
8959     } else {
8960       i::ReadOnlySpace* space = heap->read_only_space();
8961       space_statistics->space_size_ = space->CommittedMemory();
8962       space_statistics->space_used_size_ = space->Size();
8963       space_statistics->space_available_size_ = 0;
8964       space_statistics->physical_space_size_ = space->CommittedPhysicalMemory();
8965     }
8966   } else {
8967     i::Space* space = heap->space(static_cast<int>(index));
8968     space_statistics->space_size_ = space ? space->CommittedMemory() : 0;
8969     space_statistics->space_used_size_ = space ? space->SizeOfObjects() : 0;
8970     space_statistics->space_available_size_ = space ? space->Available() : 0;
8971     space_statistics->physical_space_size_ =
8972         space ? space->CommittedPhysicalMemory() : 0;
8973   }
8974   return true;
8975 }
8976 
NumberOfTrackedHeapObjectTypes()8977 size_t Isolate::NumberOfTrackedHeapObjectTypes() {
8978   i::Isolate* isolate = reinterpret_cast<i::Isolate*>(this);
8979   i::Heap* heap = isolate->heap();
8980   return heap->NumberOfTrackedHeapObjectTypes();
8981 }
8982 
GetHeapObjectStatisticsAtLastGC(HeapObjectStatistics * object_statistics,size_t type_index)8983 bool Isolate::GetHeapObjectStatisticsAtLastGC(
8984     HeapObjectStatistics* object_statistics, size_t type_index) {
8985   if (!object_statistics) return false;
8986   if (V8_LIKELY(!i::TracingFlags::is_gc_stats_enabled())) return false;
8987 
8988   i::Isolate* isolate = reinterpret_cast<i::Isolate*>(this);
8989   i::Heap* heap = isolate->heap();
8990   if (type_index >= heap->NumberOfTrackedHeapObjectTypes()) return false;
8991 
8992   const char* object_type;
8993   const char* object_sub_type;
8994   size_t object_count = heap->ObjectCountAtLastGC(type_index);
8995   size_t object_size = heap->ObjectSizeAtLastGC(type_index);
8996   if (!heap->GetObjectTypeName(type_index, &object_type, &object_sub_type)) {
8997     // There should be no objects counted when the type is unknown.
8998     DCHECK_EQ(object_count, 0U);
8999     DCHECK_EQ(object_size, 0U);
9000     return false;
9001   }
9002 
9003   object_statistics->object_type_ = object_type;
9004   object_statistics->object_sub_type_ = object_sub_type;
9005   object_statistics->object_count_ = object_count;
9006   object_statistics->object_size_ = object_size;
9007   return true;
9008 }
9009 
GetHeapCodeAndMetadataStatistics(HeapCodeStatistics * code_statistics)9010 bool Isolate::GetHeapCodeAndMetadataStatistics(
9011     HeapCodeStatistics* code_statistics) {
9012   if (!code_statistics) return false;
9013 
9014   i::Isolate* isolate = reinterpret_cast<i::Isolate*>(this);
9015   isolate->heap()->CollectCodeStatistics();
9016 
9017   code_statistics->code_and_metadata_size_ = isolate->code_and_metadata_size();
9018   code_statistics->bytecode_and_metadata_size_ =
9019       isolate->bytecode_and_metadata_size();
9020   code_statistics->external_script_source_size_ =
9021       isolate->external_script_source_size();
9022   code_statistics->cpu_profiler_metadata_size_ =
9023       i::CpuProfiler::GetAllProfilersMemorySize(
9024           reinterpret_cast<i::Isolate*>(isolate));
9025 
9026   return true;
9027 }
9028 
MeasureMemory(std::unique_ptr<MeasureMemoryDelegate> delegate,MeasureMemoryExecution execution)9029 bool Isolate::MeasureMemory(std::unique_ptr<MeasureMemoryDelegate> delegate,
9030                             MeasureMemoryExecution execution) {
9031   i::Isolate* isolate = reinterpret_cast<i::Isolate*>(this);
9032   return isolate->heap()->MeasureMemory(std::move(delegate), execution);
9033 }
9034 
Default(Isolate * isolate,Local<Context> context,Local<Promise::Resolver> promise_resolver,MeasureMemoryMode mode)9035 std::unique_ptr<MeasureMemoryDelegate> MeasureMemoryDelegate::Default(
9036     Isolate* isolate, Local<Context> context,
9037     Local<Promise::Resolver> promise_resolver, MeasureMemoryMode mode) {
9038   i::Isolate* i_isolate = reinterpret_cast<i::Isolate*>(isolate);
9039   i::Handle<i::NativeContext> native_context =
9040       handle(Utils::OpenHandle(*context)->native_context(), i_isolate);
9041   i::Handle<i::JSPromise> js_promise =
9042       i::Handle<i::JSPromise>::cast(Utils::OpenHandle(*promise_resolver));
9043   return i_isolate->heap()->MeasureMemoryDelegate(native_context, js_promise,
9044                                                   mode);
9045 }
9046 
GetStackSample(const RegisterState & state,void ** frames,size_t frames_limit,SampleInfo * sample_info)9047 void Isolate::GetStackSample(const RegisterState& state, void** frames,
9048                              size_t frames_limit, SampleInfo* sample_info) {
9049   RegisterState regs = state;
9050   i::Isolate* isolate = reinterpret_cast<i::Isolate*>(this);
9051   if (i::TickSample::GetStackSample(isolate, &regs,
9052                                     i::TickSample::kSkipCEntryFrame, frames,
9053                                     frames_limit, sample_info)) {
9054     return;
9055   }
9056   sample_info->frames_count = 0;
9057   sample_info->vm_state = OTHER;
9058   sample_info->external_callback_entry = nullptr;
9059 }
9060 
NumberOfPhantomHandleResetsSinceLastCall()9061 size_t Isolate::NumberOfPhantomHandleResetsSinceLastCall() {
9062   i::Isolate* isolate = reinterpret_cast<i::Isolate*>(this);
9063   return isolate->global_handles()->GetAndResetGlobalHandleResetCount();
9064 }
9065 
AdjustAmountOfExternalAllocatedMemory(int64_t change_in_bytes)9066 int64_t Isolate::AdjustAmountOfExternalAllocatedMemory(
9067     int64_t change_in_bytes) {
9068   // Try to check for unreasonably large or small values from the embedder.
9069   const int64_t kMaxReasonableBytes = int64_t(1) << 60;
9070   const int64_t kMinReasonableBytes = -kMaxReasonableBytes;
9071   STATIC_ASSERT(kMaxReasonableBytes >= i::JSArrayBuffer::kMaxByteLength);
9072 
9073   CHECK(kMinReasonableBytes <= change_in_bytes &&
9074         change_in_bytes < kMaxReasonableBytes);
9075 
9076   i::Isolate* i_isolate = reinterpret_cast<i::Isolate*>(this);
9077   int64_t amount = i_isolate->heap()->update_external_memory(change_in_bytes);
9078 
9079   if (change_in_bytes <= 0) return amount;
9080 
9081   if (amount > i_isolate->heap()->external_memory_limit()) {
9082     ReportExternalAllocationLimitReached();
9083   }
9084   return amount;
9085 }
9086 
SetEventLogger(LogEventCallback that)9087 void Isolate::SetEventLogger(LogEventCallback that) {
9088   // Do not overwrite the event logger if we want to log explicitly.
9089   if (i::FLAG_log_internal_timer_events) return;
9090   i::Isolate* isolate = reinterpret_cast<i::Isolate*>(this);
9091   isolate->set_event_logger(that);
9092 }
9093 
AddBeforeCallEnteredCallback(BeforeCallEnteredCallback callback)9094 void Isolate::AddBeforeCallEnteredCallback(BeforeCallEnteredCallback callback) {
9095   if (callback == nullptr) return;
9096   i::Isolate* isolate = reinterpret_cast<i::Isolate*>(this);
9097   isolate->AddBeforeCallEnteredCallback(callback);
9098 }
9099 
RemoveBeforeCallEnteredCallback(BeforeCallEnteredCallback callback)9100 void Isolate::RemoveBeforeCallEnteredCallback(
9101     BeforeCallEnteredCallback callback) {
9102   i::Isolate* isolate = reinterpret_cast<i::Isolate*>(this);
9103   isolate->RemoveBeforeCallEnteredCallback(callback);
9104 }
9105 
AddCallCompletedCallback(CallCompletedCallback callback)9106 void Isolate::AddCallCompletedCallback(CallCompletedCallback callback) {
9107   if (callback == nullptr) return;
9108   i::Isolate* isolate = reinterpret_cast<i::Isolate*>(this);
9109   isolate->AddCallCompletedCallback(callback);
9110 }
9111 
RemoveCallCompletedCallback(CallCompletedCallback callback)9112 void Isolate::RemoveCallCompletedCallback(CallCompletedCallback callback) {
9113   i::Isolate* isolate = reinterpret_cast<i::Isolate*>(this);
9114   isolate->RemoveCallCompletedCallback(callback);
9115 }
9116 
Wake()9117 void Isolate::AtomicsWaitWakeHandle::Wake() {
9118   reinterpret_cast<i::AtomicsWaitWakeHandle*>(this)->Wake();
9119 }
9120 
SetAtomicsWaitCallback(AtomicsWaitCallback callback,void * data)9121 void Isolate::SetAtomicsWaitCallback(AtomicsWaitCallback callback, void* data) {
9122   i::Isolate* isolate = reinterpret_cast<i::Isolate*>(this);
9123   isolate->SetAtomicsWaitCallback(callback, data);
9124 }
9125 
SetPromiseHook(PromiseHook hook)9126 void Isolate::SetPromiseHook(PromiseHook hook) {
9127   i::Isolate* isolate = reinterpret_cast<i::Isolate*>(this);
9128   isolate->SetPromiseHook(hook);
9129 }
9130 
SetPromiseRejectCallback(PromiseRejectCallback callback)9131 void Isolate::SetPromiseRejectCallback(PromiseRejectCallback callback) {
9132   if (callback == nullptr) return;
9133   i::Isolate* isolate = reinterpret_cast<i::Isolate*>(this);
9134   isolate->SetPromiseRejectCallback(callback);
9135 }
9136 
PerformMicrotaskCheckpoint()9137 void Isolate::PerformMicrotaskCheckpoint() {
9138   DCHECK_NE(MicrotasksPolicy::kScoped, GetMicrotasksPolicy());
9139   i::Isolate* isolate = reinterpret_cast<i::Isolate*>(this);
9140   isolate->default_microtask_queue()->PerformCheckpoint(this);
9141 }
9142 
EnqueueMicrotask(Local<Function> v8_function)9143 void Isolate::EnqueueMicrotask(Local<Function> v8_function) {
9144   i::Isolate* isolate = reinterpret_cast<i::Isolate*>(this);
9145   i::Handle<i::JSReceiver> function = Utils::OpenHandle(*v8_function);
9146   i::Handle<i::NativeContext> handler_context;
9147   if (!i::JSReceiver::GetContextForMicrotask(function).ToHandle(
9148           &handler_context))
9149     handler_context = isolate->native_context();
9150   MicrotaskQueue* microtask_queue = handler_context->microtask_queue();
9151   if (microtask_queue) microtask_queue->EnqueueMicrotask(this, v8_function);
9152 }
9153 
EnqueueMicrotask(MicrotaskCallback callback,void * data)9154 void Isolate::EnqueueMicrotask(MicrotaskCallback callback, void* data) {
9155   i::Isolate* isolate = reinterpret_cast<i::Isolate*>(this);
9156   isolate->default_microtask_queue()->EnqueueMicrotask(this, callback, data);
9157 }
9158 
SetMicrotasksPolicy(MicrotasksPolicy policy)9159 void Isolate::SetMicrotasksPolicy(MicrotasksPolicy policy) {
9160   i::Isolate* isolate = reinterpret_cast<i::Isolate*>(this);
9161   isolate->default_microtask_queue()->set_microtasks_policy(policy);
9162 }
9163 
GetMicrotasksPolicy() const9164 MicrotasksPolicy Isolate::GetMicrotasksPolicy() const {
9165   i::Isolate* isolate =
9166       reinterpret_cast<i::Isolate*>(const_cast<Isolate*>(this));
9167   return isolate->default_microtask_queue()->microtasks_policy();
9168 }
9169 
AddMicrotasksCompletedCallback(MicrotasksCompletedCallbackWithData callback,void * data)9170 void Isolate::AddMicrotasksCompletedCallback(
9171     MicrotasksCompletedCallbackWithData callback, void* data) {
9172   DCHECK(callback);
9173   i::Isolate* isolate = reinterpret_cast<i::Isolate*>(this);
9174   isolate->default_microtask_queue()->AddMicrotasksCompletedCallback(callback,
9175                                                                      data);
9176 }
9177 
RemoveMicrotasksCompletedCallback(MicrotasksCompletedCallbackWithData callback,void * data)9178 void Isolate::RemoveMicrotasksCompletedCallback(
9179     MicrotasksCompletedCallbackWithData callback, void* data) {
9180   i::Isolate* isolate = reinterpret_cast<i::Isolate*>(this);
9181   isolate->default_microtask_queue()->RemoveMicrotasksCompletedCallback(
9182       callback, data);
9183 }
9184 
SetUseCounterCallback(UseCounterCallback callback)9185 void Isolate::SetUseCounterCallback(UseCounterCallback callback) {
9186   reinterpret_cast<i::Isolate*>(this)->SetUseCounterCallback(callback);
9187 }
9188 
SetCounterFunction(CounterLookupCallback callback)9189 void Isolate::SetCounterFunction(CounterLookupCallback callback) {
9190   i::Isolate* isolate = reinterpret_cast<i::Isolate*>(this);
9191   isolate->counters()->ResetCounterFunction(callback);
9192 }
9193 
SetCreateHistogramFunction(CreateHistogramCallback callback)9194 void Isolate::SetCreateHistogramFunction(CreateHistogramCallback callback) {
9195   i::Isolate* isolate = reinterpret_cast<i::Isolate*>(this);
9196   isolate->counters()->ResetCreateHistogramFunction(callback);
9197 }
9198 
SetAddHistogramSampleFunction(AddHistogramSampleCallback callback)9199 void Isolate::SetAddHistogramSampleFunction(
9200     AddHistogramSampleCallback callback) {
9201   reinterpret_cast<i::Isolate*>(this)
9202       ->counters()
9203       ->SetAddHistogramSampleFunction(callback);
9204 }
9205 
SetMetricsRecorder(const std::shared_ptr<metrics::Recorder> & metrics_recorder)9206 void Isolate::SetMetricsRecorder(
9207     const std::shared_ptr<metrics::Recorder>& metrics_recorder) {
9208   i::Isolate* isolate = reinterpret_cast<i::Isolate*>(this);
9209   isolate->metrics_recorder()->SetEmbedderRecorder(isolate, metrics_recorder);
9210 }
9211 
SetAddCrashKeyCallback(AddCrashKeyCallback callback)9212 void Isolate::SetAddCrashKeyCallback(AddCrashKeyCallback callback) {
9213   i::Isolate* isolate = reinterpret_cast<i::Isolate*>(this);
9214   isolate->SetAddCrashKeyCallback(callback);
9215 }
9216 
IdleNotificationDeadline(double deadline_in_seconds)9217 bool Isolate::IdleNotificationDeadline(double deadline_in_seconds) {
9218   // Returning true tells the caller that it need not
9219   // continue to call IdleNotification.
9220   i::Isolate* isolate = reinterpret_cast<i::Isolate*>(this);
9221   if (!i::FLAG_use_idle_notification) return true;
9222   return isolate->heap()->IdleNotification(deadline_in_seconds);
9223 }
9224 
LowMemoryNotification()9225 void Isolate::LowMemoryNotification() {
9226   i::Isolate* isolate = reinterpret_cast<i::Isolate*>(this);
9227   {
9228     i::NestedTimedHistogramScope idle_notification_scope(
9229         isolate->counters()->gc_low_memory_notification());
9230     TRACE_EVENT0("v8", "V8.GCLowMemoryNotification");
9231     isolate->heap()->CollectAllAvailableGarbage(
9232         i::GarbageCollectionReason::kLowMemoryNotification);
9233   }
9234 }
9235 
ContextDisposedNotification(bool dependant_context)9236 int Isolate::ContextDisposedNotification(bool dependant_context) {
9237   i::Isolate* isolate = reinterpret_cast<i::Isolate*>(this);
9238 #if V8_ENABLE_WEBASSEMBLY
9239   if (!dependant_context) {
9240     if (!isolate->context().is_null()) {
9241       // We left the current context, we can abort all WebAssembly compilations
9242       // of that context.
9243       // A handle scope for the native context.
9244       i::HandleScope handle_scope(isolate);
9245       i::wasm::GetWasmEngine()->DeleteCompileJobsOnContext(
9246           isolate->native_context());
9247     }
9248   }
9249 #endif  // V8_ENABLE_WEBASSEMBLY
9250   // TODO(ahaas): move other non-heap activity out of the heap call.
9251   return isolate->heap()->NotifyContextDisposed(dependant_context);
9252 }
9253 
IsolateInForegroundNotification()9254 void Isolate::IsolateInForegroundNotification() {
9255   i::Isolate* isolate = reinterpret_cast<i::Isolate*>(this);
9256   return isolate->IsolateInForegroundNotification();
9257 }
9258 
IsolateInBackgroundNotification()9259 void Isolate::IsolateInBackgroundNotification() {
9260   i::Isolate* isolate = reinterpret_cast<i::Isolate*>(this);
9261   return isolate->IsolateInBackgroundNotification();
9262 }
9263 
MemoryPressureNotification(MemoryPressureLevel level)9264 void Isolate::MemoryPressureNotification(MemoryPressureLevel level) {
9265   i::Isolate* isolate = reinterpret_cast<i::Isolate*>(this);
9266   bool on_isolate_thread =
9267       isolate->was_locker_ever_used()
9268           ? isolate->thread_manager()->IsLockedByCurrentThread()
9269           : i::ThreadId::Current() == isolate->thread_id();
9270   isolate->heap()->MemoryPressureNotification(level, on_isolate_thread);
9271 }
9272 
ClearCachesForTesting()9273 void Isolate::ClearCachesForTesting() {
9274   i::Isolate* isolate = reinterpret_cast<i::Isolate*>(this);
9275   isolate->AbortConcurrentOptimization(i::BlockingBehavior::kBlock);
9276   isolate->ClearSerializerData();
9277 }
9278 
EnableMemorySavingsMode()9279 void Isolate::EnableMemorySavingsMode() {
9280   i::Isolate* isolate = reinterpret_cast<i::Isolate*>(this);
9281   isolate->EnableMemorySavingsMode();
9282 }
9283 
DisableMemorySavingsMode()9284 void Isolate::DisableMemorySavingsMode() {
9285   i::Isolate* isolate = reinterpret_cast<i::Isolate*>(this);
9286   isolate->DisableMemorySavingsMode();
9287 }
9288 
SetRAILMode(RAILMode rail_mode)9289 void Isolate::SetRAILMode(RAILMode rail_mode) {
9290   i::Isolate* isolate = reinterpret_cast<i::Isolate*>(this);
9291   return isolate->SetRAILMode(rail_mode);
9292 }
9293 
UpdateLoadStartTime()9294 void Isolate::UpdateLoadStartTime() {
9295   i::Isolate* isolate = reinterpret_cast<i::Isolate*>(this);
9296   isolate->UpdateLoadStartTime();
9297 }
9298 
IncreaseHeapLimitForDebugging()9299 void Isolate::IncreaseHeapLimitForDebugging() {
9300   // No-op.
9301 }
9302 
RestoreOriginalHeapLimit()9303 void Isolate::RestoreOriginalHeapLimit() {
9304   // No-op.
9305 }
9306 
IsHeapLimitIncreasedForDebugging()9307 bool Isolate::IsHeapLimitIncreasedForDebugging() { return false; }
9308 
SetJitCodeEventHandler(JitCodeEventOptions options,JitCodeEventHandler event_handler)9309 void Isolate::SetJitCodeEventHandler(JitCodeEventOptions options,
9310                                      JitCodeEventHandler event_handler) {
9311   i::Isolate* isolate = reinterpret_cast<i::Isolate*>(this);
9312   // Ensure that logging is initialized for our isolate.
9313   isolate->InitializeLoggingAndCounters();
9314   isolate->logger()->SetCodeEventHandler(options, event_handler);
9315 }
9316 
SetStackLimit(uintptr_t stack_limit)9317 void Isolate::SetStackLimit(uintptr_t stack_limit) {
9318   i::Isolate* isolate = reinterpret_cast<i::Isolate*>(this);
9319   CHECK(stack_limit);
9320   isolate->stack_guard()->SetStackLimit(stack_limit);
9321 }
9322 
GetCodeRange(void ** start,size_t * length_in_bytes)9323 void Isolate::GetCodeRange(void** start, size_t* length_in_bytes) {
9324   i::Isolate* isolate = reinterpret_cast<i::Isolate*>(this);
9325   const base::AddressRegion& code_region = isolate->heap()->code_region();
9326   *start = reinterpret_cast<void*>(code_region.begin());
9327   *length_in_bytes = code_region.size();
9328 }
9329 
GetEmbeddedCodeRange(const void ** start,size_t * length_in_bytes)9330 void Isolate::GetEmbeddedCodeRange(const void** start,
9331                                    size_t* length_in_bytes) {
9332   // Note, we should return the embedded code rande from the .text section here.
9333   i::EmbeddedData d = i::EmbeddedData::FromBlob();
9334   *start = reinterpret_cast<const void*>(d.code());
9335   *length_in_bytes = d.code_size();
9336 }
9337 
GetJSEntryStubs()9338 JSEntryStubs Isolate::GetJSEntryStubs() {
9339   JSEntryStubs entry_stubs;
9340 
9341   i::Isolate* isolate = reinterpret_cast<i::Isolate*>(this);
9342   std::array<std::pair<i::Builtin, JSEntryStub*>, 3> stubs = {
9343       {{i::Builtin::kJSEntry, &entry_stubs.js_entry_stub},
9344        {i::Builtin::kJSConstructEntry, &entry_stubs.js_construct_entry_stub},
9345        {i::Builtin::kJSRunMicrotasksEntry,
9346         &entry_stubs.js_run_microtasks_entry_stub}}};
9347   for (auto& pair : stubs) {
9348     i::Code js_entry = FromCodeT(isolate->builtins()->code(pair.first));
9349     pair.second->code.start =
9350         reinterpret_cast<const void*>(js_entry.InstructionStart());
9351     pair.second->code.length_in_bytes = js_entry.InstructionSize();
9352   }
9353 
9354   return entry_stubs;
9355 }
9356 
CopyCodePages(size_t capacity,MemoryRange * code_pages_out)9357 size_t Isolate::CopyCodePages(size_t capacity, MemoryRange* code_pages_out) {
9358 #if !defined(V8_TARGET_ARCH_64_BIT) && !defined(V8_TARGET_ARCH_ARM)
9359   // Not implemented on other platforms.
9360   UNREACHABLE();
9361 #else
9362 
9363   i::Isolate* isolate = reinterpret_cast<i::Isolate*>(this);
9364   std::vector<MemoryRange>* code_pages = isolate->GetCodePages();
9365 
9366   DCHECK_NOT_NULL(code_pages);
9367 
9368   // Copy as many elements into the output vector as we can. If the
9369   // caller-provided buffer is not big enough, we fill it, and the caller can
9370   // provide a bigger one next time. We do it this way because allocation is not
9371   // allowed in signal handlers.
9372   size_t limit = std::min(capacity, code_pages->size());
9373   for (size_t i = 0; i < limit; i++) {
9374     code_pages_out[i] = code_pages->at(i);
9375   }
9376   return code_pages->size();
9377 #endif
9378 }
9379 
9380 #define CALLBACK_SETTER(ExternalName, Type, InternalName)      \
9381   void Isolate::Set##ExternalName(Type callback) {             \
9382     i::Isolate* isolate = reinterpret_cast<i::Isolate*>(this); \
9383     isolate->set_##InternalName(callback);                     \
9384   }
9385 
CALLBACK_SETTER(FatalErrorHandler,FatalErrorCallback,exception_behavior)9386 CALLBACK_SETTER(FatalErrorHandler, FatalErrorCallback, exception_behavior)
9387 CALLBACK_SETTER(OOMErrorHandler, OOMErrorCallback, oom_behavior)
9388 CALLBACK_SETTER(ModifyCodeGenerationFromStringsCallback,
9389                 ModifyCodeGenerationFromStringsCallback2,
9390                 modify_code_gen_callback2)
9391 CALLBACK_SETTER(AllowWasmCodeGenerationCallback,
9392                 AllowWasmCodeGenerationCallback, allow_wasm_code_gen_callback)
9393 
9394 CALLBACK_SETTER(WasmModuleCallback, ExtensionCallback, wasm_module_callback)
9395 CALLBACK_SETTER(WasmInstanceCallback, ExtensionCallback, wasm_instance_callback)
9396 
9397 CALLBACK_SETTER(WasmStreamingCallback, WasmStreamingCallback,
9398                 wasm_streaming_callback)
9399 
9400 CALLBACK_SETTER(WasmLoadSourceMapCallback, WasmLoadSourceMapCallback,
9401                 wasm_load_source_map_callback)
9402 
9403 CALLBACK_SETTER(WasmSimdEnabledCallback, WasmSimdEnabledCallback,
9404                 wasm_simd_enabled_callback)
9405 
9406 CALLBACK_SETTER(WasmExceptionsEnabledCallback, WasmExceptionsEnabledCallback,
9407                 wasm_exceptions_enabled_callback)
9408 
9409 CALLBACK_SETTER(WasmDynamicTieringEnabledCallback,
9410                 WasmDynamicTieringEnabledCallback,
9411                 wasm_dynamic_tiering_enabled_callback)
9412 
9413 CALLBACK_SETTER(SharedArrayBufferConstructorEnabledCallback,
9414                 SharedArrayBufferConstructorEnabledCallback,
9415                 sharedarraybuffer_constructor_enabled_callback)
9416 
9417 void Isolate::InstallConditionalFeatures(Local<Context> context) {
9418   v8::HandleScope handle_scope(this);
9419   v8::Context::Scope context_scope(context);
9420   i::Isolate* isolate = reinterpret_cast<i::Isolate*>(this);
9421   isolate->InstallConditionalFeatures(Utils::OpenHandle(*context));
9422 #if V8_ENABLE_WEBASSEMBLY
9423   if (i::FLAG_expose_wasm) {
9424     i::WasmJs::InstallConditionalFeatures(isolate, Utils::OpenHandle(*context));
9425   }
9426 #endif  // V8_ENABLE_WEBASSEMBLY
9427 }
9428 
AddNearHeapLimitCallback(v8::NearHeapLimitCallback callback,void * data)9429 void Isolate::AddNearHeapLimitCallback(v8::NearHeapLimitCallback callback,
9430                                        void* data) {
9431   i::Isolate* isolate = reinterpret_cast<i::Isolate*>(this);
9432   isolate->heap()->AddNearHeapLimitCallback(callback, data);
9433 }
9434 
RemoveNearHeapLimitCallback(v8::NearHeapLimitCallback callback,size_t heap_limit)9435 void Isolate::RemoveNearHeapLimitCallback(v8::NearHeapLimitCallback callback,
9436                                           size_t heap_limit) {
9437   i::Isolate* isolate = reinterpret_cast<i::Isolate*>(this);
9438   isolate->heap()->RemoveNearHeapLimitCallback(callback, heap_limit);
9439 }
9440 
AutomaticallyRestoreInitialHeapLimit(double threshold_percent)9441 void Isolate::AutomaticallyRestoreInitialHeapLimit(double threshold_percent) {
9442   DCHECK_GT(threshold_percent, 0.0);
9443   DCHECK_LT(threshold_percent, 1.0);
9444   i::Isolate* isolate = reinterpret_cast<i::Isolate*>(this);
9445   isolate->heap()->AutomaticallyRestoreInitialHeapLimit(threshold_percent);
9446 }
9447 
IsDead()9448 bool Isolate::IsDead() {
9449   i::Isolate* isolate = reinterpret_cast<i::Isolate*>(this);
9450   return isolate->IsDead();
9451 }
9452 
AddMessageListener(MessageCallback that,Local<Value> data)9453 bool Isolate::AddMessageListener(MessageCallback that, Local<Value> data) {
9454   return AddMessageListenerWithErrorLevel(that, kMessageError, data);
9455 }
9456 
AddMessageListenerWithErrorLevel(MessageCallback that,int message_levels,Local<Value> data)9457 bool Isolate::AddMessageListenerWithErrorLevel(MessageCallback that,
9458                                                int message_levels,
9459                                                Local<Value> data) {
9460   i::Isolate* isolate = reinterpret_cast<i::Isolate*>(this);
9461   ENTER_V8_NO_SCRIPT_NO_EXCEPTION(isolate);
9462   i::HandleScope scope(isolate);
9463   i::Handle<i::TemplateList> list = isolate->factory()->message_listeners();
9464   i::Handle<i::FixedArray> listener = isolate->factory()->NewFixedArray(3);
9465   i::Handle<i::Foreign> foreign =
9466       isolate->factory()->NewForeign(FUNCTION_ADDR(that));
9467   listener->set(0, *foreign);
9468   listener->set(1, data.IsEmpty() ? i::ReadOnlyRoots(isolate).undefined_value()
9469                                   : *Utils::OpenHandle(*data));
9470   listener->set(2, i::Smi::FromInt(message_levels));
9471   list = i::TemplateList::Add(isolate, list, listener);
9472   isolate->heap()->SetMessageListeners(*list);
9473   return true;
9474 }
9475 
RemoveMessageListeners(MessageCallback that)9476 void Isolate::RemoveMessageListeners(MessageCallback that) {
9477   i::Isolate* isolate = reinterpret_cast<i::Isolate*>(this);
9478   ENTER_V8_NO_SCRIPT_NO_EXCEPTION(isolate);
9479   i::HandleScope scope(isolate);
9480   i::DisallowGarbageCollection no_gc;
9481   i::TemplateList listeners = isolate->heap()->message_listeners();
9482   for (int i = 0; i < listeners.length(); i++) {
9483     if (listeners.get(i).IsUndefined(isolate)) continue;  // skip deleted ones
9484     i::FixedArray listener = i::FixedArray::cast(listeners.get(i));
9485     i::Foreign callback_obj = i::Foreign::cast(listener.get(0));
9486     if (callback_obj.foreign_address() == FUNCTION_ADDR(that)) {
9487       listeners.set(i, i::ReadOnlyRoots(isolate).undefined_value());
9488     }
9489   }
9490 }
9491 
SetFailedAccessCheckCallbackFunction(FailedAccessCheckCallback callback)9492 void Isolate::SetFailedAccessCheckCallbackFunction(
9493     FailedAccessCheckCallback callback) {
9494   i::Isolate* isolate = reinterpret_cast<i::Isolate*>(this);
9495   isolate->SetFailedAccessCheckCallback(callback);
9496 }
9497 
SetCaptureStackTraceForUncaughtExceptions(bool capture,int frame_limit,StackTrace::StackTraceOptions options)9498 void Isolate::SetCaptureStackTraceForUncaughtExceptions(
9499     bool capture, int frame_limit, StackTrace::StackTraceOptions options) {
9500   i::Isolate* isolate = reinterpret_cast<i::Isolate*>(this);
9501   isolate->SetCaptureStackTraceForUncaughtExceptions(capture, frame_limit,
9502                                                      options);
9503 }
9504 
VisitExternalResources(ExternalResourceVisitor * visitor)9505 void Isolate::VisitExternalResources(ExternalResourceVisitor* visitor) {
9506   i::Isolate* isolate = reinterpret_cast<i::Isolate*>(this);
9507   isolate->heap()->VisitExternalResources(visitor);
9508 }
9509 
IsInUse()9510 bool Isolate::IsInUse() {
9511   i::Isolate* isolate = reinterpret_cast<i::Isolate*>(this);
9512   return isolate->IsInUse();
9513 }
9514 
VisitHandlesWithClassIds(PersistentHandleVisitor * visitor)9515 void Isolate::VisitHandlesWithClassIds(PersistentHandleVisitor* visitor) {
9516   i::Isolate* isolate = reinterpret_cast<i::Isolate*>(this);
9517   i::DisallowGarbageCollection no_gc;
9518   isolate->global_handles()->IterateAllRootsWithClassIds(visitor);
9519 }
9520 
VisitWeakHandles(PersistentHandleVisitor * visitor)9521 void Isolate::VisitWeakHandles(PersistentHandleVisitor* visitor) {
9522   i::Isolate* isolate = reinterpret_cast<i::Isolate*>(this);
9523   i::DisallowGarbageCollection no_gc;
9524   isolate->global_handles()->IterateYoungWeakRootsWithClassIds(visitor);
9525 }
9526 
SetAllowAtomicsWait(bool allow)9527 void Isolate::SetAllowAtomicsWait(bool allow) {
9528   i::Isolate* isolate = reinterpret_cast<i::Isolate*>(this);
9529   isolate->set_allow_atomics_wait(allow);
9530 }
9531 
DateTimeConfigurationChangeNotification(TimeZoneDetection time_zone_detection)9532 void v8::Isolate::DateTimeConfigurationChangeNotification(
9533     TimeZoneDetection time_zone_detection) {
9534   i::Isolate* i_isolate = reinterpret_cast<i::Isolate*>(this);
9535   API_RCS_SCOPE(i_isolate, Isolate, DateTimeConfigurationChangeNotification);
9536   ENTER_V8_NO_SCRIPT_NO_EXCEPTION(i_isolate);
9537   i_isolate->date_cache()->ResetDateCache(
9538       static_cast<base::TimezoneCache::TimeZoneDetection>(time_zone_detection));
9539 #ifdef V8_INTL_SUPPORT
9540   i_isolate->clear_cached_icu_object(
9541       i::Isolate::ICUObjectCacheType::kDefaultSimpleDateFormat);
9542   i_isolate->clear_cached_icu_object(
9543       i::Isolate::ICUObjectCacheType::kDefaultSimpleDateFormatForTime);
9544   i_isolate->clear_cached_icu_object(
9545       i::Isolate::ICUObjectCacheType::kDefaultSimpleDateFormatForDate);
9546 #endif  // V8_INTL_SUPPORT
9547 }
9548 
LocaleConfigurationChangeNotification()9549 void v8::Isolate::LocaleConfigurationChangeNotification() {
9550   i::Isolate* i_isolate = reinterpret_cast<i::Isolate*>(this);
9551   API_RCS_SCOPE(i_isolate, Isolate, LocaleConfigurationChangeNotification);
9552   ENTER_V8_NO_SCRIPT_NO_EXCEPTION(i_isolate);
9553 
9554 #ifdef V8_INTL_SUPPORT
9555   i_isolate->ResetDefaultLocale();
9556 #endif  // V8_INTL_SUPPORT
9557 }
9558 
IsCodeLike(v8::Isolate * isolate) const9559 bool v8::Object::IsCodeLike(v8::Isolate* isolate) const {
9560   i::Isolate* i_isolate = reinterpret_cast<i::Isolate*>(isolate);
9561   API_RCS_SCOPE(i_isolate, Object, IsCodeLike);
9562   ENTER_V8_NO_SCRIPT_NO_EXCEPTION(i_isolate);
9563   i::HandleScope scope(i_isolate);
9564   return Utils::OpenHandle(this)->IsCodeLike(i_isolate);
9565 }
9566 
9567 // static
New(Isolate * isolate,MicrotasksPolicy policy)9568 std::unique_ptr<MicrotaskQueue> MicrotaskQueue::New(Isolate* isolate,
9569                                                     MicrotasksPolicy policy) {
9570   auto microtask_queue =
9571       i::MicrotaskQueue::New(reinterpret_cast<i::Isolate*>(isolate));
9572   microtask_queue->set_microtasks_policy(policy);
9573   std::unique_ptr<MicrotaskQueue> ret(std::move(microtask_queue));
9574   return ret;
9575 }
9576 
MicrotasksScope(Isolate * isolate,MicrotasksScope::Type type)9577 MicrotasksScope::MicrotasksScope(Isolate* isolate, MicrotasksScope::Type type)
9578     : MicrotasksScope(isolate, nullptr, type) {}
9579 
MicrotasksScope(Isolate * isolate,MicrotaskQueue * microtask_queue,MicrotasksScope::Type type)9580 MicrotasksScope::MicrotasksScope(Isolate* isolate,
9581                                  MicrotaskQueue* microtask_queue,
9582                                  MicrotasksScope::Type type)
9583     : isolate_(reinterpret_cast<i::Isolate*>(isolate)),
9584       microtask_queue_(microtask_queue
9585                            ? static_cast<i::MicrotaskQueue*>(microtask_queue)
9586                            : isolate_->default_microtask_queue()),
9587       run_(type == MicrotasksScope::kRunMicrotasks) {
9588   if (run_) microtask_queue_->IncrementMicrotasksScopeDepth();
9589 #ifdef DEBUG
9590   if (!run_) microtask_queue_->IncrementDebugMicrotasksScopeDepth();
9591 #endif
9592 }
9593 
~MicrotasksScope()9594 MicrotasksScope::~MicrotasksScope() {
9595   if (run_) {
9596     microtask_queue_->DecrementMicrotasksScopeDepth();
9597     if (MicrotasksPolicy::kScoped == microtask_queue_->microtasks_policy() &&
9598         !isolate_->has_scheduled_exception()) {
9599       DCHECK_IMPLIES(isolate_->has_scheduled_exception(),
9600                      isolate_->scheduled_exception() ==
9601                          i::ReadOnlyRoots(isolate_).termination_exception());
9602       microtask_queue_->PerformCheckpoint(reinterpret_cast<Isolate*>(isolate_));
9603     }
9604   }
9605 #ifdef DEBUG
9606   if (!run_) microtask_queue_->DecrementDebugMicrotasksScopeDepth();
9607 #endif
9608 }
9609 
9610 // static
PerformCheckpoint(Isolate * v8_isolate)9611 void MicrotasksScope::PerformCheckpoint(Isolate* v8_isolate) {
9612   i::Isolate* isolate = reinterpret_cast<i::Isolate*>(v8_isolate);
9613   auto* microtask_queue = isolate->default_microtask_queue();
9614   microtask_queue->PerformCheckpoint(v8_isolate);
9615 }
9616 
9617 // static
GetCurrentDepth(Isolate * v8_isolate)9618 int MicrotasksScope::GetCurrentDepth(Isolate* v8_isolate) {
9619   i::Isolate* isolate = reinterpret_cast<i::Isolate*>(v8_isolate);
9620   auto* microtask_queue = isolate->default_microtask_queue();
9621   return microtask_queue->GetMicrotasksScopeDepth();
9622 }
9623 
9624 // static
IsRunningMicrotasks(Isolate * v8_isolate)9625 bool MicrotasksScope::IsRunningMicrotasks(Isolate* v8_isolate) {
9626   i::Isolate* isolate = reinterpret_cast<i::Isolate*>(v8_isolate);
9627   auto* microtask_queue = isolate->default_microtask_queue();
9628   return microtask_queue->IsRunningMicrotasks();
9629 }
9630 
Utf8Value(v8::Isolate * isolate,v8::Local<v8::Value> obj)9631 String::Utf8Value::Utf8Value(v8::Isolate* isolate, v8::Local<v8::Value> obj)
9632     : str_(nullptr), length_(0) {
9633   if (obj.IsEmpty()) return;
9634   i::Isolate* i_isolate = reinterpret_cast<i::Isolate*>(isolate);
9635   ENTER_V8_DO_NOT_USE(i_isolate);
9636   i::HandleScope scope(i_isolate);
9637   Local<Context> context = isolate->GetCurrentContext();
9638   TryCatch try_catch(isolate);
9639   Local<String> str;
9640   if (!obj->ToString(context).ToLocal(&str)) return;
9641   length_ = str->Utf8Length(isolate);
9642   str_ = i::NewArray<char>(length_ + 1);
9643   str->WriteUtf8(isolate, str_);
9644 }
9645 
~Utf8Value()9646 String::Utf8Value::~Utf8Value() { i::DeleteArray(str_); }
9647 
Value(v8::Isolate * isolate,v8::Local<v8::Value> obj)9648 String::Value::Value(v8::Isolate* isolate, v8::Local<v8::Value> obj)
9649     : str_(nullptr), length_(0) {
9650   if (obj.IsEmpty()) return;
9651   i::Isolate* i_isolate = reinterpret_cast<i::Isolate*>(isolate);
9652   ENTER_V8_DO_NOT_USE(i_isolate);
9653   i::HandleScope scope(i_isolate);
9654   Local<Context> context = isolate->GetCurrentContext();
9655   TryCatch try_catch(isolate);
9656   Local<String> str;
9657   if (!obj->ToString(context).ToLocal(&str)) return;
9658   length_ = str->Length();
9659   str_ = i::NewArray<uint16_t>(length_ + 1);
9660   str->Write(isolate, str_);
9661 }
9662 
~Value()9663 String::Value::~Value() { i::DeleteArray(str_); }
9664 
9665 #define DEFINE_ERROR(NAME, name)                                         \
9666   Local<Value> Exception::NAME(v8::Local<v8::String> raw_message) {      \
9667     i::Isolate* isolate = i::Isolate::Current();                         \
9668     API_RCS_SCOPE(isolate, NAME, New);                                   \
9669     ENTER_V8_NO_SCRIPT_NO_EXCEPTION(isolate);                            \
9670     i::Object error;                                                     \
9671     {                                                                    \
9672       i::HandleScope scope(isolate);                                     \
9673       i::Handle<i::String> message = Utils::OpenHandle(*raw_message);    \
9674       i::Handle<i::JSFunction> constructor = isolate->name##_function(); \
9675       error = *isolate->factory()->NewError(constructor, message);       \
9676     }                                                                    \
9677     i::Handle<i::Object> result(error, isolate);                         \
9678     return Utils::ToLocal(result);                                       \
9679   }
9680 
DEFINE_ERROR(RangeError,range_error)9681 DEFINE_ERROR(RangeError, range_error)
9682 DEFINE_ERROR(ReferenceError, reference_error)
9683 DEFINE_ERROR(SyntaxError, syntax_error)
9684 DEFINE_ERROR(TypeError, type_error)
9685 DEFINE_ERROR(WasmCompileError, wasm_compile_error)
9686 DEFINE_ERROR(WasmLinkError, wasm_link_error)
9687 DEFINE_ERROR(WasmRuntimeError, wasm_runtime_error)
9688 DEFINE_ERROR(Error, error)
9689 
9690 #undef DEFINE_ERROR
9691 
9692 Local<Message> Exception::CreateMessage(Isolate* isolate,
9693                                         Local<Value> exception) {
9694   i::Handle<i::Object> obj = Utils::OpenHandle(*exception);
9695   i::Isolate* i_isolate = reinterpret_cast<i::Isolate*>(isolate);
9696   ENTER_V8_NO_SCRIPT_NO_EXCEPTION(i_isolate);
9697   i::HandleScope scope(i_isolate);
9698   return Utils::MessageToLocal(
9699       scope.CloseAndEscape(i_isolate->CreateMessage(obj, nullptr)));
9700 }
9701 
GetStackTrace(Local<Value> exception)9702 Local<StackTrace> Exception::GetStackTrace(Local<Value> exception) {
9703   i::Handle<i::Object> obj = Utils::OpenHandle(*exception);
9704   if (!obj->IsJSObject()) return Local<StackTrace>();
9705   i::Handle<i::JSObject> js_obj = i::Handle<i::JSObject>::cast(obj);
9706   i::Isolate* isolate = js_obj->GetIsolate();
9707   ENTER_V8_NO_SCRIPT_NO_EXCEPTION(isolate);
9708   return Utils::StackTraceToLocal(isolate->GetDetailedStackTrace(js_obj));
9709 }
9710 
PreviewEntries(bool * is_key_value)9711 v8::MaybeLocal<v8::Array> v8::Object::PreviewEntries(bool* is_key_value) {
9712   if (IsMap()) {
9713     *is_key_value = true;
9714     return Map::Cast(this)->AsArray();
9715   }
9716   if (IsSet()) {
9717     *is_key_value = false;
9718     return Set::Cast(this)->AsArray();
9719   }
9720 
9721   i::Handle<i::JSReceiver> object = Utils::OpenHandle(this);
9722   i::Isolate* isolate = object->GetIsolate();
9723   Isolate* v8_isolate = reinterpret_cast<Isolate*>(isolate);
9724   ENTER_V8_NO_SCRIPT_NO_EXCEPTION(isolate);
9725   if (object->IsJSWeakCollection()) {
9726     *is_key_value = object->IsJSWeakMap();
9727     return Utils::ToLocal(i::JSWeakCollection::GetEntries(
9728         i::Handle<i::JSWeakCollection>::cast(object), 0));
9729   }
9730   if (object->IsJSMapIterator()) {
9731     i::Handle<i::JSMapIterator> it = i::Handle<i::JSMapIterator>::cast(object);
9732     MapAsArrayKind const kind =
9733         static_cast<MapAsArrayKind>(it->map().instance_type());
9734     *is_key_value = kind == MapAsArrayKind::kEntries;
9735     if (!it->HasMore()) return v8::Array::New(v8_isolate);
9736     return Utils::ToLocal(
9737         MapAsArray(isolate, it->table(), i::Smi::ToInt(it->index()), kind));
9738   }
9739   if (object->IsJSSetIterator()) {
9740     i::Handle<i::JSSetIterator> it = i::Handle<i::JSSetIterator>::cast(object);
9741     SetAsArrayKind const kind =
9742         static_cast<SetAsArrayKind>(it->map().instance_type());
9743     *is_key_value = kind == SetAsArrayKind::kEntries;
9744     if (!it->HasMore()) return v8::Array::New(v8_isolate);
9745     return Utils::ToLocal(
9746         SetAsArray(isolate, it->table(), i::Smi::ToInt(it->index()), kind));
9747   }
9748   return v8::MaybeLocal<v8::Array>();
9749 }
9750 
GetFunctionName() const9751 Local<String> CpuProfileNode::GetFunctionName() const {
9752   const i::ProfileNode* node = reinterpret_cast<const i::ProfileNode*>(this);
9753   i::Isolate* isolate = node->isolate();
9754   const i::CodeEntry* entry = node->entry();
9755   i::Handle<i::String> name =
9756       isolate->factory()->InternalizeUtf8String(entry->name());
9757   return ToApiHandle<String>(name);
9758 }
9759 
GetFunctionNameStr() const9760 const char* CpuProfileNode::GetFunctionNameStr() const {
9761   const i::ProfileNode* node = reinterpret_cast<const i::ProfileNode*>(this);
9762   return node->entry()->name();
9763 }
9764 
GetScriptId() const9765 int CpuProfileNode::GetScriptId() const {
9766   const i::ProfileNode* node = reinterpret_cast<const i::ProfileNode*>(this);
9767   const i::CodeEntry* entry = node->entry();
9768   return entry->script_id();
9769 }
9770 
GetScriptResourceName() const9771 Local<String> CpuProfileNode::GetScriptResourceName() const {
9772   const i::ProfileNode* node = reinterpret_cast<const i::ProfileNode*>(this);
9773   i::Isolate* isolate = node->isolate();
9774   return ToApiHandle<String>(isolate->factory()->InternalizeUtf8String(
9775       node->entry()->resource_name()));
9776 }
9777 
GetScriptResourceNameStr() const9778 const char* CpuProfileNode::GetScriptResourceNameStr() const {
9779   const i::ProfileNode* node = reinterpret_cast<const i::ProfileNode*>(this);
9780   return node->entry()->resource_name();
9781 }
9782 
IsScriptSharedCrossOrigin() const9783 bool CpuProfileNode::IsScriptSharedCrossOrigin() const {
9784   const i::ProfileNode* node = reinterpret_cast<const i::ProfileNode*>(this);
9785   return node->entry()->is_shared_cross_origin();
9786 }
9787 
GetLineNumber() const9788 int CpuProfileNode::GetLineNumber() const {
9789   return reinterpret_cast<const i::ProfileNode*>(this)->line_number();
9790 }
9791 
GetColumnNumber() const9792 int CpuProfileNode::GetColumnNumber() const {
9793   return reinterpret_cast<const i::ProfileNode*>(this)
9794       ->entry()
9795       ->column_number();
9796 }
9797 
GetHitLineCount() const9798 unsigned int CpuProfileNode::GetHitLineCount() const {
9799   const i::ProfileNode* node = reinterpret_cast<const i::ProfileNode*>(this);
9800   return node->GetHitLineCount();
9801 }
9802 
GetLineTicks(LineTick * entries,unsigned int length) const9803 bool CpuProfileNode::GetLineTicks(LineTick* entries,
9804                                   unsigned int length) const {
9805   const i::ProfileNode* node = reinterpret_cast<const i::ProfileNode*>(this);
9806   return node->GetLineTicks(entries, length);
9807 }
9808 
GetBailoutReason() const9809 const char* CpuProfileNode::GetBailoutReason() const {
9810   const i::ProfileNode* node = reinterpret_cast<const i::ProfileNode*>(this);
9811   return node->entry()->bailout_reason();
9812 }
9813 
GetHitCount() const9814 unsigned CpuProfileNode::GetHitCount() const {
9815   return reinterpret_cast<const i::ProfileNode*>(this)->self_ticks();
9816 }
9817 
GetNodeId() const9818 unsigned CpuProfileNode::GetNodeId() const {
9819   return reinterpret_cast<const i::ProfileNode*>(this)->id();
9820 }
9821 
GetSourceType() const9822 CpuProfileNode::SourceType CpuProfileNode::GetSourceType() const {
9823   return reinterpret_cast<const i::ProfileNode*>(this)->source_type();
9824 }
9825 
GetChildrenCount() const9826 int CpuProfileNode::GetChildrenCount() const {
9827   return static_cast<int>(
9828       reinterpret_cast<const i::ProfileNode*>(this)->children()->size());
9829 }
9830 
GetChild(int index) const9831 const CpuProfileNode* CpuProfileNode::GetChild(int index) const {
9832   const i::ProfileNode* child =
9833       reinterpret_cast<const i::ProfileNode*>(this)->children()->at(index);
9834   return reinterpret_cast<const CpuProfileNode*>(child);
9835 }
9836 
GetParent() const9837 const CpuProfileNode* CpuProfileNode::GetParent() const {
9838   const i::ProfileNode* parent =
9839       reinterpret_cast<const i::ProfileNode*>(this)->parent();
9840   return reinterpret_cast<const CpuProfileNode*>(parent);
9841 }
9842 
GetDeoptInfos() const9843 const std::vector<CpuProfileDeoptInfo>& CpuProfileNode::GetDeoptInfos() const {
9844   const i::ProfileNode* node = reinterpret_cast<const i::ProfileNode*>(this);
9845   return node->deopt_infos();
9846 }
9847 
Delete()9848 void CpuProfile::Delete() {
9849   i::CpuProfile* profile = reinterpret_cast<i::CpuProfile*>(this);
9850   i::CpuProfiler* profiler = profile->cpu_profiler();
9851   DCHECK_NOT_NULL(profiler);
9852   profiler->DeleteProfile(profile);
9853 }
9854 
GetTitle() const9855 Local<String> CpuProfile::GetTitle() const {
9856   const i::CpuProfile* profile = reinterpret_cast<const i::CpuProfile*>(this);
9857   i::Isolate* isolate = profile->top_down()->isolate();
9858   return ToApiHandle<String>(
9859       isolate->factory()->InternalizeUtf8String(profile->title()));
9860 }
9861 
GetTopDownRoot() const9862 const CpuProfileNode* CpuProfile::GetTopDownRoot() const {
9863   const i::CpuProfile* profile = reinterpret_cast<const i::CpuProfile*>(this);
9864   return reinterpret_cast<const CpuProfileNode*>(profile->top_down()->root());
9865 }
9866 
GetSample(int index) const9867 const CpuProfileNode* CpuProfile::GetSample(int index) const {
9868   const i::CpuProfile* profile = reinterpret_cast<const i::CpuProfile*>(this);
9869   return reinterpret_cast<const CpuProfileNode*>(profile->sample(index).node);
9870 }
9871 
9872 const int CpuProfileNode::kNoLineNumberInfo;
9873 const int CpuProfileNode::kNoColumnNumberInfo;
9874 
GetSampleTimestamp(int index) const9875 int64_t CpuProfile::GetSampleTimestamp(int index) const {
9876   const i::CpuProfile* profile = reinterpret_cast<const i::CpuProfile*>(this);
9877   return profile->sample(index).timestamp.since_origin().InMicroseconds();
9878 }
9879 
GetSampleState(int index) const9880 StateTag CpuProfile::GetSampleState(int index) const {
9881   const i::CpuProfile* profile = reinterpret_cast<const i::CpuProfile*>(this);
9882   return profile->sample(index).state_tag;
9883 }
9884 
GetSampleEmbedderState(int index) const9885 EmbedderStateTag CpuProfile::GetSampleEmbedderState(int index) const {
9886   const i::CpuProfile* profile = reinterpret_cast<const i::CpuProfile*>(this);
9887   return profile->sample(index).embedder_state_tag;
9888 }
9889 
GetStartTime() const9890 int64_t CpuProfile::GetStartTime() const {
9891   const i::CpuProfile* profile = reinterpret_cast<const i::CpuProfile*>(this);
9892   return profile->start_time().since_origin().InMicroseconds();
9893 }
9894 
GetEndTime() const9895 int64_t CpuProfile::GetEndTime() const {
9896   const i::CpuProfile* profile = reinterpret_cast<const i::CpuProfile*>(this);
9897   return profile->end_time().since_origin().InMicroseconds();
9898 }
9899 
ToInternal(const CpuProfile * profile)9900 static i::CpuProfile* ToInternal(const CpuProfile* profile) {
9901   return const_cast<i::CpuProfile*>(
9902       reinterpret_cast<const i::CpuProfile*>(profile));
9903 }
9904 
Serialize(OutputStream * stream,CpuProfile::SerializationFormat format) const9905 void CpuProfile::Serialize(OutputStream* stream,
9906                            CpuProfile::SerializationFormat format) const {
9907   Utils::ApiCheck(format == kJSON, "v8::CpuProfile::Serialize",
9908                   "Unknown serialization format");
9909   Utils::ApiCheck(stream->GetChunkSize() > 0, "v8::CpuProfile::Serialize",
9910                   "Invalid stream chunk size");
9911   i::CpuProfileJSONSerializer serializer(ToInternal(this));
9912   serializer.Serialize(stream);
9913 }
9914 
GetSamplesCount() const9915 int CpuProfile::GetSamplesCount() const {
9916   return reinterpret_cast<const i::CpuProfile*>(this)->samples_count();
9917 }
9918 
New(Isolate * isolate,CpuProfilingNamingMode naming_mode,CpuProfilingLoggingMode logging_mode)9919 CpuProfiler* CpuProfiler::New(Isolate* isolate,
9920                               CpuProfilingNamingMode naming_mode,
9921                               CpuProfilingLoggingMode logging_mode) {
9922   return reinterpret_cast<CpuProfiler*>(new i::CpuProfiler(
9923       reinterpret_cast<i::Isolate*>(isolate), naming_mode, logging_mode));
9924 }
9925 
CpuProfilingOptions(CpuProfilingMode mode,unsigned max_samples,int sampling_interval_us,MaybeLocal<Context> filter_context)9926 CpuProfilingOptions::CpuProfilingOptions(CpuProfilingMode mode,
9927                                          unsigned max_samples,
9928                                          int sampling_interval_us,
9929                                          MaybeLocal<Context> filter_context)
9930     : mode_(mode),
9931       max_samples_(max_samples),
9932       sampling_interval_us_(sampling_interval_us) {
9933   if (!filter_context.IsEmpty()) {
9934     Local<Context> local_filter_context = filter_context.ToLocalChecked();
9935     filter_context_.Reset(local_filter_context->GetIsolate(),
9936                           local_filter_context);
9937     filter_context_.SetWeak();
9938   }
9939 }
9940 
raw_filter_context() const9941 void* CpuProfilingOptions::raw_filter_context() const {
9942   return reinterpret_cast<void*>(
9943       i::Context::cast(*Utils::OpenPersistent(filter_context_))
9944           .native_context()
9945           .address());
9946 }
9947 
Dispose()9948 void CpuProfiler::Dispose() { delete reinterpret_cast<i::CpuProfiler*>(this); }
9949 
9950 // static
CollectSample(Isolate * isolate)9951 void CpuProfiler::CollectSample(Isolate* isolate) {
9952   i::CpuProfiler::CollectSample(reinterpret_cast<i::Isolate*>(isolate));
9953 }
9954 
SetSamplingInterval(int us)9955 void CpuProfiler::SetSamplingInterval(int us) {
9956   DCHECK_GE(us, 0);
9957   return reinterpret_cast<i::CpuProfiler*>(this)->set_sampling_interval(
9958       base::TimeDelta::FromMicroseconds(us));
9959 }
9960 
SetUsePreciseSampling(bool use_precise_sampling)9961 void CpuProfiler::SetUsePreciseSampling(bool use_precise_sampling) {
9962   reinterpret_cast<i::CpuProfiler*>(this)->set_use_precise_sampling(
9963       use_precise_sampling);
9964 }
9965 
Start(CpuProfilingOptions options,std::unique_ptr<DiscardedSamplesDelegate> delegate)9966 CpuProfilingResult CpuProfiler::Start(
9967     CpuProfilingOptions options,
9968     std::unique_ptr<DiscardedSamplesDelegate> delegate) {
9969   return reinterpret_cast<i::CpuProfiler*>(this)->StartProfiling(
9970       options, std::move(delegate));
9971 }
9972 
Start(Local<String> title,CpuProfilingOptions options,std::unique_ptr<DiscardedSamplesDelegate> delegate)9973 CpuProfilingResult CpuProfiler::Start(
9974     Local<String> title, CpuProfilingOptions options,
9975     std::unique_ptr<DiscardedSamplesDelegate> delegate) {
9976   return reinterpret_cast<i::CpuProfiler*>(this)->StartProfiling(
9977       *Utils::OpenHandle(*title), options, std::move(delegate));
9978 }
9979 
Start(Local<String> title,bool record_samples)9980 CpuProfilingResult CpuProfiler::Start(Local<String> title,
9981                                       bool record_samples) {
9982   CpuProfilingOptions options(
9983       kLeafNodeLineNumbers,
9984       record_samples ? CpuProfilingOptions::kNoSampleLimit : 0);
9985   return reinterpret_cast<i::CpuProfiler*>(this)->StartProfiling(
9986       *Utils::OpenHandle(*title), options);
9987 }
9988 
Start(Local<String> title,CpuProfilingMode mode,bool record_samples,unsigned max_samples)9989 CpuProfilingResult CpuProfiler::Start(Local<String> title,
9990                                       CpuProfilingMode mode,
9991                                       bool record_samples,
9992                                       unsigned max_samples) {
9993   CpuProfilingOptions options(mode, record_samples ? max_samples : 0);
9994   return reinterpret_cast<i::CpuProfiler*>(this)->StartProfiling(
9995       *Utils::OpenHandle(*title), options);
9996 }
9997 
StartProfiling(Local<String> title,CpuProfilingOptions options,std::unique_ptr<DiscardedSamplesDelegate> delegate)9998 CpuProfilingStatus CpuProfiler::StartProfiling(
9999     Local<String> title, CpuProfilingOptions options,
10000     std::unique_ptr<DiscardedSamplesDelegate> delegate) {
10001   return Start(title, options, std::move(delegate)).status;
10002 }
10003 
StartProfiling(Local<String> title,bool record_samples)10004 CpuProfilingStatus CpuProfiler::StartProfiling(Local<String> title,
10005                                                bool record_samples) {
10006   return Start(title, record_samples).status;
10007 }
10008 
StartProfiling(Local<String> title,CpuProfilingMode mode,bool record_samples,unsigned max_samples)10009 CpuProfilingStatus CpuProfiler::StartProfiling(Local<String> title,
10010                                                CpuProfilingMode mode,
10011                                                bool record_samples,
10012                                                unsigned max_samples) {
10013   return Start(title, mode, record_samples, max_samples).status;
10014 }
10015 
StopProfiling(Local<String> title)10016 CpuProfile* CpuProfiler::StopProfiling(Local<String> title) {
10017   return reinterpret_cast<CpuProfile*>(
10018       reinterpret_cast<i::CpuProfiler*>(this)->StopProfiling(
10019           *Utils::OpenHandle(*title)));
10020 }
10021 
Stop(ProfilerId id)10022 CpuProfile* CpuProfiler::Stop(ProfilerId id) {
10023   return reinterpret_cast<CpuProfile*>(
10024       reinterpret_cast<i::CpuProfiler*>(this)->StopProfiling(id));
10025 }
10026 
UseDetailedSourcePositionsForProfiling(Isolate * isolate)10027 void CpuProfiler::UseDetailedSourcePositionsForProfiling(Isolate* isolate) {
10028   reinterpret_cast<i::Isolate*>(isolate)
10029       ->SetDetailedSourcePositionsForProfiling(true);
10030 }
10031 
GetCodeStartAddress()10032 uintptr_t CodeEvent::GetCodeStartAddress() {
10033   return reinterpret_cast<i::CodeEvent*>(this)->code_start_address;
10034 }
10035 
GetCodeSize()10036 size_t CodeEvent::GetCodeSize() {
10037   return reinterpret_cast<i::CodeEvent*>(this)->code_size;
10038 }
10039 
GetFunctionName()10040 Local<String> CodeEvent::GetFunctionName() {
10041   return ToApiHandle<String>(
10042       reinterpret_cast<i::CodeEvent*>(this)->function_name);
10043 }
10044 
GetScriptName()10045 Local<String> CodeEvent::GetScriptName() {
10046   return ToApiHandle<String>(
10047       reinterpret_cast<i::CodeEvent*>(this)->script_name);
10048 }
10049 
GetScriptLine()10050 int CodeEvent::GetScriptLine() {
10051   return reinterpret_cast<i::CodeEvent*>(this)->script_line;
10052 }
10053 
GetScriptColumn()10054 int CodeEvent::GetScriptColumn() {
10055   return reinterpret_cast<i::CodeEvent*>(this)->script_column;
10056 }
10057 
GetCodeType()10058 CodeEventType CodeEvent::GetCodeType() {
10059   return reinterpret_cast<i::CodeEvent*>(this)->code_type;
10060 }
10061 
GetComment()10062 const char* CodeEvent::GetComment() {
10063   return reinterpret_cast<i::CodeEvent*>(this)->comment;
10064 }
10065 
GetPreviousCodeStartAddress()10066 uintptr_t CodeEvent::GetPreviousCodeStartAddress() {
10067   return reinterpret_cast<i::CodeEvent*>(this)->previous_code_start_address;
10068 }
10069 
GetCodeEventTypeName(CodeEventType code_event_type)10070 const char* CodeEvent::GetCodeEventTypeName(CodeEventType code_event_type) {
10071   switch (code_event_type) {
10072     case kUnknownType:
10073       return "Unknown";
10074 #define V(Name)       \
10075   case k##Name##Type: \
10076     return #Name;
10077       CODE_EVENTS_LIST(V)
10078 #undef V
10079   }
10080   // The execution should never pass here
10081   UNREACHABLE();
10082 }
10083 
CodeEventHandler(Isolate * isolate)10084 CodeEventHandler::CodeEventHandler(Isolate* isolate) {
10085   internal_listener_ =
10086       new i::ExternalCodeEventListener(reinterpret_cast<i::Isolate*>(isolate));
10087 }
10088 
~CodeEventHandler()10089 CodeEventHandler::~CodeEventHandler() {
10090   delete reinterpret_cast<i::ExternalCodeEventListener*>(internal_listener_);
10091 }
10092 
Enable()10093 void CodeEventHandler::Enable() {
10094   reinterpret_cast<i::ExternalCodeEventListener*>(internal_listener_)
10095       ->StartListening(this);
10096 }
10097 
Disable()10098 void CodeEventHandler::Disable() {
10099   reinterpret_cast<i::ExternalCodeEventListener*>(internal_listener_)
10100       ->StopListening();
10101 }
10102 
ToInternal(const HeapGraphEdge * edge)10103 static i::HeapGraphEdge* ToInternal(const HeapGraphEdge* edge) {
10104   return const_cast<i::HeapGraphEdge*>(
10105       reinterpret_cast<const i::HeapGraphEdge*>(edge));
10106 }
10107 
GetType() const10108 HeapGraphEdge::Type HeapGraphEdge::GetType() const {
10109   return static_cast<HeapGraphEdge::Type>(ToInternal(this)->type());
10110 }
10111 
GetName() const10112 Local<Value> HeapGraphEdge::GetName() const {
10113   i::HeapGraphEdge* edge = ToInternal(this);
10114   i::Isolate* isolate = edge->isolate();
10115   switch (edge->type()) {
10116     case i::HeapGraphEdge::kContextVariable:
10117     case i::HeapGraphEdge::kInternal:
10118     case i::HeapGraphEdge::kProperty:
10119     case i::HeapGraphEdge::kShortcut:
10120     case i::HeapGraphEdge::kWeak:
10121       return ToApiHandle<String>(
10122           isolate->factory()->InternalizeUtf8String(edge->name()));
10123     case i::HeapGraphEdge::kElement:
10124     case i::HeapGraphEdge::kHidden:
10125       return ToApiHandle<Number>(
10126           isolate->factory()->NewNumberFromInt(edge->index()));
10127     default:
10128       UNREACHABLE();
10129   }
10130 }
10131 
GetFromNode() const10132 const HeapGraphNode* HeapGraphEdge::GetFromNode() const {
10133   const i::HeapEntry* from = ToInternal(this)->from();
10134   return reinterpret_cast<const HeapGraphNode*>(from);
10135 }
10136 
GetToNode() const10137 const HeapGraphNode* HeapGraphEdge::GetToNode() const {
10138   const i::HeapEntry* to = ToInternal(this)->to();
10139   return reinterpret_cast<const HeapGraphNode*>(to);
10140 }
10141 
ToInternal(const HeapGraphNode * entry)10142 static i::HeapEntry* ToInternal(const HeapGraphNode* entry) {
10143   return const_cast<i::HeapEntry*>(
10144       reinterpret_cast<const i::HeapEntry*>(entry));
10145 }
10146 
GetType() const10147 HeapGraphNode::Type HeapGraphNode::GetType() const {
10148   return static_cast<HeapGraphNode::Type>(ToInternal(this)->type());
10149 }
10150 
GetName() const10151 Local<String> HeapGraphNode::GetName() const {
10152   i::Isolate* isolate = ToInternal(this)->isolate();
10153   return ToApiHandle<String>(
10154       isolate->factory()->InternalizeUtf8String(ToInternal(this)->name()));
10155 }
10156 
GetId() const10157 SnapshotObjectId HeapGraphNode::GetId() const { return ToInternal(this)->id(); }
10158 
GetShallowSize() const10159 size_t HeapGraphNode::GetShallowSize() const {
10160   return ToInternal(this)->self_size();
10161 }
10162 
GetChildrenCount() const10163 int HeapGraphNode::GetChildrenCount() const {
10164   return ToInternal(this)->children_count();
10165 }
10166 
GetChild(int index) const10167 const HeapGraphEdge* HeapGraphNode::GetChild(int index) const {
10168   return reinterpret_cast<const HeapGraphEdge*>(ToInternal(this)->child(index));
10169 }
10170 
ToInternal(const HeapSnapshot * snapshot)10171 static i::HeapSnapshot* ToInternal(const HeapSnapshot* snapshot) {
10172   return const_cast<i::HeapSnapshot*>(
10173       reinterpret_cast<const i::HeapSnapshot*>(snapshot));
10174 }
10175 
Delete()10176 void HeapSnapshot::Delete() {
10177   i::Isolate* isolate = ToInternal(this)->profiler()->isolate();
10178   if (isolate->heap_profiler()->GetSnapshotsCount() > 1 ||
10179       isolate->heap_profiler()->IsTakingSnapshot()) {
10180     ToInternal(this)->Delete();
10181   } else {
10182     // If this is the last snapshot, clean up all accessory data as well.
10183     isolate->heap_profiler()->DeleteAllSnapshots();
10184   }
10185 }
10186 
GetRoot() const10187 const HeapGraphNode* HeapSnapshot::GetRoot() const {
10188   return reinterpret_cast<const HeapGraphNode*>(ToInternal(this)->root());
10189 }
10190 
GetNodeById(SnapshotObjectId id) const10191 const HeapGraphNode* HeapSnapshot::GetNodeById(SnapshotObjectId id) const {
10192   return reinterpret_cast<const HeapGraphNode*>(
10193       ToInternal(this)->GetEntryById(id));
10194 }
10195 
GetNodesCount() const10196 int HeapSnapshot::GetNodesCount() const {
10197   return static_cast<int>(ToInternal(this)->entries().size());
10198 }
10199 
GetNode(int index) const10200 const HeapGraphNode* HeapSnapshot::GetNode(int index) const {
10201   return reinterpret_cast<const HeapGraphNode*>(
10202       &ToInternal(this)->entries().at(index));
10203 }
10204 
GetMaxSnapshotJSObjectId() const10205 SnapshotObjectId HeapSnapshot::GetMaxSnapshotJSObjectId() const {
10206   return ToInternal(this)->max_snapshot_js_object_id();
10207 }
10208 
Serialize(OutputStream * stream,HeapSnapshot::SerializationFormat format) const10209 void HeapSnapshot::Serialize(OutputStream* stream,
10210                              HeapSnapshot::SerializationFormat format) const {
10211   Utils::ApiCheck(format == kJSON, "v8::HeapSnapshot::Serialize",
10212                   "Unknown serialization format");
10213   Utils::ApiCheck(stream->GetChunkSize() > 0, "v8::HeapSnapshot::Serialize",
10214                   "Invalid stream chunk size");
10215   i::HeapSnapshotJSONSerializer serializer(ToInternal(this));
10216   serializer.Serialize(stream);
10217 }
10218 
10219 // static
10220 STATIC_CONST_MEMBER_DEFINITION const SnapshotObjectId
10221     HeapProfiler::kUnknownObjectId;
10222 
GetSnapshotCount()10223 int HeapProfiler::GetSnapshotCount() {
10224   return reinterpret_cast<i::HeapProfiler*>(this)->GetSnapshotsCount();
10225 }
10226 
GetHeapSnapshot(int index)10227 const HeapSnapshot* HeapProfiler::GetHeapSnapshot(int index) {
10228   return reinterpret_cast<const HeapSnapshot*>(
10229       reinterpret_cast<i::HeapProfiler*>(this)->GetSnapshot(index));
10230 }
10231 
GetObjectId(Local<Value> value)10232 SnapshotObjectId HeapProfiler::GetObjectId(Local<Value> value) {
10233   i::Handle<i::Object> obj = Utils::OpenHandle(*value);
10234   return reinterpret_cast<i::HeapProfiler*>(this)->GetSnapshotObjectId(obj);
10235 }
10236 
GetObjectId(NativeObject value)10237 SnapshotObjectId HeapProfiler::GetObjectId(NativeObject value) {
10238   return reinterpret_cast<i::HeapProfiler*>(this)->GetSnapshotObjectId(value);
10239 }
10240 
FindObjectById(SnapshotObjectId id)10241 Local<Value> HeapProfiler::FindObjectById(SnapshotObjectId id) {
10242   i::Handle<i::Object> obj =
10243       reinterpret_cast<i::HeapProfiler*>(this)->FindHeapObjectById(id);
10244   if (obj.is_null()) return Local<Value>();
10245   return Utils::ToLocal(obj);
10246 }
10247 
ClearObjectIds()10248 void HeapProfiler::ClearObjectIds() {
10249   reinterpret_cast<i::HeapProfiler*>(this)->ClearHeapObjectMap();
10250 }
10251 
TakeHeapSnapshot(ActivityControl * control,ObjectNameResolver * resolver,bool treat_global_objects_as_roots,bool capture_numeric_value)10252 const HeapSnapshot* HeapProfiler::TakeHeapSnapshot(
10253     ActivityControl* control, ObjectNameResolver* resolver,
10254     bool treat_global_objects_as_roots, bool capture_numeric_value) {
10255   return reinterpret_cast<const HeapSnapshot*>(
10256       reinterpret_cast<i::HeapProfiler*>(this)->TakeSnapshot(
10257           control, resolver, treat_global_objects_as_roots,
10258           capture_numeric_value));
10259 }
10260 
StartTrackingHeapObjects(bool track_allocations)10261 void HeapProfiler::StartTrackingHeapObjects(bool track_allocations) {
10262   reinterpret_cast<i::HeapProfiler*>(this)->StartHeapObjectsTracking(
10263       track_allocations);
10264 }
10265 
StopTrackingHeapObjects()10266 void HeapProfiler::StopTrackingHeapObjects() {
10267   reinterpret_cast<i::HeapProfiler*>(this)->StopHeapObjectsTracking();
10268 }
10269 
GetHeapStats(OutputStream * stream,int64_t * timestamp_us)10270 SnapshotObjectId HeapProfiler::GetHeapStats(OutputStream* stream,
10271                                             int64_t* timestamp_us) {
10272   i::HeapProfiler* heap_profiler = reinterpret_cast<i::HeapProfiler*>(this);
10273   return heap_profiler->PushHeapObjectsStats(stream, timestamp_us);
10274 }
10275 
StartSamplingHeapProfiler(uint64_t sample_interval,int stack_depth,SamplingFlags flags)10276 bool HeapProfiler::StartSamplingHeapProfiler(uint64_t sample_interval,
10277                                              int stack_depth,
10278                                              SamplingFlags flags) {
10279   return reinterpret_cast<i::HeapProfiler*>(this)->StartSamplingHeapProfiler(
10280       sample_interval, stack_depth, flags);
10281 }
10282 
StopSamplingHeapProfiler()10283 void HeapProfiler::StopSamplingHeapProfiler() {
10284   reinterpret_cast<i::HeapProfiler*>(this)->StopSamplingHeapProfiler();
10285 }
10286 
GetAllocationProfile()10287 AllocationProfile* HeapProfiler::GetAllocationProfile() {
10288   return reinterpret_cast<i::HeapProfiler*>(this)->GetAllocationProfile();
10289 }
10290 
DeleteAllHeapSnapshots()10291 void HeapProfiler::DeleteAllHeapSnapshots() {
10292   reinterpret_cast<i::HeapProfiler*>(this)->DeleteAllSnapshots();
10293 }
10294 
AddBuildEmbedderGraphCallback(BuildEmbedderGraphCallback callback,void * data)10295 void HeapProfiler::AddBuildEmbedderGraphCallback(
10296     BuildEmbedderGraphCallback callback, void* data) {
10297   reinterpret_cast<i::HeapProfiler*>(this)->AddBuildEmbedderGraphCallback(
10298       callback, data);
10299 }
10300 
RemoveBuildEmbedderGraphCallback(BuildEmbedderGraphCallback callback,void * data)10301 void HeapProfiler::RemoveBuildEmbedderGraphCallback(
10302     BuildEmbedderGraphCallback callback, void* data) {
10303   reinterpret_cast<i::HeapProfiler*>(this)->RemoveBuildEmbedderGraphCallback(
10304       callback, data);
10305 }
10306 
SetGetDetachednessCallback(GetDetachednessCallback callback,void * data)10307 void HeapProfiler::SetGetDetachednessCallback(GetDetachednessCallback callback,
10308                                               void* data) {
10309   reinterpret_cast<i::HeapProfiler*>(this)->SetGetDetachednessCallback(callback,
10310                                                                        data);
10311 }
10312 
SetStackStart(void * stack_start)10313 void EmbedderHeapTracer::SetStackStart(void* stack_start) {
10314   CHECK(isolate_);
10315   reinterpret_cast<i::Isolate*>(isolate_)->global_handles()->SetStackStart(
10316       stack_start);
10317 }
10318 
FinalizeTracing()10319 void EmbedderHeapTracer::FinalizeTracing() {
10320   if (isolate_) {
10321     i::Isolate* isolate = reinterpret_cast<i::Isolate*>(isolate_);
10322     if (isolate->heap()->incremental_marking()->IsMarking()) {
10323       isolate->heap()->FinalizeIncrementalMarkingAtomically(
10324           i::GarbageCollectionReason::kExternalFinalize);
10325     }
10326   }
10327 }
10328 
IncreaseAllocatedSize(size_t bytes)10329 void EmbedderHeapTracer::IncreaseAllocatedSize(size_t bytes) {
10330   if (isolate_) {
10331     i::LocalEmbedderHeapTracer* const tracer =
10332         reinterpret_cast<i::Isolate*>(isolate_)
10333             ->heap()
10334             ->local_embedder_heap_tracer();
10335     DCHECK_NOT_NULL(tracer);
10336     tracer->IncreaseAllocatedSize(bytes);
10337   }
10338 }
10339 
DecreaseAllocatedSize(size_t bytes)10340 void EmbedderHeapTracer::DecreaseAllocatedSize(size_t bytes) {
10341   if (isolate_) {
10342     i::LocalEmbedderHeapTracer* const tracer =
10343         reinterpret_cast<i::Isolate*>(isolate_)
10344             ->heap()
10345             ->local_embedder_heap_tracer();
10346     DCHECK_NOT_NULL(tracer);
10347     tracer->DecreaseAllocatedSize(bytes);
10348   }
10349 }
10350 
RegisterEmbedderReference(const BasicTracedReference<v8::Data> & ref)10351 void EmbedderHeapTracer::RegisterEmbedderReference(
10352     const BasicTracedReference<v8::Data>& ref) {
10353   if (ref.IsEmpty()) return;
10354 
10355   i::Heap* const heap = reinterpret_cast<i::Isolate*>(isolate_)->heap();
10356   heap->RegisterExternallyReferencedObject(
10357       reinterpret_cast<i::Address*>(ref.val_));
10358 }
10359 
IterateTracedGlobalHandles(TracedGlobalHandleVisitor * visitor)10360 void EmbedderHeapTracer::IterateTracedGlobalHandles(
10361     TracedGlobalHandleVisitor* visitor) {
10362   i::Isolate* isolate = reinterpret_cast<i::Isolate*>(isolate_);
10363   i::DisallowGarbageCollection no_gc;
10364   isolate->global_handles()->IterateTracedNodes(visitor);
10365 }
10366 
IsRootForNonTracingGC(const v8::TracedReference<v8::Value> & handle)10367 bool EmbedderHeapTracer::IsRootForNonTracingGC(
10368     const v8::TracedReference<v8::Value>& handle) {
10369   return true;
10370 }
10371 
ResetHandleInNonTracingGC(const v8::TracedReference<v8::Value> & handle)10372 void EmbedderHeapTracer::ResetHandleInNonTracingGC(
10373     const v8::TracedReference<v8::Value>& handle) {
10374   UNREACHABLE();
10375 }
10376 
EmbedderStateScope(Isolate * isolate,Local<v8::Context> context,EmbedderStateTag tag)10377 EmbedderStateScope::EmbedderStateScope(Isolate* isolate,
10378                                        Local<v8::Context> context,
10379                                        EmbedderStateTag tag)
10380     : embedder_state_(new internal::EmbedderState(isolate, context, tag)) {}
10381 
10382 // std::unique_ptr's destructor is not compatible with Forward declared
10383 // EmbedderState class.
10384 // Default destructor must be defined in implementation file.
10385 EmbedderStateScope::~EmbedderStateScope() = default;
10386 
CheckValue() const10387 void TracedReferenceBase::CheckValue() const {
10388 #ifdef V8_HOST_ARCH_64_BIT
10389   if (!val_) return;
10390 
10391   CHECK_NE(internal::kGlobalHandleZapValue, *reinterpret_cast<uint64_t*>(val_));
10392 #endif  // V8_HOST_ARCH_64_BIT
10393 }
10394 
CFunction(const void * address,const CFunctionInfo * type_info)10395 CFunction::CFunction(const void* address, const CFunctionInfo* type_info)
10396     : address_(address), type_info_(type_info) {
10397   CHECK_NOT_NULL(address_);
10398   CHECK_NOT_NULL(type_info_);
10399 }
10400 
CFunctionInfo(const CTypeInfo & return_info,unsigned int arg_count,const CTypeInfo * arg_info)10401 CFunctionInfo::CFunctionInfo(const CTypeInfo& return_info,
10402                              unsigned int arg_count, const CTypeInfo* arg_info)
10403     : return_info_(return_info), arg_count_(arg_count), arg_info_(arg_info) {
10404   if (arg_count_ > 0) {
10405     for (unsigned int i = 0; i < arg_count_ - 1; ++i) {
10406       DCHECK(arg_info_[i].GetType() != CTypeInfo::kCallbackOptionsType);
10407     }
10408   }
10409 }
10410 
ArgumentInfo(unsigned int index) const10411 const CTypeInfo& CFunctionInfo::ArgumentInfo(unsigned int index) const {
10412   DCHECK_LT(index, ArgumentCount());
10413   return arg_info_[index];
10414 }
10415 
ValidateIndex(size_t index) const10416 void FastApiTypedArrayBase::ValidateIndex(size_t index) const {
10417   DCHECK_LT(index, length_);
10418 }
10419 
RegisterState()10420 RegisterState::RegisterState()
10421     : pc(nullptr), sp(nullptr), fp(nullptr), lr(nullptr) {}
10422 RegisterState::~RegisterState() = default;
10423 
RegisterState(const RegisterState & other)10424 RegisterState::RegisterState(const RegisterState& other) { *this = other; }
10425 
operator =(const RegisterState & other)10426 RegisterState& RegisterState::operator=(const RegisterState& other) {
10427   if (&other != this) {
10428     pc = other.pc;
10429     sp = other.sp;
10430     fp = other.fp;
10431     lr = other.lr;
10432     if (other.callee_saved) {
10433       // Make a deep copy if {other.callee_saved} is non-null.
10434       callee_saved =
10435           std::make_unique<CalleeSavedRegisters>(*(other.callee_saved));
10436     } else {
10437       // Otherwise, set {callee_saved} to null to match {other}.
10438       callee_saved.reset();
10439     }
10440   }
10441   return *this;
10442 }
10443 
10444 #if !V8_ENABLE_WEBASSEMBLY
10445 // If WebAssembly is disabled, we still need to provide an implementation of the
10446 // WasmStreaming API. Since {WasmStreaming::Unpack} will always fail, all
10447 // methods are unreachable.
10448 
10449 class WasmStreaming::WasmStreamingImpl {};
10450 
WasmStreaming(std::unique_ptr<WasmStreamingImpl>)10451 WasmStreaming::WasmStreaming(std::unique_ptr<WasmStreamingImpl>) {
10452   UNREACHABLE();
10453 }
10454 
10455 WasmStreaming::~WasmStreaming() = default;
10456 
OnBytesReceived(const uint8_t * bytes,size_t size)10457 void WasmStreaming::OnBytesReceived(const uint8_t* bytes, size_t size) {
10458   UNREACHABLE();
10459 }
10460 
Finish(bool can_use_compiled_module)10461 void WasmStreaming::Finish(bool can_use_compiled_module) { UNREACHABLE(); }
10462 
Abort(MaybeLocal<Value> exception)10463 void WasmStreaming::Abort(MaybeLocal<Value> exception) { UNREACHABLE(); }
10464 
SetCompiledModuleBytes(const uint8_t * bytes,size_t size)10465 bool WasmStreaming::SetCompiledModuleBytes(const uint8_t* bytes, size_t size) {
10466   UNREACHABLE();
10467 }
10468 
SetClient(std::shared_ptr<Client> client)10469 void WasmStreaming::SetClient(std::shared_ptr<Client> client) { UNREACHABLE(); }
10470 
SetUrl(const char * url,size_t length)10471 void WasmStreaming::SetUrl(const char* url, size_t length) { UNREACHABLE(); }
10472 
10473 // static
Unpack(Isolate * isolate,Local<Value> value)10474 std::shared_ptr<WasmStreaming> WasmStreaming::Unpack(Isolate* isolate,
10475                                                      Local<Value> value) {
10476   FATAL("WebAssembly is disabled");
10477 }
10478 #endif  // !V8_ENABLE_WEBASSEMBLY
10479 
10480 namespace internal {
10481 
10482 const size_t HandleScopeImplementer::kEnteredContextsOffset =
10483     offsetof(HandleScopeImplementer, entered_contexts_);
10484 const size_t HandleScopeImplementer::kIsMicrotaskContextOffset =
10485     offsetof(HandleScopeImplementer, is_microtask_context_);
10486 
FreeThreadResources()10487 void HandleScopeImplementer::FreeThreadResources() { Free(); }
10488 
ArchiveThread(char * storage)10489 char* HandleScopeImplementer::ArchiveThread(char* storage) {
10490   HandleScopeData* current = isolate_->handle_scope_data();
10491   handle_scope_data_ = *current;
10492   MemCopy(storage, this, sizeof(*this));
10493 
10494   ResetAfterArchive();
10495   current->Initialize();
10496 
10497   return storage + ArchiveSpacePerThread();
10498 }
10499 
ArchiveSpacePerThread()10500 int HandleScopeImplementer::ArchiveSpacePerThread() {
10501   return sizeof(HandleScopeImplementer);
10502 }
10503 
RestoreThread(char * storage)10504 char* HandleScopeImplementer::RestoreThread(char* storage) {
10505   MemCopy(this, storage, sizeof(*this));
10506   *isolate_->handle_scope_data() = handle_scope_data_;
10507   return storage + ArchiveSpacePerThread();
10508 }
10509 
IterateThis(RootVisitor * v)10510 void HandleScopeImplementer::IterateThis(RootVisitor* v) {
10511 #ifdef DEBUG
10512   bool found_block_before_deferred = false;
10513 #endif
10514   // Iterate over all handles in the blocks except for the last.
10515   for (int i = static_cast<int>(blocks()->size()) - 2; i >= 0; --i) {
10516     Address* block = blocks()->at(i);
10517     // Cast possibly-unrelated pointers to plain Address before comparing them
10518     // to avoid undefined behavior.
10519     if (last_handle_before_deferred_block_ != nullptr &&
10520         (reinterpret_cast<Address>(last_handle_before_deferred_block_) <=
10521          reinterpret_cast<Address>(&block[kHandleBlockSize])) &&
10522         (reinterpret_cast<Address>(last_handle_before_deferred_block_) >=
10523          reinterpret_cast<Address>(block))) {
10524       v->VisitRootPointers(Root::kHandleScope, nullptr, FullObjectSlot(block),
10525                            FullObjectSlot(last_handle_before_deferred_block_));
10526       DCHECK(!found_block_before_deferred);
10527 #ifdef DEBUG
10528       found_block_before_deferred = true;
10529 #endif
10530     } else {
10531       v->VisitRootPointers(Root::kHandleScope, nullptr, FullObjectSlot(block),
10532                            FullObjectSlot(&block[kHandleBlockSize]));
10533     }
10534   }
10535 
10536   DCHECK(last_handle_before_deferred_block_ == nullptr ||
10537          found_block_before_deferred);
10538 
10539   // Iterate over live handles in the last block (if any).
10540   if (!blocks()->empty()) {
10541     v->VisitRootPointers(Root::kHandleScope, nullptr,
10542                          FullObjectSlot(blocks()->back()),
10543                          FullObjectSlot(handle_scope_data_.next));
10544   }
10545 
10546   DetachableVector<Context>* context_lists[2] = {&saved_contexts_,
10547                                                  &entered_contexts_};
10548   for (unsigned i = 0; i < arraysize(context_lists); i++) {
10549     context_lists[i]->shrink_to_fit();
10550     if (context_lists[i]->empty()) continue;
10551     FullObjectSlot start(&context_lists[i]->front());
10552     v->VisitRootPointers(Root::kHandleScope, nullptr, start,
10553                          start + static_cast<int>(context_lists[i]->size()));
10554   }
10555   // The shape of |entered_contexts_| and |is_microtask_context_| stacks must
10556   // be in sync.
10557   is_microtask_context_.shrink_to_fit();
10558   DCHECK_EQ(entered_contexts_.capacity(), is_microtask_context_.capacity());
10559   DCHECK_EQ(entered_contexts_.size(), is_microtask_context_.size());
10560 }
10561 
Iterate(RootVisitor * v)10562 void HandleScopeImplementer::Iterate(RootVisitor* v) {
10563   HandleScopeData* current = isolate_->handle_scope_data();
10564   handle_scope_data_ = *current;
10565   IterateThis(v);
10566 }
10567 
Iterate(RootVisitor * v,char * storage)10568 char* HandleScopeImplementer::Iterate(RootVisitor* v, char* storage) {
10569   HandleScopeImplementer* scope_implementer =
10570       reinterpret_cast<HandleScopeImplementer*>(storage);
10571   scope_implementer->IterateThis(v);
10572   return storage + ArchiveSpacePerThread();
10573 }
10574 
DetachPersistent(Address * prev_limit)10575 std::unique_ptr<PersistentHandles> HandleScopeImplementer::DetachPersistent(
10576     Address* prev_limit) {
10577   std::unique_ptr<PersistentHandles> ph(new PersistentHandles(isolate()));
10578   DCHECK_NOT_NULL(prev_limit);
10579 
10580   while (!blocks_.empty()) {
10581     Address* block_start = blocks_.back();
10582     Address* block_limit = &block_start[kHandleBlockSize];
10583     // We should not need to check for SealHandleScope here. Assert this.
10584     DCHECK_IMPLIES(block_start <= prev_limit && prev_limit <= block_limit,
10585                    prev_limit == block_limit);
10586     if (prev_limit == block_limit) break;
10587     ph->blocks_.push_back(blocks_.back());
10588 #if DEBUG
10589     ph->ordered_blocks_.insert(blocks_.back());
10590 #endif
10591     blocks_.pop_back();
10592   }
10593 
10594   // ph->blocks_ now contains the blocks installed on the
10595   // HandleScope stack since BeginDeferredScope was called, but in
10596   // reverse order.
10597 
10598   // Switch first and last blocks, such that the last block is the one
10599   // that is potentially half full.
10600   DCHECK(!blocks_.empty() && !ph->blocks_.empty());
10601   std::swap(ph->blocks_.front(), ph->blocks_.back());
10602 
10603   ph->block_next_ = isolate()->handle_scope_data()->next;
10604   Address* block_start = ph->blocks_.back();
10605   ph->block_limit_ = block_start + kHandleBlockSize;
10606 
10607   DCHECK_NOT_NULL(last_handle_before_deferred_block_);
10608   last_handle_before_deferred_block_ = nullptr;
10609   return ph;
10610 }
10611 
BeginDeferredScope()10612 void HandleScopeImplementer::BeginDeferredScope() {
10613   DCHECK_NULL(last_handle_before_deferred_block_);
10614   last_handle_before_deferred_block_ = isolate()->handle_scope_data()->next;
10615 }
10616 
InvokeAccessorGetterCallback(v8::Local<v8::Name> property,const v8::PropertyCallbackInfo<v8::Value> & info,v8::AccessorNameGetterCallback getter)10617 void InvokeAccessorGetterCallback(
10618     v8::Local<v8::Name> property,
10619     const v8::PropertyCallbackInfo<v8::Value>& info,
10620     v8::AccessorNameGetterCallback getter) {
10621   // Leaving JavaScript.
10622   Isolate* isolate = reinterpret_cast<Isolate*>(info.GetIsolate());
10623   RCS_SCOPE(isolate, RuntimeCallCounterId::kAccessorGetterCallback);
10624   Address getter_address = reinterpret_cast<Address>(getter);
10625   ExternalCallbackScope call_scope(isolate, getter_address);
10626   getter(property, info);
10627 }
10628 
InvokeFunctionCallback(const v8::FunctionCallbackInfo<v8::Value> & info,v8::FunctionCallback callback)10629 void InvokeFunctionCallback(const v8::FunctionCallbackInfo<v8::Value>& info,
10630                             v8::FunctionCallback callback) {
10631   Isolate* isolate = reinterpret_cast<Isolate*>(info.GetIsolate());
10632   RCS_SCOPE(isolate, RuntimeCallCounterId::kFunctionCallback);
10633   Address callback_address = reinterpret_cast<Address>(callback);
10634   ExternalCallbackScope call_scope(isolate, callback_address);
10635   callback(info);
10636 }
10637 
InvokeFinalizationRegistryCleanupFromTask(Handle<Context> context,Handle<JSFinalizationRegistry> finalization_registry,Handle<Object> callback)10638 void InvokeFinalizationRegistryCleanupFromTask(
10639     Handle<Context> context,
10640     Handle<JSFinalizationRegistry> finalization_registry,
10641     Handle<Object> callback) {
10642   Isolate* isolate = finalization_registry->native_context().GetIsolate();
10643   RCS_SCOPE(isolate,
10644             RuntimeCallCounterId::kFinalizationRegistryCleanupFromTask);
10645   // Do not use ENTER_V8 because this is always called from a running
10646   // FinalizationRegistryCleanupTask within V8 and we should not log it as an
10647   // API call. This method is implemented here to avoid duplication of the
10648   // exception handling and microtask running logic in CallDepthScope.
10649   if (IsExecutionTerminatingCheck(isolate)) return;
10650   Local<v8::Context> api_context = Utils::ToLocal(context);
10651   CallDepthScope<true> call_depth_scope(isolate, api_context);
10652   VMState<OTHER> state(isolate);
10653   Handle<Object> argv[] = {callback};
10654   if (Execution::CallBuiltin(isolate,
10655                              isolate->finalization_registry_cleanup_some(),
10656                              finalization_registry, arraysize(argv), argv)
10657           .is_null()) {
10658     call_depth_scope.Escape();
10659   }
10660 }
10661 
10662 template <>
EXPORT_TEMPLATE_DEFINE(V8_EXPORT_PRIVATE)10663 EXPORT_TEMPLATE_DEFINE(V8_EXPORT_PRIVATE)
10664 int32_t ConvertDouble(double d) {
10665   return internal::DoubleToInt32(d);
10666 }
10667 
10668 template <>
EXPORT_TEMPLATE_DEFINE(V8_EXPORT_PRIVATE)10669 EXPORT_TEMPLATE_DEFINE(V8_EXPORT_PRIVATE)
10670 uint32_t ConvertDouble(double d) {
10671   return internal::DoubleToUint32(d);
10672 }
10673 
10674 template <>
EXPORT_TEMPLATE_DEFINE(V8_EXPORT_PRIVATE)10675 EXPORT_TEMPLATE_DEFINE(V8_EXPORT_PRIVATE)
10676 float ConvertDouble(double d) {
10677   return internal::DoubleToFloat32(d);
10678 }
10679 
10680 template <>
EXPORT_TEMPLATE_DEFINE(V8_EXPORT_PRIVATE)10681 EXPORT_TEMPLATE_DEFINE(V8_EXPORT_PRIVATE)
10682 double ConvertDouble(double d) {
10683   return d;
10684 }
10685 
10686 template <>
EXPORT_TEMPLATE_DEFINE(V8_EXPORT_PRIVATE)10687 EXPORT_TEMPLATE_DEFINE(V8_EXPORT_PRIVATE)
10688 int64_t ConvertDouble(double d) {
10689   return internal::DoubleToWebIDLInt64(d);
10690 }
10691 
10692 template <>
EXPORT_TEMPLATE_DEFINE(V8_EXPORT_PRIVATE)10693 EXPORT_TEMPLATE_DEFINE(V8_EXPORT_PRIVATE)
10694 uint64_t ConvertDouble(double d) {
10695   return internal::DoubleToWebIDLUint64(d);
10696 }
10697 
10698 template <>
EXPORT_TEMPLATE_DEFINE(V8_EXPORT_PRIVATE)10699 EXPORT_TEMPLATE_DEFINE(V8_EXPORT_PRIVATE)
10700 bool ConvertDouble(double d) {
10701   // Implements https://tc39.es/ecma262/#sec-toboolean.
10702   return !std::isnan(d) && d != 0;
10703 }
10704 
10705 // Undefine macros for jumbo build.
10706 #undef SET_FIELD_WRAPPED
10707 #undef NEW_STRING
10708 #undef CALLBACK_SETTER
10709 
10710 }  // namespace internal
10711 
10712 template <>
10713 bool V8_EXPORT V8_WARN_UNUSED_RESULT
TryToCopyAndConvertArrayToCppBuffer(Local<Array> src,int32_t * dst,uint32_t max_length)10714 TryToCopyAndConvertArrayToCppBuffer<CTypeInfoBuilder<int32_t>::Build().GetId(),
10715                                     int32_t>(Local<Array> src, int32_t* dst,
10716                                              uint32_t max_length) {
10717   return CopyAndConvertArrayToCppBuffer<
10718       CTypeInfo(CTypeInfo::Type::kInt32, CTypeInfo::SequenceType::kIsSequence)
10719           .GetId(),
10720       int32_t>(src, dst, max_length);
10721 }
10722 
10723 template <>
10724 bool V8_EXPORT V8_WARN_UNUSED_RESULT
TryToCopyAndConvertArrayToCppBuffer(Local<Array> src,uint32_t * dst,uint32_t max_length)10725 TryToCopyAndConvertArrayToCppBuffer<CTypeInfoBuilder<uint32_t>::Build().GetId(),
10726                                     uint32_t>(Local<Array> src, uint32_t* dst,
10727                                               uint32_t max_length) {
10728   return CopyAndConvertArrayToCppBuffer<
10729       CTypeInfo(CTypeInfo::Type::kUint32, CTypeInfo::SequenceType::kIsSequence)
10730           .GetId(),
10731       uint32_t>(src, dst, max_length);
10732 }
10733 
10734 template <>
10735 bool V8_EXPORT V8_WARN_UNUSED_RESULT
TryToCopyAndConvertArrayToCppBuffer(Local<Array> src,float * dst,uint32_t max_length)10736 TryToCopyAndConvertArrayToCppBuffer<CTypeInfoBuilder<float>::Build().GetId(),
10737                                     float>(Local<Array> src, float* dst,
10738                                            uint32_t max_length) {
10739   return CopyAndConvertArrayToCppBuffer<
10740       CTypeInfo(CTypeInfo::Type::kFloat32, CTypeInfo::SequenceType::kIsSequence)
10741           .GetId(),
10742       float>(src, dst, max_length);
10743 }
10744 
10745 template <>
10746 bool V8_EXPORT V8_WARN_UNUSED_RESULT
TryToCopyAndConvertArrayToCppBuffer(Local<Array> src,double * dst,uint32_t max_length)10747 TryToCopyAndConvertArrayToCppBuffer<CTypeInfoBuilder<double>::Build().GetId(),
10748                                     double>(Local<Array> src, double* dst,
10749                                             uint32_t max_length) {
10750   return CopyAndConvertArrayToCppBuffer<
10751       CTypeInfo(CTypeInfo::Type::kFloat64, CTypeInfo::SequenceType::kIsSequence)
10752           .GetId(),
10753       double>(src, dst, max_length);
10754 }
10755 
10756 }  // namespace v8
10757 
10758 #undef TRACE_BS
10759 #include "src/api/api-macros-undef.h"
10760