1. Abstract
Change the iterator requirements for non-Ranges algorithms. For forward iterators and above that are constant iterators, instead of requiring that iterators meet certain Cpp17...Iterator requirements, require that the iterators model certain iterator concepts. This makes iterators from several standard views usable with non-Ranges algorithms that require forward iterators or above, such as the parallel overloads of most 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.
2.2. R1
Add the section § 8 Sentinels based on discussion in SG9 (Ranges) on 9 Aug 2021. This is just background information; there is no change to what is being proposed.
2.3. R2
Change the title from "Ranges views as inputs to non-Ranges algorithms" to "Ranges iterators as inputs to non-Ranges algorithms".
Based on discussion in SG9 (Ranges) on 13 Sep 2021, withdraw the portion of the paper that changed input iterators and output iterators to use iterator concepts. The requirements for input and output iterators remain unchanged from the current standard, using Cpp17Iterator requirements.
Add the section Mutable Iterators. This is just background information; there is no change to what is being proposed.
In § 4 Analysis, discuss the iterators for 
2.4. R3
Based on discussion in SG9 (Ranges) on 11 Oct 2021, change the proposal to handle proxy iterators correctly. This turned out to be a significant change, both to the wording and to the design section. In the wording, the change from Cpp17...Iterator requirements to iterator concepts now only happens for constant iterators, not all iterators. In the design section, § 5.3 Algorithms require concepts instead of categories and § 6 Impact and Details had significant changes and § 7 Mutable Iterators and Proxy Iterators was completely rewritten.
2.5. R4
Gather implementation experience, described in § 9 Implementation Experience. Update the § 6.4 Implementation impact section based on the implementation experience.
2.6. R5
Removed the issue about using "shall" for uncheckable requirements on user code based on discussion in LWG on 2022-04-22. I had thought it was an issue because I was not aware that Library has a different meaning for "shall" than Core does.
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 iterator category of the view’s iterator is 
- 
     iota_view :: iterator input_iterator_tag random_access_iterator_tag 
- 
     lazy_split_view :: iterator input_iterator_tag forward_iterator_tag 
- 
     split_view :: iterator input_iterator_tag forward_iterator_tag 
- 
     elements_view :: iterator transform_view :: iterator std :: get < N > input_iterator_tag std :: get < N > std :: get < N > 
