1. Abstract
Change the iterator requirements for non-Ranges algorithms. Instead of requiring that iterators meet certain Cpp17Iterator requirements, require that the iterators model certain iterator concepts. This makes iterators from several standard views usable with non-Ranges algorithms.
2. Revision history
2.1. R0
This paper arose out of a private e-mail discussion where Bryce Adelstein Lelbach questioned why code that is similar to the first example in § 3 Motivation didn’t compile with MSVC.
3. Motivation
These two snippets of code should be well-formed:
std :: vector < int > data = ...; auto v = data | std :: views :: transform ([]( int x ){ return x * x ; }); int sum_of_squares = std :: reduce ( std :: execution :: par , begin ( v ), end ( v )); 
auto idxs = std :: views :: iota ( 0 , N ); std :: transform ( std :: execution :: par , begin ( idxs ), end ( idxs ), begin ( sqrts ), []( int x ) { return std :: sqrt ( float ( x )); }); 
It should be possible, in most cases, to use the iterators from ranges and views as the inputs to C++17 parallel algorithms and to other algorithms that require forward iterators or greater.
4. Analysis
In this example, the lambda transformation function returns a value, so the iterators passed to 
Several other views have the same issue, where the 
- 
     iota_view iterator_category input_iterator_tag iterator_concept random_access_iterator_tag 
- 
     lazy_split_view iterator_category input_iterator_tag iterator_concept forward_iterator_tag 
- 
     split_view iterator_category input_iterator_tag iterator_concept forward_iterator_tag 
- 
     elements_view transform_view iterator_category std :: get < N > iterator_category input_iterator_tag std :: get < N > iterator_concept std :: get < N > 
4.1. Current situation
MSVC checks that iterator arguments to parallel algorithms have the correct iterator category. If either of the code snippets in § 3 Motivation are compiled, the compilation will fail with "error C2338: Parallel algorithms require forward iterators or stronger."
libstd++ does not do any up-front checking of the iterator category of algorithm arguments. Its implementation of algorithms will accept iterators of any category as long as the iterators support the operations that are actually used in the implementation. The code snippets in § 3 Motivation can be successfully compiled with GCC 11.1.
5. Possible Solutions
5.1. Do nothing
We could simply wait for parallel versions of Range-based algorithms to make their way into the standard. Until then, users who want to use certain view iterators as inputs to parallel algorithms are out of luck.
This solution might be worth considering if parallel Range-based algorithms were on track for C++23. But [P2214] "A Plan for C++23 Ranges" puts parallel Range-based algorithms in Tier 2 of Ranges work, with the caveat that the work needs to be coordinated with Executors to make sure the interface is correct. That pushes the work well past C++23.
Even if parallel Range-based algorithms were already in the draft IS, it would still be worth investigating how to get Ranges iterators to work better with non-Ranges algorithms, in the interest of having the various parts of the Standard work well together.
5.2. Relax the Cpp17 iterator requirements
We could change the definition of Cpp17ForwardIterator ([forward.iterators]), removing the requirement that 
The authors do not consider this option to be feasible.  This change would break existing code that assumes that 
5.3. Algorithms require concepts instead of categories
We could change the wording in [algorithms.requirements] to say that the template arguments of the algorithms shall model an iterator concept rather than meet a set of Cpp17 requirements. For example:
If an algorithm’s template parameter is named,ForwardIterator , orForwardIterator1 , the template argument shallForwardIterator2 meet the Cpp17ForwardIterator requirements ([forward.iterators])model([iterator.concept.forward]) .forward_iterator 
This change relaxes the requirements on the template arguments for non-Ranges algorithms.  Implementations may need to change to no longer depend on requirements that are no longer required.  It is not expected that this will be a big burden, since a well-written algorithm shouldn’t be depending on 
This change will not break any existing code because the iterator concept imposes fewer requirements on the type than the corresponding Cpp17...Iterator requirements. Any well-formed program will continue to be well-formed and have the same behavior. Some programs that were not well-formed may become well-formed, such as the two motivating examples in this paper; these will have reasonable, non-surprising behavior.
This is the solution proposed by this paper. See immediately below for more details and analysis of this solution.
6. Impact and Details
6.1. Changes
In every bullet of [algorithms.requirements]/p4 except for the one about 
For 
This 
- 
     Concept forward_iterator equality_comparable incrementable regular forward_iterator 
