Viewing file: v8-memory-span.h (10.5 KB) -rw-r--r-- Select action/file-type: (+) | (+) | (+) | Code (+) | Session (+) | (+) | SDB (+) | (+) | (+) | (+) | (+) | (+) |
// 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_
|