|  | //===- llvm/Support/YAMLTraits.h --------------------------------*- C++ -*-===// | 
|  | // | 
|  | //                             The LLVM Linker | 
|  | // | 
|  | // This file is distributed under the University of Illinois Open Source | 
|  | // License. See LICENSE.TXT for details. | 
|  | // | 
|  | //===----------------------------------------------------------------------===// | 
|  |  | 
|  | #ifndef LLVM_SUPPORT_YAMLTRAITS_H | 
|  | #define LLVM_SUPPORT_YAMLTRAITS_H | 
|  |  | 
|  | #include "llvm/ADT/Optional.h" | 
|  | #include "llvm/ADT/SmallVector.h" | 
|  | #include "llvm/ADT/StringMap.h" | 
|  | #include "llvm/ADT/StringRef.h" | 
|  | #include "llvm/ADT/Twine.h" | 
|  | #include "llvm/Support/AlignOf.h" | 
|  | #include "llvm/Support/Allocator.h" | 
|  | #include "llvm/Support/Endian.h" | 
|  | #include "llvm/Support/Regex.h" | 
|  | #include "llvm/Support/SourceMgr.h" | 
|  | #include "llvm/Support/YAMLParser.h" | 
|  | #include "llvm/Support/raw_ostream.h" | 
|  | #include <cassert> | 
|  | #include <cctype> | 
|  | #include <cstddef> | 
|  | #include <cstdint> | 
|  | #include <memory> | 
|  | #include <new> | 
|  | #include <string> | 
|  | #include <system_error> | 
|  | #include <type_traits> | 
|  | #include <vector> | 
|  |  | 
|  | namespace llvm { | 
|  | namespace yaml { | 
|  |  | 
|  | struct EmptyContext {}; | 
|  |  | 
|  | /// This class should be specialized by any type that needs to be converted | 
|  | /// to/from a YAML mapping.  For example: | 
|  | /// | 
|  | ///     struct MappingTraits<MyStruct> { | 
|  | ///       static void mapping(IO &io, MyStruct &s) { | 
|  | ///         io.mapRequired("name", s.name); | 
|  | ///         io.mapRequired("size", s.size); | 
|  | ///         io.mapOptional("age",  s.age); | 
|  | ///       } | 
|  | ///     }; | 
|  | template<class T> | 
|  | struct MappingTraits { | 
|  | // Must provide: | 
|  | // static void mapping(IO &io, T &fields); | 
|  | // Optionally may provide: | 
|  | // static StringRef validate(IO &io, T &fields); | 
|  | // | 
|  | // The optional flow flag will cause generated YAML to use a flow mapping | 
|  | // (e.g. { a: 0, b: 1 }): | 
|  | // static const bool flow = true; | 
|  | }; | 
|  |  | 
|  | /// This class is similar to MappingTraits<T> but allows you to pass in | 
|  | /// additional context for each map operation.  For example: | 
|  | /// | 
|  | ///     struct MappingContextTraits<MyStruct, MyContext> { | 
|  | ///       static void mapping(IO &io, MyStruct &s, MyContext &c) { | 
|  | ///         io.mapRequired("name", s.name); | 
|  | ///         io.mapRequired("size", s.size); | 
|  | ///         io.mapOptional("age",  s.age); | 
|  | ///         ++c.TimesMapped; | 
|  | ///       } | 
|  | ///     }; | 
|  | template <class T, class Context> struct MappingContextTraits { | 
|  | // Must provide: | 
|  | // static void mapping(IO &io, T &fields, Context &Ctx); | 
|  | // Optionally may provide: | 
|  | // static StringRef validate(IO &io, T &fields, Context &Ctx); | 
|  | // | 
|  | // The optional flow flag will cause generated YAML to use a flow mapping | 
|  | // (e.g. { a: 0, b: 1 }): | 
|  | // static const bool flow = true; | 
|  | }; | 
|  |  | 
|  | /// This class should be specialized by any integral type that converts | 
|  | /// to/from a YAML scalar where there is a one-to-one mapping between | 
|  | /// in-memory values and a string in YAML.  For example: | 
|  | /// | 
|  | ///     struct ScalarEnumerationTraits<Colors> { | 
|  | ///         static void enumeration(IO &io, Colors &value) { | 
|  | ///           io.enumCase(value, "red",   cRed); | 
|  | ///           io.enumCase(value, "blue",  cBlue); | 
|  | ///           io.enumCase(value, "green", cGreen); | 
|  | ///         } | 
|  | ///       }; | 
|  | template<typename T> | 
|  | struct ScalarEnumerationTraits { | 
|  | // Must provide: | 
|  | // static void enumeration(IO &io, T &value); | 
|  | }; | 
|  |  | 
|  | /// This class should be specialized by any integer type that is a union | 
|  | /// of bit values and the YAML representation is a flow sequence of | 
|  | /// strings.  For example: | 
|  | /// | 
|  | ///      struct ScalarBitSetTraits<MyFlags> { | 
|  | ///        static void bitset(IO &io, MyFlags &value) { | 
|  | ///          io.bitSetCase(value, "big",   flagBig); | 
|  | ///          io.bitSetCase(value, "flat",  flagFlat); | 
|  | ///          io.bitSetCase(value, "round", flagRound); | 
|  | ///        } | 
|  | ///      }; | 
|  | template<typename T> | 
|  | struct ScalarBitSetTraits { | 
|  | // Must provide: | 
|  | // static void bitset(IO &io, T &value); | 
|  | }; | 
|  |  | 
|  | /// This class should be specialized by type that requires custom conversion | 
|  | /// to/from a yaml scalar.  For example: | 
|  | /// | 
|  | ///    template<> | 
|  | ///    struct ScalarTraits<MyType> { | 
|  | ///      static void output(const MyType &val, void*, llvm::raw_ostream &out) { | 
|  | ///        // stream out custom formatting | 
|  | ///        out << llvm::format("%x", val); | 
|  | ///      } | 
|  | ///      static StringRef input(StringRef scalar, void*, MyType &value) { | 
|  | ///        // parse scalar and set `value` | 
|  | ///        // return empty string on success, or error string | 
|  | ///        return StringRef(); | 
|  | ///      } | 
|  | ///      static bool mustQuote(StringRef) { return true; } | 
|  | ///    }; | 
|  | template<typename T> | 
|  | struct ScalarTraits { | 
|  | // Must provide: | 
|  | // | 
|  | // Function to write the value as a string: | 
|  | //static void output(const T &value, void *ctxt, llvm::raw_ostream &out); | 
|  | // | 
|  | // Function to convert a string to a value.  Returns the empty | 
|  | // StringRef on success or an error string if string is malformed: | 
|  | //static StringRef input(StringRef scalar, void *ctxt, T &value); | 
|  | // | 
|  | // Function to determine if the value should be quoted. | 
|  | //static bool mustQuote(StringRef); | 
|  | }; | 
|  |  | 
|  | /// This class should be specialized by type that requires custom conversion | 
|  | /// to/from a YAML literal block scalar. For example: | 
|  | /// | 
|  | ///    template <> | 
|  | ///    struct BlockScalarTraits<MyType> { | 
|  | ///      static void output(const MyType &Value, void*, llvm::raw_ostream &Out) | 
|  | ///      { | 
|  | ///        // stream out custom formatting | 
|  | ///        Out << Val; | 
|  | ///      } | 
|  | ///      static StringRef input(StringRef Scalar, void*, MyType &Value) { | 
|  | ///        // parse scalar and set `value` | 
|  | ///        // return empty string on success, or error string | 
|  | ///        return StringRef(); | 
|  | ///      } | 
|  | ///    }; | 
|  | template <typename T> | 
|  | struct BlockScalarTraits { | 
|  | // Must provide: | 
|  | // | 
|  | // Function to write the value as a string: | 
|  | // static void output(const T &Value, void *ctx, llvm::raw_ostream &Out); | 
|  | // | 
|  | // Function to convert a string to a value.  Returns the empty | 
|  | // StringRef on success or an error string if string is malformed: | 
|  | // static StringRef input(StringRef Scalar, void *ctxt, T &Value); | 
|  | }; | 
|  |  | 
|  | /// This class should be specialized by any type that needs to be converted | 
|  | /// to/from a YAML sequence.  For example: | 
|  | /// | 
|  | ///    template<> | 
|  | ///    struct SequenceTraits< std::vector<MyType>> { | 
|  | ///      static size_t size(IO &io, std::vector<MyType> &seq) { | 
|  | ///        return seq.size(); | 
|  | ///      } | 
|  | ///      static MyType& element(IO &, std::vector<MyType> &seq, size_t index) { | 
|  | ///        if ( index >= seq.size() ) | 
|  | ///          seq.resize(index+1); | 
|  | ///        return seq[index]; | 
|  | ///      } | 
|  | ///    }; | 
|  | template<typename T> | 
|  | struct SequenceTraits { | 
|  | // Must provide: | 
|  | // static size_t size(IO &io, T &seq); | 
|  | // static T::value_type& element(IO &io, T &seq, size_t index); | 
|  | // | 
|  | // The following is option and will cause generated YAML to use | 
|  | // a flow sequence (e.g. [a,b,c]). | 
|  | // static const bool flow = true; | 
|  | }; | 
|  |  | 
|  | /// This class should be specialized by any type that needs to be converted | 
|  | /// to/from a list of YAML documents. | 
|  | template<typename T> | 
|  | struct DocumentListTraits { | 
|  | // Must provide: | 
|  | // static size_t size(IO &io, T &seq); | 
|  | // static T::value_type& element(IO &io, T &seq, size_t index); | 
|  | }; | 
|  |  | 
|  | // Only used for better diagnostics of missing traits | 
|  | template <typename T> | 
|  | struct MissingTrait; | 
|  |  | 
|  | // Test if ScalarEnumerationTraits<T> is defined on type T. | 
|  | template <class T> | 
|  | struct has_ScalarEnumerationTraits | 
|  | { | 
|  | typedef void (*Signature_enumeration)(class IO&, T&); | 
|  |  | 
|  | template <typename U> | 
|  | static char test(SameType<Signature_enumeration, &U::enumeration>*); | 
|  |  | 
|  | template <typename U> | 
|  | static double test(...); | 
|  |  | 
|  | public: | 
|  | static bool const value = | 
|  | (sizeof(test<ScalarEnumerationTraits<T>>(nullptr)) == 1); | 
|  | }; | 
|  |  | 
|  | // Test if ScalarBitSetTraits<T> is defined on type T. | 
|  | template <class T> | 
|  | struct has_ScalarBitSetTraits | 
|  | { | 
|  | typedef void (*Signature_bitset)(class IO&, T&); | 
|  |  | 
|  | template <typename U> | 
|  | static char test(SameType<Signature_bitset, &U::bitset>*); | 
|  |  | 
|  | template <typename U> | 
|  | static double test(...); | 
|  |  | 
|  | public: | 
|  | static bool const value = (sizeof(test<ScalarBitSetTraits<T>>(nullptr)) == 1); | 
|  | }; | 
|  |  | 
|  | // Test if ScalarTraits<T> is defined on type T. | 
|  | template <class T> | 
|  | struct has_ScalarTraits | 
|  | { | 
|  | typedef StringRef (*Signature_input)(StringRef, void*, T&); | 
|  | typedef void (*Signature_output)(const T&, void*, llvm::raw_ostream&); | 
|  | typedef bool (*Signature_mustQuote)(StringRef); | 
|  |  | 
|  | template <typename U> | 
|  | static char test(SameType<Signature_input, &U::input> *, | 
|  | SameType<Signature_output, &U::output> *, | 
|  | SameType<Signature_mustQuote, &U::mustQuote> *); | 
|  |  | 
|  | template <typename U> | 
|  | static double test(...); | 
|  |  | 
|  | public: | 
|  | static bool const value = | 
|  | (sizeof(test<ScalarTraits<T>>(nullptr, nullptr, nullptr)) == 1); | 
|  | }; | 
|  |  | 
|  | // Test if BlockScalarTraits<T> is defined on type T. | 
|  | template <class T> | 
|  | struct has_BlockScalarTraits | 
|  | { | 
|  | typedef StringRef (*Signature_input)(StringRef, void *, T &); | 
|  | typedef void (*Signature_output)(const T &, void *, llvm::raw_ostream &); | 
|  |  | 
|  | template <typename U> | 
|  | static char test(SameType<Signature_input, &U::input> *, | 
|  | SameType<Signature_output, &U::output> *); | 
|  |  | 
|  | template <typename U> | 
|  | static double test(...); | 
|  |  | 
|  | public: | 
|  | static bool const value = | 
|  | (sizeof(test<BlockScalarTraits<T>>(nullptr, nullptr)) == 1); | 
|  | }; | 
|  |  | 
|  | // Test if MappingContextTraits<T> is defined on type T. | 
|  | template <class T, class Context> struct has_MappingTraits { | 
|  | typedef void (*Signature_mapping)(class IO &, T &, Context &); | 
|  |  | 
|  | template <typename U> | 
|  | static char test(SameType<Signature_mapping, &U::mapping>*); | 
|  |  | 
|  | template <typename U> | 
|  | static double test(...); | 
|  |  | 
|  | public: | 
|  | static bool const value = | 
|  | (sizeof(test<MappingContextTraits<T, Context>>(nullptr)) == 1); | 
|  | }; | 
|  |  | 
|  | // Test if MappingTraits<T> is defined on type T. | 
|  | template <class T> struct has_MappingTraits<T, EmptyContext> { | 
|  | typedef void (*Signature_mapping)(class IO &, T &); | 
|  |  | 
|  | template <typename U> | 
|  | static char test(SameType<Signature_mapping, &U::mapping> *); | 
|  |  | 
|  | template <typename U> static double test(...); | 
|  |  | 
|  | public: | 
|  | static bool const value = (sizeof(test<MappingTraits<T>>(nullptr)) == 1); | 
|  | }; | 
|  |  | 
|  | // Test if MappingContextTraits<T>::validate() is defined on type T. | 
|  | template <class T, class Context> struct has_MappingValidateTraits { | 
|  | typedef StringRef (*Signature_validate)(class IO &, T &, Context &); | 
|  |  | 
|  | template <typename U> | 
|  | static char test(SameType<Signature_validate, &U::validate>*); | 
|  |  | 
|  | template <typename U> | 
|  | static double test(...); | 
|  |  | 
|  | public: | 
|  | static bool const value = | 
|  | (sizeof(test<MappingContextTraits<T, Context>>(nullptr)) == 1); | 
|  | }; | 
|  |  | 
|  | // Test if MappingTraits<T>::validate() is defined on type T. | 
|  | template <class T> struct has_MappingValidateTraits<T, EmptyContext> { | 
|  | typedef StringRef (*Signature_validate)(class IO &, T &); | 
|  |  | 
|  | template <typename U> | 
|  | static char test(SameType<Signature_validate, &U::validate> *); | 
|  |  | 
|  | template <typename U> static double test(...); | 
|  |  | 
|  | public: | 
|  | static bool const value = (sizeof(test<MappingTraits<T>>(nullptr)) == 1); | 
|  | }; | 
|  |  | 
|  | // Test if SequenceTraits<T> is defined on type T. | 
|  | template <class T> | 
|  | struct has_SequenceMethodTraits | 
|  | { | 
|  | typedef size_t (*Signature_size)(class IO&, T&); | 
|  |  | 
|  | template <typename U> | 
|  | static char test(SameType<Signature_size, &U::size>*); | 
|  |  | 
|  | template <typename U> | 
|  | static double test(...); | 
|  |  | 
|  | public: | 
|  | static bool const value =  (sizeof(test<SequenceTraits<T>>(nullptr)) == 1); | 
|  | }; | 
|  |  | 
|  | // has_FlowTraits<int> will cause an error with some compilers because | 
|  | // it subclasses int.  Using this wrapper only instantiates the | 
|  | // real has_FlowTraits only if the template type is a class. | 
|  | template <typename T, bool Enabled = std::is_class<T>::value> | 
|  | class has_FlowTraits | 
|  | { | 
|  | public: | 
|  | static const bool value = false; | 
|  | }; | 
|  |  | 
|  | // Some older gcc compilers don't support straight forward tests | 
|  | // for members, so test for ambiguity cause by the base and derived | 
|  | // classes both defining the member. | 
|  | template <class T> | 
|  | struct has_FlowTraits<T, true> | 
|  | { | 
|  | struct Fallback { bool flow; }; | 
|  | struct Derived : T, Fallback { }; | 
|  |  | 
|  | template<typename C> | 
|  | static char (&f(SameType<bool Fallback::*, &C::flow>*))[1]; | 
|  |  | 
|  | template<typename C> | 
|  | static char (&f(...))[2]; | 
|  |  | 
|  | public: | 
|  | static bool const value = sizeof(f<Derived>(nullptr)) == 2; | 
|  | }; | 
|  |  | 
|  | // Test if SequenceTraits<T> is defined on type T | 
|  | template<typename T> | 
|  | struct has_SequenceTraits : public std::integral_constant<bool, | 
|  | has_SequenceMethodTraits<T>::value > { }; | 
|  |  | 
|  | // Test if DocumentListTraits<T> is defined on type T | 
|  | template <class T> | 
|  | struct has_DocumentListTraits | 
|  | { | 
|  | typedef size_t (*Signature_size)(class IO&, T&); | 
|  |  | 
|  | template <typename U> | 
|  | static char test(SameType<Signature_size, &U::size>*); | 
|  |  | 
|  | template <typename U> | 
|  | static double test(...); | 
|  |  | 
|  | public: | 
|  | static bool const value = (sizeof(test<DocumentListTraits<T>>(nullptr))==1); | 
|  | }; | 
|  |  | 
|  | inline bool isNumber(StringRef S) { | 
|  | static const char OctalChars[] = "01234567"; | 
|  | if (S.startswith("0") && | 
|  | S.drop_front().find_first_not_of(OctalChars) == StringRef::npos) | 
|  | return true; | 
|  |  | 
|  | if (S.startswith("0o") && | 
|  | S.drop_front(2).find_first_not_of(OctalChars) == StringRef::npos) | 
|  | return true; | 
|  |  | 
|  | static const char HexChars[] = "0123456789abcdefABCDEF"; | 
|  | if (S.startswith("0x") && | 
|  | S.drop_front(2).find_first_not_of(HexChars) == StringRef::npos) | 
|  | return true; | 
|  |  | 
|  | static const char DecChars[] = "0123456789"; | 
|  | if (S.find_first_not_of(DecChars) == StringRef::npos) | 
|  | return true; | 
|  |  | 
|  | if (S.equals(".inf") || S.equals(".Inf") || S.equals(".INF")) | 
|  | return true; | 
|  |  | 
|  | Regex FloatMatcher("^(\\.[0-9]+|[0-9]+(\\.[0-9]*)?)([eE][-+]?[0-9]+)?$"); | 
|  | if (FloatMatcher.match(S)) | 
|  | return true; | 
|  |  | 
|  | return false; | 
|  | } | 
|  |  | 
|  | inline bool isNumeric(StringRef S) { | 
|  | if ((S.front() == '-' || S.front() == '+') && isNumber(S.drop_front())) | 
|  | return true; | 
|  |  | 
|  | if (isNumber(S)) | 
|  | return true; | 
|  |  | 
|  | if (S.equals(".nan") || S.equals(".NaN") || S.equals(".NAN")) | 
|  | return true; | 
|  |  | 
|  | return false; | 
|  | } | 
|  |  | 
|  | inline bool isNull(StringRef S) { | 
|  | return S.equals("null") || S.equals("Null") || S.equals("NULL") || | 
|  | S.equals("~"); | 
|  | } | 
|  |  | 
|  | inline bool isBool(StringRef S) { | 
|  | return S.equals("true") || S.equals("True") || S.equals("TRUE") || | 
|  | S.equals("false") || S.equals("False") || S.equals("FALSE"); | 
|  | } | 
|  |  | 
|  | inline bool needsQuotes(StringRef S) { | 
|  | if (S.empty()) | 
|  | return true; | 
|  | if (isspace(S.front()) || isspace(S.back())) | 
|  | return true; | 
|  | if (S.front() == ',') | 
|  | return true; | 
|  |  | 
|  | static const char ScalarSafeChars[] = | 
|  | "abcdefghijklmnopqrstuvwxyz" | 
|  | "ABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789_-/^., \t"; | 
|  | if (S.find_first_not_of(ScalarSafeChars) != StringRef::npos) | 
|  | return true; | 
|  |  | 
|  | if (isNull(S)) | 
|  | return true; | 
|  | if (isBool(S)) | 
|  | return true; | 
|  | if (isNumeric(S)) | 
|  | return true; | 
|  |  | 
|  | return false; | 
|  | } | 
|  |  | 
|  | template <typename T, typename Context> | 
|  | struct missingTraits | 
|  | : public std::integral_constant<bool, | 
|  | !has_ScalarEnumerationTraits<T>::value && | 
|  | !has_ScalarBitSetTraits<T>::value && | 
|  | !has_ScalarTraits<T>::value && | 
|  | !has_BlockScalarTraits<T>::value && | 
|  | !has_MappingTraits<T, Context>::value && | 
|  | !has_SequenceTraits<T>::value && | 
|  | !has_DocumentListTraits<T>::value> {}; | 
|  |  | 
|  | template <typename T, typename Context> | 
|  | struct validatedMappingTraits | 
|  | : public std::integral_constant< | 
|  | bool, has_MappingTraits<T, Context>::value && | 
|  | has_MappingValidateTraits<T, Context>::value> {}; | 
|  |  | 
|  | template <typename T, typename Context> | 
|  | struct unvalidatedMappingTraits | 
|  | : public std::integral_constant< | 
|  | bool, has_MappingTraits<T, Context>::value && | 
|  | !has_MappingValidateTraits<T, Context>::value> {}; | 
|  |  | 
|  | // Base class for Input and Output. | 
|  | class IO { | 
|  | public: | 
|  | IO(void *Ctxt=nullptr); | 
|  | virtual ~IO(); | 
|  |  | 
|  | virtual bool outputting() = 0; | 
|  |  | 
|  | virtual unsigned beginSequence() = 0; | 
|  | virtual bool preflightElement(unsigned, void *&) = 0; | 
|  | virtual void postflightElement(void*) = 0; | 
|  | virtual void endSequence() = 0; | 
|  | virtual bool canElideEmptySequence() = 0; | 
|  |  | 
|  | virtual unsigned beginFlowSequence() = 0; | 
|  | virtual bool preflightFlowElement(unsigned, void *&) = 0; | 
|  | virtual void postflightFlowElement(void*) = 0; | 
|  | virtual void endFlowSequence() = 0; | 
|  |  | 
|  | virtual bool mapTag(StringRef Tag, bool Default=false) = 0; | 
|  | virtual void beginMapping() = 0; | 
|  | virtual void endMapping() = 0; | 
|  | virtual bool preflightKey(const char*, bool, bool, bool &, void *&) = 0; | 
|  | virtual void postflightKey(void*) = 0; | 
|  |  | 
|  | virtual void beginFlowMapping() = 0; | 
|  | virtual void endFlowMapping() = 0; | 
|  |  | 
|  | virtual void beginEnumScalar() = 0; | 
|  | virtual bool matchEnumScalar(const char*, bool) = 0; | 
|  | virtual bool matchEnumFallback() = 0; | 
|  | virtual void endEnumScalar() = 0; | 
|  |  | 
|  | virtual bool beginBitSetScalar(bool &) = 0; | 
|  | virtual bool bitSetMatch(const char*, bool) = 0; | 
|  | virtual void endBitSetScalar() = 0; | 
|  |  | 
|  | virtual void scalarString(StringRef &, bool) = 0; | 
|  | virtual void blockScalarString(StringRef &) = 0; | 
|  |  | 
|  | virtual void setError(const Twine &) = 0; | 
|  |  | 
|  | template <typename T> | 
|  | void enumCase(T &Val, const char* Str, const T ConstVal) { | 
|  | if ( matchEnumScalar(Str, outputting() && Val == ConstVal) ) { | 
|  | Val = ConstVal; | 
|  | } | 
|  | } | 
|  |  | 
|  | // allow anonymous enum values to be used with LLVM_YAML_STRONG_TYPEDEF | 
|  | template <typename T> | 
|  | void enumCase(T &Val, const char* Str, const uint32_t ConstVal) { | 
|  | if ( matchEnumScalar(Str, outputting() && Val == static_cast<T>(ConstVal)) ) { | 
|  | Val = ConstVal; | 
|  | } | 
|  | } | 
|  |  | 
|  | template <typename FBT, typename T> | 
|  | void enumFallback(T &Val) { | 
|  | if (matchEnumFallback()) { | 
|  | EmptyContext Context; | 
|  | // FIXME: Force integral conversion to allow strong typedefs to convert. | 
|  | FBT Res = static_cast<typename FBT::BaseType>(Val); | 
|  | yamlize(*this, Res, true, Context); | 
|  | Val = static_cast<T>(static_cast<typename FBT::BaseType>(Res)); | 
|  | } | 
|  | } | 
|  |  | 
|  | template <typename T> | 
|  | void bitSetCase(T &Val, const char* Str, const T ConstVal) { | 
|  | if ( bitSetMatch(Str, outputting() && (Val & ConstVal) == ConstVal) ) { | 
|  | Val = Val | ConstVal; | 
|  | } | 
|  | } | 
|  |  | 
|  | // allow anonymous enum values to be used with LLVM_YAML_STRONG_TYPEDEF | 
|  | template <typename T> | 
|  | void bitSetCase(T &Val, const char* Str, const uint32_t ConstVal) { | 
|  | if ( bitSetMatch(Str, outputting() && (Val & ConstVal) == ConstVal) ) { | 
|  | Val = Val | ConstVal; | 
|  | } | 
|  | } | 
|  |  | 
|  | template <typename T> | 
|  | void maskedBitSetCase(T &Val, const char *Str, T ConstVal, T Mask) { | 
|  | if (bitSetMatch(Str, outputting() && (Val & Mask) == ConstVal)) | 
|  | Val = Val | ConstVal; | 
|  | } | 
|  |  | 
|  | template <typename T> | 
|  | void maskedBitSetCase(T &Val, const char *Str, uint32_t ConstVal, | 
|  | uint32_t Mask) { | 
|  | if (bitSetMatch(Str, outputting() && (Val & Mask) == ConstVal)) | 
|  | Val = Val | ConstVal; | 
|  | } | 
|  |  | 
|  | void *getContext(); | 
|  | void setContext(void *); | 
|  |  | 
|  | template <typename T> void mapRequired(const char *Key, T &Val) { | 
|  | EmptyContext Ctx; | 
|  | this->processKey(Key, Val, true, Ctx); | 
|  | } | 
|  | template <typename T, typename Context> | 
|  | void mapRequired(const char *Key, T &Val, Context &Ctx) { | 
|  | this->processKey(Key, Val, true, Ctx); | 
|  | } | 
|  |  | 
|  | template <typename T> void mapOptional(const char *Key, T &Val) { | 
|  | EmptyContext Ctx; | 
|  | mapOptionalWithContext(Key, Val, Ctx); | 
|  | } | 
|  |  | 
|  | template <typename T> | 
|  | void mapOptional(const char *Key, T &Val, const T &Default) { | 
|  | EmptyContext Ctx; | 
|  | mapOptionalWithContext(Key, Val, Default, Ctx); | 
|  | } | 
|  |  | 
|  | template <typename T, typename Context> | 
|  | typename std::enable_if<has_SequenceTraits<T>::value, void>::type | 
|  | mapOptionalWithContext(const char *Key, T &Val, Context &Ctx) { | 
|  | // omit key/value instead of outputting empty sequence | 
|  | if (this->canElideEmptySequence() && !(Val.begin() != Val.end())) | 
|  | return; | 
|  | this->processKey(Key, Val, false, Ctx); | 
|  | } | 
|  |  | 
|  | template <typename T, typename Context> | 
|  | void mapOptionalWithContext(const char *Key, Optional<T> &Val, Context &Ctx) { | 
|  | this->processKeyWithDefault(Key, Val, Optional<T>(), /*Required=*/false, | 
|  | Ctx); | 
|  | } | 
|  |  | 
|  | template <typename T, typename Context> | 
|  | typename std::enable_if<!has_SequenceTraits<T>::value, void>::type | 
|  | mapOptionalWithContext(const char *Key, T &Val, Context &Ctx) { | 
|  | this->processKey(Key, Val, false, Ctx); | 
|  | } | 
|  |  | 
|  | template <typename T, typename Context> | 
|  | void mapOptionalWithContext(const char *Key, T &Val, const T &Default, | 
|  | Context &Ctx) { | 
|  | this->processKeyWithDefault(Key, Val, Default, false, Ctx); | 
|  | } | 
|  |  | 
|  | private: | 
|  | template <typename T, typename Context> | 
|  | void processKeyWithDefault(const char *Key, Optional<T> &Val, | 
|  | const Optional<T> &DefaultValue, bool Required, | 
|  | Context &Ctx) { | 
|  | assert(DefaultValue.hasValue() == false && | 
|  | "Optional<T> shouldn't have a value!"); | 
|  | void *SaveInfo; | 
|  | bool UseDefault; | 
|  | const bool sameAsDefault = outputting() && !Val.hasValue(); | 
|  | if (!outputting() && !Val.hasValue()) | 
|  | Val = T(); | 
|  | if (this->preflightKey(Key, Required, sameAsDefault, UseDefault, | 
|  | SaveInfo)) { | 
|  | yamlize(*this, Val.getValue(), Required, Ctx); | 
|  | this->postflightKey(SaveInfo); | 
|  | } else { | 
|  | if (UseDefault) | 
|  | Val = DefaultValue; | 
|  | } | 
|  | } | 
|  |  | 
|  | template <typename T, typename Context> | 
|  | void processKeyWithDefault(const char *Key, T &Val, const T &DefaultValue, | 
|  | bool Required, Context &Ctx) { | 
|  | void *SaveInfo; | 
|  | bool UseDefault; | 
|  | const bool sameAsDefault = outputting() && Val == DefaultValue; | 
|  | if ( this->preflightKey(Key, Required, sameAsDefault, UseDefault, | 
|  | SaveInfo) ) { | 
|  | yamlize(*this, Val, Required, Ctx); | 
|  | this->postflightKey(SaveInfo); | 
|  | } | 
|  | else { | 
|  | if ( UseDefault ) | 
|  | Val = DefaultValue; | 
|  | } | 
|  | } | 
|  |  | 
|  | template <typename T, typename Context> | 
|  | void processKey(const char *Key, T &Val, bool Required, Context &Ctx) { | 
|  | void *SaveInfo; | 
|  | bool UseDefault; | 
|  | if ( this->preflightKey(Key, Required, false, UseDefault, SaveInfo) ) { | 
|  | yamlize(*this, Val, Required, Ctx); | 
|  | this->postflightKey(SaveInfo); | 
|  | } | 
|  | } | 
|  |  | 
|  | private: | 
|  | void  *Ctxt; | 
|  | }; | 
|  |  | 
|  | namespace detail { | 
|  |  | 
|  | template <typename T, typename Context> | 
|  | void doMapping(IO &io, T &Val, Context &Ctx) { | 
|  | MappingContextTraits<T, Context>::mapping(io, Val, Ctx); | 
|  | } | 
|  |  | 
|  | template <typename T> void doMapping(IO &io, T &Val, EmptyContext &Ctx) { | 
|  | MappingTraits<T>::mapping(io, Val); | 
|  | } | 
|  |  | 
|  | } // end namespace detail | 
|  |  | 
|  | template <typename T> | 
|  | typename std::enable_if<has_ScalarEnumerationTraits<T>::value, void>::type | 
|  | yamlize(IO &io, T &Val, bool, EmptyContext &Ctx) { | 
|  | io.beginEnumScalar(); | 
|  | ScalarEnumerationTraits<T>::enumeration(io, Val); | 
|  | io.endEnumScalar(); | 
|  | } | 
|  |  | 
|  | template <typename T> | 
|  | typename std::enable_if<has_ScalarBitSetTraits<T>::value, void>::type | 
|  | yamlize(IO &io, T &Val, bool, EmptyContext &Ctx) { | 
|  | bool DoClear; | 
|  | if ( io.beginBitSetScalar(DoClear) ) { | 
|  | if ( DoClear ) | 
|  | Val = static_cast<T>(0); | 
|  | ScalarBitSetTraits<T>::bitset(io, Val); | 
|  | io.endBitSetScalar(); | 
|  | } | 
|  | } | 
|  |  | 
|  | template <typename T> | 
|  | typename std::enable_if<has_ScalarTraits<T>::value, void>::type | 
|  | yamlize(IO &io, T &Val, bool, EmptyContext &Ctx) { | 
|  | if ( io.outputting() ) { | 
|  | std::string Storage; | 
|  | llvm::raw_string_ostream Buffer(Storage); | 
|  | ScalarTraits<T>::output(Val, io.getContext(), Buffer); | 
|  | StringRef Str = Buffer.str(); | 
|  | io.scalarString(Str, ScalarTraits<T>::mustQuote(Str)); | 
|  | } | 
|  | else { | 
|  | StringRef Str; | 
|  | io.scalarString(Str, ScalarTraits<T>::mustQuote(Str)); | 
|  | StringRef Result = ScalarTraits<T>::input(Str, io.getContext(), Val); | 
|  | if ( !Result.empty() ) { | 
|  | io.setError(llvm::Twine(Result)); | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | template <typename T> | 
|  | typename std::enable_if<has_BlockScalarTraits<T>::value, void>::type | 
|  | yamlize(IO &YamlIO, T &Val, bool, EmptyContext &Ctx) { | 
|  | if (YamlIO.outputting()) { | 
|  | std::string Storage; | 
|  | llvm::raw_string_ostream Buffer(Storage); | 
|  | BlockScalarTraits<T>::output(Val, YamlIO.getContext(), Buffer); | 
|  | StringRef Str = Buffer.str(); | 
|  | YamlIO.blockScalarString(Str); | 
|  | } else { | 
|  | StringRef Str; | 
|  | YamlIO.blockScalarString(Str); | 
|  | StringRef Result = | 
|  | BlockScalarTraits<T>::input(Str, YamlIO.getContext(), Val); | 
|  | if (!Result.empty()) | 
|  | YamlIO.setError(llvm::Twine(Result)); | 
|  | } | 
|  | } | 
|  |  | 
|  | template <typename T, typename Context> | 
|  | typename std::enable_if<validatedMappingTraits<T, Context>::value, void>::type | 
|  | yamlize(IO &io, T &Val, bool, Context &Ctx) { | 
|  | if (has_FlowTraits<MappingTraits<T>>::value) | 
|  | io.beginFlowMapping(); | 
|  | else | 
|  | io.beginMapping(); | 
|  | if (io.outputting()) { | 
|  | StringRef Err = MappingTraits<T>::validate(io, Val); | 
|  | if (!Err.empty()) { | 
|  | llvm::errs() << Err << "\n"; | 
|  | assert(Err.empty() && "invalid struct trying to be written as yaml"); | 
|  | } | 
|  | } | 
|  | detail::doMapping(io, Val, Ctx); | 
|  | if (!io.outputting()) { | 
|  | StringRef Err = MappingTraits<T>::validate(io, Val); | 
|  | if (!Err.empty()) | 
|  | io.setError(Err); | 
|  | } | 
|  | if (has_FlowTraits<MappingTraits<T>>::value) | 
|  | io.endFlowMapping(); | 
|  | else | 
|  | io.endMapping(); | 
|  | } | 
|  |  | 
|  | template <typename T, typename Context> | 
|  | typename std::enable_if<unvalidatedMappingTraits<T, Context>::value, void>::type | 
|  | yamlize(IO &io, T &Val, bool, Context &Ctx) { | 
|  | if (has_FlowTraits<MappingTraits<T>>::value) { | 
|  | io.beginFlowMapping(); | 
|  | detail::doMapping(io, Val, Ctx); | 
|  | io.endFlowMapping(); | 
|  | } else { | 
|  | io.beginMapping(); | 
|  | detail::doMapping(io, Val, Ctx); | 
|  | io.endMapping(); | 
|  | } | 
|  | } | 
|  |  | 
|  | template <typename T> | 
|  | typename std::enable_if<missingTraits<T, EmptyContext>::value, void>::type | 
|  | yamlize(IO &io, T &Val, bool, EmptyContext &Ctx) { | 
|  | char missing_yaml_trait_for_type[sizeof(MissingTrait<T>)]; | 
|  | } | 
|  |  | 
|  | template <typename T, typename Context> | 
|  | typename std::enable_if<has_SequenceTraits<T>::value, void>::type | 
|  | yamlize(IO &io, T &Seq, bool, Context &Ctx) { | 
|  | if ( has_FlowTraits< SequenceTraits<T> >::value ) { | 
|  | unsigned incnt = io.beginFlowSequence(); | 
|  | unsigned count = io.outputting() ? SequenceTraits<T>::size(io, Seq) : incnt; | 
|  | for(unsigned i=0; i < count; ++i) { | 
|  | void *SaveInfo; | 
|  | if ( io.preflightFlowElement(i, SaveInfo) ) { | 
|  | yamlize(io, SequenceTraits<T>::element(io, Seq, i), true, Ctx); | 
|  | io.postflightFlowElement(SaveInfo); | 
|  | } | 
|  | } | 
|  | io.endFlowSequence(); | 
|  | } | 
|  | else { | 
|  | unsigned incnt = io.beginSequence(); | 
|  | unsigned count = io.outputting() ? SequenceTraits<T>::size(io, Seq) : incnt; | 
|  | for(unsigned i=0; i < count; ++i) { | 
|  | void *SaveInfo; | 
|  | if ( io.preflightElement(i, SaveInfo) ) { | 
|  | yamlize(io, SequenceTraits<T>::element(io, Seq, i), true, Ctx); | 
|  | io.postflightElement(SaveInfo); | 
|  | } | 
|  | } | 
|  | io.endSequence(); | 
|  | } | 
|  | } | 
|  |  | 
|  | template<> | 
|  | struct ScalarTraits<bool> { | 
|  | static void output(const bool &, void*, llvm::raw_ostream &); | 
|  | static StringRef input(StringRef, void*, bool &); | 
|  | static bool mustQuote(StringRef) { return false; } | 
|  | }; | 
|  |  | 
|  | template<> | 
|  | struct ScalarTraits<StringRef> { | 
|  | static void output(const StringRef &, void*, llvm::raw_ostream &); | 
|  | static StringRef input(StringRef, void*, StringRef &); | 
|  | static bool mustQuote(StringRef S) { return needsQuotes(S); } | 
|  | }; | 
|  |  | 
|  | template<> | 
|  | struct ScalarTraits<std::string> { | 
|  | static void output(const std::string &, void*, llvm::raw_ostream &); | 
|  | static StringRef input(StringRef, void*, std::string &); | 
|  | static bool mustQuote(StringRef S) { return needsQuotes(S); } | 
|  | }; | 
|  |  | 
|  | template<> | 
|  | struct ScalarTraits<uint8_t> { | 
|  | static void output(const uint8_t &, void*, llvm::raw_ostream &); | 
|  | static StringRef input(StringRef, void*, uint8_t &); | 
|  | static bool mustQuote(StringRef) { return false; } | 
|  | }; | 
|  |  | 
|  | template<> | 
|  | struct ScalarTraits<uint16_t> { | 
|  | static void output(const uint16_t &, void*, llvm::raw_ostream &); | 
|  | static StringRef input(StringRef, void*, uint16_t &); | 
|  | static bool mustQuote(StringRef) { return false; } | 
|  | }; | 
|  |  | 
|  | template<> | 
|  | struct ScalarTraits<uint32_t> { | 
|  | static void output(const uint32_t &, void*, llvm::raw_ostream &); | 
|  | static StringRef input(StringRef, void*, uint32_t &); | 
|  | static bool mustQuote(StringRef) { return false; } | 
|  | }; | 
|  |  | 
|  | template<> | 
|  | struct ScalarTraits<uint64_t> { | 
|  | static void output(const uint64_t &, void*, llvm::raw_ostream &); | 
|  | static StringRef input(StringRef, void*, uint64_t &); | 
|  | static bool mustQuote(StringRef) { return false; } | 
|  | }; | 
|  |  | 
|  | template<> | 
|  | struct ScalarTraits<int8_t> { | 
|  | static void output(const int8_t &, void*, llvm::raw_ostream &); | 
|  | static StringRef input(StringRef, void*, int8_t &); | 
|  | static bool mustQuote(StringRef) { return false; } | 
|  | }; | 
|  |  | 
|  | template<> | 
|  | struct ScalarTraits<int16_t> { | 
|  | static void output(const int16_t &, void*, llvm::raw_ostream &); | 
|  | static StringRef input(StringRef, void*, int16_t &); | 
|  | static bool mustQuote(StringRef) { return false; } | 
|  | }; | 
|  |  | 
|  | template<> | 
|  | struct ScalarTraits<int32_t> { | 
|  | static void output(const int32_t &, void*, llvm::raw_ostream &); | 
|  | static StringRef input(StringRef, void*, int32_t &); | 
|  | static bool mustQuote(StringRef) { return false; } | 
|  | }; | 
|  |  | 
|  | template<> | 
|  | struct ScalarTraits<int64_t> { | 
|  | static void output(const int64_t &, void*, llvm::raw_ostream &); | 
|  | static StringRef input(StringRef, void*, int64_t &); | 
|  | static bool mustQuote(StringRef) { return false; } | 
|  | }; | 
|  |  | 
|  | template<> | 
|  | struct ScalarTraits<float> { | 
|  | static void output(const float &, void*, llvm::raw_ostream &); | 
|  | static StringRef input(StringRef, void*, float &); | 
|  | static bool mustQuote(StringRef) { return false; } | 
|  | }; | 
|  |  | 
|  | template<> | 
|  | struct ScalarTraits<double> { | 
|  | static void output(const double &, void*, llvm::raw_ostream &); | 
|  | static StringRef input(StringRef, void*, double &); | 
|  | static bool mustQuote(StringRef) { return false; } | 
|  | }; | 
|  |  | 
|  | // For endian types, we just use the existing ScalarTraits for the underlying | 
|  | // type.  This way endian aware types are supported whenever a ScalarTraits | 
|  | // is defined for the underlying type. | 
|  | template <typename value_type, support::endianness endian, size_t alignment> | 
|  | struct ScalarTraits<support::detail::packed_endian_specific_integral< | 
|  | value_type, endian, alignment>> { | 
|  | typedef support::detail::packed_endian_specific_integral<value_type, endian, | 
|  | alignment> | 
|  | endian_type; | 
|  |  | 
|  | static void output(const endian_type &E, void *Ctx, | 
|  | llvm::raw_ostream &Stream) { | 
|  | ScalarTraits<value_type>::output(static_cast<value_type>(E), Ctx, Stream); | 
|  | } | 
|  |  | 
|  | static StringRef input(StringRef Str, void *Ctx, endian_type &E) { | 
|  | value_type V; | 
|  | auto R = ScalarTraits<value_type>::input(Str, Ctx, V); | 
|  | E = static_cast<endian_type>(V); | 
|  | return R; | 
|  | } | 
|  |  | 
|  | static bool mustQuote(StringRef Str) { | 
|  | return ScalarTraits<value_type>::mustQuote(Str); | 
|  | } | 
|  | }; | 
|  |  | 
|  | // Utility for use within MappingTraits<>::mapping() method | 
|  | // to [de]normalize an object for use with YAML conversion. | 
|  | template <typename TNorm, typename TFinal> | 
|  | struct MappingNormalization { | 
|  | MappingNormalization(IO &i_o, TFinal &Obj) | 
|  | : io(i_o), BufPtr(nullptr), Result(Obj) { | 
|  | if ( io.outputting() ) { | 
|  | BufPtr = new (&Buffer) TNorm(io, Obj); | 
|  | } | 
|  | else { | 
|  | BufPtr = new (&Buffer) TNorm(io); | 
|  | } | 
|  | } | 
|  |  | 
|  | ~MappingNormalization() { | 
|  | if ( ! io.outputting() ) { | 
|  | Result = BufPtr->denormalize(io); | 
|  | } | 
|  | BufPtr->~TNorm(); | 
|  | } | 
|  |  | 
|  | TNorm* operator->() { return BufPtr; } | 
|  |  | 
|  | private: | 
|  | typedef llvm::AlignedCharArrayUnion<TNorm> Storage; | 
|  |  | 
|  | Storage       Buffer; | 
|  | IO           &io; | 
|  | TNorm        *BufPtr; | 
|  | TFinal       &Result; | 
|  | }; | 
|  |  | 
|  | // Utility for use within MappingTraits<>::mapping() method | 
|  | // to [de]normalize an object for use with YAML conversion. | 
|  | template <typename TNorm, typename TFinal> | 
|  | struct MappingNormalizationHeap { | 
|  | MappingNormalizationHeap(IO &i_o, TFinal &Obj, | 
|  | llvm::BumpPtrAllocator *allocator) | 
|  | : io(i_o), BufPtr(nullptr), Result(Obj) { | 
|  | if ( io.outputting() ) { | 
|  | BufPtr = new (&Buffer) TNorm(io, Obj); | 
|  | } | 
|  | else if (allocator) { | 
|  | BufPtr = allocator->Allocate<TNorm>(); | 
|  | new (BufPtr) TNorm(io); | 
|  | } else { | 
|  | BufPtr = new TNorm(io); | 
|  | } | 
|  | } | 
|  |  | 
|  | ~MappingNormalizationHeap() { | 
|  | if ( io.outputting() ) { | 
|  | BufPtr->~TNorm(); | 
|  | } | 
|  | else { | 
|  | Result = BufPtr->denormalize(io); | 
|  | } | 
|  | } | 
|  |  | 
|  | TNorm* operator->() { return BufPtr; } | 
|  |  | 
|  | private: | 
|  | typedef llvm::AlignedCharArrayUnion<TNorm> Storage; | 
|  |  | 
|  | Storage       Buffer; | 
|  | IO           &io; | 
|  | TNorm        *BufPtr; | 
|  | TFinal       &Result; | 
|  | }; | 
|  |  | 
|  | /// | 
|  | /// The Input class is used to parse a yaml document into in-memory structs | 
|  | /// and vectors. | 
|  | /// | 
|  | /// It works by using YAMLParser to do a syntax parse of the entire yaml | 
|  | /// document, then the Input class builds a graph of HNodes which wraps | 
|  | /// each yaml Node.  The extra layer is buffering.  The low level yaml | 
|  | /// parser only lets you look at each node once.  The buffering layer lets | 
|  | /// you search and interate multiple times.  This is necessary because | 
|  | /// the mapRequired() method calls may not be in the same order | 
|  | /// as the keys in the document. | 
|  | /// | 
|  | class Input : public IO { | 
|  | public: | 
|  | // Construct a yaml Input object from a StringRef and optional | 
|  | // user-data. The DiagHandler can be specified to provide | 
|  | // alternative error reporting. | 
|  | Input(StringRef InputContent, | 
|  | void *Ctxt = nullptr, | 
|  | SourceMgr::DiagHandlerTy DiagHandler = nullptr, | 
|  | void *DiagHandlerCtxt = nullptr); | 
|  | ~Input() override; | 
|  |  | 
|  | // Check if there was an syntax or semantic error during parsing. | 
|  | std::error_code error(); | 
|  |  | 
|  | private: | 
|  | bool outputting() override; | 
|  | bool mapTag(StringRef, bool) override; | 
|  | void beginMapping() override; | 
|  | void endMapping() override; | 
|  | bool preflightKey(const char *, bool, bool, bool &, void *&) override; | 
|  | void postflightKey(void *) override; | 
|  | void beginFlowMapping() override; | 
|  | void endFlowMapping() override; | 
|  | unsigned beginSequence() override; | 
|  | void endSequence() override; | 
|  | bool preflightElement(unsigned index, void *&) override; | 
|  | void postflightElement(void *) override; | 
|  | unsigned beginFlowSequence() override; | 
|  | bool preflightFlowElement(unsigned , void *&) override; | 
|  | void postflightFlowElement(void *) override; | 
|  | void endFlowSequence() override; | 
|  | void beginEnumScalar() override; | 
|  | bool matchEnumScalar(const char*, bool) override; | 
|  | bool matchEnumFallback() override; | 
|  | void endEnumScalar() override; | 
|  | bool beginBitSetScalar(bool &) override; | 
|  | bool bitSetMatch(const char *, bool ) override; | 
|  | void endBitSetScalar() override; | 
|  | void scalarString(StringRef &, bool) override; | 
|  | void blockScalarString(StringRef &) override; | 
|  | void setError(const Twine &message) override; | 
|  | bool canElideEmptySequence() override; | 
|  |  | 
|  | class HNode { | 
|  | virtual void anchor(); | 
|  |  | 
|  | public: | 
|  | HNode(Node *n) : _node(n) { } | 
|  | virtual ~HNode() = default; | 
|  |  | 
|  | static inline bool classof(const HNode *) { return true; } | 
|  |  | 
|  | Node *_node; | 
|  | }; | 
|  |  | 
|  | class EmptyHNode : public HNode { | 
|  | void anchor() override; | 
|  |  | 
|  | public: | 
|  | EmptyHNode(Node *n) : HNode(n) { } | 
|  |  | 
|  | static inline bool classof(const HNode *n) { | 
|  | return NullNode::classof(n->_node); | 
|  | } | 
|  |  | 
|  | static inline bool classof(const EmptyHNode *) { return true; } | 
|  | }; | 
|  |  | 
|  | class ScalarHNode : public HNode { | 
|  | void anchor() override; | 
|  |  | 
|  | public: | 
|  | ScalarHNode(Node *n, StringRef s) : HNode(n), _value(s) { } | 
|  |  | 
|  | StringRef value() const { return _value; } | 
|  |  | 
|  | static inline bool classof(const HNode *n) { | 
|  | return ScalarNode::classof(n->_node) || | 
|  | BlockScalarNode::classof(n->_node); | 
|  | } | 
|  |  | 
|  | static inline bool classof(const ScalarHNode *) { return true; } | 
|  |  | 
|  | protected: | 
|  | StringRef _value; | 
|  | }; | 
|  |  | 
|  | class MapHNode : public HNode { | 
|  | void anchor() override; | 
|  |  | 
|  | public: | 
|  | MapHNode(Node *n) : HNode(n) { } | 
|  |  | 
|  | static inline bool classof(const HNode *n) { | 
|  | return MappingNode::classof(n->_node); | 
|  | } | 
|  |  | 
|  | static inline bool classof(const MapHNode *) { return true; } | 
|  |  | 
|  | typedef llvm::StringMap<std::unique_ptr<HNode>> NameToNode; | 
|  |  | 
|  | bool isValidKey(StringRef key); | 
|  |  | 
|  | NameToNode                        Mapping; | 
|  | llvm::SmallVector<const char*, 6> ValidKeys; | 
|  | }; | 
|  |  | 
|  | class SequenceHNode : public HNode { | 
|  | void anchor() override; | 
|  |  | 
|  | public: | 
|  | SequenceHNode(Node *n) : HNode(n) { } | 
|  |  | 
|  | static inline bool classof(const HNode *n) { | 
|  | return SequenceNode::classof(n->_node); | 
|  | } | 
|  |  | 
|  | static inline bool classof(const SequenceHNode *) { return true; } | 
|  |  | 
|  | std::vector<std::unique_ptr<HNode>> Entries; | 
|  | }; | 
|  |  | 
|  | std::unique_ptr<Input::HNode> createHNodes(Node *node); | 
|  | void setError(HNode *hnode, const Twine &message); | 
|  | void setError(Node *node, const Twine &message); | 
|  |  | 
|  | public: | 
|  | // These are only used by operator>>. They could be private | 
|  | // if those templated things could be made friends. | 
|  | bool setCurrentDocument(); | 
|  | bool nextDocument(); | 
|  |  | 
|  | /// Returns the current node that's being parsed by the YAML Parser. | 
|  | const Node *getCurrentNode() const; | 
|  |  | 
|  | private: | 
|  | llvm::SourceMgr                     SrcMgr; // must be before Strm | 
|  | std::unique_ptr<llvm::yaml::Stream> Strm; | 
|  | std::unique_ptr<HNode>              TopNode; | 
|  | std::error_code                     EC; | 
|  | llvm::BumpPtrAllocator              StringAllocator; | 
|  | llvm::yaml::document_iterator       DocIterator; | 
|  | std::vector<bool>                   BitValuesUsed; | 
|  | HNode                              *CurrentNode; | 
|  | bool                                ScalarMatchFound; | 
|  | }; | 
|  |  | 
|  | /// | 
|  | /// The Output class is used to generate a yaml document from in-memory structs | 
|  | /// and vectors. | 
|  | /// | 
|  | class Output : public IO { | 
|  | public: | 
|  | Output(llvm::raw_ostream &, void *Ctxt = nullptr, int WrapColumn = 70); | 
|  | ~Output() override; | 
|  |  | 
|  | bool outputting() override; | 
|  | bool mapTag(StringRef, bool) override; | 
|  | void beginMapping() override; | 
|  | void endMapping() override; | 
|  | bool preflightKey(const char *key, bool, bool, bool &, void *&) override; | 
|  | void postflightKey(void *) override; | 
|  | void beginFlowMapping() override; | 
|  | void endFlowMapping() override; | 
|  | unsigned beginSequence() override; | 
|  | void endSequence() override; | 
|  | bool preflightElement(unsigned, void *&) override; | 
|  | void postflightElement(void *) override; | 
|  | unsigned beginFlowSequence() override; | 
|  | bool preflightFlowElement(unsigned, void *&) override; | 
|  | void postflightFlowElement(void *) override; | 
|  | void endFlowSequence() override; | 
|  | void beginEnumScalar() override; | 
|  | bool matchEnumScalar(const char*, bool) override; | 
|  | bool matchEnumFallback() override; | 
|  | void endEnumScalar() override; | 
|  | bool beginBitSetScalar(bool &) override; | 
|  | bool bitSetMatch(const char *, bool ) override; | 
|  | void endBitSetScalar() override; | 
|  | void scalarString(StringRef &, bool) override; | 
|  | void blockScalarString(StringRef &) override; | 
|  | void setError(const Twine &message) override; | 
|  | bool canElideEmptySequence() override; | 
|  |  | 
|  | // These are only used by operator<<. They could be private | 
|  | // if that templated operator could be made a friend. | 
|  | void beginDocuments(); | 
|  | bool preflightDocument(unsigned); | 
|  | void postflightDocument(); | 
|  | void endDocuments(); | 
|  |  | 
|  | private: | 
|  | void output(StringRef s); | 
|  | void outputUpToEndOfLine(StringRef s); | 
|  | void newLineCheck(); | 
|  | void outputNewLine(); | 
|  | void paddedKey(StringRef key); | 
|  | void flowKey(StringRef Key); | 
|  |  | 
|  | enum InState { | 
|  | inSeq, | 
|  | inFlowSeq, | 
|  | inMapFirstKey, | 
|  | inMapOtherKey, | 
|  | inFlowMapFirstKey, | 
|  | inFlowMapOtherKey | 
|  | }; | 
|  |  | 
|  | llvm::raw_ostream       &Out; | 
|  | int                      WrapColumn; | 
|  | SmallVector<InState, 8>  StateStack; | 
|  | int                      Column; | 
|  | int                      ColumnAtFlowStart; | 
|  | int                      ColumnAtMapFlowStart; | 
|  | bool                     NeedBitValueComma; | 
|  | bool                     NeedFlowSequenceComma; | 
|  | bool                     EnumerationMatchFound; | 
|  | bool                     NeedsNewLine; | 
|  | }; | 
|  |  | 
|  | /// YAML I/O does conversion based on types. But often native data types | 
|  | /// are just a typedef of built in intergral types (e.g. int).  But the C++ | 
|  | /// type matching system sees through the typedef and all the typedefed types | 
|  | /// look like a built in type. This will cause the generic YAML I/O conversion | 
|  | /// to be used. To provide better control over the YAML conversion, you can | 
|  | /// use this macro instead of typedef.  It will create a class with one field | 
|  | /// and automatic conversion operators to and from the base type. | 
|  | /// Based on BOOST_STRONG_TYPEDEF | 
|  | #define LLVM_YAML_STRONG_TYPEDEF(_base, _type)                                 \ | 
|  | struct _type {                                                             \ | 
|  | _type() = default;                                                     \ | 
|  | _type(const _base v) : value(v) {}                                     \ | 
|  | _type(const _type &v) = default;                                       \ | 
|  | _type &operator=(const _type &rhs) = default;                          \ | 
|  | _type &operator=(const _base &rhs) { value = rhs; return *this; }      \ | 
|  | operator const _base & () const { return value; }                      \ | 
|  | bool operator==(const _type &rhs) const { return value == rhs.value; } \ | 
|  | bool operator==(const _base &rhs) const { return value == rhs; }       \ | 
|  | bool operator<(const _type &rhs) const { return value < rhs.value; }   \ | 
|  | _base value;                                                           \ | 
|  | typedef _base BaseType;                                                \ | 
|  | }; | 
|  |  | 
|  | /// | 
|  | /// Use these types instead of uintXX_t in any mapping to have | 
|  | /// its yaml output formatted as hexadecimal. | 
|  | /// | 
|  | LLVM_YAML_STRONG_TYPEDEF(uint8_t, Hex8) | 
|  | LLVM_YAML_STRONG_TYPEDEF(uint16_t, Hex16) | 
|  | LLVM_YAML_STRONG_TYPEDEF(uint32_t, Hex32) | 
|  | LLVM_YAML_STRONG_TYPEDEF(uint64_t, Hex64) | 
|  |  | 
|  | template<> | 
|  | struct ScalarTraits<Hex8> { | 
|  | static void output(const Hex8 &, void*, llvm::raw_ostream &); | 
|  | static StringRef input(StringRef, void*, Hex8 &); | 
|  | static bool mustQuote(StringRef) { return false; } | 
|  | }; | 
|  |  | 
|  | template<> | 
|  | struct ScalarTraits<Hex16> { | 
|  | static void output(const Hex16 &, void*, llvm::raw_ostream &); | 
|  | static StringRef input(StringRef, void*, Hex16 &); | 
|  | static bool mustQuote(StringRef) { return false; } | 
|  | }; | 
|  |  | 
|  | template<> | 
|  | struct ScalarTraits<Hex32> { | 
|  | static void output(const Hex32 &, void*, llvm::raw_ostream &); | 
|  | static StringRef input(StringRef, void*, Hex32 &); | 
|  | static bool mustQuote(StringRef) { return false; } | 
|  | }; | 
|  |  | 
|  | template<> | 
|  | struct ScalarTraits<Hex64> { | 
|  | static void output(const Hex64 &, void*, llvm::raw_ostream &); | 
|  | static StringRef input(StringRef, void*, Hex64 &); | 
|  | static bool mustQuote(StringRef) { return false; } | 
|  | }; | 
|  |  | 
|  | // Define non-member operator>> so that Input can stream in a document list. | 
|  | template <typename T> | 
|  | inline | 
|  | typename std::enable_if<has_DocumentListTraits<T>::value, Input &>::type | 
|  | operator>>(Input &yin, T &docList) { | 
|  | int i = 0; | 
|  | EmptyContext Ctx; | 
|  | while ( yin.setCurrentDocument() ) { | 
|  | yamlize(yin, DocumentListTraits<T>::element(yin, docList, i), true, Ctx); | 
|  | if ( yin.error() ) | 
|  | return yin; | 
|  | yin.nextDocument(); | 
|  | ++i; | 
|  | } | 
|  | return yin; | 
|  | } | 
|  |  | 
|  | // Define non-member operator>> so that Input can stream in a map as a document. | 
|  | template <typename T> | 
|  | inline typename std::enable_if<has_MappingTraits<T, EmptyContext>::value, | 
|  | Input &>::type | 
|  | operator>>(Input &yin, T &docMap) { | 
|  | EmptyContext Ctx; | 
|  | yin.setCurrentDocument(); | 
|  | yamlize(yin, docMap, true, Ctx); | 
|  | return yin; | 
|  | } | 
|  |  | 
|  | // Define non-member operator>> so that Input can stream in a sequence as | 
|  | // a document. | 
|  | template <typename T> | 
|  | inline | 
|  | typename std::enable_if<has_SequenceTraits<T>::value, Input &>::type | 
|  | operator>>(Input &yin, T &docSeq) { | 
|  | EmptyContext Ctx; | 
|  | if (yin.setCurrentDocument()) | 
|  | yamlize(yin, docSeq, true, Ctx); | 
|  | return yin; | 
|  | } | 
|  |  | 
|  | // Define non-member operator>> so that Input can stream in a block scalar. | 
|  | template <typename T> | 
|  | inline | 
|  | typename std::enable_if<has_BlockScalarTraits<T>::value, Input &>::type | 
|  | operator>>(Input &In, T &Val) { | 
|  | EmptyContext Ctx; | 
|  | if (In.setCurrentDocument()) | 
|  | yamlize(In, Val, true, Ctx); | 
|  | return In; | 
|  | } | 
|  |  | 
|  | // Provide better error message about types missing a trait specialization | 
|  | template <typename T> | 
|  | inline typename std::enable_if<missingTraits<T, EmptyContext>::value, | 
|  | Input &>::type | 
|  | operator>>(Input &yin, T &docSeq) { | 
|  | char missing_yaml_trait_for_type[sizeof(MissingTrait<T>)]; | 
|  | return yin; | 
|  | } | 
|  |  | 
|  | // Define non-member operator<< so that Output can stream out document list. | 
|  | template <typename T> | 
|  | inline | 
|  | typename std::enable_if<has_DocumentListTraits<T>::value, Output &>::type | 
|  | operator<<(Output &yout, T &docList) { | 
|  | EmptyContext Ctx; | 
|  | yout.beginDocuments(); | 
|  | const size_t count = DocumentListTraits<T>::size(yout, docList); | 
|  | for(size_t i=0; i < count; ++i) { | 
|  | if ( yout.preflightDocument(i) ) { | 
|  | yamlize(yout, DocumentListTraits<T>::element(yout, docList, i), true, | 
|  | Ctx); | 
|  | yout.postflightDocument(); | 
|  | } | 
|  | } | 
|  | yout.endDocuments(); | 
|  | return yout; | 
|  | } | 
|  |  | 
|  | // Define non-member operator<< so that Output can stream out a map. | 
|  | template <typename T> | 
|  | inline typename std::enable_if<has_MappingTraits<T, EmptyContext>::value, | 
|  | Output &>::type | 
|  | operator<<(Output &yout, T &map) { | 
|  | EmptyContext Ctx; | 
|  | yout.beginDocuments(); | 
|  | if ( yout.preflightDocument(0) ) { | 
|  | yamlize(yout, map, true, Ctx); | 
|  | yout.postflightDocument(); | 
|  | } | 
|  | yout.endDocuments(); | 
|  | return yout; | 
|  | } | 
|  |  | 
|  | // Define non-member operator<< so that Output can stream out a sequence. | 
|  | template <typename T> | 
|  | inline | 
|  | typename std::enable_if<has_SequenceTraits<T>::value, Output &>::type | 
|  | operator<<(Output &yout, T &seq) { | 
|  | EmptyContext Ctx; | 
|  | yout.beginDocuments(); | 
|  | if ( yout.preflightDocument(0) ) { | 
|  | yamlize(yout, seq, true, Ctx); | 
|  | yout.postflightDocument(); | 
|  | } | 
|  | yout.endDocuments(); | 
|  | return yout; | 
|  | } | 
|  |  | 
|  | // Define non-member operator<< so that Output can stream out a block scalar. | 
|  | template <typename T> | 
|  | inline | 
|  | typename std::enable_if<has_BlockScalarTraits<T>::value, Output &>::type | 
|  | operator<<(Output &Out, T &Val) { | 
|  | EmptyContext Ctx; | 
|  | Out.beginDocuments(); | 
|  | if (Out.preflightDocument(0)) { | 
|  | yamlize(Out, Val, true, Ctx); | 
|  | Out.postflightDocument(); | 
|  | } | 
|  | Out.endDocuments(); | 
|  | return Out; | 
|  | } | 
|  |  | 
|  | // Provide better error message about types missing a trait specialization | 
|  | template <typename T> | 
|  | inline typename std::enable_if<missingTraits<T, EmptyContext>::value, | 
|  | Output &>::type | 
|  | operator<<(Output &yout, T &seq) { | 
|  | char missing_yaml_trait_for_type[sizeof(MissingTrait<T>)]; | 
|  | return yout; | 
|  | } | 
|  |  | 
|  | template <typename T> struct SequenceTraitsImpl { | 
|  | typedef typename T::value_type _type; | 
|  | static size_t size(IO &io, T &seq) { return seq.size(); } | 
|  | static _type &element(IO &io, T &seq, size_t index) { | 
|  | if (index >= seq.size()) | 
|  | seq.resize(index + 1); | 
|  | return seq[index]; | 
|  | } | 
|  | }; | 
|  |  | 
|  | } // end namespace yaml | 
|  | } // end namespace llvm | 
|  |  | 
|  | /// Utility for declaring that a std::vector of a particular type | 
|  | /// should be considered a YAML sequence. | 
|  | #define LLVM_YAML_IS_SEQUENCE_VECTOR(_type)                                    \ | 
|  | namespace llvm {                                                             \ | 
|  | namespace yaml {                                                             \ | 
|  | template <>                                                                  \ | 
|  | struct SequenceTraits<std::vector<_type>>                                    \ | 
|  | : public SequenceTraitsImpl<std::vector<_type>> {};                      \ | 
|  | template <unsigned N>                                                        \ | 
|  | struct SequenceTraits<SmallVector<_type, N>>                                 \ | 
|  | : public SequenceTraitsImpl<SmallVector<_type, N>> {};                   \ | 
|  | }                                                                            \ | 
|  | } | 
|  |  | 
|  | /// Utility for declaring that a std::vector of a particular type | 
|  | /// should be considered a YAML flow sequence. | 
|  | /// We need to do a partial specialization on the vector version, not a full. | 
|  | /// If this is a full specialization, the compiler is a bit too "smart" and | 
|  | /// decides to warn on -Wunused-const-variable.  This workaround can be | 
|  | /// removed and we can do a full specialization on std::vector<T> once | 
|  | /// PR28878 is fixed. | 
|  | #define LLVM_YAML_IS_FLOW_SEQUENCE_VECTOR(_type)                               \ | 
|  | namespace llvm {                                                             \ | 
|  | namespace yaml {                                                             \ | 
|  | template <unsigned N>                                                        \ | 
|  | struct SequenceTraits<SmallVector<_type, N>>                                 \ | 
|  | : public SequenceTraitsImpl<SmallVector<_type, N>> {                     \ | 
|  | static const bool flow = true;                                             \ | 
|  | };                                                                           \ | 
|  | template <typename Allocator>                                                \ | 
|  | struct SequenceTraits<std::vector<_type, Allocator>>                         \ | 
|  | : public SequenceTraitsImpl<std::vector<_type, Allocator>> {             \ | 
|  | static const bool flow = true;                                             \ | 
|  | };                                                                           \ | 
|  | }                                                                            \ | 
|  | } | 
|  |  | 
|  | /// Utility for declaring that a std::vector of a particular type | 
|  | /// should be considered a YAML document list. | 
|  | #define LLVM_YAML_IS_DOCUMENT_LIST_VECTOR(_type)                               \ | 
|  | namespace llvm {                                                             \ | 
|  | namespace yaml {                                                             \ | 
|  | template <unsigned N>                                                        \ | 
|  | struct DocumentListTraits<SmallVector<_type, N>>                             \ | 
|  | : public SequenceTraitsImpl<SmallVector<_type, N>> {};                   \ | 
|  | template <>                                                                  \ | 
|  | struct DocumentListTraits<std::vector<_type>>                                \ | 
|  | : public SequenceTraitsImpl<std::vector<_type>> {};                      \ | 
|  | }                                                                            \ | 
|  | } | 
|  |  | 
|  | #endif // LLVM_SUPPORT_YAMLTRAITS_H |