forked from OSchip/llvm-project
				
			
		
			
				
	
	
		
			250 lines
		
	
	
		
			7.1 KiB
		
	
	
	
		
			C++
		
	
	
	
			
		
		
	
	
			250 lines
		
	
	
		
			7.1 KiB
		
	
	
	
		
			C++
		
	
	
	
| //===-- MsgPackDocumentYAML.cpp - MsgPack Document YAML interface -------*-===//
 | |
| //
 | |
| //                     The LLVM Compiler Infrastructure
 | |
| //
 | |
| // This file is distributed under the University of Illinois Open Source
 | |
| // License. See LICENSE.TXT for details.
 | |
| //
 | |
| //===----------------------------------------------------------------------===//
 | |
| //
 | |
| /// This file implements YAMLIO on a msgpack::Document.
 | |
| //
 | |
| //===----------------------------------------------------------------------===//
 | |
| 
 | |
| #include "llvm/BinaryFormat/MsgPackDocument.h"
 | |
| #include "llvm/Support/YAMLTraits.h"
 | |
| 
 | |
| using namespace llvm;
 | |
| using namespace msgpack;
 | |
| 
 | |
| namespace {
 | |
| 
 | |
| // Struct used to represent scalar node. (MapDocNode and ArrayDocNode already
 | |
| // exist in MsgPackDocument.h.)
 | |
| struct ScalarDocNode : DocNode {
 | |
|   ScalarDocNode(DocNode N) : DocNode(N) {}
 | |
| 
 | |
|   /// Get the YAML tag for this ScalarDocNode. This normally returns ""; it only
 | |
|   /// returns something else if the result of toString would be ambiguous, e.g.
 | |
|   /// a string that parses as a number or boolean.
 | |
|   StringRef getYAMLTag() const;
 | |
| };
 | |
| 
 | |
| } // namespace
 | |
| 
 | |
| /// Convert this DocNode to a string, assuming it is scalar.
 | |
| std::string DocNode::toString() const {
 | |
|   std::string S;
 | |
|   raw_string_ostream OS(S);
 | |
|   switch (getKind()) {
 | |
|   case msgpack::Type::String:
 | |
|     OS << Raw;
 | |
|     break;
 | |
|   case msgpack::Type::Nil:
 | |
|     break;
 | |
|   case msgpack::Type::Boolean:
 | |
|     OS << (Bool ? "true" : "false");
 | |
|     break;
 | |
|   case msgpack::Type::Int:
 | |
|     OS << Int;
 | |
|     break;
 | |
|   case msgpack::Type::UInt:
 | |
|     if (getDocument()->getHexMode())
 | |
|       OS << format("%#llx", (unsigned long long)UInt);
 | |
|     else
 | |
|       OS << UInt;
 | |
|     break;
 | |
|   case msgpack::Type::Float:
 | |
|     OS << Float;
 | |
|     break;
 | |
|   default:
 | |
|     llvm_unreachable("not scalar");
 | |
|     break;
 | |
|   }
 | |
|   return OS.str();
 | |
| }
 | |
| 
 | |
| /// Convert the StringRef and use it to set this DocNode (assuming scalar). If
 | |
| /// it is a string, copy the string into the Document's strings list so we do
 | |
| /// not rely on S having a lifetime beyond this call. Tag is "" or a YAML tag.
 | |
