1 1.1 mrg // <experimental/functional> -*- C++ -*- 2 1.1 mrg 3 1.1.1.11 mrg // Copyright (C) 2014-2024 Free Software Foundation, Inc. 4 1.1 mrg // 5 1.1 mrg // This file is part of the GNU ISO C++ Library. This library is free 6 1.1 mrg // software; you can redistribute it and/or modify it under the 7 1.1 mrg // terms of the GNU General Public License as published by the 8 1.1 mrg // Free Software Foundation; either version 3, or (at your option) 9 1.1 mrg // any later version. 10 1.1 mrg 11 1.1 mrg // This library is distributed in the hope that it will be useful, 12 1.1 mrg // but WITHOUT ANY WARRANTY; without even the implied warranty of 13 1.1 mrg // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the 14 1.1 mrg // GNU General Public License for more details. 15 1.1 mrg 16 1.1 mrg // Under Section 7 of GPL version 3, you are granted additional 17 1.1 mrg // permissions described in the GCC Runtime Library Exception, version 18 1.1 mrg // 3.1, as published by the Free Software Foundation. 19 1.1 mrg 20 1.1 mrg // You should have received a copy of the GNU General Public License and 21 1.1 mrg // a copy of the GCC Runtime Library Exception along with this program; 22 1.1 mrg // see the files COPYING3 and COPYING.RUNTIME respectively. If not, see 23 1.1 mrg // <http://www.gnu.org/licenses/>. 24 1.1 mrg 25 1.1 mrg /** @file experimental/functional 26 1.1 mrg * This is a TS C++ Library header. 27 1.1.1.9 mrg * @ingroup libfund-ts 28 1.1 mrg */ 29 1.1 mrg 30 1.1 mrg #ifndef _GLIBCXX_EXPERIMENTAL_FUNCTIONAL 31 1.1 mrg #define _GLIBCXX_EXPERIMENTAL_FUNCTIONAL 1 32 1.1 mrg 33 1.1 mrg #pragma GCC system_header 34 1.1 mrg 35 1.1.1.11 mrg #include <bits/requires_hosted.h> // experimental is currently omitted 36 1.1.1.11 mrg 37 1.1.1.4 mrg #if __cplusplus >= 201402L 38 1.1 mrg 39 1.1 mrg #include <functional> 40 1.1 mrg #include <tuple> 41 1.1 mrg #include <iterator> 42 1.1 mrg #include <unordered_map> 43 1.1 mrg #include <vector> 44 1.1 mrg #include <array> 45 1.1.1.11 mrg #include <bits/stl_algobase.h> // std::max, std::search 46 1.1.1.3 mrg #include <experimental/bits/lfts_config.h> 47 1.1 mrg 48 1.1 mrg namespace std _GLIBCXX_VISIBILITY(default) 49 1.1 mrg { 50 1.1.1.7 mrg _GLIBCXX_BEGIN_NAMESPACE_VERSION 51 1.1.1.7 mrg 52 1.1 mrg namespace experimental 53 1.1 mrg { 54 1.1 mrg inline namespace fundamentals_v1 55 1.1 mrg { 56 1.1.1.8 mrg // See C++14 20.9.9, Function object binders 57 1.1 mrg 58 1.1 mrg /// Variable template for std::is_bind_expression 59 1.1 mrg template<typename _Tp> 60 1.1 mrg constexpr bool is_bind_expression_v = std::is_bind_expression<_Tp>::value; 61 1.1 mrg 62 1.1 mrg /// Variable template for std::is_placeholder 63 1.1 mrg template<typename _Tp> 64 1.1 mrg constexpr int is_placeholder_v = std::is_placeholder<_Tp>::value; 65 1.1 mrg 66 1.1 mrg #define __cpp_lib_experimental_boyer_moore_searching 201411 67 1.1 mrg 68 1.1 mrg // Searchers 69 1.1 mrg 70 1.1 mrg template<typename _ForwardIterator1, typename _BinaryPredicate = equal_to<>> 71 1.1 mrg class default_searcher 72 1.1 mrg { 73 1.1 mrg public: 74 1.1 mrg default_searcher(_ForwardIterator1 __pat_first, 75 1.1 mrg _ForwardIterator1 __pat_last, 76 1.1 mrg _BinaryPredicate __pred = _BinaryPredicate()) 77 1.1 mrg : _M_m(__pat_first, __pat_last, std::move(__pred)) 78 1.1 mrg { } 79 1.1 mrg 80 1.1 mrg template<typename _ForwardIterator2> 81 1.1 mrg _ForwardIterator2 82 1.1 mrg operator()(_ForwardIterator2 __first, _ForwardIterator2 __last) const 83 1.1 mrg { 84 1.1 mrg return std::search(__first, __last, 85 1.1 mrg std::get<0>(_M_m), std::get<1>(_M_m), 86 1.1 mrg std::get<2>(_M_m)); 87 1.1 mrg } 88 1.1 mrg 89 1.1 mrg private: 90 1.1 mrg std::tuple<_ForwardIterator1, _ForwardIterator1, _BinaryPredicate> _M_m; 91 1.1 mrg }; 92 1.1 mrg 93 1.1 mrg template<typename _Key, typename _Tp, typename _Hash, typename _Pred> 94 1.1 mrg struct __boyer_moore_map_base 95 1.1 mrg { 96 1.1 mrg template<typename _RAIter> 97 1.1 mrg __boyer_moore_map_base(_RAIter __pat, size_t __patlen, 98 1.1 mrg _Hash&& __hf, _Pred&& __pred) 99 1.1 mrg : _M_bad_char{ __patlen, std::move(__hf), std::move(__pred) } 100 1.1 mrg { 101 1.1 mrg if (__patlen > 0) 102 1.1 mrg for (__diff_type __i = 0; __i < __patlen - 1; ++__i) 103 1.1 mrg _M_bad_char[__pat[__i]] = __patlen - 1 - __i; 104 1.1 mrg } 105 1.1 mrg 106 1.1 mrg using __diff_type = _Tp; 107 1.1 mrg 108 1.1 mrg __diff_type 109 1.1 mrg _M_lookup(_Key __key, __diff_type __not_found) const 110 1.1 mrg { 111 1.1 mrg auto __iter = _M_bad_char.find(__key); 112 1.1 mrg if (__iter == _M_bad_char.end()) 113 1.1 mrg return __not_found; 114 1.1 mrg return __iter->second; 115 1.1 mrg } 116 1.1 mrg 117 1.1 mrg _Pred 118 1.1 mrg _M_pred() const { return _M_bad_char.key_eq(); } 119 1.1 mrg 120 1.1.1.2 mrg _GLIBCXX_STD_C::unordered_map<_Key, _Tp, _Hash, _Pred> _M_bad_char; 121 1.1 mrg }; 122 1.1 mrg 123 1.1 mrg template<typename _Tp, size_t _Len, typename _Pred> 124 1.1 mrg struct __boyer_moore_array_base 125 1.1 mrg { 126 1.1 mrg template<typename _RAIter, typename _Unused> 127 1.1 mrg __boyer_moore_array_base(_RAIter __pat, size_t __patlen, 128 1.1 mrg _Unused&&, _Pred&& __pred) 129 1.1.1.10 mrg : _M_bad_char{ std::array<_Tp, _Len>{}, std::move(__pred) } 130 1.1 mrg { 131 1.1 mrg std::get<0>(_M_bad_char).fill(__patlen); 132 1.1 mrg if (__patlen > 0) 133 1.1 mrg for (__diff_type __i = 0; __i < __patlen - 1; ++__i) 134 1.1 mrg { 135 1.1 mrg auto __ch = __pat[__i]; 136 1.1 mrg using _UCh = std::make_unsigned_t<decltype(__ch)>; 137 1.1 mrg auto __uch = static_cast<_UCh>(__ch); 138 1.1 mrg std::get<0>(_M_bad_char)[__uch] = __patlen - 1 - __i; 139 1.1 mrg } 140 1.1 mrg } 141 1.1 mrg 142 1.1 mrg using __diff_type = _Tp; 143 1.1 mrg 144 1.1 mrg template<typename _Key> 145 1.1 mrg __diff_type 146 1.1 mrg _M_lookup(_Key __key, __diff_type __not_found) const 147 1.1 mrg { 148 1.1 mrg auto __ukey = static_cast<std::make_unsigned_t<_Key>>(__key); 149 1.1 mrg if (__ukey >= _Len) 150 1.1 mrg return __not_found; 151 1.1 mrg return std::get<0>(_M_bad_char)[__ukey]; 152 1.1 mrg } 153 1.1 mrg 154 1.1 mrg const _Pred& 155 1.1 mrg _M_pred() const { return std::get<1>(_M_bad_char); } 156 1.1 mrg 157 1.1.1.10 mrg std::tuple<std::array<_Tp, _Len>, _Pred> _M_bad_char; 158 1.1 mrg }; 159 1.1 mrg 160 1.1 mrg // Use __boyer_moore_array_base when pattern consists of narrow characters 161 1.1.1.7 mrg // (or std::byte) and uses std::equal_to as the predicate. 162 1.1 mrg template<typename _RAIter, typename _Hash, typename _Pred, 163 1.1 mrg typename _Val = typename iterator_traits<_RAIter>::value_type, 164 1.1 mrg typename _Diff = typename iterator_traits<_RAIter>::difference_type> 165 1.1 mrg using __boyer_moore_base_t 166 1.1.1.10 mrg = std::__conditional_t<std::__is_byte_like<_Val, _Pred>::value, 167 1.1.1.10 mrg __boyer_moore_array_base<_Diff, 256, _Pred>, 168 1.1.1.10 mrg __boyer_moore_map_base<_Val, _Diff, _Hash, _Pred>>; 169 1.1 mrg 170 1.1 mrg template<typename _RAIter, typename _Hash 171 1.1 mrg = std::hash<typename std::iterator_traits<_RAIter>::value_type>, 172 1.1 mrg typename _BinaryPredicate = std::equal_to<>> 173 1.1 mrg class boyer_moore_searcher 174 1.1 mrg : __boyer_moore_base_t<_RAIter, _Hash, _BinaryPredicate> 175 1.1 mrg { 176 1.1 mrg using _Base = __boyer_moore_base_t<_RAIter, _Hash, _BinaryPredicate>; 177 1.1 mrg using typename _Base::__diff_type; 178 1.1 mrg 179 1.1 mrg public: 180 1.1 mrg boyer_moore_searcher(_RAIter __pat_first, _RAIter __pat_last, 181 1.1 mrg _Hash __hf = _Hash(), 182 1.1 mrg _BinaryPredicate __pred = _BinaryPredicate()); 183 1.1 mrg 184 1.1 mrg template<typename _RandomAccessIterator2> 185 1.1 mrg _RandomAccessIterator2 186 1.1 mrg operator()(_RandomAccessIterator2 __first, 187 1.1 mrg _RandomAccessIterator2 __last) const; 188 1.1 mrg 189 1.1 mrg private: 190 1.1 mrg bool 191 1.1 mrg _M_is_prefix(_RAIter __word, __diff_type __len, 192 1.1 mrg __diff_type __pos) 193 1.1 mrg { 194 1.1 mrg const auto& __pred = this->_M_pred(); 195 1.1 mrg __diff_type __suffixlen = __len - __pos; 196 1.1 mrg for (__diff_type __i = 0; __i < __suffixlen; ++__i) 197 1.1 mrg if (!__pred(__word[__i], __word[__pos + __i])) 198 1.1 mrg return false; 199 1.1 mrg return true; 200 1.1 mrg } 201 1.1 mrg 202 1.1 mrg __diff_type 203 1.1 mrg _M_suffix_length(_RAIter __word, __diff_type __len, 204 1.1 mrg __diff_type __pos) 205 1.1 mrg { 206 1.1 mrg const auto& __pred = this->_M_pred(); 207 1.1 mrg __diff_type __i = 0; 208 1.1 mrg while (__pred(__word[__pos - __i], __word[__len - 1 - __i]) 209 1.1 mrg && __i < __pos) 210 1.1 mrg { 211 1.1 mrg ++__i; 212 1.1 mrg } 213 1.1 mrg return __i; 214 1.1 mrg } 215 1.1 mrg 216 1.1 mrg template<typename _Tp> 217 1.1 mrg __diff_type 218 1.1 mrg _M_bad_char_shift(_Tp __c) const 219 1.1 mrg { return this->_M_lookup(__c, _M_pat_end - _M_pat); } 220 1.1 mrg 221 1.1 mrg _RAIter _M_pat; 222 1.1 mrg _RAIter _M_pat_end; 223 1.1.1.2 mrg _GLIBCXX_STD_C::vector<__diff_type> _M_good_suffix; 224 1.1 mrg }; 225 1.1 mrg 226 1.1 mrg template<typename _RAIter, typename _Hash 227 1.1 mrg = std::hash<typename std::iterator_traits<_RAIter>::value_type>, 228 1.1 mrg typename _BinaryPredicate = std::equal_to<>> 229 1.1 mrg class boyer_moore_horspool_searcher 230 1.1 mrg : __boyer_moore_base_t<_RAIter, _Hash, _BinaryPredicate> 231 1.1 mrg { 232 1.1 mrg using _Base = __boyer_moore_base_t<_RAIter, _Hash, _BinaryPredicate>; 233 1.1 mrg using typename _Base::__diff_type; 234 1.1 mrg 235 1.1 mrg public: 236 1.1 mrg boyer_moore_horspool_searcher(_RAIter __pat, 237 1.1 mrg _RAIter __pat_end, 238 1.1 mrg _Hash __hf = _Hash(), 239 1.1 mrg _BinaryPredicate __pred 240 1.1 mrg = _BinaryPredicate()) 241 1.1 mrg : _Base(__pat, __pat_end - __pat, std::move(__hf), std::move(__pred)), 242 1.1 mrg _M_pat(__pat), _M_pat_end(__pat_end) 243 1.1 mrg { } 244 1.1 mrg 245 1.1 mrg template<typename _RandomAccessIterator2> 246 1.1 mrg _RandomAccessIterator2 247 1.1 mrg operator()(_RandomAccessIterator2 __first, 248 1.1 mrg _RandomAccessIterator2 __last) const 249 1.1 mrg { 250 1.1 mrg const auto& __pred = this->_M_pred(); 251 1.1 mrg auto __patlen = _M_pat_end - _M_pat; 252 1.1 mrg if (__patlen == 0) 253 1.1 mrg return __first; 254 1.1 mrg auto __len = __last - __first; 255 1.1 mrg while (__len >= __patlen) 256 1.1 mrg { 257 1.1 mrg for (auto __scan = __patlen - 1; 258 1.1 mrg __pred(__first[__scan], _M_pat[__scan]); --__scan) 259 1.1 mrg if (__scan == 0) 260 1.1 mrg return __first; 261 1.1 mrg auto __shift = _M_bad_char_shift(__first[__patlen - 1]); 262 1.1 mrg __len -= __shift; 263 1.1 mrg __first += __shift; 264 1.1 mrg } 265 1.1 mrg return __last; 266 1.1 mrg } 267 1.1 mrg 268 1.1 mrg private: 269 1.1 mrg template<typename _Tp> 270 1.1 mrg __diff_type 271 1.1 mrg _M_bad_char_shift(_Tp __c) const 272 1.1 mrg { return this->_M_lookup(__c, _M_pat_end - _M_pat); } 273 1.1 mrg 274 1.1 mrg _RAIter _M_pat; 275 1.1 mrg _RAIter _M_pat_end; 276 1.1 mrg }; 277 1.1 mrg 278 1.1 mrg /// Generator function for default_searcher 279 1.1 mrg template<typename _ForwardIterator, 280 1.1 mrg typename _BinaryPredicate = std::equal_to<>> 281 1.1 mrg inline default_searcher<_ForwardIterator, _BinaryPredicate> 282 1.1 mrg make_default_searcher(_ForwardIterator __pat_first, 283 1.1 mrg _ForwardIterator __pat_last, 284 1.1 mrg _BinaryPredicate __pred = _BinaryPredicate()) 285 1.1 mrg { return { __pat_first, __pat_last, __pred }; } 286 1.1 mrg 287 1.1 mrg /// Generator function for boyer_moore_searcher 288 1.1 mrg template<typename _RAIter, typename _Hash 289 1.1 mrg = std::hash<typename std::iterator_traits<_RAIter>::value_type>, 290 1.1 mrg typename _BinaryPredicate = equal_to<>> 291 1.1 mrg inline boyer_moore_searcher<_RAIter, _Hash, _BinaryPredicate> 292 1.1 mrg make_boyer_moore_searcher(_RAIter __pat_first, _RAIter __pat_last, 293 1.1 mrg _Hash __hf = _Hash(), 294 1.1 mrg _BinaryPredicate __pred = _BinaryPredicate()) 295 1.1 mrg { return { __pat_first, __pat_last, std::move(__hf), std::move(__pred) }; } 296 1.1 mrg 297 1.1 mrg /// Generator function for boyer_moore_horspool_searcher 298 1.1 mrg template<typename _RAIter, typename _Hash 299 1.1 mrg = std::hash<typename std::iterator_traits<_RAIter>::value_type>, 300 1.1 mrg typename _BinaryPredicate = equal_to<>> 301 1.1 mrg inline boyer_moore_horspool_searcher<_RAIter, _Hash, _BinaryPredicate> 302 1.1 mrg make_boyer_moore_horspool_searcher(_RAIter __pat_first, _RAIter __pat_last, 303 1.1 mrg _Hash __hf = _Hash(), 304 1.1 mrg _BinaryPredicate __pred 305 1.1 mrg = _BinaryPredicate()) 306 1.1 mrg { return { __pat_first, __pat_last, std::move(__hf), std::move(__pred) }; } 307 1.1 mrg 308 1.1 mrg template<typename _RAIter, typename _Hash, typename _BinaryPredicate> 309 1.1 mrg boyer_moore_searcher<_RAIter, _Hash, _BinaryPredicate>:: 310 1.1 mrg boyer_moore_searcher(_RAIter __pat, _RAIter __pat_end, 311 1.1 mrg _Hash __hf, _BinaryPredicate __pred) 312 1.1 mrg : _Base(__pat, __pat_end - __pat, std::move(__hf), std::move(__pred)), 313 1.1 mrg _M_pat(__pat), _M_pat_end(__pat_end), _M_good_suffix(__pat_end - __pat) 314 1.1 mrg { 315 1.1 mrg auto __patlen = __pat_end - __pat; 316 1.1 mrg if (__patlen == 0) 317 1.1 mrg return; 318 1.1 mrg __diff_type __last_prefix = __patlen - 1; 319 1.1 mrg for (__diff_type __p = __patlen - 1; __p >= 0; --__p) 320 1.1 mrg { 321 1.1 mrg if (_M_is_prefix(__pat, __patlen, __p + 1)) 322 1.1 mrg __last_prefix = __p + 1; 323 1.1 mrg _M_good_suffix[__p] = __last_prefix + (__patlen - 1 - __p); 324 1.1 mrg } 325 1.1 mrg for (__diff_type __p = 0; __p < __patlen - 1; ++__p) 326 1.1 mrg { 327 1.1 mrg auto __slen = _M_suffix_length(__pat, __patlen, __p); 328 1.1 mrg auto __pos = __patlen - 1 - __slen; 329 1.1 mrg if (!__pred(__pat[__p - __slen], __pat[__pos])) 330 1.1 mrg _M_good_suffix[__pos] = __patlen - 1 - __p + __slen; 331 1.1 mrg } 332 1.1 mrg } 333 1.1 mrg 334 1.1 mrg template<typename _RAIter, typename _Hash, typename _BinaryPredicate> 335 1.1 mrg template<typename _RandomAccessIterator2> 336 1.1 mrg _RandomAccessIterator2 337 1.1 mrg boyer_moore_searcher<_RAIter, _Hash, _BinaryPredicate>:: 338 1.1 mrg operator()(_RandomAccessIterator2 __first, 339 1.1 mrg _RandomAccessIterator2 __last) const 340 1.1 mrg { 341 1.1 mrg auto __patlen = _M_pat_end - _M_pat; 342 1.1 mrg if (__patlen == 0) 343 1.1 mrg return __first; 344 1.1 mrg const auto& __pred = this->_M_pred(); 345 1.1 mrg __diff_type __i = __patlen - 1; 346 1.1 mrg auto __stringlen = __last - __first; 347 1.1 mrg while (__i < __stringlen) 348 1.1 mrg { 349 1.1 mrg __diff_type __j = __patlen - 1; 350 1.1 mrg while (__j >= 0 && __pred(__first[__i], _M_pat[__j])) 351 1.1 mrg { 352 1.1 mrg --__i; 353 1.1 mrg --__j; 354 1.1 mrg } 355 1.1 mrg if (__j < 0) 356 1.1 mrg return __first + __i + 1; 357 1.1 mrg __i += std::max(_M_bad_char_shift(__first[__i]), 358 1.1 mrg _M_good_suffix[__j]); 359 1.1 mrg } 360 1.1 mrg return __last; 361 1.1 mrg } 362 1.1 mrg } // namespace fundamentals_v1 363 1.1 mrg 364 1.1 mrg inline namespace fundamentals_v2 365 1.1 mrg { 366 1.1 mrg #define __cpp_lib_experimental_not_fn 201406 367 1.1 mrg 368 1.1 mrg /// [func.not_fn] Function template not_fn 369 1.1 mrg template<typename _Fn> 370 1.1 mrg inline auto 371 1.1 mrg not_fn(_Fn&& __fn) 372 1.1 mrg noexcept(std::is_nothrow_constructible<std::decay_t<_Fn>, _Fn&&>::value) 373 1.1 mrg { 374 1.1.1.4 mrg return std::_Not_fn<std::decay_t<_Fn>>{std::forward<_Fn>(__fn), 0}; 375 1.1 mrg } 376 1.1.1.6 mrg } // namespace fundamentals_v2 377 1.1.1.6 mrg } // namespace experimental 378 1.1.1.7 mrg 379 1.1.1.7 mrg _GLIBCXX_END_NAMESPACE_VERSION 380 1.1 mrg } // namespace std 381 1.1 mrg 382 1.1 mrg #endif // C++14 383 1.1 mrg 384 1.1 mrg #endif // _GLIBCXX_EXPERIMENTAL_FUNCTIONAL 385