1ff3989e6SKonstantin Varlamov //===----------------------------------------------------------------------===// 2ff3989e6SKonstantin Varlamov // 3ff3989e6SKonstantin Varlamov // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. 4ff3989e6SKonstantin Varlamov // See https://llvm.org/LICENSE.txt for license information. 5ff3989e6SKonstantin Varlamov // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception 6ff3989e6SKonstantin Varlamov // 7ff3989e6SKonstantin Varlamov //===----------------------------------------------------------------------===// 8ff3989e6SKonstantin Varlamov 9ff3989e6SKonstantin Varlamov #ifndef _LIBCPP___ALGORITHM_MAKE_PROJECTED_H 10ff3989e6SKonstantin Varlamov #define _LIBCPP___ALGORITHM_MAKE_PROJECTED_H 11ff3989e6SKonstantin Varlamov 12ff3989e6SKonstantin Varlamov #include <__config> 13ff3989e6SKonstantin Varlamov #include <__functional/identity.h> 14ff3989e6SKonstantin Varlamov #include <__functional/invoke.h> 151cdec6c9SHui Xie #include <__type_traits/decay.h> 16db7d7959SKonstantin Varlamov #include <__type_traits/enable_if.h> 1709e3a360SLouis Dionne #include <__type_traits/invoke.h> 181cdec6c9SHui Xie #include <__type_traits/is_member_pointer.h> 19db7d7959SKonstantin Varlamov #include <__utility/declval.h> 20ff3989e6SKonstantin Varlamov #include <__utility/forward.h> 21ff3989e6SKonstantin Varlamov 22ff3989e6SKonstantin Varlamov #if !defined(_LIBCPP_HAS_NO_PRAGMA_SYSTEM_HEADER) 23ff3989e6SKonstantin Varlamov # pragma GCC system_header 24ff3989e6SKonstantin Varlamov #endif 25ff3989e6SKonstantin Varlamov 26db7d7959SKonstantin Varlamov _LIBCPP_BEGIN_NAMESPACE_STD 27db7d7959SKonstantin Varlamov 28db7d7959SKonstantin Varlamov template <class _Pred, class _Proj> 29db7d7959SKonstantin Varlamov struct _ProjectedPred { 30db7d7959SKonstantin Varlamov _Pred& __pred; // Can be a unary or a binary predicate. 31db7d7959SKonstantin Varlamov _Proj& __proj; 32db7d7959SKonstantin Varlamov 3383ce1397SNikolas Klauser _LIBCPP_CONSTEXPR _LIBCPP_HIDE_FROM_ABI _ProjectedPred(_Pred& __pred_arg, _Proj& __proj_arg) 3483ce1397SNikolas Klauser : __pred(__pred_arg), __proj(__proj_arg) {} 35db7d7959SKonstantin Varlamov 36db7d7959SKonstantin Varlamov template <class _Tp> 37*0fa05456SNikolas Klauser __invoke_result_t<_Pred&, decltype(std::__invoke(std::declval<_Proj&>(), std::declval<_Tp>()))> _LIBCPP_CONSTEXPR 38*0fa05456SNikolas Klauser _LIBCPP_HIDE_FROM_ABI 399783f28cSLouis Dionne operator()(_Tp&& __v) const { 40db7d7959SKonstantin Varlamov return std::__invoke(__pred, std::__invoke(__proj, std::forward<_Tp>(__v))); 41db7d7959SKonstantin Varlamov } 42db7d7959SKonstantin Varlamov 43db7d7959SKonstantin Varlamov template <class _T1, class _T2> 44*0fa05456SNikolas Klauser __invoke_result_t<_Pred&, 45db7d7959SKonstantin Varlamov decltype(std::__invoke(std::declval<_Proj&>(), std::declval<_T1>())), 46*0fa05456SNikolas Klauser decltype(std::__invoke(std::declval<_Proj&>(), std::declval<_T2>()))> _LIBCPP_CONSTEXPR 47e2c2ffbeSLouis Dionne _LIBCPP_HIDE_FROM_ABI 489783f28cSLouis Dionne operator()(_T1&& __lhs, _T2&& __rhs) const { 499783f28cSLouis Dionne return std::__invoke( 509783f28cSLouis Dionne __pred, std::__invoke(__proj, std::forward<_T1>(__lhs)), std::__invoke(__proj, std::forward<_T2>(__rhs))); 51db7d7959SKonstantin Varlamov } 52db7d7959SKonstantin Varlamov }; 53db7d7959SKonstantin Varlamov 549783f28cSLouis Dionne template < 559783f28cSLouis Dionne class _Pred, 56b4ecfd3cSNikolas Klauser class _Proj, 579783f28cSLouis Dionne __enable_if_t<!(!is_member_pointer<__decay_t<_Pred> >::value && __is_identity<__decay_t<_Proj> >::value), int> = 0> 589783f28cSLouis Dionne _LIBCPP_HIDE_FROM_ABI _LIBCPP_CONSTEXPR _ProjectedPred<_Pred, _Proj> __make_projected(_Pred& __pred, _Proj& __proj) { 59db7d7959SKonstantin Varlamov return _ProjectedPred<_Pred, _Proj>(__pred, __proj); 60db7d7959SKonstantin Varlamov } 61db7d7959SKonstantin Varlamov 62db7d7959SKonstantin Varlamov // Avoid creating the functor and just use the pristine comparator -- for certain algorithms, this would enable 63db7d7959SKonstantin Varlamov // optimizations that rely on the type of the comparator. Additionally, this results in less layers of indirection in 64db7d7959SKonstantin Varlamov // the call stack when the comparator is invoked, even in an unoptimized build. 659783f28cSLouis Dionne template < 669783f28cSLouis Dionne class _Pred, 67b4ecfd3cSNikolas Klauser class _Proj, 689783f28cSLouis Dionne __enable_if_t<!is_member_pointer<__decay_t<_Pred> >::value && __is_identity<__decay_t<_Proj> >::value, int> = 0> 69e65cd4ceSNikolas Klauser _LIBCPP_HIDE_FROM_ABI _LIBCPP_CONSTEXPR _Pred& __make_projected(_Pred& __pred, _Proj&) { 70db7d7959SKonstantin Varlamov return __pred; 71db7d7959SKonstantin Varlamov } 72db7d7959SKonstantin Varlamov 73db7d7959SKonstantin Varlamov _LIBCPP_END_NAMESPACE_STD 74db7d7959SKonstantin Varlamov 754f15267dSNikolas Klauser #if _LIBCPP_STD_VER >= 20 76ff3989e6SKonstantin Varlamov 77ff3989e6SKonstantin Varlamov _LIBCPP_BEGIN_NAMESPACE_STD 78ff3989e6SKonstantin Varlamov 79ff3989e6SKonstantin Varlamov namespace ranges { 80ff3989e6SKonstantin Varlamov 811cdec6c9SHui Xie template <class _Comp, class _Proj1, class _Proj2> 829783f28cSLouis Dionne _LIBCPP_HIDE_FROM_ABI constexpr decltype(auto) __make_projected_comp(_Comp& __comp, _Proj1& __proj1, _Proj2& __proj2) { 83b4ecfd3cSNikolas Klauser if constexpr (__is_identity<decay_t<_Proj1>>::value && __is_identity<decay_t<_Proj2>>::value && 841cdec6c9SHui Xie !is_member_pointer_v<decay_t<_Comp>>) { 851cdec6c9SHui Xie // Avoid creating the lambda and just use the pristine comparator -- for certain algorithms, this would enable 861cdec6c9SHui Xie // optimizations that rely on the type of the comparator. 871cdec6c9SHui Xie return __comp; 881cdec6c9SHui Xie 891cdec6c9SHui Xie } else { 9002540b2fSLouis Dionne return [&](auto&& __lhs, auto&& __rhs) -> bool { 911cdec6c9SHui Xie return std::invoke(__comp, 921cdec6c9SHui Xie std::invoke(__proj1, std::forward<decltype(__lhs)>(__lhs)), 931cdec6c9SHui Xie std::invoke(__proj2, std::forward<decltype(__rhs)>(__rhs))); 941cdec6c9SHui Xie }; 951cdec6c9SHui Xie } 961cdec6c9SHui Xie } 971cdec6c9SHui Xie 98ff3989e6SKonstantin Varlamov } // namespace ranges 99ff3989e6SKonstantin Varlamov 100ff3989e6SKonstantin Varlamov _LIBCPP_END_NAMESPACE_STD 101ff3989e6SKonstantin Varlamov 1024f15267dSNikolas Klauser #endif // _LIBCPP_STD_VER >= 20 103ff3989e6SKonstantin Varlamov 104ff3989e6SKonstantin Varlamov #endif // _LIBCPP___ALGORITHM_MAKE_PROJECTED_H 105