/* * Copyright (c) 2021 The WebRTC project authors. All Rights Reserved. * * Use of this source code is governed by a BSD-style license * that can be found in the LICENSE file in the root of the source * tree. An additional intellectual property rights grant can be found * in the file PATENTS. All contributing project authors may * be found in the AUTHORS file in the root of the source tree. */ // This implementation is borrowed from Chromium. #ifndef RTC_BASE_CONTAINERS_INVOKE_H_ #define RTC_BASE_CONTAINERS_INVOKE_H_ #include #include namespace webrtc { namespace invoke_internal { // Helper struct and alias to deduce the class type from a member function // pointer or member object pointer. template struct member_pointer_class {}; template struct member_pointer_class { using type = ClassT; }; template using member_pointer_class_t = typename member_pointer_class::type; // Utility struct to detect specializations of std::reference_wrapper. template struct is_reference_wrapper : std::false_type {}; template struct is_reference_wrapper> : std::true_type {}; // Small helpers used below in invoke_internal::invoke to make the SFINAE more // concise. template const bool& IsMemFunPtr = std::is_member_function_pointer>::value; template const bool& IsMemObjPtr = std::is_member_object_pointer>::value; template >> const bool& IsMemPtrToBaseOf = std::is_base_of>::value; template const bool& IsRefWrapper = is_reference_wrapper>::value; template using EnableIf = std::enable_if_t; // Invokes a member function pointer on a reference to an object of a suitable // type. Covers bullet 1 of the INVOKE definition. // // Reference: https://wg21.link/func.require#1.1 template && IsMemPtrToBaseOf> = true> constexpr decltype(auto) InvokeImpl(F&& f, T1&& t1, Args&&... args) { return (std::forward(t1).*f)(std::forward(args)...); } // Invokes a member function pointer on a std::reference_wrapper to an object of // a suitable type. Covers bullet 2 of the INVOKE definition. // // Reference: https://wg21.link/func.require#1.2 template && IsRefWrapper> = true> constexpr decltype(auto) InvokeImpl(F&& f, T1&& t1, Args&&... args) { return (t1.get().*f)(std::forward(args)...); } // Invokes a member function pointer on a pointer-like type to an object of a // suitable type. Covers bullet 3 of the INVOKE definition. // // Reference: https://wg21.link/func.require#1.3 template && !IsMemPtrToBaseOf && !IsRefWrapper> = true> constexpr decltype(auto) InvokeImpl(F&& f, T1&& t1, Args&&... args) { return ((*std::forward(t1)).*f)(std::forward(args)...); } // Invokes a member object pointer on a reference to an object of a suitable // type. Covers bullet 4 of the INVOKE definition. // // Reference: https://wg21.link/func.require#1.4 template && IsMemPtrToBaseOf> = true> constexpr decltype(auto) InvokeImpl(F&& f, T1&& t1) { return std::forward(t1).*f; } // Invokes a member object pointer on a std::reference_wrapper to an object of // a suitable type. Covers bullet 5 of the INVOKE definition. // // Reference: https://wg21.link/func.require#1.5 template && IsRefWrapper> = true> constexpr decltype(auto) InvokeImpl(F&& f, T1&& t1) { return t1.get().*f; } // Invokes a member object pointer on a pointer-like type to an object of a // suitable type. Covers bullet 6 of the INVOKE definition. // // Reference: https://wg21.link/func.require#1.6 template && !IsMemPtrToBaseOf && !IsRefWrapper> = true> constexpr decltype(auto) InvokeImpl(F&& f, T1&& t1) { return (*std::forward(t1)).*f; } // Invokes a regular function or function object. Covers bullet 7 of the INVOKE // definition. // // Reference: https://wg21.link/func.require#1.7 template constexpr decltype(auto) InvokeImpl(F&& f, Args&&... args) { return std::forward(f)(std::forward(args)...); } } // namespace invoke_internal // Implementation of C++17's std::invoke. This is not based on implementation // referenced in original std::invoke proposal, but rather a manual // implementation, so that it can be constexpr. // // References: // - https://wg21.link/n4169#implementability // - https://en.cppreference.com/w/cpp/utility/functional/invoke // - https://wg21.link/func.invoke template constexpr decltype(auto) invoke(F&& f, Args&&... args) { return invoke_internal::InvokeImpl(std::forward(f), std::forward(args)...); } } // namespace webrtc #endif // RTC_BASE_CONTAINERS_INVOKE_H_