1 //===- unittest/Tooling/SourceCodeTest.cpp --------------------------------===//
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
7 //===----------------------------------------------------------------------===//
9 #include "clang/Tooling/Transformer/SourceCode.h"
10 #include "TestVisitor.h"
11 #include "clang/Basic/Diagnostic.h"
12 #include "clang/Basic/SourceLocation.h"
13 #include "clang/Lex/Lexer.h"
14 #include "llvm/Testing/Support/Annotations.h"
15 #include "llvm/Testing/Support/Error.h"
16 #include "llvm/Testing/Support/SupportHelpers.h"
17 #include <gmock/gmock.h>
18 #include <gtest/gtest.h>
20 using namespace clang
;
23 using llvm::Succeeded
;
25 using tooling::getAssociatedRange
;
26 using tooling::getExtendedRange
;
27 using tooling::getExtendedText
;
28 using tooling::getRangeForEdit
;
29 using tooling::getText
;
30 using tooling::maybeExtendRange
;
31 using tooling::validateEditRange
;
35 struct IntLitVisitor
: TestVisitor
<IntLitVisitor
> {
36 bool VisitIntegerLiteral(IntegerLiteral
*Expr
) {
37 OnIntLit(Expr
, Context
);
41 std::function
<void(IntegerLiteral
*, ASTContext
*Context
)> OnIntLit
;
44 struct CallsVisitor
: TestVisitor
<CallsVisitor
> {
45 bool VisitCallExpr(CallExpr
*Expr
) {
46 OnCall(Expr
, Context
);
50 std::function
<void(CallExpr
*, ASTContext
*Context
)> OnCall
;
53 // Equality matcher for `clang::CharSourceRange`, which lacks `operator==`.
54 MATCHER_P(EqualsRange
, R
, "") {
55 return arg
.isTokenRange() == R
.isTokenRange() &&
56 arg
.getBegin() == R
.getBegin() && arg
.getEnd() == R
.getEnd();
59 MATCHER_P2(EqualsAnnotatedRange
, Context
, R
, "") {
60 if (arg
.getBegin().isMacroID()) {
61 *result_listener
<< "which starts in a macro";
64 if (arg
.getEnd().isMacroID()) {
65 *result_listener
<< "which ends in a macro";
69 CharSourceRange Range
= Lexer::getAsCharRange(
70 arg
, Context
->getSourceManager(), Context
->getLangOpts());
71 unsigned Begin
= Context
->getSourceManager().getFileOffset(Range
.getBegin());
72 unsigned End
= Context
->getSourceManager().getFileOffset(Range
.getEnd());
74 *result_listener
<< "which is a " << (arg
.isTokenRange() ? "Token" : "Char")
75 << " range [" << Begin
<< "," << End
<< ")";
76 return Begin
== R
.Begin
&& End
== R
.End
;
79 static ::testing::Matcher
<CharSourceRange
> AsRange(const SourceManager
&SM
,
80 llvm::Annotations::Range R
) {
81 return EqualsRange(CharSourceRange::getCharRange(
82 SM
.getLocForStartOfFile(SM
.getMainFileID()).getLocWithOffset(R
.Begin
),
83 SM
.getLocForStartOfFile(SM
.getMainFileID()).getLocWithOffset(R
.End
)));
86 // Base class for visitors that expect a single match corresponding to a
87 // specific annotated range.
88 template <typename T
> class AnnotatedCodeVisitor
: public TestVisitor
<T
> {
91 llvm::Annotations Code
;
94 AnnotatedCodeVisitor() : Code("$r[[]]") {}
95 // Helper for tests of `getAssociatedRange`.
96 bool VisitDeclHelper(Decl
*Decl
) {
97 // Only consider explicit declarations.
98 if (Decl
->isImplicit())
102 EXPECT_THAT(getAssociatedRange(*Decl
, *this->Context
),
103 EqualsAnnotatedRange(this->Context
, Code
.range("r")))
108 bool runOverAnnotated(llvm::StringRef AnnotatedCode
,
109 std::vector
<std::string
> Args
= {}) {
110 Code
= llvm::Annotations(AnnotatedCode
);
112 Args
.push_back("-std=c++11");
113 Args
.push_back("-fno-delayed-template-parsing");
114 bool result
= tooling::runToolOnCodeWithArgs(this->CreateTestAction(),
116 EXPECT_EQ(MatchCount
, 1) << AnnotatedCode
;
121 TEST(SourceCodeTest
, getText
) {
122 CallsVisitor Visitor
;
124 Visitor
.OnCall
= [](CallExpr
*CE
, ASTContext
*Context
) {
125 EXPECT_EQ("foo(x, y)", getText(*CE
, *Context
));
127 Visitor
.runOver("void foo(int x, int y) { foo(x, y); }");
129 Visitor
.OnCall
= [](CallExpr
*CE
, ASTContext
*Context
) {
130 EXPECT_EQ("APPLY(foo, x, y)", getText(*CE
, *Context
));
132 Visitor
.runOver("#define APPLY(f, x, y) f(x, y)\n"
133 "void foo(int x, int y) { APPLY(foo, x, y); }");
136 TEST(SourceCodeTest
, getTextWithMacro
) {
137 CallsVisitor Visitor
;
139 Visitor
.OnCall
= [](CallExpr
*CE
, ASTContext
*Context
) {
140 EXPECT_EQ("F OO", getText(*CE
, *Context
));
141 Expr
*P0
= CE
->getArg(0);
142 Expr
*P1
= CE
->getArg(1);
143 EXPECT_EQ("", getText(*P0
, *Context
));
144 EXPECT_EQ("", getText(*P1
, *Context
));
146 Visitor
.runOver("#define F foo(\n"
148 "void foo(int x, int y) { F OO ; }");
150 Visitor
.OnCall
= [](CallExpr
*CE
, ASTContext
*Context
) {
151 EXPECT_EQ("", getText(*CE
, *Context
));
152 Expr
*P0
= CE
->getArg(0);
153 Expr
*P1
= CE
->getArg(1);
154 EXPECT_EQ("x", getText(*P0
, *Context
));
155 EXPECT_EQ("y", getText(*P1
, *Context
));
157 Visitor
.runOver("#define FOO(x, y) (void)x; (void)y; foo(x, y);\n"
158 "void foo(int x, int y) { FOO(x,y) }");
161 TEST(SourceCodeTest
, getExtendedText
) {
162 CallsVisitor Visitor
;
164 Visitor
.OnCall
= [](CallExpr
*CE
, ASTContext
*Context
) {
165 EXPECT_EQ("foo(x, y);",
166 getExtendedText(*CE
, tok::TokenKind::semi
, *Context
));
168 Expr
*P0
= CE
->getArg(0);
169 Expr
*P1
= CE
->getArg(1);
170 EXPECT_EQ("x", getExtendedText(*P0
, tok::TokenKind::semi
, *Context
));
171 EXPECT_EQ("x,", getExtendedText(*P0
, tok::TokenKind::comma
, *Context
));
172 EXPECT_EQ("y", getExtendedText(*P1
, tok::TokenKind::semi
, *Context
));
174 Visitor
.runOver("void foo(int x, int y) { foo(x, y); }");
175 Visitor
.runOver("void foo(int x, int y) { if (true) foo(x, y); }");
176 Visitor
.runOver("int foo(int x, int y) { if (true) return 3 + foo(x, y); }");
177 Visitor
.runOver("void foo(int x, int y) { for (foo(x, y);;) ++x; }");
179 "bool foo(int x, int y) { for (;foo(x, y);) x = 1; return true; }");
181 Visitor
.OnCall
= [](CallExpr
*CE
, ASTContext
*Context
) {
182 EXPECT_EQ("foo()", getExtendedText(*CE
, tok::TokenKind::semi
, *Context
));
184 Visitor
.runOver("bool foo() { if (foo()) return true; return false; }");
185 Visitor
.runOver("void foo() { int x; for (;; foo()) ++x; }");
186 Visitor
.runOver("int foo() { return foo() + 3; }");
189 TEST(SourceCodeTest
, maybeExtendRange_TokenRange
) {
190 struct ExtendTokenRangeVisitor
191 : AnnotatedCodeVisitor
<ExtendTokenRangeVisitor
> {
192 bool VisitCallExpr(CallExpr
*CE
) {
194 EXPECT_THAT(getExtendedRange(*CE
, tok::TokenKind::semi
, *Context
),
195 EqualsAnnotatedRange(Context
, Code
.range("r")));
200 ExtendTokenRangeVisitor Visitor
;
201 // Extends to include semicolon.
202 Visitor
.runOverAnnotated("void f(int x, int y) { $r[[f(x, y);]] }");
203 // Does not extend to include semicolon.
204 Visitor
.runOverAnnotated(
205 "int f(int x, int y) { if (0) return $r[[f(x, y)]] + 3; }");
208 TEST(SourceCodeTest
, maybeExtendRange_CharRange
) {
209 struct ExtendCharRangeVisitor
: AnnotatedCodeVisitor
<ExtendCharRangeVisitor
> {
210 bool VisitCallExpr(CallExpr
*CE
) {
212 CharSourceRange Call
= Lexer::getAsCharRange(CE
->getSourceRange(),
213 Context
->getSourceManager(),
214 Context
->getLangOpts());
215 EXPECT_THAT(maybeExtendRange(Call
, tok::TokenKind::semi
, *Context
),
216 EqualsAnnotatedRange(Context
, Code
.range("r")));
220 ExtendCharRangeVisitor Visitor
;
221 // Extends to include semicolon.
222 Visitor
.runOverAnnotated("void f(int x, int y) { $r[[f(x, y);]] }");
223 // Does not extend to include semicolon.
224 Visitor
.runOverAnnotated(
225 "int f(int x, int y) { if (0) return $r[[f(x, y)]] + 3; }");
228 TEST(SourceCodeTest
, getAssociatedRange
) {
229 struct VarDeclsVisitor
: AnnotatedCodeVisitor
<VarDeclsVisitor
> {
230 bool VisitVarDecl(VarDecl
*Decl
) { return VisitDeclHelper(Decl
); }
232 VarDeclsVisitor Visitor
;
234 // Includes semicolon.
235 Visitor
.runOverAnnotated("$r[[int x = 4;]]");
237 // Includes newline and semicolon.
238 Visitor
.runOverAnnotated("$r[[int x = 4;\n]]");
240 // Includes trailing comments.
241 Visitor
.runOverAnnotated("$r[[int x = 4; // Comment\n]]");
242 Visitor
.runOverAnnotated("$r[[int x = 4; /* Comment */\n]]");
244 // Does *not* include trailing comments when another entity appears between
245 // the decl and the comment.
246 Visitor
.runOverAnnotated("$r[[int x = 4;]] class C {}; // Comment\n");
248 // Includes attributes.
249 Visitor
.runOverAnnotated(R
"cpp(
250 #define ATTR __attribute__((deprecated("message
")))
254 // Includes attributes and comments together.
255 Visitor
.runOverAnnotated(R
"cpp(
256 #define ATTR __attribute__((deprecated("message
")))
262 TEST(SourceCodeTest
, getAssociatedRangeClasses
) {
263 struct RecordDeclsVisitor
: AnnotatedCodeVisitor
<RecordDeclsVisitor
> {
264 bool VisitRecordDecl(RecordDecl
*Decl
) { return VisitDeclHelper(Decl
); }
266 RecordDeclsVisitor Visitor
;
268 Visitor
.runOverAnnotated("$r[[class A;]]");
269 Visitor
.runOverAnnotated("$r[[class A {};]]");
271 // Includes leading template annotation.
272 Visitor
.runOverAnnotated("$r[[template <typename T> class A;]]");
273 Visitor
.runOverAnnotated("$r[[template <typename T> class A {};]]");
276 TEST(SourceCodeTest
, getAssociatedRangeClassTemplateSpecializations
) {
277 struct CXXRecordDeclsVisitor
: AnnotatedCodeVisitor
<CXXRecordDeclsVisitor
> {
278 bool VisitCXXRecordDecl(CXXRecordDecl
*Decl
) {
279 return Decl
->getTemplateSpecializationKind() !=
280 TSK_ExplicitSpecialization
||
281 VisitDeclHelper(Decl
);
284 CXXRecordDeclsVisitor Visitor
;
286 Visitor
.runOverAnnotated(R
"cpp(
287 template <typename T> class A{};
288 $r[[template <> class A<int>;]])cpp");
289 Visitor
.runOverAnnotated(R
"cpp(
290 template <typename T> class A{};
291 $r[[template <> class A<int> {};]])cpp");
294 TEST(SourceCodeTest
, getAssociatedRangeFunctions
) {
295 struct FunctionDeclsVisitor
: AnnotatedCodeVisitor
<FunctionDeclsVisitor
> {
296 bool VisitFunctionDecl(FunctionDecl
*Decl
) { return VisitDeclHelper(Decl
); }
298 FunctionDeclsVisitor Visitor
;
300 Visitor
.runOverAnnotated("$r[[int f();]]");
301 Visitor
.runOverAnnotated("$r[[int f() { return 0; }]]");
302 // Includes leading template annotation.
303 Visitor
.runOverAnnotated("$r[[template <typename T> int f();]]");
304 Visitor
.runOverAnnotated("$r[[template <typename T> int f() { return 0; }]]");
307 TEST(SourceCodeTest
, getAssociatedRangeMemberTemplates
) {
308 struct CXXMethodDeclsVisitor
: AnnotatedCodeVisitor
<CXXMethodDeclsVisitor
> {
309 bool VisitCXXMethodDecl(CXXMethodDecl
*Decl
) {
310 // Only consider the definition of the template.
311 return !Decl
->doesThisDeclarationHaveABody() || VisitDeclHelper(Decl
);
314 CXXMethodDeclsVisitor Visitor
;
316 Visitor
.runOverAnnotated(R
"cpp(
317 template <typename C>
318 struct A { template <typename T> int member(T v); };
320 $r[[template <typename C>
321 template <typename T>
322 int A<C>::member(T v) { return 0; }]])cpp");
325 TEST(SourceCodeTest
, getAssociatedRangeWithComments
) {
326 struct VarDeclsVisitor
: AnnotatedCodeVisitor
<VarDeclsVisitor
> {
327 bool VisitVarDecl(VarDecl
*Decl
) { return VisitDeclHelper(Decl
); }
330 VarDeclsVisitor Visitor
;
331 auto Visit
= [&](llvm::StringRef AnnotatedCode
) {
332 Visitor
.runOverAnnotated(AnnotatedCode
, {"-fparse-all-comments"});
335 // Includes leading comments.
336 Visit("$r[[// Comment.\nint x = 4;]]");
337 Visit("$r[[// Comment.\nint x = 4;\n]]");
338 Visit("$r[[/* Comment.*/\nint x = 4;\n]]");
339 // ... even if separated by (extra) horizontal whitespace.
340 Visit("$r[[/* Comment.*/ \nint x = 4;\n]]");
342 // Includes comments even in the presence of trailing whitespace.
343 Visit("$r[[// Comment.\nint x = 4;]] ");
345 // Includes comments when the declaration is followed by the beginning or end
346 // of a compound statement.
359 // Includes comments inside macros (when decl is in the same macro).
361 #define DECL /* Comment */ int x
364 // Does not include comments when only the decl or the comment come from a
366 // FIXME: Change code to allow this.
372 #define COMMENT /* Comment */
376 // Includes multi-line comments.
389 // Does not include comments separated by multiple empty lines.
390 Visit("// Comment.\n\n\n$r[[int x = 4;\n]]");
391 Visit("/* Comment.*/\n\n\n$r[[int x = 4;\n]]");
393 // Does not include comments before a *series* of declarations.
397 ]]class foo {};)cpp");
399 // Does not include IfThisThenThat comments
400 Visit("// LINT.IfChange.\n$r[[int x = 4;]]");
401 Visit("// LINT.ThenChange.\n$r[[int x = 4;]]");
403 // Includes attributes.
405 #define ATTR __attribute__((deprecated("message
")))
409 // Includes attributes and comments together.
411 #define ATTR __attribute__((deprecated("message
")))
417 TEST(SourceCodeTest
, getAssociatedRangeInvalidForPartialExpansions
) {
418 struct FailingVarDeclsVisitor
: TestVisitor
<FailingVarDeclsVisitor
> {
419 FailingVarDeclsVisitor() {}
420 bool VisitVarDecl(VarDecl
*Decl
) {
421 EXPECT_TRUE(getAssociatedRange(*Decl
, *Context
).isInvalid());
426 FailingVarDeclsVisitor Visitor
;
427 // Should fail because it only includes a part of the expansion.
428 std::string Code
= R
"cpp(
429 #define DECL class foo { }; int x
431 Visitor
.runOver(Code
);
434 TEST(SourceCodeTest
, EditRangeWithMacroExpansionsShouldSucceed
) {
435 // The call expression, whose range we are extracting, includes two macro
437 llvm::Annotations
Code(R
"cpp(
439 int foo(int x, int y);
440 int a = $r[[foo(M(1), M(2))]];
443 CallsVisitor Visitor
;
445 Visitor
.OnCall
= [&Code
](CallExpr
*CE
, ASTContext
*Context
) {
446 auto Range
= CharSourceRange::getTokenRange(CE
->getSourceRange());
447 EXPECT_THAT(getRangeForEdit(Range
, *Context
),
448 ValueIs(AsRange(Context
->getSourceManager(), Code
.range("r"))));
450 Visitor
.runOver(Code
.code());
453 TEST(SourceCodeTest
, EditWholeMacroExpansionShouldSucceed
) {
454 llvm::Annotations
Code(R
"cpp(
459 IntLitVisitor Visitor
;
460 Visitor
.OnIntLit
= [&Code
](IntegerLiteral
*Expr
, ASTContext
*Context
) {
461 auto Range
= CharSourceRange::getTokenRange(Expr
->getSourceRange());
462 EXPECT_THAT(getRangeForEdit(Range
, *Context
),
463 ValueIs(AsRange(Context
->getSourceManager(), Code
.range("r"))));
465 Visitor
.runOver(Code
.code());
468 TEST(SourceCodeTest
, EditPartialMacroExpansionShouldFail
) {
469 std::string Code
= R
"cpp(
474 IntLitVisitor Visitor
;
475 Visitor
.OnIntLit
= [](IntegerLiteral
*Expr
, ASTContext
*Context
) {
476 auto Range
= CharSourceRange::getTokenRange(Expr
->getSourceRange());
477 EXPECT_FALSE(getRangeForEdit(Range
, *Context
).hasValue());
479 Visitor
.runOver(Code
);
482 TEST(SourceCodeTest
, EditWholeMacroArgShouldSucceed
) {
483 llvm::Annotations
Code(R
"cpp(
484 #define FOO(a) a + 7.0;
485 int a = FOO($r[[10]]);
488 IntLitVisitor Visitor
;
489 Visitor
.OnIntLit
= [&Code
](IntegerLiteral
*Expr
, ASTContext
*Context
) {
490 auto Range
= CharSourceRange::getTokenRange(Expr
->getSourceRange());
491 EXPECT_THAT(getRangeForEdit(Range
, *Context
),
492 ValueIs(AsRange(Context
->getSourceManager(), Code
.range("r"))));
494 Visitor
.runOver(Code
.code());
497 TEST(SourceCodeTest
, EditPartialMacroArgShouldSucceed
) {
498 llvm::Annotations
Code(R
"cpp(
499 #define FOO(a) a + 7.0;
500 int a = FOO($r[[10]] + 10.0);
503 IntLitVisitor Visitor
;
504 Visitor
.OnIntLit
= [&Code
](IntegerLiteral
*Expr
, ASTContext
*Context
) {
505 auto Range
= CharSourceRange::getTokenRange(Expr
->getSourceRange());
506 EXPECT_THAT(getRangeForEdit(Range
, *Context
),
507 ValueIs(AsRange(Context
->getSourceManager(), Code
.range("r"))));
509 Visitor
.runOver(Code
.code());
512 TEST(SourceCodeTest
, EditRangeWithMacroExpansionsIsValid
) {
513 // The call expression, whose range we are extracting, includes two macro
515 llvm::StringRef Code
= R
"cpp(
517 int foo(int x, int y);
518 int a = foo(M(1), M(2));
521 CallsVisitor Visitor
;
523 Visitor
.OnCall
= [](CallExpr
*CE
, ASTContext
*Context
) {
524 auto Range
= CharSourceRange::getTokenRange(CE
->getSourceRange());
525 EXPECT_THAT_ERROR(validateEditRange(Range
, Context
->getSourceManager()),
528 Visitor
.runOver(Code
);
531 TEST(SourceCodeTest
, SpellingRangeOfMacroArgIsValid
) {
532 llvm::StringRef Code
= R
"cpp(
533 #define FOO(a) a + 7.0;
537 IntLitVisitor Visitor
;
538 Visitor
.OnIntLit
= [](IntegerLiteral
*Expr
, ASTContext
*Context
) {
539 SourceLocation ArgLoc
=
540 Context
->getSourceManager().getSpellingLoc(Expr
->getBeginLoc());
541 // The integer literal is a single token.
542 auto ArgRange
= CharSourceRange::getTokenRange(ArgLoc
);
543 EXPECT_THAT_ERROR(validateEditRange(ArgRange
, Context
->getSourceManager()),
546 Visitor
.runOver(Code
);
549 TEST(SourceCodeTest
, InvalidEditRangeIsInvalid
) {
550 llvm::StringRef Code
= "int c = 10;";
552 // We use the visitor just to get a valid context.
553 IntLitVisitor Visitor
;
554 Visitor
.OnIntLit
= [](IntegerLiteral
*, ASTContext
*Context
) {
555 CharSourceRange Invalid
;
556 EXPECT_THAT_ERROR(validateEditRange(Invalid
, Context
->getSourceManager()),
559 Visitor
.runOver(Code
);
562 TEST(SourceCodeTest
, InvertedEditRangeIsInvalid
) {
563 llvm::StringRef Code
= R
"cpp(
568 CallsVisitor Visitor
;
569 Visitor
.OnCall
= [](CallExpr
*Expr
, ASTContext
*Context
) {
570 auto InvertedRange
= CharSourceRange::getTokenRange(
571 SourceRange(Expr
->getEndLoc(), Expr
->getBeginLoc()));
573 validateEditRange(InvertedRange
, Context
->getSourceManager()),
576 Visitor
.runOver(Code
);
579 TEST(SourceCodeTest
, MacroArgIsInvalid
) {
580 llvm::StringRef Code
= R
"cpp(
581 #define FOO(a) a + 7.0;
585 IntLitVisitor Visitor
;
586 Visitor
.OnIntLit
= [](IntegerLiteral
*Expr
, ASTContext
*Context
) {
587 auto Range
= CharSourceRange::getTokenRange(Expr
->getSourceRange());
588 EXPECT_THAT_ERROR(validateEditRange(Range
, Context
->getSourceManager()),
591 Visitor
.runOver(Code
);
594 TEST(SourceCodeTest
, EditWholeMacroExpansionIsInvalid
) {
595 llvm::StringRef Code
= R
"cpp(
600 IntLitVisitor Visitor
;
601 Visitor
.OnIntLit
= [](IntegerLiteral
*Expr
, ASTContext
*Context
) {
602 auto Range
= CharSourceRange::getTokenRange(Expr
->getSourceRange());
603 EXPECT_THAT_ERROR(validateEditRange(Range
, Context
->getSourceManager()),
607 Visitor
.runOver(Code
);
610 TEST(SourceCodeTest
, EditPartialMacroExpansionIsInvalid
) {
611 llvm::StringRef Code
= R
"cpp(
616 IntLitVisitor Visitor
;
617 Visitor
.OnIntLit
= [](IntegerLiteral
*Expr
, ASTContext
*Context
) {
618 auto Range
= CharSourceRange::getTokenRange(Expr
->getSourceRange());
619 EXPECT_THAT_ERROR(validateEditRange(Range
, Context
->getSourceManager()),
622 Visitor
.runOver(Code
);
624 } // end anonymous namespace