blob: fbbcd8848f1cd26a3e273bc1eb2fe6e971b22ed3 [file] [log] [blame]
Daniel Dunbarafcf5b32009-07-24 07:04:27 +00001//===-- Twine.cpp - Fast Temporary String Concatenation -------------------===//
2//
Chandler Carruth2946cd72019-01-19 08:50:56 +00003// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4// See https://llvm.org/LICENSE.txt for license information.
5// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
Daniel Dunbarafcf5b32009-07-24 07:04:27 +00006//
7//===----------------------------------------------------------------------===//
8
9#include "llvm/ADT/Twine.h"
Daniel Dunbarea579e32009-08-19 18:09:47 +000010#include "llvm/ADT/SmallString.h"
Nico Weber432a3882018-04-30 14:59:11 +000011#include "llvm/Config/llvm-config.h"
David Greenef92e6712010-01-05 01:28:40 +000012#include "llvm/Support/Debug.h"
Zachary Turnerde4be352016-12-17 00:38:15 +000013#include "llvm/Support/FormatVariadic.h"
Daniel Dunbarafcf5b32009-07-24 07:04:27 +000014#include "llvm/Support/raw_ostream.h"
15using namespace llvm;
16
17std::string Twine::str() const {
Frits van Bommelf8bf4c22011-07-15 11:05:37 +000018 // If we're storing only a std::string, just return it.
19 if (LHSKind == StdStringKind && RHSKind == EmptyKind)
Chris Lattner9650f062011-07-24 20:44:30 +000020 return *LHS.stdString;
Frits van Bommelf8bf4c22011-07-15 11:05:37 +000021
Zachary Turnerde4be352016-12-17 00:38:15 +000022 // If we're storing a formatv_object, we can avoid an extra copy by formatting
23 // it immediately and returning the result.
24 if (LHSKind == FormatvObjectKind && RHSKind == EmptyKind)
25 return LHS.formatvObject->str();
26
Frits van Bommelf8bf4c22011-07-15 11:05:37 +000027 // Otherwise, flatten and copy the contents first.
Daniel Dunbarea579e32009-08-19 18:09:47 +000028 SmallString<256> Vec;
Benjamin Kramer2e06b932010-01-13 12:45:23 +000029 return toStringRef(Vec).str();
Daniel Dunbarafcf5b32009-07-24 07:04:27 +000030}
31
32void Twine::toVector(SmallVectorImpl<char> &Out) const {
33 raw_svector_ostream OS(Out);
34 print(OS);
35}
36
Michael J. Spencerf2cc8282010-12-01 20:37:30 +000037StringRef Twine::toNullTerminatedStringRef(SmallVectorImpl<char> &Out) const {
Michael J. Spencer73e60d02010-12-03 05:42:25 +000038 if (isUnary()) {
39 switch (getLHSKind()) {
40 case CStringKind:
41 // Already null terminated, yay!
Chris Lattner9650f062011-07-24 20:44:30 +000042 return StringRef(LHS.cString);
Michael J. Spencer73e60d02010-12-03 05:42:25 +000043 case StdStringKind: {
Chris Lattner9650f062011-07-24 20:44:30 +000044 const std::string *str = LHS.stdString;
Frits van Bommelf8bf4c22011-07-15 11:05:37 +000045 return StringRef(str->c_str(), str->size());
46 }
Michael J. Spencer73e60d02010-12-03 05:42:25 +000047 default:
48 break;
49 }
Michael J. Spencerf2cc8282010-12-01 20:37:30 +000050 }
51 toVector(Out);
52 Out.push_back(0);
53 Out.pop_back();
54 return StringRef(Out.data(), Out.size());
55}
56
Chris Lattner9650f062011-07-24 20:44:30 +000057void Twine::printOneChild(raw_ostream &OS, Child Ptr,
Daniel Dunbarafcf5b32009-07-24 07:04:27 +000058 NodeKind Kind) const {
59 switch (Kind) {
60 case Twine::NullKind: break;
61 case Twine::EmptyKind: break;
Daniel Dunbarb49994a2009-07-29 07:08:44 +000062 case Twine::TwineKind:
Chris Lattner9650f062011-07-24 20:44:30 +000063 Ptr.twine->print(OS);
Daniel Dunbarb49994a2009-07-29 07:08:44 +000064 break;
Michael J. Spencerf13f4422010-11-26 04:16:08 +000065 case Twine::CStringKind:
Chris Lattner9650f062011-07-24 20:44:30 +000066 OS << Ptr.cString;
Daniel Dunbarafcf5b32009-07-24 07:04:27 +000067 break;
68 case Twine::StdStringKind:
Chris Lattner9650f062011-07-24 20:44:30 +000069 OS << *Ptr.stdString;
Daniel Dunbarafcf5b32009-07-24 07:04:27 +000070 break;
71 case Twine::StringRefKind:
Chris Lattner9650f062011-07-24 20:44:30 +000072 OS << *Ptr.stringRef;
73 break;
Yaron Keren1ee89fc2015-03-17 09:51:17 +000074 case Twine::SmallStringKind:
75 OS << *Ptr.smallString;
76 break;
Zachary Turnerde4be352016-12-17 00:38:15 +000077 case Twine::FormatvObjectKind:
78 OS << *Ptr.formatvObject;
79 break;
Chris Lattner9650f062011-07-24 20:44:30 +000080 case Twine::CharKind:
81 OS << Ptr.character;
Daniel Dunbarafcf5b32009-07-24 07:04:27 +000082 break;
Daniel Dunbara91dd9b2009-07-30 21:15:14 +000083 case Twine::DecUIKind:
Chris Lattner9650f062011-07-24 20:44:30 +000084 OS << Ptr.decUI;
Daniel Dunbare8b32362009-07-30 03:47:15 +000085 break;
Daniel Dunbara91dd9b2009-07-30 21:15:14 +000086 case Twine::DecIKind:
Chris Lattner9650f062011-07-24 20:44:30 +000087 OS << Ptr.decI;
Daniel Dunbare8b32362009-07-30 03:47:15 +000088 break;
Daniel Dunbara91dd9b2009-07-30 21:15:14 +000089 case Twine::DecULKind:
Chris Lattner9650f062011-07-24 20:44:30 +000090 OS << *Ptr.decUL;
Daniel Dunbarb49994a2009-07-29 07:08:44 +000091 break;
Daniel Dunbara91dd9b2009-07-30 21:15:14 +000092 case Twine::DecLKind:
Chris Lattner9650f062011-07-24 20:44:30 +000093 OS << *Ptr.decL;
Daniel Dunbara91dd9b2009-07-30 21:15:14 +000094 break;
95 case Twine::DecULLKind:
Chris Lattner9650f062011-07-24 20:44:30 +000096 OS << *Ptr.decULL;
Daniel Dunbara91dd9b2009-07-30 21:15:14 +000097 break;
98 case Twine::DecLLKind:
Chris Lattner9650f062011-07-24 20:44:30 +000099 OS << *Ptr.decLL;
Daniel Dunbarb49994a2009-07-29 07:08:44 +0000100 break;
101 case Twine::UHexKind:
Chris Lattner9650f062011-07-24 20:44:30 +0000102 OS.write_hex(*Ptr.uHex);
Daniel Dunbarafcf5b32009-07-24 07:04:27 +0000103 break;
104 }
105}
106
Chris Lattner9650f062011-07-24 20:44:30 +0000107void Twine::printOneChildRepr(raw_ostream &OS, Child Ptr,
Daniel Dunbarafcf5b32009-07-24 07:04:27 +0000108 NodeKind Kind) const {
109 switch (Kind) {
110 case Twine::NullKind:
111 OS << "null"; break;
112 case Twine::EmptyKind:
113 OS << "empty"; break;
Daniel Dunbarafcf5b32009-07-24 07:04:27 +0000114 case Twine::TwineKind:
115 OS << "rope:";
Chris Lattner9650f062011-07-24 20:44:30 +0000116 Ptr.twine->printRepr(OS);
Daniel Dunbarafcf5b32009-07-24 07:04:27 +0000117 break;
Daniel Dunbarb49994a2009-07-29 07:08:44 +0000118 case Twine::CStringKind:
119 OS << "cstring:\""
Chris Lattner9650f062011-07-24 20:44:30 +0000120 << Ptr.cString << "\"";
Daniel Dunbarb49994a2009-07-29 07:08:44 +0000121 break;
122 case Twine::StdStringKind:
Zachary Turner41a9ee92017-10-11 23:54:34 +0000123 OS << "std::string:\""
124 << Ptr.stdString << "\"";
Daniel Dunbarb49994a2009-07-29 07:08:44 +0000125 break;
126 case Twine::StringRefKind:
Zachary Turner41a9ee92017-10-11 23:54:34 +0000127 OS << "stringref:\""
128 << Ptr.stringRef << "\"";
Chris Lattner9650f062011-07-24 20:44:30 +0000129 break;
NAKAMURA Takumifb3bd712015-05-25 01:43:23 +0000130 case Twine::SmallStringKind:
NAKAMURA Takumi5582a6a2015-05-25 01:43:34 +0000131 OS << "smallstring:\"" << *Ptr.smallString << "\"";
NAKAMURA Takumifb3bd712015-05-25 01:43:23 +0000132 break;
Zachary Turnerde4be352016-12-17 00:38:15 +0000133 case Twine::FormatvObjectKind:
134 OS << "formatv:\"" << *Ptr.formatvObject << "\"";
135 break;
Chris Lattner9650f062011-07-24 20:44:30 +0000136 case Twine::CharKind:
137 OS << "char:\"" << Ptr.character << "\"";
Daniel Dunbarb49994a2009-07-29 07:08:44 +0000138 break;
Daniel Dunbara91dd9b2009-07-30 21:15:14 +0000139 case Twine::DecUIKind:
Chris Lattner9650f062011-07-24 20:44:30 +0000140 OS << "decUI:\"" << Ptr.decUI << "\"";
Daniel Dunbarb49994a2009-07-29 07:08:44 +0000141 break;
Daniel Dunbara91dd9b2009-07-30 21:15:14 +0000142 case Twine::DecIKind:
Chris Lattner9650f062011-07-24 20:44:30 +0000143 OS << "decI:\"" << Ptr.decI << "\"";
Daniel Dunbare8b32362009-07-30 03:47:15 +0000144 break;
Daniel Dunbara91dd9b2009-07-30 21:15:14 +0000145 case Twine::DecULKind:
Chris Lattner9650f062011-07-24 20:44:30 +0000146 OS << "decUL:\"" << *Ptr.decUL << "\"";
Daniel Dunbare8b32362009-07-30 03:47:15 +0000147 break;
Daniel Dunbara91dd9b2009-07-30 21:15:14 +0000148 case Twine::DecLKind:
Chris Lattner9650f062011-07-24 20:44:30 +0000149 OS << "decL:\"" << *Ptr.decL << "\"";
Daniel Dunbara91dd9b2009-07-30 21:15:14 +0000150 break;
151 case Twine::DecULLKind:
Chris Lattner9650f062011-07-24 20:44:30 +0000152 OS << "decULL:\"" << *Ptr.decULL << "\"";
Daniel Dunbara91dd9b2009-07-30 21:15:14 +0000153 break;
154 case Twine::DecLLKind:
Chris Lattner9650f062011-07-24 20:44:30 +0000155 OS << "decLL:\"" << *Ptr.decLL << "\"";
Daniel Dunbarb49994a2009-07-29 07:08:44 +0000156 break;
157 case Twine::UHexKind:
Chris Lattner9650f062011-07-24 20:44:30 +0000158 OS << "uhex:\"" << Ptr.uHex << "\"";
Daniel Dunbarb49994a2009-07-29 07:08:44 +0000159 break;
Daniel Dunbarafcf5b32009-07-24 07:04:27 +0000160 }
161}
162
163void Twine::print(raw_ostream &OS) const {
164 printOneChild(OS, LHS, getLHSKind());
165 printOneChild(OS, RHS, getRHSKind());
166}
167
168void Twine::printRepr(raw_ostream &OS) const {
169 OS << "(Twine ";
170 printOneChildRepr(OS, LHS, getLHSKind());
171 OS << " ";
172 printOneChildRepr(OS, RHS, getRHSKind());
173 OS << ")";
174}
175
Aaron Ballman615eb472017-10-15 14:32:27 +0000176#if !defined(NDEBUG) || defined(LLVM_ENABLE_DUMP)
Yaron Kereneb2a2542016-01-29 20:50:44 +0000177LLVM_DUMP_METHOD void Twine::dump() const {
Eric Christophera13839f2014-02-26 23:27:16 +0000178 print(dbgs());
Daniel Dunbarafcf5b32009-07-24 07:04:27 +0000179}
180
Matthias Braun8c209aa2017-01-28 02:02:38 +0000181LLVM_DUMP_METHOD void Twine::dumpRepr() const {
Eric Christophera13839f2014-02-26 23:27:16 +0000182 printRepr(dbgs());
Daniel Dunbarafcf5b32009-07-24 07:04:27 +0000183}
Matthias Braun8c209aa2017-01-28 02:02:38 +0000184#endif