| StringRef DocNode::fromString(StringRef S, StringRef Tag) {
 | |
|   if (Tag == "tag:yaml.org,2002:str")
 | |
|     Tag = "";
 | |
|   if (Tag == "!int" || Tag == "") {
 | |
|     // Try unsigned int then signed int.
 | |
|     *this = getDocument()->getNode(uint64_t(0));
 | |
|     StringRef Err = yaml::ScalarTraits<uint64_t>::input(S, nullptr, getUInt());
 | |
|     if (Err != "") {
 | |
|       *this = getDocument()->getNode(int64_t(0));
 | |
|       Err = yaml::ScalarTraits<int64_t>::input(S, nullptr, getInt());
 | |
|     }
 | |
|     if (Err == "" || Tag != "")
 | |
|       return Err;
 | |
|   }
 | |
|   if (Tag == "!nil") {
 | |
|     *this = getDocument()->getNode();
 | |
|     return "";
 | |
|   }
 | |
|   if (Tag == "!bool" || Tag == "") {
 | |
|     *this = getDocument()->getNode(false);
 | |
|     StringRef Err = yaml::ScalarTraits<bool>::input(S, nullptr, getBool());
 | |
|     if (Err == "" || Tag != "")
 | |
|       return Err;
 | |
|   }
 | |
|   if (Tag == "!float" || Tag == "") {
 | |
|     *this = getDocument()->getNode(0.0);
 | |
|     StringRef Err = yaml::ScalarTraits<double>::input(S, nullptr, getFloat());
 | |
|     if (Err == "" || Tag != "")
 | |
|       return Err;
 | |
|   }
 | |
|   assert((Tag == "!str" || Tag == "") && "unsupported tag");
 | |
|   std::string V;
 | |
|   StringRef Err = yaml::ScalarTraits<std::string>::input(S, nullptr, V);
 | |
|   if (Err == "")
 | |
|     *this = getDocument()->getNode(V, /*Copy=*/true);
 | |
|   return Err;
 | |
| }
 | |
| 
 | |
| /// Get the YAML tag for this ScalarDocNode. This normally returns ""; it only
 | |
| /// returns something else if the result of toString would be ambiguous, e.g.
 | |
| /// a string that parses as a number or boolean.
 | |
| StringRef ScalarDocNode::getYAMLTag() const {
 | |
|   if (getKind() == msgpack::Type::Nil)
 | |
|     return "!nil";
 | |
|   // Try converting both ways and see if we get the same kind. If not, we need
 | |
|   // a tag.
 | |
|   ScalarDocNode N = getDocument()->getNode();
 | |
|   N.fromString(toString(), "");
 | |
|   if (N.getKind() == getKind())
 | |
|     return "";
 | |
|   // Tolerate signedness of int changing, as tags do not differentiate between
 | |
|   // them anyway.
 | |
|   if (N.getKind() == msgpack::Type::UInt && getKind() == msgpack::Type::Int)
 | |
|     return "";
 | |
|   if (N.getKind() == msgpack::Type::Int && getKind() == msgpack::Type::UInt)
 | |
|     return "";
 | |
|   // We do need a tag.
 | |
|   switch (getKind()) {
 | |
|   case msgpack::Type::String:
 | |
|     return "!str";
 | |
|   case msgpack::Type::Int:
 | |
|     return "!int";
 | |
|   case msgpack::Type::UInt:
 | |
|     return "!int";
 | |
|   case msgpack::Type::Boolean:
 | |
|     return "!bool";
 | |
|   case msgpack::Type::Float:
 | |
|     return "!float";
 | |
|   default:
 | |
|     llvm_unreachable("unrecognized kind");
 | |
|   }
 | |
| }
 | |
| 
 | |
