Run DCE after a LoopFlatten test to reduce spurious output [nfc]
[llvm-project.git] / clang / unittests / Tooling / StencilTest.cpp
blob26257cf2ca3a5f8312d8c64f94e75797aff3e175
1 //===- unittest/Tooling/StencilTest.cpp -----------------------------------===//
2 //
3 // 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
6 //
7 //===----------------------------------------------------------------------===//
9 #include "clang/Tooling/Transformer/Stencil.h"
10 #include "clang/AST/ASTTypeTraits.h"
11 #include "clang/AST/Expr.h"
12 #include "clang/ASTMatchers/ASTMatchers.h"
13 #include "clang/Tooling/FixIt.h"
14 #include "clang/Tooling/Tooling.h"
15 #include "llvm/Support/Error.h"
16 #include "llvm/Testing/Support/Error.h"
17 #include "gmock/gmock.h"
18 #include "gtest/gtest.h"
19 #include <optional>
21 using namespace clang;
22 using namespace transformer;
23 using namespace ast_matchers;
25 namespace {
26 using ::llvm::Failed;
27 using ::llvm::HasValue;
28 using ::llvm::StringError;
29 using ::testing::AllOf;
30 using ::testing::HasSubstr;
31 using MatchResult = MatchFinder::MatchResult;
33 // Create a valid translation-unit from a statement.
34 static std::string wrapSnippet(StringRef ExtraPreface,
35 StringRef StatementCode) {
36 constexpr char Preface[] = R"cc(
37 namespace N { class C {}; }
38 namespace { class AnonC {}; }
39 struct S { int Field; };
40 namespace std {
41 template <typename T>
42 struct unique_ptr {
43 T* operator->() const;
44 T& operator*() const;
47 template<class T> T desugar() { return T(); };
48 )cc";
49 return (Preface + ExtraPreface + "auto stencil_test_snippet = []{" +
50 StatementCode + "};")
51 .str();
54 static DeclarationMatcher wrapMatcher(const StatementMatcher &Matcher) {
55 return varDecl(hasName("stencil_test_snippet"),
56 hasDescendant(compoundStmt(hasAnySubstatement(Matcher))));
59 struct TestMatch {
60 // The AST unit from which `result` is built. We bundle it because it backs
61 // the result. Users are not expected to access it.
62 std::unique_ptr<ASTUnit> AstUnit;
63 // The result to use in the test. References `ast_unit`.
64 MatchResult Result;
67 // Matches `Matcher` against the statement `StatementCode` and returns the
68 // result. Handles putting the statement inside a function and modifying the
69 // matcher correspondingly. `Matcher` should match one of the statements in
70 // `StatementCode` exactly -- that is, produce exactly one match. However,
71 // `StatementCode` may contain other statements not described by `Matcher`.
72 // `ExtraPreface` (optionally) adds extra decls to the TU, before the code.
73 static std::optional<TestMatch> matchStmt(StringRef StatementCode,
74 StatementMatcher Matcher,
75 StringRef ExtraPreface = "") {
76 auto AstUnit = tooling::buildASTFromCodeWithArgs(
77 wrapSnippet(ExtraPreface, StatementCode), {"-Wno-unused-value"});
78 if (AstUnit == nullptr) {
79 ADD_FAILURE() << "AST construction failed";
80 return std::nullopt;
82 ASTContext &Context = AstUnit->getASTContext();
83 auto Matches = ast_matchers::match(wrapMatcher(Matcher), Context);
84 // We expect a single, exact match for the statement.
85 if (Matches.size() != 1) {
86 ADD_FAILURE() << "Wrong number of matches: " << Matches.size();
87 return std::nullopt;
89 return TestMatch{std::move(AstUnit), MatchResult(Matches[0], &Context)};
92 class StencilTest : public ::testing::Test {
93 protected:
94 // Verifies that the given stencil fails when evaluated on a valid match
95 // result. Binds a statement to "stmt", a (non-member) ctor-initializer to
96 // "init", an expression to "expr" and a (nameless) declaration to "decl".
97 void testError(const Stencil &Stencil,
98 ::testing::Matcher<std::string> Matcher) {
99 const std::string Snippet = R"cc(
100 struct A {};
101 class F : public A {
102 public:
103 F(int) {}
105 F(1);
106 )cc";
107 auto StmtMatch = matchStmt(
108 Snippet,
109 stmt(hasDescendant(
110 cxxConstructExpr(
111 hasDeclaration(decl(hasDescendant(cxxCtorInitializer(
112 isBaseInitializer())
113 .bind("init")))
114 .bind("decl")))
115 .bind("expr")))
116 .bind("stmt"));
117 ASSERT_TRUE(StmtMatch);
118 if (auto ResultOrErr = Stencil->eval(StmtMatch->Result)) {
119 ADD_FAILURE() << "Expected failure but succeeded: " << *ResultOrErr;
120 } else {
121 auto Err = llvm::handleErrors(ResultOrErr.takeError(),
122 [&Matcher](const StringError &Err) {
123 EXPECT_THAT(Err.getMessage(), Matcher);
125 if (Err) {
126 ADD_FAILURE() << "Unhandled error: " << llvm::toString(std::move(Err));
131 // Tests failures caused by references to unbound nodes. `unbound_id` is the
132 // id that will cause the failure.
133 void testUnboundNodeError(const Stencil &Stencil, StringRef UnboundId) {
134 testError(Stencil,
135 AllOf(HasSubstr(std::string(UnboundId)), HasSubstr("not bound")));
139 TEST_F(StencilTest, SingleStatement) {
140 StringRef Condition("C"), Then("T"), Else("E");
141 const std::string Snippet = R"cc(
142 if (true)
143 return 1;
144 else
145 return 0;
146 )cc";
147 auto StmtMatch = matchStmt(
148 Snippet, ifStmt(hasCondition(expr().bind(Condition)),
149 hasThen(stmt().bind(Then)), hasElse(stmt().bind(Else))));
150 ASSERT_TRUE(StmtMatch);
151 // Invert the if-then-else.
152 auto Stencil =
153 cat("if (!", node(std::string(Condition)), ") ",
154 statement(std::string(Else)), " else ", statement(std::string(Then)));
155 EXPECT_THAT_EXPECTED(Stencil->eval(StmtMatch->Result),
156 HasValue("if (!true) return 0; else return 1;"));
159 TEST_F(StencilTest, UnboundNode) {
160 const std::string Snippet = R"cc(
161 if (true)
162 return 1;
163 else
164 return 0;
165 )cc";
166 auto StmtMatch = matchStmt(Snippet, ifStmt(hasCondition(stmt().bind("a1")),
167 hasThen(stmt().bind("a2"))));
168 ASSERT_TRUE(StmtMatch);
169 auto Stencil = cat("if(!", node("a1"), ") ", node("UNBOUND"), ";");
170 auto ResultOrErr = Stencil->eval(StmtMatch->Result);
171 EXPECT_TRUE(llvm::errorToBool(ResultOrErr.takeError()))
172 << "Expected unbound node, got " << *ResultOrErr;
175 // Tests that a stencil with a single parameter (`Id`) evaluates to the expected
176 // string, when `Id` is bound to the expression-statement in `Snippet`.
177 void testExpr(StringRef Id, StringRef Snippet, const Stencil &Stencil,
178 StringRef Expected) {
179 auto StmtMatch = matchStmt(Snippet, expr().bind(Id));
180 ASSERT_TRUE(StmtMatch);
181 EXPECT_THAT_EXPECTED(Stencil->eval(StmtMatch->Result),
182 HasValue(std::string(Expected)));
185 void testFailure(StringRef Id, StringRef Snippet, const Stencil &Stencil,
186 testing::Matcher<std::string> MessageMatcher) {
187 auto StmtMatch = matchStmt(Snippet, expr().bind(Id));
188 ASSERT_TRUE(StmtMatch);
189 EXPECT_THAT_EXPECTED(Stencil->eval(StmtMatch->Result),
190 Failed<StringError>(testing::Property(
191 &StringError::getMessage, MessageMatcher)));
194 TEST_F(StencilTest, SelectionOp) {
195 StringRef Id = "id";
196 testExpr(Id, "3;", cat(node(std::string(Id))), "3");
199 TEST_F(StencilTest, IfBoundOpBound) {
200 StringRef Id = "id";
201 testExpr(Id, "3;", ifBound(Id, cat("5"), cat("7")), "5");
204 TEST_F(StencilTest, IfBoundOpUnbound) {
205 StringRef Id = "id";
206 testExpr(Id, "3;", ifBound("other", cat("5"), cat("7")), "7");
209 static auto selectMatcher() {
210 // The `anything` matcher is not bound, to test for none of the cases
211 // matching.
212 return expr(anyOf(integerLiteral().bind("int"), cxxBoolLiteral().bind("bool"),
213 floatLiteral().bind("float"), anything()));
216 static auto selectStencil() {
217 return selectBound({
218 {"int", cat("I")},
219 {"bool", cat("B")},
220 {"bool", cat("redundant")},
221 {"float", cat("F")},
225 TEST_F(StencilTest, SelectBoundChooseDetectedMatch) {
226 std::string Input = "3;";
227 auto StmtMatch = matchStmt(Input, selectMatcher());
228 ASSERT_TRUE(StmtMatch);
229 EXPECT_THAT_EXPECTED(selectStencil()->eval(StmtMatch->Result),
230 HasValue(std::string("I")));
233 TEST_F(StencilTest, SelectBoundChooseFirst) {
234 std::string Input = "true;";
235 auto StmtMatch = matchStmt(Input, selectMatcher());
236 ASSERT_TRUE(StmtMatch);
237 EXPECT_THAT_EXPECTED(selectStencil()->eval(StmtMatch->Result),
238 HasValue(std::string("B")));
241 TEST_F(StencilTest, SelectBoundDiesOnExhaustedCases) {
242 std::string Input = "\"string\";";
243 auto StmtMatch = matchStmt(Input, selectMatcher());
244 ASSERT_TRUE(StmtMatch);
245 EXPECT_THAT_EXPECTED(
246 selectStencil()->eval(StmtMatch->Result),
247 Failed<StringError>(testing::Property(
248 &StringError::getMessage,
249 AllOf(HasSubstr("selectBound failed"), HasSubstr("no default")))));
252 TEST_F(StencilTest, SelectBoundSucceedsWithDefault) {
253 std::string Input = "\"string\";";
254 auto StmtMatch = matchStmt(Input, selectMatcher());
255 ASSERT_TRUE(StmtMatch);
256 auto Stencil = selectBound({{"int", cat("I")}}, cat("D"));
257 EXPECT_THAT_EXPECTED(Stencil->eval(StmtMatch->Result),
258 HasValue(std::string("D")));
261 TEST_F(StencilTest, ExpressionOpNoParens) {
262 StringRef Id = "id";
263 testExpr(Id, "3;", expression(Id), "3");
266 // Don't parenthesize a parens expression.
267 TEST_F(StencilTest, ExpressionOpNoParensParens) {
268 StringRef Id = "id";
269 testExpr(Id, "(3);", expression(Id), "(3)");
272 TEST_F(StencilTest, ExpressionOpBinaryOpParens) {
273 StringRef Id = "id";
274 testExpr(Id, "3+4;", expression(Id), "(3+4)");
277 // `expression` shares code with other ops, so we get sufficient coverage of the
278 // error handling code with this test. If that changes in the future, more error
279 // tests should be added.
280 TEST_F(StencilTest, ExpressionOpUnbound) {
281 StringRef Id = "id";
282 testFailure(Id, "3;", expression("ACACA"),
283 AllOf(HasSubstr("ACACA"), HasSubstr("not bound")));
286 TEST_F(StencilTest, DerefPointer) {
287 StringRef Id = "id";
288 testExpr(Id, "int *x; x;", deref(Id), "*x");
291 TEST_F(StencilTest, DerefBinOp) {
292 StringRef Id = "id";
293 testExpr(Id, "int *x; x + 1;", deref(Id), "*(x + 1)");
296 TEST_F(StencilTest, DerefAddressExpr) {
297 StringRef Id = "id";
298 testExpr(Id, "int x; &x;", deref(Id), "x");
301 TEST_F(StencilTest, AddressOfValue) {
302 StringRef Id = "id";
303 testExpr(Id, "int x; x;", addressOf(Id), "&x");
306 TEST_F(StencilTest, AddressOfDerefExpr) {
307 StringRef Id = "id";
308 testExpr(Id, "int *x; *x;", addressOf(Id), "x");
311 TEST_F(StencilTest, MaybeDerefValue) {
312 StringRef Id = "id";
313 testExpr(Id, "int x; x;", maybeDeref(Id), "x");
316 TEST_F(StencilTest, MaybeDerefPointer) {
317 StringRef Id = "id";
318 testExpr(Id, "int *x; x;", maybeDeref(Id), "*x");
321 TEST_F(StencilTest, MaybeDerefBinOp) {
322 StringRef Id = "id";
323 testExpr(Id, "int *x; x + 1;", maybeDeref(Id), "*(x + 1)");
326 TEST_F(StencilTest, MaybeDerefAddressExpr) {
327 StringRef Id = "id";
328 testExpr(Id, "int x; &x;", maybeDeref(Id), "x");
331 TEST_F(StencilTest, MaybeDerefSmartPointer) {
332 StringRef Id = "id";
333 std::string Snippet = R"cc(
334 std::unique_ptr<S> x;
336 )cc";
337 testExpr(Id, Snippet, maybeDeref(Id), "*x");
340 TEST_F(StencilTest, MaybeDerefSmartPointerFromMemberExpr) {
341 StringRef Id = "id";
342 std::string Snippet = "std::unique_ptr<S> x; x->Field;";
343 auto StmtMatch =
344 matchStmt(Snippet, memberExpr(hasObjectExpression(expr().bind(Id))));
345 ASSERT_TRUE(StmtMatch);
346 const Stencil Stencil = maybeDeref(Id);
347 EXPECT_THAT_EXPECTED(Stencil->eval(StmtMatch->Result), HasValue("*x"));
350 TEST_F(StencilTest, MaybeAddressOfPointer) {
351 StringRef Id = "id";
352 testExpr(Id, "int *x; x;", maybeAddressOf(Id), "x");
355 TEST_F(StencilTest, MaybeAddressOfValue) {
356 StringRef Id = "id";
357 testExpr(Id, "int x; x;", addressOf(Id), "&x");
360 TEST_F(StencilTest, MaybeAddressOfBinOp) {
361 StringRef Id = "id";
362 testExpr(Id, "int x; x + 1;", maybeAddressOf(Id), "&(x + 1)");
365 TEST_F(StencilTest, MaybeAddressOfDerefExpr) {
366 StringRef Id = "id";
367 testExpr(Id, "int *x; *x;", addressOf(Id), "x");
370 TEST_F(StencilTest, MaybeAddressOfSmartPointer) {
371 StringRef Id = "id";
372 testExpr(Id, "std::unique_ptr<S> x; x;", maybeAddressOf(Id), "x");
375 TEST_F(StencilTest, MaybeAddressOfSmartPointerFromMemberCall) {
376 StringRef Id = "id";
377 std::string Snippet = "std::unique_ptr<S> x; x->Field;";
378 auto StmtMatch =
379 matchStmt(Snippet, memberExpr(hasObjectExpression(expr().bind(Id))));
380 ASSERT_TRUE(StmtMatch);
381 const Stencil Stencil = maybeAddressOf(Id);
382 EXPECT_THAT_EXPECTED(Stencil->eval(StmtMatch->Result), HasValue("x"));
385 TEST_F(StencilTest, MaybeAddressOfSmartPointerDerefNoCancel) {
386 StringRef Id = "id";
387 testExpr(Id, "std::unique_ptr<S> x; *x;", maybeAddressOf(Id), "&*x");
390 TEST_F(StencilTest, AccessOpValue) {
391 StringRef Snippet = R"cc(
392 S x;
394 )cc";
395 StringRef Id = "id";
396 testExpr(Id, Snippet, access(Id, "field"), "x.field");
399 TEST_F(StencilTest, AccessOpValueExplicitText) {
400 StringRef Snippet = R"cc(
401 S x;
403 )cc";
404 StringRef Id = "id";
405 testExpr(Id, Snippet, access(Id, cat("field")), "x.field");
408 TEST_F(StencilTest, AccessOpValueAddress) {
409 StringRef Snippet = R"cc(
410 S x;
412 )cc";
413 StringRef Id = "id";
414 testExpr(Id, Snippet, access(Id, "field"), "x.field");
417 TEST_F(StencilTest, AccessOpPointer) {
418 StringRef Snippet = R"cc(
419 S *x;
421 )cc";
422 StringRef Id = "id";
423 testExpr(Id, Snippet, access(Id, "field"), "x->field");
426 TEST_F(StencilTest, AccessOpPointerDereference) {
427 StringRef Snippet = R"cc(
428 S *x;
430 )cc";
431 StringRef Id = "id";
432 testExpr(Id, Snippet, access(Id, "field"), "x->field");
435 TEST_F(StencilTest, AccessOpSmartPointer) {
436 StringRef Snippet = R"cc(
437 std::unique_ptr<S> x;
439 )cc";
440 StringRef Id = "id";
441 testExpr(Id, Snippet, access(Id, "field"), "x->field");
444 TEST_F(StencilTest, AccessOpSmartPointerDereference) {
445 StringRef Snippet = R"cc(
446 std::unique_ptr<S> x;
448 )cc";
449 StringRef Id = "id";
450 testExpr(Id, Snippet, access(Id, "field"), "x->field");
453 TEST_F(StencilTest, AccessOpSmartPointerMemberCall) {
454 StringRef Snippet = R"cc(
455 std::unique_ptr<S> x;
456 x->Field;
457 )cc";
458 StringRef Id = "id";
459 auto StmtMatch =
460 matchStmt(Snippet, memberExpr(hasObjectExpression(expr().bind(Id))));
461 ASSERT_TRUE(StmtMatch);
462 EXPECT_THAT_EXPECTED(access(Id, "field")->eval(StmtMatch->Result),
463 HasValue("x->field"));
466 TEST_F(StencilTest, AccessOpExplicitThis) {
467 using clang::ast_matchers::hasObjectExpression;
468 using clang::ast_matchers::memberExpr;
470 // Set up the code so we can bind to a use of this.
471 StringRef Snippet = R"cc(
472 class C {
473 public:
474 int x;
475 int foo() { return this->x; }
477 )cc";
478 auto StmtMatch = matchStmt(
479 Snippet,
480 traverse(TK_AsIs, returnStmt(hasReturnValue(ignoringImplicit(memberExpr(
481 hasObjectExpression(expr().bind("obj"))))))));
482 ASSERT_TRUE(StmtMatch);
483 const Stencil Stencil = access("obj", "field");
484 EXPECT_THAT_EXPECTED(Stencil->eval(StmtMatch->Result),
485 HasValue("this->field"));
488 TEST_F(StencilTest, AccessOpImplicitThis) {
489 using clang::ast_matchers::hasObjectExpression;
490 using clang::ast_matchers::memberExpr;
492 // Set up the code so we can bind to a use of (implicit) this.
493 StringRef Snippet = R"cc(
494 class C {
495 public:
496 int x;
497 int foo() { return x; }
499 )cc";
500 auto StmtMatch =
501 matchStmt(Snippet, returnStmt(hasReturnValue(ignoringImplicit(memberExpr(
502 hasObjectExpression(expr().bind("obj")))))));
503 ASSERT_TRUE(StmtMatch);
504 const Stencil Stencil = access("obj", "field");
505 EXPECT_THAT_EXPECTED(Stencil->eval(StmtMatch->Result), HasValue("field"));
508 TEST_F(StencilTest, DescribeType) {
509 std::string Snippet = "int *x; x;";
510 std::string Expected = "int *";
511 auto StmtMatch =
512 matchStmt(Snippet, declRefExpr(hasType(qualType().bind("type"))));
513 ASSERT_TRUE(StmtMatch);
514 EXPECT_THAT_EXPECTED(describe("type")->eval(StmtMatch->Result),
515 HasValue(std::string(Expected)));
518 TEST_F(StencilTest, DescribeSugaredType) {
519 std::string Snippet = "using Ty = int; Ty *x; x;";
520 std::string Expected = "Ty *";
521 auto StmtMatch =
522 matchStmt(Snippet, declRefExpr(hasType(qualType().bind("type"))));
523 ASSERT_TRUE(StmtMatch);
524 EXPECT_THAT_EXPECTED(describe("type")->eval(StmtMatch->Result),
525 HasValue(std::string(Expected)));
528 TEST_F(StencilTest, DescribeDeclType) {
529 std::string Snippet = "S s; s;";
530 std::string Expected = "S";
531 auto StmtMatch =
532 matchStmt(Snippet, declRefExpr(hasType(qualType().bind("type"))));
533 ASSERT_TRUE(StmtMatch);
534 EXPECT_THAT_EXPECTED(describe("type")->eval(StmtMatch->Result),
535 HasValue(std::string(Expected)));
538 TEST_F(StencilTest, DescribeQualifiedType) {
539 std::string Snippet = "N::C c; c;";
540 std::string Expected = "N::C";
541 auto StmtMatch =
542 matchStmt(Snippet, declRefExpr(hasType(qualType().bind("type"))));
543 ASSERT_TRUE(StmtMatch);
544 EXPECT_THAT_EXPECTED(describe("type")->eval(StmtMatch->Result),
545 HasValue(std::string(Expected)));
548 TEST_F(StencilTest, DescribeUnqualifiedType) {
549 std::string Snippet = "using N::C; C c; c;";
550 std::string Expected = "C";
551 auto StmtMatch =
552 matchStmt(Snippet, declRefExpr(hasType(qualType().bind("type"))));
553 ASSERT_TRUE(StmtMatch);
554 EXPECT_THAT_EXPECTED(describe("type")->eval(StmtMatch->Result),
555 HasValue(std::string(Expected)));
558 TEST_F(StencilTest, DescribeAnonNamespaceType) {
559 std::string Snippet = "auto c = desugar<AnonC>(); c;";
560 std::string Expected = "(anonymous namespace)::AnonC";
561 auto StmtMatch =
562 matchStmt(Snippet, declRefExpr(hasType(qualType().bind("type"))));
563 ASSERT_TRUE(StmtMatch);
564 EXPECT_THAT_EXPECTED(describe("type")->eval(StmtMatch->Result),
565 HasValue(std::string(Expected)));
568 TEST_F(StencilTest, RunOp) {
569 StringRef Id = "id";
570 auto SimpleFn = [Id](const MatchResult &R) {
571 return std::string(R.Nodes.getNodeAs<Stmt>(Id) != nullptr ? "Bound"
572 : "Unbound");
574 testExpr(Id, "3;", run(SimpleFn), "Bound");
577 TEST_F(StencilTest, CatOfMacroRangeSucceeds) {
578 StringRef Snippet = R"cpp(
579 #define MACRO 3.77
580 double foo(double d);
581 foo(MACRO);)cpp";
583 auto StmtMatch =
584 matchStmt(Snippet, callExpr(callee(functionDecl(hasName("foo"))),
585 argumentCountIs(1),
586 hasArgument(0, expr().bind("arg"))));
587 ASSERT_TRUE(StmtMatch);
588 Stencil S = cat(node("arg"));
589 EXPECT_THAT_EXPECTED(S->eval(StmtMatch->Result), HasValue("MACRO"));
592 TEST_F(StencilTest, CatOfMacroArgRangeSucceeds) {
593 StringRef Snippet = R"cpp(
594 #define MACRO(a, b) a + b
595 MACRO(2, 3);)cpp";
597 auto StmtMatch =
598 matchStmt(Snippet, binaryOperator(hasRHS(expr().bind("rhs"))));
599 ASSERT_TRUE(StmtMatch);
600 Stencil S = cat(node("rhs"));
601 EXPECT_THAT_EXPECTED(S->eval(StmtMatch->Result), HasValue("3"));
604 TEST_F(StencilTest, CatOfMacroArgSubRangeSucceeds) {
605 StringRef Snippet = R"cpp(
606 #define MACRO(a, b) a + b
607 int foo(int);
608 MACRO(2, foo(3));)cpp";
610 auto StmtMatch = matchStmt(
611 Snippet, binaryOperator(hasRHS(callExpr(
612 callee(functionDecl(hasName("foo"))), argumentCountIs(1),
613 hasArgument(0, expr().bind("arg"))))));
614 ASSERT_TRUE(StmtMatch);
615 Stencil S = cat(node("arg"));
616 EXPECT_THAT_EXPECTED(S->eval(StmtMatch->Result), HasValue("3"));
619 TEST_F(StencilTest, CatOfInvalidRangeFails) {
620 StringRef Snippet = R"cpp(
621 #define MACRO (3.77)
622 double foo(double d);
623 foo(MACRO);)cpp";
625 auto StmtMatch =
626 matchStmt(Snippet, callExpr(callee(functionDecl(hasName("foo"))),
627 argumentCountIs(1),
628 hasArgument(0, expr().bind("arg"))));
629 ASSERT_TRUE(StmtMatch);
630 Stencil S = cat(node("arg"));
631 Expected<std::string> Result = S->eval(StmtMatch->Result);
632 ASSERT_FALSE(Result);
633 llvm::handleAllErrors(Result.takeError(), [](const llvm::StringError &E) {
634 EXPECT_THAT(E.getMessage(), AllOf(HasSubstr("selected range"),
635 HasSubstr("macro expansion")));
639 // The `StencilToStringTest` tests verify that the string representation of the
640 // stencil combinator matches (as best possible) the spelling of the
641 // combinator's construction. Exceptions include those combinators that have no
642 // explicit spelling (like raw text) and those supporting non-printable
643 // arguments (like `run`, `selection`).
645 TEST(StencilToStringTest, RawTextOp) {
646 auto S = cat("foo bar baz");
647 StringRef Expected = R"("foo bar baz")";
648 EXPECT_EQ(S->toString(), Expected);
651 TEST(StencilToStringTest, RawTextOpEscaping) {
652 auto S = cat("foo \"bar\" baz\\n");
653 StringRef Expected = R"("foo \"bar\" baz\\n")";
654 EXPECT_EQ(S->toString(), Expected);
657 TEST(StencilToStringTest, DescribeOp) {
658 auto S = describe("Id");
659 StringRef Expected = R"repr(describe("Id"))repr";
660 EXPECT_EQ(S->toString(), Expected);
663 TEST(StencilToStringTest, DebugPrintNodeOp) {
664 auto S = dPrint("Id");
665 StringRef Expected = R"repr(dPrint("Id"))repr";
666 EXPECT_EQ(S->toString(), Expected);
669 TEST(StencilToStringTest, ExpressionOp) {
670 auto S = expression("Id");
671 StringRef Expected = R"repr(expression("Id"))repr";
672 EXPECT_EQ(S->toString(), Expected);
675 TEST(StencilToStringTest, DerefOp) {
676 auto S = deref("Id");
677 StringRef Expected = R"repr(deref("Id"))repr";
678 EXPECT_EQ(S->toString(), Expected);
681 TEST(StencilToStringTest, AddressOfOp) {
682 auto S = addressOf("Id");
683 StringRef Expected = R"repr(addressOf("Id"))repr";
684 EXPECT_EQ(S->toString(), Expected);
687 TEST(StencilToStringTest, SelectionOp) {
688 auto S1 = cat(node("node1"));
689 EXPECT_EQ(S1->toString(), "selection(...)");
692 TEST(StencilToStringTest, AccessOpText) {
693 auto S = access("Id", "memberData");
694 StringRef Expected = R"repr(access("Id", "memberData"))repr";
695 EXPECT_EQ(S->toString(), Expected);
698 TEST(StencilToStringTest, AccessOpSelector) {
699 auto S = access("Id", cat(name("otherId")));
700 StringRef Expected = R"repr(access("Id", selection(...)))repr";
701 EXPECT_EQ(S->toString(), Expected);
704 TEST(StencilToStringTest, AccessOpStencil) {
705 auto S = access("Id", cat("foo_", "bar"));
706 StringRef Expected = R"repr(access("Id", seq("foo_", "bar")))repr";
707 EXPECT_EQ(S->toString(), Expected);
710 TEST(StencilToStringTest, IfBoundOp) {
711 auto S = ifBound("Id", cat("trueText"), access("exprId", "memberData"));
712 StringRef Expected =
713 R"repr(ifBound("Id", "trueText", access("exprId", "memberData")))repr";
714 EXPECT_EQ(S->toString(), Expected);
717 TEST(StencilToStringTest, SelectBoundOp) {
718 auto S = selectBound({
719 {"int", cat("I")},
720 {"float", cat("F")},
722 StringRef Expected = R"repr(selectBound({{"int", "I"}, {"float", "F"}}))repr";
723 EXPECT_EQ(S->toString(), Expected);
726 TEST(StencilToStringTest, SelectBoundOpWithOneCase) {
727 auto S = selectBound({{"int", cat("I")}});
728 StringRef Expected = R"repr(selectBound({{"int", "I"}}))repr";
729 EXPECT_EQ(S->toString(), Expected);
732 TEST(StencilToStringTest, SelectBoundOpWithDefault) {
733 auto S = selectBound({{"int", cat("I")}, {"float", cat("F")}}, cat("D"));
734 StringRef Expected =
735 R"cc(selectBound({{"int", "I"}, {"float", "F"}}, "D"))cc";
736 EXPECT_EQ(S->toString(), Expected);
739 TEST(StencilToStringTest, RunOp) {
740 auto F1 = [](const MatchResult &R) { return "foo"; };
741 auto S1 = run(F1);
742 EXPECT_EQ(S1->toString(), "run(...)");
745 TEST(StencilToStringTest, Sequence) {
746 auto S = cat("foo", access("x", "m()"), "bar",
747 ifBound("x", cat("t"), access("e", "f")));
748 StringRef Expected = R"repr(seq("foo", access("x", "m()"), "bar", )repr"
749 R"repr(ifBound("x", "t", access("e", "f"))))repr";
750 EXPECT_EQ(S->toString(), Expected);
753 TEST(StencilToStringTest, SequenceEmpty) {
754 auto S = cat();
755 StringRef Expected = "seq()";
756 EXPECT_EQ(S->toString(), Expected);
759 TEST(StencilToStringTest, SequenceSingle) {
760 auto S = cat("foo");
761 StringRef Expected = "\"foo\"";
762 EXPECT_EQ(S->toString(), Expected);
765 TEST(StencilToStringTest, SequenceFromVector) {
766 auto S = catVector({cat("foo"), access("x", "m()"), cat("bar"),
767 ifBound("x", cat("t"), access("e", "f"))});
768 StringRef Expected = R"repr(seq("foo", access("x", "m()"), "bar", )repr"
769 R"repr(ifBound("x", "t", access("e", "f"))))repr";
770 EXPECT_EQ(S->toString(), Expected);
772 } // namespace