xref: /llvm-project/clang-tools-extra/clangd/unittests/SourceCodeTests.cpp (revision 777180a32b61070a10dd330b4f038bf24e916af1)
1 //===-- SourceCodeTests.cpp  ------------------------------------*- C++ -*-===//
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 //===----------------------------------------------------------------------===//
8 #include "Annotations.h"
9 #include "Context.h"
10 #include "Protocol.h"
11 #include "SourceCode.h"
12 #include "TestTU.h"
13 #include "clang/Basic/LangOptions.h"
14 #include "clang/Basic/SourceLocation.h"
15 #include "clang/Format/Format.h"
16 #include "llvm/Support/Error.h"
17 #include "llvm/Support/raw_os_ostream.h"
18 #include "llvm/Testing/Support/Annotations.h"
19 #include "llvm/Testing/Support/Error.h"
20 #include "gmock/gmock.h"
21 #include "gtest/gtest.h"
22 #include <tuple>
23 
24 namespace clang {
25 namespace clangd {
26 namespace {
27 
28 using llvm::Failed;
29 using llvm::HasValue;
30 
31 MATCHER_P2(Pos, Line, Col, "") {
32   return arg.line == int(Line) && arg.character == int(Col);
33 }
34 
35 MATCHER_P(MacroName, Name, "") { return arg.Name == Name; }
36 
37 /// A helper to make tests easier to read.
38 Position position(int Line, int Character) {
39   Position Pos;
40   Pos.line = Line;
41   Pos.character = Character;
42   return Pos;
43 }
44 
45 Range range(const std::pair<int, int> &P1, const std::pair<int, int> &P2) {
46   Range Range;
47   Range.start = position(P1.first, P1.second);
48   Range.end = position(P2.first, P2.second);
49   return Range;
50 }
51 
52 TEST(SourceCodeTests, lspLength) {
53   EXPECT_EQ(lspLength(""), 0UL);
54   EXPECT_EQ(lspLength("ascii"), 5UL);
55   // BMP
56   EXPECT_EQ(lspLength("↓"), 1UL);
57   EXPECT_EQ(lspLength("¥"), 1UL);
58   // astral
59   EXPECT_EQ(lspLength("��"), 2UL);
60 
61   WithContextValue UTF8(kCurrentOffsetEncoding, OffsetEncoding::UTF8);
62   EXPECT_EQ(lspLength(""), 0UL);
63   EXPECT_EQ(lspLength("ascii"), 5UL);
64   // BMP
65   EXPECT_EQ(lspLength("↓"), 3UL);
66   EXPECT_EQ(lspLength("¥"), 2UL);
67   // astral
68   EXPECT_EQ(lspLength("��"), 4UL);
69 
70   WithContextValue UTF32(kCurrentOffsetEncoding, OffsetEncoding::UTF32);
71   EXPECT_EQ(lspLength(""), 0UL);
72   EXPECT_EQ(lspLength("ascii"), 5UL);
73   // BMP
74   EXPECT_EQ(lspLength("↓"), 1UL);
75   EXPECT_EQ(lspLength("¥"), 1UL);
76   // astral
77   EXPECT_EQ(lspLength("��"), 1UL);
78 }
79 
80 // The = → �� below are ASCII (1 byte), BMP (3 bytes), and astral (4 bytes).
81 const char File[] = R"(0:0 = 0
82 1:0 → 8
83 2:0 �� 18)";
84 struct Line {
85   unsigned Number;
86   unsigned Offset;
87   unsigned Length;
88 };
89 Line FileLines[] = {Line{0, 0, 7}, Line{1, 8, 9}, Line{2, 18, 11}};
90 
91 TEST(SourceCodeTests, PositionToOffset) {
92   // line out of bounds
93   EXPECT_THAT_EXPECTED(positionToOffset(File, position(-1, 2)), llvm::Failed());
94   // first line
95   EXPECT_THAT_EXPECTED(positionToOffset(File, position(0, -1)),
96                        llvm::Failed()); // out of range
97   EXPECT_THAT_EXPECTED(positionToOffset(File, position(0, 0)),
98                        llvm::HasValue(0)); // first character
99   EXPECT_THAT_EXPECTED(positionToOffset(File, position(0, 3)),
100                        llvm::HasValue(3)); // middle character
101   EXPECT_THAT_EXPECTED(positionToOffset(File, position(0, 6)),
102                        llvm::HasValue(6)); // last character
103   EXPECT_THAT_EXPECTED(positionToOffset(File, position(0, 7)),
104                        llvm::HasValue(7)); // the newline itself
105   EXPECT_THAT_EXPECTED(positionToOffset(File, position(0, 7), false),
106                        llvm::HasValue(7));
107   EXPECT_THAT_EXPECTED(positionToOffset(File, position(0, 8)),
108                        llvm::HasValue(7)); // out of range
109   EXPECT_THAT_EXPECTED(positionToOffset(File, position(0, 8), false),
110                        llvm::Failed()); // out of range
111   // middle line
112   EXPECT_THAT_EXPECTED(positionToOffset(File, position(1, -1)),
113                        llvm::Failed()); // out of range
114   EXPECT_THAT_EXPECTED(positionToOffset(File, position(1, 0)),
115                        llvm::HasValue(8)); // first character
116   EXPECT_THAT_EXPECTED(positionToOffset(File, position(1, 3)),
117                        llvm::HasValue(11)); // middle character
118   EXPECT_THAT_EXPECTED(positionToOffset(File, position(1, 3), false),
119                        llvm::HasValue(11));
120   EXPECT_THAT_EXPECTED(positionToOffset(File, position(1, 6)),
121                        llvm::HasValue(16)); // last character
122   EXPECT_THAT_EXPECTED(positionToOffset(File, position(1, 7)),
123                        llvm::HasValue(17)); // the newline itself
124   EXPECT_THAT_EXPECTED(positionToOffset(File, position(1, 8)),
125                        llvm::HasValue(17)); // out of range
126   EXPECT_THAT_EXPECTED(positionToOffset(File, position(1, 8), false),
127                        llvm::Failed()); // out of range
128   // last line
129   EXPECT_THAT_EXPECTED(positionToOffset(File, position(2, -1)),
130                        llvm::Failed()); // out of range
131   EXPECT_THAT_EXPECTED(positionToOffset(File, position(2, 0)),
132                        llvm::HasValue(18)); // first character
133   EXPECT_THAT_EXPECTED(positionToOffset(File, position(2, 3)),
134                        llvm::HasValue(21)); // middle character
135   EXPECT_THAT_EXPECTED(positionToOffset(File, position(2, 5), false),
136                        llvm::Failed()); // middle of surrogate pair
137   EXPECT_THAT_EXPECTED(positionToOffset(File, position(2, 5)),
138                        llvm::HasValue(26)); // middle of surrogate pair
139   EXPECT_THAT_EXPECTED(positionToOffset(File, position(2, 6), false),
140                        llvm::HasValue(26)); // end of surrogate pair
141   EXPECT_THAT_EXPECTED(positionToOffset(File, position(2, 8)),
142                        llvm::HasValue(28)); // last character
143   EXPECT_THAT_EXPECTED(positionToOffset(File, position(2, 9)),
144                        llvm::HasValue(29)); // EOF
145   EXPECT_THAT_EXPECTED(positionToOffset(File, position(2, 10), false),
146                        llvm::Failed()); // out of range
147   // line out of bounds
148   EXPECT_THAT_EXPECTED(positionToOffset(File, position(3, 0)), llvm::Failed());
149   EXPECT_THAT_EXPECTED(positionToOffset(File, position(3, 1)), llvm::Failed());
150 
151   // Codepoints are similar, except near astral characters.
152   WithContextValue UTF32(kCurrentOffsetEncoding, OffsetEncoding::UTF32);
153   // line out of bounds
154   EXPECT_THAT_EXPECTED(positionToOffset(File, position(-1, 2)), llvm::Failed());
155   // first line
156   EXPECT_THAT_EXPECTED(positionToOffset(File, position(0, -1)),
157                        llvm::Failed()); // out of range
158   EXPECT_THAT_EXPECTED(positionToOffset(File, position(0, 0)),
159                        llvm::HasValue(0)); // first character
160   EXPECT_THAT_EXPECTED(positionToOffset(File, position(0, 3)),
161                        llvm::HasValue(3)); // middle character
162   EXPECT_THAT_EXPECTED(positionToOffset(File, position(0, 6)),
163                        llvm::HasValue(6)); // last character
164   EXPECT_THAT_EXPECTED(positionToOffset(File, position(0, 7)),
165                        llvm::HasValue(7)); // the newline itself
166   EXPECT_THAT_EXPECTED(positionToOffset(File, position(0, 7), false),
167                        llvm::HasValue(7));
168   EXPECT_THAT_EXPECTED(positionToOffset(File, position(0, 8)),
169                        llvm::HasValue(7)); // out of range
170   EXPECT_THAT_EXPECTED(positionToOffset(File, position(0, 8), false),
171                        llvm::Failed()); // out of range
172   // middle line
173   EXPECT_THAT_EXPECTED(positionToOffset(File, position(1, -1)),
174                        llvm::Failed()); // out of range
175   EXPECT_THAT_EXPECTED(positionToOffset(File, position(1, 0)),
176                        llvm::HasValue(8)); // first character
177   EXPECT_THAT_EXPECTED(positionToOffset(File, position(1, 3)),
178                        llvm::HasValue(11)); // middle character
179   EXPECT_THAT_EXPECTED(positionToOffset(File, position(1, 3), false),
180                        llvm::HasValue(11));
181   EXPECT_THAT_EXPECTED(positionToOffset(File, position(1, 6)),
182                        llvm::HasValue(16)); // last character
183   EXPECT_THAT_EXPECTED(positionToOffset(File, position(1, 7)),
184                        llvm::HasValue(17)); // the newline itself
185   EXPECT_THAT_EXPECTED(positionToOffset(File, position(1, 8)),
186                        llvm::HasValue(17)); // out of range
187   EXPECT_THAT_EXPECTED(positionToOffset(File, position(1, 8), false),
188                        llvm::Failed()); // out of range
189   // last line
190   EXPECT_THAT_EXPECTED(positionToOffset(File, position(2, -1)),
191                        llvm::Failed()); // out of range
192   EXPECT_THAT_EXPECTED(positionToOffset(File, position(2, 0)),
193                        llvm::HasValue(18)); // first character
194   EXPECT_THAT_EXPECTED(positionToOffset(File, position(2, 4)),
195                        llvm::HasValue(22)); // Before astral character.
196   EXPECT_THAT_EXPECTED(positionToOffset(File, position(2, 5), false),
197                        llvm::HasValue(26)); // after astral character
198   EXPECT_THAT_EXPECTED(positionToOffset(File, position(2, 7)),
199                        llvm::HasValue(28)); // last character
200   EXPECT_THAT_EXPECTED(positionToOffset(File, position(2, 8)),
201                        llvm::HasValue(29)); // EOF
202   EXPECT_THAT_EXPECTED(positionToOffset(File, position(2, 9), false),
203                        llvm::Failed()); // out of range
204   // line out of bounds
205   EXPECT_THAT_EXPECTED(positionToOffset(File, position(3, 0)), llvm::Failed());
206   EXPECT_THAT_EXPECTED(positionToOffset(File, position(3, 1)), llvm::Failed());
207 
208   // Test UTF-8, where transformations are trivial.
209   WithContextValue UTF8(kCurrentOffsetEncoding, OffsetEncoding::UTF8);
210   EXPECT_THAT_EXPECTED(positionToOffset(File, position(-1, 2)), llvm::Failed());
211   EXPECT_THAT_EXPECTED(positionToOffset(File, position(3, 0)), llvm::Failed());
212   for (Line L : FileLines) {
213     EXPECT_THAT_EXPECTED(positionToOffset(File, position(L.Number, -1)),
214                          llvm::Failed()); // out of range
215     for (unsigned I = 0; I <= L.Length; ++I)
216       EXPECT_THAT_EXPECTED(positionToOffset(File, position(L.Number, I)),
217                            llvm::HasValue(L.Offset + I));
218     EXPECT_THAT_EXPECTED(positionToOffset(File, position(L.Number, L.Length+1)),
219                          llvm::HasValue(L.Offset + L.Length));
220     EXPECT_THAT_EXPECTED(
221         positionToOffset(File, position(L.Number, L.Length + 1), false),
222         llvm::Failed()); // out of range
223   }
224 }
225 
226 TEST(SourceCodeTests, OffsetToPosition) {
227   EXPECT_THAT(offsetToPosition(File, 0), Pos(0, 0)) << "start of file";
228   EXPECT_THAT(offsetToPosition(File, 3), Pos(0, 3)) << "in first line";
229   EXPECT_THAT(offsetToPosition(File, 6), Pos(0, 6)) << "end of first line";
230   EXPECT_THAT(offsetToPosition(File, 7), Pos(0, 7)) << "first newline";
231   EXPECT_THAT(offsetToPosition(File, 8), Pos(1, 0)) << "start of second line";
232   EXPECT_THAT(offsetToPosition(File, 12), Pos(1, 4)) << "before BMP char";
233   EXPECT_THAT(offsetToPosition(File, 13), Pos(1, 5)) << "in BMP char";
234   EXPECT_THAT(offsetToPosition(File, 15), Pos(1, 5)) << "after BMP char";
235   EXPECT_THAT(offsetToPosition(File, 16), Pos(1, 6)) << "end of second line";
236   EXPECT_THAT(offsetToPosition(File, 17), Pos(1, 7)) << "second newline";
237   EXPECT_THAT(offsetToPosition(File, 18), Pos(2, 0)) << "start of last line";
238   EXPECT_THAT(offsetToPosition(File, 21), Pos(2, 3)) << "in last line";
239   EXPECT_THAT(offsetToPosition(File, 22), Pos(2, 4)) << "before astral char";
240   EXPECT_THAT(offsetToPosition(File, 24), Pos(2, 6)) << "in astral char";
241   EXPECT_THAT(offsetToPosition(File, 26), Pos(2, 6)) << "after astral char";
242   EXPECT_THAT(offsetToPosition(File, 28), Pos(2, 8)) << "end of last line";
243   EXPECT_THAT(offsetToPosition(File, 29), Pos(2, 9)) << "EOF";
244   EXPECT_THAT(offsetToPosition(File, 30), Pos(2, 9)) << "out of bounds";
245 
246   // Codepoints are similar, except near astral characters.
247   WithContextValue UTF32(kCurrentOffsetEncoding, OffsetEncoding::UTF32);
248   EXPECT_THAT(offsetToPosition(File, 0), Pos(0, 0)) << "start of file";
249   EXPECT_THAT(offsetToPosition(File, 3), Pos(0, 3)) << "in first line";
250   EXPECT_THAT(offsetToPosition(File, 6), Pos(0, 6)) << "end of first line";
251   EXPECT_THAT(offsetToPosition(File, 7), Pos(0, 7)) << "first newline";
252   EXPECT_THAT(offsetToPosition(File, 8), Pos(1, 0)) << "start of second line";
253   EXPECT_THAT(offsetToPosition(File, 12), Pos(1, 4)) << "before BMP char";
254   EXPECT_THAT(offsetToPosition(File, 13), Pos(1, 5)) << "in BMP char";
255   EXPECT_THAT(offsetToPosition(File, 15), Pos(1, 5)) << "after BMP char";
256   EXPECT_THAT(offsetToPosition(File, 16), Pos(1, 6)) << "end of second line";
257   EXPECT_THAT(offsetToPosition(File, 17), Pos(1, 7)) << "second newline";
258   EXPECT_THAT(offsetToPosition(File, 18), Pos(2, 0)) << "start of last line";
259   EXPECT_THAT(offsetToPosition(File, 21), Pos(2, 3)) << "in last line";
260   EXPECT_THAT(offsetToPosition(File, 22), Pos(2, 4)) << "before astral char";
261   EXPECT_THAT(offsetToPosition(File, 24), Pos(2, 5)) << "in astral char";
262   EXPECT_THAT(offsetToPosition(File, 26), Pos(2, 5)) << "after astral char";
263   EXPECT_THAT(offsetToPosition(File, 28), Pos(2, 7)) << "end of last line";
264   EXPECT_THAT(offsetToPosition(File, 29), Pos(2, 8)) << "EOF";
265   EXPECT_THAT(offsetToPosition(File, 30), Pos(2, 8)) << "out of bounds";
266 
267   WithContextValue UTF8(kCurrentOffsetEncoding, OffsetEncoding::UTF8);
268   for (Line L : FileLines) {
269     for (unsigned I = 0; I <= L.Length; ++I)
270       EXPECT_THAT(offsetToPosition(File, L.Offset + I), Pos(L.Number, I));
271   }
272   EXPECT_THAT(offsetToPosition(File, 30), Pos(2, 11)) << "out of bounds";
273 }
274 
275 TEST(SourceCodeTests, IsRangeConsecutive) {
276   EXPECT_TRUE(isRangeConsecutive(range({2, 2}, {2, 3}), range({2, 3}, {2, 4})));
277   EXPECT_FALSE(
278       isRangeConsecutive(range({0, 2}, {0, 3}), range({2, 3}, {2, 4})));
279   EXPECT_FALSE(
280       isRangeConsecutive(range({2, 2}, {2, 3}), range({2, 4}, {2, 5})));
281 }
282 
283 TEST(SourceCodeTests, SourceLocationInMainFile) {
284   Annotations Source(R"cpp(
285     ^in^t ^foo
286     ^bar
287     ^baz ^() {}  {} {} {} { }^
288 )cpp");
289 
290   SourceManagerForFile Owner("foo.cpp", Source.code());
291   SourceManager &SM = Owner.get();
292 
293   SourceLocation StartOfFile = SM.getLocForStartOfFile(SM.getMainFileID());
294   EXPECT_THAT_EXPECTED(sourceLocationInMainFile(SM, position(0, 0)),
295                        HasValue(StartOfFile));
296   // End of file.
297   EXPECT_THAT_EXPECTED(
298       sourceLocationInMainFile(SM, position(4, 0)),
299       HasValue(StartOfFile.getLocWithOffset(Source.code().size())));
300   // Column number is too large.
301   EXPECT_THAT_EXPECTED(sourceLocationInMainFile(SM, position(0, 1)), Failed());
302   EXPECT_THAT_EXPECTED(sourceLocationInMainFile(SM, position(0, 100)),
303                        Failed());
304   EXPECT_THAT_EXPECTED(sourceLocationInMainFile(SM, position(4, 1)), Failed());
305   // Line number is too large.
306   EXPECT_THAT_EXPECTED(sourceLocationInMainFile(SM, position(5, 0)), Failed());
307   // Check all positions mentioned in the test return valid results.
308   for (auto P : Source.points()) {
309     size_t Offset = llvm::cantFail(positionToOffset(Source.code(), P));
310     EXPECT_THAT_EXPECTED(sourceLocationInMainFile(SM, P),
311                          HasValue(StartOfFile.getLocWithOffset(Offset)));
312   }
313 }
314 
315 TEST(SourceCodeTests, GetBeginningOfIdentifier) {
316   std::string Preamble = R"cpp(
317 struct Bar { int func(); };
318 #define MACRO(X) void f() { X; }
319 Bar* bar;
320   )cpp";
321   // First ^ is the expected beginning, last is the search position.
322   for (const std::string &Text : std::vector<std::string>{
323            "int ^f^oo();", // inside identifier
324            "int ^foo();",  // beginning of identifier
325            "int ^foo^();", // end of identifier
326            "int foo(^);",  // non-identifier
327            "^int foo();",  // beginning of file (can't back up)
328            "int ^f0^0();", // after a digit (lexing at N-1 is wrong)
329            "/^/ comments", // non-interesting token
330            "void f(int abc) { abc ^ ++; }",    // whitespace
331            "void f(int abc) { ^abc^++; }",     // range of identifier
332            "void f(int abc) { ++^abc^; }",     // range of identifier
333            "void f(int abc) { ++^abc; }",      // range of identifier
334            "void f(int abc) { ^+^+abc; }",     // range of operator
335            "void f(int abc) { ^abc^ ++; }",    // range of identifier
336            "void f(int abc) { abc ^++^; }",    // range of operator
337            "void f(int abc) { ^++^ abc; }",    // range of operator
338            "void f(int abc) { ++ ^abc^; }",    // range of identifier
339            "void f(int abc) { ^++^/**/abc; }", // range of operator
340            "void f(int abc) { ++/**/^abc; }",  // range of identifier
341            "void f(int abc) { ^abc^/**/++; }", // range of identifier
342            "void f(int abc) { abc/**/^++; }",  // range of operator
343            "void f() {^ }", // outside of identifier and operator
344            "int ^λλ^λ();",  // UTF-8 handled properly when backing up
345 
346            // identifier in macro arg
347            "MACRO(bar->^func())",  // beginning of identifier
348            "MACRO(bar->^fun^c())", // inside identifier
349            "MACRO(bar->^func^())", // end of identifier
350            "MACRO(^bar->func())",  // begin identifier
351            "MACRO(^bar^->func())", // end identifier
352            "^MACRO(bar->func())",  // beginning of macro name
353            "^MAC^RO(bar->func())", // inside macro name
354            "^MACRO^(bar->func())", // end of macro name
355        }) {
356     std::string WithPreamble = Preamble + Text;
357     Annotations TestCase(WithPreamble);
358     auto AST = TestTU::withCode(TestCase.code()).build();
359     const auto &SourceMgr = AST.getSourceManager();
360     SourceLocation Actual = getBeginningOfIdentifier(
361         TestCase.points().back(), SourceMgr, AST.getLangOpts());
362     Position ActualPos = offsetToPosition(
363         TestCase.code(),
364         SourceMgr.getFileOffset(SourceMgr.getSpellingLoc(Actual)));
365     EXPECT_EQ(TestCase.points().front(), ActualPos) << Text;
366   }
367 }
368 
369 TEST(SourceCodeTests, CollectIdentifiers) {
370   auto Style = format::getLLVMStyle();
371   auto IDs = collectIdentifiers(R"cpp(
372   #include "a.h"
373   void foo() { int xyz; int abc = xyz; return foo(); }
374   )cpp",
375                                 Style);
376   EXPECT_EQ(IDs.size(), 7u);
377   EXPECT_EQ(IDs["include"], 1u);
378   EXPECT_EQ(IDs["void"], 1u);
379   EXPECT_EQ(IDs["int"], 2u);
380   EXPECT_EQ(IDs["xyz"], 2u);
381   EXPECT_EQ(IDs["abc"], 1u);
382   EXPECT_EQ(IDs["return"], 1u);
383   EXPECT_EQ(IDs["foo"], 2u);
384 }
385 
386 TEST(SourceCodeTests, CollectWords) {
387   auto Words = collectWords(R"cpp(
388   #define FIZZ_BUZZ
389   // this is a comment
390   std::string getSomeText() { return "magic word"; }
391   )cpp");
392   std::set<StringRef> ActualWords(Words.keys().begin(), Words.keys().end());
393   std::set<StringRef> ExpectedWords = {"define",  "fizz",   "buzz", "this",
394                                        "comment", "string", "some", "text",
395                                        "return",  "magic",  "word"};
396   EXPECT_EQ(ActualWords, ExpectedWords);
397 }
398 
399 TEST(SourceCodeTests, VisibleNamespaces) {
400   std::vector<std::pair<const char *, std::vector<std::string>>> Cases = {
401       {
402           R"cpp(
403             // Using directive resolved against enclosing namespaces.
404             using namespace foo;
405             namespace ns {
406             using namespace bar;
407           )cpp",
408           {"ns", "", "bar", "foo", "ns::bar"},
409       },
410       {
411           R"cpp(
412             // Don't include namespaces we've closed, ignore namespace aliases.
413             using namespace clang;
414             using std::swap;
415             namespace clang {
416             namespace clangd {}
417             namespace ll = ::llvm;
418             }
419             namespace clang {
420           )cpp",
421           {"clang", ""},
422       },
423       {
424           R"cpp(
425             // Using directives visible even if a namespace is reopened.
426             // Ignore anonymous namespaces.
427             namespace foo{ using namespace bar; }
428             namespace foo{ namespace {
429           )cpp",
430           {"foo", "", "bar", "foo::bar"},
431       },
432       {
433           R"cpp(
434             // Mismatched braces
435             namespace foo{}
436             }}}
437             namespace bar{
438           )cpp",
439           {"bar", ""},
440       },
441       {
442           R"cpp(
443             // Namespaces with multiple chunks.
444             namespace a::b {
445               using namespace c::d;
446               namespace e::f {
447           )cpp",
448           {
449               "a::b::e::f",
450               "",
451               "a",
452               "a::b",
453               "a::b::c::d",
454               "a::b::e",
455               "a::c::d",
456               "c::d",
457           },
458       },
459       {
460           "",
461           {""},
462       },
463       {
464           R"cpp(
465             // Parse until EOF
466             namespace bar{})cpp",
467           {""},
468       },
469   };
470   for (const auto& Case : Cases) {
471     EXPECT_EQ(Case.second,
472               visibleNamespaces(Case.first, format::getLLVMStyle()))
473         << Case.first;
474   }
475 }
476 
477 TEST(SourceCodeTests, GetMacros) {
478   Annotations Code(R"cpp(
479      #define MACRO 123
480      int abc = MA^CRO;
481    )cpp");
482   TestTU TU = TestTU::withCode(Code.code());
483   auto AST = TU.build();
484   auto Loc = getBeginningOfIdentifier(Code.point(), AST.getSourceManager(),
485                                       AST.getLangOpts());
486   auto Result = locateMacroAt(Loc, AST.getPreprocessor());
487   ASSERT_TRUE(Result);
488   EXPECT_THAT(*Result, MacroName("MACRO"));
489 }
490 
491 TEST(SourceCodeTests, IsInsideMainFile){
492   TestTU TU;
493   TU.HeaderCode = R"cpp(
494     #define DEFINE_CLASS(X) class X {};
495     #define DEFINE_YY DEFINE_CLASS(YY)
496 
497     class Header1 {};
498     DEFINE_CLASS(Header2)
499     class Header {};
500   )cpp";
501   TU.Code = R"cpp(
502     class Main1 {};
503     DEFINE_CLASS(Main2)
504     DEFINE_YY
505     class Main {};
506   )cpp";
507   TU.ExtraArgs.push_back("-DHeader=Header3");
508   TU.ExtraArgs.push_back("-DMain=Main3");
509   auto AST = TU.build();
510   const auto& SM = AST.getSourceManager();
511   auto DeclLoc = [&AST](llvm::StringRef Name) {
512     return findDecl(AST, Name).getLocation();
513   };
514   for (const auto *HeaderDecl : {"Header1", "Header2", "Header3"})
515     EXPECT_FALSE(isInsideMainFile(DeclLoc(HeaderDecl), SM));
516 
517   for (const auto *MainDecl : {"Main1", "Main2", "Main3", "YY"})
518     EXPECT_TRUE(isInsideMainFile(DeclLoc(MainDecl), SM));
519 }
520 
521 // Test for functions toHalfOpenFileRange and getHalfOpenFileRange
522 TEST(SourceCodeTests, HalfOpenFileRange) {
523   // Each marked range should be the file range of the decl with the same name
524   // and each name should be unique.
525   Annotations Test(R"cpp(
526     #define FOO(X, Y) int Y = ++X
527     #define BAR(X) X + 1
528     #define ECHO(X) X
529 
530     #define BUZZ BAZZ(ADD)
531     #define BAZZ(m) m(1)
532     #define ADD(a) int f = a + 1;
533     template<typename T>
534     class P {};
535 
536     int main() {
537       $a[[P<P<P<P<P<int>>>>> a]];
538       $b[[int b = 1]];
539       $c[[FOO(b, c)]];
540       $d[[FOO(BAR(BAR(b)), d)]];
541       // FIXME: We might want to select everything inside the outer ECHO.
542       ECHO(ECHO($e[[int) ECHO(e]]));
543       // Shouldn't crash.
544       $f[[BUZZ]];
545     }
546   )cpp");
547 
548   ParsedAST AST = TestTU::withCode(Test.code()).build();
549   llvm::errs() << Test.code();
550   const SourceManager &SM = AST.getSourceManager();
551   const LangOptions &LangOpts = AST.getLangOpts();
552   // Turn a SourceLocation into a pair of positions
553   auto SourceRangeToRange = [&SM](SourceRange SrcRange) {
554     return Range{sourceLocToPosition(SM, SrcRange.getBegin()),
555                  sourceLocToPosition(SM, SrcRange.getEnd())};
556   };
557   auto CheckRange = [&](llvm::StringRef Name) {
558     const NamedDecl &Decl = findUnqualifiedDecl(AST, Name);
559     auto FileRange = toHalfOpenFileRange(SM, LangOpts, Decl.getSourceRange());
560     SCOPED_TRACE("Checking range: " + Name);
561     ASSERT_NE(FileRange, llvm::None);
562     Range HalfOpenRange = SourceRangeToRange(*FileRange);
563     EXPECT_EQ(HalfOpenRange, Test.ranges(Name)[0]);
564   };
565 
566   CheckRange("a");
567   CheckRange("b");
568   CheckRange("c");
569   CheckRange("d");
570   CheckRange("e");
571   CheckRange("f");
572 }
573 
574 TEST(SourceCodeTests, HalfOpenFileRangePathologicalPreprocessor) {
575   const char *Case = R"cpp(
576 #define MACRO while(1)
577     void test() {
578 [[#include "Expand.inc"
579         br^eak]];
580     }
581   )cpp";
582   Annotations Test(Case);
583   auto TU = TestTU::withCode(Test.code());
584   TU.AdditionalFiles["Expand.inc"] = "MACRO\n";
585   auto AST = TU.build();
586 
587   const auto &Func = cast<FunctionDecl>(findDecl(AST, "test"));
588   const auto &Body = cast<CompoundStmt>(Func.getBody());
589   const auto &Loop = cast<WhileStmt>(*Body->child_begin());
590   llvm::Optional<SourceRange> Range = toHalfOpenFileRange(
591       AST.getSourceManager(), AST.getLangOpts(), Loop->getSourceRange());
592   ASSERT_TRUE(Range) << "Failed to get file range";
593   EXPECT_EQ(AST.getSourceManager().getFileOffset(Range->getBegin()),
594             Test.llvm::Annotations::range().Begin);
595   EXPECT_EQ(AST.getSourceManager().getFileOffset(Range->getEnd()),
596             Test.llvm::Annotations::range().End);
597 }
598 
599 TEST(SourceCodeTests, IncludeHashLoc) {
600   const char *Case = R"cpp(
601 $foo^#include "foo.inc"
602 #define HEADER "bar.inc"
603   $bar^#  include HEADER
604   )cpp";
605   Annotations Test(Case);
606   auto TU = TestTU::withCode(Test.code());
607   TU.AdditionalFiles["foo.inc"] = "int foo;\n";
608   TU.AdditionalFiles["bar.inc"] = "int bar;\n";
609   auto AST = TU.build();
610   const auto& SM = AST.getSourceManager();
611 
612   FileID Foo = SM.getFileID(findDecl(AST, "foo").getLocation());
613   EXPECT_EQ(SM.getFileOffset(includeHashLoc(Foo, SM)),
614             Test.llvm::Annotations::point("foo"));
615   FileID Bar = SM.getFileID(findDecl(AST, "bar").getLocation());
616   EXPECT_EQ(SM.getFileOffset(includeHashLoc(Bar, SM)),
617             Test.llvm::Annotations::point("bar"));
618 }
619 
620 TEST(SourceCodeTests, GetEligiblePoints) {
621   constexpr struct {
622     const char *Code;
623     const char *FullyQualifiedName;
624     const char *EnclosingNamespace;
625   } Cases[] = {
626       {R"cpp(// FIXME: We should also mark positions before and after
627                  //declarations/definitions as eligible.
628               namespace ns1 {
629               namespace a { namespace ns2 {} }
630               namespace ns2 {^
631               void foo();
632               namespace {}
633               void bar() {}
634               namespace ns3 {}
635               class T {};
636               ^}
637               using namespace ns2;
638               })cpp",
639        "ns1::ns2::symbol", "ns1::ns2::"},
640       {R"cpp(
641               namespace ns1 {^
642               namespace a { namespace ns2 {} }
643               namespace b {}
644               namespace ns {}
645               ^})cpp",
646        "ns1::ns2::symbol", "ns1::"},
647       {R"cpp(
648               namespace x {
649               namespace a { namespace ns2 {} }
650               namespace b {}
651               namespace ns {}
652               }^)cpp",
653        "ns1::ns2::symbol", ""},
654       {R"cpp(
655               namespace ns1 {
656               namespace ns2 {^^}
657               namespace b {}
658               namespace ns2 {^^}
659               }
660               namespace ns1 {namespace ns2 {^^}})cpp",
661        "ns1::ns2::symbol", "ns1::ns2::"},
662       {R"cpp(
663               namespace ns1 {^
664               namespace ns {}
665               namespace b {}
666               namespace ns {}
667               ^}
668               namespace ns1 {^namespace ns {}^})cpp",
669        "ns1::ns2::symbol", "ns1::"},
670   };
671   for (auto Case : Cases) {
672     Annotations Test(Case.Code);
673 
674     auto Res = getEligiblePoints(Test.code(), Case.FullyQualifiedName,
675                                  format::getLLVMStyle());
676     EXPECT_THAT(Res.EligiblePoints, testing::ElementsAreArray(Test.points()))
677         << Test.code();
678     EXPECT_EQ(Res.EnclosingNamespace, Case.EnclosingNamespace) << Test.code();
679   }
680 }
681 
682 TEST(SourceCodeTests, IdentifierRanges) {
683   Annotations Code(R"cpp(
684    class [[Foo]] {};
685    // Foo
686    /* Foo */
687    void f([[Foo]]* foo1) {
688      [[Foo]] foo2;
689      auto S = [[Foo]]();
690 // cross-line identifier is not supported.
691 F\
692 o\
693 o foo2;
694    }
695   )cpp");
696   LangOptions LangOpts;
697   LangOpts.CPlusPlus = true;
698   EXPECT_EQ(Code.ranges(),
699             collectIdentifierRanges("Foo", Code.code(), LangOpts));
700 }
701 
702 TEST(SourceCodeTests, isHeaderFile) {
703   // Without lang options.
704   EXPECT_TRUE(isHeaderFile("foo.h"));
705   EXPECT_TRUE(isHeaderFile("foo.hh"));
706   EXPECT_TRUE(isHeaderFile("foo.hpp"));
707 
708   EXPECT_FALSE(isHeaderFile("foo.cpp"));
709   EXPECT_FALSE(isHeaderFile("foo.c++"));
710   EXPECT_FALSE(isHeaderFile("foo.cxx"));
711   EXPECT_FALSE(isHeaderFile("foo.cc"));
712   EXPECT_FALSE(isHeaderFile("foo.c"));
713   EXPECT_FALSE(isHeaderFile("foo.mm"));
714   EXPECT_FALSE(isHeaderFile("foo.m"));
715 
716   // With lang options
717   LangOptions LangOpts;
718   LangOpts.IsHeaderFile = true;
719   EXPECT_TRUE(isHeaderFile("string", LangOpts));
720   // Emulate cases where there is no "-x header" flag for a .h file, we still
721   // want to treat it as a header.
722   LangOpts.IsHeaderFile = false;
723   EXPECT_TRUE(isHeaderFile("header.h", LangOpts));
724 }
725 
726 } // namespace
727 } // namespace clangd
728 } // namespace clang
729