This page is a snapshot from the LWG issues list, see the Library Active Issues List for more information and the meaning of C++23 status.
elements_view::begin() and elements_view::end() have incompatible
constraintsSection: 25.7.23.2 [range.elements.view] Status: C++23 Submitter: Patrick Palka Opened: 2020-02-21 Last modified: 2023-11-22
Priority: 1
View all other issues in [range.elements.view].
View all issues with C++23 status.
Discussion:
P1994R1 (elements_view needs its own
sentinel) introduces a distinct sentinel type for elements_view.
In doing so, it replaces the two existing overloads of elements_view::end() with four new ones:
- constexpr auto end() requires (!simple-view<V>) - { return ranges::end(base_); } - - constexpr auto end() const requires simple-view<V> - { return ranges::end(base_); }+ constexpr auto end() + { return sentinel<false>{ranges::end(base_)}; } + + constexpr auto end() requires common_range<V> + { return iterator<false>{ranges::end(base_)}; } + + constexpr auto end() const + requires range<const V> + { return sentinel<true>{ranges::end(base_)}; } + + constexpr auto end() const + requires common_range<const V> + { return iterator<true>{ranges::end(base_)}; }
But now these new overloads of elements_view::end() have constraints
that are no longer consistent with the constraints of elements_view::begin():
constexpr auto begin() requires (!simple-view<V>)
{ return iterator<false>(ranges::begin(base_)); }
constexpr auto begin() const requires simple-view<V>
{ return iterator<true>(ranges::begin(base_)); }
This inconsistency means that we can easily come up with a view V for
which elements_view<V>::begin() returns an iterator<true>
and elements_view<V>::end() returns an sentinel<false>,
i.e. incomparable things of opposite constness. For example:
tuple<int, int> x[] = {{0,0}};
ranges::subrange r = {counted_iterator(x, 1), default_sentinel};
auto v = r | views::elements<0>;
v.begin() == v.end(); // ill-formed
Here, overload resolution for begin() selects the const overload because
the subrange r models simple-view. But overload resolution for
end() selects the non-const non-common_range overload. Hence
the last line of this snippet is ill-formed because it is comparing an iterator and
sentinel of opposite constness, for which we have no matching operator==
overload. So in this example v does not even model range because its begin()
and end() are incomparable.
elements_view::begin()
and on elements_view::end() are consistent and compatible. The following proposed
resolution seems to be one way to achieve that and takes inspiration from the design of
transform_view.
[2020-04-04 Issue Prioritization]
Priority to 1 after reflector discussion.
[2020-07-17; telecon]
Should be considered together with 3448(i) and 3449(i).
Previous resolution [SUPERSEDED]:
This wording is relative to N4849 after application of P1994R1.
Modify 25.7.23.2 [range.elements.view], class template
elements_viewsynopsis, as indicated:namespace std::ranges { […] template<input_range V, size_t N> requires view<V> && has-tuple-element<range_value_t<V>, N> && has-tuple-element<remove_reference_t<range_reference_t<V>>, N> class elements_view : public view_interface<elements_view<V, N>> { public: […] constexpr V base() && { return std::move(base_); } constexpr auto begin()requires (!simple-view<V>){ return iterator<false>(ranges::begin(base_)); } constexpr auto begin() const requiressimple-view<V>range<const V> { return iterator<true>(ranges::begin(base_)); } […] }; }
[2020-06-05 Tim updates P/R in light of reflector discussions and LWG 3448(i) and comments]
The fact that, as currently specified, sentinel<false> is not
comparable with iterator<true> is a problem with the specification
of this comparison, as noted in LWG 3448(i). The P/R below repairs
this problem along the lines suggested in that issue. The constraint mismatch does
make this problem easier to observe for elements_view, but the mismatch
is not independently a problem: since begin can only add constness on
simple-views for which constness is immaterial, whether
end also adds constness or not ought not to matter.
begin overload set: if const V is a range,
but V is not a simple-view, then a const elements_view<V, N>
has no viable begin at all (the simplest example of such non-simple-views
is probably single_view). That's simply broken; the fix is to constrain
the const overload of begin with just range<const V>
instead of simple-view<V>. Notably, this is how many other
uses of simple-view work already (see, e.g., take_view in
25.7.10.2 [range.take.view]).
The previous simple-view constraint on end served a
useful purpose (when done correctly): it reduces template instantiations if
the underlying view is const-agnostic. This was lost in P1994 because that paper
modeled the overload set on transform_view; however, discussion with
Eric Niebler confirmed that the reason transform_view doesn't have the
simple-view optimization is because it would add constness to
the callable object as well, which can make a material difference in the result.
Such concerns are not present in elements_view where the "callable
object" is effectively hard-coded into the type and unaffected by
const-qualification. The revised P/R below therefore restores the
simple-view optimization.
I have implemented this P/R together with P1994 on top of libstdc++ trunk and
can confirm that no existing test cases were broken by these changes, and that
it fixes the issue reported here as well as in LWG 3448 (as it relates to
elements_view).
[2020-10-02; Status to Tentatively Ready after five positive votes on the reflector]
[2020-11-09 Approved In November virtual meeting. Status changed: Tentatively Ready → WP.]
Proposed resolution:
This wording is relative to N4861. It assumes
the maybe-const helper introduced by the P/R of LWG 3448(i).
Modify 25.7.23.2 [range.elements.view], class template elements_view synopsis, as indicated:
namespace std::ranges {
[…]
template<input_range V, size_t N>
requires view<V> && has-tuple-element<range_value_t<V>, N> &&
has-tuple-element<remove_reference_t<range_reference_t<V>>, N>
class elements_view : public view_interface<elements_view<V, N>> {
public:
[…]
constexpr V base() && { return std::move(base_); }
constexpr auto begin() requires (!simple-view<V>)
{ return iterator<false>(ranges::begin(base_)); }
constexpr auto begin() const requires simple-view<V>range<const V>
{ return iterator<true>(ranges::begin(base_)); }
constexpr auto end() requires (!simple-view<V> && !common_range<V>)
{ return sentinel<false>{ranges::end(base_)}; }
constexpr auto end() requires (!simple-view<V> && common_range<V>)
{ return iterator<false>{ranges::end(base_)}; }
constexpr auto end() const requires range<const V>
{ return sentinel<true>{ranges::end(base_)}; }
constexpr auto end() const requires common_range<const V>
{ return iterator<true>{ranges::end(base_)}; }
[…]
};
}
Modify 25.7.23.4 [range.elements.sentinel] as indicated:
[Drafting note: Unlike the current P/R of LWG 3448(i), this P/R also changes the return type of
operator-to depend on the constness of iterator rather than that of the sentinel. This is consistent withsized_sentinel_for<S, I>(24.3.4.8 [iterator.concept.sizedsentinel]), which requiresdecltype(i - s)to beiter_difference_t<I>.]
[…]namespace std::ranges { template<input_range V, size_t N> requires view<V> && has-tuple-element<range_value_t<V>, N> && has-tuple-element<remove_reference_t<range_reference_t<V>>, N> template<bool Const> class elements_view<V, F>::sentinel { […] constexpr sentinel_t<Base> base() const; template<bool OtherConst> requires sentinel_for<sentinel_t<Base>, iterator_t<maybe-const<OtherConst, V>>> friend constexpr bool operator==(const iterator<OtherConst>& x, const sentinel& y); template<bool OtherConst> requires sized_sentinel_for<sentinel_t<Base>, iterator_t<maybe-const<OtherConst, V>>> friend constexpr range_difference_t<Basemaybe-const<OtherConst, V>> operator-(const iterator<OtherConst>& x, const sentinel& y)requires sized_sentinel_for<sentinel_t<Base>, iterator_t<Base>>; template<bool OtherConst> requires sized_sentinel_for<sentinel_t<Base>, iterator_t<maybe-const<OtherConst, V>>> friend constexpr range_difference_t<Basemaybe-const<OtherConst, V>> operator-(const sentinel& x, const iterator<OtherConst>& y)requires sized_sentinel_for<sentinel_t<Base>, iterator_t<Base>>; }; }template<bool OtherConst> requires sentinel_for<sentinel_t<Base>, iterator_t<maybe-const<OtherConst, V>>> friend constexpr bool operator==(const iterator<OtherConst>& x, const sentinel& y);-4- Effects: Equivalent to:
return x.current_ == y.end_;template<bool OtherConst> requires sized_sentinel_for<sentinel_t<Base>, iterator_t<maybe-const<OtherConst, V>>> friend constexpr range_difference_t<Basemaybe-const<OtherConst, V>> operator-(const iterator<OtherConst>& x, const sentinel& y)requires sized_sentinel_for<sentinel_t<Base>, iterator_t<Base>>;-5- Effects: Equivalent to:
return x.current_ - y.end_;template<bool OtherConst> requires sized_sentinel_for<sentinel_t<Base>, iterator_t<maybe-const<OtherConst, V>>> friend constexpr range_difference_t<Basemaybe-const<OtherConst, V>> operator-(const sentinel& x, const iterator<OtherConst>& y)requires sized_sentinel_for<sentinel_t<Base>, iterator_t<Base>>;-6- Effects: Equivalent to:
return x.end_ - y.current_;