aboutsummaryrefslogtreecommitdiff
path: root/libcxx/include/__utility/try_key_extraction.h
blob: 755c08214019fc05c270dbda7026d6d7114e72b9 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
//===----------------------------------------------------------------------===//
//
// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
// See https://llvm.org/LICENSE.txt for license information.
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
//
//===----------------------------------------------------------------------===//

#ifndef _LIBCPP___UTILITY_TRY_EXTRACT_KEY_H
#define _LIBCPP___UTILITY_TRY_EXTRACT_KEY_H

#include <__config>
#include <__fwd/pair.h>
#include <__fwd/tuple.h>
#include <__type_traits/enable_if.h>
#include <__type_traits/is_same.h>
#include <__type_traits/remove_const.h>
#include <__type_traits/remove_const_ref.h>
#include <__utility/declval.h>
#include <__utility/forward.h>
#include <__utility/piecewise_construct.h>
#include <__utility/priority_tag.h>

#if !defined(_LIBCPP_HAS_NO_PRAGMA_SYSTEM_HEADER)
#  pragma GCC system_header
#endif

_LIBCPP_BEGIN_NAMESPACE_STD

template <class _KeyT, class _Ret, class _WithKey, class _WithoutKey, class... _Args>
_LIBCPP_HIDE_FROM_ABI _Ret
__try_key_extraction_impl(__priority_tag<0>, _WithKey, _WithoutKey __without_key, _Args&&... __args) {
  return __without_key(std::forward<_Args>(__args)...);
}

template <class _KeyT,
          class _Ret,
          class _WithKey,
          class _WithoutKey,
          class _Arg,
          __enable_if_t<is_same<_KeyT, __remove_const_ref_t<_Arg> >::value, int> = 0>
_LIBCPP_HIDE_FROM_ABI _Ret
__try_key_extraction_impl(__priority_tag<1>, _WithKey __with_key, _WithoutKey, _Arg&& __arg) {
  return __with_key(__arg, std::forward<_Arg>(__arg));
}

template <class _KeyT,
          class _Ret,
          class _WithKey,
          class _WithoutKey,
          class _Arg,
          __enable_if_t<__is_pair_v<__remove_const_ref_t<_Arg> > &&
                            is_same<__remove_const_t<typename __remove_const_ref_t<_Arg>::first_type>, _KeyT>::value,
                        int> = 0>
_LIBCPP_HIDE_FROM_ABI _Ret
__try_key_extraction_impl(__priority_tag<1>, _WithKey __with_key, _WithoutKey, _Arg&& __arg) {
  return __with_key(__arg.first, std::forward<_Arg>(__arg));
}

template <class _KeyT,
          class _Ret,
          class _WithKey,
          class _WithoutKey,
          class _Arg1,
          class _Arg2,
          __enable_if_t<is_same<_KeyT, __remove_const_ref_t<_Arg1> >::value, int> = 0>
_LIBCPP_HIDE_FROM_ABI _Ret
__try_key_extraction_impl(__priority_tag<1>, _WithKey __with_key, _WithoutKey, _Arg1&& __arg1, _Arg2&& __arg2) {
  return __with_key(__arg1, std::forward<_Arg1>(__arg1), std::forward<_Arg2>(__arg2));
}

#ifndef _LIBCPP_CXX03_LANG
template <class _KeyT,
          class _Ret,
          class _WithKey,
          class _WithoutKey,
          class _PiecewiseConstruct,
          class _Tuple1,
          class _Tuple2,
          __enable_if_t<is_same<__remove_const_ref_t<_PiecewiseConstruct>, piecewise_construct_t>::value &&
                            __is_tuple_v<_Tuple1> && tuple_size<_Tuple1>::value == 1 &&
                            is_same<__remove_const_ref_t<typename tuple_element<0, _Tuple1>::type>, _KeyT>::value,
                        int> = 0>
_LIBCPP_HIDE_FROM_ABI _Ret __try_key_extraction_impl(
    __priority_tag<1>,
    _WithKey __with_key,
    _WithoutKey,
    _PiecewiseConstruct&& __pc,
    _Tuple1&& __tuple1,
    _Tuple2&& __tuple2) {
  return __with_key(
      std::get<0>(__tuple1),
      std::forward<_PiecewiseConstruct>(__pc),
      std::forward<_Tuple1>(__tuple1),
      std::forward<_Tuple2>(__tuple2));
}
#endif // _LIBCPP_CXX03_LANG

// This function tries extracting the given _KeyT from _Args...
// If it succeeds to extract the key, it calls the `__with_key` function with the extracted key and all of the
// arguments. Otherwise it calls the `__without_key` function with all of the arguments.
//
// Both `__with_key` and `__without_key` must take all arguments by reference.
template <class _KeyT, class _WithKey, class _WithoutKey, class... _Args>
_LIBCPP_HIDE_FROM_ABI decltype(std::declval<_WithoutKey>()(std::declval<_Args>()...))
__try_key_extraction(_WithKey __with_key, _WithoutKey __without_key, _Args&&... __args) {
  using _Ret = decltype(__without_key(std::forward<_Args>(__args)...));
  return std::__try_key_extraction_impl<_KeyT, _Ret>(
      __priority_tag<1>(), __with_key, __without_key, std::forward<_Args>(__args)...);
}

_LIBCPP_END_NAMESPACE_STD

#endif // _LIBCPP___UTILITY_TRY_EXTRACT_KEY_H