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
|