File Explorer

/proc/thread-self/root/proc/self/root/proc/12/root/node24/include/node

This explorer reads the filesystem of the server it runs on, so /workspace/user isn't present here. Browsing and the terminal still work against this server's own disk from /.

v8-memory-span.h10.5 KB · 324 lines
// 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_