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