aboutsummaryrefslogtreecommitdiff
path: root/deps/v8/src/builtins/builtins-weak-refs.cc
diff options
context:
space:
mode:
Diffstat (limited to 'deps/v8/src/builtins/builtins-weak-refs.cc')
-rw-r--r--deps/v8/src/builtins/builtins-weak-refs.cc169
1 files changed, 169 insertions, 0 deletions
diff --git a/deps/v8/src/builtins/builtins-weak-refs.cc b/deps/v8/src/builtins/builtins-weak-refs.cc
new file mode 100644
index 0000000000..e89deb705b
--- /dev/null
+++ b/deps/v8/src/builtins/builtins-weak-refs.cc
@@ -0,0 +1,169 @@
+// Copyright 2018 the V8 project authors. All rights reserved.
+// Use of this source code is governed by a BSD-style license that can be
+// found in the LICENSE file.
+
+#include "src/builtins/builtins-utils-inl.h"
+#include "src/counters.h"
+#include "src/objects/js-weak-refs-inl.h"
+
+namespace v8 {
+namespace internal {
+
+BUILTIN(WeakFactoryConstructor) {
+ HandleScope scope(isolate);
+ Handle<JSFunction> target = args.target();
+ if (args.new_target()->IsUndefined(isolate)) { // [[Call]]
+ THROW_NEW_ERROR_RETURN_FAILURE(
+ isolate, NewTypeError(MessageTemplate::kConstructorNotFunction,
+ handle(target->shared()->Name(), isolate)));
+ }
+ // [[Construct]]
+ Handle<JSReceiver> new_target = Handle<JSReceiver>::cast(args.new_target());
+ Handle<Object> cleanup = args.atOrUndefined(isolate, 1);
+
+ if (!cleanup->IsCallable()) {
+ THROW_NEW_ERROR_RETURN_FAILURE(
+ isolate, NewTypeError(MessageTemplate::kWeakRefsCleanupMustBeCallable));
+ }
+
+ Handle<JSObject> result;
+ ASSIGN_RETURN_FAILURE_ON_EXCEPTION(
+ isolate, result,
+ JSObject::New(target, new_target, Handle<AllocationSite>::null()));
+
+ Handle<JSWeakFactory> weak_factory = Handle<JSWeakFactory>::cast(result);
+ weak_factory->set_native_context(*isolate->native_context());
+ weak_factory->set_cleanup(*cleanup);
+ weak_factory->set_flags(
+ JSWeakFactory::ScheduledForCleanupField::encode(false));
+ return *weak_factory;
+}
+
+BUILTIN(WeakFactoryMakeCell) {
+ HandleScope scope(isolate);
+ const char* method_name = "WeakFactory.prototype.makeCell";
+
+ CHECK_RECEIVER(JSWeakFactory, weak_factory, method_name);
+
+ Handle<Object> target = args.atOrUndefined(isolate, 1);
+ if (!target->IsJSReceiver()) {
+ THROW_NEW_ERROR_RETURN_FAILURE(
+ isolate,
+ NewTypeError(MessageTemplate::kWeakRefsMakeCellTargetMustBeObject));
+ }
+ Handle<JSReceiver> target_receiver = Handle<JSReceiver>::cast(target);
+ Handle<Object> holdings = args.atOrUndefined(isolate, 2);
+ if (target->SameValue(*holdings)) {
+ THROW_NEW_ERROR_RETURN_FAILURE(
+ isolate,
+ NewTypeError(
+ MessageTemplate::kWeakRefsMakeCellTargetAndHoldingsMustNotBeSame));
+ }
+
+ // TODO(marja): Realms.
+
+ Handle<Map> weak_cell_map(isolate->native_context()->js_weak_cell_map(),
+ isolate);
+
+ // Allocate the JSWeakCell object in the old space, because 1) JSWeakCell
+ // weakness handling is only implemented in the old space 2) they're
+ // supposedly long-living. TODO(marja): Support JSWeakCells in Scavenger.
+ Handle<JSWeakCell> weak_cell =
+ Handle<JSWeakCell>::cast(isolate->factory()->NewJSObjectFromMap(
+ weak_cell_map, TENURED, Handle<AllocationSite>::null()));
+ weak_cell->set_target(*target_receiver);
+ weak_cell->set_holdings(*holdings);
+ weak_factory->AddWeakCell(*weak_cell);
+ return *weak_cell;
+}
+
+BUILTIN(WeakFactoryCleanupSome) {
+ HandleScope scope(isolate);
+ const char* method_name = "WeakFactory.prototype.cleanupSome";
+
+ CHECK_RECEIVER(JSWeakFactory, weak_factory, method_name);
+
+ // Don't do set_scheduled_for_cleanup(false); we still have the microtask
+ // scheduled and don't want to schedule another one in case the user never
+ // executes microtasks.
+ JSWeakFactory::Cleanup(weak_factory, isolate);
+ return ReadOnlyRoots(isolate).undefined_value();
+}
+
+BUILTIN(WeakFactoryCleanupIteratorNext) {
+ HandleScope scope(isolate);
+ CHECK_RECEIVER(JSWeakFactoryCleanupIterator, iterator, "next");
+
+ Handle<JSWeakFactory> weak_factory(iterator->factory(), isolate);
+ if (!weak_factory->NeedsCleanup()) {
+ return *isolate->factory()->NewJSIteratorResult(
+ handle(ReadOnlyRoots(isolate).undefined_value(), isolate), true);
+ }
+ Handle<JSWeakCell> weak_cell_object =
+ handle(weak_factory->PopClearedCell(isolate), isolate);
+
+ return *isolate->factory()->NewJSIteratorResult(weak_cell_object, false);
+}
+
+BUILTIN(WeakCellHoldingsGetter) {
+ HandleScope scope(isolate);
+ CHECK_RECEIVER(JSWeakCell, weak_cell, "get WeakCell.holdings");
+ return weak_cell->holdings();
+}
+
+BUILTIN(WeakCellClear) {
+ HandleScope scope(isolate);
+ CHECK_RECEIVER(JSWeakCell, weak_cell, "WeakCell.prototype.clear");
+ weak_cell->Clear(isolate);
+ return ReadOnlyRoots(isolate).undefined_value();
+}
+
+BUILTIN(WeakRefConstructor) {
+ HandleScope scope(isolate);
+ Handle<JSFunction> target = args.target();
+ if (args.new_target()->IsUndefined(isolate)) { // [[Call]]
+ THROW_NEW_ERROR_RETURN_FAILURE(
+ isolate, NewTypeError(MessageTemplate::kConstructorNotFunction,
+ handle(target->shared()->Name(), isolate)));
+ }
+ // [[Construct]]
+ Handle<JSReceiver> new_target = Handle<JSReceiver>::cast(args.new_target());
+ Handle<Object> target_object = args.atOrUndefined(isolate, 1);
+ if (!target_object->IsJSReceiver()) {
+ THROW_NEW_ERROR_RETURN_FAILURE(
+ isolate,
+ NewTypeError(
+ MessageTemplate::kWeakRefsWeakRefConstructorTargetMustBeObject));
+ }
+ isolate->heap()->AddKeepDuringJobTarget(
+ Handle<JSReceiver>::cast(target_object));
+
+ // TODO(marja): Realms.
+
+ Handle<JSObject> result;
+ ASSIGN_RETURN_FAILURE_ON_EXCEPTION(
+ isolate, result,
+ JSObject::New(target, new_target, Handle<AllocationSite>::null()));
+
+ Handle<JSWeakRef> weak_ref = Handle<JSWeakRef>::cast(result);
+ weak_ref->set_target(*target_object);
+ return *weak_ref;
+}
+
+BUILTIN(WeakRefDeref) {
+ HandleScope scope(isolate);
+ CHECK_RECEIVER(JSWeakRef, weak_ref, "WeakRef.prototype.deref");
+ if (weak_ref->target()->IsJSReceiver()) {
+ Handle<JSReceiver> target =
+ handle(JSReceiver::cast(weak_ref->target()), isolate);
+ // AddKeepDuringJobTarget might allocate and cause a GC, but it won't clear
+ // weak_ref since we hold a Handle to its target.
+ isolate->heap()->AddKeepDuringJobTarget(target);
+ } else {
+ DCHECK(weak_ref->target()->IsUndefined(isolate));
+ }
+ return weak_ref->target();
+}
+
+} // namespace internal
+} // namespace v8