- 
     Output iterators are generally not compared for equality with other iterators in well-behaved algorithms. 
6.2. Relaxed requirements
The Cpp17...Iterator requirements and the iterator concepts are worded very differently, making it challenging to compare them directly. As far as I can determine, no iterator concept requires anything of its type that is not also required by the corresponding Cpp17...Iterator set of requirements. In the other direction, these are the things required by the Cpp17...Iterator requirements that are not required by the concepts:
- 
     Cpp17InputIterator and Cpp17OutputIterator require that the iterator be copyable, but input_iterator output_iterator 
- 
     Cpp17InputIterator requires that the iterator be Cpp17EqualityComparable, but input_iterator equality_comparable 
- 
     Cpp17ForwardIterator requires that reference T & const T & forward_iterator iter_reference_t < I > 
6.3. Other uses
The Cpp17...Iterator requirements are used in several other places in the standard besides [algorithms.requirements]. It is proposed that in places where the requirements are on iterator types that the program passes to standard algorithms, the wording is changed from meeting Cpp17...Iterator requirements to modeling an iterator concept, just like is being done for [algorithms.requirements]. See § 9 Wording for all the places where this happens.
Other uses of Cpp17...Iterator requirements are not changed by this proposal. In some of those cases the requirement is on a standard iterator, not a program iterator, so relaxing the requirements could break existing code. The other uses are in [sequence.reqmts], [associative.reqmts.general], [move.iter.requirements], [locale.category], [fs.req], [fs.path.req], [fs.class.directory.iterator.general], [time.zone.db.list], [reverse.iter.requirements], [allocator.requirements.general], [stacktrace.basic.obs], [string.view.iterators], [container.requirements.general], [span.iterators], [iterator.operations], [alg.equal], and [valarray.range].
6.4. Implementation impact
If any standard algorithm implementations rely on the requirements listed in § 6.2 Relaxed requirements, those implementations will have to change to not rely on those requirements. It is assumed, though not yet verified, that not many algorithm implementations will have to change, and that the changes will not be difficult to make. The hardest part might be identifying places that need to be changed.
Any implementation that checks the iterator category of algorithm iterator arguments, such as MSVC, will have to change the way those checks happen. That should be a straightforward, if somewhat tedious, change.
6.5. User impact
No well-formed programs will become ill-formed or have different behavior as a result of this change. Some ill-formed programs will become well-formed with the behaviors users would expect. This change will reduce the number of surprises by making code that users reasonably expect to be well-formed actually well-formed. It is not expected that any users would be negatively impacted by this change.
7. Implementation Experience
None so far. Some implementation experience will be attempted if LEWG likes the proposal and is considering forwarding it on.
Because this is a change to a specification that is not directly implemented in code, it is not a simple matter to make the change to a standard library implementation and test it. I want to find out if the committee is at all interested in this idea before spending much time on implementing it.
8. Feature Test Macro
This change affects the well-formed-ness of certain code.  Some users might want to write their code two different ways based on whether or not a standard library has implemented this change.  Therefore, I believe that the benefit to users of a feature test macro is greater than the cost to implementers of defining it.  This proposal adds the new macro 
9. Wording
Changes are relative to N4888 from June 2021.
Change 25.2 "Algorithms requirements" [algorithms.requirements] paragraph 4 as follows:
Throughout this Clause, where the template parameters are not constrained, the names of template parameters are used to express type requirements.
If an algorithm’s template parameter is named
,InputIterator , orInputIterator1 , the template argument shallInputIterator2 meet the Cpp17InputIterator requirements ([input.iterators])model([iterator.concept.input]) andinput_iterator ([concept.equality.comparable]) .equality_comparable 
If an algorithm’s template parameter is named
,OutputIterator , orOutputIterator1 , the template argument shallOutputIterator2 meet the Cpp17OutputIterator requirements ([output.iterators])model([iterator.concept.output]) .output_iterator 
If an algorithm’s template parameter is named
,ForwardIterator , orForwardIterator1 , the template argument shallForwardIterator2 meet the Cpp17ForwardIterator requirements ([forward.iterators])model([iterator.concept.forward]) .forward_iterator 
If an algorithm’s template parameter is named
, the template argument shallNoThrowForwardIterator meet the Cpp17ForwardIterator requirements ([forward.iterators])model, and is required to have the property that no exceptions are thrown from increment, assignment, or comparison of, or indirection through, valid iterators.forward_iterator 
If an algorithm’s template parameter is named
,BidirectionalIterator , orBidirectionalIterator1 , the template argument shallBidirectionalIterator2 meet the Cpp17BidirectionalIterator requirements ([bidirectional.iterators])model([iterator.concept.bidir]) .bidirectional_iterator 
If an algorithm’s template parameter is named
,RandomAccessIterator , orRandomAccessIterator1 , the template argument shallRandomAccessIterator2 meet the Cpp17RandomAccessIterator requirements ([random.access.iterators])model([iterator.concept.random.access]) .random_access_iterator 
As currently worded, before these changes, the implementation is required to issue a diagnostic if the iterator type doesn’t meet the given requirements. The requirements contain some semantic elements, which the implementation cannot reliably check at compile time, making it difficult to always issue the required diagnostic. The proposed wording change does not fix this issue, since the concepts also have semantic requirements that can’t be reliably checked at compile time. Should the wording be changed to make the failure to meet the iterator requirements IFNDR or undefined behavior?
Change 25.7.12 "Sample" [alg.random.sample] paragraphs 2 and 5 as follows:
Paragraph 2:
Preconditions:
is not in the rangeout . For the overload in namespace[ first , last ) :std 
PopulationIterator meets the Cpp17InputIterator requirements ([input.iterators])models([iterator.concept.input]) andinput_iterator ([concept.equality.comparable]) .equality_comparable 
SampleIterator meets the Cpp17OutputIterator requirements ([output.iterators])models([iterator.concept.output]) .output_iterator 
SampleIterator meets the Cpp17RandomAccessIterator requirements ([random.access.iterators])models([iterator.concept.random.access]) unlessrandom_access_iterator PopulationIterator meets the Cpp17ForwardIterator requirements ([forward.iterators])models([iterator.concept.forward]).forward_iterator 
meets the requirements of a uniform random bit generator type ([rand.req.urng]).remove_reference_t < UniformRandomBitGenerator > 
Paragraph 5:
Remarks:
For the overload in namespace
, stable if and only ifstd PopulationIterator meets the Cpp17ForwardIterator requirementsmodels. For the first overload in namespaceforward_iterator , stable if and only ifranges modelsI .forward_iterator 
To the extent that the implementation of this function makes use of random numbers, the object
serves as the implementation’s source of randomness.g 
Change 26.6.8.1 "Class 
Paragraph 6:
Preconditions:
InputIterator meets the Cpp17InputIterator requirements ([input.iterators])models([iterator.concept.input]) andinput_iterator ([concept.equality.comparable]) .equality_comparable 
Paragraph 9:
Preconditions:
RandomAccessIterator meets the Cpp17RandomAccessIterator requirements ([random.access.iterators]) andmodels([iterator.concept.random.access]) and meets the requirements of a mutable iterator.random_access_iterator 
Paragraph 15:
Preconditions:
OutputIterator meets the Cpp17OutputIterator requirements ([output.iterators])models([iterator.concept.output]) .output_iterator 
Change 26.6.9.6.1 "Class template 
Preconditions:
InputIterator meets the Cpp17InputIterator requirements ([input.iterators])models([iterator.concept.input]) andinput_iterator ([concept.equality.comparable]) . Ifequality_comparable , let n = 1 and w0 = 1. Otherwise,firstW == lastW forms a sequence w of length n > 0.[ firstW , lastW ) 
Change 26.6.9.2 "Class template 
Preconditions:
andInputIteratorB eachInputIteratorW meet the Cpp17InputIterator requirements ([input.iterators])model([iterator.concept.input]).input_iterator modelsInputIteratorB ([concept.equality.comparable]) . Ifequality_comparable orfirstB == lastB , let n = 1, w0 = 1, b0 = 0, and b1 = 1. Otherwise,++ firstB == lastB forms a sequence b of length n + 1, the length of the sequence w starting from[ firstB , lastB ) is at least n, and any wk for k ≥ n are ignored by the distribution.firstW 
Change 26.6.9.3 "Class template 
Preconditions:
andInputIteratorB eachInputIteratorW meet the Cpp17InputIterator requirements ([input.iterators])model([iterator.concept.input])input_iterator modelsInputIteratorB ([concept.equality.comparable]) . Ifequality_comparable orfirstB == lastB , let n = 1, ρ0 = ρ1 = 1, b0 = 0, and b1 = 1. Otherwise,++ firstB == lastB forms a sequence b of length n + 1, the length of the sequence w starting from[ firstB , lastB ) is at least n + 1, and any wk for k ≥ n + 1 are ignored by the distribution.firstW 
Change 25.7.9 "Unique" [alg.unique] paragraph 8.2.2 as follows:
For the overloads with no
, letExecutionPolicy be the value type ofT . IfInputIterator InputIterator meets the Cpp17ForwardIterator requirementsmodels([iterator.concept.forward]) , then there are no additional requirements forforward_iterator . Otherwise, ifT OutputIterator meets the Cpp17ForwardIterator requirementsmodelsand its value type is the same asforward_iterator , thenT meets the Cpp17CopyAssignable (Table 31) requirements. Otherwise,T meets both the Cpp17CopyConstructible (Table 29) and Cpp17CopyAssignable requirements.T 
Change 25.7.14 "Shift" [alg.shift] paragraphs 5 and 6 as follows:
Preconditions:
isn >= 0 true. The type ofmeets the Cpp17MoveAssignable requirements.* first ForwardIterator meets the Cpp17BidirectionalIterator requirements ([bidirectional.iterators]) ormodels([iterator.concept.bidir]) or meets the Cpp17ValueSwappable requirements.bidirectional_iterator Effects: If
orn == 0 , does nothing. Otherwise, moves the element from positionn >= last - first into positionfirst + i for each non-negative integerfirst + n + i . In the first overload case, ifi < ( last - first ) - n ForwardIterator meets themodelsrequirementsCpp17BidirectionalIterator , does so in order starting frombidirectional_iterator and proceeding toi = ( last - first ) - n - 1 .i = 0 
Change 25.8.5 "Partitions" [alg.partitions] paragraph 8.1 as follows:
For the overload with no
, exactly N applications of the predicate and projection. At most N/2 swaps if the type ofExecutionPolicy first meets the Cpp17BidirectionalIterator requirements for the overloads in namespacemodelsorstd bidirectional_iterator for the overloads in namespace, and at most N swaps otherwise.ranges 
Change 30.9.6 "Formatting" [re.results.form] paragraph 1 as follows:
Preconditions:
andready () == trueOutputIter meets the requirements for a Cpp17OutputIterator ([output.iterators])models([iterator.concept.output]) .output_iterator 
Change 30.10.2 "
Preconditions:
BidirectionalIterator meets the Cpp17BidirectionalIterator requirements ([bidirectional.iterators])models([iterator.concept.bidir]) .bidirectional_iterator 
Change 30.10.3 "
Preconditions:
BidirectionalIterator meets the Cpp17BidirectionalIterator requirements ([bidirectional.iterators])models([iterator.concept.bidir]) .bidirectional_iterator 
Add the following feature test macro to [version.syn]:
date // also in#define __cpp_lib_algorithm_iterator_requirements ,< algorithm > ,< numeric > < memory >