//===--- span- The span class -----------------------------------*- C++ -*-===// // // The LLVM Compiler Infrastructure // // This file is distributed under the University of Illinois Open Source // License. See LICENSE.TXT for details. // //===----------------------------------------------------------------------===// #ifndef ACXXEL_SPAN_H #define ACXXEL_SPAN_H #include #include #include #include #include namespace acxxel { /// Value used to indicate slicing to the end of the span. static constexpr std::ptrdiff_t dynamic_extent = -1; // NOLINT class SpanBase {}; /// Implementation of the proposed C++17 std::span class. /// /// Based on the paper: /// http://www.open-std.org/jtc1/sc22/wg21/docs/papers/2016/p0122r1.pdf template class Span : public SpanBase { public: /// \name constants and types /// \{ using element_type = ElementType; using index_type = std::ptrdiff_t; using pointer = element_type *; using reference = element_type &; using iterator = element_type *; using const_iterator = const element_type *; using value_type = typename std::remove_const::type; /// \} /// \name constructors, copy, assignment, and destructor. /// \{ /// Constructs an empty span with null pointer data. Span() : Data(nullptr), Size(0) {} /// Constructs an empty span with null pointer data. // Intentionally implicit. Span(std::nullptr_t) : Data(nullptr), Size(0) {} /// Constructs a span from a pointer and element count. Span(pointer Ptr, index_type Count) : Data(Ptr), Size(Count) { if (Count < 0 || (!Ptr && Count)) std::terminate(); } /// Constructs a span from a pointer to the fist element in the range and a /// pointer to one past the last element in the range. Span(pointer FirstElem, pointer LastElem) : Data(FirstElem), Size(std::distance(FirstElem, LastElem)) { if (Size < 0) std::terminate(); } /// Constructs a span from an array. // Intentionally implicit. template Span(T (&Arr)[N]) : Data(Arr), Size(N) {} /// Constructs a span from a std::array. // Intentionally implicit. template Span(const std::array::type, N> &Arr) : Data(Arr.data()), Size(N) {} /// Constructs a span from a container such as a std::vector. // TODO(jhen): Put in a check to make sure this constructor does not // participate in overload resolution unless Container meets the following // requirements: // * Container is a contiguous container and a sequence container. // Intentionally implicit. template Span(Container &Cont, typename std::enable_if< std::is_same< typename std::remove_const::type, typename std::remove_const::type>::value && !std::is_array::value && !std::is_base_of::value && std::is_convertible::value>::type * = nullptr) : Data(Cont.data()), Size(Cont.size()) {} /// Avoids creating spans from expiring temporary objects. // TODO(jhen): Put in a check to make sure this constructor does not // participate in overload resolution unless Container meets the following // requirements: // * Container is a contiguous container and a sequence container. template Span(Container &&Cont, typename std::enable_if< std::is_same< typename std::remove_const::type, typename std::remove_const::type>::value && !std::is_array::value && !std::is_base_of::value && std::is_convertible::value>::type * = nullptr) = delete; Span(const Span &) noexcept = default; Span(Span &&) noexcept; /// Constructs a span from copying a span of another type that can be /// implicitly converted to the type stored by the constructed span. // Intentionally implicit. template Span(const Span &Other) : Data(Other.Data), Size(Other.Size) {} /// Constructs a span from moving a span of another type that can be /// implicitly converted to the type stored by the constructed span. // Intentionally implicit. template Span(Span &&Other) : Data(Other.Data), Size(Other.Size) {} ~Span() = default; Span &operator=(const Span &) noexcept = default; Span &operator=(Span &&) noexcept; /// \} /// \name subviews /// \{ /// Creates a span out of the first Count elements of this span. Span first(index_type Count) const { bool Valid = Count >= 0 && Count <= size(); if (!Valid) std::terminate(); return Span(data(), Count); } /// Creates a span out of the last Count elements of this span. Span last(index_type Count) const { bool Valid = Count >= 0 && Count <= size(); if (!Valid) std::terminate(); return Span(Count == 0 ? data() : data() + (size() - Count), Count); } /// Creates a span out of the Count elements of this span beginning at Offset. /// /// If no arguments is provided for Count, the new span will extend to the end /// of the current span. Span subspan(index_type Offset, index_type Count = dynamic_extent) const { bool Valid = (Offset == 0 || (Offset > 0 && Offset <= size())) && (Count == dynamic_extent || (Count >= 0 && Offset + Count <= size())); if (!Valid) std::terminate(); return Span( data() + Offset, Count == dynamic_extent ? size() - Offset : Count); } /// \} /// \name observers /// \{ index_type length() const { return Size; } index_type size() const { return Size; } bool empty() const { return size() == 0; } /// \} /// \name element access /// \{ reference operator[](index_type Idx) const { bool Valid = Idx >= 0 && Idx < size(); if (!Valid) std::terminate(); return Data[Idx]; } reference operator()(index_type Idx) const { return operator[](Idx); } pointer data() const noexcept { return Data; } /// \} /// \name iterator support /// \{ iterator begin() const noexcept { return Data; } iterator end() const noexcept { return Data + Size; } const_iterator cbegin() const noexcept { return Data; } const_iterator cend() const noexcept { return Data + Size; } /// \} private: template friend class Span; pointer Data; index_type Size; }; template Span::Span(Span &&) noexcept = default; template Span &Span::operator=(Span &&) noexcept = default; } // namespace acxxel #endif // ACXXEL_SPAN_H