diff options
Diffstat (limited to 'js/src/util/TrailingArray.h')
-rw-r--r-- | js/src/util/TrailingArray.h | 91 |
1 files changed, 91 insertions, 0 deletions
diff --git a/js/src/util/TrailingArray.h b/js/src/util/TrailingArray.h new file mode 100644 index 0000000000..bfda9931d1 --- /dev/null +++ b/js/src/util/TrailingArray.h @@ -0,0 +1,91 @@ +/* -*- 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. +template <typename Base> +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>(static_cast<Base*>(this)); + return reinterpret_cast<T*>(base + offset); + } + template <typename T> + const T* offsetToPointer(Offset offset) const { + uintptr_t base = + reinterpret_cast<uintptr_t>(static_cast<const Base*>(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>(static_cast<Base*>(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 |