summaryrefslogtreecommitdiffstats
path: root/js/src/gc/Allocator-inl.h
diff options
context:
space:
mode:
authorDaniel Baumann <daniel.baumann@progress-linux.org>2024-04-19 00:47:55 +0000
committerDaniel Baumann <daniel.baumann@progress-linux.org>2024-04-19 00:47:55 +0000
commit26a029d407be480d791972afb5975cf62c9360a6 (patch)
treef435a8308119effd964b339f76abb83a57c29483 /js/src/gc/Allocator-inl.h
parentInitial commit. (diff)
downloadfirefox-26a029d407be480d791972afb5975cf62c9360a6.tar.xz
firefox-26a029d407be480d791972afb5975cf62c9360a6.zip
Adding upstream version 124.0.1.upstream/124.0.1
Signed-off-by: Daniel Baumann <daniel.baumann@progress-linux.org>
Diffstat (limited to 'js/src/gc/Allocator-inl.h')
-rw-r--r--js/src/gc/Allocator-inl.h162
1 files changed, 162 insertions, 0 deletions
diff --git a/js/src/gc/Allocator-inl.h b/js/src/gc/Allocator-inl.h
new file mode 100644
index 0000000000..2fb0220925
--- /dev/null
+++ b/js/src/gc/Allocator-inl.h
@@ -0,0 +1,162 @@
+/* -*- 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/. */
+
+/*
+ * Inline definitions of the CellAllocator methods.
+ *
+ * This is included from JSContext-inl.h for the definiton of JSContext::newCell
+ * and shouldn't need to be included elsewhere.
+ */
+
+#ifndef gc_Allocator_inl_h
+#define gc_Allocator_inl_h
+
+#include "gc/Allocator.h"
+
+#include "gc/Cell.h"
+#include "gc/Zone.h"
+#include "js/Class.h"
+#include "js/RootingAPI.h"
+
+#include "gc/Nursery-inl.h"
+
+namespace js {
+namespace gc {
+
+template <typename T, AllowGC allowGC, typename... Args>
+T* CellAllocator::NewCell(JSContext* cx, Args&&... args) {
+ static_assert(std::is_base_of_v<gc::Cell, T>);
+
+ // Objects. See the valid parameter list in NewObject, above.
+ if constexpr (std::is_base_of_v<JSObject, T>) {
+ return NewObject<T, allowGC>(cx, std::forward<Args>(args)...);
+ }
+
+ // BigInt
+ else if constexpr (std::is_base_of_v<JS::BigInt, T>) {
+ return NewBigInt<T, allowGC>(cx, std::forward<Args>(args)...);
+ }
+
+ // "Normal" strings (all of which can be nursery allocated). Atoms and
+ // external strings will fall through to the generic code below. All other
+ // strings go through NewString, which will forward the arguments to the
+ // appropriate string class's constructor.
+ else if constexpr (std::is_base_of_v<JSString, T> &&
+ !std::is_base_of_v<JSAtom, T> &&
+ !std::is_base_of_v<JSExternalString, T>) {
+ return NewString<T, allowGC>(cx, std::forward<Args>(args)...);
+ }
+
+ else {
+ // Allocate a new tenured GC thing that's not nursery-allocatable. Use
+ // cx->newCell<T>(...), where the parameters are forwarded to the type's
+ // constructor.
+ return NewTenuredCell<T, allowGC>(cx, std::forward<Args>(args)...);
+ }
+}
+
+template <typename T, AllowGC allowGC, typename... Args>
+/* static */
+T* CellAllocator::NewString(JSContext* cx, gc::Heap heap, Args&&... args) {
+ static_assert(std::is_base_of_v<JSString, T>);
+ gc::AllocKind kind = gc::MapTypeToAllocKind<T>::kind;
+ void* ptr = AllocNurseryOrTenuredCell<JS::TraceKind::String, allowGC>(
+ cx, kind, sizeof(T), heap, nullptr);
+ if (MOZ_UNLIKELY(!ptr)) {
+ return nullptr;
+ }
+ return new (mozilla::KnownNotNull, ptr) T(std::forward<Args>(args)...);
+}
+
+template <typename T, AllowGC allowGC>
+/* static */
+T* CellAllocator::NewBigInt(JSContext* cx, Heap heap) {
+ void* ptr = AllocNurseryOrTenuredCell<JS::TraceKind::BigInt, allowGC>(
+ cx, gc::AllocKind::BIGINT, sizeof(T), heap, nullptr);
+ if (MOZ_UNLIKELY(!ptr)) {
+ return nullptr;
+ }
+ return new (mozilla::KnownNotNull, ptr) T();
+}
+
+template <typename T, AllowGC allowGC>
+/* static */
+T* CellAllocator::NewObject(JSContext* cx, gc::AllocKind kind, gc::Heap heap,
+ const JSClass* clasp, gc::AllocSite* site) {
+ MOZ_ASSERT(IsObjectAllocKind(kind));
+ MOZ_ASSERT_IF(heap != gc::Heap::Tenured && clasp->hasFinalize() &&
+ !clasp->isProxyObject(),
+ CanNurseryAllocateFinalizedClass(clasp));
+ size_t thingSize = JSObject::thingSize(kind);
+ void* cell = AllocNurseryOrTenuredCell<JS::TraceKind::Object, allowGC>(
+ cx, kind, thingSize, heap, site);
+ if (MOZ_UNLIKELY(!cell)) {
+ return nullptr;
+ }
+ return new (mozilla::KnownNotNull, cell) T();
+}
+
+template <JS::TraceKind traceKind, AllowGC allowGC>
+/* static */
+void* CellAllocator::AllocNurseryOrTenuredCell(JSContext* cx,
+ gc::AllocKind allocKind,
+ size_t thingSize, gc::Heap heap,
+ AllocSite* site) {
+ MOZ_ASSERT(IsNurseryAllocable(allocKind));
+ MOZ_ASSERT(MapAllocToTraceKind(allocKind) == traceKind);
+ MOZ_ASSERT(thingSize == Arena::thingSize(allocKind));
+ MOZ_ASSERT_IF(site && site->initialHeap() == Heap::Tenured,
+ heap == Heap::Tenured);
+
+ if (!PreAllocChecks<allowGC>(cx, allocKind)) {
+ return nullptr;
+ }
+
+ JS::Zone* zone = cx->zone();
+ gc::Heap minHeapToTenure = CheckedHeap(zone->minHeapToTenure(traceKind));
+ if (CheckedHeap(heap) < minHeapToTenure) {
+ if (!site) {
+ site = zone->unknownAllocSite(traceKind);
+ }
+
+ void* ptr = cx->nursery().tryAllocateCell(site, thingSize, traceKind);
+ if (MOZ_LIKELY(ptr)) {
+ return ptr;
+ }
+
+ return RetryNurseryAlloc<allowGC>(cx, traceKind, allocKind, thingSize,
+ site);
+ }
+
+ return TryNewTenuredCell<allowGC>(cx, allocKind, thingSize);
+}
+
+/* static */
+MOZ_ALWAYS_INLINE gc::Heap CellAllocator::CheckedHeap(gc::Heap heap) {
+ if (heap > Heap::Tenured) {
+ // This helps the compiler to see that nursery allocation is never
+ // possible if Heap::Tenured is specified.
+ MOZ_MAKE_COMPILER_ASSUME_IS_UNREACHABLE("Bad gc::Heap value");
+ }
+
+ return heap;
+}
+
+template <typename T, AllowGC allowGC, typename... Args>
+/* static */
+T* CellAllocator::NewTenuredCell(JSContext* cx, Args&&... args) {
+ gc::AllocKind kind = gc::MapTypeToAllocKind<T>::kind;
+ void* cell = AllocTenuredCell<allowGC>(cx, kind, sizeof(T));
+ if (MOZ_UNLIKELY(!cell)) {
+ return nullptr;
+ }
+ return new (mozilla::KnownNotNull, cell) T(std::forward<Args>(args)...);
+}
+
+} // namespace gc
+} // namespace js
+
+#endif // gc_Allocator_inl_h