summaryrefslogtreecommitdiffstats
path: root/js/src/vm/Realm-inl.h
diff options
context:
space:
mode:
Diffstat (limited to 'js/src/vm/Realm-inl.h')
-rw-r--r--js/src/vm/Realm-inl.h110
1 files changed, 110 insertions, 0 deletions
diff --git a/js/src/vm/Realm-inl.h b/js/src/vm/Realm-inl.h
new file mode 100644
index 0000000000..698d0b3f74
--- /dev/null
+++ b/js/src/vm/Realm-inl.h
@@ -0,0 +1,110 @@
+/* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 2 -*-
+ * vim: set ts=8 sts=2 et sw=2 tw=80:
+ * This Source Code Form is subject to the terms of the Mozilla Public
+ * License, v. 2.0. If a copy of the MPL was not distributed with this
+ * file, You can obtain one at http://mozilla.org/MPL/2.0/. */
+
+#ifndef vm_Realm_inl_h
+#define vm_Realm_inl_h
+
+#include "vm/Realm.h"
+
+#include "gc/Barrier.h"
+#include "gc/Marking.h"
+#include "vm/GlobalObject.h"
+
+#include "vm/JSContext-inl.h"
+
+inline void JS::Realm::initGlobal(js::GlobalObject& global) {
+ MOZ_ASSERT(global.realm() == this);
+ MOZ_ASSERT(!global_);
+ global_.set(&global);
+}
+
+js::GlobalObject* JS::Realm::maybeGlobal() const {
+ MOZ_ASSERT_IF(global_, global_->realm() == this);
+ return global_;
+}
+
+inline bool JS::Realm::hasLiveGlobal() const {
+ // The global is swept by traceWeakGlobalEdge when we start sweeping a zone
+ // group. This frees the GlobalObjectData, so the realm must live at least as
+ // long as the global.
+ MOZ_ASSERT_IF(global_, !js::gc::IsAboutToBeFinalized(global_));
+ return bool(global_);
+}
+
+inline bool JS::Realm::hasInitializedGlobal() const {
+ return hasLiveGlobal() && !initializingGlobal_;
+}
+
+inline bool JS::Realm::marked() const {
+ // The Realm survives in the following cases:
+ // - its global is live
+ // - it has been entered (to ensure we don't destroy the Realm while we're
+ // allocating its global)
+ // - it was allocated after the start of an incremental GC (as there may be
+ // pointers to it from other GC things)
+ return hasLiveGlobal() || hasBeenEnteredIgnoringJit() ||
+ allocatedDuringIncrementalGC_;
+}
+
+/* static */ inline js::ObjectRealm& js::ObjectRealm::get(const JSObject* obj) {
+ // Note: obj might be a CCW if we're accessing ObjectRealm::enumerators.
+ // CCWs here are fine because we always return the same ObjectRealm for a
+ // particular (CCW) object.
+ return obj->maybeCCWRealm()->objects_;
+}
+
+template <typename T>
+js::AutoRealm::AutoRealm(JSContext* cx, const T& target)
+ : cx_(cx), origin_(cx->realm()) {
+ cx_->enterRealmOf(target);
+}
+
+// Protected constructor that bypasses assertions in enterRealmOf.
+js::AutoRealm::AutoRealm(JSContext* cx, JS::Realm* target)
+ : cx_(cx), origin_(cx->realm()) {
+ cx_->enterRealm(target);
+}
+
+js::AutoRealm::~AutoRealm() { cx_->leaveRealm(origin_); }
+
+js::AutoFunctionOrCurrentRealm::AutoFunctionOrCurrentRealm(JSContext* cx,
+ HandleObject fun) {
+ JS::Realm* realm = JS::GetFunctionRealm(cx, fun);
+ if (!realm) {
+ cx->clearPendingException();
+ return;
+ }
+
+ // Enter the function's realm.
+ ar_.emplace(cx, realm);
+}
+
+js::AutoAllocInAtomsZone::AutoAllocInAtomsZone(JSContext* cx)
+ : cx_(cx), origin_(cx->realm()) {
+ cx_->enterAtomsZone();
+}
+
+js::AutoAllocInAtomsZone::~AutoAllocInAtomsZone() {
+ cx_->leaveAtomsZone(origin_);
+}
+
+js::AutoMaybeLeaveAtomsZone::AutoMaybeLeaveAtomsZone(JSContext* cx)
+ : cx_(cx), wasInAtomsZone_(cx->zone() && cx->zone()->isAtomsZone()) {
+ if (wasInAtomsZone_) {
+ cx_->leaveAtomsZone(nullptr);
+ }
+}
+
+js::AutoMaybeLeaveAtomsZone::~AutoMaybeLeaveAtomsZone() {
+ if (wasInAtomsZone_) {
+ cx_->enterAtomsZone();
+ }
+}
+
+js::AutoRealmUnchecked::AutoRealmUnchecked(JSContext* cx, JS::Realm* target)
+ : AutoRealm(cx, target) {}
+
+#endif /* vm_Realm_inl_h */