PR-URL: https://github.com/nodejs/node/pull/58070 Reviewed-By: Antoine du Hamel <duhamelantoine1995@gmail.com> Reviewed-By: Darshan Sen <raisinten@gmail.com> Reviewed-By: Joyee Cheung <joyeec9h3@gmail.com> Reviewed-By: Rafael Gonzaga <rafael.nunu@hotmail.com>
324 lines
10 KiB
C++
324 lines
10 KiB
C++
// Copyright 2021 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.
|
|
|
|
#ifndef INCLUDE_V8_MEMORY_SPAN_H_
|
|
#define INCLUDE_V8_MEMORY_SPAN_H_
|
|
|
|
#include <stddef.h>
|
|
|
|
#include <array>
|
|
#include <cstddef>
|
|
#include <iterator>
|
|
#include <type_traits>
|
|
|
|
#include "v8config.h" // NOLINT(build/include_directory)
|
|
|
|
// TODO(pkasting): Use <compare>/spaceship unconditionally after dropping
|
|
// support for old libstdc++ versions.
|
|
#if __has_include(<version>)
|
|
#include <version>
|
|
#endif
|
|
#if defined(__cpp_lib_three_way_comparison) && \
|
|
__cpp_lib_three_way_comparison >= 201711L
|
|
#define V8_HAVE_SPACESHIP_OPERATOR 1
|
|
#else
|
|
#define V8_HAVE_SPACESHIP_OPERATOR 0
|
|
#endif
|
|
|
|
// TODO(pkasting): Make this block unconditional after dropping support for old
|
|
// libstdc++ versions.
|
|
#if __has_include(<ranges>)
|
|
#include <ranges>
|
|
|
|
namespace v8 {
|
|
|
|
template <typename T>
|
|
class V8_EXPORT MemorySpan;
|
|
|
|
} // namespace v8
|
|
|
|
// Mark `MemorySpan` as satisfying the `view` and `borrowed_range` concepts.
|
|
// This should be done before the definition of `MemorySpan`, so that any
|
|
// inlined calls to range functionality use the correct specializations.
|
|
template <typename T>
|
|
inline constexpr bool std::ranges::enable_view<v8::MemorySpan<T>> = true;
|
|
template <typename T>
|
|
inline constexpr bool std::ranges::enable_borrowed_range<v8::MemorySpan<T>> =
|
|
true;
|
|
#endif
|
|
|
|
namespace v8 {
|
|
|
|
/**
|
|
* Points to an unowned contiguous buffer holding a known number of elements.
|
|
*
|
|
* This is similar to std::span (under consideration for C++20), but does not
|
|
* require advanced C++ support. In the (far) future, this may be replaced with
|
|
* or aliased to std::span.
|
|
*
|
|
* To facilitate future migration, this class exposes a subset of the interface
|
|
* implemented by std::span.
|
|
*/
|
|
template <typename T>
|
|
class V8_EXPORT MemorySpan {
|
|
private:
|
|
/** Some C++ machinery, brought from the future. */
|
|
template <typename From, typename To>
|
|
using is_array_convertible = std::is_convertible<From (*)[], To (*)[]>;
|
|
template <typename From, typename To>
|
|
static constexpr bool is_array_convertible_v =
|
|
is_array_convertible<From, To>::value;
|
|
|
|
template <typename It>
|
|
using iter_reference_t = decltype(*std::declval<It&>());
|
|
|
|
template <typename It, typename = void>
|
|
struct is_compatible_iterator : std::false_type {};
|
|
template <typename It>
|
|
struct is_compatible_iterator<
|
|
It,
|
|
std::void_t<
|
|
std::is_base_of<std::random_access_iterator_tag,
|
|
typename std::iterator_traits<It>::iterator_category>,
|
|
is_array_convertible<std::remove_reference_t<iter_reference_t<It>>,
|
|
T>>> : std::true_type {};
|
|
template <typename It>
|
|
static constexpr bool is_compatible_iterator_v =
|
|
is_compatible_iterator<It>::value;
|
|
|
|
template <typename U>
|
|
[[nodiscard]] static constexpr U* to_address(U* p) noexcept {
|
|
return p;
|
|
}
|
|
|
|
template <typename It,
|
|
typename = std::void_t<decltype(std::declval<It&>().operator->())>>
|
|
[[nodiscard]] static constexpr auto to_address(It it) noexcept {
|
|
return it.operator->();
|
|
}
|
|
|
|
public:
|
|
/** The default constructor creates an empty span. */
|
|
constexpr MemorySpan() = default;
|
|
|
|
/** Constructor from nullptr and count, for backwards compatibility.
|
|
* This is not compatible with C++20 std::span.
|
|
*/
|
|
constexpr MemorySpan(std::nullptr_t, size_t) {}
|
|
|
|
/** Constructor from "iterator" and count. */
|
|
template <typename Iterator,
|
|
std::enable_if_t<is_compatible_iterator_v<Iterator>, bool> = true>
|
|
constexpr MemorySpan(Iterator first,
|
|
size_t count) // NOLINT(runtime/explicit)
|
|
: data_(to_address(first)), size_(count) {}
|
|
|
|
/** Constructor from two "iterators". */
|
|
template <typename Iterator,
|
|
std::enable_if_t<is_compatible_iterator_v<Iterator> &&
|
|
!std::is_convertible_v<Iterator, size_t>,
|
|
bool> = true>
|
|
constexpr MemorySpan(Iterator first,
|
|
Iterator last) // NOLINT(runtime/explicit)
|
|
: data_(to_address(first)), size_(last - first) {}
|
|
|
|
/** Implicit conversion from C-style array. */
|
|
template <size_t N>
|
|
constexpr MemorySpan(T (&a)[N]) noexcept // NOLINT(runtime/explicit)
|
|
: data_(a), size_(N) {}
|
|
|
|
/** Implicit conversion from std::array. */
|
|
template <typename U, size_t N,
|
|
std::enable_if_t<is_array_convertible_v<U, T>, bool> = true>
|
|
constexpr MemorySpan(
|
|
std::array<U, N>& a) noexcept // NOLINT(runtime/explicit)
|
|
: data_(a.data()), size_{N} {}
|
|
|
|
/** Implicit conversion from const std::array. */
|
|
template <typename U, size_t N,
|
|
std::enable_if_t<is_array_convertible_v<const U, T>, bool> = true>
|
|
constexpr MemorySpan(
|
|
const std::array<U, N>& a) noexcept // NOLINT(runtime/explicit)
|
|
: data_(a.data()), size_{N} {}
|
|
|
|
/** Returns a pointer to the beginning of the buffer. */
|
|
[[nodiscard]] constexpr T* data() const { return data_; }
|
|
/** Returns the number of elements that the buffer holds. */
|
|
[[nodiscard]] constexpr size_t size() const { return size_; }
|
|
|
|
[[nodiscard]] constexpr T& operator[](size_t i) const { return data_[i]; }
|
|
|
|
/** Returns true if the buffer is empty. */
|
|
[[nodiscard]] constexpr bool empty() const { return size() == 0; }
|
|
|
|
class Iterator {
|
|
public:
|
|
using difference_type = std::ptrdiff_t;
|
|
using value_type = T;
|
|
using pointer = value_type*;
|
|
using reference = value_type&;
|
|
using iterator_category = std::random_access_iterator_tag;
|
|
// There seems to be no feature-test macro covering this, so use the
|
|
// presence of `<ranges>` as a crude proxy, since it was added to the
|
|
// standard as part of the Ranges papers.
|
|
// TODO(pkasting): Add this unconditionally after dropping support for old
|
|
// libstdc++ versions.
|
|
#if __has_include(<ranges>)
|
|
using iterator_concept = std::contiguous_iterator_tag;
|
|
#endif
|
|
|
|
// Required to satisfy `std::semiregular<>`.
|
|
constexpr Iterator() = default;
|
|
|
|
[[nodiscard]] friend constexpr bool operator==(const Iterator& a,
|
|
const Iterator& b) {
|
|
// TODO(pkasting): Replace this body with `= default` after dropping
|
|
// support for old gcc versions.
|
|
return a.ptr_ == b.ptr_;
|
|
}
|
|
#if V8_HAVE_SPACESHIP_OPERATOR
|
|
[[nodiscard]] friend constexpr auto operator<=>(const Iterator&,
|
|
const Iterator&) = default;
|
|
#else
|
|
// Assume that if spaceship isn't present, operator rewriting might not be
|
|
// either.
|
|
[[nodiscard]] friend constexpr bool operator!=(const Iterator& a,
|
|
const Iterator& b) {
|
|
return a.ptr_ != b.ptr_;
|
|
}
|
|
|
|
[[nodiscard]] friend constexpr bool operator<(const Iterator& a,
|
|
const Iterator& b) {
|
|
return a.ptr_ < b.ptr_;
|
|
}
|
|
[[nodiscard]] friend constexpr bool operator<=(const Iterator& a,
|
|
const Iterator& b) {
|
|
return a.ptr_ <= b.ptr_;
|
|
}
|
|
[[nodiscard]] friend constexpr bool operator>(const Iterator& a,
|
|
const Iterator& b) {
|
|
return a.ptr_ > b.ptr_;
|
|
}
|
|
[[nodiscard]] friend constexpr bool operator>=(const Iterator& a,
|
|
const Iterator& b) {
|
|
return a.ptr_ >= b.ptr_;
|
|
}
|
|
#endif
|
|
|
|
constexpr Iterator& operator++() {
|
|
++ptr_;
|
|
return *this;
|
|
}
|
|
|
|
constexpr Iterator operator++(int) {
|
|
Iterator temp = *this;
|
|
++*this;
|
|
return temp;
|
|
}
|
|
|
|
constexpr Iterator& operator--() {
|
|
--ptr_;
|
|
return *this;
|
|
}
|
|
|
|
constexpr Iterator operator--(int) {
|
|
Iterator temp = *this;
|
|
--*this;
|
|
return temp;
|
|
}
|
|
|
|
constexpr Iterator& operator+=(difference_type rhs) {
|
|
ptr_ += rhs;
|
|
return *this;
|
|
}
|
|
|
|
[[nodiscard]] friend constexpr Iterator operator+(Iterator lhs,
|
|
difference_type rhs) {
|
|
lhs += rhs;
|
|
return lhs;
|
|
}
|
|
|
|
[[nodiscard]] friend constexpr Iterator operator+(difference_type lhs,
|
|
const Iterator& rhs) {
|
|
return rhs + lhs;
|
|
}
|
|
|
|
constexpr Iterator& operator-=(difference_type rhs) {
|
|
ptr_ -= rhs;
|
|
return *this;
|
|
}
|
|
|
|
[[nodiscard]] friend constexpr Iterator operator-(Iterator lhs,
|
|
difference_type rhs) {
|
|
lhs -= rhs;
|
|
return lhs;
|
|
}
|
|
|
|
[[nodiscard]] friend constexpr difference_type operator-(
|
|
const Iterator& lhs, const Iterator& rhs) {
|
|
return lhs.ptr_ - rhs.ptr_;
|
|
}
|
|
|
|
[[nodiscard]] constexpr reference operator*() const { return *ptr_; }
|
|
[[nodiscard]] constexpr pointer operator->() const { return ptr_; }
|
|
[[nodiscard]] constexpr reference operator[](size_t offset) const {
|
|
return ptr_[offset];
|
|
}
|
|
|
|
private:
|
|
friend class MemorySpan<T>;
|
|
|
|
constexpr explicit Iterator(T* ptr) : ptr_(ptr) {}
|
|
|
|
T* ptr_ = nullptr;
|
|
};
|
|
|
|
[[nodiscard]] Iterator begin() const { return Iterator(data_); }
|
|
[[nodiscard]] Iterator end() const { return Iterator(data_ + size_); }
|
|
|
|
private:
|
|
T* data_ = nullptr;
|
|
size_t size_ = 0;
|
|
};
|
|
|
|
/**
|
|
* Helper function template to create an array of fixed length, initialized by
|
|
* the provided initializer list, without explicitly specifying the array size,
|
|
* e.g.
|
|
*
|
|
* auto arr = v8::to_array<Local<String>>({v8_str("one"), v8_str("two")});
|
|
*
|
|
* In the future, this may be replaced with or aliased to std::to_array (under
|
|
* consideration for C++20).
|
|
*/
|
|
|
|
namespace detail {
|
|
template <class T, std::size_t N, std::size_t... I>
|
|
[[nodiscard]] constexpr std::array<std::remove_cv_t<T>, N> to_array_lvalue_impl(
|
|
T (&a)[N], std::index_sequence<I...>) {
|
|
return {{a[I]...}};
|
|
}
|
|
|
|
template <class T, std::size_t N, std::size_t... I>
|
|
[[nodiscard]] constexpr std::array<std::remove_cv_t<T>, N> to_array_rvalue_impl(
|
|
T (&&a)[N], std::index_sequence<I...>) {
|
|
return {{std::move(a[I])...}};
|
|
}
|
|
} // namespace detail
|
|
|
|
template <class T, std::size_t N>
|
|
[[nodiscard]] constexpr std::array<std::remove_cv_t<T>, N> to_array(T (&a)[N]) {
|
|
return detail::to_array_lvalue_impl(a, std::make_index_sequence<N>{});
|
|
}
|
|
|
|
template <class T, std::size_t N>
|
|
[[nodiscard]] constexpr std::array<std::remove_cv_t<T>, N> to_array(
|
|
T (&&a)[N]) {
|
|
return detail::to_array_rvalue_impl(std::move(a),
|
|
std::make_index_sequence<N>{});
|
|
}
|
|
|
|
} // namespace v8
|
|
#endif // INCLUDE_V8_MEMORY_SPAN_H_
|