2009-07-24 15:04:27 +08:00
|
|
|
//===- TwineTest.cpp - Twine unit tests -----------------------------------===//
|
|
|
|
//
|
2019-01-19 16:50:56 +08:00
|
|
|
// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
|
|
|
|
// See https://llvm.org/LICENSE.txt for license information.
|
|
|
|
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
|
2009-07-24 15:04:27 +08:00
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
2016-12-17 08:38:15 +08:00
|
|
|
#include "llvm/ADT/Twine.h"
|
2017-06-06 19:06:56 +08:00
|
|
|
#include "llvm/ADT/SmallString.h"
|
2016-12-17 08:38:15 +08:00
|
|
|
#include "llvm/Support/FormatAdapters.h"
|
|
|
|
#include "llvm/Support/FormatVariadic.h"
|
2009-07-24 15:04:27 +08:00
|
|
|
#include "llvm/Support/raw_ostream.h"
|
2012-12-04 18:23:08 +08:00
|
|
|
#include "gtest/gtest.h"
|
2009-07-24 15:04:27 +08:00
|
|
|
using namespace llvm;
|
|
|
|
|
|
|
|
namespace {
|
|
|
|
|
|
|
|
std::string repr(const Twine &Value) {
|
|
|
|
std::string res;
|
|
|
|
llvm::raw_string_ostream OS(res);
|
|
|
|
Value.printRepr(OS);
|
|
|
|
return OS.str();
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST(TwineTest, Construction) {
|
|
|
|
EXPECT_EQ("", Twine().str());
|
|
|
|
EXPECT_EQ("hi", Twine("hi").str());
|
|
|
|
EXPECT_EQ("hi", Twine(std::string("hi")).str());
|
|
|
|
EXPECT_EQ("hi", Twine(StringRef("hi")).str());
|
|
|
|
EXPECT_EQ("hi", Twine(StringRef(std::string("hi"))).str());
|
|
|
|
EXPECT_EQ("hi", Twine(StringRef("hithere", 2)).str());
|
2015-03-17 17:51:17 +08:00
|
|
|
EXPECT_EQ("hi", Twine(SmallString<4>("hi")).str());
|
2016-12-17 08:38:15 +08:00
|
|
|
EXPECT_EQ("hi", Twine(formatv("{0}", "hi")).str());
|
2009-07-24 15:04:27 +08:00
|
|
|
}
|
|
|
|
|
2009-07-29 15:08:44 +08:00
|
|
|
TEST(TwineTest, Numbers) {
|
2009-07-30 11:47:15 +08:00
|
|
|
EXPECT_EQ("123", Twine(123U).str());
|
|
|
|
EXPECT_EQ("123", Twine(123).str());
|
|
|
|
EXPECT_EQ("-123", Twine(-123).str());
|
|
|
|
EXPECT_EQ("123", Twine(123).str());
|
|
|
|
EXPECT_EQ("-123", Twine(-123).str());
|
2009-07-29 15:08:44 +08:00
|
|
|
|
2009-07-31 02:30:19 +08:00
|
|
|
EXPECT_EQ("7b", Twine::utohexstr(123).str());
|
2009-07-29 15:08:44 +08:00
|
|
|
}
|
|
|
|
|
2011-07-25 04:44:30 +08:00
|
|
|
TEST(TwineTest, Characters) {
|
|
|
|
EXPECT_EQ("x", Twine('x').str());
|
|
|
|
EXPECT_EQ("x", Twine(static_cast<unsigned char>('x')).str());
|
|
|
|
EXPECT_EQ("x", Twine(static_cast<signed char>('x')).str());
|
|
|
|
}
|
|
|
|
|
2009-07-24 15:04:27 +08:00
|
|
|
TEST(TwineTest, Concat) {
|
|
|
|
// Check verse repr, since we care about the actual representation not just
|
|
|
|
// the result.
|
|
|
|
|
|
|
|
// Concat with null.
|
|
|
|
EXPECT_EQ("(Twine null empty)",
|
|
|
|
repr(Twine("hi").concat(Twine::createNull())));
|
|
|
|
EXPECT_EQ("(Twine null empty)",
|
|
|
|
repr(Twine::createNull().concat(Twine("hi"))));
|
|
|
|
|
|
|
|
// Concat with empty.
|
|
|
|
EXPECT_EQ("(Twine cstring:\"hi\" empty)",
|
|
|
|
repr(Twine("hi").concat(Twine())));
|
|
|
|
EXPECT_EQ("(Twine cstring:\"hi\" empty)",
|
|
|
|
repr(Twine().concat(Twine("hi"))));
|
2015-03-17 17:51:17 +08:00
|
|
|
EXPECT_EQ("(Twine smallstring:\"hi\" empty)",
|
|
|
|
repr(Twine().concat(Twine(SmallString<5>("hi")))));
|
2016-12-17 08:38:15 +08:00
|
|
|
EXPECT_EQ("(Twine formatv:\"howdy\" empty)",
|
|
|
|
repr(Twine(formatv("howdy")).concat(Twine())));
|
|
|
|
EXPECT_EQ("(Twine formatv:\"howdy\" empty)",
|
|
|
|
repr(Twine().concat(Twine(formatv("howdy")))));
|
2015-03-17 17:51:17 +08:00
|
|
|
EXPECT_EQ("(Twine smallstring:\"hey\" cstring:\"there\")",
|
|
|
|
repr(Twine(SmallString<7>("hey")).concat(Twine("there"))));
|
2009-07-24 15:04:27 +08:00
|
|
|
|
|
|
|
// Concatenation of unary ropes.
|
|
|
|
EXPECT_EQ("(Twine cstring:\"a\" cstring:\"b\")",
|
|
|
|
repr(Twine("a").concat(Twine("b"))));
|
|
|
|
|
|
|
|
// Concatenation of other ropes.
|
|
|
|
EXPECT_EQ("(Twine rope:(Twine cstring:\"a\" cstring:\"b\") cstring:\"c\")",
|
|
|
|
repr(Twine("a").concat(Twine("b")).concat(Twine("c"))));
|
|
|
|
EXPECT_EQ("(Twine cstring:\"a\" rope:(Twine cstring:\"b\" cstring:\"c\"))",
|
|
|
|
repr(Twine("a").concat(Twine("b").concat(Twine("c")))));
|
2015-03-17 17:51:17 +08:00
|
|
|
EXPECT_EQ("(Twine cstring:\"a\" rope:(Twine smallstring:\"b\" cstring:\"c\"))",
|
|
|
|
repr(Twine("a").concat(Twine(SmallString<3>("b")).concat(Twine("c")))));
|
2009-07-24 15:04:27 +08:00
|
|
|
}
|
|
|
|
|
2010-12-02 04:37:30 +08:00
|
|
|
TEST(TwineTest, toNullTerminatedStringRef) {
|
|
|
|
SmallString<8> storage;
|
|
|
|
EXPECT_EQ(0, *Twine("hello").toNullTerminatedStringRef(storage).end());
|
|
|
|
EXPECT_EQ(0,
|
|
|
|
*Twine(StringRef("hello")).toNullTerminatedStringRef(storage).end());
|
2015-03-17 17:51:17 +08:00
|
|
|
EXPECT_EQ(0, *Twine(SmallString<11>("hello"))
|
|
|
|
.toNullTerminatedStringRef(storage)
|
|
|
|
.end());
|
2016-12-17 08:38:15 +08:00
|
|
|
EXPECT_EQ(0, *Twine(formatv("{0}{1}", "how", "dy"))
|
|
|
|
.toNullTerminatedStringRef(storage)
|
|
|
|
.end());
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST(TwineTest, LazyEvaluation) {
|
|
|
|
struct formatter : FormatAdapter<int> {
|
2016-12-17 09:31:46 +08:00
|
|
|
explicit formatter(int &Count) : FormatAdapter(0), Count(Count) {}
|
2016-12-17 08:38:15 +08:00
|
|
|
int &Count;
|
|
|
|
|
2020-07-17 11:36:46 +08:00
|
|
|
void format(raw_ostream &OS, StringRef Style) override { ++Count; }
|
2016-12-17 08:38:15 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
int Count = 0;
|
|
|
|
formatter Formatter(Count);
|
|
|
|
(void)Twine(formatv("{0}", Formatter));
|
|
|
|
EXPECT_EQ(0, Count);
|
|
|
|
(void)Twine(formatv("{0}", Formatter)).str();
|
|
|
|
EXPECT_EQ(1, Count);
|
2010-12-02 04:37:30 +08:00
|
|
|
}
|
|
|
|
|
2009-07-24 15:04:27 +08:00
|
|
|
// I suppose linking in the entire code generator to add a unit test to check
|
|
|
|
// the code size of the concat operation is overkill... :)
|
|
|
|
|
|
|
|
} // end anonymous namespace
|