blob: d62123cc985e4a32931d5f91cbe70b0aee1ed075 [file] [log] [blame]
Daniel Dunbar2538f7a2009-07-24 07:04:27 +00001//===-- Twine.cpp - Fast Temporary String Concatenation -------------------===//
2//
3// The LLVM Compiler Infrastructure
4//
5// This file is distributed under the University of Illinois Open Source
6// License. See LICENSE.TXT for details.
7//
8//===----------------------------------------------------------------------===//
9
10#include "llvm/ADT/Twine.h"
Daniel Dunbarb7be0e82009-08-19 18:09:47 +000011#include "llvm/ADT/SmallString.h"
David Greene2b965b02010-01-05 01:28:40 +000012#include "llvm/Support/Debug.h"
Daniel Dunbar2538f7a2009-07-24 07:04:27 +000013#include "llvm/Support/raw_ostream.h"
14using namespace llvm;
15
16std::string Twine::str() const {
Frits van Bommel331dbca2011-07-15 11:05:37 +000017 // If we're storing only a std::string, just return it.
18 if (LHSKind == StdStringKind && RHSKind == EmptyKind)
19 return *static_cast<const std::string*>(LHS);
20
21 // Otherwise, flatten and copy the contents first.
Daniel Dunbarb7be0e82009-08-19 18:09:47 +000022 SmallString<256> Vec;
Benjamin Kramerb357e062010-01-13 12:45:23 +000023 return toStringRef(Vec).str();
Daniel Dunbar2538f7a2009-07-24 07:04:27 +000024}
25
26void Twine::toVector(SmallVectorImpl<char> &Out) const {
27 raw_svector_ostream OS(Out);
28 print(OS);
29}
30
Benjamin Kramerb357e062010-01-13 12:45:23 +000031StringRef Twine::toStringRef(SmallVectorImpl<char> &Out) const {
32 if (isSingleStringRef())
33 return getSingleStringRef();
34 toVector(Out);
35 return StringRef(Out.data(), Out.size());
36}
37
Michael J. Spencer7dc7ac32010-12-01 20:37:30 +000038StringRef Twine::toNullTerminatedStringRef(SmallVectorImpl<char> &Out) const {
Michael J. Spencer0dda5432010-12-03 05:42:25 +000039 if (isUnary()) {
40 switch (getLHSKind()) {
41 case CStringKind:
42 // Already null terminated, yay!
43 return StringRef(static_cast<const char*>(LHS));
44 case StdStringKind: {
Frits van Bommel331dbca2011-07-15 11:05:37 +000045 const std::string *str = static_cast<const std::string*>(LHS);
46 return StringRef(str->c_str(), str->size());
47 }
Michael J. Spencer0dda5432010-12-03 05:42:25 +000048 default:
49 break;
50 }
Michael J. Spencer7dc7ac32010-12-01 20:37:30 +000051 }
52 toVector(Out);
53 Out.push_back(0);
54 Out.pop_back();
55 return StringRef(Out.data(), Out.size());
56}
57
Michael J. Spencer326990f2010-11-26 04:16:08 +000058void Twine::printOneChild(raw_ostream &OS, const void *Ptr,
Daniel Dunbar2538f7a2009-07-24 07:04:27 +000059 NodeKind Kind) const {
60 switch (Kind) {
61 case Twine::NullKind: break;
62 case Twine::EmptyKind: break;
Daniel Dunbar763457e2009-07-29 07:08:44 +000063 case Twine::TwineKind:
Michael J. Spencer326990f2010-11-26 04:16:08 +000064 static_cast<const Twine*>(Ptr)->print(OS);
Daniel Dunbar763457e2009-07-29 07:08:44 +000065 break;
Michael J. Spencer326990f2010-11-26 04:16:08 +000066 case Twine::CStringKind:
67 OS << static_cast<const char*>(Ptr);
Daniel Dunbar2538f7a2009-07-24 07:04:27 +000068 break;
69 case Twine::StdStringKind:
Michael J. Spencer326990f2010-11-26 04:16:08 +000070 OS << *static_cast<const std::string*>(Ptr);
Daniel Dunbar2538f7a2009-07-24 07:04:27 +000071 break;
72 case Twine::StringRefKind:
Michael J. Spencer326990f2010-11-26 04:16:08 +000073 OS << *static_cast<const StringRef*>(Ptr);
Daniel Dunbar2538f7a2009-07-24 07:04:27 +000074 break;
Daniel Dunbar2d8bc0f2009-07-30 21:15:14 +000075 case Twine::DecUIKind:
Chris Lattnerea03e102010-05-05 18:40:33 +000076 OS << (unsigned)(uintptr_t)Ptr;
Daniel Dunbar0165a2c2009-07-30 03:47:15 +000077 break;
Daniel Dunbar2d8bc0f2009-07-30 21:15:14 +000078 case Twine::DecIKind:
Chris Lattnerea03e102010-05-05 18:40:33 +000079 OS << (int)(intptr_t)Ptr;
Daniel Dunbar0165a2c2009-07-30 03:47:15 +000080 break;
Daniel Dunbar2d8bc0f2009-07-30 21:15:14 +000081 case Twine::DecULKind:
82 OS << *static_cast<const unsigned long*>(Ptr);
Daniel Dunbar763457e2009-07-29 07:08:44 +000083 break;
Daniel Dunbar2d8bc0f2009-07-30 21:15:14 +000084 case Twine::DecLKind:
85 OS << *static_cast<const long*>(Ptr);
86 break;
87 case Twine::DecULLKind:
88 OS << *static_cast<const unsigned long long*>(Ptr);
89 break;
90 case Twine::DecLLKind:
91 OS << *static_cast<const long long*>(Ptr);
Daniel Dunbar763457e2009-07-29 07:08:44 +000092 break;
93 case Twine::UHexKind:
Daniel Dunbar0fffbaf2009-07-30 18:30:19 +000094 OS.write_hex(*static_cast<const uint64_t*>(Ptr));
Daniel Dunbar2538f7a2009-07-24 07:04:27 +000095 break;
96 }
97}
98
Michael J. Spencer326990f2010-11-26 04:16:08 +000099void Twine::printOneChildRepr(raw_ostream &OS, const void *Ptr,
Daniel Dunbar2538f7a2009-07-24 07:04:27 +0000100 NodeKind Kind) const {
101 switch (Kind) {
102 case Twine::NullKind:
103 OS << "null"; break;
104 case Twine::EmptyKind:
105 OS << "empty"; break;
Daniel Dunbar2538f7a2009-07-24 07:04:27 +0000106 case Twine::TwineKind:
107 OS << "rope:";
108 static_cast<const Twine*>(Ptr)->printRepr(OS);
109 break;
Daniel Dunbar763457e2009-07-29 07:08:44 +0000110 case Twine::CStringKind:
111 OS << "cstring:\""
112 << static_cast<const char*>(Ptr) << "\"";
113 break;
114 case Twine::StdStringKind:
115 OS << "std::string:\""
116 << static_cast<const std::string*>(Ptr) << "\"";
117 break;
118 case Twine::StringRefKind:
119 OS << "stringref:\""
120 << static_cast<const StringRef*>(Ptr) << "\"";
121 break;
Daniel Dunbar2d8bc0f2009-07-30 21:15:14 +0000122 case Twine::DecUIKind:
Chris Lattnerea03e102010-05-05 18:40:33 +0000123 OS << "decUI:\"" << (unsigned)(uintptr_t)Ptr << "\"";
Daniel Dunbar763457e2009-07-29 07:08:44 +0000124 break;
Daniel Dunbar2d8bc0f2009-07-30 21:15:14 +0000125 case Twine::DecIKind:
Chris Lattnerea03e102010-05-05 18:40:33 +0000126 OS << "decI:\"" << (int)(intptr_t)Ptr << "\"";
Daniel Dunbar0165a2c2009-07-30 03:47:15 +0000127 break;
Daniel Dunbar2d8bc0f2009-07-30 21:15:14 +0000128 case Twine::DecULKind:
129 OS << "decUL:\"" << *static_cast<const unsigned long*>(Ptr) << "\"";
Daniel Dunbar0165a2c2009-07-30 03:47:15 +0000130 break;
Daniel Dunbar2d8bc0f2009-07-30 21:15:14 +0000131 case Twine::DecLKind:
132 OS << "decL:\"" << *static_cast<const long*>(Ptr) << "\"";
133 break;
134 case Twine::DecULLKind:
135 OS << "decULL:\"" << *static_cast<const unsigned long long*>(Ptr) << "\"";
136 break;
137 case Twine::DecLLKind:
138 OS << "decLL:\"" << *static_cast<const long long*>(Ptr) << "\"";
Daniel Dunbar763457e2009-07-29 07:08:44 +0000139 break;
140 case Twine::UHexKind:
Daniel Dunbar2d8bc0f2009-07-30 21:15:14 +0000141 OS << "uhex:\"" << static_cast<const uint64_t*>(Ptr) << "\"";
Daniel Dunbar763457e2009-07-29 07:08:44 +0000142 break;
Daniel Dunbar2538f7a2009-07-24 07:04:27 +0000143 }
144}
145
146void Twine::print(raw_ostream &OS) const {
147 printOneChild(OS, LHS, getLHSKind());
148 printOneChild(OS, RHS, getRHSKind());
149}
150
151void Twine::printRepr(raw_ostream &OS) const {
152 OS << "(Twine ";
153 printOneChildRepr(OS, LHS, getLHSKind());
154 OS << " ";
155 printOneChildRepr(OS, RHS, getRHSKind());
156 OS << ")";
157}
158
159void Twine::dump() const {
David Greene2b965b02010-01-05 01:28:40 +0000160 print(llvm::dbgs());
Daniel Dunbar2538f7a2009-07-24 07:04:27 +0000161}
162
163void Twine::dumpRepr() const {
David Greene2b965b02010-01-05 01:28:40 +0000164 printRepr(llvm::dbgs());
Daniel Dunbar2538f7a2009-07-24 07:04:27 +0000165}