blob: 787cfdd4e0c44cae2f0207cfabe0a920d0cdf4e5 [file] [log] [blame]
// -*- C++ -*-
//===------------------------------ span ---------------------------------===//
//
// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
// See https://llvm.org/LICENSE.txt for license information.
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
//
//===---------------------------------------------------------------------===//
// UNSUPPORTED: c++98, c++03, c++11, c++14, c++17
// <span>
// template<class ElementType, size_t Extent = dynamic_extent>
// class span {
// public:
// // constants and types
// using element_type = ElementType;
// using value_type = remove_cv_t<ElementType>;
// using index_type = size_t;
// using difference_type = ptrdiff_t;
// using pointer = element_type *;
// using reference = element_type &;
// using const_pointe = const element_type *;
// using const_reference = const element_type &;
// using iterator = implementation-defined;
// using const_iterator = implementation-defined;
// using reverse_iterator = std::reverse_iterator<iterator>;
// using const_reverse_iterator = std::reverse_iterator<const_iterator>;
//
// static constexpr index_type extent = Extent;
//
#include <span>
#include <cassert>
#include <iterator>
#include <string>
#include "test_macros.h"
template <typename S, typename Iter>
void testIterator()
{
typedef std::iterator_traits<Iter> ItT;
ASSERT_SAME_TYPE(typename ItT::iterator_category, std::random_access_iterator_tag);
ASSERT_SAME_TYPE(typename ItT::value_type, typename S::value_type);
ASSERT_SAME_TYPE(typename ItT::reference, typename S::reference);
ASSERT_SAME_TYPE(typename ItT::pointer, typename S::pointer);
ASSERT_SAME_TYPE(typename ItT::difference_type, typename S::difference_type);
}
template <typename S, typename Iter>
void testConstIterator()
{
typedef std::iterator_traits<Iter> ItT;
ASSERT_SAME_TYPE(typename ItT::iterator_category, std::random_access_iterator_tag);
ASSERT_SAME_TYPE(typename ItT::value_type, typename S::value_type);
// I'd like to say 'const typename S::pointer' here, but that gives me
// a const pointer to a non-const value, which is not what I want.
ASSERT_SAME_TYPE(typename ItT::reference, typename S::element_type const &);
ASSERT_SAME_TYPE(typename ItT::pointer, typename S::element_type const *);
ASSERT_SAME_TYPE(typename ItT::difference_type, typename S::difference_type);
}
template <typename S, typename ElementType, std::size_t Size>
void testSpan()
{
ASSERT_SAME_TYPE(typename S::element_type, ElementType);
ASSERT_SAME_TYPE(typename S::value_type, std::remove_cv_t<ElementType>);
ASSERT_SAME_TYPE(typename S::index_type, std::size_t);
ASSERT_SAME_TYPE(typename S::difference_type, std::ptrdiff_t);
ASSERT_SAME_TYPE(typename S::pointer, ElementType *);
ASSERT_SAME_TYPE(typename S::const_pointer, const ElementType *);
ASSERT_SAME_TYPE(typename S::reference, ElementType &);
ASSERT_SAME_TYPE(typename S::const_reference, const ElementType &);
static_assert(S::extent == Size); // check that it exists
testIterator<S, typename S::iterator>();
testIterator<S, typename S::reverse_iterator>();
testConstIterator<S, typename S::const_iterator>();
testConstIterator<S, typename S::const_reverse_iterator>();
}
template <typename T>
void test()
{
testSpan<std::span< T>, T, std::dynamic_extent>();
testSpan<std::span<const T>, const T, std::dynamic_extent>();
testSpan<std::span< volatile T>, volatile T, std::dynamic_extent>();
testSpan<std::span<const volatile T>, const volatile T, std::dynamic_extent>();
testSpan<std::span< T, 5>, T, 5>();
testSpan<std::span<const T, 5>, const T, 5>();
testSpan<std::span< volatile T, 5>, volatile T, 5>();
testSpan<std::span<const volatile T, 5>, const volatile T, 5>();
}
struct A{};
int main(int, char**)
{
test<int>();
test<long>();
test<double>();
test<std::string>();
test<A>();
return 0;
}