diff options
author | Daniel Baumann <daniel.baumann@progress-linux.org> | 2024-04-07 19:33:14 +0000 |
---|---|---|
committer | Daniel Baumann <daniel.baumann@progress-linux.org> | 2024-04-07 19:33:14 +0000 |
commit | 36d22d82aa202bb199967e9512281e9a53db42c9 (patch) | |
tree | 105e8c98ddea1c1e4784a60a5a6410fa416be2de /js/src/util/TrailingArray.h | |
parent | Initial commit. (diff) | |
download | firefox-esr-36d22d82aa202bb199967e9512281e9a53db42c9.tar.xz firefox-esr-36d22d82aa202bb199967e9512281e9a53db42c9.zip |
Adding upstream version 115.7.0esr.upstream/115.7.0esr
Signed-off-by: Daniel Baumann <daniel.baumann@progress-linux.org>
Diffstat (limited to 'js/src/util/TrailingArray.h')
-rw-r--r-- | js/src/util/TrailingArray.h | 88 |
1 files changed, 88 insertions, 0 deletions
diff --git a/js/src/util/TrailingArray.h b/js/src/util/TrailingArray.h new file mode 100644 index 0000000000..574f9aa300 --- /dev/null +++ b/js/src/util/TrailingArray.h @@ -0,0 +1,88 @@ +/* -*- 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 util_TrailingArray_h +#define util_TrailingArray_h + +#include "mozilla/Assertions.h" // MOZ_ASSERT + +#include <stddef.h> // size_t +#include <stdint.h> // uint32_t, uintptr_t + +namespace js { + +// This is a mixin class to use for types that have trailing arrays and use +// offsets to delimit them. It provides helper methods to do casting and +// initialization while avoiding C++ undefined behaviour. +class TrailingArray { + protected: + // Offsets are measured in bytes relative to 'this'. + using Offset = uint32_t; + + // Test if offset is correctly aligned for type. + template <typename T> + static constexpr bool isAlignedOffset(Offset offset) { + return offset % alignof(T) == 0; + } + template <size_t N> + static constexpr bool isAlignedOffset(Offset offset) { + return offset % N == 0; + } + + // Translate an offset into a concrete pointer. + template <typename T> + T* offsetToPointer(Offset offset) { + uintptr_t base = reinterpret_cast<uintptr_t>(this); + return reinterpret_cast<T*>(base + offset); + } + template <typename T> + const T* offsetToPointer(Offset offset) const { + uintptr_t base = reinterpret_cast<uintptr_t>(this); + return reinterpret_cast<const T*>(base + offset); + } + + // Placement-new the elements of an array. This optimizes away for types with + // trivial default initialization and plays nicely with compiler vectorization + // passes. + template <typename T> + void initElements(Offset offset, size_t nelem) { + MOZ_ASSERT(isAlignedOffset<T>(offset)); + + // Address of first array element. + uintptr_t elem = reinterpret_cast<uintptr_t>(this) + offset; + + for (size_t i = 0; i < nelem; ++i) { + void* raw = reinterpret_cast<void*>(elem); + new (raw) T; + elem += sizeof(T); + } + } + + // Compute the length of an array from its start and end offset. + template <typename T> + size_t numElements(Offset start, Offset end) const { + constexpr size_t ElemSize = sizeof(T); + return numElements<ElemSize>(start, end); + } + template <size_t ElemSize> + size_t numElements(Offset start, Offset end) const { + MOZ_ASSERT(start <= end); + MOZ_ASSERT((end - start) % ElemSize == 0); + return (end - start) / ElemSize; + } + + // Constructor is protected so a derived type is required. + TrailingArray() = default; + + public: + // Type has trailing data so isn't copyable or movable. + TrailingArray(const TrailingArray&) = delete; + TrailingArray& operator=(const TrailingArray&) = delete; +}; + +} // namespace js + +#endif // util_TrailingArray_h |