- 
     In [P2321], the iterators for zip_view adjacent_view input_iterator_tag zip_transform_view adjacent_transform_view transform_view :: iterator 
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 when the iterator is used as a constant iterator. For example:
If an algorithm’s template parameter is named,ForwardIterator , orForwardIterator1 , the template argument shall meet the Cpp17ForwardIterator requirements ([forward.iterators]) if it is required to be a mutable iterator, or modelForwardIterator2 ([iterator.concept.forward]) otherwise .forward_iterator 
This change relaxes the requirements on some of 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.
Making this change only for constant iterators, not mutable iterators, is important. See § 7 Mutable Iterators and Proxy Iterators for details.
For reasons explained below, this change is proposed only for forward, bidirectional, and random access iterators. The requirements for input and output iterators remain unchanged.
6. Impact and Details
6.1. Changes
In the bullets of [algorithms.requirements]/p4 for forward, bidirectional, and random access iterators leave the requirements unchanged for mutable iterators and change the requirements on constant iterators from meeting a set of Cpp17...Iterator requirements to modeling the corresponding iterator concept.
6.2. Relaxed requirements
The Cpp17...Iterator requirements and the iterator concepts are worded very differently, making it challenging to compare them directly.
The only thing that I can find that the iterator concepts require that the Cpp17...Iterator requirements do not is that iterator destructors must be declared 
The one thing that Cpp17ForwardIterator requires that the 
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 forward or above non-mutable 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 § 11 Wording for the five places where this happens. Sections where the only requirements on program-supplied iterators are Cpp17InputIterator or Cpp17OutputIterator are not changed. These are [rand.dist.samp.discrete], [rand.dist.samp.pconst], [rand.dist.samp.plinear], and [re.results.form]. Sections where the requirements on forward or above iterators are unchanged because they are required to be mutable are [rand.util.seedseq], [alg.shift], [alg.partitions].
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 return type of dereferencing a forward (or higher), non-mutable iterator being a reference type (see § 6.2 Relaxed requirements) or the iterator’s 
Any implementation, such as MSVC, that checks the iterator category of forward iterators passed to algorithms will have to change the way those checks happen. That change, while a little bit tedious, is not at all hard.
If an algorithm checks the iterator category to choose the most efficient implementation, in some cases that code should be changed to check the iterator concept instead. Testing turned up two cases of this in libstdc++ (see § 9 Implementation Experience), but testing probably won’t find all cases like this. Visual code inspection might be necessary.
I expect that the biggest impact on implementations will be expanding their test suites to cover the code patterns that will become well-formed once this proposal is adopted.
6.5. User impact
No well-formed programs will become ill-formed (unless there are user-defined iterators with 
7. Mutable Iterators and Proxy Iterators
The change from Cpp17...Iterator requirements to iterator concepts is only being done for constant iterators. If the requirements for mutable iterators were changed in the same way, that would allow the passing of proxy iterators to algorithms that wouldn’t handle them correctly.
C++ has always had at least one standard proxy iterator: 
Ranges algorithms are designed to correctly handle proxy iterators.  For example, their specifications say that they call 
Proxy iterators can be used as constant iterators without problems. It is only their use as mutable iterators in algorithms that aren’t designed for them that causes problems.
Keeping the Cpp17...Iterator requirements for mutable iterators passed to non-Ranges algorithms prevents using proxy iterators in those situations from being well-formed, because proxy iterators do not satisfy the Cpp17ForwardIterator requirement that 
8. Sentinels
Many ranges use iterator/sentinel pairs, rather than a pair of iterators, as the bounds of the range. Non-Ranges algorithms always require a pair of iterators that have the same type, and will not work with iterator/sentinel pairs. If this proposal is adopted, more ranges than before will be usable with classic algorithms, but the sentinel issue will still prevent some ranges from being useful as inputs to classic algorithms.
The two examples in § 3 Motivation use iterator pairs and don’t suffer from the sentinel issue. 
If a range with iterator/sentinel pairs needs to be used with a classic algorithm, there is a good chance that the user can simply wrap the range in a 
I am in favor of changing the non-Ranges algorithms to use iterator/sentinel pairs. But that change can be done independently, not as part of this paper. Both changes are useful on their own, independent of the other, and wouldn’t benefit from being tied together.
9. Implementation Experience
I took a smallish test suite of the C++ parallel algorithms that I wrote a few years ago and adapted it to test this paper.  I made two copies of the test suite.  In one copy, I used 
I ran this test suite with GCC 11 in C++20 mode using its libstdc++.  That turned up three places that will need to be changed if this proposal is adopted.  In two algorithms, 
I also ran the test suite with Visual Studio 2022 with /std:c++latest, with the standard library cloned from the STL project on GitHub.  I edited the parallel algorithms in the library (see § 4.1 Current situation), changing the requirements checks on the iterators to match this proposal, which meant changing from checking that the iterator’s category was at least 
The parallel algorithms that were covered by my test suite are 
Based on these test results, this proposal is implementable with reasonable effort. Some changes will have to be made to the compiler’s standard library, but those changes should not be difficult to make. I expect that most of the effort will be spent writing tests, or adapting existing tests, to cover this area.
10. 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 
11. Wording
Changes are relative to N4888 from June 2021.
Change 25.2 "Algorithms requirements" [algorithms.requirements] paragraphs 4 and 5 as follows, combining them into a single paragraph:
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 Effects: element specifies that a value pointed to by any iterator passed as an argument is modified, then the type of that argument shall meet the requirements of a mutable iterator ([iterator.requirements]).
If an algorithm’s template parameter is named
,InputIterator , orInputIterator1 , the template argument shall meet the Cpp17InputIterator requirements ([input.iterators]).InputIterator2 
If an algorithm’s template parameter is named
,OutputIterator , orOutputIterator1 , the template argument shall meet the Cpp17OutputIterator requirements ([output.iterators]).OutputIterator2 
If an algorithm’s template parameter is named
,ForwardIterator ,ForwardIterator1 or, orForwardIterator2 , the template argument shall meet the Cpp17ForwardIterator requirements ([forward.iterators]) if it is required to be a mutable iterator, or modelNoThrowForwardIterator ([iterator.concept.forward]) otherwise .forward_iterator 
If an algorithm’s template parameter is named
, the template argumentNoThrowForwardIterator shall meet the Cpp17ForwardIterator requirements ([forward.iterators]), andis also required to have the property that no exceptions are thrown from increment, assignment, or comparison of, or indirection through, valid iterators.
If an algorithm’s template parameter is named
,BidirectionalIterator , orBidirectionalIterator1 , the template argument shall meet the Cpp17BidirectionalIterator requirements ([bidirectional.iterators]) if it is required to be a mutable iterator, or modelBidirectionalIterator2 ([iterator.concept.bidir]) otherwise .bidirectional_iterator 
If an algorithm’s template parameter is named
,RandomAccessIterator , orRandomAccessIterator1 , the template argument shall meet the Cpp17RandomAccessIterator requirements ([random.access.iterators]) if it is required to be a mutable iterator, or modelRandomAccessIterator2 ([iterator.concept.random.access]) otherwise .random_access_iterator If an algorithm’s Effects: element specifies that a value pointed to by any iterator passed as an argument is modified, then that algorithm has an additional type requirement: The type of that argument shall meet the requirements of a mutable iterator ([iterator.requirements]).[Note 1:
This requirement doesThese requirements do not affect iterator arguments thatare namedare constrained, for which iterator category and mutability requirements are expressed explicitly. — end note],OutputIterator , orOutputIterator1 , because output iterators must always be mutable, nor does it affect arguments thatOutputIterator2 
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 
meets the Cpp17InputIterator requirements ([input.iterators]).PopulationIterator 
meets the Cpp17OutputIterator requirements ([output.iterators]).SampleIterator 
meets the Cpp17RandomAccessIterator requirements ([random.access.iterators]) unlessSampleIterator 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 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 meets the Cpp17ForwardIterator requirements and its value type is the same asOutputIterator , thenT meets the Cpp17CopyAssignable (Table 31) requirements. Otherwise,T meets both the Cpp17CopyConstructible (Table 29) and Cpp17CopyAssignable requirements.T 
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 >