10#include <userver/utils/assert.hpp>
12USERVER_NAMESPACE_BEGIN
19struct TypeIdentityImpl final {
24using TypeIdentity =
typename TypeIdentityImpl<T>::type;
33 using value_type = std::remove_cv_t<T>;
35 constexpr span()
noexcept : span(
nullptr,
nullptr) {}
37 constexpr span(T* begin, T* end)
noexcept : begin_(begin), end_(end) {
38 UASSERT((begin !=
nullptr && end !=
nullptr && begin <= end) || (begin ==
nullptr && end ==
nullptr));
41 constexpr span(T* begin, std::size_t size)
noexcept : begin_(begin), end_(begin + size) {
42 UASSERT(begin !=
nullptr || size == 0);
45#if defined(__GNUC__
) && !defined(__clang__
)
47#pragma GCC diagnostic push
48#pragma GCC diagnostic ignored "-Winit-list-lifetime"
51 template <
typename Void =
void,
typename = std::enable_if_t<std::is_const_v<T> && std::is_void_v<Void>>>
52 constexpr span(std::initializer_list<value_type> il)
56#if defined(__GNUC__
) && !defined(__clang__
)
58#pragma GCC diagnostic pop
64 typename = std::enable_if_t<
66 (std::is_lvalue_reference_v<Container> || std::is_const_v<T>)&&
68 !std::is_same_v<std::remove_cv_t<std::remove_reference_t<Container>>, span> &&
70 std::is_convertible_v<
decltype(std::data(std::declval<Container&>())), T*>>>
72 constexpr span(Container&& cont)
noexcept : span(std::data(cont), std::size(cont)) {}
74 template <std::size_t Size>
75 constexpr span(
impl::TypeIdentity<T> (&array)[Size])
noexcept
76 : span(std::data(array), std::size(array)) {}
78 constexpr T* begin()
const noexcept {
return begin_; }
79 constexpr T* end()
const noexcept {
return end_; }
81 constexpr T* data()
const noexcept {
return begin_; }
82 constexpr std::size_t size()
const noexcept {
return end_ - begin_; }
83 constexpr bool empty()
const noexcept {
return size() == 0; }
85 constexpr span<T> first(std::size_t count)
const noexcept {
87 return span{begin_, begin_ + count};
90 constexpr span<T> last(std::size_t count)
const noexcept {
92 return span{end_ - count, end_};
95 constexpr span<T> subspan(std::size_t offset)
const noexcept {
97 return span{begin_ + offset, end_};
100 constexpr span<T> subspan(
104 UASSERT(offset + count <= size());
105 return span{begin_ + offset, begin_ + offset + count};
108 constexpr T& operator[](std::size_t index)
const noexcept {
110 return begin_[index];
118template <
typename Container>
119span(Container&& cont) -> span<std::remove_reference_t<
decltype(*std::begin(cont))>>;
123span<
const std::byte>
as_bytes(span<T> s)
noexcept {
124 const auto*
const data =
reinterpret_cast<
const std::byte*>(s.data());
125 return {data, data + s.size() *
sizeof(T)};
129template <
typename T,
typename = std::enable_if_t<!std::is_const_v<T>>>
131 auto*
const data =
reinterpret_cast<std::byte*>(s.data());
132 return {data, data + s.size() *
sizeof(T)};
145template <
typename T,
typename Enabler>
146struct range_const_iterator;
149struct range_const_iterator<USERVER_NAMESPACE::utils::span<T>,
void> {
150 using type =
typename USERVER_NAMESPACE::utils::span<T>::iterator;