///\file /****************************************************************************** The MIT License(MIT) Embedded Template Library. https://github.com/ETLCPP/etl https://www.etlcpp.com Copyright(c) 2017 John Wellbelove Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files(the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and / or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions : The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software. THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. ******************************************************************************/ #ifndef ETL_ITERATOR_INCLUDED #define ETL_ITERATOR_INCLUDED #include "platform.h" #include "invoke.h" #include "type_traits.h" #include "utility.h" #include "private/addressof.h" #if ETL_USING_STL || defined(ETL_IN_UNIT_TEST) #include #endif ///\defgroup iterator iterator ///\ingroup utilities namespace etl { //*************************************************************************** // iterator tags struct input_iterator_tag { }; struct output_iterator_tag { }; struct forward_iterator_tag : public input_iterator_tag { }; struct bidirectional_iterator_tag : public forward_iterator_tag { }; struct random_access_iterator_tag : public bidirectional_iterator_tag { }; // struct contiguous_iterator_tag : public random_access_iterator_tag {}; //*************************************************************************** // iterator_traits // For anything not a fundamental type. template ::value, void>::type> struct iterator_traits { typedef typename TIterator::iterator_category iterator_category; typedef typename TIterator::value_type value_type; typedef typename TIterator::difference_type difference_type; typedef typename TIterator::pointer pointer; typedef typename TIterator::reference reference; }; // For pointers. template struct iterator_traits { typedef ETL_OR_STD::random_access_iterator_tag iterator_category; typedef T value_type; typedef ptrdiff_t difference_type; typedef typename etl::remove_cv::type* pointer; typedef T& reference; }; // For const pointers. template struct iterator_traits { typedef ETL_OR_STD::random_access_iterator_tag iterator_category; typedef T value_type; typedef ptrdiff_t difference_type; typedef const typename etl::remove_cv::type* pointer; typedef const T& reference; }; //*************************************************************************** // advance template ETL_CONSTEXPR14 void advance_helper(TIterator& itr, TDistance n, ETL_OR_STD::input_iterator_tag) { while (n-- > 0) { ++itr; } } template ETL_CONSTEXPR14 void advance_helper(TIterator& itr, TDistance n, ETL_OR_STD::output_iterator_tag) { while (n-- > 0) { ++itr; } } template ETL_CONSTEXPR14 void advance_helper(TIterator& itr, TDistance n, ETL_OR_STD::forward_iterator_tag) { while (n-- > 0) { ++itr; } } template ETL_CONSTEXPR14 void advance_helper(TIterator& itr, TDistance n, ETL_OR_STD::bidirectional_iterator_tag) { if (n > 0) { while (n--) { ++itr; } } else { while (n++) { --itr; } } } template ETL_CONSTEXPR14 void advance_helper(TIterator& itr, TDistance n, ETL_OR_STD::random_access_iterator_tag) { typedef typename etl::iterator_traits::difference_type diff_t; itr += static_cast(n); } template ETL_CONSTEXPR14 void advance(TIterator& itr, TDistance n) { typedef typename etl::iterator_traits::iterator_category tag; advance_helper(itr, n, tag()); } //*************************************************************************** // distance template ETL_CONSTEXPR14 typename etl::iterator_traits::difference_type distance_helper(TIterator first, TIterator last, ETL_OR_STD::input_iterator_tag) { typename etl::iterator_traits::difference_type d = 0; while (first != last) { ++d; ++first; } return d; } template ETL_CONSTEXPR14 typename etl::iterator_traits::difference_type distance_helper(TIterator first, TIterator last, ETL_OR_STD::forward_iterator_tag) { typename etl::iterator_traits::difference_type d = 0; while (first != last) { ++d; ++first; } return d; } template ETL_CONSTEXPR14 typename etl::iterator_traits::difference_type distance_helper(TIterator first, TIterator last, ETL_OR_STD::bidirectional_iterator_tag) { typename etl::iterator_traits::difference_type d = 0; while (first != last) { ++d; ++first; } return d; } template ETL_CONSTEXPR14 typename etl::iterator_traits::difference_type distance_helper(TIterator first, TIterator last, ETL_OR_STD::random_access_iterator_tag) { return last - first; } template ETL_CONSTEXPR14 typename etl::iterator_traits::difference_type distance(TIterator first, TIterator last) { typedef typename etl::iterator_traits::iterator_category tag; return distance_helper(first, last, tag()); } //*************************************************************************** // Previous template ETL_CONSTEXPR14 TIterator prev(TIterator itr, typename etl::iterator_traits::difference_type n = 1) { etl::advance(itr, -n); return itr; } //*************************************************************************** // Next template ETL_CONSTEXPR14 TIterator next(TIterator itr, typename etl::iterator_traits::difference_type n = 1) { etl::advance(itr, n); return itr; } //*************************************************************************** // reverse_iterator template class reverse_iterator { public: typedef typename iterator_traits::iterator_category iterator_category; typedef typename iterator_traits::value_type value_type; typedef typename iterator_traits::difference_type difference_type; typedef typename iterator_traits::pointer pointer; typedef typename iterator_traits::reference reference; typedef TIterator iterator_type; ETL_CONSTEXPR14 reverse_iterator() : current() { } ETL_CONSTEXPR14 explicit reverse_iterator(TIterator itr) : current(itr) { } template ETL_CONSTEXPR14 reverse_iterator(const reverse_iterator& other) : current(other.base()) { } template ETL_CONSTEXPR14 reverse_iterator& operator=(const reverse_iterator& other) { current = other.base(); return (*this); } ETL_CONSTEXPR14 TIterator base() const { return current; } ETL_NODISCARD ETL_CONSTEXPR14 reference operator*() const { TIterator temp = current; return *(--temp); } ETL_NODISCARD ETL_CONSTEXPR14 pointer operator->() const { TIterator temp = current; return &(*--temp); } ETL_CONSTEXPR14 reverse_iterator& operator++() { --current; return *this; } ETL_CONSTEXPR14 reverse_iterator operator++(int) { reverse_iterator temp = *this; --current; return temp; } ETL_CONSTEXPR14 reverse_iterator& operator--() { ++current; return (*this); } ETL_CONSTEXPR14 reverse_iterator operator--(int) { reverse_iterator temp = *this; ++current; return temp; } ETL_CONSTEXPR14 reverse_iterator& operator+=(const difference_type offset) { current -= offset; return (*this); } ETL_CONSTEXPR14 reverse_iterator& operator-=(const difference_type offset) { current += offset; return (*this); } ETL_NODISCARD ETL_CONSTEXPR14 reverse_iterator operator+(const difference_type offset) const { return reverse_iterator(current - offset); } ETL_NODISCARD ETL_CONSTEXPR14 reverse_iterator operator-(const difference_type offset) const { return (reverse_iterator(current + offset)); } ETL_NODISCARD ETL_CONSTEXPR14 reference operator[](const difference_type offset) const { return (*(*this + offset)); } protected: TIterator current; }; template ETL_CONSTEXPR14 bool operator==(const reverse_iterator& lhs, const reverse_iterator& rhs) { return lhs.base() == rhs.base(); } template ETL_CONSTEXPR14 bool operator!=(const reverse_iterator& lhs, const reverse_iterator& rhs) { return !(lhs == rhs); } template ETL_CONSTEXPR14 bool operator<(const reverse_iterator& lhs, const reverse_iterator& rhs) { return rhs.base() < lhs.base(); } template ETL_CONSTEXPR14 bool operator>(const reverse_iterator& lhs, const reverse_iterator& rhs) { return rhs < lhs; } template ETL_CONSTEXPR14 bool operator<=(const reverse_iterator& lhs, const reverse_iterator& rhs) { return !(rhs < lhs); } template ETL_CONSTEXPR14 bool operator>=(const reverse_iterator& lhs, const reverse_iterator& rhs) { return !(lhs < rhs); } template ETL_CONSTEXPR14 typename reverse_iterator::difference_type operator-(const reverse_iterator& lhs, const reverse_iterator& rhs) { return rhs.base() - lhs.base(); } template ETL_CONSTEXPR14 reverse_iterator operator+(TDifference n, const reverse_iterator& itr) { return itr.operator+(n); } //*************************************************************************** /// iterator //*************************************************************************** template struct iterator { typedef T value_type; typedef TDistance difference_type; typedef TPointer pointer; typedef TReference reference; typedef TCategory iterator_category; }; #if ETL_USING_CPP11 //*************************************************************************** // move_iterator template class move_iterator { public: typedef typename iterator_traits::iterator_category iterator_category; typedef typename iterator_traits::value_type value_type; typedef typename iterator_traits::difference_type difference_type; typedef TIterator iterator_type; typedef TIterator pointer; typedef value_type&& reference; move_iterator() {} explicit move_iterator(TIterator itr) : current(itr) { } template move_iterator(const move_iterator& itr) : current(itr.base()) { } template move_iterator& operator=(const move_iterator& itr) { current = itr.current; return *this; } iterator_type base() const { return current; } pointer operator->() const { return current; } reference operator*() const { return etl::move(*current); } move_iterator& operator++() { ++current; return *this; } move_iterator& operator--() { --current; return *this; } move_iterator operator++(int) { move_iterator temp = *this; ++current; return temp; } move_iterator operator--(int) { move_iterator temp = *this; --current; return temp; } move_iterator operator+(difference_type n) const { return move_iterator(current + n); } move_iterator operator-(difference_type n) const { return move_iterator(current - n); } move_iterator operator+=(difference_type n) { current += n; return *this; } move_iterator operator-=(difference_type n) { current -= n; return *this; } reference operator[](difference_type n) const { return etl::move(current[n]); } private: TIterator current; }; template bool operator==(const etl::move_iterator& lhs, const etl::move_iterator& rhs) { return lhs.base() == rhs.base(); } template bool operator!=(const etl::move_iterator& lhs, const etl::move_iterator& rhs) { return !(lhs == rhs); } template bool operator<(const etl::move_iterator& lhs, const etl::move_iterator& rhs) { return lhs.base() < rhs.base(); } template bool operator<=(const etl::move_iterator& lhs, const etl::move_iterator& rhs) { return !(rhs < lhs); } template bool operator>(const etl::move_iterator& lhs, const etl::move_iterator& rhs) { return (rhs < lhs); } template bool operator>=(const etl::move_iterator& lhs, const etl::move_iterator& rhs) { return !(lhs < rhs); } template move_iterator operator+(typename move_iterator::difference_type n, const move_iterator& rhs) { return rhs + n; } template auto operator-(const move_iterator& lhs, const move_iterator& rhs) -> decltype(lhs.base() - rhs.base()) { return lhs.base() - rhs.base(); } template etl::move_iterator make_move_iterator(TIterator itr) { return etl::move_iterator(itr); } #endif // ETL_USING_CPP11 //*************************************************************************** // back_insert_iterator //*************************************************************************** //*************************************************************************** /// Turns assignment into push_back //*************************************************************************** template class back_insert_iterator : public etl::iterator { public: typedef TContainer container_type; //*************************************************************************** /// Constructor //*************************************************************************** explicit ETL_CONSTEXPR14 back_insert_iterator(TContainer& c) : container(etl::addressof(c)) { } //*************************************************************************** /// Assignment operator //*************************************************************************** ETL_CONSTEXPR14 back_insert_iterator& operator=(const typename TContainer::value_type& value) { container->push_back(value); return (*this); } #if ETL_USING_CPP11 //*************************************************************************** /// Move assignment operator. //*************************************************************************** ETL_CONSTEXPR14 back_insert_iterator& operator=(typename TContainer::value_type&& value) { container->push_back(etl::move(value)); return (*this); } #endif // ETL_USING_CPP11 //*************************************************************************** /// Dereference operator. //*************************************************************************** ETL_NODISCARD ETL_CONSTEXPR14 back_insert_iterator& operator*() { return (*this); } //*************************************************************************** /// Pre-increment operator. //*************************************************************************** ETL_CONSTEXPR14 back_insert_iterator& operator++() { return (*this); } //*************************************************************************** /// Post-increment operator. //*************************************************************************** ETL_CONSTEXPR14 back_insert_iterator operator++(int) { return (*this); } protected: TContainer* container; }; //*************************************************************************** /// Creates a back_insert_iterator from a container. //*************************************************************************** template ETL_NODISCARD ETL_CONSTEXPR14 etl::back_insert_iterator back_inserter(TContainer& container) { return etl::back_insert_iterator(container); } //*************************************************************************** // front_insert_iterator //*************************************************************************** //*************************************************************************** /// Turns assignment into a push_front. //*************************************************************************** template class front_insert_iterator : public etl::iterator { public: typedef TContainer container_type; //*************************************************************************** /// Constructor //*************************************************************************** explicit ETL_CONSTEXPR14 front_insert_iterator(TContainer& c) : container(etl::addressof(c)) { } //*************************************************************************** /// Assignment operator //*************************************************************************** ETL_CONSTEXPR14 front_insert_iterator& operator=(const typename TContainer::value_type& value) { container->push_front(value); return (*this); } #if ETL_USING_CPP11 //*************************************************************************** /// Move assignment operator. //*************************************************************************** ETL_CONSTEXPR14 front_insert_iterator& operator=(typename TContainer::value_type&& value) { container->push_front(etl::move(value)); return (*this); } #endif // ETL_USING_CPP11 //*************************************************************************** /// Dereference operator. //*************************************************************************** ETL_NODISCARD ETL_CONSTEXPR14 front_insert_iterator& operator*() { return (*this); } //*************************************************************************** /// Pre-increment operator. //*************************************************************************** ETL_CONSTEXPR14 front_insert_iterator& operator++() { return (*this); } //*************************************************************************** /// Post-increment operator. //*************************************************************************** ETL_CONSTEXPR14 front_insert_iterator operator++(int) { return (*this); } protected: TContainer* container; }; //*************************************************************************** /// Creates a front_insert_iterator from a container. //*************************************************************************** template ETL_NODISCARD ETL_CONSTEXPR14 etl::front_insert_iterator front_inserter(TContainer& container) { return etl::front_insert_iterator(container); } //*************************************************************************** // push_insert_iterator //*************************************************************************** //*************************************************************************** /// Turns assignment into a push. //*************************************************************************** template class push_insert_iterator : public etl::iterator { public: typedef TContainer container_type; //*************************************************************************** /// Constructor //*************************************************************************** explicit ETL_CONSTEXPR14 push_insert_iterator(TContainer& c) : container(etl::addressof(c)) { } //*************************************************************************** /// Assignment operator //*************************************************************************** ETL_CONSTEXPR14 push_insert_iterator& operator=(const typename TContainer::value_type& value) { container->push(value); return (*this); } #if ETL_USING_CPP11 //*************************************************************************** /// Move assignment operator. //*************************************************************************** ETL_CONSTEXPR14 push_insert_iterator& operator=(typename TContainer::value_type&& value) { container->push(etl::move(value)); return (*this); } #endif // ETL_USING_CPP11 //*************************************************************************** /// Dereference operator. //*************************************************************************** ETL_NODISCARD ETL_CONSTEXPR14 push_insert_iterator& operator*() { return (*this); } //*************************************************************************** /// Pre-increment operator. //*************************************************************************** ETL_CONSTEXPR14 push_insert_iterator& operator++() { return (*this); } //*************************************************************************** /// Post-increment operator. //*************************************************************************** ETL_CONSTEXPR14 push_insert_iterator operator++(int) { return (*this); } protected: TContainer* container; }; //*************************************************************************** /// Creates a push_insert_iterator from a container. //*************************************************************************** template ETL_NODISCARD ETL_CONSTEXPR14 etl::push_insert_iterator push_inserter(TContainer& container) { return etl::push_insert_iterator(container); } //*************************************************************************** // Helper templates. //*************************************************************************** template struct is_input_iterator { static ETL_CONSTANT bool value = etl::is_same::iterator_category, ETL_OR_STD::input_iterator_tag>::value; }; template ETL_CONSTANT bool is_input_iterator::value; template struct is_output_iterator { static ETL_CONSTANT bool value = etl::is_same::iterator_category, ETL_OR_STD::output_iterator_tag>::value; }; template ETL_CONSTANT bool is_output_iterator::value; template struct is_forward_iterator { static ETL_CONSTANT bool value = etl::is_same::iterator_category, ETL_OR_STD::forward_iterator_tag>::value; }; template ETL_CONSTANT bool is_forward_iterator::value; template struct is_bidirectional_iterator { static ETL_CONSTANT bool value = etl::is_same::iterator_category, ETL_OR_STD::bidirectional_iterator_tag>::value; }; template ETL_CONSTANT bool is_bidirectional_iterator::value; // Deprecated template struct is_random_iterator { static ETL_CONSTANT bool value = etl::is_same::iterator_category, ETL_OR_STD::random_access_iterator_tag>::value; }; template ETL_CONSTANT bool is_random_iterator::value; template struct is_random_access_iterator { static ETL_CONSTANT bool value = etl::is_same::iterator_category, ETL_OR_STD::random_access_iterator_tag>::value; }; template ETL_CONSTANT bool is_random_access_iterator::value; template struct is_input_iterator_concept { static ETL_CONSTANT bool value = etl::is_input_iterator::value || etl::is_forward_iterator::value || etl::is_bidirectional_iterator::value || etl::is_random_iterator::value; }; template ETL_CONSTANT bool is_input_iterator_concept::value; template struct is_output_iterator_concept { static ETL_CONSTANT bool value = etl::is_output_iterator::value || etl::is_forward_iterator::value || etl::is_bidirectional_iterator::value || etl::is_random_iterator::value; }; template ETL_CONSTANT bool is_output_iterator_concept::value; template struct is_forward_iterator_concept { static ETL_CONSTANT bool value = etl::is_forward_iterator::value || etl::is_bidirectional_iterator::value || etl::is_random_iterator::value; }; template ETL_CONSTANT bool is_forward_iterator_concept::value; template struct is_bidirectional_iterator_concept { static ETL_CONSTANT bool value = etl::is_bidirectional_iterator::value || etl::is_random_iterator::value; }; template ETL_CONSTANT bool is_bidirectional_iterator_concept::value; // Deprecated template struct is_random_iterator_concept { static ETL_CONSTANT bool value = etl::is_random_iterator::value; }; // Deprecated template ETL_CONSTANT bool is_random_iterator_concept::value; // Deprecated template struct is_random_access_iterator_concept { static ETL_CONSTANT bool value = etl::is_random_access_iterator::value; }; // Deprecated template ETL_CONSTANT bool is_random_access_iterator_concept::value; #if ETL_NOT_USING_STL || ETL_CPP11_NOT_SUPPORTED //***************************************************************************** /// Get the 'begin' iterator. ///\ingroup container //***************************************************************************** template ETL_CONSTEXPR typename TContainer::iterator begin(TContainer& container) { return container.begin(); } //***************************************************************************** /// Get the 'begin' const_iterator for a container. ///\ingroup container //***************************************************************************** template ETL_CONSTEXPR typename TContainer::const_iterator begin(const TContainer& container) { return container.begin(); } //***************************************************************************** /// Get the 'begin' const_iterator for a container. ///\ingroup container //***************************************************************************** template ETL_CONSTEXPR typename TContainer::const_iterator cbegin(const TContainer& container) { return container.cbegin(); } //***************************************************************************** /// Get the 'end' iterator for a container. ///\ingroup container //***************************************************************************** template ETL_CONSTEXPR typename TContainer::iterator end(TContainer& container) { return container.end(); } //***************************************************************************** /// Get the 'end' const_iterator for a container. ///\ingroup container //***************************************************************************** template ETL_CONSTEXPR typename TContainer::const_iterator end(const TContainer& container) { return container.end(); } //***************************************************************************** /// Get the 'end' const_iterator for a container. ///\ingroup container //***************************************************************************** template ETL_CONSTEXPR typename TContainer::const_iterator cend(const TContainer& container) { return container.cend(); } //***************************************************************************** /// Get the 'begin' pointer for an array. ///\ingroup container //***************************************************************************** template ETL_CONSTEXPR TValue* begin(TValue (&data)[Array_Size]) { return &data[0]; } //***************************************************************************** /// Get the 'begin' const iterator for an array. ///\ingroup container //***************************************************************************** template ETL_CONSTEXPR const TValue* begin(const TValue (&data)[Array_Size]) { return &data[0]; } //***************************************************************************** /// Get the 'begin' const iterator for an array. ///\ingroup container //***************************************************************************** template ETL_CONSTEXPR const TValue* cbegin(const TValue (&data)[Array_Size]) { return &data[0]; } //***************************************************************************** /// Get the 'end' iterator for an array. ///\ingroup container //***************************************************************************** template ETL_CONSTEXPR TValue* end(TValue (&data)[Array_Size]) { return &data[Array_Size]; } //***************************************************************************** /// Get the 'end' const iterator for an array. ///\ingroup container //***************************************************************************** template ETL_CONSTEXPR const TValue* end(const TValue (&data)[Array_Size]) { return &data[Array_Size]; } //***************************************************************************** /// Get the 'end' const iterator for an array. ///\ingroup container //***************************************************************************** template ETL_CONSTEXPR const TValue* cend(const TValue (&data)[Array_Size]) { return &data[Array_Size]; } #endif #if ETL_NOT_USING_STL || ETL_CPP14_NOT_SUPPORTED //***************************************************************************** /// Get the 'begin' reverse_iterator for a container. ///\ingroup container //***************************************************************************** template ETL_CONSTEXPR typename TContainer::reverse_iterator rbegin(TContainer& container) { return container.rbegin(); } //***************************************************************************** /// Get the 'begin' reverse_iterator for a container. ///\ingroup container //***************************************************************************** template ETL_CONSTEXPR typename TContainer::const_reverse_iterator rbegin(const TContainer& container) { return container.rbegin(); } //***************************************************************************** /// Get the 'begin' reverse_iterator for a container. ///\ingroup container //***************************************************************************** template ETL_CONSTEXPR typename TContainer::const_reverse_iterator crbegin(const TContainer& container) { return container.crbegin(); } //***************************************************************************** /// Get the 'end' reverse_iterator for a container. ///\ingroup container //***************************************************************************** template ETL_CONSTEXPR typename TContainer::reverse_iterator rend(TContainer& container) { return container.rend(); } //***************************************************************************** /// Get the 'end' reverse_iterator for a container. ///\ingroup container //***************************************************************************** template ETL_CONSTEXPR typename TContainer::const_reverse_iterator rend(const TContainer& container) { return container.rend(); } //***************************************************************************** /// Get the 'end' reverse_iterator for a container. ///\ingroup container //***************************************************************************** template ETL_CONSTEXPR typename TContainer::const_reverse_iterator crend(const TContainer& container) { return container.crend(); } //***************************************************************************** /// Get the 'begin' reverse_iterator for an array. ///\ingroup container //***************************************************************************** template ETL_OR_STD::reverse_iterator rbegin(TValue (&data)[Array_Size]) { return ETL_OR_STD::reverse_iterator(&data[Array_Size]); } //***************************************************************************** /// Get the 'begin' const reverse_iterator for an array. ///\ingroup container //***************************************************************************** template ETL_CONSTEXPR ETL_OR_STD::reverse_iterator crbegin(const TValue (&data)[Array_Size]) { return ETL_OR_STD::reverse_iterator(&data[Array_Size]); } //***************************************************************************** /// Get the 'end' reverse_iterator for an array. ///\ingroup container //***************************************************************************** template ETL_CONSTEXPR ETL_OR_STD::reverse_iterator rend(TValue (&data)[Array_Size]) { return ETL_OR_STD::reverse_iterator(&data[0]); } //***************************************************************************** /// Get the 'end' const reverse_iterator for an array. ///\ingroup container //***************************************************************************** template ETL_CONSTEXPR ETL_OR_STD::reverse_iterator crend(const TValue (&data)[Array_Size]) { return ETL_OR_STD::reverse_iterator(&data[0]); } #endif #if ETL_NOT_USING_STL || ETL_CPP17_NOT_SUPPORTED //************************************************************************** /// Get the size of a container. /// Expects the container to have defined 'size_type'. ///\ingroup container //************************************************************************** template ETL_CONSTEXPR typename TContainer::size_type size(const TContainer& container) { return container.size(); } ///************************************************************************** /// Get the size of an array in elements at run time, or compile time if C++11 /// or above. ///\ingroup container ///************************************************************************** template ETL_CONSTEXPR size_t size(TValue (&)[Array_Size]) { return Array_Size; } #endif //************************************************************************** /// Get the size of an array in elements at compile time for C++03 ///\code /// sizeof(array_size(array)) ///\endcode ///\ingroup container //************************************************************************** template char (&array_size(T (&array)[Array_Size]))[Array_Size]; #define ETL_ARRAY_SIZE(a) sizeof(etl::array_size(a)) #if ETL_NOT_USING_STL || ETL_CPP17_NOT_SUPPORTED //************************************************************************** /// Returns a pointer to the block of memory containing the elements of the /// range. ///\ingroup container //************************************************************************** template ETL_CONSTEXPR typename TContainer::pointer data(TContainer& container) { return container.data(); } //************************************************************************** /// Returns a const_pointer to the block of memory containing the elements of /// the range. ///\ingroup container //************************************************************************** template ETL_CONSTEXPR typename TContainer::const_pointer data(const TContainer& container) { return container.data(); } ///************************************************************************** /// Returns a pointer to the block of memory containing the elements of the /// range. ///\ingroup container ///************************************************************************** template ETL_CONSTEXPR TValue* data(TValue (&a)[Array_Size]) { return a; } ///************************************************************************** /// Returns a const pointer to the block of memory containing the elements of /// the range. ///\ingroup container ///************************************************************************** template ETL_CONSTEXPR const TValue* data(const TValue (&a)[Array_Size]) { return a; } #endif #if ETL_USING_CPP17 template using iter_value_t = typename etl::iterator_traits>::value_type; template using iter_reference_t = decltype(*etl::declval()); #if ETL_USING_CPP20 template using iter_const_reference_t = typename etl::common_reference_t&&, etl::iter_reference_t>; #endif template using iter_difference_t = typename etl::iterator_traits>::difference_type; template using projected_value_t = etl::remove_cvref_t>>; namespace ranges { namespace private_ranges { struct begin { template constexpr auto operator()(T& t) const { return ETL_OR_STD::begin(t); } }; struct end { template constexpr auto operator()(T& t) const { return ETL_OR_STD::end(t); } }; struct cbegin { template constexpr auto operator()(T& t) const { return ETL_OR_STD::cbegin(t); } }; struct cend { template constexpr auto operator()(T& t) const { return ETL_OR_STD::cend(t); } }; struct rbegin { template constexpr auto operator()(T& t) const { return ETL_OR_STD::rbegin(t); } }; struct rend { template constexpr auto operator()(T& t) const { return ETL_OR_STD::rend(t); } }; struct crbegin { template constexpr auto operator()(T& t) const { return ETL_OR_STD::crbegin(t); } }; struct crend { template constexpr auto operator()(T& t) const { return ETL_OR_STD::crend(t); } }; template struct has_size_member : etl::false_type { }; template struct has_size_member< T, etl::void_t().size())>> : etl::true_type { }; template struct has_empty_member : etl::false_type { }; template struct has_empty_member< T, etl::void_t().empty())>> : etl::true_type { }; struct distance { // Overload for common ranges (iterator == sentinel type) template ::value || etl::is_output_iterator_concept::value>> constexpr etl::iter_difference_t operator()(I first, I last) const { if constexpr (etl::is_random_access_iterator_concept::value) { return last - first; } else { etl::iter_difference_t n = 0; while (!(first == last)) { ++first; ++n; } return n; } } // Overload for non-common ranges (iterator != sentinel type) template ::value || etl::is_output_iterator_concept::value) && !etl::is_same::value>> constexpr etl::iter_difference_t operator()(I first, S last) const { etl::iter_difference_t n = 0; while (!(first == last)) { ++first; ++n; } return n; } }; struct size { template constexpr size_t operator()(T&& t) const { using U = etl::remove_cvref_t; if constexpr (has_size_member::value) { return static_cast(t.size()); } else { using iter_type = decltype(ETL_OR_STD::begin(t)); static_assert(etl::is_forward_iterator_concept::value, "ranges::size requires a sized range or at least a forward range; " "single-pass input ranges are not supported"); return static_cast(distance{}(ETL_OR_STD::begin(t), ETL_OR_STD::end(t))); } } }; struct ssize { template constexpr auto operator()(T&& t) const { using U = etl::remove_cvref_t; if constexpr (has_size_member::value) { return static_cast(t.size()); } else { using iter_type = decltype(ETL_OR_STD::begin(t)); static_assert(etl::is_forward_iterator_concept::value, "ranges::ssize requires a sized range or at least a forward range; " "single-pass input ranges are not supported"); return static_cast(distance{}(ETL_OR_STD::begin(t), ETL_OR_STD::end(t))); } } }; struct empty { template constexpr auto operator()(T&& t) const { using U = etl::remove_cvref_t; if constexpr (has_empty_member::value) { return t.empty(); } else { return ETL_OR_STD::cbegin(t) == ETL_OR_STD::cend(t); } } }; struct data { template constexpr auto operator()(T& t) const { return ETL_OR_STD::data(t); } }; struct cdata { template constexpr etl::add_pointer_t< etl::add_const_t()))>>> operator()(T& t) const { return ETL_OR_STD::data(t); } }; } // namespace private_ranges inline constexpr private_ranges::begin begin; inline constexpr private_ranges::end end; inline constexpr private_ranges::cbegin cbegin; inline constexpr private_ranges::cend cend; inline constexpr private_ranges::rbegin rbegin; inline constexpr private_ranges::rend rend; inline constexpr private_ranges::crbegin crbegin; inline constexpr private_ranges::crend crend; inline constexpr private_ranges::size size; inline constexpr private_ranges::ssize ssize; inline constexpr private_ranges::empty empty; inline constexpr private_ranges::data data; inline constexpr private_ranges::cdata cdata; inline constexpr private_ranges::distance distance; //************************************************************************* /// Range primitives. //************************************************************************* template using iterator_t = decltype(etl::ranges::begin(etl::declval())); template using const_iterator_t = decltype(etl::ranges::cbegin(etl::declval())); template using sentinel_t = decltype(etl::ranges::end(etl::declval())); template using const_sentinel_t = decltype(etl::ranges::cend(etl::declval())); template using range_size_t = decltype(etl::ranges::size(etl::declval())); template using range_difference_t = etl::iter_difference_t>; template using range_value_t = etl::iter_value_t>; template using range_reference_t = etl::iter_reference_t>; struct advance_fn { template ::value || etl::is_output_iterator_concept::value) && etl::is_integral>::value>> constexpr void operator()(I& i, etl::iter_difference_t n) const { if constexpr (etl::is_random_access_iterator_concept::value) { i += n; } else { while (n > 0) { --n; ++i; } if constexpr (etl::is_bidirectional_iterator_concept::value) { while (n < 0) { ++n; --i; } } } } template ::value || etl::is_output_iterator_concept::value) && !etl::is_integral::value>> constexpr void operator()(I& i, S bound) const { if constexpr (etl::is_assignable_v) { i = etl::move(bound); } else if constexpr (etl::is_same_v && etl::is_random_access_iterator_concept::value) { (*this)(i, bound - i); } else { while (!(i == bound)) { ++i; } } } template ::value || etl::is_output_iterator_concept::value>> constexpr etl::iter_difference_t operator()(I& i, etl::iter_difference_t n, S bound) const { if constexpr (etl::is_same_v && etl::is_random_access_iterator_concept::value) { const auto dist = bound - i; if ((n >= 0 && dist >= 0 && n >= dist) || (n <= 0 && dist <= 0 && n <= dist)) { (*this)(i, bound); return n - dist; } (*this)(i, n); return 0; } else { while (n > 0 && !(i == bound)) { --n; ++i; } if constexpr (etl::is_bidirectional_iterator_concept::value) { while (n < 0 && !(i == bound)) { ++n; --i; } } return n; } } }; inline constexpr auto advance = advance_fn(); struct prev_fn { template ::value>> constexpr I operator()(I i) const { --i; return i; } template ::value>> constexpr I operator()(I i, etl::iter_difference_t n) const { ranges::advance(i, -n); return i; } template ::value>> constexpr I operator()(I i, etl::iter_difference_t n, I bound) const { ranges::advance(i, -n, bound); return i; } }; inline constexpr auto prev = prev_fn(); struct next_fn { template ::value || etl::is_output_iterator_concept::value>> constexpr I operator()(I i) const { ++i; return i; } template ::value || etl::is_output_iterator_concept::value) && etl::is_integral>::value>> constexpr I operator()(I i, etl::iter_difference_t n) const { ranges::advance(i, n); return i; } template ::value || etl::is_output_iterator_concept::value) && !etl::is_integral::value>> constexpr I operator()(I i, S bound) const { ranges::advance(i, bound); return i; } template ::value || etl::is_output_iterator_concept::value) && !etl::is_integral::value>> constexpr I operator()(I i, etl::iter_difference_t n, S bound) const { ranges::advance(i, n, bound); return i; } }; inline constexpr auto next = next_fn(); } // namespace ranges struct unreachable_sentinel_t { }; inline constexpr unreachable_sentinel_t unreachable_sentinel{}; template constexpr bool operator==(unreachable_sentinel_t, const I&) noexcept { return false; } template constexpr bool operator==(const I&, unreachable_sentinel_t) noexcept { return false; } template constexpr bool operator!=(unreachable_sentinel_t, const I& i) noexcept { return !(unreachable_sentinel_t{} == i); } template constexpr bool operator!=(const I& i, unreachable_sentinel_t) noexcept { return !(i == unreachable_sentinel_t{}); } struct default_sentinel_t { }; inline constexpr default_sentinel_t default_sentinel{}; namespace private_iterator { template struct has_arrow_operator : etl::false_type { }; template struct has_arrow_operator< T, etl::void_t().operator->())>> : etl::true_type { }; //*********************************** /// Proxy that owns a copy of the dereferenced value so that operator-> /// can safely return a pointer to it. Used when the wrapped iterator /// has no member operator-> and is not a raw pointer (i.e. *it may /// yield a prvalue / proxy whose address would otherwise dangle). //*********************************** template struct arrow_proxy { TValue stored; constexpr arrow_proxy(TValue value) : stored(etl::move(value)) { } constexpr const TValue* operator->() const noexcept { return etl::addressof(stored); } }; } // namespace private_iterator template class counted_iterator { template friend class counted_iterator; public: using iterator_type = I; using value_type = etl::iter_value_t; using difference_type = etl::iter_difference_t; using iterator_category = typename etl::iterator_traits::iterator_category; using pointer = typename etl::iterator_traits::pointer; using reference = typename etl::iterator_traits::reference; constexpr counted_iterator() = default; constexpr counted_iterator(I x, etl::iter_difference_t n) : current(etl::move(x)) , length(n) { } template constexpr counted_iterator(const counted_iterator& other) : current(other.current) , length(other.length) { } template constexpr counted_iterator& operator=(const counted_iterator& other) { current = other.current; length = other.length; return *this; } constexpr const I& base() const& noexcept { return current; } constexpr I base() && { return etl::move(current); } constexpr etl::iter_difference_t count() const noexcept { return length; } constexpr decltype(auto) operator*() const { return *current; } // operator-> for iterator types that provide a member operator-> template < typename J = I, etl::enable_if_t<(etl::is_input_iterator_concept::value || etl::is_output_iterator_concept::value) && private_iterator::has_arrow_operator::value, int> = 0> constexpr auto operator->() const noexcept { return current.operator->(); } // operator-> fallback for raw-pointer iterators (addressof is always safe) template < typename J = I, etl::enable_if_t<(etl::is_input_iterator_concept::value || etl::is_output_iterator_concept::value) && !private_iterator::has_arrow_operator::value && etl::is_pointer::value, int> = 0> constexpr auto operator->() const noexcept { return current; } // operator-> fallback for class-type iterators without member operator-> // When *current yields an lvalue reference, just take its address. template < typename J = I, etl::enable_if_t< (etl::is_input_iterator_concept::value || etl::is_output_iterator_concept::value) && !private_iterator::has_arrow_operator::value && !etl::is_pointer::value && etl::is_lvalue_reference< decltype(*etl::declval())>::value, int> = 0> constexpr auto operator->() const noexcept { return etl::addressof(*current); } // operator-> fallback for class-type iterators without member operator-> // When *current yields a prvalue / proxy, use an owning proxy so the // address remains valid. template < typename J = I, etl::enable_if_t< (etl::is_input_iterator_concept::value || etl::is_output_iterator_concept::value) && !private_iterator::has_arrow_operator::value && !etl::is_pointer::value && !etl::is_lvalue_reference< decltype(*etl::declval())>::value, int> = 0> constexpr auto operator->() const { return private_iterator::arrow_proxy{*current}; } template < typename J = I, etl::enable_if_t::value, int> = 0> constexpr decltype(auto) operator[](etl::iter_difference_t n) const { return current[n]; } constexpr counted_iterator& operator++() { ++current; --length; return *this; } constexpr counted_iterator operator++(int) { counted_iterator tmp = *this; current++; length--; return tmp; } template < typename J = I, etl::enable_if_t::value, int> = 0> constexpr counted_iterator& operator+=(etl::iter_difference_t n) { current += n; length -= n; return *this; } template < typename J = I, etl::enable_if_t::value, int> = 0> constexpr counted_iterator operator+(etl::iter_difference_t n) const { counted_iterator result{*this}; result += n; return result; } constexpr counted_iterator& operator--() { --current; ++length; return *this; } constexpr counted_iterator operator--(int) { counted_iterator tmp = *this; current--; length++; return tmp; } template < typename J = I, etl::enable_if_t::value, int> = 0> constexpr counted_iterator& operator-=(etl::iter_difference_t n) { current -= n; length += n; return *this; } template < typename J = I, etl::enable_if_t::value, int> = 0> constexpr counted_iterator operator-(etl::iter_difference_t n) const { counted_iterator result{*this}; result -= n; return result; } friend constexpr bool operator==(const counted_iterator& x, const counted_iterator& y) { return x.length == y.length; } friend constexpr bool operator==(const counted_iterator& x, etl::default_sentinel_t) { return x.count() == 0; } friend constexpr bool operator==(etl::default_sentinel_t, const counted_iterator& x) { return x.count() == 0; } friend constexpr bool operator!=(const counted_iterator& x, etl::default_sentinel_t) { return x.count() != 0; } friend constexpr bool operator!=(etl::default_sentinel_t, const counted_iterator& y) { return y.count() != 0; } template < typename J = I, etl::enable_if_t::value, int> = 0> friend constexpr counted_iterator operator+(etl::iter_difference_t n, const counted_iterator& x) { return counted_iterator(x.current + n, x.length - n); } friend constexpr etl::iter_difference_t operator-(const counted_iterator& x, const counted_iterator& y) { return y.length - x.length; } friend constexpr etl::iter_difference_t operator-(const counted_iterator& x, etl::default_sentinel_t) { return -x.length; } friend constexpr etl::iter_difference_t operator-(etl::default_sentinel_t, const counted_iterator& y) { return y.length; } private: I current{}; difference_type length{}; }; template , TIterator>::value>> constexpr typename etl::iterator_traits::difference_type distance(TIterator first, etl::default_sentinel_t) { return first.count(); } #endif #if ETL_USING_CPP14 template struct is_range : etl::false_type { }; template struct is_range< T, etl::void_t())), decltype(ETL_OR_STD::end(etl::declval()))>> : etl::true_type { }; #if ETL_USING_CPP17 template inline constexpr bool is_range_v = is_range::value; #endif #endif } // namespace etl #endif