1 1.1 mrg // <experimental/functional> -*- C++ -*- 2 1.1 mrg 3 1.7 mrg // Copyright (C) 2014-2022 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.6 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.3 mrg #if __cplusplus >= 201402L 36 1.1 mrg 37 1.1 mrg #include <functional> 38 1.1 mrg #include <tuple> 39 1.1 mrg #include <iterator> 40 1.1 mrg #include <unordered_map> 41 1.1 mrg #include <vector> 42 1.1 mrg #include <array> 43 1.1 mrg #include <bits/stl_algo.h> 44 1.3 mrg #ifdef _GLIBCXX_PARALLEL 45 1.3 mrg # include <parallel/algorithm> // For std::__parallel::search 46 1.3 mrg #endif 47 1.3 mrg #include <experimental/bits/lfts_config.h> 48 1.1 mrg 49 1.1 mrg namespace std _GLIBCXX_VISIBILITY(default) 50 1.1 mrg { 51 1.4 mrg _GLIBCXX_BEGIN_NAMESPACE_VERSION 52 1.4 mrg 53 1.1 mrg namespace experimental 54 1.1 mrg { 55 1.1 mrg inline namespace fundamentals_v1 56 1.1 mrg { 57 1.5 mrg // See C++14 20.9.9, Function object binders 58 1.1 mrg 59 1.1 mrg /// Variable template for std::is_bind_expression 60 1.1 mrg template<typename _Tp> 61 1.1 mrg constexpr bool is_bind_expression_v = std::is_bind_expression<_Tp>::value; 62 1.1 mrg 63 1.1 mrg /// Variable template for std::is_placeholder 64 1.1 mrg template<typename _Tp> 65 1.1 mrg constexpr int is_placeholder_v = std::is_placeholder<_Tp>::value; 66 1.1 mrg 67 1.1 mrg #define __cpp_lib_experimental_boyer_moore_searching 201411 68 1.1 mrg 69 1.1 mrg // Searchers 70 1.1 mrg 71 1.1 mrg template<typename _ForwardIterator1, typename _BinaryPredicate = equal_to<>> 72 1.1 mrg class default_searcher 73 1.1 mrg { 74 1.1 mrg public: 75 1.1 mrg default_searcher(_ForwardIterator1 __pat_first, 76 1.1 mrg _ForwardIterator1 __pat_last, 77 1.1 mrg _BinaryPredicate __pred = _BinaryPredicate()) 78 1.1 mrg : _M_m(__pat_first, __pat_last, std::move(__pred)) 79 1.1 mrg { } 80 1.1 mrg 81 1.1 mrg template<typename _ForwardIterator2> 82 1.1 mrg _ForwardIterator2 83 1.1 mrg operator()(_ForwardIterator2 __first, _ForwardIterator2 __last) const 84 1.1 mrg { 85 1.1 mrg return std::search(__first, __last, 86 1.1 mrg std::get<0>(_M_m), std::get<1>(_M_m), 87 1.1 mrg std::get<2>(_M_m)); 88 1.1 mrg } 89 1.1 mrg 90 1.1 mrg private: 91 1.1 mrg std::tuple<_ForwardIterator1, _ForwardIterator1, _BinaryPredicate> _M_m; 92 1.1 mrg }; 93 1.1 mrg 94 1.1 mrg template<typename _Key, typename _Tp, typename _Hash, typename _Pred> 95 1.1 mrg struct __boyer_moore_map_base 96 1.1 mrg { 97 1.1 mrg template<typename _RAIter> 98 1.1 mrg __boyer_moore_map_base(_RAIter __pat, size_t __patlen, 99 1.1 mrg _Hash&& __hf, _Pred&& __pred) 100 1.1 mrg : _M_bad_char{ __patlen, std::move(__hf), std::move(__pred) } 101 1.1 mrg { 102 1.1 mrg if (__patlen > 0) 103 1.1 mrg for (__diff_type __i = 0; __i < __patlen - 1; ++__i) 104 1.1 mrg _M_bad_char[__pat[__i]] = __patlen - 1 - __i; 105 1.1 mrg } 106 1.1 mrg 107 1.1 mrg using __diff_type = _Tp; 108 1.1 mrg 109 1.1 mrg __diff_type 110 1.1 mrg _M_lookup(_Key __key, __diff_type __not_found) const 111 1.1 mrg { 112 1.1 mrg auto __iter = _M_bad_char.find(__key); 113 1.1 mrg if (__iter == _M_bad_char.end()) 114 1.1 mrg return __not_found; 115 1.1 mrg return __iter->second; 116 1.1 mrg } 117 1.1 mrg 118 1.1 mrg _Pred 119 1.1 mrg _M_pred() const { return _M_bad_char.key_eq(); } 120 1.1 mrg 121 1.3 mrg _GLIBCXX_STD_C::unordered_map<_Key, _Tp, _Hash, _Pred> _M_bad_char; 122 1.1 mrg }; 123 1.1 mrg 124 1.1 mrg template<typename _Tp, size_t _Len, typename _Pred> 125 1.1 mrg struct __boyer_moore_array_base 126 1.1 mrg { 127 1.1 mrg template<typename _RAIter, typename _Unused> 128 1.1 mrg __boyer_moore_array_base(_RAIter __pat, size_t __patlen, 129 1.1 mrg _Unused&&, _Pred&& __pred) 130 1.7 mrg : _M_bad_char{ std::array<_Tp, _Len>{}, std::move(__pred) } 131 1.1 mrg { 132 1.1 mrg std::get<0>(_M_bad_char).fill(__patlen); 133 1.1 mrg if (__patlen > 0) 134 1.1 mrg for (__diff_type __i = 0; __i < __patlen - 1; ++__i) 135 1.1 mrg { 136 1.1 mrg auto __ch = __pat[__i]; 137 1.1 mrg using _UCh = std::make_unsigned_t<decltype(__ch)>; 138 1.1 mrg auto __uch = static_cast<_UCh>(__ch); 139 1.1 mrg std::get<0>(_M_bad_char)[__uch] = __patlen - 1 - __i; 140 1.1 mrg } 141 1.1 mrg } 142 1.1 mrg 143 1.1 mrg using __diff_type = _Tp; 144 1.1 mrg 145 1.1 mrg template<typename _Key> 146 1.1 mrg __diff_type 147 1.1 mrg _M_lookup(_Key __key, __diff_type __not_found) const 148 1.1 mrg { 149 1.1 mrg auto __ukey = static_cast<std::make_unsigned_t<_Key>>(__key); 150 1.1 mrg if (__ukey >= _Len) 151 1.1 mrg return __not_found; 152 1.1 mrg return std::get<0>(_M_bad_char)[__ukey]; 153 1.1 mrg } 154 1.1 mrg 155 1.1 mrg const _Pred& 156 1.1 mrg _M_pred() const { return std::get<1>(_M_bad_char); } 157 1.1 mrg 158 1.7 mrg std::tuple<std::array<_Tp, _Len>, _Pred> _M_bad_char; 159 1.1 mrg }; 160 1.1 mrg 161 1.1 mrg // Use __boyer_moore_array_base when pattern consists of narrow characters 162 1.4 mrg // (or std::byte) and uses std::equal_to as the predicate. 163 1.1 mrg template<typename _RAIter, typename _Hash, typename _Pred, 164 1.1 mrg typename _Val = typename iterator_traits<_RAIter>::value_type, 165 1.1 mrg typename _Diff = typename iterator_traits<_RAIter>::difference_type> 166 1.1 mrg using __boyer_moore_base_t 167 1.7 mrg = std::__conditional_t<std::__is_byte_like<_Val, _Pred>::value, 168 1.7 mrg __boyer_moore_array_base<_Diff, 256, _Pred>, 169 1.7 mrg __boyer_moore_map_base<_Val, _Diff, _Hash, _Pred>>; 170 1.1 mrg 171 1.1 mrg template<typename _RAIter, typename _Hash 172 1.1 mrg = std::hash<typename std::iterator_traits<_RAIter>::value_type>, 173 1.1 mrg typename _BinaryPredicate = std::equal_to<>> 174 1.1 mrg class boyer_moore_searcher 175 1.1 mrg : __boyer_moore_base_t<_RAIter, _Hash, _BinaryPredicate> 176 1.1 mrg { 177 1.1 mrg using _Base = __boyer_moore_base_t<_RAIter, _Hash, _BinaryPredicate>; 178 1.1 mrg using typename _Base::__diff_type; 179 1.1 mrg 180 1.1 mrg public: 181 1.1 mrg boyer_moore_searcher(_RAIter __pat_first, _RAIter __pat_last, 182 1.1 mrg _Hash __hf = _Hash(), 183 1.1 mrg _BinaryPredicate __pred = _BinaryPredicate()); 184 1.1 mrg 185 1.1 mrg template<typename _RandomAccessIterator2> 186 1.1 mrg _RandomAccessIterator2 187 1.1 mrg operator()(_RandomAccessIterator2 __first, 188 1.1 mrg _RandomAccessIterator2 __last) const; 189 1.1 mrg 190 1.1 mrg private: 191 1.1 mrg bool 192 1.1 mrg _M_is_prefix(_RAIter __word, __diff_type __len, 193 1.1 mrg __diff_type __pos) 194 1.1 mrg { 195 1.1 mrg const auto& __pred = this->_M_pred(); 196 1.1 mrg __diff_type __suffixlen = __len - __pos; 197 1.1 mrg for (__diff_type __i = 0; __i < __suffixlen; ++__i) 198 1.1 mrg if (!__pred(__word[__i], __word[__pos + __i])) 199 1.1 mrg return false; 200 1.1 mrg return true; 201 1.1 mrg } 202 1.1 mrg 203 1.1 mrg __diff_type 204 1.1 mrg _M_suffix_length(_RAIter __word, __diff_type __len, 205 1.1 mrg __diff_type __pos) 206 1.1 mrg { 207 1.1 mrg const auto& __pred = this->_M_pred(); 208 1.1 mrg __diff_type __i = 0; 209 1.1 mrg while (__pred(__word[__pos - __i], __word[__len - 1 - __i]) 210 1.1 mrg && __i < __pos) 211 1.1 mrg { 212 1.1 mrg ++__i; 213 1.1 mrg } 214 1.1 mrg return __i; 215 1.1 mrg } 216 1.1 mrg 217 1.1 mrg template<typename _Tp> 218 1.1 mrg __diff_type 219 1.1 mrg _M_bad_char_shift(_Tp __c) const 220 1.1 mrg { return this->_M_lookup(__c, _M_pat_end - _M_pat); } 221 1.1 mrg 222 1.1 mrg _RAIter _M_pat; 223 1.1 mrg _RAIter _M_pat_end; 224 1.3 mrg _GLIBCXX_STD_C::vector<__diff_type> _M_good_suffix; 225 1.1 mrg }; 226 1.1 mrg 227 1.1 mrg template<typename _RAIter, typename _Hash 228 1.1 mrg = std::hash<typename std::iterator_traits<_RAIter>::value_type>, 229 1.1 mrg typename _BinaryPredicate = std::equal_to<>> 230 1.1 mrg class boyer_moore_horspool_searcher 231 1.1 mrg : __boyer_moore_base_t<_RAIter, _Hash, _BinaryPredicate> 232 1.1 mrg { 233 1.1 mrg using _Base = __boyer_moore_base_t<_RAIter, _Hash, _BinaryPredicate>; 234 1.1 mrg using typename _Base::__diff_type; 235 1.1 mrg 236 1.1 mrg public: 237 1.1 mrg boyer_moore_horspool_searcher(_RAIter __pat, 238 1.1 mrg _RAIter __pat_end, 239 1.1 mrg _Hash __hf = _Hash(), 240 1.1 mrg _BinaryPredicate __pred 241 1.1 mrg = _BinaryPredicate()) 242 1.1 mrg : _Base(__pat, __pat_end - __pat, std::move(__hf), std::move(__pred)), 243 1.1 mrg _M_pat(__pat), _M_pat_end(__pat_end) 244 1.1 mrg { } 245 1.1 mrg 246 1.1 mrg template<typename _RandomAccessIterator2> 247 1.1 mrg _RandomAccessIterator2 248 1.1 mrg operator()(_RandomAccessIterator2 __first, 249 1.1 mrg _RandomAccessIterator2 __last) const 250 1.1 mrg { 251 1.1 mrg const auto& __pred = this->_M_pred(); 252 1.1 mrg auto __patlen = _M_pat_end - _M_pat; 253 1.1 mrg if (__patlen == 0) 254 1.1 mrg return __first; 255 1.1 mrg auto __len = __last - __first; 256 1.1 mrg while (__len >= __patlen) 257 1.1 mrg { 258 1.1 mrg for (auto __scan = __patlen - 1; 259 1.1 mrg __pred(__first[__scan], _M_pat[__scan]); --__scan) 260 1.1 mrg if (__scan == 0) 261 1.1 mrg return __first; 262 1.1 mrg auto __shift = _M_bad_char_shift(__first[__patlen - 1]); 263 1.1 mrg __len -= __shift; 264 1.1 mrg __first += __shift; 265 1.1 mrg } 266 1.1 mrg return __last; 267 1.1 mrg } 268 1.1 mrg 269 1.1 mrg private: 270 1.1 mrg template<typename _Tp> 271 1.1 mrg __diff_type 272 1.1 mrg _M_bad_char_shift(_Tp __c) const 273 1.1 mrg { return this->_M_lookup(__c, _M_pat_end - _M_pat); } 274 1.1 mrg 275 1.1 mrg _RAIter _M_pat; 276 1.1 mrg _RAIter _M_pat_end; 277 1.1 mrg }; 278 1.1 mrg 279 1.1 mrg /// Generator function for default_searcher 280 1.1 mrg template<typename _ForwardIterator, 281 1.1 mrg typename _BinaryPredicate = std::equal_to<>> 282 1.1 mrg inline default_searcher<_ForwardIterator, _BinaryPredicate> 283 1.1 mrg make_default_searcher(_ForwardIterator __pat_first, 284 1.1 mrg _ForwardIterator __pat_last, 285 1.1 mrg _BinaryPredicate __pred = _BinaryPredicate()) 286 1.1 mrg { return { __pat_first, __pat_last, __pred }; } 287 1.1 mrg 288 1.1 mrg /// Generator function for boyer_moore_searcher 289 1.1 mrg template<typename _RAIter, typename _Hash 290 1.1 mrg = std::hash<typename std::iterator_traits<_RAIter>::value_type>, 291 1.1 mrg typename _BinaryPredicate = equal_to<>> 292 1.1 mrg inline boyer_moore_searcher<_RAIter, _Hash, _BinaryPredicate> 293 1.1 mrg make_boyer_moore_searcher(_RAIter __pat_first, _RAIter __pat_last, 294 1.1 mrg _Hash __hf = _Hash(), 295 1.1 mrg _BinaryPredicate __pred = _BinaryPredicate()) 296 1.1 mrg { return { __pat_first, __pat_last, std::move(__hf), std::move(__pred) }; } 297 1.1 mrg 298 1.1 mrg /// Generator function for boyer_moore_horspool_searcher 299 1.1 mrg template<typename _RAIter, typename _Hash 300 1.1 mrg = std::hash<typename std::iterator_traits<_RAIter>::value_type>, 301 1.1 mrg typename _BinaryPredicate = equal_to<>> 302 1.1 mrg inline boyer_moore_horspool_searcher<_RAIter, _Hash, _BinaryPredicate> 303 1.1 mrg make_boyer_moore_horspool_searcher(_RAIter __pat_first, _RAIter __pat_last, 304 1.1 mrg _Hash __hf = _Hash(), 305 1.1 mrg _BinaryPredicate __pred 306 1.1 mrg = _BinaryPredicate()) 307 1.1 mrg { return { __pat_first, __pat_last, std::move(__hf), std::move(__pred) }; } 308 1.1 mrg 309 1.1 mrg template<typename _RAIter, typename _Hash, typename _BinaryPredicate> 310 1.1 mrg boyer_moore_searcher<_RAIter, _Hash, _BinaryPredicate>:: 311 1.1 mrg boyer_moore_searcher(_RAIter __pat, _RAIter __pat_end, 312 1.1 mrg _Hash __hf, _BinaryPredicate __pred) 313 1.1 mrg : _Base(__pat, __pat_end - __pat, std::move(__hf), std::move(__pred)), 314 1.1 mrg _M_pat(__pat), _M_pat_end(__pat_end), _M_good_suffix(__pat_end - __pat) 315 1.1 mrg { 316 1.1 mrg auto __patlen = __pat_end - __pat; 317 1.1 mrg if (__patlen == 0) 318 1.1 mrg return; 319 1.1 mrg __diff_type __last_prefix = __patlen - 1; 320 1.1 mrg for (__diff_type __p = __patlen - 1; __p >= 0; --__p) 321 1.1 mrg { 322 1.1 mrg if (_M_is_prefix(__pat, __patlen, __p + 1)) 323 1.1 mrg __last_prefix = __p + 1; 324 1.1 mrg _M_good_suffix[__p] = __last_prefix + (__patlen - 1 - __p); 325 1.1 mrg } 326 1.1 mrg for (__diff_type __p = 0; __p < __patlen - 1; ++__p) 327 1.1 mrg { 328 1.1 mrg auto __slen = _M_suffix_length(__pat, __patlen, __p); 329 1.1 mrg auto __pos = __patlen - 1 - __slen; 330 1.1 mrg if (!__pred(__pat[__p - __slen], __pat[__pos])) 331 1.1 mrg _M_good_suffix[__pos] = __patlen - 1 - __p + __slen; 332 1.1 mrg } 333 1.1 mrg } 334 1.1 mrg 335 1.1 mrg template<typename _RAIter, typename _Hash, typename _BinaryPredicate> 336 1.1 mrg template<typename _RandomAccessIterator2> 337 1.1 mrg _RandomAccessIterator2 338 1.1 mrg boyer_moore_searcher<_RAIter, _Hash, _BinaryPredicate>:: 339 1.1 mrg operator()(_RandomAccessIterator2 __first, 340 1.1 mrg _RandomAccessIterator2 __last) const 341 1.1 mrg { 342 1.1 mrg auto __patlen = _M_pat_end - _M_pat; 343 1.1 mrg if (__patlen == 0) 344 1.1 mrg return __first; 345 1.1 mrg const auto& __pred = this->_M_pred(); 346 1.1 mrg __diff_type __i = __patlen - 1; 347 1.1 mrg auto __stringlen = __last - __first; 348 1.1 mrg while (__i < __stringlen) 349 1.1 mrg { 350 1.1 mrg __diff_type __j = __patlen - 1; 351 1.1 mrg while (__j >= 0 && __pred(__first[__i], _M_pat[__j])) 352 1.1 mrg { 353 1.1 mrg --__i; 354 1.1 mrg --__j; 355 1.1 mrg } 356 1.1 mrg if (__j < 0) 357 1.1 mrg return __first + __i + 1; 358 1.1 mrg __i += std::max(_M_bad_char_shift(__first[__i]), 359 1.1 mrg _M_good_suffix[__j]); 360 1.1 mrg } 361 1.1 mrg return __last; 362 1.1 mrg } 363 1.1 mrg } // namespace fundamentals_v1 364 1.1 mrg 365 1.1 mrg inline namespace fundamentals_v2 366 1.1 mrg { 367 1.1 mrg #define __cpp_lib_experimental_not_fn 201406 368 1.1 mrg 369 1.1 mrg /// [func.not_fn] Function template not_fn 370 1.1 mrg template<typename _Fn> 371 1.1 mrg inline auto 372 1.1 mrg not_fn(_Fn&& __fn) 373 1.1 mrg noexcept(std::is_nothrow_constructible<std::decay_t<_Fn>, _Fn&&>::value) 374 1.1 mrg { 375 1.3 mrg return std::_Not_fn<std::decay_t<_Fn>>{std::forward<_Fn>(__fn), 0}; 376 1.1 mrg } 377 1.4 mrg } // namespace fundamentals_v2 378 1.4 mrg } // namespace experimental 379 1.1 mrg 380 1.1 mrg _GLIBCXX_END_NAMESPACE_VERSION 381 1.1 mrg } // namespace std 382 1.1 mrg 383 1.1 mrg #endif // C++14 384 1.1 mrg 385 1.1 mrg #endif // _GLIBCXX_EXPERIMENTAL_FUNCTIONAL 386