// Copyright (C) 2019 T. Zachary Laine // // Distributed under the Boost Software License, Version 1.0. (See // accompanying file LICENSE_1_0.txt or copy at // http://www.boost.org/LICENSE_1_0.txt) #include #include "ill_formed.hpp" #include #include #include #include #include template using decrementable_t = decltype(--std::declval()); struct basic_forward_iter : boost::stl_interfaces:: iterator_interface { basic_forward_iter() : it_(nullptr) {} basic_forward_iter(int * it) : it_(it) {} int & operator*() const { return *it_; } basic_forward_iter & operator++() { ++it_; return *this; } friend bool operator==(basic_forward_iter lhs, basic_forward_iter rhs) noexcept { return lhs.it_ == rhs.it_; } using base_type = boost::stl_interfaces:: iterator_interface; using base_type::operator++; private: int * it_; }; BOOST_STL_INTERFACES_STATIC_ASSERT_CONCEPT( basic_forward_iter, std::forward_iterator) BOOST_STL_INTERFACES_STATIC_ASSERT_ITERATOR_TRAITS( basic_forward_iter, std::forward_iterator_tag, std::forward_iterator_tag, int, int &, int *, std::ptrdiff_t) static_assert(ill_formed::value, ""); template struct forward_iter : boost::stl_interfaces::iterator_interface< forward_iter, std::forward_iterator_tag, ValueType> { forward_iter() : it_(nullptr) {} forward_iter(ValueType * it) : it_(it) {} template< typename ValueType2, typename E = std::enable_if_t< std::is_convertible::value>> forward_iter(forward_iter it) : it_(it.it_) {} ValueType & operator*() const { return *it_; } forward_iter & operator++() { ++it_; return *this; } friend bool operator==(forward_iter lhs, forward_iter rhs) noexcept { return lhs.it_ == rhs.it_; } using base_type = boost::stl_interfaces::iterator_interface< forward_iter, std::forward_iterator_tag, ValueType>; using base_type::operator++; private: ValueType * it_; template friend struct forward_iter; }; using forward = forward_iter; using const_forward = forward_iter; static_assert(ill_formed::value, ""); static_assert(ill_formed::value, ""); BOOST_STL_INTERFACES_STATIC_ASSERT_CONCEPT(forward, std::forward_iterator) BOOST_STL_INTERFACES_STATIC_ASSERT_ITERATOR_TRAITS( forward, std::forward_iterator_tag, std::forward_iterator_tag, int, int &, int *, std::ptrdiff_t) BOOST_STL_INTERFACES_STATIC_ASSERT_CONCEPT(const_forward, std::forward_iterator) BOOST_STL_INTERFACES_STATIC_ASSERT_ITERATOR_TRAITS( const_forward, std::forward_iterator_tag, std::forward_iterator_tag, int, int const &, int const *, std::ptrdiff_t) std::array ints = {{0, 1, 2, 3, 4, 5, 6, 7, 8, 9}}; //////////////////// // view_interface // //////////////////// #include "view_tests.hpp" template using data_t = decltype(std::declval().data()); static_assert( ill_formed< data_t, subrange< basic_forward_iter, basic_forward_iter, boost::stl_interfaces::v1::element_layout::discontiguous>>::value, ""); static_assert( ill_formed< data_t, subrange< basic_forward_iter, basic_forward_iter, boost::stl_interfaces::v1::element_layout::discontiguous> const>:: value, ""); template using size_t_ = decltype(std::declval().size()); static_assert( ill_formed< size_t_, subrange< basic_forward_iter, basic_forward_iter, boost::stl_interfaces::v1::element_layout::discontiguous>>::value, ""); static_assert( ill_formed< size_t_, subrange< basic_forward_iter, basic_forward_iter, boost::stl_interfaces::v1::element_layout::discontiguous> const>:: value, ""); template using back_t_ = decltype(std::declval().back()); static_assert( ill_formed< back_t_, subrange< basic_forward_iter, basic_forward_iter, boost::stl_interfaces::v1::element_layout::discontiguous>>::value, ""); static_assert( ill_formed< back_t_, subrange< basic_forward_iter, basic_forward_iter, boost::stl_interfaces::v1::element_layout::discontiguous> const>:: value, ""); template using index_operator_t = decltype(std::declval()[0]); static_assert( ill_formed< index_operator_t, subrange< basic_forward_iter, basic_forward_iter, boost::stl_interfaces::v1::element_layout::discontiguous>>::value, ""); static_assert( ill_formed< index_operator_t, subrange< basic_forward_iter, basic_forward_iter, boost::stl_interfaces::v1::element_layout::discontiguous> const>:: value, ""); int main() { { basic_forward_iter first(ints.data()); basic_forward_iter last(ints.data() + ints.size()); { std::array ints_copy; std::copy(first, last, ints_copy.begin()); BOOST_TEST(ints_copy == ints); } { std::array iota_ints; basic_forward_iter first(iota_ints.data()); basic_forward_iter last(iota_ints.data() + iota_ints.size()); std::iota(first, last, 0); BOOST_TEST(iota_ints == ints); } } { forward first(ints.data()); forward last(ints.data() + ints.size()); const_forward first_copy(first); const_forward last_copy(last); std::equal(first, last, first_copy, last_copy); } { forward first(ints.data()); forward last(ints.data() + ints.size()); while (first != last) first++; } { forward first(ints.data()); forward last(ints.data() + ints.size()); { std::array ints_copy; std::copy(first, last, ints_copy.begin()); BOOST_TEST(ints_copy == ints); } { std::array iota_ints; forward first(iota_ints.data()); forward last(iota_ints.data() + iota_ints.size()); std::iota(first, last, 0); BOOST_TEST(iota_ints == ints); } } { const_forward first(ints.data()); const_forward last(ints.data() + ints.size()); { std::array ints_copy; std::copy(first, last, ints_copy.begin()); BOOST_TEST(ints_copy == ints); } { BOOST_TEST(std::binary_search(first, last, 3)); } } { basic_forward_iter first(ints.data()); basic_forward_iter last(ints.data() + ints.size()); auto r = range( first, last); auto empty = range( first, first); // range begin/end { std::array ints_copy; std::copy(r.begin(), r.end(), ints_copy.begin()); BOOST_TEST(ints_copy == ints); BOOST_TEST(empty.begin() == empty.end()); } // empty/op bool { BOOST_TEST(!r.empty()); BOOST_TEST(r); BOOST_TEST(empty.empty()); BOOST_TEST(!empty); auto const cr = r; BOOST_TEST(!cr.empty()); BOOST_TEST(cr); auto const cempty = empty; BOOST_TEST(cempty.empty()); BOOST_TEST(!cempty); } // front/back { BOOST_TEST(r.front() == 0); auto const cr = r; BOOST_TEST(cr.front() == 0); } } return boost::report_errors(); }