aboutsummaryrefslogtreecommitdiff
path: root/include/EASTL/algorithm.h
diff options
context:
space:
mode:
Diffstat (limited to 'include/EASTL/algorithm.h')
-rw-r--r--include/EASTL/algorithm.h4342
1 files changed, 0 insertions, 4342 deletions
diff --git a/include/EASTL/algorithm.h b/include/EASTL/algorithm.h
deleted file mode 100644
index 6257514..0000000
--- a/include/EASTL/algorithm.h
+++ /dev/null
@@ -1,4342 +0,0 @@
-/////////////////////////////////////////////////////////////////////////////
-// Copyright (c) Electronic Arts Inc. All rights reserved.
-/////////////////////////////////////////////////////////////////////////////
-
-///////////////////////////////////////////////////////////////////////////////
-// This file implements some of the primary algorithms from the C++ STL
-// algorithm library. These versions are just like that STL versions and so
-// are redundant. They are provided solely for the purpose of projects that
-// either cannot use standard C++ STL or want algorithms that have guaranteed
-// identical behaviour across platforms.
-///////////////////////////////////////////////////////////////////////////////
-
-
-///////////////////////////////////////////////////////////////////////////////
-// Definitions
-//
-// You will notice that we are very particular about the templated typenames
-// we use here. You will notice that we follow the C++ standard closely in
-// these respects. Each of these typenames have a specific meaning;
-// this is why we don't just label templated arguments with just letters
-// such as T, U, V, A, B. Here we provide a quick reference for the typenames
-// we use. See the C++ standard, section 25-8 for more details.
-// --------------------------------------------------------------
-// typename Meaning
-// --------------------------------------------------------------
-// T The value type.
-// Compare A function which takes two arguments and returns the lesser of the two.
-// Predicate A function which takes one argument returns true if the argument meets some criteria.
-// BinaryPredicate A function which takes two arguments and returns true if some criteria is met (e.g. they are equal).
-// StrickWeakOrdering A BinaryPredicate that compares two objects, returning true if the first precedes the second. Like Compare but has additional requirements. Used for sorting routines.
-// Function A function which takes one argument and applies some operation to the target.
-// Size A count or size.
-// Generator A function which takes no arguments and returns a value (which will usually be assigned to an object).
-// UnaryOperation A function which takes one argument and returns a value (which will usually be assigned to second object).
-// BinaryOperation A function which takes two arguments and returns a value (which will usually be assigned to a third object).
-// InputIterator An input iterator (iterator you read from) which allows reading each element only once and only in a forward direction.
-// ForwardIterator An input iterator which is like InputIterator except it can be reset back to the beginning.
-// BidirectionalIterator An input iterator which is like ForwardIterator except it can be read in a backward direction as well.
-// RandomAccessIterator An input iterator which can be addressed like an array. It is a superset of all other input iterators.
-// OutputIterator An output iterator (iterator you write to) which allows writing each element only once in only in a forward direction.
-//
-// Note that with iterators that a function which takes an InputIterator will
-// also work with a ForwardIterator, BidirectionalIterator, or RandomAccessIterator.
-// The given iterator type is merely the -minimum- supported functionality the
-// iterator must support.
-///////////////////////////////////////////////////////////////////////////////
-
-
-///////////////////////////////////////////////////////////////////////////////
-// Optimizations
-//
-// There are a number of opportunities for opptimizations that we take here
-// in this library. The most obvious kinds are those that subsitute memcpy
-// in the place of a conventional loop for data types with which this is
-// possible. The algorithms here are optimized to a higher level than currently
-// available C++ STL algorithms from vendors such as Microsoft. This is especially
-// so for game programming on console devices, as we do things such as reduce
-// branching relative to other STL algorithm implementations. However, the
-// proper implementation of these algorithm optimizations is a fairly tricky
-// thing.
-//
-// The various things we look to take advantage of in order to implement
-// optimizations include:
-// - Taking advantage of random access iterators.
-// - Taking advantage of POD (plain old data) data types.
-// - Taking advantage of type_traits in general.
-// - Reducing branching and taking advantage of likely branch predictions.
-// - Taking advantage of issues related to pointer and reference aliasing.
-// - Improving cache coherency during memory accesses.
-// - Making code more likely to be inlinable by the compiler.
-//
-///////////////////////////////////////////////////////////////////////////////
-
-
-///////////////////////////////////////////////////////////////////////////////
-// Supported Algorithms
-//
-// Algorithms that we implement are listed here. Note that these items are not
-// all within this header file, as we split up the header files in order to
-// improve compilation performance. Items marked with '+' are items that are
-// extensions which don't exist in the C++ standard.
-//
-// -------------------------------------------------------------------------------
-// Algorithm Notes
-// -------------------------------------------------------------------------------
-// adjacent_find
-// adjacent_find<Compare>
-// all_of C++11
-// any_of C++11
-// none_of C++11
-// binary_search
-// binary_search<Compare>
-// +binary_search_i
-// +binary_search_i<Compare>
-// +change_heap Found in heap.h
-// +change_heap<Compare> Found in heap.h
-// clamp
-// copy
-// copy_if C++11
-// copy_n C++11
-// copy_backward
-// count
-// count_if
-// equal
-// equal<Compare>
-// equal_range
-// equal_range<Compare>
-// fill
-// fill_n
-// find
-// find_end
-// find_end<Compare>
-// find_first_of
-// find_first_of<Compare>
-// +find_first_not_of
-// +find_first_not_of<Compare>
-// +find_last_of
-// +find_last_of<Compare>
-// +find_last_not_of
-// +find_last_not_of<Compare>
-// find_if
-// find_if_not
-// for_each
-// generate
-// generate_n
-// +identical
-// +identical<Compare>
-// iter_swap
-// lexicographical_compare
-// lexicographical_compare<Compare>
-// lexicographical_compare_three_way
-// lower_bound
-// lower_bound<Compare>
-// make_heap Found in heap.h
-// make_heap<Compare> Found in heap.h
-// min
-// min<Compare>
-// max
-// max<Compare>
-// +min_alt Exists to work around the problem of conflicts with min/max #defines on some systems.
-// +min_alt<Compare>
-// +max_alt
-// +max_alt<Compare>
-// +median
-// +median<Compare>
-// merge Found in sort.h
-// merge<Compare> Found in sort.h
-// min_element
-// min_element<Compare>
-// max_element
-// max_element<Compare>
-// mismatch
-// mismatch<Compare>
-// move
-// move_backward
-// nth_element Found in sort.h
-// nth_element<Compare> Found in sort.h
-// partial_sort Found in sort.h
-// partial_sort<Compare> Found in sort.h
-// push_heap Found in heap.h
-// push_heap<Compare> Found in heap.h
-// pop_heap Found in heap.h
-// pop_heap<Compare> Found in heap.h
-// random_shuffle<Random>
-// remove
-// remove_if
-// +apply_and_remove
-// +apply_and_remove_if
-// remove_copy
-// remove_copy_if
-// +remove_heap Found in heap.h
-// +remove_heap<Compare> Found in heap.h
-// replace
-// replace_if
-// replace_copy
-// replace_copy_if
-// reverse_copy
-// reverse
-// random_shuffle
-// rotate
-// rotate_copy
-// search
-// search<Compare>
-// search_n
-// set_difference
-// set_difference<Compare>
-// set_difference_2
-// set_difference_2<Compare>
-// set_decomposition
-// set_decomposition<Compare>
-// set_intersection
-// set_intersection<Compare>
-// set_symmetric_difference
-// set_symmetric_difference<Compare>
-// set_union
-// set_union<Compare>
-// sort Found in sort.h
-// sort<Compare> Found in sort.h
-// sort_heap Found in heap.h
-// sort_heap<Compare> Found in heap.h
-// stable_sort Found in sort.h
-// stable_sort<Compare> Found in sort.h
-// swap
-// swap_ranges
-// transform
-// transform<Operation>
-// unique
-// unique<Compare>
-// upper_bound
-// upper_bound<Compare>
-// is_permutation
-// is_permutation<Predicate>
-// next_permutation
-// next_permutation<Compare>
-//
-// Algorithms from the C++ standard that we don't implement are listed here.
-// Most of these items are absent because they aren't used very often.
-// They also happen to be the more complicated than other algorithms.
-// However, we can implement any of these functions for users that might
-// need them.
-// includes
-// includes<Compare>
-// inplace_merge
-// inplace_merge<Compare>
-// partial_sort_copy
-// partial_sort_copy<Compare>
-// paritition
-// prev_permutation
-// prev_permutation<Compare>
-// search_n<Compare>
-// stable_partition
-// unique_copy
-// unique_copy<Compare>
-//
-///////////////////////////////////////////////////////////////////////////////
-
-
-#ifndef EASTL_ALGORITHM_H
-#define EASTL_ALGORITHM_H
-
-
-#include <EASTL/internal/config.h>
-#include <EASTL/type_traits.h>
-#include <EASTL/internal/move_help.h>
-#include <EASTL/internal/copy_help.h>
-#include <EASTL/internal/fill_help.h>
-#include <EASTL/initializer_list.h>
-#include <EASTL/iterator.h>
-#include <EASTL/functional.h>
-#include <EASTL/utility.h>
-#include <EASTL/internal/generic_iterator.h>
-#include <EASTL/random.h>
-#include <EASTL/compare.h>
-
-EA_DISABLE_ALL_VC_WARNINGS();
-
- #if defined(EA_COMPILER_MSVC) && (defined(EA_PROCESSOR_X86) || defined(EA_PROCESSOR_X86_64))
- #include <intrin.h>
- #endif
-
- #include <stddef.h>
- #include <string.h> // memcpy, memcmp, memmove
-
-EA_RESTORE_ALL_VC_WARNINGS();
-
-#if defined(EA_PRAGMA_ONCE_SUPPORTED)
- #pragma once // Some compilers (e.g. VC++) benefit significantly from using this. We've measured 3-4% build speed improvements in apps as a result.
-#endif
-
-
-
-///////////////////////////////////////////////////////////////////////////////
-// min/max workaround
-//
-// MSVC++ has #defines for min/max which collide with the min/max algorithm
-// declarations. The following may still not completely resolve some kinds of
-// problems with MSVC++ #defines, though it deals with most cases in production
-// game code.
-//
-#if EASTL_NOMINMAX
- #ifdef min
- #undef min
- #endif
- #ifdef max
- #undef max
- #endif
-#endif
-
-
-
-
-namespace eastl
-{
- /// min_element
- ///
- /// min_element finds the smallest element in the range [first, last).
- /// It returns the first iterator i in [first, last) such that no other
- /// iterator in [first, last) points to a value smaller than *i.
- /// The return value is last if and only if [first, last) is an empty range.
- ///
- /// Returns: The first iterator i in the range [first, last) such that
- /// for any iterator j in the range [first, last) the following corresponding
- /// condition holds: !(*j < *i).
- ///
- /// Complexity: Exactly 'max((last - first) - 1, 0)' applications of the
- /// corresponding comparisons.
- ///
- template <typename ForwardIterator>
- ForwardIterator min_element(ForwardIterator first, ForwardIterator last)
- {
- if(first != last)
- {
- ForwardIterator currentMin = first;
-
- while(++first != last)
- {
- if(*first < *currentMin)
- currentMin = first;
- }
- return currentMin;
- }
- return first;
- }
-
-
- /// min_element
- ///
- /// min_element finds the smallest element in the range [first, last).
- /// It returns the first iterator i in [first, last) such that no other
- /// iterator in [first, last) points to a value smaller than *i.
- /// The return value is last if and only if [first, last) is an empty range.
- ///
- /// Returns: The first iterator i in the range [first, last) such that
- /// for any iterator j in the range [first, last) the following corresponding
- /// conditions hold: compare(*j, *i) == false.
- ///
- /// Complexity: Exactly 'max((last - first) - 1, 0)' applications of the
- /// corresponding comparisons.
- ///
- template <typename ForwardIterator, typename Compare>
- ForwardIterator min_element(ForwardIterator first, ForwardIterator last, Compare compare)
- {
- if(first != last)
- {
- ForwardIterator currentMin = first;
-
- while(++first != last)
- {
- if(compare(*first, *currentMin))
- currentMin = first;
- }
- return currentMin;
- }
- return first;
- }
-
-
- /// max_element
- ///
- /// max_element finds the largest element in the range [first, last).
- /// It returns the first iterator i in [first, last) such that no other
- /// iterator in [first, last) points to a value greater than *i.
- /// The return value is last if and only if [first, last) is an empty range.
- ///
- /// Returns: The first iterator i in the range [first, last) such that
- /// for any iterator j in the range [first, last) the following corresponding
- /// condition holds: !(*i < *j).
- ///
- /// Complexity: Exactly 'max((last - first) - 1, 0)' applications of the
- /// corresponding comparisons.
- ///
- template <typename ForwardIterator>
- ForwardIterator max_element(ForwardIterator first, ForwardIterator last)
- {
- if(first != last)
- {
- ForwardIterator currentMax = first;
-
- while(++first != last)
- {
- if(*currentMax < *first)
- currentMax = first;
- }
- return currentMax;
- }
- return first;
- }
-
-
- /// max_element
- ///
- /// max_element finds the largest element in the range [first, last).
- /// It returns the first iterator i in [first, last) such that no other
- /// iterator in [first, last) points to a value greater than *i.
- /// The return value is last if and only if [first, last) is an empty range.
- ///
- /// Returns: The first iterator i in the range [first, last) such that
- /// for any iterator j in the range [first, last) the following corresponding
- /// condition holds: compare(*i, *j) == false.
- ///
- /// Complexity: Exactly 'max((last - first) - 1, 0)' applications of the
- /// corresponding comparisons.
- ///
- template <typename ForwardIterator, typename Compare>
- ForwardIterator max_element(ForwardIterator first, ForwardIterator last, Compare compare)
- {
- if(first != last)
- {
- ForwardIterator currentMax = first;
-
- while(++first != last)
- {
- if(compare(*currentMax, *first))
- currentMax = first;
- }
- return currentMax;
- }
- return first;
- }
-
-
- #if EASTL_MINMAX_ENABLED
-
- /// min
- ///
- /// Min returns the lesser of its two arguments; it returns the first
- /// argument if neither is less than the other. The two arguments are
- /// compared with operator <.
- ///
- /// This min and our other min implementations are defined as returning:
- /// b < a ? b : a
- /// which for example may in practice result in something different than:
- /// b <= a ? b : a
- /// in the case where b is different from a (though they compare as equal).
- /// We choose the specific ordering here because that's the ordering
- /// done by other STL implementations.
- ///
- /// Some compilers (e.g. VS20003 - VS2013) generate poor code for the case of
- /// scalars returned by reference, so we provide a specialization for those cases.
- /// The specialization returns T by value instead of reference, which is
- /// not that the Standard specifies. The Standard allows you to use
- /// an expression like &max(x, y), which would be impossible in this case.
- /// However, we have found no actual code that uses min or max like this and
- /// this specialization causes no problems in practice. Microsoft has acknowledged
- /// the problem and may fix it for a future VS version.
- ///
- template <typename T>
- inline EA_CONSTEXPR typename eastl::enable_if<eastl::is_scalar<T>::value, T>::type
- min(T a, T b)
- {
- return b < a ? b : a;
- }
-
- template <typename T>
- inline EA_CONSTEXPR typename eastl::enable_if<!eastl::is_scalar<T>::value, const T&>::type
- min(const T& a, const T& b)
- {
- return b < a ? b : a;
- }
-
- inline EA_CONSTEXPR float min(float a, float b) { return b < a ? b : a; }
- inline EA_CONSTEXPR double min(double a, double b) { return b < a ? b : a; }
- inline EA_CONSTEXPR long double min(long double a, long double b) { return b < a ? b : a; }
-
- #endif // EASTL_MINMAX_ENABLED
-
-
- /// min_alt
- ///
- /// This is an alternative version of min that avoids any possible
- /// collisions with Microsoft #defines of min and max.
- ///
- /// See min(a, b) for detailed specifications.
- ///
- template <typename T>
- inline EA_CONSTEXPR typename eastl::enable_if<eastl::is_scalar<T>::value, T>::type
- min_alt(T a, T b)
- {
- return b < a ? b : a;
- }
-
- template <typename T>
- inline typename eastl::enable_if<!eastl::is_scalar<T>::value, const T&>::type
- min_alt(const T& a, const T& b)
- {
- return b < a ? b : a;
- }
-
- inline EA_CONSTEXPR float min_alt(float a, float b) { return b < a ? b : a; }
- inline EA_CONSTEXPR double min_alt(double a, double b) { return b < a ? b : a; }
- inline EA_CONSTEXPR long double min_alt(long double a, long double b) { return b < a ? b : a; }
-
-
- #if EASTL_MINMAX_ENABLED
-
- /// min
- ///
- /// Min returns the lesser of its two arguments; it returns the first
- /// argument if neither is less than the other. The two arguments are
- /// compared with the Compare function (or function object), which
- /// takes two arguments and returns true if the first is less than
- /// the second.
- ///
- /// See min(a, b) for detailed specifications.
- ///
- /// Example usage:
- /// struct A{ int a; };
- /// struct Struct{ bool operator()(const A& a1, const A& a2){ return a1.a < a2.a; } };
- ///
- /// A a1, a2, a3;
- /// a3 = min(a1, a2, Struct());
- ///
- /// Example usage:
- /// struct B{ int b; };
- /// inline bool Function(const B& b1, const B& b2){ return b1.b < b2.b; }
- ///
- /// B b1, b2, b3;
- /// b3 = min(b1, b2, Function);
- ///
- template <typename T, typename Compare>
- inline const T&
- min(const T& a, const T& b, Compare compare)
- {
- return compare(b, a) ? b : a;
- }
-
- #endif // EASTL_MINMAX_ENABLED
-
-
- /// min_alt
- ///
- /// This is an alternative version of min that avoids any possible
- /// collisions with Microsoft #defines of min and max.
- ///
- /// See min(a, b) for detailed specifications.
- ///
- template <typename T, typename Compare>
- inline const T&
- min_alt(const T& a, const T& b, Compare compare)
- {
- return compare(b, a) ? b : a;
- }
-
-
- #if EASTL_MINMAX_ENABLED
-
- /// max
- ///
- /// Max returns the greater of its two arguments; it returns the first
- /// argument if neither is greater than the other. The two arguments are
- /// compared with operator < (and not operator >).
- ///
- /// This min and our other min implementations are defined as returning:
- /// a < b ? b : a
- /// which for example may in practice result in something different than:
- /// a <= b ? b : a
- /// in the case where b is different from a (though they compare as equal).
- /// We choose the specific ordering here because that's the ordering
- /// done by other STL implementations.
- ///
- template <typename T>
- inline EA_CONSTEXPR typename eastl::enable_if<eastl::is_scalar<T>::value, T>::type
- max(T a, T b)
- {
- return a < b ? b : a;
- }
-
- template <typename T>
- inline EA_CONSTEXPR typename eastl::enable_if<!eastl::is_scalar<T>::value, const T&>::type
- max(const T& a, const T& b)
- {
- return a < b ? b : a;
- }
-
- inline EA_CONSTEXPR float max(float a, float b) { return a < b ? b : a; }
- inline EA_CONSTEXPR double max(double a, double b) { return a < b ? b : a; }
- inline EA_CONSTEXPR long double max(long double a, long double b) { return a < b ? b : a; }
-
- #endif // EASTL_MINMAX_ENABLED
-
-
- /// max_alt
- ///
- /// This is an alternative version of max that avoids any possible
- /// collisions with Microsoft #defines of min and max.
- ///
- template <typename T>
- inline EA_CONSTEXPR typename eastl::enable_if<eastl::is_scalar<T>::value, T>::type
- max_alt(T a, T b)
- {
- return a < b ? b : a;
- }
-
- template <typename T>
- inline EA_CONSTEXPR typename eastl::enable_if<!eastl::is_scalar<T>::value, const T&>::type
- max_alt(const T& a, const T& b)
- {
- return a < b ? b : a;
- }
-
- inline EA_CONSTEXPR float max_alt(float a, float b) { return a < b ? b : a; }
- inline EA_CONSTEXPR double max_alt(double a, double b) { return a < b ? b : a; }
- inline EA_CONSTEXPR long double max_alt(long double a, long double b) { return a < b ? b : a; }
-
-
- #if EASTL_MINMAX_ENABLED
- /// max
- ///
- /// Min returns the lesser of its two arguments; it returns the first
- /// argument if neither is less than the other. The two arguments are
- /// compared with the Compare function (or function object), which
- /// takes two arguments and returns true if the first is less than
- /// the second.
- ///
- template <typename T, typename Compare>
- inline const T&
- max(const T& a, const T& b, Compare compare)
- {
- return compare(a, b) ? b : a;
- }
- #endif
-
-
- /// max_alt
- ///
- /// This is an alternative version of max that avoids any possible
- /// collisions with Microsoft #defines of min and max.
- ///
- template <typename T, typename Compare>
- inline const T&
- max_alt(const T& a, const T& b, Compare compare)
- {
- return compare(a, b) ? b : a;
- }
-
-
- /// min(std::initializer_list)
- ///
- template <typename T >
- T min(std::initializer_list<T> ilist)
- {
- return *eastl::min_element(ilist.begin(), ilist.end());
- }
-
- /// min(std::initializer_list, Compare)
- ///
- template <typename T, typename Compare>
- T min(std::initializer_list<T> ilist, Compare compare)
- {
- return *eastl::min_element(ilist.begin(), ilist.end(), compare);
- }
-
-
- /// max(std::initializer_list)
- ///
- template <typename T >
- T max(std::initializer_list<T> ilist)
- {
- return *eastl::max_element(ilist.begin(), ilist.end());
- }
-
- /// max(std::initializer_list, Compare)
- ///
- template <typename T, typename Compare>
- T max(std::initializer_list<T> ilist, Compare compare)
- {
- return *eastl::max_element(ilist.begin(), ilist.end(), compare);
- }
-
-
- /// minmax_element
- ///
- /// Returns: make_pair(first, first) if [first, last) is empty, otherwise make_pair(m, M),
- /// where m is the first iterator in [first,last) such that no iterator in the range
- /// refers to a smaller element, and where M is the last iterator in [first,last) such
- /// that no iterator in the range refers to a larger element.
- ///
- /// Complexity: At most max([(3/2)*(N - 1)], 0) applications of the corresponding predicate,
- /// where N is distance(first, last).
- ///
- template <typename ForwardIterator, typename Compare>
- eastl::pair<ForwardIterator, ForwardIterator>
- minmax_element(ForwardIterator first, ForwardIterator last, Compare compare)
- {
- eastl::pair<ForwardIterator, ForwardIterator> result(first, first);
-
- if(!(first == last) && !(++first == last))
- {
- if(compare(*first, *result.first))
- {
- result.second = result.first;
- result.first = first;
- }
- else
- result.second = first;
-
- while(++first != last)
- {
- ForwardIterator i = first;
-
- if(++first == last)
- {
- if(compare(*i, *result.first))
- result.first = i;
- else if(!compare(*i, *result.second))
- result.second = i;
- break;
- }
- else
- {
- if(compare(*first, *i))
- {
- if(compare(*first, *result.first))
- result.first = first;
-
- if(!compare(*i, *result.second))
- result.second = i;
- }
- else
- {
- if(compare(*i, *result.first))
- result.first = i;
-
- if(!compare(*first, *result.second))
- result.second = first;
- }
- }
- }
- }
-
- return result;
- }
-
-
- template <typename ForwardIterator>
- eastl::pair<ForwardIterator, ForwardIterator>
- minmax_element(ForwardIterator first, ForwardIterator last)
- {
- typedef typename eastl::iterator_traits<ForwardIterator>::value_type value_type;
-
- return eastl::minmax_element(first, last, eastl::less<value_type>());
- }
-
-
-
- /// minmax
- ///
- /// Requires: Type T shall be LessThanComparable.
- /// Returns: pair<const T&, const T&>(b, a) if b is smaller than a, and pair<const T&, const T&>(a, b) otherwise.
- /// Remarks: Returns pair<const T&, const T&>(a, b) when the arguments are equivalent.
- /// Complexity: Exactly one comparison.
- ///
-
- // The following optimization is a problem because it changes the return value in a way that would break
- // users unless they used auto (e.g. auto result = minmax(17, 33); )
- //
- // template <typename T>
- // inline EA_CONSTEXPR typename eastl::enable_if<eastl::is_scalar<T>::value, eastl::pair<T, T> >::type
- // minmax(T a, T b)
- // {
- // return (b < a) ? eastl::make_pair(b, a) : eastl::make_pair(a, b);
- // }
- //
- // template <typename T>
- // inline typename eastl::enable_if<!eastl::is_scalar<T>::value, eastl::pair<const T&, const T&> >::type
- // minmax(const T& a, const T& b)
- // {
- // return (b < a) ? eastl::make_pair(b, a) : eastl::make_pair(a, b);
- // }
-
- // It turns out that the following conforming definition of minmax generates a warning when used with VC++ up
- // to at least VS2012. The VS2012 version of minmax is a broken and non-conforming definition, and we don't
- // want to do that. We could do it for scalars alone, though we'd have to decide if we are going to do that
- // for all compilers, because it changes the return value from a pair of references to a pair of values.
- template <typename T>
- inline eastl::pair<const T&, const T&>
- minmax(const T& a, const T& b)
- {
- return (b < a) ? eastl::make_pair(b, a) : eastl::make_pair(a, b);
- }
-
-
- template <typename T, typename Compare>
- eastl::pair<const T&, const T&>
- minmax(const T& a, const T& b, Compare compare)
- {
- return compare(b, a) ? eastl::make_pair(b, a) : eastl::make_pair(a, b);
- }
-
-
-
- template <typename T>
- eastl::pair<T, T>
- minmax(std::initializer_list<T> ilist)
- {
- typedef typename std::initializer_list<T>::iterator iterator_type;
- eastl::pair<iterator_type, iterator_type> iteratorPair = eastl::minmax_element(ilist.begin(), ilist.end());
- return eastl::make_pair(*iteratorPair.first, *iteratorPair.second);
- }
-
- template <typename T, class Compare>
- eastl::pair<T, T>
- minmax(std::initializer_list<T> ilist, Compare compare)
- {
- typedef typename std::initializer_list<T>::iterator iterator_type;
- eastl::pair<iterator_type, iterator_type> iteratorPair = eastl::minmax_element(ilist.begin(), ilist.end(), compare);
- return eastl::make_pair(*iteratorPair.first, *iteratorPair.second);
- }
-
- template <typename T>
- inline T&& median_impl(T&& a, T&& b, T&& c)
- {
- if(eastl::less<T>()(a, b))
- {
- if(eastl::less<T>()(b, c))
- return eastl::forward<T>(b);
- else if(eastl::less<T>()(a, c))
- return eastl::forward<T>(c);
- else
- return eastl::forward<T>(a);
- }
- else if(eastl::less<T>()(a, c))
- return eastl::forward<T>(a);
- else if(eastl::less<T>()(b, c))
- return eastl::forward<T>(c);
- return eastl::forward<T>(b);
- }
-
- /// median
- ///
- /// median finds which element of three (a, b, d) is in-between the other two.
- /// If two or more elements are equal, the first (e.g. a before b) is chosen.
- ///
- /// Complexity: Either two or three comparisons will be required, depending
- /// on the values.
- ///
- template <typename T>
- inline const T& median(const T& a, const T& b, const T& c)
- {
- return median_impl(a, b, c);
- }
-
- /// median
- ///
- /// median finds which element of three (a, b, d) is in-between the other two.
- /// If two or more elements are equal, the first (e.g. a before b) is chosen.
- ///
- /// Complexity: Either two or three comparisons will be required, depending
- /// on the values.
- ///
- template <typename T>
- inline T&& median(T&& a, T&& b, T&& c)
- {
- return eastl::forward<T>(median_impl(eastl::forward<T>(a), eastl::forward<T>(b), eastl::forward<T>(c)));
- }
-
-
- template <typename T, typename Compare>
- inline T&& median_impl(T&& a, T&& b, T&& c, Compare compare)
- {
- if(compare(a, b))
- {
- if(compare(b, c))
- return eastl::forward<T>(b);
- else if(compare(a, c))
- return eastl::forward<T>(c);
- else
- return eastl::forward<T>(a);
- }
- else if(compare(a, c))
- return eastl::forward<T>(a);
- else if(compare(b, c))
- return eastl::forward<T>(c);
- return eastl::forward<T>(b);
- }
-
-
- /// median
- ///
- /// median finds which element of three (a, b, d) is in-between the other two.
- /// If two or more elements are equal, the first (e.g. a before b) is chosen.
- ///
- /// Complexity: Either two or three comparisons will be required, depending
- /// on the values.
- ///
- template <typename T, typename Compare>
- inline const T& median(const T& a, const T& b, const T& c, Compare compare)
- {
- return median_impl<const T&, Compare>(a, b, c, compare);
- }
-
- /// median
- ///
- /// median finds which element of three (a, b, d) is in-between the other two.
- /// If two or more elements are equal, the first (e.g. a before b) is chosen.
- ///
- /// Complexity: Either two or three comparisons will be required, depending
- /// on the values.
- ///
- template <typename T, typename Compare>
- inline T&& median(T&& a, T&& b, T&& c, Compare compare)
- {
- return eastl::forward<T>(median_impl<T&&, Compare>(eastl::forward<T>(a), eastl::forward<T>(b), eastl::forward<T>(c), compare));
- }
-
-
-
-
- /// all_of
- ///
- /// Returns: true if the unary predicate p returns true for all elements in the range [first, last)
- ///
- template <typename InputIterator, typename Predicate>
- inline bool all_of(InputIterator first, InputIterator last, Predicate p)
- {
- for(; first != last; ++first)
- {
- if(!p(*first))
- return false;
- }
- return true;
- }
-
-
- /// any_of
- ///
- /// Returns: true if the unary predicate p returns true for any of the elements in the range [first, last)
- ///
- template <typename InputIterator, typename Predicate>
- inline bool any_of(InputIterator first, InputIterator last, Predicate p)
- {
- for(; first != last; ++first)
- {
- if(p(*first))
- return true;
- }
- return false;
- }
-
-
- /// none_of
- ///
- /// Returns: true if the unary predicate p returns true for none of the elements in the range [first, last)
- ///
- template <typename InputIterator, typename Predicate>
- inline bool none_of(InputIterator first, InputIterator last, Predicate p)
- {
- for(; first != last; ++first)
- {
- if(p(*first))
- return false;
- }
- return true;
- }
-
-
- /// adjacent_find
- ///
- /// Returns: The first iterator i such that both i and i + 1 are in the range
- /// [first, last) for which the following corresponding conditions hold: *i == *(i + 1).
- /// Returns last if no such iterator is found.
- ///
- /// Complexity: Exactly 'find(first, last, value) - first' applications of the corresponding predicate.
- ///
- template <typename ForwardIterator>
- inline ForwardIterator
- adjacent_find(ForwardIterator first, ForwardIterator last)
- {
- if(first != last)
- {
- ForwardIterator i = first;
-
- for(++i; i != last; ++i)
- {
- if(*first == *i)
- return first;
- first = i;
- }
- }
- return last;
- }
-
-
-
- /// adjacent_find
- ///
- /// Returns: The first iterator i such that both i and i + 1 are in the range
- /// [first, last) for which the following corresponding conditions hold: predicate(*i, *(i + 1)) != false.
- /// Returns last if no such iterator is found.
- ///
- /// Complexity: Exactly 'find(first, last, value) - first' applications of the corresponding predicate.
- ///
- template <typename ForwardIterator, typename BinaryPredicate>
- inline ForwardIterator
- adjacent_find(ForwardIterator first, ForwardIterator last, BinaryPredicate predicate)
- {
- if(first != last)
- {
- ForwardIterator i = first;
-
- for(++i; i != last; ++i)
- {
- if(predicate(*first, *i))
- return first;
- first = i;
- }
- }
- return last;
- }
-
-
- /// shuffle
- ///
- /// New for C++11
- /// Randomizes a sequence of values via a user-supplied UniformRandomNumberGenerator.
- /// The difference between this and the original random_shuffle function is that this uses the more
- /// advanced and flexible UniformRandomNumberGenerator interface as opposed to the more
- /// limited RandomNumberGenerator interface of random_shuffle.
- ///
- /// Effects: Shuffles the elements in the range [first, last) with uniform distribution.
- ///
- /// Complexity: Exactly '(last - first) - 1' swaps.
- ///
- /// Example usage:
- /// struct Rand{ eastl_size_t operator()(eastl_size_t n) { return (eastl_size_t)(rand() % n); } }; // Note: The C rand function is poor and slow.
- /// Rand randInstance;
- /// shuffle(pArrayBegin, pArrayEnd, randInstance);
- ///
- // See the C++11 Standard, 26.5.1.3, Uniform random number generator requirements.
- // Also http://en.cppreference.com/w/cpp/numeric/random/uniform_int_distribution
-
- template <typename RandomAccessIterator, typename UniformRandomNumberGenerator>
- void shuffle(RandomAccessIterator first, RandomAccessIterator last, UniformRandomNumberGenerator&& urng)
- {
- if(first != last)
- {
- typedef typename eastl::iterator_traits<RandomAccessIterator>::difference_type difference_type;
- typedef typename eastl::make_unsigned<difference_type>::type unsigned_difference_type;
- typedef typename eastl::uniform_int_distribution<unsigned_difference_type> uniform_int_distribution;
- typedef typename uniform_int_distribution::param_type uniform_int_distribution_param_type;
-
- uniform_int_distribution uid;
-
- for(RandomAccessIterator i = first + 1; i != last; ++i)
- iter_swap(i, first + uid(urng, uniform_int_distribution_param_type(0, i - first)));
- }
- }
-
-
- /// random_shuffle
- ///
- /// Randomizes a sequence of values.
- ///
- /// Effects: Shuffles the elements in the range [first, last) with uniform distribution.
- ///
- /// Complexity: Exactly '(last - first) - 1' swaps.
- ///
- /// Example usage:
- /// eastl_size_t Rand(eastl_size_t n) { return (eastl_size_t)(rand() % n); } // Note: The C rand function is poor and slow.
- /// pointer_to_unary_function<eastl_size_t, eastl_size_t> randInstance(Rand);
- /// random_shuffle(pArrayBegin, pArrayEnd, randInstance);
- ///
- /// Example usage:
- /// struct Rand{ eastl_size_t operator()(eastl_size_t n) { return (eastl_size_t)(rand() % n); } }; // Note: The C rand function is poor and slow.
- /// Rand randInstance;
- /// random_shuffle(pArrayBegin, pArrayEnd, randInstance);
- ///
- template <typename RandomAccessIterator, typename RandomNumberGenerator>
- inline void random_shuffle(RandomAccessIterator first, RandomAccessIterator last, RandomNumberGenerator&& rng)
- {
- typedef typename eastl::iterator_traits<RandomAccessIterator>::difference_type difference_type;
-
- // We must do 'rand((i - first) + 1)' here and cannot do 'rand(last - first)',
- // as it turns out that the latter results in unequal distribution probabilities.
- // http://www.cigital.com/papers/download/developer_gambling.php
-
- for(RandomAccessIterator i = first + 1; i < last; ++i)
- iter_swap(i, first + (difference_type)rng((eastl_size_t)((i - first) + 1)));
- }
-
-
- /// random_shuffle
- ///
- /// Randomizes a sequence of values.
- ///
- /// Effects: Shuffles the elements in the range [first, last) with uniform distribution.
- ///
- /// Complexity: Exactly '(last - first) - 1' swaps.
- ///
- /// Example usage:
- /// random_shuffle(pArrayBegin, pArrayEnd);
- ///
- /// *** Disabled until we decide if we want to get into the business of writing random number generators. ***
- ///
- /// template <typename RandomAccessIterator>
- /// inline void random_shuffle(RandomAccessIterator first, RandomAccessIterator last)
- /// {
- /// for(RandomAccessIterator i = first + 1; i < last; ++i)
- /// iter_swap(i, first + SomeRangedRandomNumberGenerator((i - first) + 1));
- /// }
-
-
-
-
-
-
- /// move_n
- ///
- /// Same as move(InputIterator, InputIterator, OutputIterator) except based on count instead of iterator range.
- ///
- template <typename InputIterator, typename Size, typename OutputIterator>
- inline OutputIterator
- move_n_impl(InputIterator first, Size n, OutputIterator result, EASTL_ITC_NS::input_iterator_tag)
- {
- for(; n > 0; --n)
- *result++ = eastl::move(*first++);
- return result;
- }
-
- template <typename RandomAccessIterator, typename Size, typename OutputIterator>
- inline OutputIterator
- move_n_impl(RandomAccessIterator first, Size n, OutputIterator result, EASTL_ITC_NS::random_access_iterator_tag)
- {
- return eastl::move(first, first + n, result); // Take advantage of the optimizations present in the move algorithm.
- }
-
-
- template <typename InputIterator, typename Size, typename OutputIterator>
- inline OutputIterator
- move_n(InputIterator first, Size n, OutputIterator result)
- {
- typedef typename eastl::iterator_traits<InputIterator>::iterator_category IC;
- return eastl::move_n_impl(first, n, result, IC());
- }
-
-
-
- /// copy_n
- ///
- /// Same as copy(InputIterator, InputIterator, OutputIterator) except based on count instead of iterator range.
- /// Effects: Copies exactly count values from the range beginning at first to the range beginning at result, if count > 0. Does nothing otherwise.
- /// Returns: Iterator in the destination range, pointing past the last element copied if count>0 or first otherwise.
- /// Complexity: Exactly count assignments, if count > 0.
- ///
- template <typename InputIterator, typename Size, typename OutputIterator>
- inline OutputIterator
- copy_n_impl(InputIterator first, Size n, OutputIterator result, EASTL_ITC_NS::input_iterator_tag)
- {
- for(; n > 0; --n)
- *result++ = *first++;
- return result;
- }
-
- template <typename RandomAccessIterator, typename Size, typename OutputIterator>
- inline OutputIterator
- copy_n_impl(RandomAccessIterator first, Size n, OutputIterator result, EASTL_ITC_NS::random_access_iterator_tag)
- {
- return eastl::copy(first, first + n, result); // Take advantage of the optimizations present in the copy algorithm.
- }
-
-
- template <typename InputIterator, typename Size, typename OutputIterator>
- inline OutputIterator
- copy_n(InputIterator first, Size n, OutputIterator result)
- {
- typedef typename eastl::iterator_traits<InputIterator>::iterator_category IC;
- return eastl::copy_n_impl(first, n, result, IC());
- }
-
-
- /// copy_if
- ///
- /// Effects: Assigns to the result iterator only if the predicate is true.
- ///
- template <typename InputIterator, typename OutputIterator, typename Predicate>
- inline OutputIterator
- copy_if(InputIterator first, InputIterator last, OutputIterator result, Predicate predicate)
- {
- // This implementation's performance could be improved by taking a more complicated approach like with the copy algorithm.
- for(; first != last; ++first)
- {
- if(predicate(*first))
- *result++ = *first;
- }
-
- return result;
- }
-
-
-
-
- // Implementation moving copying both trivial and non-trivial data via a lesser iterator than random-access.
- template <typename /*BidirectionalIterator1Category*/, bool /*isMove*/, bool /*canMemmove*/>
- struct move_and_copy_backward_helper
- {
- template <typename BidirectionalIterator1, typename BidirectionalIterator2>
- static BidirectionalIterator2 move_or_copy_backward(BidirectionalIterator1 first, BidirectionalIterator1 last, BidirectionalIterator2 resultEnd)
- {
- while(first != last)
- *--resultEnd = *--last;
- return resultEnd; // resultEnd now points to the beginning of the destination sequence instead of the end.
- }
- };
-
- // Specialization for moving non-trivial data via a lesser iterator than random-access.
- template <typename BidirectionalIterator1Category>
- struct move_and_copy_backward_helper<BidirectionalIterator1Category, true, false>
- {
- template <typename BidirectionalIterator1, typename BidirectionalIterator2>
- static BidirectionalIterator2 move_or_copy_backward(BidirectionalIterator1 first, BidirectionalIterator1 last, BidirectionalIterator2 resultEnd)
- {
- while(first != last)
- *--resultEnd = eastl::move(*--last);
- return resultEnd; // resultEnd now points to the beginning of the destination sequence instead of the end.
- }
- };
-
- // Specialization for moving non-trivial data via a random-access iterator. It's theoretically faster because the compiler can see the count when its a compile-time const.
- template<>
- struct move_and_copy_backward_helper<EASTL_ITC_NS::random_access_iterator_tag, true, false>
- {
- template<typename BidirectionalIterator1, typename BidirectionalIterator2>
- static BidirectionalIterator2 move_or_copy_backward(BidirectionalIterator1 first, BidirectionalIterator1 last, BidirectionalIterator2 resultEnd)
- {
- typedef typename eastl::iterator_traits<BidirectionalIterator1>::difference_type difference_type;
-
- for(difference_type n = (last - first); n > 0; --n)
- *--resultEnd = eastl::move(*--last);
- return resultEnd; // resultEnd now points to the beginning of the destination sequence instead of the end.
- }
- };
-
- // Specialization for copying non-trivial data via a random-access iterator. It's theoretically faster because the compiler can see the count when its a compile-time const.
- // This specialization converts the random access BidirectionalIterator1 last-first to an integral type. There's simple way for us to take advantage of a random access output iterator,
- // as the range is specified by the input instead of the output, and distance(first, last) for a non-random-access iterator is potentially slow.
- template <>
- struct move_and_copy_backward_helper<EASTL_ITC_NS::random_access_iterator_tag, false, false>
- {
- template <typename BidirectionalIterator1, typename BidirectionalIterator2>
- static BidirectionalIterator2 move_or_copy_backward(BidirectionalIterator1 first, BidirectionalIterator1 last, BidirectionalIterator2 resultEnd)
- {
- typedef typename eastl::iterator_traits<BidirectionalIterator1>::difference_type difference_type;
-
- for(difference_type n = (last - first); n > 0; --n)
- *--resultEnd = *--last;
- return resultEnd; // resultEnd now points to the beginning of the destination sequence instead of the end.
- }
- };
-
- // Specialization for when we can use memmove/memcpy. See the notes above for what conditions allow this.
- template <bool isMove>
- struct move_and_copy_backward_helper<EASTL_ITC_NS::random_access_iterator_tag, isMove, true>
- {
- template <typename T>
- static T* move_or_copy_backward(const T* first, const T* last, T* resultEnd)
- {
- return (T*)memmove(resultEnd - (last - first), first, (size_t)((uintptr_t)last - (uintptr_t)first));
- // We could use memcpy here if there's no range overlap, but memcpy is rarely much faster than memmove.
- }
- };
-
- template <bool isMove, typename BidirectionalIterator1, typename BidirectionalIterator2>
- inline BidirectionalIterator2 move_and_copy_backward_chooser(BidirectionalIterator1 first, BidirectionalIterator1 last, BidirectionalIterator2 resultEnd)
- {
- typedef typename eastl::iterator_traits<BidirectionalIterator1>::iterator_category IIC;
-
- const bool canBeMemmoved = internal::can_be_memmoved_helper<BidirectionalIterator1, BidirectionalIterator2>::value;
-
- return eastl::move_and_copy_backward_helper<IIC, isMove, canBeMemmoved>::move_or_copy_backward(first, last, resultEnd); // Need to chose based on the input iterator tag and not the output iterator tag, because containers accept input ranges of iterator types different than self.
- }
-
-
- // We have a second layer of unwrap_iterator calls because the original iterator might be something like move_iterator<generic_iterator<int*> > (i.e. doubly-wrapped).
- template <bool isMove, typename BidirectionalIterator1, typename BidirectionalIterator2>
- inline BidirectionalIterator2 move_and_copy_backward_unwrapper(BidirectionalIterator1 first, BidirectionalIterator1 last, BidirectionalIterator2 resultEnd)
- {
- return BidirectionalIterator2(eastl::move_and_copy_backward_chooser<isMove>(eastl::unwrap_iterator(first), eastl::unwrap_iterator(last), eastl::unwrap_iterator(resultEnd))); // Have to convert to BidirectionalIterator2 because result.base() could be a T*
- }
-
-
- /// move_backward
- ///
- /// The elements are moved in reverse order (the last element is moved first), but their relative order is preserved.
- /// After this operation the elements in the moved-from range will still contain valid values of the
- /// appropriate type, but not necessarily the same values as before the move.
- /// Returns the beginning of the result range.
- /// Note: When moving between containers, the dest range must be valid; this function doesn't resize containers.
- /// Note: If result is within [first, last), move must be used instead of move_backward.
- ///
- /// Example usage:
- /// eastl::move_backward(myArray.begin(), myArray.end(), myDestArray.end());
- ///
- /// Reference implementation:
- /// template <typename BidirectionalIterator1, typename BidirectionalIterator2>
- /// BidirectionalIterator2 move_backward(BidirectionalIterator1 first, BidirectionalIterator1 last, BidirectionalIterator2 resultEnd)
- /// {
- /// while(last != first)
- /// *--resultEnd = eastl::move(*--last);
- /// return resultEnd;
- /// }
- ///
- template <typename BidirectionalIterator1, typename BidirectionalIterator2>
- inline BidirectionalIterator2 move_backward(BidirectionalIterator1 first, BidirectionalIterator1 last, BidirectionalIterator2 resultEnd)
- {
- return eastl::move_and_copy_backward_unwrapper<true>(eastl::unwrap_iterator(first), eastl::unwrap_iterator(last), resultEnd);
- }
-
-
- /// copy_backward
- ///
- /// copies memory in the range of [first, last) to the range *ending* with result.
- ///
- /// Effects: Copies elements in the range [first, last) into the range
- /// [result - (last - first), result) starting from last 1 and proceeding to first.
- /// For each positive integer n <= (last - first), performs *(result n) = *(last - n).
- ///
- /// Requires: result shall not be in the range [first, last).
- ///
- /// Returns: result - (last - first). That is, returns the beginning of the result range.
- ///
- /// Complexity: Exactly 'last - first' assignments.
- ///
- template <typename BidirectionalIterator1, typename BidirectionalIterator2>
- inline BidirectionalIterator2 copy_backward(BidirectionalIterator1 first, BidirectionalIterator1 last, BidirectionalIterator2 resultEnd)
- {
- const bool isMove = eastl::is_move_iterator<BidirectionalIterator1>::value; EA_UNUSED(isMove);
-
- return eastl::move_and_copy_backward_unwrapper<isMove>(eastl::unwrap_iterator(first), eastl::unwrap_iterator(last), resultEnd);
- }
-
-
- /// count
- ///
- /// Counts the number of items in the range of [first, last) which equal the input value.
- ///
- /// Effects: Returns the number of iterators i in the range [first, last) for which the
- /// following corresponding conditions hold: *i == value.
- ///
- /// Complexity: At most 'last - first' applications of the corresponding predicate.
- ///
- /// Note: The predicate version of count is count_if and not another variation of count.
- /// This is because both versions would have three parameters and there could be ambiguity.
- ///
- template <typename InputIterator, typename T>
- inline typename eastl::iterator_traits<InputIterator>::difference_type
- count(InputIterator first, InputIterator last, const T& value)
- {
- typename eastl::iterator_traits<InputIterator>::difference_type result = 0;
-
- for(; first != last; ++first)
- {
- if(*first == value)
- ++result;
- }
- return result;
- }
-
-
- // C++ doesn't define a count with predicate, as it can effectively be synthesized via count_if
- // with an appropriate predicate. However, it's often simpler to just have count with a predicate.
- template <typename InputIterator, typename T, typename Predicate>
- inline typename eastl::iterator_traits<InputIterator>::difference_type
- count(InputIterator first, InputIterator last, const T& value, Predicate predicate)
- {
- typename eastl::iterator_traits<InputIterator>::difference_type result = 0;
-
- for(; first != last; ++first)
- {
- if(predicate(*first, value))
- ++result;
- }
- return result;
- }
-
-
- /// count_if
- ///
- /// Counts the number of items in the range of [first, last) which match
- /// the input value as defined by the input predicate function.
- ///
- /// Effects: Returns the number of iterators i in the range [first, last) for which the
- /// following corresponding conditions hold: predicate(*i) != false.
- ///
- /// Complexity: At most 'last - first' applications of the corresponding predicate.
- ///
- /// Note: The non-predicate version of count_if is count and not another variation of count_if.
- /// This is because both versions would have three parameters and there could be ambiguity.
- ///
- template <typename InputIterator, typename Predicate>
- inline typename eastl::iterator_traits<InputIterator>::difference_type
- count_if(InputIterator first, InputIterator last, Predicate predicate)
- {
- typename eastl::iterator_traits<InputIterator>::difference_type result = 0;
-
- for(; first != last; ++first)
- {
- if(predicate(*first))
- ++result;
- }
- return result;
- }
-
-
- /// find
- ///
- /// finds the value within the unsorted range of [first, last).
- ///
- /// Returns: The first iterator i in the range [first, last) for which
- /// the following corresponding conditions hold: *i == value.
- /// Returns last if no such iterator is found.
- ///
- /// Complexity: At most 'last - first' applications of the corresponding predicate.
- /// This is a linear search and not a binary one.
- ///
- /// Note: The predicate version of find is find_if and not another variation of find.
- /// This is because both versions would have three parameters and there could be ambiguity.
- ///
- template <typename InputIterator, typename T>
- inline InputIterator
- find(InputIterator first, InputIterator last, const T& value)
- {
- while((first != last) && !(*first == value)) // Note that we always express value comparisons in terms of < or ==.
- ++first;
- return first;
- }
-
-
- // C++ doesn't define a find with predicate, as it can effectively be synthesized via find_if
- // with an appropriate predicate. However, it's often simpler to just have find with a predicate.
- template <typename InputIterator, typename T, typename Predicate>
- inline InputIterator
- find(InputIterator first, InputIterator last, const T& value, Predicate predicate)
- {
- while((first != last) && !predicate(*first, value))
- ++first;
- return first;
- }
-
-
-
- /// find_if
- ///
- /// finds the value within the unsorted range of [first, last).
- ///
- /// Returns: The first iterator i in the range [first, last) for which
- /// the following corresponding conditions hold: pred(*i) != false.
- /// Returns last if no such iterator is found.
- /// If the sequence of elements to search for (i.e. first2 - last2) is empty,
- /// the find always fails and last1 will be returned.
- ///
- /// Complexity: At most 'last - first' applications of the corresponding predicate.
- ///
- /// Note: The non-predicate version of find_if is find and not another variation of find_if.
- /// This is because both versions would have three parameters and there could be ambiguity.
- ///
- template <typename InputIterator, typename Predicate>
- inline InputIterator
- find_if(InputIterator first, InputIterator last, Predicate predicate)
- {
- while((first != last) && !predicate(*first))
- ++first;
- return first;
- }
-
-
-
- /// find_if_not
- ///
- /// find_if_not works the same as find_if except it tests for if the predicate
- /// returns false for the elements instead of true.
- ///
- template <typename InputIterator, typename Predicate>
- inline InputIterator
- find_if_not(InputIterator first, InputIterator last, Predicate predicate)
- {
- for(; first != last; ++first)
- {
- if(!predicate(*first))
- return first;
- }
- return last;
- }
-
-
-
-
- /// find_first_of
- ///
- /// find_first_of is similar to find in that it performs linear search through
- /// a range of ForwardIterators. The difference is that while find searches
- /// for one particular value, find_first_of searches for any of several values.
- /// Specifically, find_first_of searches for the first occurrance in the
- /// range [first1, last1) of any of the elements in [first2, last2).
- /// This function is thus similar to the strpbrk standard C string function.
- /// If the sequence of elements to search for (i.e. first2-last2) is empty,
- /// the find always fails and last1 will be returned.
- ///
- /// Effects: Finds an element that matches one of a set of values.
- ///
- /// Returns: The first iterator i in the range [first1, last1) such that for some
- /// integer j in the range [first2, last2) the following conditions hold: *i == *j.
- /// Returns last1 if no such iterator is found.
- ///
- /// Complexity: At most '(last1 - first1) * (last2 - first2)' applications of the
- /// corresponding predicate.
- ///
- template <typename ForwardIterator1, typename ForwardIterator2>
- ForwardIterator1
- find_first_of(ForwardIterator1 first1, ForwardIterator1 last1,
- ForwardIterator2 first2, ForwardIterator2 last2)
- {
- for(; first1 != last1; ++first1)
- {
- for(ForwardIterator2 i = first2; i != last2; ++i)
- {
- if(*first1 == *i)
- return first1;
- }
- }
- return last1;
- }
-
-
- /// find_first_of
- ///
- /// find_first_of is similar to find in that it performs linear search through
- /// a range of ForwardIterators. The difference is that while find searches
- /// for one particular value, find_first_of searches for any of several values.
- /// Specifically, find_first_of searches for the first occurrance in the
- /// range [first1, last1) of any of the elements in [first2, last2).
- /// This function is thus similar to the strpbrk standard C string function.
- ///
- /// Effects: Finds an element that matches one of a set of values.
- ///
- /// Returns: The first iterator i in the range [first1, last1) such that for some
- /// integer j in the range [first2, last2) the following conditions hold: pred(*i, *j) != false.
- /// Returns last1 if no such iterator is found.
- ///
- /// Complexity: At most '(last1 - first1) * (last2 - first2)' applications of the
- /// corresponding predicate.
- ///
- template <typename ForwardIterator1, typename ForwardIterator2, typename BinaryPredicate>
- ForwardIterator1
- find_first_of(ForwardIterator1 first1, ForwardIterator1 last1,
- ForwardIterator2 first2, ForwardIterator2 last2,
- BinaryPredicate predicate)
- {
- for(; first1 != last1; ++first1)
- {
- for(ForwardIterator2 i = first2; i != last2; ++i)
- {
- if(predicate(*first1, *i))
- return first1;
- }
- }
- return last1;
- }
-
-
- /// find_first_not_of
- ///
- /// Searches through first range for the first element that does not belong the second input range.
- /// This is very much like the C++ string find_first_not_of function.
- ///
- /// Returns: The first iterator i in the range [first1, last1) such that for some
- /// integer j in the range [first2, last2) the following conditions hold: !(*i == *j).
- /// Returns last1 if no such iterator is found.
- ///
- /// Complexity: At most '(last1 - first1) * (last2 - first2)' applications of the
- /// corresponding predicate.
- ///
- template <class ForwardIterator1, class ForwardIterator2>
- ForwardIterator1
- find_first_not_of(ForwardIterator1 first1, ForwardIterator1 last1,
- ForwardIterator2 first2, ForwardIterator2 last2)
- {
- for(; first1 != last1; ++first1)
- {
- if(eastl::find(first2, last2, *first1) == last2)
- break;
- }
-
- return first1;
- }
-
-
-
- /// find_first_not_of
- ///
- /// Searches through first range for the first element that does not belong the second input range.
- /// This is very much like the C++ string find_first_not_of function.
- ///
- /// Returns: The first iterator i in the range [first1, last1) such that for some
- /// integer j in the range [first2, last2) the following conditions hold: pred(*i, *j) == false.
- /// Returns last1 if no such iterator is found.
- ///
- /// Complexity: At most '(last1 - first1) * (last2 - first2)' applications of the
- /// corresponding predicate.
- ///
- template <class ForwardIterator1, class ForwardIterator2, class BinaryPredicate>
- inline ForwardIterator1
- find_first_not_of(ForwardIterator1 first1, ForwardIterator1 last1,
- ForwardIterator2 first2, ForwardIterator2 last2,
- BinaryPredicate predicate)
- {
- typedef typename eastl::iterator_traits<ForwardIterator1>::value_type value_type;
-
- for(; first1 != last1; ++first1)
- {
- if(eastl::find_if(first2, last2, eastl::bind1st<BinaryPredicate, value_type>(predicate, *first1)) == last2)
- break;
- }
-
- return first1;
- }
-
-
- template <class BidirectionalIterator1, class ForwardIterator2>
- inline BidirectionalIterator1
- find_last_of(BidirectionalIterator1 first1, BidirectionalIterator1 last1,
- ForwardIterator2 first2, ForwardIterator2 last2)
- {
- if((first1 != last1) && (first2 != last2))
- {
- BidirectionalIterator1 it1(last1);
-
- while((--it1 != first1) && (eastl::find(first2, last2, *it1) == last2))
- ; // Do nothing
-
- if((it1 != first1) || (eastl::find(first2, last2, *it1) != last2))
- return it1;
- }
-
- return last1;
- }
-
-
- template <class BidirectionalIterator1, class ForwardIterator2, class BinaryPredicate>
- BidirectionalIterator1
- find_last_of(BidirectionalIterator1 first1, BidirectionalIterator1 last1,
- ForwardIterator2 first2, ForwardIterator2 last2,
- BinaryPredicate predicate)
- {
- typedef typename eastl::iterator_traits<BidirectionalIterator1>::value_type value_type;
-
- if((first1 != last1) && (first2 != last2))
- {
- BidirectionalIterator1 it1(last1);
-
- while((--it1 != first1) && (eastl::find_if(first2, last2, eastl::bind1st<BinaryPredicate, value_type>(predicate, *it1)) == last2))
- ; // Do nothing
-
- if((it1 != first1) || (eastl::find_if(first2, last2, eastl::bind1st<BinaryPredicate, value_type>(predicate, *it1)) != last2))
- return it1;
- }
-
- return last1;
- }
-
-
- template <class BidirectionalIterator1, class ForwardIterator2>
- inline BidirectionalIterator1
- find_last_not_of(BidirectionalIterator1 first1, BidirectionalIterator1 last1,
- ForwardIterator2 first2, ForwardIterator2 last2)
- {
- if((first1 != last1) && (first2 != last2))
- {
- BidirectionalIterator1 it1(last1);
-
- while((--it1 != first1) && (eastl::find(first2, last2, *it1) != last2))
- ; // Do nothing
-
- if((it1 != first1) || (eastl::find( first2, last2, *it1) == last2))
- return it1;
- }
-
- return last1;
- }
-
-
- template <class BidirectionalIterator1, class ForwardIterator2, class BinaryPredicate>
- inline BidirectionalIterator1
- find_last_not_of(BidirectionalIterator1 first1, BidirectionalIterator1 last1,
- ForwardIterator2 first2, ForwardIterator2 last2,
- BinaryPredicate predicate)
- {
- typedef typename eastl::iterator_traits<BidirectionalIterator1>::value_type value_type;
-
- if((first1 != last1) && (first2 != last2))
- {
- BidirectionalIterator1 it1(last1);
-
- while((--it1 != first1) && (eastl::find_if(first2, last2, eastl::bind1st<BinaryPredicate, value_type>(predicate, *it1)) != last2))
- ; // Do nothing
-
- if((it1 != first1) || (eastl::find_if(first2, last2, eastl::bind1st<BinaryPredicate, value_type>(predicate, *it1))) != last2)
- return it1;
- }
-
- return last1;
- }
-
-
-
-
- /// for_each
- ///
- /// Calls the Function function for each value in the range [first, last).
- /// Function takes a single parameter: the current value.
- ///
- /// Effects: Applies function to the result of dereferencing every iterator in
- /// the range [first, last), starting from first and proceeding to last 1.
- ///
- /// Returns: function.
- ///
- /// Complexity: Applies function exactly 'last - first' times.
- ///
- /// Note: If function returns a result, the result is ignored.
- ///
- template <typename InputIterator, typename Function>
- inline Function
- for_each(InputIterator first, InputIterator last, Function function)
- {
- for(; first != last; ++first)
- function(*first);
- return function;
- }
-
- /// for_each_n
- ///
- /// Calls the Function function for each value in the range [first, first + n).
- /// Function takes a single parameter: the current value.
- ///
- /// Effects: Applies function to the result of dereferencing every iterator in
- /// the range [first, first + n), starting from first and proceeding to last 1.
- ///
- /// Returns: first + n.
- ///
- /// Complexity: Applies function exactly 'first + n' times.
- ///
- /// Note:
- //// * If function returns a result, the result is ignored.
- //// * If n < 0, behaviour is undefined.
- ///
- template <typename InputIterator, typename Size, typename Function>
- EA_CPP14_CONSTEXPR inline InputIterator
- for_each_n(InputIterator first, Size n, Function function)
- {
- for (Size i = 0; i < n; ++first, i++)
- function(*first);
- return first;
- }
-
-
- /// generate
- ///
- /// Iterates the range of [first, last) and assigns to each element the
- /// result of the function generator. Generator is a function which takes
- /// no arguments.
- ///
- /// Complexity: Exactly 'last - first' invocations of generator and assignments.
- ///
- template <typename ForwardIterator, typename Generator>
- inline void
- generate(ForwardIterator first, ForwardIterator last, Generator generator)
- {
- for(; first != last; ++first) // We cannot call generate_n(first, last-first, generator)
- *first = generator(); // because the 'last-first' might not be supported by the
- } // given iterator.
-
-
- /// generate_n
- ///
- /// Iterates an interator n times and assigns the result of generator
- /// to each succeeding element. Generator is a function which takes
- /// no arguments.
- ///
- /// Complexity: Exactly n invocations of generator and assignments.
- ///
- template <typename OutputIterator, typename Size, typename Generator>
- inline OutputIterator
- generate_n(OutputIterator first, Size n, Generator generator)
- {
- for(; n > 0; --n, ++first)
- *first = generator();
- return first;
- }
-
-
- /// transform
- ///
- /// Iterates the input range of [first, last) and the output iterator result
- /// and assigns the result of unaryOperation(input) to result.
- ///
- /// Effects: Assigns through every iterator i in the range [result, result + (last1 - first1))
- /// a new corresponding value equal to unaryOperation(*(first1 + (i - result)).
- ///
- /// Requires: op shall not have any side effects.
- ///
- /// Returns: result + (last1 - first1). That is, returns the end of the output range.
- ///
- /// Complexity: Exactly 'last1 - first1' applications of unaryOperation.
- ///
- /// Note: result may be equal to first.
- ///
- template <typename InputIterator, typename OutputIterator, typename UnaryOperation>
- inline OutputIterator
- transform(InputIterator first, InputIterator last, OutputIterator result, UnaryOperation unaryOperation)
- {
- for(; first != last; ++first, ++result)
- *result = unaryOperation(*first);
- return result;
- }
-
-
- /// transform
- ///
- /// Iterates the input range of [first, last) and the output iterator result
- /// and assigns the result of binaryOperation(input1, input2) to result.
- ///
- /// Effects: Assigns through every iterator i in the range [result, result + (last1 - first1))
- /// a new corresponding value equal to binaryOperation(*(first1 + (i - result), *(first2 + (i - result))).
- ///
- /// Requires: binaryOperation shall not have any side effects.
- ///
- /// Returns: result + (last1 - first1). That is, returns the end of the output range.
- ///
- /// Complexity: Exactly 'last1 - first1' applications of binaryOperation.
- ///
- /// Note: result may be equal to first1 or first2.
- ///
- template <typename InputIterator1, typename InputIterator2, typename OutputIterator, typename BinaryOperation>
- inline OutputIterator
- transform(InputIterator1 first1, InputIterator1 last1, InputIterator2 first2, OutputIterator result, BinaryOperation binaryOperation)
- {
- for(; first1 != last1; ++first1, ++first2, ++result)
- *result = binaryOperation(*first1, *first2);
- return result;
- }
-
-
- /// equal
- ///
- /// Returns: true if for every iterator i in the range [first1, last1) the
- /// following corresponding conditions hold: predicate(*i, *(first2 + (i - first1))) != false.
- /// Otherwise, returns false.
- ///
- /// Complexity: At most last1 first1 applications of the corresponding predicate.
- ///
- /// To consider: Make specializations of this for scalar types and random access
- /// iterators that uses memcmp or some trick memory comparison function.
- /// We should verify that such a thing results in an improvement.
- ///
- template <typename InputIterator1, typename InputIterator2>
- EA_CPP14_CONSTEXPR inline bool equal(InputIterator1 first1, InputIterator1 last1, InputIterator2 first2)
- {
- for(; first1 != last1; ++first1, ++first2)
- {
- if(!(*first1 == *first2)) // Note that we always express value comparisons in terms of < or ==.
- return false;
- }
- return true;
- }
-
- /* Enable the following if there was shown to be some benefit. A glance and Microsoft VC++ memcmp
- shows that it is not optimized in any way, much less one that would benefit us here.
-
- inline bool equal(const bool* first1, const bool* last1, const bool* first2)
- { return (memcmp(first1, first2, (size_t)((uintptr_t)last1 - (uintptr_t)first1)) == 0); }
-
- inline bool equal(const char* first1, const char* last1, const char* first2)
- { return (memcmp(first1, first2, (size_t)((uintptr_t)last1 - (uintptr_t)first1)) == 0); }
-
- inline bool equal(const unsigned char* first1, const unsigned char* last1, const unsigned char* first2)
- { return (memcmp(first1, first2, (size_t)((uintptr_t)last1 - (uintptr_t)first1)) == 0); }
-
- inline bool equal(const signed char* first1, const signed char* last1, const signed char* first2)
- { return (memcmp(first1, first2, (size_t)((uintptr_t)last1 - (uintptr_t)first1)) == 0); }
-
- #ifndef EA_WCHAR_T_NON_NATIVE
- inline bool equal(const wchar_t* first1, const wchar_t* last1, const wchar_t* first2)
- { return (memcmp(first1, first2, (size_t)((uintptr_t)last1 - (uintptr_t)first1)) == 0); }
- #endif
-
- inline bool equal(const int16_t* first1, const int16_t* last1, const int16_t* first2)
- { return (memcmp(first1, first2, (size_t)((uintptr_t)last1 - (uintptr_t)first1)) == 0); }
-
- inline bool equal(const uint16_t* first1, const uint16_t* last1, const uint16_t* first2)
- { return (memcmp(first1, first2, (size_t)((uintptr_t)last1 - (uintptr_t)first1)) == 0); }
-
- inline bool equal(const int32_t* first1, const int32_t* last1, const int32_t* first2)
- { return (memcmp(first1, first2, (size_t)((uintptr_t)last1 - (uintptr_t)first1)) == 0); }
-
- inline bool equal(const uint32_t* first1, const uint32_t* last1, const uint32_t* first2)
- { return (memcmp(first1, first2, (size_t)((uintptr_t)last1 - (uintptr_t)first1)) == 0); }
-
- inline bool equal(const int64_t* first1, const int64_t* last1, const int64_t* first2)
- { return (memcmp(first1, first2, (size_t)((uintptr_t)last1 - (uintptr_t)first1)) == 0); }
-
- inline bool equal(const uint64_t* first1, const uint64_t* last1, const uint64_t* first2)
- { return (memcmp(first1, first2, (size_t)((uintptr_t)last1 - (uintptr_t)first1)) == 0); }
- */
-
-
-
- /// equal
- ///
- /// Returns: true if for every iterator i in the range [first1, last1) the
- /// following corresponding conditions hold: pred(*i, *(first2 + (i first1))) != false.
- /// Otherwise, returns false.
- ///
- /// Complexity: At most last1 first1 applications of the corresponding predicate.
- ///
- template <typename InputIterator1, typename InputIterator2, typename BinaryPredicate>
- inline bool
- equal(InputIterator1 first1, InputIterator1 last1, InputIterator2 first2, BinaryPredicate predicate)
- {
- for(; first1 != last1; ++first1, ++first2)
- {
- if(!predicate(*first1, *first2))
- return false;
- }
- return true;
- }
-
-
-
- /// identical
- ///
- /// Returns true if the two input ranges are equivalent.
- /// There is a subtle difference between this algorithm and
- /// the 'equal' algorithm. The equal algorithm assumes the
- /// two ranges are of equal length. This algorithm efficiently
- /// compares two ranges for both length equality and for
- /// element equality. There is no other standard algorithm
- /// that can do this.
- ///
- /// Returns: true if the sequence of elements defined by the range
- /// [first1, last1) is of the same length as the sequence of
- /// elements defined by the range of [first2, last2) and if
- /// the elements in these ranges are equal as per the
- /// equal algorithm.
- ///
- /// Complexity: At most 'min((last1 - first1), (last2 - first2))' applications
- /// of the corresponding comparison.
- ///
- template <typename InputIterator1, typename InputIterator2>
- bool identical(InputIterator1 first1, InputIterator1 last1,
- InputIterator2 first2, InputIterator2 last2)
- {
- while((first1 != last1) && (first2 != last2) && (*first1 == *first2))
- {
- ++first1;
- ++first2;
- }
- return (first1 == last1) && (first2 == last2);
- }
-
-
- /// identical
- ///
- template <typename InputIterator1, typename InputIterator2, typename BinaryPredicate>
- bool identical(InputIterator1 first1, InputIterator1 last1,
- InputIterator2 first2, InputIterator2 last2, BinaryPredicate predicate)
- {
- while((first1 != last1) && (first2 != last2) && predicate(*first1, *first2))
- {
- ++first1;
- ++first2;
- }
- return (first1 == last1) && (first2 == last2);
- }
-
-
-
- /// lexicographical_compare
- ///
- /// Returns: true if the sequence of elements defined by the range
- /// [first1, last1) is lexicographically less than the sequence of
- /// elements defined by the range [first2, last2). Returns false otherwise.
- ///
- /// Complexity: At most 'min((last1 - first1), (last2 - first2))' applications
- /// of the corresponding comparison.
- ///
- /// Note: If two sequences have the same number of elements and their
- /// corresponding elements are equivalent, then neither sequence is
- /// lexicographically less than the other. If one sequence is a prefix
- /// of the other, then the shorter sequence is lexicographically less
- /// than the longer sequence. Otherwise, the lexicographical comparison
- /// of the sequences yields the same result as the comparison of the first
- /// corresponding pair of elements that are not equivalent.
- ///
- template <typename InputIterator1, typename InputIterator2>
- inline bool
- lexicographical_compare(InputIterator1 first1, InputIterator1 last1, InputIterator2 first2, InputIterator2 last2)
- {
- for(; (first1 != last1) && (first2 != last2); ++first1, ++first2)
- {
- if(*first1 < *first2)
- return true;
- if(*first2 < *first1)
- return false;
- }
- return (first1 == last1) && (first2 != last2);
- }
-
- inline bool // Specialization for const char*.
- lexicographical_compare(const char* first1, const char* last1, const char* first2, const char* last2)
- {
- const ptrdiff_t n1(last1 - first1), n2(last2 - first2);
- const int result = memcmp(first1, first2, (size_t)eastl::min_alt(n1, n2));
- return result ? (result < 0) : (n1 < n2);
- }
-
- inline bool // Specialization for char*.
- lexicographical_compare(char* first1, char* last1, char* first2, char* last2)
- {
- const ptrdiff_t n1(last1 - first1), n2(last2 - first2);
- const int result = memcmp(first1, first2, (size_t)eastl::min_alt(n1, n2));
- return result ? (result < 0) : (n1 < n2);
- }
-
- inline bool // Specialization for const unsigned char*.
- lexicographical_compare(const unsigned char* first1, const unsigned char* last1, const unsigned char* first2, const unsigned char* last2)
- {
- const ptrdiff_t n1(last1 - first1), n2(last2 - first2);
- const int result = memcmp(first1, first2, (size_t)eastl::min_alt(n1, n2));
- return result ? (result < 0) : (n1 < n2);
- }
-
- inline bool // Specialization for unsigned char*.
- lexicographical_compare(unsigned char* first1, unsigned char* last1, unsigned char* first2, unsigned char* last2)
- {
- const ptrdiff_t n1(last1 - first1), n2(last2 - first2);
- const int result = memcmp(first1, first2, (size_t)eastl::min_alt(n1, n2));
- return result ? (result < 0) : (n1 < n2);
- }
-
- inline bool // Specialization for const signed char*.
- lexicographical_compare(const signed char* first1, const signed char* last1, const signed char* first2, const signed char* last2)
- {
- const ptrdiff_t n1(last1 - first1), n2(last2 - first2);
- const int result = memcmp(first1, first2, (size_t)eastl::min_alt(n1, n2));
- return result ? (result < 0) : (n1 < n2);
- }
-
- inline bool // Specialization for signed char*.
- lexicographical_compare(signed char* first1, signed char* last1, signed char* first2, signed char* last2)
- {
- const ptrdiff_t n1(last1 - first1), n2(last2 - first2);
- const int result = memcmp(first1, first2, (size_t)eastl::min_alt(n1, n2));
- return result ? (result < 0) : (n1 < n2);
- }
-
- #if defined(_MSC_VER) // If using the VC++ compiler (and thus bool is known to be a single byte)...
- //Not sure if this is a good idea.
- //inline bool // Specialization for const bool*.
- //lexicographical_compare(const bool* first1, const bool* last1, const bool* first2, const bool* last2)
- //{
- // const ptrdiff_t n1(last1 - first1), n2(last2 - first2);
- // const int result = memcmp(first1, first2, (size_t)eastl::min_alt(n1, n2));
- // return result ? (result < 0) : (n1 < n2);
- //}
- //
- //inline bool // Specialization for bool*.
- //lexicographical_compare(bool* first1, bool* last1, bool* first2, bool* last2)
- //{
- // const ptrdiff_t n1(last1 - first1), n2(last2 - first2);
- // const int result = memcmp(first1, first2, (size_t)eastl::min_alt(n1, n2));
- // return result ? (result < 0) : (n1 < n2);
- //}
- #endif
-
-
-
- /// lexicographical_compare
- ///
- /// Returns: true if the sequence of elements defined by the range
- /// [first1, last1) is lexicographically less than the sequence of
- /// elements defined by the range [first2, last2). Returns false otherwise.
- ///
- /// Complexity: At most 'min((last1 -first1), (last2 - first2))' applications
- /// of the corresponding comparison.
- ///
- /// Note: If two sequences have the same number of elements and their
- /// corresponding elements are equivalent, then neither sequence is
- /// lexicographically less than the other. If one sequence is a prefix
- /// of the other, then the shorter sequence is lexicographically less
- /// than the longer sequence. Otherwise, the lexicographical comparison
- /// of the sequences yields the same result as the comparison of the first
- /// corresponding pair of elements that are not equivalent.
- ///
- /// Note: False is always returned if range 1 is exhausted before range 2.
- /// The result of this is that you can't do a successful reverse compare
- /// (e.g. use greater<> as the comparison instead of less<>) unless the
- /// two sequences are of identical length. What you want to do is reverse
- /// the order of the arguments in order to get the desired effect.
- ///
- template <typename InputIterator1, typename InputIterator2, typename Compare>
- inline bool
- lexicographical_compare(InputIterator1 first1, InputIterator1 last1,
- InputIterator2 first2, InputIterator2 last2, Compare compare)
- {
- for(; (first1 != last1) && (first2 != last2); ++first1, ++first2)
- {
- if(compare(*first1, *first2))
- return true;
- if(compare(*first2, *first1))
- return false;
- }
- return (first1 == last1) && (first2 != last2);
- }
-
-
-#if defined(EA_COMPILER_HAS_THREE_WAY_COMPARISON)
-
- /// lexicographical_compare_three_way
- ///
- /// Returns: The comparison category ordering between both ranges. For the first non-equivalent pair in the ranges,
- /// the comparison will be returned. Else if the first range is a subset (superset) of the second range, then the
- /// less (greater) ordering will be returned.
- ///
- /// Complexity: At most N iterations, where N = min(last1-first1, last2-first2) of the applications
- /// of the corresponding comparison.
- ///
- /// Note: If two sequences have the same number of elements and their
- /// corresponding elements are equivalent, then neither sequence is
- /// lexicographically less than the other. If one sequence is a prefix
- /// of the other, then the shorter sequence is lexicographically less
- /// than the longer sequence. Otherwise, the lexicographical comparison
- /// of the sequences yields the same result as the comparison of the first
- /// corresponding pair of elements that are not equivalent.
- ///
- template <typename InputIterator1, typename InputIterator2, typename Compare>
- constexpr auto lexicographical_compare_three_way(InputIterator1 first1, InputIterator1 last1,
- InputIterator2 first2, InputIterator2 last2,
- Compare compare) -> decltype(compare(*first1, *first2))
- {
- for (; (first1 != last1) && (first2 != last2); ++first1, ++first2)
- {
- if (auto c = compare(*first1, *first2); c != 0)
- return c;
- }
-
- return (first1 != last1) ? std::strong_ordering::greater :
- (first2 != last2) ? std::strong_ordering::less :
- std::strong_ordering::equal;
- }
-#endif
-
- /// mismatch
- ///
- /// Finds the first position where the two ranges [first1, last1) and
- /// [first2, first2 + (last1 - first1)) differ. The two versions of
- /// mismatch use different tests for whether elements differ.
- ///
- /// Returns: A pair of iterators i and j such that j == first2 + (i - first1)
- /// and i is the first iterator in the range [first1, last1) for which the
- /// following corresponding condition holds: !(*i == *(first2 + (i - first1))).
- /// Returns the pair last1 and first2 + (last1 - first1) if such an iterator
- /// i is not found.
- ///
- /// Complexity: At most last1 first1 applications of the corresponding predicate.
- ///
- template <class InputIterator1, class InputIterator2>
- inline eastl::pair<InputIterator1, InputIterator2>
- mismatch(InputIterator1 first1, InputIterator1 last1,
- InputIterator2 first2) // , InputIterator2 last2)
- {
- while((first1 != last1) && (*first1 == *first2)) // && (first2 != last2) <- C++ standard mismatch function doesn't check first2/last2.
- {
- ++first1;
- ++first2;
- }
-
- return eastl::pair<InputIterator1, InputIterator2>(first1, first2);
- }
-
-
- /// mismatch
- ///
- /// Finds the first position where the two ranges [first1, last1) and
- /// [first2, first2 + (last1 - first1)) differ. The two versions of
- /// mismatch use different tests for whether elements differ.
- ///
- /// Returns: A pair of iterators i and j such that j == first2 + (i - first1)
- /// and i is the first iterator in the range [first1, last1) for which the
- /// following corresponding condition holds: pred(*i, *(first2 + (i - first1))) == false.
- /// Returns the pair last1 and first2 + (last1 - first1) if such an iterator
- /// i is not found.
- ///
- /// Complexity: At most last1 first1 applications of the corresponding predicate.
- ///
- template <class InputIterator1, class InputIterator2, class BinaryPredicate>
- inline eastl::pair<InputIterator1, InputIterator2>
- mismatch(InputIterator1 first1, InputIterator1 last1,
- InputIterator2 first2, // InputIterator2 last2,
- BinaryPredicate predicate)
- {
- while((first1 != last1) && predicate(*first1, *first2)) // && (first2 != last2) <- C++ standard mismatch function doesn't check first2/last2.
- {
- ++first1;
- ++first2;
- }
-
- return eastl::pair<InputIterator1, InputIterator2>(first1, first2);
- }
-
-
- /// lower_bound
- ///
- /// Finds the position of the first element in a sorted range that has a value
- /// greater than or equivalent to a specified value.
- ///
- /// Effects: Finds the first position into which value can be inserted without
- /// violating the ordering.
- ///
- /// Returns: The furthermost iterator i in the range [first, last) such that
- /// for any iterator j in the range [first, i) the following corresponding
- /// condition holds: *j < value.
- ///
- /// Complexity: At most 'log(last - first) + 1' comparisons.
- ///
- /// Optimizations: We have no need to specialize this implementation for random
- /// access iterators (e.g. contiguous array), as the code below will already
- /// take advantage of them.
- ///
- template <typename ForwardIterator, typename T>
- ForwardIterator
- lower_bound(ForwardIterator first, ForwardIterator last, const T& value)
- {
- typedef typename eastl::iterator_traits<ForwardIterator>::difference_type DifferenceType;
-
- DifferenceType d = eastl::distance(first, last); // This will be efficient for a random access iterator such as an array.
-
- while(d > 0)
- {
- ForwardIterator i = first;
- DifferenceType d2 = d >> 1; // We use '>>1' here instead of '/2' because MSVC++ for some reason generates significantly worse code for '/2'. Go figure.
-
- eastl::advance(i, d2); // This will be efficient for a random access iterator such as an array.
-
- if(*i < value)
- {
- // Disabled because std::lower_bound doesn't specify (23.3.3.3, p3) this can be done: EASTL_VALIDATE_COMPARE(!(value < *i)); // Validate that the compare function is sane.
- first = ++i;
- d -= d2 + 1;
- }
- else
- d = d2;
- }
- return first;
- }
-
-
- /// lower_bound
- ///
- /// Finds the position of the first element in a sorted range that has a value
- /// greater than or equivalent to a specified value. The input Compare function
- /// takes two arguments and returns true if the first argument is less than
- /// the second argument.
- ///
- /// Effects: Finds the first position into which value can be inserted without
- /// violating the ordering.
- ///
- /// Returns: The furthermost iterator i in the range [first, last) such that
- /// for any iterator j in the range [first, i) the following corresponding
- /// condition holds: compare(*j, value) != false.
- ///
- /// Complexity: At most 'log(last - first) + 1' comparisons.
- ///
- /// Optimizations: We have no need to specialize this implementation for random
- /// access iterators (e.g. contiguous array), as the code below will already
- /// take advantage of them.
- ///
- template <typename ForwardIterator, typename T, typename Compare>
- ForwardIterator
- lower_bound(ForwardIterator first, ForwardIterator last, const T& value, Compare compare)
- {
- typedef typename eastl::iterator_traits<ForwardIterator>::difference_type DifferenceType;
-
- DifferenceType d = eastl::distance(first, last); // This will be efficient for a random access iterator such as an array.
-
- while(d > 0)
- {
- ForwardIterator i = first;
- DifferenceType d2 = d >> 1; // We use '>>1' here instead of '/2' because MSVC++ for some reason generates significantly worse code for '/2'. Go figure.
-
- eastl::advance(i, d2); // This will be efficient for a random access iterator such as an array.
-
- if(compare(*i, value))
- {
- // Disabled because std::lower_bound doesn't specify (23.3.3.1, p3) this can be done: EASTL_VALIDATE_COMPARE(!compare(value, *i)); // Validate that the compare function is sane.
- first = ++i;
- d -= d2 + 1;
- }
- else
- d = d2;
- }
- return first;
- }
-
-
-
- /// upper_bound
- ///
- /// Finds the position of the first element in a sorted range that has a
- /// value that is greater than a specified value.
- ///
- /// Effects: Finds the furthermost position into which value can be inserted
- /// without violating the ordering.
- ///
- /// Returns: The furthermost iterator i in the range [first, last) such that
- /// for any iterator j in the range [first, i) the following corresponding
- /// condition holds: !(value < *j).
- ///
- /// Complexity: At most 'log(last - first) + 1' comparisons.
- ///
- template <typename ForwardIterator, typename T>
- ForwardIterator
- upper_bound(ForwardIterator first, ForwardIterator last, const T& value)
- {
- typedef typename eastl::iterator_traits<ForwardIterator>::difference_type DifferenceType;
-
- DifferenceType len = eastl::distance(first, last);
-
- while(len > 0)
- {
- ForwardIterator i = first;
- DifferenceType len2 = len >> 1; // We use '>>1' here instead of '/2' because MSVC++ for some reason generates significantly worse code for '/2'. Go figure.
-
- eastl::advance(i, len2);
-
- if(!(value < *i)) // Note that we always express value comparisons in terms of < or ==.
- {
- first = ++i;
- len -= len2 + 1;
- }
- else
- {
- // Disabled because std::upper_bound doesn't specify (23.3.3.2, p3) this can be done: EASTL_VALIDATE_COMPARE(!(*i < value)); // Validate that the compare function is sane.
- len = len2;
- }
- }
- return first;
- }
-
-
- /// upper_bound
- ///
- /// Finds the position of the first element in a sorted range that has a
- /// value that is greater than a specified value. The input Compare function
- /// takes two arguments and returns true if the first argument is less than
- /// the second argument.
- ///
- /// Effects: Finds the furthermost position into which value can be inserted
- /// without violating the ordering.
- ///
- /// Returns: The furthermost iterator i in the range [first, last) such that
- /// for any iterator j in the range [first, i) the following corresponding
- /// condition holds: compare(value, *j) == false.
- ///
- /// Complexity: At most 'log(last - first) + 1' comparisons.
- ///
- template <typename ForwardIterator, typename T, typename Compare>
- ForwardIterator
- upper_bound(ForwardIterator first, ForwardIterator last, const T& value, Compare compare)
- {
- typedef typename eastl::iterator_traits<ForwardIterator>::difference_type DifferenceType;
-
- DifferenceType len = eastl::distance(first, last);
-
- while(len > 0)
- {
- ForwardIterator i = first;
- DifferenceType len2 = len >> 1; // We use '>>1' here instead of '/2' because MSVC++ for some reason generates significantly worse code for '/2'. Go figure.
-
- eastl::advance(i, len2);
-
- if(!compare(value, *i))
- {
- first = ++i;
- len -= len2 + 1;
- }
- else
- {
- // Disabled because std::upper_bound doesn't specify (23.3.3.2, p3) this can be done: EASTL_VALIDATE_COMPARE(!compare(*i, value)); // Validate that the compare function is sane.
- len = len2;
- }
- }
- return first;
- }
-
-
- /// equal_range
- ///
- /// Effects: Finds the largest subrange [i, j) such that the value can be inserted
- /// at any iterator k in it without violating the ordering. k satisfies the
- /// corresponding conditions: !(*k < value) && !(value < *k).
- ///
- /// Complexity: At most '2 * log(last - first) + 1' comparisons.
- ///
- template <typename ForwardIterator, typename T>
- pair<ForwardIterator, ForwardIterator>
- equal_range(ForwardIterator first, ForwardIterator last, const T& value)
- {
- typedef pair<ForwardIterator, ForwardIterator> ResultType;
- typedef typename eastl::iterator_traits<ForwardIterator>::difference_type DifferenceType;
-
- DifferenceType d = eastl::distance(first, last);
-
- while(d > 0)
- {
- ForwardIterator i(first);
- DifferenceType d2 = d >> 1; // We use '>>1' here instead of '/2' because MSVC++ for some reason generates significantly worse code for '/2'. Go figure.
-
- eastl::advance(i, d2);
-
- if(*i < value)
- {
- EASTL_VALIDATE_COMPARE(!(value < *i)); // Validate that the compare function is sane.
- first = ++i;
- d -= d2 + 1;
- }
- else if(value < *i)
- {
- EASTL_VALIDATE_COMPARE(!(*i < value)); // Validate that the compare function is sane.
- d = d2;
- last = i;
- }
- else
- {
- ForwardIterator j(i);
-
- return ResultType(eastl::lower_bound(first, i, value),
- eastl::upper_bound(++j, last, value));
- }
- }
- return ResultType(first, first);
- }
-
-
- /// equal_range
- ///
- /// Effects: Finds the largest subrange [i, j) such that the value can be inserted
- /// at any iterator k in it without violating the ordering. k satisfies the
- /// corresponding conditions: compare(*k, value) == false && compare(value, *k) == false.
- ///
- /// Complexity: At most '2 * log(last - first) + 1' comparisons.
- ///
- template <typename ForwardIterator, typename T, typename Compare>
- pair<ForwardIterator, ForwardIterator>
- equal_range(ForwardIterator first, ForwardIterator last, const T& value, Compare compare)
- {
- typedef pair<ForwardIterator, ForwardIterator> ResultType;
- typedef typename eastl::iterator_traits<ForwardIterator>::difference_type DifferenceType;
-
- DifferenceType d = eastl::distance(first, last);
-
- while(d > 0)
- {
- ForwardIterator i(first);
- DifferenceType d2 = d >> 1; // We use '>>1' here instead of '/2' because MSVC++ for some reason generates significantly worse code for '/2'. Go figure.
-
- eastl::advance(i, d2);
-
- if(compare(*i, value))
- {
- EASTL_VALIDATE_COMPARE(!compare(value, *i)); // Validate that the compare function is sane.
- first = ++i;
- d -= d2 + 1;
- }
- else if(compare(value, *i))
- {
- EASTL_VALIDATE_COMPARE(!compare(*i, value)); // Validate that the compare function is sane.
- d = d2;
- last = i;
- }
- else
- {
- ForwardIterator j(i);
-
- return ResultType(eastl::lower_bound(first, i, value, compare),
- eastl::upper_bound(++j, last, value, compare));
- }
- }
- return ResultType(first, first);
- }
-
-
- /// replace
- ///
- /// Effects: Substitutes elements referred by the iterator i in the range [first, last)
- /// with new_value, when the following corresponding conditions hold: *i == old_value.
- ///
- /// Complexity: Exactly 'last - first' applications of the corresponding predicate.
- ///
- /// Note: The predicate version of replace is replace_if and not another variation of replace.
- /// This is because both versions would have the same parameter count and there could be ambiguity.
- ///
- template <typename ForwardIterator, typename T>
- inline void
- replace(ForwardIterator first, ForwardIterator last, const T& old_value, const T& new_value)
- {
- for(; first != last; ++first)
- {
- if(*first == old_value)
- *first = new_value;
- }
- }
-
-
- /// replace_if
- ///
- /// Effects: Substitutes elements referred by the iterator i in the range [first, last)
- /// with new_value, when the following corresponding conditions hold: predicate(*i) != false.
- ///
- /// Complexity: Exactly 'last - first' applications of the corresponding predicate.
- ///
- /// Note: The predicate version of replace_if is replace and not another variation of replace_if.
- /// This is because both versions would have the same parameter count and there could be ambiguity.
- ///
- template <typename ForwardIterator, typename Predicate, typename T>
- inline void
- replace_if(ForwardIterator first, ForwardIterator last, Predicate predicate, const T& new_value)
- {
- for(; first != last; ++first)
- {
- if(predicate(*first))
- *first = new_value;
- }
- }
-
-
- /// remove_copy
- ///
- /// Effects: Copies all the elements referred to by the iterator i in the range
- /// [first, last) for which the following corresponding condition does not hold:
- /// *i == value.
- ///
- /// Requires: The ranges [first, last) and [result, result + (last - first)) shall not overlap.
- ///
- /// Returns: The end of the resulting range.
- ///
- /// Complexity: Exactly 'last - first' applications of the corresponding predicate.
- ///
- template <typename InputIterator, typename OutputIterator, typename T>
- inline OutputIterator
- remove_copy(InputIterator first, InputIterator last, OutputIterator result, const T& value)
- {
- for(; first != last; ++first)
- {
- if(!(*first == value)) // Note that we always express value comparisons in terms of < or ==.
- {
- *result = eastl::move(*first);
- ++result;
- }
- }
- return result;
- }
-
-
- /// remove_copy_if
- ///
- /// Effects: Copies all the elements referred to by the iterator i in the range
- /// [first, last) for which the following corresponding condition does not hold:
- /// predicate(*i) != false.
- ///
- /// Requires: The ranges [first, last) and [result, result + (last - first)) shall not overlap.
- ///
- /// Returns: The end of the resulting range.
- ///
- /// Complexity: Exactly 'last - first' applications of the corresponding predicate.
- ///
- template <typename InputIterator, typename OutputIterator, typename Predicate>
- inline OutputIterator
- remove_copy_if(InputIterator first, InputIterator last, OutputIterator result, Predicate predicate)
- {
- for(; first != last; ++first)
- {
- if(!predicate(*first))
- {
- *result = eastl::move(*first);
- ++result;
- }
- }
- return result;
- }
-
-
- /// remove
- ///
- /// Effects: Eliminates all the elements referred to by iterator i in the
- /// range [first, last) for which the following corresponding condition
- /// holds: *i == value.
- ///
- /// Returns: The end of the resulting range.
- ///
- /// Complexity: Exactly 'last - first' applications of the corresponding predicate.
- ///
- /// Note: The predicate version of remove is remove_if and not another variation of remove.
- /// This is because both versions would have the same parameter count and there could be ambiguity.
- ///
- /// Note: Since this function moves the element to the back of the heap and
- /// doesn't actually remove it from the given container, the user must call
- /// the container erase function if the user wants to erase the element
- /// from the container.
- ///
- /// Example usage:
- /// vector<int> intArray;
- /// ...
- /// intArray.erase(remove(intArray.begin(), intArray.end(), 4), intArray.end()); // Erase all elements of value 4.
- ///
- template <typename ForwardIterator, typename T>
- inline ForwardIterator
- remove(ForwardIterator first, ForwardIterator last, const T& value)
- {
- first = eastl::find(first, last, value);
- if(first != last)
- {
- ForwardIterator i(first);
- return eastl::remove_copy(++i, last, first, value);
- }
- return first;
- }
-
-
- /// remove_if
- ///
- /// Effects: Eliminates all the elements referred to by iterator i in the
- /// range [first, last) for which the following corresponding condition
- /// holds: predicate(*i) != false.
- ///
- /// Returns: The end of the resulting range.
- ///
- /// Complexity: Exactly 'last - first' applications of the corresponding predicate.
- ///
- /// Note: The predicate version of remove_if is remove and not another variation of remove_if.
- /// This is because both versions would have the same parameter count and there could be ambiguity.
- ///
- /// Note: Since this function moves the element to the back of the heap and
- /// doesn't actually remove it from the given container, the user must call
- /// the container erase function if the user wants to erase the element
- /// from the container.
- ///
- /// Example usage:
- /// vector<int> intArray;
- /// ...
- /// intArray.erase(remove(intArray.begin(), intArray.end(), bind2nd(less<int>(), (int)3)), intArray.end()); // Erase all elements less than 3.
- ///
- template <typename ForwardIterator, typename Predicate>
- inline ForwardIterator
- remove_if(ForwardIterator first, ForwardIterator last, Predicate predicate)
- {
- first = eastl::find_if(first, last, predicate);
- if(first != last)
- {
- ForwardIterator i(first);
- return eastl::remove_copy_if<ForwardIterator, ForwardIterator, Predicate>(++i, last, first, predicate);
- }
- return first;
- }
-
-
- /// apply_and_remove_if
- ///
- /// Calls the Function function for all elements referred to my iterator i in the range
- /// [first, last) for which the following corresponding condition holds:
- /// predicate(*i) == true
- /// and then left shift moves potential non-matching elements over it.
- ///
- /// Returns: a past-the-end iterator for the new end of the range.
- ///
- /// Complexity: Exactly 'last - first' applications of the corresponding predicate + applies
- /// function once for every time the condition holds.
- ///
- /// Note: Since removing is done by shifting (by means of copy move assignment) the elements
- /// in the range in such a way that the elements that are not to be removed appear in the
- /// beginning of the range doesn't actually remove it from the given container, the user must call
- /// the container erase function if the user wants to erase the element
- /// from the container. I.e. in the same they as for remove_if the excess elements
- /// are left in a valid but possibly moved from state.
- ///
- template <typename ForwardIterator, typename Function, typename Predicate>
- inline ForwardIterator apply_and_remove_if(ForwardIterator first,
- ForwardIterator last,
- Function function,
- Predicate predicate)
- {
- first = eastl::find_if(first, last, predicate);
- if (first != last)
- {
- function(*first);
- for (auto i = next(first); i != last; ++i)
- {
- if (predicate(*i))
- {
- function(*i);
- continue;
- }
- *first = eastl::move(*i);
- ++first;
- }
- }
- return first;
- }
-
-
- /// apply_and_remove
- ///
- /// Calls the Function function for all elements referred to my iterator i in the range
- /// [first, last) for which the following corresponding condition holds:
- /// value == *i
- /// and then left shift moves potential non-matching elements over it.
- ///
- /// Returns: a past-the-end iterator for the new end of the range.
- ///
- /// Complexity: Exactly 'last - first' applications of the corresponding equality test
- /// + applies function once for every time the condition holds.
- ///
- /// Note: Since removing is done by shifting (by means of copy move assignment) the elements
- /// in the range in such a way that the elements that are not to be removed appear in the
- /// beginning of the range doesn't actually remove it from the given container, the user must call
- /// the container erase function if the user wants to erase the element
- /// from the container. I.e. in the same they as for remove_if the excess elements
- /// are left in a valid but possibly moved from state.
- ///
- template <typename ForwardIterator, typename Function, typename T>
- inline ForwardIterator apply_and_remove(ForwardIterator first,
- ForwardIterator last,
- Function function,
- const T& value)
- {
- first = eastl::find(first, last, value);
- if (first != last)
- {
- function(*first);
- for (auto i = next(first); i != last; ++i)
- {
- if (value == *i)
- {
- function(*i);
- continue;
- }
- *first = eastl::move(*i);
- ++first;
- }
- }
- return first;
- }
-
-
- /// replace_copy
- ///
- /// Effects: Assigns to every iterator i in the range [result, result + (last - first))
- /// either new_value or *(first + (i - result)) depending on whether the following
- /// corresponding conditions hold: *(first + (i - result)) == old_value.
- ///
- /// Requires: The ranges [first, last) and [result, result + (last - first)) shall not overlap.
- ///
- /// Returns: result + (last - first).
- ///
- /// Complexity: Exactly 'last - first' applications of the corresponding predicate.
- ///
- /// Note: The predicate version of replace_copy is replace_copy_if and not another variation of replace_copy.
- /// This is because both versions would have the same parameter count and there could be ambiguity.
- ///
- template <typename InputIterator, typename OutputIterator, typename T>
- inline OutputIterator
- replace_copy(InputIterator first, InputIterator last, OutputIterator result, const T& old_value, const T& new_value)
- {
- for(; first != last; ++first, ++result)
- *result = (*first == old_value) ? new_value : *first;
- return result;
- }
-
-
- /// replace_copy_if
- ///
- /// Effects: Assigns to every iterator i in the range [result, result + (last - first))
- /// either new_value or *(first + (i - result)) depending on whether the following
- /// corresponding conditions hold: predicate(*(first + (i - result))) != false.
- ///
- /// Requires: The ranges [first, last) and [result, result+(lastfirst)) shall not overlap.
- ///
- /// Returns: result + (last - first).
- ///
- /// Complexity: Exactly 'last - first' applications of the corresponding predicate.
- ///
- /// Note: The predicate version of replace_copy_if is replace_copy and not another variation of replace_copy_if.
- /// This is because both versions would have the same parameter count and there could be ambiguity.
- ///
- template <typename InputIterator, typename OutputIterator, typename Predicate, typename T>
- inline OutputIterator
- replace_copy_if(InputIterator first, InputIterator last, OutputIterator result, Predicate predicate, const T& new_value)
- {
- for(; first != last; ++first, ++result)
- *result = predicate(*first) ? new_value : *first;
- return result;
- }
-
-
-
-
- // reverse
- //
- // We provide helper functions which allow reverse to be implemented more
- // efficiently for some types of iterators and types.
- //
- template <typename BidirectionalIterator>
- inline void reverse_impl(BidirectionalIterator first, BidirectionalIterator last, EASTL_ITC_NS::bidirectional_iterator_tag)
- {
- for(; (first != last) && (first != --last); ++first) // We are not allowed to use operator <, <=, >, >= with a
- eastl::iter_swap(first, last); // generic (bidirectional or otherwise) iterator.
- }
-
- template <typename RandomAccessIterator>
- inline void reverse_impl(RandomAccessIterator first, RandomAccessIterator last, EASTL_ITC_NS::random_access_iterator_tag)
- {
- if(first != last)
- {
- for(; first < --last; ++first) // With a random access iterator, we can use operator < to more efficiently implement
- eastl::iter_swap(first, last); // this algorithm. A generic iterator doesn't necessarily have an operator < defined.
- }
- }
-
- /// reverse
- ///
- /// Reverses the values within the range [first, last).
- ///
- /// Effects: For each nonnegative integer i <= (last - first) / 2,
- /// applies swap to all pairs of iterators first + i, (last i) - 1.
- ///
- /// Complexity: Exactly '(last - first) / 2' swaps.
- ///
- template <typename BidirectionalIterator>
- inline void reverse(BidirectionalIterator first, BidirectionalIterator last)
- {
- typedef typename eastl::iterator_traits<BidirectionalIterator>::iterator_category IC;
- eastl::reverse_impl(first, last, IC());
- }
-
-
-
- /// reverse_copy
- ///
- /// Copies the range [first, last) in reverse order to the result.
- ///
- /// Effects: Copies the range [first, last) to the range
- /// [result, result + (last - first)) such that for any nonnegative
- /// integer i < (last - first) the following assignment takes place:
- /// *(result + (last - first) - i) = *(first + i)
- ///
- /// Requires: The ranges [first, last) and [result, result + (last - first))
- /// shall not overlap.
- ///
- /// Returns: result + (last - first). That is, returns the end of the output range.
- ///
- /// Complexity: Exactly 'last - first' assignments.
- ///
- template <typename BidirectionalIterator, typename OutputIterator>
- inline OutputIterator
- reverse_copy(BidirectionalIterator first, BidirectionalIterator last, OutputIterator result)
- {
- for(; first != last; ++result)
- *result = *--last;
- return result;
- }
-
-
-
- /// search
- ///
- /// Search finds a subsequence within the range [first1, last1) that is identical to [first2, last2)
- /// when compared element-by-element. It returns an iterator pointing to the beginning of that
- /// subsequence, or else last1 if no such subsequence exists. As such, it is very much like
- /// the C strstr function, with the primary difference being that strstr uses 0-terminated strings
- /// whereas search uses an end iterator to specify the end of a string.
- ///
- /// Returns: The first iterator i in the range [first1, last1 - (last2 - first2)) such that for
- /// any nonnegative integer n less than 'last2 - first2' the following corresponding condition holds:
- /// *(i + n) == *(first2 + n). Returns last1 if no such iterator is found.
- ///
- /// Complexity: At most (last1 first1) * (last2 first2) applications of the corresponding predicate.
- ///
- template <typename ForwardIterator1, typename ForwardIterator2>
- ForwardIterator1
- search(ForwardIterator1 first1, ForwardIterator1 last1,
- ForwardIterator2 first2, ForwardIterator2 last2)
- {
- if(first2 != last2) // If there is anything to search for...
- {
- // We need to make a special case for a pattern of one element,
- // as the logic below prevents one element patterns from working.
- ForwardIterator2 temp2(first2);
- ++temp2;
-
- if(temp2 != last2) // If what we are searching for has a length > 1...
- {
- ForwardIterator1 cur1(first1);
- ForwardIterator2 p2;
-
- while(first1 != last1)
- {
- // The following loop is the equivalent of eastl::find(first1, last1, *first2)
- while((first1 != last1) && !(*first1 == *first2))
- ++first1;
-
- if(first1 != last1)
- {
- p2 = temp2;
- cur1 = first1;
-
- if(++cur1 != last1)
- {
- while(*cur1 == *p2)
- {
- if(++p2 == last2)
- return first1;
-
- if(++cur1 == last1)
- return last1;
- }
-
- ++first1;
- continue;
- }
- }
- return last1;
- }
-
- // Fall through to the end.
- }
- else
- return eastl::find(first1, last1, *first2);
- }
-
- return first1;
-
-
- #if 0
- /* Another implementation which is a little more simpler but executes a little slower on average.
- typedef typename eastl::iterator_traits<ForwardIterator1>::difference_type difference_type_1;
- typedef typename eastl::iterator_traits<ForwardIterator2>::difference_type difference_type_2;
-
- const difference_type_2 d2 = eastl::distance(first2, last2);
-
- for(difference_type_1 d1 = eastl::distance(first1, last1); d1 >= d2; ++first1, --d1)
- {
- ForwardIterator1 temp1 = first1;
-
- for(ForwardIterator2 temp2 = first2; ; ++temp1, ++temp2)
- {
- if(temp2 == last2)
- return first1;
- if(!(*temp1 == *temp2))
- break;
- }
- }
-
- return last1;
- */
- #endif
- }
-
-
- /// search
- ///
- /// Search finds a subsequence within the range [first1, last1) that is identical to [first2, last2)
- /// when compared element-by-element. It returns an iterator pointing to the beginning of that
- /// subsequence, or else last1 if no such subsequence exists. As such, it is very much like
- /// the C strstr function, with the only difference being that strstr uses 0-terminated strings
- /// whereas search uses an end iterator to specify the end of a string.
- ///
- /// Returns: The first iterator i in the range [first1, last1 - (last2 - first2)) such that for
- /// any nonnegative integer n less than 'last2 - first2' the following corresponding condition holds:
- /// predicate(*(i + n), *(first2 + n)) != false. Returns last1 if no such iterator is found.
- ///
- /// Complexity: At most (last1 first1) * (last2 first2) applications of the corresponding predicate.
- ///
- template <typename ForwardIterator1, typename ForwardIterator2, typename BinaryPredicate>
- ForwardIterator1
- search(ForwardIterator1 first1, ForwardIterator1 last1,
- ForwardIterator2 first2, ForwardIterator2 last2,
- BinaryPredicate predicate)
- {
- typedef typename eastl::iterator_traits<ForwardIterator1>::difference_type difference_type_1;
- typedef typename eastl::iterator_traits<ForwardIterator2>::difference_type difference_type_2;
-
- difference_type_2 d2 = eastl::distance(first2, last2);
-
- if(d2 != 0)
- {
- ForwardIterator1 i(first1);
- eastl::advance(i, d2);
-
- for(difference_type_1 d1 = eastl::distance(first1, last1); d1 >= d2; --d1)
- {
- if(eastl::equal<ForwardIterator1, ForwardIterator2, BinaryPredicate>(first1, i, first2, predicate))
- return first1;
- if(d1 > d2) // To do: Find a way to make the algorithm more elegant.
- {
- ++first1;
- ++i;
- }
- }
- return last1;
- }
- return first1; // Just like with strstr, we return first1 if the match string is empty.
- }
-
-
-
- // search_n helper functions
- //
- template <typename ForwardIterator, typename Size, typename T>
- ForwardIterator // Generic implementation.
- search_n_impl(ForwardIterator first, ForwardIterator last, Size count, const T& value, EASTL_ITC_NS::forward_iterator_tag)
- {
- if(count <= 0)
- return first;
-
- Size d1 = (Size)eastl::distance(first, last); // Should d1 be of type Size, ptrdiff_t, or iterator_traits<ForwardIterator>::difference_type?
- // The problem with using iterator_traits<ForwardIterator>::difference_type is that
- if(count > d1) // ForwardIterator may not be a true iterator but instead something like a pointer.
- return last;
-
- for(; d1 >= count; ++first, --d1)
- {
- ForwardIterator i(first);
-
- for(Size n = 0; n < count; ++n, ++i, --d1)
- {
- if(!(*i == value)) // Note that we always express value comparisons in terms of < or ==.
- goto not_found;
- }
- return first;
-
- not_found:
- first = i;
- }
- return last;
- }
-
- template <typename RandomAccessIterator, typename Size, typename T> inline
- RandomAccessIterator // Random access iterator implementation. Much faster than generic implementation.
- search_n_impl(RandomAccessIterator first, RandomAccessIterator last, Size count, const T& value, EASTL_ITC_NS::random_access_iterator_tag)
- {
- if(count <= 0)
- return first;
- else if(count == 1)
- return eastl::find(first, last, value);
- else if(last > first)
- {
- RandomAccessIterator lookAhead;
- RandomAccessIterator backTrack;
-
- Size skipOffset = (count - 1);
- Size tailSize = (Size)(last - first);
- Size remainder;
- Size prevRemainder;
-
- for(lookAhead = first + skipOffset; tailSize >= count; lookAhead += count)
- {
- tailSize -= count;
-
- if(*lookAhead == value)
- {
- remainder = skipOffset;
-
- for(backTrack = lookAhead - 1; *backTrack == value; --backTrack)
- {
- if(--remainder == 0)
- return (lookAhead - skipOffset); // success
- }
-
- if(remainder <= tailSize)
- {
- prevRemainder = remainder;
-
- while(*(++lookAhead) == value)
- {
- if(--remainder == 0)
- return (backTrack + 1); // success
- }
- tailSize -= (prevRemainder - remainder);
- }
- else
- return last; // failure
- }
-
- // lookAhead here is always pointing to the element of the last mismatch.
- }
- }
-
- return last; // failure
- }
-
-
- /// search_n
- ///
- /// Returns: The first iterator i in the range [first, last count) such that
- /// for any nonnegative integer n less than count the following corresponding
- /// conditions hold: *(i + n) == value, pred(*(i + n),value) != false.
- /// Returns last if no such iterator is found.
- ///
- /// Complexity: At most '(last1 - first1) * count' applications of the corresponding predicate.
- ///
- template <typename ForwardIterator, typename Size, typename T>
- ForwardIterator
- search_n(ForwardIterator first, ForwardIterator last, Size count, const T& value)
- {
- typedef typename eastl::iterator_traits<ForwardIterator>::iterator_category IC;
- return eastl::search_n_impl(first, last, count, value, IC());
- }
-
-
- /// binary_search
- ///
- /// Returns: true if there is an iterator i in the range [first last) that
- /// satisfies the corresponding conditions: !(*i < value) && !(value < *i).
- ///
- /// Complexity: At most 'log(last - first) + 2' comparisons.
- ///
- /// Note: The reason binary_search returns bool instead of an iterator is
- /// that search_n, lower_bound, or equal_range already return an iterator.
- /// However, there are arguments that binary_search should return an iterator.
- /// Note that we provide binary_search_i (STL extension) to return an iterator.
- ///
- /// To use search_n to find an item, do this:
- /// iterator i = search_n(begin, end, 1, value);
- /// To use lower_bound to find an item, do this:
- /// iterator i = lower_bound(begin, end, value);
- /// if((i != last) && !(value < *i))
- /// <use the iterator>
- /// It turns out that the above lower_bound method is as fast as binary_search
- /// would be if it returned an iterator.
- ///
- template <typename ForwardIterator, typename T>
- inline bool
- binary_search(ForwardIterator first, ForwardIterator last, const T& value)
- {
- // To do: This can be made slightly faster by not using lower_bound.
- ForwardIterator i(eastl::lower_bound<ForwardIterator, T>(first, last, value));
- return ((i != last) && !(value < *i)); // Note that we always express value comparisons in terms of < or ==.
- }
-
-
- /// binary_search
- ///
- /// Returns: true if there is an iterator i in the range [first last) that
- /// satisfies the corresponding conditions: compare(*i, value) == false &&
- /// compare(value, *i) == false.
- ///
- /// Complexity: At most 'log(last - first) + 2' comparisons.
- ///
- /// Note: See comments above regarding the bool return value of binary_search.
- ///
- template <typename ForwardIterator, typename T, typename Compare>
- inline bool
- binary_search(ForwardIterator first, ForwardIterator last, const T& value, Compare compare)
- {
- // To do: This can be made slightly faster by not using lower_bound.
- ForwardIterator i(eastl::lower_bound<ForwardIterator, T, Compare>(first, last, value, compare));
- return ((i != last) && !compare(value, *i));
- }
-
-
- /// binary_search_i
- ///
- /// Returns: iterator if there is an iterator i in the range [first last) that
- /// satisfies the corresponding conditions: !(*i < value) && !(value < *i).
- /// Returns last if the value is not found.
- ///
- /// Complexity: At most 'log(last - first) + 2' comparisons.
- ///
- template <typename ForwardIterator, typename T>
- inline ForwardIterator
- binary_search_i(ForwardIterator first, ForwardIterator last, const T& value)
- {
- // To do: This can be made slightly faster by not using lower_bound.
- ForwardIterator i(eastl::lower_bound<ForwardIterator, T>(first, last, value));
- if((i != last) && !(value < *i)) // Note that we always express value comparisons in terms of < or ==.
- return i;
- return last;
- }
-
-
- /// binary_search_i
- ///
- /// Returns: iterator if there is an iterator i in the range [first last) that
- /// satisfies the corresponding conditions: !(*i < value) && !(value < *i).
- /// Returns last if the value is not found.
- ///
- /// Complexity: At most 'log(last - first) + 2' comparisons.
- ///
- template <typename ForwardIterator, typename T, typename Compare>
- inline ForwardIterator
- binary_search_i(ForwardIterator first, ForwardIterator last, const T& value, Compare compare)
- {
- // To do: This can be made slightly faster by not using lower_bound.
- ForwardIterator i(eastl::lower_bound<ForwardIterator, T, Compare>(first, last, value, compare));
- if((i != last) && !compare(value, *i))
- return i;
- return last;
- }
-
-
- /// unique
- ///
- /// Given a sorted range, this function removes duplicated items.
- /// Note that if you have a container then you will probably want
- /// to call erase on the container with the return value if your
- /// goal is to remove the duplicated items from the container.
- ///
- /// Effects: Eliminates all but the first element from every consecutive
- /// group of equal elements referred to by the iterator i in the range
- /// [first, last) for which the following corresponding condition holds:
- /// *i == *(i - 1).
- ///
- /// Returns: The end of the resulting range.
- ///
- /// Complexity: If the range (last - first) is not empty, exactly (last - first)
- /// applications of the corresponding predicate, otherwise no applications of the predicate.
- ///
- /// Example usage:
- /// vector<int> intArray;
- /// ...
- /// intArray.erase(unique(intArray.begin(), intArray.end()), intArray.end());
- ///
- template <typename ForwardIterator>
- ForwardIterator unique(ForwardIterator first, ForwardIterator last)
- {
- first = eastl::adjacent_find<ForwardIterator>(first, last);
-
- if(first != last) // We expect that there are duplicated items, else the user wouldn't be calling this function.
- {
- ForwardIterator dest(first);
-
- for(++first; first != last; ++first)
- {
- if(!(*dest == *first)) // Note that we always express value comparisons in terms of < or ==.
- *++dest = *first;
- }
- return ++dest;
- }
- return last;
- }
-
-
- /// unique
- ///
- /// Given a sorted range, this function removes duplicated items.
- /// Note that if you have a container then you will probably want
- /// to call erase on the container with the return value if your
- /// goal is to remove the duplicated items from the container.
- ///
- /// Effects: Eliminates all but the first element from every consecutive
- /// group of equal elements referred to by the iterator i in the range
- /// [first, last) for which the following corresponding condition holds:
- /// predicate(*i, *(i - 1)) != false.
- ///
- /// Returns: The end of the resulting range.
- ///
- /// Complexity: If the range (last - first) is not empty, exactly (last - first)
- /// applications of the corresponding predicate, otherwise no applications of the predicate.
- ///
- template <typename ForwardIterator, typename BinaryPredicate>
- ForwardIterator unique(ForwardIterator first, ForwardIterator last, BinaryPredicate predicate)
- {
- first = eastl::adjacent_find<ForwardIterator, BinaryPredicate>(first, last, predicate);
-
- if(first != last) // We expect that there are duplicated items, else the user wouldn't be calling this function.
- {
- ForwardIterator dest(first);
-
- for(++first; first != last; ++first)
- {
- if(!predicate(*dest, *first))
- *++dest = *first;
- }
- return ++dest;
- }
- return last;
- }
-
-
-
- // find_end
- //
- // We provide two versions here, one for a bidirectional iterators and one for
- // regular forward iterators. Given that we are searching backward, it's a bit
- // more efficient if we can use backwards iteration to implement our search,
- // though this requires an iterator that can be reversed.
- //
- template <typename ForwardIterator1, typename ForwardIterator2>
- ForwardIterator1
- find_end_impl(ForwardIterator1 first1, ForwardIterator1 last1,
- ForwardIterator2 first2, ForwardIterator2 last2,
- EASTL_ITC_NS::forward_iterator_tag, EASTL_ITC_NS::forward_iterator_tag)
- {
- if(first2 != last2) // We have to do this check because the search algorithm below will return first1 (and not last1) if the first2/last2 range is empty.
- {
- for(ForwardIterator1 result(last1); ; )
- {
- const ForwardIterator1 resultNext(eastl::search(first1, last1, first2, last2));
-
- if(resultNext != last1) // If another sequence was found...
- {
- first1 = result = resultNext;
- ++first1;
- }
- else
- return result;
- }
- }
- return last1;
- }
-
- template <typename BidirectionalIterator1, typename BidirectionalIterator2>
- BidirectionalIterator1
- find_end_impl(BidirectionalIterator1 first1, BidirectionalIterator1 last1,
- BidirectionalIterator2 first2, BidirectionalIterator2 last2,
- EASTL_ITC_NS::bidirectional_iterator_tag, EASTL_ITC_NS::bidirectional_iterator_tag)
- {
- typedef eastl::reverse_iterator<BidirectionalIterator1> reverse_iterator1;
- typedef eastl::reverse_iterator<BidirectionalIterator2> reverse_iterator2;
-
- reverse_iterator1 rresult(eastl::search(reverse_iterator1(last1), reverse_iterator1(first1),
- reverse_iterator2(last2), reverse_iterator2(first2)));
- if(rresult.base() != first1) // If we found something...
- {
- BidirectionalIterator1 result(rresult.base());
-
- eastl::advance(result, -eastl::distance(first2, last2)); // We have an opportunity to optimize this, as the
- return result; // search function already calculates this distance.
- }
- return last1;
- }
-
- /// find_end
- ///
- /// Finds the last occurrence of the second sequence in the first sequence.
- /// As such, this function is much like the C string function strrstr and it
- /// is also the same as a reversed version of 'search'. It is called find_end
- /// instead of the possibly more consistent search_end simply because the C++
- /// standard algorithms have such naming.
- ///
- /// Returns an iterator between first1 and last1 if the sequence is found.
- /// returns last1 (the end of the first seqence) if the sequence is not found.
- ///
- template <typename ForwardIterator1, typename ForwardIterator2>
- inline ForwardIterator1
- find_end(ForwardIterator1 first1, ForwardIterator1 last1,
- ForwardIterator2 first2, ForwardIterator2 last2)
- {
- typedef typename eastl::iterator_traits<ForwardIterator1>::iterator_category IC1;
- typedef typename eastl::iterator_traits<ForwardIterator2>::iterator_category IC2;
-
- return eastl::find_end_impl(first1, last1, first2, last2, IC1(), IC2());
- }
-
-
-
-
- // To consider: Fold the predicate and non-predicate versions of
- // this algorithm into a single function.
- template <typename ForwardIterator1, typename ForwardIterator2, typename BinaryPredicate>
- ForwardIterator1
- find_end_impl(ForwardIterator1 first1, ForwardIterator1 last1,
- ForwardIterator2 first2, ForwardIterator2 last2,
- BinaryPredicate predicate,
- EASTL_ITC_NS::forward_iterator_tag, EASTL_ITC_NS::forward_iterator_tag)
- {
- if(first2 != last2) // We have to do this check because the search algorithm below will return first1 (and not last1) if the first2/last2 range is empty.
- {
- for(ForwardIterator1 result = last1; ; )
- {
- const ForwardIterator1 resultNext(eastl::search<ForwardIterator1, ForwardIterator2, BinaryPredicate>(first1, last1, first2, last2, predicate));
-
- if(resultNext != last1) // If another sequence was found...
- {
- first1 = result = resultNext;
- ++first1;
- }
- else
- return result;
- }
- }
- return last1;
- }
-
- template <typename BidirectionalIterator1, typename BidirectionalIterator2, typename BinaryPredicate>
- BidirectionalIterator1
- find_end_impl(BidirectionalIterator1 first1, BidirectionalIterator1 last1,
- BidirectionalIterator2 first2, BidirectionalIterator2 last2,
- BinaryPredicate predicate,
- EASTL_ITC_NS::bidirectional_iterator_tag, EASTL_ITC_NS::bidirectional_iterator_tag)
- {
- typedef eastl::reverse_iterator<BidirectionalIterator1> reverse_iterator1;
- typedef eastl::reverse_iterator<BidirectionalIterator2> reverse_iterator2;
-
- reverse_iterator1 rresult(eastl::search<reverse_iterator1, reverse_iterator2, BinaryPredicate>
- (reverse_iterator1(last1), reverse_iterator1(first1),
- reverse_iterator2(last2), reverse_iterator2(first2),
- predicate));
- if(rresult.base() != first1) // If we found something...
- {
- BidirectionalIterator1 result(rresult.base());
- eastl::advance(result, -eastl::distance(first2, last2));
- return result;
- }
- return last1;
- }
-
-
- /// find_end
- ///
- /// Effects: Finds a subsequence of equal values in a sequence.
- ///
- /// Returns: The last iterator i in the range [first1, last1 - (last2 - first2))
- /// such that for any nonnegative integer n < (last2 - first2), the following
- /// corresponding conditions hold: pred(*(i+n),*(first2+n)) != false. Returns
- /// last1 if no such iterator is found.
- ///
- /// Complexity: At most (last2 - first2) * (last1 - first1 - (last2 - first2) + 1)
- /// applications of the corresponding predicate.
- ///
- template <typename ForwardIterator1, typename ForwardIterator2, typename BinaryPredicate>
- inline ForwardIterator1
- find_end(ForwardIterator1 first1, ForwardIterator1 last1,
- ForwardIterator2 first2, ForwardIterator2 last2,
- BinaryPredicate predicate)
- {
- typedef typename eastl::iterator_traits<ForwardIterator1>::iterator_category IC1;
- typedef typename eastl::iterator_traits<ForwardIterator2>::iterator_category IC2;
-
- return eastl::find_end_impl<ForwardIterator1, ForwardIterator2, BinaryPredicate>
- (first1, last1, first2, last2, predicate, IC1(), IC2());
- }
-
-
- /// set_difference
- ///
- /// set_difference iterates over both input ranges and copies elements present
- /// in the first range but not the second to the output range.
- ///
- /// Effects: Copies the elements of the range [first1, last1) which are not
- /// present in the range [first2, last2) to the range beginning at result.
- /// The elements in the constructed range are sorted.
- ///
- /// Requires: The input ranges must be sorted.
- /// Requires: The output range shall not overlap with either of the original ranges.
- ///
- /// Returns: The end of the output range.
- ///
- /// Complexity: At most (2 * ((last1 - first1) + (last2 - first2)) - 1) comparisons.
- ///
- template <typename InputIterator1, typename InputIterator2, typename OutputIterator>
- OutputIterator set_difference(InputIterator1 first1, InputIterator1 last1,
- InputIterator2 first2, InputIterator2 last2,
- OutputIterator result)
- {
- while((first1 != last1) && (first2 != last2))
- {
- if(*first1 < *first2)
- {
- *result = *first1;
- ++first1;
- ++result;
- }
- else if(*first2 < *first1)
- ++first2;
- else
- {
- ++first1;
- ++first2;
- }
- }
-
- return eastl::copy(first1, last1, result);
- }
-
-
- template <typename InputIterator1, typename InputIterator2, typename OutputIterator, typename Compare>
- OutputIterator set_difference(InputIterator1 first1, InputIterator1 last1,
- InputIterator2 first2, InputIterator2 last2,
- OutputIterator result, Compare compare)
- {
- while((first1 != last1) && (first2 != last2))
- {
- if(compare(*first1, *first2))
- {
- EASTL_VALIDATE_COMPARE(!compare(*first2, *first1)); // Validate that the compare function is sane.
- *result = *first1;
- ++first1;
- ++result;
- }
- else if(compare(*first2, *first1))
- {
- EASTL_VALIDATE_COMPARE(!compare(*first1, *first2)); // Validate that the compare function is sane.
- ++first2;
- }
- else
- {
- ++first1;
- ++first2;
- }
- }
-
- return eastl::copy(first1, last1, result);
- }
-
-
- /// set_difference_2
- ///
- /// set_difference_2 iterates over both input ranges and copies elements present
- /// in the first range but not the second to the first output range and copies
- /// elements present in the second range but not in the first to the second output
- /// range.
- ///
- /// Effects: Copies the elements of the range [first1, last1) which are not
- /// present in the range [first2, last2) to the first output range beginning at
- /// result1 AND copies the element of range [first2, last2) which are not present
- /// in the range [first1, last) to the second output range beginning at result2.
- /// The elements in the constructed range are sorted.
- ///
- /// Requires: The input ranges must be sorted.
- /// Requires: The output ranges shall not overlap with either of the original ranges.
- ///
- /// Returns: Nothing.
- ///
- /// Complexity: At most (2 * ((last1 - first1) + (last2 - first2)) - 1) comparisons.
- ///
- template <typename InputIterator1, typename InputIterator2, typename OutputIterator, typename Compare>
- void set_difference_2(InputIterator1 first1, InputIterator1 last1,
- InputIterator2 first2, InputIterator2 last2,
- OutputIterator result1, OutputIterator result2, Compare compare)
- {
- while ((first1 != last1) && (first2 != last2))
- {
- if (compare(*first1, *first2))
- {
- EASTL_VALIDATE_COMPARE(!compare(*first2, *first1)); // Validate that the compare function is sane.
- *result1++ = *first1++;
- }
- else if (compare(*first2, *first1))
- {
- EASTL_VALIDATE_COMPARE(!compare(*first1, *first2)); // Validate that the compare function is sane.
- *result2++ = *first2++;
- }
- else
- {
- ++first1;
- ++first2;
- }
- }
-
- eastl::copy(first2, last2, result2);
- eastl::copy(first1, last1, result1);
- }
-
- /// set_difference_2
- ///
- /// set_difference_2 with the default comparison object is eastl::less<>.
- ///
- template <typename InputIterator1, typename InputIterator2, typename OutputIterator>
- void set_difference_2(InputIterator1 first1, InputIterator1 last1,
- InputIterator2 first2, InputIterator2 last2,
- OutputIterator result1, OutputIterator result2)
- {
- eastl::set_difference_2(first1, last1, first2, last2, result1, result2, eastl::less<>{});
- }
-
-
- /// set_symmetric_difference
- ///
- /// set_difference iterates over both input ranges and copies elements present
- /// in the either range but not the other to the output range.
- ///
- /// Effects: Copies the elements of the range [first1, last1) which are not
- /// present in the range [first2, last2), and the elements of the range [first2, last2)
- /// which are not present in the range [first1, last1) to the range beginning at result.
- /// The elements in the constructed range are sorted.
- ///
- /// Requires: The input ranges must be sorted.
- /// Requires: The resulting range shall not overlap with either of the original ranges.
- ///
- /// Returns: The end of the constructed range.
- ///
- /// Complexity: At most (2 * ((last1 - first1) + (last2 - first2)) - 1) comparisons.
- ///
- template <typename InputIterator1, typename InputIterator2, typename OutputIterator>
- OutputIterator set_symmetric_difference(InputIterator1 first1, InputIterator1 last1,
- InputIterator2 first2, InputIterator2 last2,
- OutputIterator result)
- {
- while((first1 != last1) && (first2 != last2))
- {
- if(*first1 < *first2)
- {
- *result = *first1;
- ++first1;
- ++result;
- }
- else if(*first2 < *first1)
- {
- *result = *first2;
- ++first2;
- ++result;
- }
- else
- {
- ++first1;
- ++first2;
- }
- }
-
- return eastl::copy(first2, last2, eastl::copy(first1, last1, result));
- }
-
-
- template <typename InputIterator1, typename InputIterator2, typename OutputIterator, typename Compare>
- OutputIterator set_symmetric_difference(InputIterator1 first1, InputIterator1 last1,
- InputIterator2 first2, InputIterator2 last2,
- OutputIterator result, Compare compare)
- {
- while((first1 != last1) && (first2 != last2))
- {
- if(compare(*first1, *first2))
- {
- EASTL_VALIDATE_COMPARE(!compare(*first2, *first1)); // Validate that the compare function is sane.
- *result = *first1;
- ++first1;
- ++result;
- }
- else if(compare(*first2, *first1))
- {
- EASTL_VALIDATE_COMPARE(!compare(*first1, *first2)); // Validate that the compare function is sane.
- *result = *first2;
- ++first2;
- ++result;
- }
- else
- {
- ++first1;
- ++first2;
- }
- }
-
- return eastl::copy(first2, last2, eastl::copy(first1, last1, result));
- }
-
-
- /// set_intersection
- ///
- /// set_intersection over both ranges and copies elements present in
- /// both ranges to the output range.
- ///
- /// Effects: Constructs a sorted intersection of the elements from the
- /// two ranges; that is, the set of elements that are present in both of the ranges.
- ///
- /// Requires: The input ranges must be sorted.
- /// Requires: The resulting range shall not overlap with either of the original ranges.
- ///
- /// Returns: The end of the constructed range.
- ///
- /// Complexity: At most 2 * ((last1 - first1) + (last2 - first2)) - 1) comparisons.
- ///
- /// Note: The copying operation is stable; if an element is present in both ranges,
- /// the one from the first range is copied.
- ///
- template <typename InputIterator1, typename InputIterator2, typename OutputIterator>
- OutputIterator set_intersection(InputIterator1 first1, InputIterator1 last1,
- InputIterator2 first2, InputIterator2 last2,
- OutputIterator result)
- {
- while((first1 != last1) && (first2 != last2))
- {
- if(*first1 < *first2)
- ++first1;
- else if(*first2 < *first1)
- ++first2;
- else
- {
- *result = *first1;
- ++first1;
- ++first2;
- ++result;
- }
- }
-
- return result;
- }
-
-
- template <typename InputIterator1, typename InputIterator2, typename OutputIterator, typename Compare>
- OutputIterator set_intersection(InputIterator1 first1, InputIterator1 last1,
- InputIterator2 first2, InputIterator2 last2,
- OutputIterator result, Compare compare)
- {
- while((first1 != last1) && (first2 != last2))
- {
- if(compare(*first1, *first2))
- {
- EASTL_VALIDATE_COMPARE(!compare(*first2, *first1)); // Validate that the compare function is sane.
- ++first1;
- }
- else if(compare(*first2, *first1))
- {
- EASTL_VALIDATE_COMPARE(!compare(*first1, *first2)); // Validate that the compare function is sane.
- ++first2;
- }
- else
- {
- *result = *first1;
- ++first1;
- ++first2;
- ++result;
- }
- }
-
- return result;
- }
-
-
-
- /// set_union
- ///
- /// set_union iterates over both ranges and copies elements present in
- /// both ranges to the output range.
- ///
- /// Effects: Constructs a sorted union of the elements from the two ranges;
- /// that is, the set of elements that are present in one or both of the ranges.
- ///
- /// Requires: The input ranges must be sorted.
- /// Requires: The resulting range shall not overlap with either of the original ranges.
- ///
- /// Returns: The end of the constructed range.
- ///
- /// Complexity: At most (2 * ((last1 - first1) + (last2 - first2)) - 1) comparisons.
- ///
- /// Note: The copying operation is stable; if an element is present in both ranges,
- /// the one from the first range is copied.
- ///
- template <typename InputIterator1, typename InputIterator2, typename OutputIterator>
- OutputIterator set_union(InputIterator1 first1, InputIterator1 last1,
- InputIterator2 first2, InputIterator2 last2,
- OutputIterator result)
- {
- while((first1 != last1) && (first2 != last2))
- {
- if(*first1 < *first2)
- {
- *result = *first1;
- ++first1;
- }
- else if(*first2 < *first1)
- {
- *result = *first2;
- ++first2;
- }
- else
- {
- *result = *first1;
- ++first1;
- ++first2;
- }
- ++result;
- }
-
- return eastl::copy(first2, last2, eastl::copy(first1, last1, result));
- }
-
-
- template <typename InputIterator1, typename InputIterator2, typename OutputIterator, typename Compare>
- OutputIterator set_union(InputIterator1 first1, InputIterator1 last1,
- InputIterator2 first2, InputIterator2 last2,
- OutputIterator result, Compare compare)
- {
- while((first1 != last1) && (first2 != last2))
- {
- if(compare(*first1, *first2))
- {
- EASTL_VALIDATE_COMPARE(!compare(*first2, *first1)); // Validate that the compare function is sane.
- *result = *first1;
- ++first1;
- }
- else if(compare(*first2, *first1))
- {
- EASTL_VALIDATE_COMPARE(!compare(*first1, *first2)); // Validate that the compare function is sane.
- *result = *first2;
- ++first2;
- }
- else
- {
- *result = *first1;
- ++first1;
- ++first2;
- }
- ++result;
- }
-
- return eastl::copy(first2, last2, eastl::copy(first1, last1, result));
- }
-
-
- /// set_decomposition
- ///
- /// set_decomposition iterates over both ranges and copies elements to one of the three
- /// categories of output ranges.
- ///
- /// Effects: Constructs three sorted containers of the elements from the two ranges.
- /// * OutputIterator1 is elements only in Container1.
- /// * OutputIterator2 is elements only in Container2.
- /// * OutputIterator3 is elements that are in both Container1 and Container2.
- ///
- /// Requires: The input ranges must be sorted.
- /// Requires: The resulting ranges shall not overlap with either of the original ranges.
- ///
- /// Returns: The end of the constructed range of elements in both Container1 and Container2.
- ///
- /// Complexity: At most (2 * ((last1 - first1) + (last2 - first2)) - 1) comparisons.
- ///
- template <typename InputIterator1, typename InputIterator2,
- typename OutputIterator1, typename OutputIterator2, typename OutputIterator3, typename Compare>
- OutputIterator3 set_decomposition(InputIterator1 first1, InputIterator1 last1,
- InputIterator2 first2, InputIterator2 last2,
- OutputIterator1 result1, OutputIterator2 result2, OutputIterator3 result3, Compare compare)
- {
- while ((first1 != last1) && (first2 != last2))
- {
- if (compare(*first1, *first2))
- {
- EASTL_VALIDATE_COMPARE(!compare(*first2, *first1)); // Validate that the compare function is sane.
- *result1++ = *first1++;
- }
- else if (compare(*first2, *first1))
- {
- EASTL_VALIDATE_COMPARE(!compare(*first1, *first2)); // Validate that the compare function is sane.
- *result2++ = *first2++;
- }
- else
- {
- *result3++ = *first1++;
- ++first2;
- }
- }
-
- eastl::copy(first1, last1, result1);
- eastl::copy(first2, last2, result2);
-
- return result3;
- }
-
- /// set_decomposition
- ///
- /// set_decomposition with the default comparison object is eastl::less<>.
- ///
- template <typename InputIterator1, typename InputIterator2,
- typename OutputIterator1, typename OutputIterator2, typename OutputIterator3>
- OutputIterator3 set_decomposition(InputIterator1 first1, InputIterator1 last1, InputIterator2 first2, InputIterator2 last2,
- OutputIterator1 result1, OutputIterator2 result2, OutputIterator3 result3)
- {
- return eastl::set_decomposition(first1, last1, first2, last2, result1, result2, result3, eastl::less<>{});
- }
-
-
- /// is_permutation
- ///
- template<typename ForwardIterator1, typename ForwardIterator2>
- bool is_permutation(ForwardIterator1 first1, ForwardIterator1 last1, ForwardIterator2 first2)
- {
- typedef typename eastl::iterator_traits<ForwardIterator1>::difference_type difference_type;
-
- // Skip past any equivalent initial elements.
- while((first1 != last1) && (*first1 == *first2))
- {
- ++first1;
- ++first2;
- }
-
- if(first1 != last1)
- {
- const difference_type first1Size = eastl::distance(first1, last1);
- ForwardIterator2 last2 = first2;
- eastl::advance(last2, first1Size);
-
- for(ForwardIterator1 i = first1; i != last1; ++i)
- {
- if(i == eastl::find(first1, i, *i))
- {
- const difference_type c = eastl::count(first2, last2, *i);
-
- if((c == 0) || (c != eastl::count(i, last1, *i)))
- return false;
- }
- }
- }
-
- return true;
- }
-
- /// is_permutation
- ///
- template<typename ForwardIterator1, typename ForwardIterator2, class BinaryPredicate>
- bool is_permutation(ForwardIterator1 first1, ForwardIterator1 last1, ForwardIterator2 first2, BinaryPredicate predicate)
- {
- typedef typename eastl::iterator_traits<ForwardIterator1>::difference_type difference_type;
-
- // Skip past any equivalent initial elements.
- while((first1 != last1) && predicate(*first1, *first2))
- {
- ++first1;
- ++first2;
- }
-
- if(first1 != last1)
- {
- const difference_type first1Size = eastl::distance(first1, last1);
- ForwardIterator2 last2 = first2;
- eastl::advance(last2, first1Size);
-
- for(ForwardIterator1 i = first1; i != last1; ++i)
- {
- if(i == eastl::find(first1, i, *i, predicate))
- {
- const difference_type c = eastl::count(first2, last2, *i, predicate);
-
- if((c == 0) || (c != eastl::count(i, last1, *i, predicate)))
- return false;
- }
- }
- }
-
- return true;
- }
-
-
- /// next_permutation
- ///
- /// mutates the range [first, last) to the next permutation. Returns true if the
- /// new range is not the final permutation (sorted like the starting permutation).
- /// Permutations start with a sorted range, and false is returned when next_permutation
- /// results in the initial sorted range, or if the range has <= 1 element.
- /// Note that elements are compared by operator < (as usual) and that elements deemed
- /// equal via this are not rearranged.
- ///
- /// http://marknelson.us/2002/03/01/next-permutation/
- /// Basically we start with an ordered range and reverse it's order one specifically
- /// chosen swap and reverse at a time. It happens that this require going through every
- /// permutation of the range. We use the same variable names as the document above.
- ///
- /// To consider: Significantly improved permutation/combination functionality:
- /// http://home.roadrunner.com/~hinnant/combinations.html
- ///
- /// Example usage:
- /// vector<int> intArray;
- /// // <populate intArray>
- /// sort(intArray.begin(), intArray.end());
- /// do {
- /// // <do something with intArray>
- /// } while(next_permutation(intArray.begin(), intArray.end()));
- ///
-
- template<typename BidirectionalIterator, typename Compare>
- bool next_permutation(BidirectionalIterator first, BidirectionalIterator last, Compare compare)
- {
- if(first != last) // If there is anything in the range...
- {
- BidirectionalIterator i = last;
-
- if(first != --i) // If the range has more than one item...
- {
- for(;;)
- {
- BidirectionalIterator ii(i), j;
-
- if(compare(*--i, *ii)) // Find two consecutive values where the first is less than the second.
- {
- j = last;
- while(!compare(*i, *--j)) // Find the final value that's greater than the first (it may be equal to the second).
- {}
- eastl::iter_swap(i, j); // Swap the first and the final.
- eastl::reverse(ii, last); // Reverse the ranget from second to last.
- return true;
- }
-
- if(i == first) // There are no two consecutive values where the first is less than the second, meaning the range is in reverse order. The reverse ordered range is always the last permutation.
- {
- eastl::reverse(first, last);
- break; // We are done.
- }
- }
- }
- }
-
- return false;
- }
-
- template<typename BidirectionalIterator>
- bool next_permutation(BidirectionalIterator first, BidirectionalIterator last)
- {
- typedef typename eastl::iterator_traits<BidirectionalIterator>::value_type value_type;
-
- return eastl::next_permutation(first, last, eastl::less<value_type>());
- }
-
-
-
- /// rotate
- ///
- /// Effects: For each non-negative integer i < (last - first), places the element from the
- /// position first + i into position first + (i + (last - middle)) % (last - first).
- ///
- /// Returns: first + (last - middle). That is, returns where first went to.
- ///
- /// Remarks: This is a left rotate.
- ///
- /// Requires: [first,middle) and [middle,last) shall be valid ranges. ForwardIterator shall
- /// satisfy the requirements of ValueSwappable (17.6.3.2). The type of *first shall satisfy
- /// the requirements of MoveConstructible (Table 20) and the requirements of MoveAssignable.
- ///
- /// Complexity: At most last - first swaps.
- ///
- /// Note: While rotate works on ForwardIterators (e.g. slist) and BidirectionalIterators (e.g. list),
- /// you can get much better performance (O(1) instead of O(n)) with slist and list rotation by
- /// doing splice operations on those lists instead of calling this rotate function.
- ///
- /// http://www.cs.bell-labs.com/cm/cs/pearls/s02b.pdf / http://books.google.com/books?id=kse_7qbWbjsC&pg=PA14&lpg=PA14&dq=Programming+Pearls+flipping+hands
- /// http://books.google.com/books?id=tjOlkl7ecVQC&pg=PA189&lpg=PA189&dq=stepanov+Elements+of+Programming+rotate
- /// http://stackoverflow.com/questions/21160875/why-is-stdrotate-so-fast
- ///
- /// Strategy:
- /// - We handle the special case of (middle == first) and (middle == last) no-ops
- /// up front in the main rotate entry point.
- /// - There's a basic ForwardIterator implementation (rotate_general_impl) which is
- /// a fallback implementation that's not as fast as others but works for all cases.
- /// - There's a slightly better BidirectionalIterator implementation.
- /// - We have specialized versions for rotating elements that are is_trivially_move_assignable.
- /// These versions will use memmove for when we have a RandomAccessIterator.
- /// - We have a specialized version for rotating by only a single position, as that allows us
- /// (with any iterator type) to avoid a lot of logic involved with algorithms like "flipping hands"
- /// and achieve near optimal O(n) behavior. it turns out that rotate-by-one is a common use
- /// case in practice.
- ///
- namespace Internal
- {
- template<typename ForwardIterator>
- ForwardIterator rotate_general_impl(ForwardIterator first, ForwardIterator middle, ForwardIterator last)
- {
- using eastl::swap;
-
- ForwardIterator current = middle;
-
- do {
- swap(*first++, *current++);
-
- if(first == middle)
- middle = current;
- } while(current != last);
-
- ForwardIterator result = first;
- current = middle;
-
- while(current != last)
- {
- swap(*first++, *current++);
-
- if(first == middle)
- middle = current;
- else if(current == last)
- current = middle;
- }
-
- return result; // result points to first + (last - middle).
- }
-
-
- template <typename ForwardIterator>
- ForwardIterator move_rotate_left_by_one(ForwardIterator first, ForwardIterator last)
- {
- typedef typename eastl::iterator_traits<ForwardIterator>::value_type value_type;
-
- value_type temp(eastl::move(*first));
- ForwardIterator result = eastl::move(eastl::next(first), last, first); // Note that while our template type is BidirectionalIterator, if the actual
- *result = eastl::move(temp); // iterator is a RandomAccessIterator then this move will be a memmove for trivial types.
-
- return result; // result points to the final element in the range.
- }
-
-
- template <typename BidirectionalIterator>
- BidirectionalIterator move_rotate_right_by_one(BidirectionalIterator first, BidirectionalIterator last)
- {
- typedef typename eastl::iterator_traits<BidirectionalIterator>::value_type value_type;
-
- BidirectionalIterator beforeLast = eastl::prev(last);
- value_type temp(eastl::move(*beforeLast));
- BidirectionalIterator result = eastl::move_backward(first, beforeLast, last); // Note that while our template type is BidirectionalIterator, if the actual
- *first = eastl::move(temp); // iterator is a RandomAccessIterator then this move will be a memmove for trivial types.
-
- return result; // result points to the first element in the range.
- }
-
- template <typename /*IteratorCategory*/, bool /*is_trivially_move_assignable*/>
- struct rotate_helper
- {
- template <typename ForwardIterator>
- static ForwardIterator rotate_impl(ForwardIterator first, ForwardIterator middle, ForwardIterator last)
- { return Internal::rotate_general_impl(first, middle, last); }
- };
-
- template <>
- struct rotate_helper<EASTL_ITC_NS::forward_iterator_tag, true>
- {
- template <typename ForwardIterator>
- static ForwardIterator rotate_impl(ForwardIterator first, ForwardIterator middle, ForwardIterator last)
- {
- if(eastl::next(first) == middle) // If moving trivial types by a single element, memcpy is fast for that case.
- return Internal::move_rotate_left_by_one(first, last);
- return Internal::rotate_general_impl(first, middle, last);
- }
- };
-
- template <>
- struct rotate_helper<EASTL_ITC_NS::bidirectional_iterator_tag, false>
- {
- template <typename BidirectionalIterator>
- static BidirectionalIterator rotate_impl(BidirectionalIterator first, BidirectionalIterator middle, BidirectionalIterator last)
- { return Internal::rotate_general_impl(first, middle, last); } // rotate_general_impl outperforms the flipping hands algorithm.
-
- /*
- // Simplest "flipping hands" implementation. Disabled because it's slower on average than rotate_general_impl.
- template <typename BidirectionalIterator>
- static BidirectionalIterator rotate_impl(BidirectionalIterator first, BidirectionalIterator middle, BidirectionalIterator last)
- {
- eastl::reverse(first, middle);
- eastl::reverse(middle, last);
- eastl::reverse(first, last);
- return first + (last - middle); // This can be slow for large ranges because operator + and - are O(n).
- }
-
- // Smarter "flipping hands" implementation, but still disabled because benchmarks are showing it to be slower than rotate_general_impl.
- template <typename BidirectionalIterator>
- static BidirectionalIterator rotate_impl(BidirectionalIterator first, BidirectionalIterator middle, BidirectionalIterator last)
- {
- // This is the "flipping hands" algorithm.
- eastl::reverse_impl(first, middle, EASTL_ITC_NS::bidirectional_iterator_tag()); // Reverse the left side.
- eastl::reverse_impl(middle, last, EASTL_ITC_NS::bidirectional_iterator_tag()); // Reverse the right side.
-
- // Reverse the entire range.
- while((first != middle) && (middle != last))
- {
- eastl::iter_swap(first, --last);
- ++first;
- }
-
- if(first == middle) // Finish reversing the entire range.
- {
- eastl::reverse_impl(middle, last, bidirectional_iterator_tag());
- return last;
- }
- else
- {
- eastl::reverse_impl(first, middle, bidirectional_iterator_tag());
- return first;
- }
- }
- */
- };
-
- template <>
- struct rotate_helper<EASTL_ITC_NS::bidirectional_iterator_tag, true>
- {
- template <typename BidirectionalIterator>
- static BidirectionalIterator rotate_impl(BidirectionalIterator first, BidirectionalIterator middle, BidirectionalIterator last)
- {
- if(eastl::next(first) == middle) // If moving trivial types by a single element, memcpy is fast for that case.
- return Internal::move_rotate_left_by_one(first, last);
- if(eastl::next(middle) == last)
- return Internal::move_rotate_right_by_one(first, last);
- return Internal::rotate_general_impl(first, middle, last);
- }
- };
-
- template <typename Integer>
- inline Integer greatest_common_divisor(Integer x, Integer y)
- {
- do {
- Integer t = (x % y);
- x = y;
- y = t;
- } while(y);
-
- return x;
- }
-
- template <>
- struct rotate_helper<EASTL_ITC_NS::random_access_iterator_tag, false>
- {
- // This is the juggling algorithm, using move operations.
- // In practice this implementation is about 25% faster than rotate_general_impl. We may want to
- // consider sticking with just rotate_general_impl and avoid the code generation of this function.
- template <typename RandomAccessIterator>
- static RandomAccessIterator rotate_impl(RandomAccessIterator first, RandomAccessIterator middle, RandomAccessIterator last)
- {
- typedef typename iterator_traits<RandomAccessIterator>::difference_type difference_type;
- typedef typename iterator_traits<RandomAccessIterator>::value_type value_type;
-
- const difference_type m1 = (middle - first);
- const difference_type m2 = (last - middle);
- const difference_type g = Internal::greatest_common_divisor(m1, m2);
- value_type temp;
-
- for(RandomAccessIterator p = first + g; p != first;)
- {
- temp = eastl::move(*--p);
- RandomAccessIterator p1 = p;
- RandomAccessIterator p2 = p + m1;
- do
- {
- *p1 = eastl::move(*p2);
- p1 = p2;
- const difference_type d = (last - p2);
-
- if(m1 < d)
- p2 += m1;
- else
- p2 = first + (m1 - d);
- } while(p2 != p);
-
- *p1 = eastl::move(temp);
- }
-
- return first + m2;
- }
- };
-
- template <>
- struct rotate_helper<EASTL_ITC_NS::random_access_iterator_tag, true>
- {
- // Experiments were done which tested the performance of using an intermediate buffer
- // to do memcpy's to as opposed to executing a swapping algorithm. It turns out this is
- // actually slower than even rotate_general_impl, partly because the average case involves
- // memcpy'ing a quarter of the element range twice. Experiments were done with various kinds
- // of PODs with various element counts.
-
- template <typename RandomAccessIterator>
- static RandomAccessIterator rotate_impl(RandomAccessIterator first, RandomAccessIterator middle, RandomAccessIterator last)
- {
- if(eastl::next(first) == middle) // If moving trivial types by a single element, memcpy is fast for that case.
- return Internal::move_rotate_left_by_one(first, last);
- if(eastl::next(middle) == last)
- return Internal::move_rotate_right_by_one(first, last);
- if((last - first) < 32) // For small ranges rotate_general_impl is faster.
- return Internal::rotate_general_impl(first, middle, last);
- return Internal::rotate_helper<EASTL_ITC_NS::random_access_iterator_tag, false>::rotate_impl(first, middle, last);
- }
- };
-
- } // namespace Internal
-
-
- template <typename ForwardIterator>
- ForwardIterator rotate(ForwardIterator first, ForwardIterator middle, ForwardIterator last)
- {
- if(middle != first)
- {
- if(middle != last)
- {
- typedef typename eastl::iterator_traits<ForwardIterator>::iterator_category IC;
- typedef typename eastl::iterator_traits<ForwardIterator>::value_type value_type;
-
- return Internal::rotate_helper<IC, eastl::is_trivially_move_assignable<value_type>::value || // This is the best way of telling if we can move types via memmove, but without a conforming C++11 compiler it usually returns false.
- eastl::is_pod<value_type>::value || // This is a more conservative way of telling if we can move types via memmove, and most compilers support it, but it doesn't have as full of coverage as is_trivially_move_assignable.
- eastl::is_scalar<value_type>::value> // This is the most conservative means and works with all compilers, but works only for scalars.
- ::rotate_impl(first, middle, last);
- }
-
- return first;
- }
-
- return last;
- }
-
-
-
- /// rotate_copy
- ///
- /// Similar to rotate except writes the output to the OutputIterator and
- /// returns an OutputIterator to the element past the last element copied
- /// (i.e. result + (last - first))
- ///
- template <typename ForwardIterator, typename OutputIterator>
- OutputIterator rotate_copy(ForwardIterator first, ForwardIterator middle, ForwardIterator last, OutputIterator result)
- {
- return eastl::copy(first, middle, eastl::copy(middle, last, result));
- }
-
-
-
- /// clamp
- ///
- /// Returns a reference to a clamped value within the range of [lo, hi].
- ///
- /// http://en.cppreference.com/w/cpp/algorithm/clamp
- ///
- template <class T, class Compare>
- EA_CONSTEXPR const T& clamp(const T& v, const T& lo, const T& hi, Compare comp)
- {
- EASTL_ASSERT(!comp(hi, lo));
- return comp(v, lo) ? lo : comp(hi, v) ? hi : v;
- }
-
- template <class T>
- EA_CONSTEXPR const T& clamp(const T& v, const T& lo, const T& hi)
- {
- return eastl::clamp(v, lo, hi, eastl::less<>());
- }
-
-
-
-} // namespace eastl
-
-
-#endif // Header include guard
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-