1 //===--- PreambleTests.cpp --------------------------------------*- C++ -*-===//
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 "Annotations.h"
12 #include "Diagnostics.h"
15 #include "ParsedAST.h"
18 #include "SourceCode.h"
22 #include "support/Context.h"
23 #include "clang/Basic/SourceManager.h"
24 #include "clang/Format/Format.h"
25 #include "clang/Frontend/FrontendActions.h"
26 #include "clang/Frontend/PrecompiledPreamble.h"
27 #include "llvm/ADT/StringMap.h"
28 #include "llvm/ADT/StringRef.h"
29 #include "llvm/Support/Error.h"
30 #include "llvm/Support/MemoryBuffer.h"
31 #include "llvm/Support/ScopedPrinter.h"
32 #include "llvm/Support/VirtualFileSystem.h"
33 #include "llvm/Testing/Annotations/Annotations.h"
34 #include "gmock/gmock.h"
35 #include "gtest/gtest-matchers.h"
36 #include "gtest/gtest.h"
44 using testing::Contains
;
45 using testing::ElementsAre
;
47 using testing::IsEmpty
;
48 using testing::Matcher
;
49 using testing::MatchesRegex
;
50 using testing::UnorderedElementsAre
;
51 using testing::UnorderedElementsAreArray
;
57 MATCHER_P2(Distance
, File
, D
, "") {
58 return arg
.first() == File
&& arg
.second
== D
;
61 // Builds a preamble for BaselineContents, patches it for ModifiedContents and
62 // returns the includes in the patch.
64 collectPatchedIncludes(llvm::StringRef ModifiedContents
,
65 llvm::StringRef BaselineContents
,
66 llvm::StringRef MainFileName
= "main.cpp") {
68 auto TU
= TestTU::withCode(BaselineContents
);
69 TU
.Filename
= MainFileName
.str();
70 // ms-compatibility changes meaning of #import, make sure it is turned off.
71 TU
.ExtraArgs
= {"-fno-ms-compatibility"};
72 auto BaselinePreamble
= TU
.preamble();
74 TU
.Code
= ModifiedContents
.str();
75 auto PI
= TU
.inputs(FS
);
76 auto PP
= PreamblePatch::createFullPatch(testPath(TU
.Filename
), PI
,
78 // Collect patch contents.
79 IgnoreDiagnostics Diags
;
80 auto CI
= buildCompilerInvocation(PI
, Diags
);
82 // Run preprocessor over the modified contents with patched Invocation. We
83 // provide a preamble and trim contents to ensure only the implicit header
84 // introduced by the patch is parsed and nothing else.
85 // We don't run PP directly over the patch cotents to test production
87 auto Bounds
= Lexer::ComputePreamble(ModifiedContents
, CI
->getLangOpts());
89 prepareCompilerInstance(std::move(CI
), &BaselinePreamble
->Preamble
,
90 llvm::MemoryBuffer::getMemBufferCopy(
91 ModifiedContents
.slice(0, Bounds
.Size
).str()),
92 PI
.TFS
->view(PI
.CompileCommand
.Directory
), Diags
);
93 PreprocessOnlyAction Action
;
94 if (!Action
.BeginSourceFile(*Clang
, Clang
->getFrontendOpts().Inputs
[0])) {
95 ADD_FAILURE() << "failed begin source file";
98 IncludeStructure Includes
;
99 Includes
.collect(*Clang
);
100 if (llvm::Error Err
= Action
.Execute()) {
101 ADD_FAILURE() << "failed to execute action: " << std::move(Err
);
104 Action
.EndSourceFile();
108 // Check preamble lexing logic by building an empty preamble and patching it
109 // with all the contents.
110 TEST(PreamblePatchTest
, IncludeParsing
) {
111 // We expect any line with a point to show up in the patch.
112 llvm::StringRef Cases
[] = {
114 R
"cpp(^#include "a
.h
")cpp",
115 // Both preamble and mainfile
118 garbage, finishes preamble
134 ^#include <b.h>)cpp",
135 // Directive is not part of preamble if it is not the token immediately
136 // followed by the hash (#).
139 #/**/include <b.h>)cpp",
142 for (const auto &Case : Cases) {
143 Annotations Test(Case);
144 const auto Code = Test.code();
148 collectPatchedIncludes(Code, /*BaselineContents=*/"").MainFileIncludes;
149 auto Points = Test.points();
150 ASSERT_EQ(Includes.size(), Points.size());
151 for (size_t I = 0, E = Includes.size(); I != E; ++I)
152 EXPECT_EQ(Includes[I].HashLine, Points[I].line);
156 TEST(PreamblePatchTest, ContainsNewIncludes) {
157 constexpr llvm::StringLiteral BaselineContents = R"cpp(
159 #include <b.h> // This will be removed
162 constexpr llvm::StringLiteral ModifiedContents = R"cpp(
164 #include <c.h> // This has changed a line.
165 #include <c.h> // This is a duplicate.
166 #include <d.h> // This is newly introduced.
168 auto Includes = collectPatchedIncludes(ModifiedContents, BaselineContents)
170 EXPECT_THAT(Includes, ElementsAre(AllOf(Field(&Inclusion::Written, "<d
.h
>"),
171 Field(&Inclusion::HashLine, 4))));
174 TEST(PreamblePatchTest, MainFileIsEscaped) {
175 auto Includes = collectPatchedIncludes("#include <a.h>", "", "file\"name.cpp")
177 EXPECT_THAT(Includes, ElementsAre(AllOf(Field(&Inclusion::Written, "<a.h>"),
178 Field(&Inclusion::HashLine, 0))));
181 TEST(PreamblePatchTest, PatchesPreambleIncludes) {
183 IgnoreDiagnostics Diags;
184 auto TU = TestTU::withCode(R"cpp(
185 #include "a.h" // IWYU pragma: keep
191 TU.AdditionalFiles["a
.h
"] = "#include \"b.h\"";
192 TU.AdditionalFiles["b.h"] = "";
193 TU.AdditionalFiles["c.h"] = "";
194 auto PI = TU.inputs(FS);
195 auto BaselinePreamble = buildPreamble(
196 TU.Filename, *buildCompilerInvocation(PI, Diags), PI, true, nullptr);
197 // We drop c.h from modified and add a new header. Since the latter is patched
198 // we should only get a.h in preamble includes. d.h shouldn't be part of the
199 // preamble, as it's coming from a disabled region.
207 auto PP = PreamblePatch::createFullPatch(testPath(TU.Filename), TU.inputs(FS),
209 // Only a.h should exists in the preamble, as c.h has been dropped and b.h was
212 PP.preambleIncludes(),
214 Field(&Inclusion::Written, "\"a
.h
\""),
215 Field(&Inclusion::Resolved, testPath("a
.h
")),
216 Field(&Inclusion::HeaderID, testing::Not(testing::Eq(std::nullopt))),
217 Field(&Inclusion::FileKind, SrcMgr::CharacteristicKind::C_User))));
220 std::optional<ParsedAST>
221 createPatchedAST(llvm::StringRef Baseline, llvm::StringRef Modified,
222 llvm::StringMap<std::string> AdditionalFiles = {}) {
223 auto TU = TestTU::withCode(Baseline);
224 TU.AdditionalFiles = std::move(AdditionalFiles);
225 auto BaselinePreamble = TU.preamble();
226 if (!BaselinePreamble) {
227 ADD_FAILURE() << "Failed to build baseline preamble
";
231 IgnoreDiagnostics Diags;
233 TU.Code = Modified.str();
234 auto CI = buildCompilerInvocation(TU.inputs(FS), Diags);
236 ADD_FAILURE() << "Failed to build compiler invocation
";
239 return ParsedAST::build(testPath(TU.Filename), TU.inputs(FS), std::move(CI),
240 {}, BaselinePreamble);
243 std::string getPreamblePatch(llvm::StringRef Baseline,
244 llvm::StringRef Modified) {
245 auto BaselinePreamble = TestTU::withCode(Baseline).preamble();
246 if (!BaselinePreamble) {
247 ADD_FAILURE() << "Failed to build baseline preamble
";
251 auto TU = TestTU::withCode(Modified);
252 return PreamblePatch::createFullPatch(testPath("main
.cpp
"), TU.inputs(FS),
258 TEST(PreamblePatchTest, IncludesArePreserved) {
259 llvm::StringLiteral Baseline = R"(//error-ok
263 llvm::StringLiteral Modified = R"(//error-ok
268 auto Includes = createPatchedAST(Baseline, Modified.str())
269 ->getIncludeStructure()
271 EXPECT_TRUE(!Includes.empty());
272 EXPECT_EQ(Includes, TestTU::withCode(Baseline)
274 .getIncludeStructure()
278 TEST(PreamblePatchTest, Define) {
279 // BAR should be defined while parsing the AST.
281 const char *const Contents;
282 const char *const ExpectedPatch;
288 R"cpp(#line 0 ".*main.cpp"
300 R"cpp(#line 0 ".*main.cpp"
312 R"cpp(#line 0 ".*main.cpp"
320 for (const auto &Case : Cases) {
321 SCOPED_TRACE(Case.Contents);
322 llvm::Annotations Modified(Case.Contents);
323 EXPECT_THAT(getPreamblePatch("", Modified.code()),
324 MatchesRegex(Case.ExpectedPatch));
326 auto AST = createPatchedAST("", Modified.code());
328 std::vector<llvm::Annotations::Range> MacroRefRanges;
329 for (auto &M : AST->getMacros().MacroRefs) {
330 for (auto &O : M.getSecond())
331 MacroRefRanges.push_back({O.StartOffset, O.EndOffset});
333 EXPECT_THAT(MacroRefRanges, Contains(Modified.range()));
337 TEST(PreamblePatchTest, OrderingPreserved) {
338 llvm::StringLiteral Baseline = "#define BAR(X) X";
339 Annotations Modified(R"cpp(
340 #define BAR(X, Y) X Y
345 llvm::StringLiteral ExpectedPatch(R"cpp(#line 0 ".*main.cpp"
348 #define BAR\(X, Y\) X Y
353 EXPECT_THAT(getPreamblePatch(Baseline, Modified.code()),
354 MatchesRegex(ExpectedPatch.str()));
356 auto AST = createPatchedAST(Baseline, Modified.code());
360 TEST(PreamblePatchTest, LocateMacroAtWorks) {
362 const char *const Baseline;
363 const char *const Modified;
365 // Addition of new directive
372 // Available inside preamble section
377 #undef $use^FOO)cpp",
379 // Available after undef, as we don't patch those
387 // Identifier on a different line
395 // In presence of comment tokens
413 for (const auto &Case : Cases) {
414 SCOPED_TRACE(Case.Modified);
415 llvm::Annotations Modified(Case.Modified);
416 auto AST = createPatchedAST(Case.Baseline, Modified.code());
419 const auto &SM = AST->getSourceManager();
420 auto *MacroTok = AST->getTokens().spelledTokenAt(
421 SM.getComposedLoc(SM.getMainFileID(), Modified.point("use
")));
422 ASSERT_TRUE(MacroTok);
424 auto FoundMacro = locateMacroAt(*MacroTok, AST->getPreprocessor());
425 ASSERT_TRUE(FoundMacro);
426 EXPECT_THAT(FoundMacro->Name, "FOO
");
428 auto MacroLoc = FoundMacro->NameLoc;
429 EXPECT_EQ(SM.getFileID(MacroLoc), SM.getMainFileID());
430 EXPECT_EQ(SM.getFileOffset(MacroLoc), Modified.point("def
"));
434 TEST(PreamblePatchTest, LocateMacroAtDeletion) {
436 // We don't patch deleted define directives, make sure we don't crash.
437 llvm::StringLiteral Baseline = "#define FOO";
438 llvm::Annotations Modified("^FOO");
440 auto AST = createPatchedAST(Baseline, Modified.code());
443 const auto &SM = AST->getSourceManager();
444 auto *MacroTok = AST->getTokens().spelledTokenAt(
445 SM.getComposedLoc(SM.getMainFileID(), Modified.point()));
446 ASSERT_TRUE(MacroTok);
448 auto FoundMacro = locateMacroAt(*MacroTok, AST->getPreprocessor());
449 ASSERT_TRUE(FoundMacro);
450 EXPECT_THAT(FoundMacro->Name, "FOO");
452 getHover(*AST, offsetToPosition(Modified.code(), Modified.point()),
453 format::getLLVMStyle(), nullptr);
455 EXPECT_THAT(HI->Definition, testing::IsEmpty());
459 // Offset is valid, but underlying text is different.
460 llvm::StringLiteral Baseline = "#define FOO";
461 Annotations Modified(R"cpp(#define BAR
464 auto AST
= createPatchedAST(Baseline
, Modified
.code());
467 auto HI
= getHover(*AST
, Modified
.point(), format::getLLVMStyle(), nullptr);
469 EXPECT_THAT(HI
->Definition
, "#define BAR");
473 MATCHER_P(referenceRangeIs
, R
, "") { return arg
.Loc
.range
== R
; }
475 TEST(PreamblePatchTest
, RefsToMacros
) {
477 const char *const Baseline
;
478 const char *const Modified
;
495 // Ref in preamble section
504 for (const auto &Case
: Cases
) {
505 Annotations
Modified(Case
.Modified
);
506 auto AST
= createPatchedAST("", Modified
.code());
509 const auto &SM
= AST
->getSourceManager();
510 std::vector
<Matcher
<ReferencesResult::Reference
>> ExpectedLocations
;
511 for (const auto &R
: Modified
.ranges())
512 ExpectedLocations
.push_back(referenceRangeIs(R
));
514 for (const auto &P
: Modified
.points()) {
515 auto *MacroTok
= AST
->getTokens().spelledTokenAt(SM
.getComposedLoc(
517 llvm::cantFail(positionToOffset(Modified
.code(), P
))));
518 ASSERT_TRUE(MacroTok
);
519 EXPECT_THAT(findReferences(*AST
, P
, 0).References
,
520 testing::ElementsAreArray(ExpectedLocations
));
525 TEST(TranslatePreamblePatchLocation
, Simple
) {
526 auto TU
= TestTU::withHeaderCode(R
"cpp(
529 // Presumed line/col needs to be valid in the main file.
530 TU
.Code
= R
"cpp(// line 1
534 TU
.Filename
= "main.cpp";
535 TU
.HeaderFilename
= "__preamble_patch__.h";
536 TU
.ImplicitHeaderGuard
= false;
538 auto AST
= TU
.build();
539 auto &SM
= AST
.getSourceManager();
540 auto &ND
= findDecl(AST
, "foo");
541 EXPECT_NE(SM
.getFileID(ND
.getLocation()), SM
.getMainFileID());
543 auto TranslatedLoc
= translatePreamblePatchLocation(ND
.getLocation(), SM
);
544 auto DecompLoc
= SM
.getDecomposedLoc(TranslatedLoc
);
545 EXPECT_EQ(DecompLoc
.first
, SM
.getMainFileID());
546 EXPECT_EQ(SM
.getLineNumber(DecompLoc
.first
, DecompLoc
.second
), 3U);
549 TEST(PreamblePatch
, ModifiedBounds
) {
551 const char *const Baseline
;
552 const char *const Modified
;
562 {"#define FOO", "#define BAR"},
571 for (const auto &Case
: Cases
) {
572 auto TU
= TestTU::withCode(Case
.Baseline
);
573 auto BaselinePreamble
= TU
.preamble();
574 ASSERT_TRUE(BaselinePreamble
);
576 Annotations
Modified(Case
.Modified
);
577 TU
.Code
= Modified
.code().str();
579 auto PP
= PreamblePatch::createFullPatch(testPath(TU
.Filename
),
580 TU
.inputs(FS
), *BaselinePreamble
);
582 IgnoreDiagnostics Diags
;
583 auto CI
= buildCompilerInvocation(TU
.inputs(FS
), Diags
);
586 const auto ExpectedBounds
=
587 Lexer::ComputePreamble(Case
.Modified
, CI
->getLangOpts());
588 EXPECT_EQ(PP
.modifiedBounds().Size
, ExpectedBounds
.Size
);
589 EXPECT_EQ(PP
.modifiedBounds().PreambleEndsAtStartOfLine
,
590 ExpectedBounds
.PreambleEndsAtStartOfLine
);
594 TEST(PreamblePatch
, MacroLoc
) {
595 llvm::StringLiteral Baseline
= "\n#define MACRO 12\nint num = MACRO;";
596 llvm::StringLiteral Modified
= " \n#define MACRO 12\nint num = MACRO;";
597 auto AST
= createPatchedAST(Baseline
, Modified
);
601 TEST(PreamblePatch
, NoopWhenNotRequested
) {
602 llvm::StringLiteral Baseline
= "#define M\nint num = M;";
603 llvm::StringLiteral Modified
= "#define M\n#include <foo.h>\nint num = M;";
604 auto TU
= TestTU::withCode(Baseline
);
605 auto BaselinePreamble
= TU
.preamble();
606 ASSERT_TRUE(BaselinePreamble
);
608 TU
.Code
= Modified
.str();
610 auto PP
= PreamblePatch::createMacroPatch(testPath(TU
.Filename
),
611 TU
.inputs(FS
), *BaselinePreamble
);
612 EXPECT_TRUE(PP
.text().empty());
615 ::testing::Matcher
<const Diag
&>
616 withNote(::testing::Matcher
<Note
> NoteMatcher
) {
617 return Field(&Diag::Notes
, ElementsAre(NoteMatcher
));
619 MATCHER_P(Diag
, Range
, "Diag at " + llvm::to_string(Range
)) {
620 return arg
.Range
== Range
;
622 MATCHER_P2(Diag
, Range
, Name
,
623 "Diag at " + llvm::to_string(Range
) + " = [" + Name
+ "]") {
624 return arg
.Range
== Range
&& arg
.Name
== Name
;
627 TEST(PreamblePatch
, DiagnosticsFromMainASTAreInRightPlace
) {
629 Annotations
Code("#define FOO");
630 // Check with removals from preamble.
631 Annotations
NewCode("[[x]];/* error-ok */");
632 auto AST
= createPatchedAST(Code
.code(), NewCode
.code());
633 EXPECT_THAT(AST
->getDiagnostics(),
634 ElementsAre(Diag(NewCode
.range(), "missing_type_specifier")));
637 // Check with additions to preamble.
638 Annotations
Code("#define FOO");
639 Annotations
NewCode(R
"(
642 [[x]];/* error-ok */)");
643 auto AST
= createPatchedAST(Code
.code(), NewCode
.code());
644 EXPECT_THAT(AST
->getDiagnostics(),
645 ElementsAre(Diag(NewCode
.range(), "missing_type_specifier")));
649 TEST(PreamblePatch
, DiagnosticsToPreamble
) {
651 Cfg
.Diagnostics
.UnusedIncludes
= Config::IncludesPolicy::Strict
;
652 Cfg
.Diagnostics
.MissingIncludes
= Config::IncludesPolicy::Strict
;
653 WithContextValue
WithCfg(Config::Key
, std::move(Cfg
));
655 llvm::StringMap
<std::string
> AdditionalFiles
;
656 AdditionalFiles
["foo.h"] = "#pragma once";
657 AdditionalFiles
["bar.h"] = "#pragma once";
661 [[#include "foo
.h
"]])");
662 // Check with removals from preamble.
663 Annotations
NewCode(R
"([[# include "foo
.h
"]])");
664 auto AST
= createPatchedAST(Code
.code(), NewCode
.code(), AdditionalFiles
);
665 EXPECT_THAT(AST
->getDiagnostics(),
666 ElementsAre(Diag(NewCode
.range(), "unused-includes")));
669 // Check with additions to preamble.
672 [[#include "foo
.h
"]])");
673 Annotations
NewCode(R
"(
674 $bar[[#include "bar
.h
"]]
676 $foo[[#include "foo
.h
"]])");
677 auto AST
= createPatchedAST(Code
.code(), NewCode
.code(), AdditionalFiles
);
679 AST
->getDiagnostics(),
680 UnorderedElementsAre(Diag(NewCode
.range("bar"), "unused-includes"),
681 Diag(NewCode
.range("foo"), "unused-includes")));
684 Annotations
Code("#define [[FOO]] 1\n");
685 // Check ranges for notes.
686 // This also makes sure we don't generate missing-include diagnostics
687 // because macros are redefined in preamble-patch.
688 Annotations
NewCode(R
"(#define BARXYZ 1
689 #define $foo1[[FOO]] 1
691 #define $foo2[[FOO]] 2)");
692 auto AST
= createPatchedAST(Code
.code(), NewCode
.code(), AdditionalFiles
);
694 AST
->getDiagnostics(),
695 ElementsAre(AllOf(Diag(NewCode
.range("foo2"), "-Wmacro-redefined"),
696 withNote(Diag(NewCode
.range("foo1"))))));
700 TEST(PreamblePatch
, TranslatesDiagnosticsInPreamble
) {
702 // Check with additions to preamble.
703 Annotations
Code("#include [[<foo>]]");
704 Annotations
NewCode(R
"(
706 #include [[<foo>]])");
707 auto AST
= createPatchedAST(Code
.code(), NewCode
.code());
708 EXPECT_THAT(AST
->getDiagnostics(),
709 ElementsAre(Diag(NewCode
.range(), "pp_file_not_found")));
712 // Check with removals from preamble.
715 #include [[<foo>]])");
716 Annotations
NewCode("#include [[<foo>]]");
717 auto AST
= createPatchedAST(Code
.code(), NewCode
.code());
718 EXPECT_THAT(AST
->getDiagnostics(),
719 ElementsAre(Diag(NewCode
.range(), "pp_file_not_found")));
722 // Drop line with diags.
723 Annotations
Code("#include [[<foo>]]");
724 Annotations
NewCode("#define BAR\n#define BAZ\n");
725 auto AST
= createPatchedAST(Code
.code(), NewCode
.code());
726 EXPECT_THAT(AST
->getDiagnostics(), IsEmpty());
729 // Picks closest line in case of multiple alternatives.
730 Annotations
Code("#include [[<foo>]]");
731 Annotations
NewCode(R
"(
736 auto AST
= createPatchedAST(Code
.code(), NewCode
.code());
737 EXPECT_THAT(AST
->getDiagnostics(),
738 ElementsAre(Diag(NewCode
.range(), "pp_file_not_found")));
741 // Drop diag if line spelling has changed.
742 Annotations
Code("#include [[<foo>]]");
743 Annotations
NewCode(" # include <foo>");
744 auto AST
= createPatchedAST(Code
.code(), NewCode
.code());
745 EXPECT_THAT(AST
->getDiagnostics(), IsEmpty());
753 Annotations
NewCode(R
"(#include [[<fo\
755 auto AST
= createPatchedAST(Code
.code(), NewCode
.code());
756 EXPECT_THAT(AST
->getDiagnostics(),
757 ElementsAre(Diag(NewCode
.range(), "pp_file_not_found")));
760 // Multiple lines with change.
766 Annotations
NewCode(R
"(#include <fo\
768 auto AST
= createPatchedAST(Code
.code(), NewCode
.code());
769 EXPECT_THAT(AST
->getDiagnostics(), IsEmpty());
774 #define $note[[BAR]] 1
775 #define $main[[BAR]] 2)");
776 Annotations
NewCode(R
"(
778 #define $note[[BAR]] 1
780 #define $main[[BAR]] 2)");
781 auto AST
= createPatchedAST(Code
.code(), NewCode
.code());
783 AST
->getDiagnostics(),
784 ElementsAre(AllOf(Diag(NewCode
.range("main"), "-Wmacro-redefined"),
785 withNote(Diag(NewCode
.range("note"))))));
788 // Preserves diag without note.
790 #define $note[[BAR]] 1
791 #define $main[[BAR]] 2)");
792 Annotations
NewCode(R
"(
793 #define $main[[BAR]] 2)");
794 auto AST
= createPatchedAST(Code
.code(), NewCode
.code());
796 AST
->getDiagnostics(),
797 ElementsAre(AllOf(Diag(NewCode
.range("main"), "-Wmacro-redefined"),
798 Field(&Diag::Notes
, IsEmpty()))));
801 // Make sure orphaned notes are not promoted to diags.
803 #define $note[[BAR]] 1
804 #define $main[[BAR]] 2)");
805 Annotations
NewCode(R
"(
808 auto AST
= createPatchedAST(Code
.code(), NewCode
.code());
809 EXPECT_THAT(AST
->getDiagnostics(), IsEmpty());
817 // This code will emit a diagnostic for unterminated #ifndef (as stale
818 // preamble has the conditional but main file doesn't terminate it).
819 // We shouldn't emit any diagnotiscs (and shouldn't crash).
820 Annotations
NewCode("");
821 auto AST
= createPatchedAST(Code
.code(), NewCode
.code());
822 EXPECT_THAT(AST
->getDiagnostics(), IsEmpty());
830 // This code will emit a diagnostic for unterminated #ifndef (as stale
831 // preamble has the conditional but main file doesn't terminate it).
832 // We shouldn't emit any diagnotiscs (and shouldn't crash).
833 // FIXME: Patch/ignore diagnostics in such cases.
834 Annotations
NewCode(R
"(
837 auto AST
= createPatchedAST(Code
.code(), NewCode
.code());
839 AST
->getDiagnostics(),
840 ElementsAre(Diag(NewCode
.range(), "pp_unterminated_conditional")));
844 MATCHER_P2(Mark
, Range
, Text
, "") {
845 return std::tie(arg
.Rng
, arg
.Trivia
) == std::tie(Range
, Text
);
848 TEST(PreamblePatch
, MacroAndMarkHandling
) {
850 Annotations
Code(R
"cpp(
858 Annotations
NewCode(R
"cpp(
869 auto AST
= createPatchedAST(Code
.code(), NewCode
.code());
870 EXPECT_THAT(AST
->getMacros().Names
.keys(),
871 UnorderedElementsAreArray({"FOO", "BAR", "BAZ"}));
872 EXPECT_THAT(AST
->getMarks(),
873 UnorderedElementsAre(Mark(NewCode
.range("x"), " XX"),
874 Mark(NewCode
.range("y"), " YY")));
878 TEST(PreamblePatch
, PatchFileEntry
) {
879 Annotations
Code(R
"cpp(#define FOO)cpp");
880 Annotations
NewCode(R
"cpp(
884 auto AST
= createPatchedAST(Code
.code(), Code
.code());
886 PreamblePatch::getPatchEntry(AST
->tuPath(), AST
->getSourceManager()),
890 auto AST
= createPatchedAST(Code
.code(), NewCode
.code());
892 PreamblePatch::getPatchEntry(AST
->tuPath(), AST
->getSourceManager());
893 ASSERT_NE(FE
, std::nullopt
);
894 EXPECT_THAT(FE
->getName().str(),
895 testing::EndsWith(PreamblePatch::HeaderName
.str()));
900 } // namespace clangd