xref: /llvm-project/libcxx/include/__cxx03/__algorithm/make_projected.h (revision ce7771902dc50d900de639d499a60486b83f70e0)
1e78f53d1SNikolas Klauser //===----------------------------------------------------------------------===//
2e78f53d1SNikolas Klauser //
3e78f53d1SNikolas Klauser // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4e78f53d1SNikolas Klauser // See https://llvm.org/LICENSE.txt for license information.
5e78f53d1SNikolas Klauser // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6e78f53d1SNikolas Klauser //
7e78f53d1SNikolas Klauser //===----------------------------------------------------------------------===//
8e78f53d1SNikolas Klauser 
9*ce777190SNikolas Klauser #ifndef _LIBCPP___CXX03___ALGORITHM_MAKE_PROJECTED_H
10*ce777190SNikolas Klauser #define _LIBCPP___CXX03___ALGORITHM_MAKE_PROJECTED_H
11e78f53d1SNikolas Klauser 
1273fbae83SNikolas Klauser #include <__cxx03/__concepts/same_as.h>
1373fbae83SNikolas Klauser #include <__cxx03/__config>
1473fbae83SNikolas Klauser #include <__cxx03/__functional/identity.h>
1573fbae83SNikolas Klauser #include <__cxx03/__functional/invoke.h>
1673fbae83SNikolas Klauser #include <__cxx03/__type_traits/decay.h>
1773fbae83SNikolas Klauser #include <__cxx03/__type_traits/enable_if.h>
1873fbae83SNikolas Klauser #include <__cxx03/__type_traits/integral_constant.h>
1973fbae83SNikolas Klauser #include <__cxx03/__type_traits/is_member_pointer.h>
2073fbae83SNikolas Klauser #include <__cxx03/__type_traits/is_same.h>
2173fbae83SNikolas Klauser #include <__cxx03/__utility/declval.h>
2273fbae83SNikolas Klauser #include <__cxx03/__utility/forward.h>
23e78f53d1SNikolas Klauser 
24e78f53d1SNikolas Klauser #if !defined(_LIBCPP_HAS_NO_PRAGMA_SYSTEM_HEADER)
25e78f53d1SNikolas Klauser #  pragma GCC system_header
26e78f53d1SNikolas Klauser #endif
27e78f53d1SNikolas Klauser 
28e78f53d1SNikolas Klauser _LIBCPP_BEGIN_NAMESPACE_STD
29e78f53d1SNikolas Klauser 
30e78f53d1SNikolas Klauser template <class _Pred, class _Proj>
31e78f53d1SNikolas Klauser struct _ProjectedPred {
32e78f53d1SNikolas Klauser   _Pred& __pred; // Can be a unary or a binary predicate.
33e78f53d1SNikolas Klauser   _Proj& __proj;
34e78f53d1SNikolas Klauser 
35e78f53d1SNikolas Klauser   _LIBCPP_CONSTEXPR _LIBCPP_HIDE_FROM_ABI _ProjectedPred(_Pred& __pred_arg, _Proj& __proj_arg)
36e78f53d1SNikolas Klauser       : __pred(__pred_arg), __proj(__proj_arg) {}
37e78f53d1SNikolas Klauser 
38e78f53d1SNikolas Klauser   template <class _Tp>
39e78f53d1SNikolas Klauser   typename __invoke_of<_Pred&, decltype(std::__invoke(std::declval<_Proj&>(), std::declval<_Tp>()))>::type
40e78f53d1SNikolas Klauser       _LIBCPP_CONSTEXPR _LIBCPP_HIDE_FROM_ABI
41e78f53d1SNikolas Klauser       operator()(_Tp&& __v) const {
42e78f53d1SNikolas Klauser     return std::__invoke(__pred, std::__invoke(__proj, std::forward<_Tp>(__v)));
43e78f53d1SNikolas Klauser   }
44e78f53d1SNikolas Klauser 
45e78f53d1SNikolas Klauser   template <class _T1, class _T2>
46e78f53d1SNikolas Klauser   typename __invoke_of<_Pred&,
47e78f53d1SNikolas Klauser                        decltype(std::__invoke(std::declval<_Proj&>(), std::declval<_T1>())),
48e78f53d1SNikolas Klauser                        decltype(std::__invoke(std::declval<_Proj&>(), std::declval<_T2>()))>::type _LIBCPP_CONSTEXPR
49e78f53d1SNikolas Klauser   _LIBCPP_HIDE_FROM_ABI
50e78f53d1SNikolas Klauser   operator()(_T1&& __lhs, _T2&& __rhs) const {
51e78f53d1SNikolas Klauser     return std::__invoke(
52e78f53d1SNikolas Klauser         __pred, std::__invoke(__proj, std::forward<_T1>(__lhs)), std::__invoke(__proj, std::forward<_T2>(__rhs)));
53e78f53d1SNikolas Klauser   }
54e78f53d1SNikolas Klauser };
55e78f53d1SNikolas Klauser 
56e78f53d1SNikolas Klauser template <
57e78f53d1SNikolas Klauser     class _Pred,
58e78f53d1SNikolas Klauser     class _Proj,
59e78f53d1SNikolas Klauser     __enable_if_t<!(!is_member_pointer<__decay_t<_Pred> >::value && __is_identity<__decay_t<_Proj> >::value), int> = 0>
60e78f53d1SNikolas Klauser _LIBCPP_HIDE_FROM_ABI _LIBCPP_CONSTEXPR _ProjectedPred<_Pred, _Proj> __make_projected(_Pred& __pred, _Proj& __proj) {
61e78f53d1SNikolas Klauser   return _ProjectedPred<_Pred, _Proj>(__pred, __proj);
62e78f53d1SNikolas Klauser }
63e78f53d1SNikolas Klauser 
64e78f53d1SNikolas Klauser // Avoid creating the functor and just use the pristine comparator -- for certain algorithms, this would enable
65e78f53d1SNikolas Klauser // optimizations that rely on the type of the comparator. Additionally, this results in less layers of indirection in
66e78f53d1SNikolas Klauser // the call stack when the comparator is invoked, even in an unoptimized build.
67e78f53d1SNikolas Klauser template <
68e78f53d1SNikolas Klauser     class _Pred,
69e78f53d1SNikolas Klauser     class _Proj,
70e78f53d1SNikolas Klauser     __enable_if_t<!is_member_pointer<__decay_t<_Pred> >::value && __is_identity<__decay_t<_Proj> >::value, int> = 0>
71e78f53d1SNikolas Klauser _LIBCPP_HIDE_FROM_ABI _LIBCPP_CONSTEXPR _Pred& __make_projected(_Pred& __pred, _Proj&) {
72e78f53d1SNikolas Klauser   return __pred;
73e78f53d1SNikolas Klauser }
74e78f53d1SNikolas Klauser 
75e78f53d1SNikolas Klauser _LIBCPP_END_NAMESPACE_STD
76e78f53d1SNikolas Klauser 
77e78f53d1SNikolas Klauser #if _LIBCPP_STD_VER >= 20
78e78f53d1SNikolas Klauser 
79e78f53d1SNikolas Klauser _LIBCPP_BEGIN_NAMESPACE_STD
80e78f53d1SNikolas Klauser 
81e78f53d1SNikolas Klauser namespace ranges {
82e78f53d1SNikolas Klauser 
83e78f53d1SNikolas Klauser template <class _Comp, class _Proj1, class _Proj2>
84e78f53d1SNikolas Klauser _LIBCPP_HIDE_FROM_ABI constexpr decltype(auto) __make_projected_comp(_Comp& __comp, _Proj1& __proj1, _Proj2& __proj2) {
85e78f53d1SNikolas Klauser   if constexpr (__is_identity<decay_t<_Proj1>>::value && __is_identity<decay_t<_Proj2>>::value &&
86e78f53d1SNikolas Klauser                 !is_member_pointer_v<decay_t<_Comp>>) {
87e78f53d1SNikolas Klauser     // Avoid creating the lambda and just use the pristine comparator -- for certain algorithms, this would enable
88e78f53d1SNikolas Klauser     // optimizations that rely on the type of the comparator.
89e78f53d1SNikolas Klauser     return __comp;
90e78f53d1SNikolas Klauser 
91e78f53d1SNikolas Klauser   } else {
92e78f53d1SNikolas Klauser     return [&](auto&& __lhs, auto&& __rhs) -> bool {
93e78f53d1SNikolas Klauser       return std::invoke(__comp,
94e78f53d1SNikolas Klauser                          std::invoke(__proj1, std::forward<decltype(__lhs)>(__lhs)),
95e78f53d1SNikolas Klauser                          std::invoke(__proj2, std::forward<decltype(__rhs)>(__rhs)));
96e78f53d1SNikolas Klauser     };
97e78f53d1SNikolas Klauser   }
98e78f53d1SNikolas Klauser }
99e78f53d1SNikolas Klauser 
100e78f53d1SNikolas Klauser } // namespace ranges
101e78f53d1SNikolas Klauser 
102e78f53d1SNikolas Klauser _LIBCPP_END_NAMESPACE_STD
103e78f53d1SNikolas Klauser 
104e78f53d1SNikolas Klauser #endif // _LIBCPP_STD_VER >= 20
105e78f53d1SNikolas Klauser 
106*ce777190SNikolas Klauser #endif // _LIBCPP___CXX03___ALGORITHM_MAKE_PROJECTED_H
107