The Gaudi Framework  v30r3 (a5ef0a68)
invoke.h
Go to the documentation of this file.
1 #ifndef GAUDIKERNEL_INVOKE_H
2 #define GAUDIKERNEL_INVOKE_H
3 #include <functional>
4 
5 #if __cplusplus > 201402L
6 namespace Gaudi
7 {
8  using std::invoke;
9 }
10 #else
11 
12 #include <type_traits>
13 #include <utility>
14 
15 namespace Gaudi
16 {
17  namespace detail2
18  {
19 
20  // An implemention of std::invoke based on
21  // http://en.cppreference.com/w/cpp/utility/functional/invoke
22  template <class T>
24  };
25  template <class U>
26  struct is_reference_wrapper<std::reference_wrapper<U>> : std::true_type {
27  };
28 
29  template <class Base, class T, class Derived, class... Args>
30  auto INVOKE( T Base::*pmf, Derived&& ref, Args&&... args ) noexcept(
31  noexcept( ( std::forward<Derived>( ref ).*pmf )( std::forward<Args>( args )... ) ) )
32  -> std::enable_if_t<std::is_function<T>::value && std::is_base_of<Base, std::decay_t<Derived>>::value,
33  decltype( ( std::forward<Derived>( ref ).*pmf )( std::forward<Args>( args )... ) )>
34  {
35  return ( std::forward<Derived>( ref ).*pmf )( std::forward<Args>( args )... );
36  }
37 
38  template <class Base, class T, class RefWrap, class... Args>
39  auto INVOKE( T Base::*pmf, RefWrap&& ref,
40  Args&&... args ) noexcept( noexcept( ( ref.get().*pmf )( std::forward<Args>( args )... ) ) )
41  -> std::enable_if_t<std::is_function<T>::value && is_reference_wrapper<std::decay_t<RefWrap>>::value,
42  decltype( ( ref.get().*pmf )( std::forward<Args>( args )... ) )>
43 
44  {
45  return ( ref.get().*pmf )( std::forward<Args>( args )... );
46  }
47 
48  template <class Base, class T, class Pointer, class... Args>
49  auto INVOKE( T Base::*pmf, Pointer&& ptr, Args&&... args ) noexcept(
50  noexcept( ( ( *std::forward<Pointer>( ptr ) ).*pmf )( std::forward<Args>( args )... ) ) )
51  -> std::enable_if_t<std::is_function<T>::value && !is_reference_wrapper<std::decay_t<Pointer>>::value &&
52  !std::is_base_of<Base, std::decay_t<Pointer>>::value,
53  decltype( ( ( *std::forward<Pointer>( ptr ) ).*pmf )( std::forward<Args>( args )... ) )>
54  {
55  return ( ( *std::forward<Pointer>( ptr ) ).*pmf )( std::forward<Args>( args )... );
56  }
57 
58  template <class Base, class T, class Derived>
59  auto INVOKE( T Base::*pmd, Derived&& ref ) noexcept( noexcept( std::forward<Derived>( ref ).*pmd ) )
60  -> std::enable_if_t<!std::is_function<T>::value && std::is_base_of<Base, std::decay_t<Derived>>::value,
61  decltype( std::forward<Derived>( ref ).*pmd )>
62  {
63  return std::forward<Derived>( ref ).*pmd;
64  }
65 
66  template <class Base, class T, class RefWrap>
67  auto INVOKE( T Base::*pmd, RefWrap&& ref ) noexcept( noexcept( ref.get().*pmd ) )
68  -> std::enable_if_t<!std::is_function<T>::value && is_reference_wrapper<std::decay_t<RefWrap>>::value,
69  decltype( ref.get().*pmd )>
70  {
71  return ref.get().*pmd;
72  }
73 
74  template <class Base, class T, class Pointer>
75  auto INVOKE( T Base::*pmd, Pointer&& ptr ) noexcept( noexcept( ( *std::forward<Pointer>( ptr ) ).*pmd ) )
76  -> std::enable_if_t<!std::is_function<T>::value && !is_reference_wrapper<std::decay_t<Pointer>>::value &&
77  !std::is_base_of<Base, std::decay_t<Pointer>>::value,
78  decltype( ( *std::forward<Pointer>( ptr ) ).*pmd )>
79  {
80  return ( *std::forward<Pointer>( ptr ) ).*pmd;
81  }
82 
83  template <class F, class... Args>
84  auto INVOKE( F&& f, Args&&... args ) noexcept( noexcept( std::forward<F>( f )( std::forward<Args>( args )... ) ) )
85  -> std::enable_if_t<!std::is_member_pointer<std::decay_t<F>>::value,
86  decltype( std::forward<F>( f )( std::forward<Args>( args )... ) )>
87  {
88  return std::forward<F>( f )( std::forward<Args>( args )... );
89  }
90  } // namespace detail2
91 
92  template <class F, class... ArgTypes>
93  auto invoke( F&& f, ArgTypes&&... args )
94  // exception specification for QoI
95  noexcept( noexcept( detail2::INVOKE( std::forward<F>( f ), std::forward<ArgTypes>( args )... ) ) )
96  -> decltype( detail2::INVOKE( std::forward<F>( f ), std::forward<ArgTypes>( args )... ) )
97  {
98  return detail2::INVOKE( std::forward<F>( f ), std::forward<ArgTypes>( args )... );
99  }
100 }
101 #endif
102 #endif
STL namespace.
auto INVOKE(T Base::*pmf, Derived &&ref, Args &&...args) noexcept( noexcept((std::forward< Derived >(ref).*pmf)(std::forward< Args >(args)...))) -> std::enable_if_t< std::is_function< T >::value &&std::is_base_of< Base, std::decay_t< Derived >>::value, decltype((std::forward< Derived >(ref).*pmf)(std::forward< Args >(args)...))>
Definition: invoke.h:30
auto invoke(F &&f, ArgTypes &&...args) noexcept(noexcept(detail2::INVOKE(std::forward< F >(f), std::forward< ArgTypes >(args)...))) -> decltype(detail2::INVOKE(std::forward< F >(f), std::forward< ArgTypes >(args)...))
Definition: invoke.h:93
Helper functions to set/get the application return code.
Definition: __init__.py:1