| //===- FormatVariadicDetails.h - Helpers for FormatVariadic.h ----*- C++-*-===// |
| // |
| // The LLVM Compiler Infrastructure |
| // |
| // This file is distributed under the University of Illinois Open Source |
| // License. See LICENSE.TXT for details. |
| // |
| //===----------------------------------------------------------------------===// |
| |
| #ifndef LLVM_SUPPORT_FORMATVARIADIC_DETAILS_H |
| #define LLVM_SUPPORT_FORMATVARIADIC_DETAILS_H |
| |
| #include "llvm/ADT/StringRef.h" |
| #include "llvm/Support/raw_ostream.h" |
| |
| #include <type_traits> |
| |
| namespace llvm { |
| template <typename T, typename Enable = void> struct format_provider {}; |
| |
| namespace detail { |
| class format_adapter { |
| protected: |
| virtual ~format_adapter() {} |
| |
| public: |
| virtual void format(raw_ostream &S, StringRef Options) = 0; |
| }; |
| |
| template <typename T> class provider_format_adapter : public format_adapter { |
| T Item; |
| |
| public: |
| explicit provider_format_adapter(T &&Item) : Item(std::forward<T>(Item)) {} |
| |
| void format(llvm::raw_ostream &S, StringRef Options) override { |
| format_provider<typename std::decay<T>::type>::format(Item, S, Options); |
| } |
| }; |
| |
| template <typename T> class missing_format_adapter; |
| |
| // Test if format_provider<T> is defined on T and contains a member function |
| // with the signature: |
| // static void format(const T&, raw_stream &, StringRef); |
| // |
| template <class T> class has_FormatProvider { |
| public: |
| using Decayed = typename std::decay<T>::type; |
| typedef void (*Signature_format)(const Decayed &, llvm::raw_ostream &, |
| StringRef); |
| |
| template <typename U> |
| static char test(SameType<Signature_format, &U::format> *); |
| |
| template <typename U> static double test(...); |
| |
| static bool const value = |
| (sizeof(test<llvm::format_provider<Decayed>>(nullptr)) == 1); |
| }; |
| |
| // Simple template that decides whether a type T should use the member-function |
| // based format() invocation. |
| template <typename T> |
| struct uses_format_member |
| : public std::integral_constant< |
| bool, |
| std::is_base_of<format_adapter, |
| typename std::remove_reference<T>::type>::value> {}; |
| |
| // Simple template that decides whether a type T should use the format_provider |
| // based format() invocation. The member function takes priority, so this test |
| // will only be true if there is not ALSO a format member. |
| template <typename T> |
| struct uses_format_provider |
| : public std::integral_constant< |
| bool, !uses_format_member<T>::value && has_FormatProvider<T>::value> { |
| }; |
| |
| // Simple template that decides whether a type T has neither a member-function |
| // nor format_provider based implementation that it can use. Mostly used so |
| // that the compiler spits out a nice diagnostic when a type with no format |
| // implementation can be located. |
| template <typename T> |
| struct uses_missing_provider |
| : public std::integral_constant<bool, |
| !uses_format_member<T>::value && |
| !uses_format_provider<T>::value> {}; |
| |
| template <typename T> |
| typename std::enable_if<uses_format_member<T>::value, T>::type |
| build_format_adapter(T &&Item) { |
| return std::forward<T>(Item); |
| } |
| |
| template <typename T> |
| typename std::enable_if<uses_format_provider<T>::value, |
| provider_format_adapter<T>>::type |
| build_format_adapter(T &&Item) { |
| return provider_format_adapter<T>(std::forward<T>(Item)); |
| } |
| |
| template <typename T> |
| typename std::enable_if<uses_missing_provider<T>::value, |
| missing_format_adapter<T>>::type |
| build_format_adapter(T &&Item) { |
| return missing_format_adapter<T>(); |
| } |
| } |
| } |
| |
| #endif |