| namespace llvm {
 | |
| namespace yaml {
 | |
| 
 | |
| /// YAMLIO for DocNode
 | |
| template <> struct PolymorphicTraits<DocNode> {
 | |
| 
 | |
|   static NodeKind getKind(const DocNode &N) {
 | |
|     switch (N.getKind()) {
 | |
|     case msgpack::Type::Map:
 | |
|       return NodeKind::Map;
 | |
|     case msgpack::Type::Array:
 | |
|       return NodeKind::Sequence;
 | |
|     default:
 | |
|       return NodeKind::Scalar;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   static MapDocNode &getAsMap(DocNode &N) { return N.getMap(/*Convert=*/true); }
 | |
| 
 | |
|   static ArrayDocNode &getAsSequence(DocNode &N) {
 | |
|     N.getArray(/*Convert=*/true);
 | |
|     return *static_cast<ArrayDocNode *>(&N);
 | |
|   }
 | |
| 
 | |
|   static ScalarDocNode &getAsScalar(DocNode &N) {
 | |
|     return *static_cast<ScalarDocNode *>(&N);
 | |
|   }
 | |
| };
 | |
| 
 | |
| /// YAMLIO for ScalarDocNode
 | |
| template <> struct TaggedScalarTraits<ScalarDocNode> {
 | |
| 
 | |
|   static void output(const ScalarDocNode &S, void *Ctxt, raw_ostream &OS,
 | |
|                      raw_ostream &TagOS) {
 | |
|     TagOS << S.getYAMLTag();
 | |
|     OS << S.toString();
 | |
|   }
 | |
| 
 | |
|   static StringRef input(StringRef Str, StringRef Tag, void *Ctxt,
 | |
|                          ScalarDocNode &S) {
 | |
|     return S.fromString(Str, Tag);
 | |
|   }
 | |
| 
 | |
|   static QuotingType mustQuote(const ScalarDocNode &S, StringRef ScalarStr) {
 | |
|     switch (S.getKind()) {
 | |
|     case Type::Int:
 | |
|       return ScalarTraits<int64_t>::mustQuote(ScalarStr);
 | |
|     case Type::UInt:
 | |
|       return ScalarTraits<uint64_t>::mustQuote(ScalarStr);
 | |
|     case Type::Nil:
 | |
|       return ScalarTraits<StringRef>::mustQuote(ScalarStr);
 | |
|     case Type::Boolean:
 | |
|       return ScalarTraits<bool>::mustQuote(ScalarStr);
 | |
|     case Type::Float:
 | |
|       return ScalarTraits<double>::mustQuote(ScalarStr);
 | |
|     case Type::Binary:
 | |
|     case Type::String:
 | |
|       return ScalarTraits<std::string>::mustQuote(ScalarStr);
 | |
|     default:
 | |
|       llvm_unreachable("unrecognized ScalarKind");
 | |
|     }
 | |
|   }
 | |
| };
 | |
| 
 | |
| /// YAMLIO for MapDocNode
 | |
| template <> struct CustomMappingTraits<MapDocNode> {
 | |
| 
 | |
|   static void inputOne(IO &IO, StringRef Key, MapDocNode &M) {
 | |
|     ScalarDocNode KeyObj = M.getDocument()->getNode();
 | |
|     KeyObj.fromString(Key, "");
 | |
|     IO.mapRequired(Key.str().c_str(), M.getMap()[KeyObj]);
 | |
|   }
 | |
| 
 | |
|   static void output(IO &IO, MapDocNode &M) {
 | |
|     for (auto I : M.getMap()) {
 | |
|       IO.mapRequired(I.first.toString().c_str(), I.second);
 | |
|     }
 | |
|   }
 | |
| };
 | |
| 
 | |
| /// YAMLIO for ArrayNode
 | |
| template <> struct SequenceTraits<ArrayDocNode> {
 | |
| 
 | |
|   static size_t size(IO &IO, ArrayDocNode &A) { return A.size(); }
 | |
| 
 | |
|   static DocNode &element(IO &IO, ArrayDocNode &A, size_t Index) {
 | |
|     return A[Index];
 | |
|   }
 | |
| };
 | |
| 
 | |
| } // namespace yaml
 | |
| } // namespace llvm
 | |
| 
 | |
| /// Convert MsgPack Document to YAML text.
 | |
| void msgpack::Document::toYAML(raw_ostream &OS) {
 | |
|   yaml::Output Yout(OS);
 | |
|   Yout << getRoot();
 | |
| }
 | |
| 
 | |
| /// Read YAML text into the MsgPack document. Returns false on failure.
 | |
| bool msgpack::Document::fromYAML(StringRef S) {
 | |
|   clear();
 | |
|   yaml::Input Yin(S);
 | |
|   Yin >> getRoot();
 | |
|   return !Yin.error();
 | |
| }
 | |
| 
 |