// Copyright 2017 The Chromium Authors. All rights reserved. // Use of this source code is governed by a BSD-style license that can be // found in the LICENSE file. #ifndef BASE_TASK_SCHEDULER_TASK_TRAITS_DETAILS_H_ #define BASE_TASK_SCHEDULER_TASK_TRAITS_DETAILS_H_ #include <type_traits> #include <utility> namespace base { namespace internal { // HasArgOfType<CheckedType, ArgTypes...>::value is true iff a type in ArgTypes // matches CheckedType. template <class...> struct HasArgOfType : std::false_type {}; template <class CheckedType, class FirstArgType, class... ArgTypes> struct HasArgOfType<CheckedType, FirstArgType, ArgTypes...> : std::conditional<std::is_same<CheckedType, FirstArgType>::value, std::true_type, HasArgOfType<CheckedType, ArgTypes...>>::type {}; // When the following call is made: // GetValueFromArgListImpl(CallFirstTag(), GetterType(), args...); // If |args| is empty, the compiler selects the first overload. This overload // returns getter.GetDefaultValue(). If |args| is not empty, the compiler // prefers using the second overload because the type of the first argument // matches exactly. This overload returns getter.GetValueFromArg(first_arg), // where |first_arg| is the first element in |args|. If // getter.GetValueFromArg(first_arg) isn't defined, the compiler uses the third // overload instead. This overload discards the first argument in |args| and // makes a recursive call to GetValueFromArgListImpl() with CallFirstTag() as // first argument. // Tag dispatching. struct CallSecondTag {}; struct CallFirstTag : CallSecondTag {}; // Overload 1: Default value. template <class GetterType> constexpr typename GetterType::ValueType GetValueFromArgListImpl( CallFirstTag, GetterType getter) { return getter.GetDefaultValue(); } // Overload 2: Get value from first argument. Check that no argument in |args| // has the same type as |first_arg|. template <class GetterType, class FirstArgType, class... ArgTypes, class TestGetValueFromArgDefined = decltype(std::declval<GetterType>().GetValueFromArg( std::declval<FirstArgType>()))> constexpr typename GetterType::ValueType GetValueFromArgListImpl( CallFirstTag, GetterType getter, const FirstArgType& first_arg, const ArgTypes&... args) { static_assert(!HasArgOfType<FirstArgType, ArgTypes...>::value, "Multiple arguments of the same type were provided to the " "constructor of TaskTraits."); return getter.GetValueFromArg(first_arg); } // Overload 3: Discard first argument. template <class GetterType, class FirstArgType, class... ArgTypes> constexpr typename GetterType::ValueType GetValueFromArgListImpl( CallSecondTag, GetterType getter, const FirstArgType&, const ArgTypes&... args) { return GetValueFromArgListImpl(CallFirstTag(), getter, args...); } // If there is an argument |arg_of_type| of type Getter::ArgType in |args|, // returns getter.GetValueFromArg(arg_of_type). If there are more than one // argument of type Getter::ArgType in |args|, generates a compile-time error. // Otherwise, returns getter.GetDefaultValue(). // // |getter| must provide: // // ValueType: // The return type of GetValueFromArgListImpl(). // // ArgType: // The type of the argument from which GetValueFromArgListImpl() derives its // return value. // // ValueType GetValueFromArg(ArgType): // Converts an argument of type ArgType into a value returned by // GetValueFromArgListImpl(). // // ValueType GetDefaultValue(): // Returns the value returned by GetValueFromArgListImpl() if none of its // arguments is of type ArgType. template <class GetterType, class... ArgTypes> constexpr typename GetterType::ValueType GetValueFromArgList( GetterType getter, const ArgTypes&... args) { return GetValueFromArgListImpl(CallFirstTag(), getter, args...); } template <typename ArgType> struct BooleanArgGetter { using ValueType = bool; constexpr ValueType GetValueFromArg(ArgType) const { return true; } constexpr ValueType GetDefaultValue() const { return false; } }; template <typename ArgType, ArgType DefaultValue> struct EnumArgGetter { using ValueType = ArgType; constexpr ValueType GetValueFromArg(ArgType arg) const { return arg; } constexpr ValueType GetDefaultValue() const { return DefaultValue; } }; // Allows instantiation of multiple types in one statement. Used to prevent // instantiation of the constructor of TaskTraits with inappropriate argument // types. template <class...> struct InitTypes {}; } // namespace internal } // namespace base #endif // BASE_TASK_SCHEDULER_TASK_TRAITS_DETAILS_H_