xref: /openbsd-src/gnu/llvm/libcxx/include/__ranges/istream_view.h (revision 4bdff4bed0e3d54e55670334c7d0077db4170f86)
1*4bdff4beSrobert // -*- C++ -*-
2*4bdff4beSrobert //===----------------------------------------------------------------------===//
3*4bdff4beSrobert //
4*4bdff4beSrobert // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
5*4bdff4beSrobert // See https://llvm.org/LICENSE.txt for license information.
6*4bdff4beSrobert // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
7*4bdff4beSrobert //
8*4bdff4beSrobert //===----------------------------------------------------------------------===//
9*4bdff4beSrobert 
10*4bdff4beSrobert #ifndef _LIBCPP___RANGES_ISTREAM_VIEW_H
11*4bdff4beSrobert #define _LIBCPP___RANGES_ISTREAM_VIEW_H
12*4bdff4beSrobert 
13*4bdff4beSrobert #include <__concepts/constructible.h>
14*4bdff4beSrobert #include <__concepts/derived_from.h>
15*4bdff4beSrobert #include <__concepts/movable.h>
16*4bdff4beSrobert #include <__config>
17*4bdff4beSrobert #include <__iterator/default_sentinel.h>
18*4bdff4beSrobert #include <__iterator/iterator_traits.h>
19*4bdff4beSrobert #include <__memory/addressof.h>
20*4bdff4beSrobert #include <__ranges/view_interface.h>
21*4bdff4beSrobert #include <__type_traits/remove_cvref.h>
22*4bdff4beSrobert #include <__utility/forward.h>
23*4bdff4beSrobert #include <cstddef>
24*4bdff4beSrobert #include <iosfwd>
25*4bdff4beSrobert 
26*4bdff4beSrobert #if !defined(_LIBCPP_HAS_NO_PRAGMA_SYSTEM_HEADER)
27*4bdff4beSrobert #  pragma GCC system_header
28*4bdff4beSrobert #endif
29*4bdff4beSrobert 
30*4bdff4beSrobert #if _LIBCPP_STD_VER >= 20
31*4bdff4beSrobert 
32*4bdff4beSrobert _LIBCPP_BEGIN_NAMESPACE_STD
33*4bdff4beSrobert 
34*4bdff4beSrobert namespace ranges {
35*4bdff4beSrobert 
36*4bdff4beSrobert template <class _Val, class _CharT, class _Traits>
requires(basic_istream<_CharT,_Traits> & __is,_Val & __t)37*4bdff4beSrobert concept __stream_extractable = requires(basic_istream<_CharT, _Traits>& __is, _Val& __t) { __is >> __t; };
38*4bdff4beSrobert 
39*4bdff4beSrobert template <movable _Val, class _CharT, class _Traits = char_traits<_CharT>>
40*4bdff4beSrobert   requires default_initializable<_Val> && __stream_extractable<_Val, _CharT, _Traits>
41*4bdff4beSrobert class basic_istream_view : public view_interface<basic_istream_view<_Val, _CharT, _Traits>> {
42*4bdff4beSrobert   class __iterator;
43*4bdff4beSrobert 
44*4bdff4beSrobert public:
basic_istream_view(basic_istream<_CharT,_Traits> & __stream)45*4bdff4beSrobert   _LIBCPP_HIDE_FROM_ABI constexpr explicit basic_istream_view(basic_istream<_CharT, _Traits>& __stream)
46*4bdff4beSrobert       : __stream_(std::addressof(__stream)) {}
47*4bdff4beSrobert 
begin()48*4bdff4beSrobert   _LIBCPP_HIDE_FROM_ABI constexpr auto begin() {
49*4bdff4beSrobert     *__stream_ >> __value_;
50*4bdff4beSrobert     return __iterator{*this};
51*4bdff4beSrobert   }
52*4bdff4beSrobert 
end()53*4bdff4beSrobert   _LIBCPP_HIDE_FROM_ABI constexpr default_sentinel_t end() const noexcept { return default_sentinel; }
54*4bdff4beSrobert 
55*4bdff4beSrobert private:
56*4bdff4beSrobert   basic_istream<_CharT, _Traits>* __stream_;
57*4bdff4beSrobert   _LIBCPP_NO_UNIQUE_ADDRESS _Val __value_ = _Val();
58*4bdff4beSrobert };
59*4bdff4beSrobert 
60*4bdff4beSrobert template <movable _Val, class _CharT, class _Traits>
61*4bdff4beSrobert   requires default_initializable<_Val> && __stream_extractable<_Val, _CharT, _Traits>
62*4bdff4beSrobert class basic_istream_view<_Val, _CharT, _Traits>::__iterator {
63*4bdff4beSrobert public:
64*4bdff4beSrobert   using iterator_concept = input_iterator_tag;
65*4bdff4beSrobert   using difference_type  = ptrdiff_t;
66*4bdff4beSrobert   using value_type       = _Val;
67*4bdff4beSrobert 
__iterator(basic_istream_view<_Val,_CharT,_Traits> & __parent)68*4bdff4beSrobert   _LIBCPP_HIDE_FROM_ABI constexpr explicit __iterator(
69*4bdff4beSrobert       basic_istream_view<_Val, _CharT, _Traits>& __parent) noexcept
70*4bdff4beSrobert       : __parent_(std::addressof(__parent)) {}
71*4bdff4beSrobert 
72*4bdff4beSrobert   __iterator(const __iterator&)                  = delete;
73*4bdff4beSrobert   _LIBCPP_HIDE_FROM_ABI __iterator(__iterator&&) = default;
74*4bdff4beSrobert 
75*4bdff4beSrobert   __iterator& operator=(const __iterator&)                  = delete;
76*4bdff4beSrobert   _LIBCPP_HIDE_FROM_ABI __iterator& operator=(__iterator&&) = default;
77*4bdff4beSrobert 
78*4bdff4beSrobert   _LIBCPP_HIDE_FROM_ABI __iterator& operator++() {
79*4bdff4beSrobert     *__parent_->__stream_ >> __parent_->__value_;
80*4bdff4beSrobert     return *this;
81*4bdff4beSrobert   }
82*4bdff4beSrobert 
83*4bdff4beSrobert   _LIBCPP_HIDE_FROM_ABI void operator++(int) { ++*this; }
84*4bdff4beSrobert 
85*4bdff4beSrobert   _LIBCPP_HIDE_FROM_ABI _Val& operator*() const { return __parent_->__value_; }
86*4bdff4beSrobert 
87*4bdff4beSrobert   _LIBCPP_HIDE_FROM_ABI friend bool operator==(const __iterator& __x, default_sentinel_t) {
88*4bdff4beSrobert     return !*__x.__get_parent_stream();
89*4bdff4beSrobert   }
90*4bdff4beSrobert 
91*4bdff4beSrobert private:
92*4bdff4beSrobert   basic_istream_view<_Val, _CharT, _Traits>* __parent_;
93*4bdff4beSrobert 
__get_parent_stream()94*4bdff4beSrobert   _LIBCPP_HIDE_FROM_ABI constexpr basic_istream<_CharT, _Traits>* __get_parent_stream() const {
95*4bdff4beSrobert     return __parent_->__stream_;
96*4bdff4beSrobert   }
97*4bdff4beSrobert };
98*4bdff4beSrobert 
99*4bdff4beSrobert template <class _Val>
100*4bdff4beSrobert using istream_view = basic_istream_view<_Val, char>;
101*4bdff4beSrobert 
102*4bdff4beSrobert #  ifndef _LIBCPP_HAS_NO_WIDE_CHARACTERS
103*4bdff4beSrobert template <class _Val>
104*4bdff4beSrobert using wistream_view = basic_istream_view<_Val, wchar_t>;
105*4bdff4beSrobert #  endif
106*4bdff4beSrobert 
107*4bdff4beSrobert namespace views {
108*4bdff4beSrobert namespace __istream {
109*4bdff4beSrobert 
110*4bdff4beSrobert // clang-format off
111*4bdff4beSrobert template <class _Tp>
112*4bdff4beSrobert struct __fn {
113*4bdff4beSrobert   template <class _Up, class _UnCVRef = remove_cvref_t<_Up>>
114*4bdff4beSrobert     requires derived_from<_UnCVRef, basic_istream<typename _UnCVRef::char_type,
115*4bdff4beSrobert                                                   typename _UnCVRef::traits_type>>
116*4bdff4beSrobert   _LIBCPP_HIDE_FROM_ABI constexpr auto operator()(_Up&& __u) const
117*4bdff4beSrobert     noexcept(noexcept(basic_istream_view<_Tp, typename _UnCVRef::char_type,
118*4bdff4beSrobert                                               typename _UnCVRef::traits_type>(std::forward<_Up>(__u))))
119*4bdff4beSrobert     -> decltype(      basic_istream_view<_Tp, typename _UnCVRef::char_type,
120*4bdff4beSrobert                                               typename _UnCVRef::traits_type>(std::forward<_Up>(__u)))
121*4bdff4beSrobert     {   return        basic_istream_view<_Tp, typename _UnCVRef::char_type,
122*4bdff4beSrobert                                               typename _UnCVRef::traits_type>(std::forward<_Up>(__u));
123*4bdff4beSrobert     }
124*4bdff4beSrobert };
125*4bdff4beSrobert // clang-format on
126*4bdff4beSrobert 
127*4bdff4beSrobert } // namespace __istream
128*4bdff4beSrobert 
129*4bdff4beSrobert inline namespace __cpo {
130*4bdff4beSrobert template <class _Tp>
131*4bdff4beSrobert inline constexpr auto istream = __istream::__fn<_Tp>{};
132*4bdff4beSrobert } // namespace __cpo
133*4bdff4beSrobert } // namespace views
134*4bdff4beSrobert 
135*4bdff4beSrobert } // namespace ranges
136*4bdff4beSrobert 
137*4bdff4beSrobert _LIBCPP_END_NAMESPACE_STD
138*4bdff4beSrobert 
139*4bdff4beSrobert #endif // _LIBCPP_STD_VER >= 20
140*4bdff4beSrobert 
141*4bdff4beSrobert #endif // _LIBCPP___RANGES_ISTREAM_VIEW_H
142