16#include "llvm/ADT/StringRef.h"
17#include "llvm/Support/ScopedPrinter.h"
18#include "llvm/Support/raw_ostream.h"
19#include "gmock/gmock.h"
20#include "gtest/gtest.h"
36using ::testing::ElementsAre;
37using ::testing::IsEmpty;
39constexpr InlayHintOptions DefaultOptsForTests{2};
41std::vector<InlayHint> hintsOfKind(ParsedAST &
AST, InlayHintKind Kind,
42 InlayHintOptions Opts) {
43 std::vector<InlayHint> Result;
45 if (Hint.kind == Kind)
46 Result.push_back(Hint);
55 std::string RangeName;
58 friend llvm::raw_ostream &
operator<<(llvm::raw_ostream &Stream,
59 const ExpectedHint &Hint) {
60 return Stream << Hint.Label <<
"@$" << Hint.RangeName;
64MATCHER_P2(HintMatcher, Expected, Code, llvm::to_string(Expected)) {
65 llvm::StringRef ExpectedView(Expected.Label);
66 std::string ResultLabel = arg.joinLabels();
67 if (ResultLabel != ExpectedView.trim(
" ") ||
68 arg.paddingLeft != ExpectedView.starts_with(
" ") ||
69 arg.paddingRight != ExpectedView.ends_with(
" ")) {
70 *result_listener <<
"label is '" << ResultLabel <<
"'";
73 if (arg.range != Code.range(Expected.RangeName)) {
74 *result_listener <<
"range is " << llvm::to_string(arg.range) <<
" but $"
75 << Expected.RangeName <<
" is "
76 << llvm::to_string(Code.range(Expected.RangeName));
86 C.InlayHints.Parameters =
false;
87 C.InlayHints.DeducedTypes =
false;
88 C.InlayHints.Designators =
false;
89 C.InlayHints.BlockEnd =
false;
90 C.InlayHints.DefaultArguments =
false;
94template <
typename... ExpectedHints>
95void assertHintsWithHeader(
InlayHintKind Kind, llvm::StringRef AnnotatedSource,
97 ExpectedHints... Expected) {
100 TU.ExtraArgs.push_back(
"-std=c++23");
101 TU.HeaderCode = HeaderContent;
102 auto AST = TU.build();
104 EXPECT_THAT(hintsOfKind(
AST, Kind, Opts),
105 ElementsAre(HintMatcher(Expected, Source)...));
112template <
typename... ExpectedHints>
113void assertHints(
InlayHintKind Kind, llvm::StringRef AnnotatedSource,
115 return assertHintsWithHeader(Kind, AnnotatedSource,
"", Opts,
116 std::move(Expected)...);
120template <
typename...
T>
void ignore(T &&...) {}
122template <
typename... ExpectedHints>
123void assertParameterHints(llvm::StringRef AnnotatedSource,
124 ExpectedHints... Expected) {
125 ignore(Expected.Side =
Left...);
130template <
typename... ExpectedHints>
131void assertTypeHints(llvm::StringRef AnnotatedSource,
132 ExpectedHints... Expected) {
133 ignore(Expected.Side =
Right...);
138template <
typename... ExpectedHints>
139void assertDesignatorHints(llvm::StringRef AnnotatedSource,
140 ExpectedHints... Expected) {
148template <
typename... ExpectedHints>
149void assertBlockEndHintsWithOpts(llvm::StringRef AnnotatedSource,
151 ExpectedHints... Expected) {
158template <
typename... ExpectedHints>
159void assertBlockEndHints(llvm::StringRef AnnotatedSource,
160 ExpectedHints... Expected) {
161 assertBlockEndHintsWithOpts(AnnotatedSource, DefaultOptsForTests,
165TEST(ParameterHints, Smoke) {
166 assertParameterHints(R
"cpp(
172 ExpectedHint{"param: ",
"param"});
175TEST(ParameterHints, NoName) {
177 assertParameterHints(R
"cpp(
185TEST(ParameterHints, NoNameConstReference) {
187 assertParameterHints(R
"cpp(
188 void foo(const int&);
195TEST(ParameterHints, NoNameReference) {
197 assertParameterHints(R
"cpp(
204 ExpectedHint{"&: ",
"param"});
207TEST(ParameterHints, NoNameRValueReference) {
209 assertParameterHints(R
"cpp(
217TEST(ParameterHints, NoNameVariadicDeclaration) {
219 assertParameterHints(R
"cpp(
220 template <typename... Args>
221 void foo(Args&& ...);
228TEST(ParameterHints, NoNameVariadicForwarded) {
231 assertParameterHints(R
"cpp(
232 namespace std { template <typename T> T&& forward(T&); }
234 template <typename... Args>
235 void bar(Args&&... args) { return foo(std::forward<Args>(args)...); }
242TEST(ParameterHints, NoNameVariadicPlain) {
244 assertParameterHints(R
"cpp(
246 template <typename... Args>
247 void bar(Args&&... args) { return foo(args...); }
254TEST(ParameterHints, NameInDefinition) {
256 assertParameterHints(R
"cpp(
261 void foo(int param) {};
263 ExpectedHint{"param: ",
"param"});
266TEST(ParameterHints, NamePartiallyInDefinition) {
268 assertParameterHints(R
"cpp(
269 void foo(int, int b);
271 foo($param1[[42]], $param2[[42]]);
273 void foo(int a, int) {};
275 ExpectedHint{"a: ",
"param1"},
276 ExpectedHint{
"b: ",
"param2"});
279TEST(ParameterHints, NameInDefinitionVariadic) {
281 assertParameterHints(R
"cpp(
283 template <typename... Args>
284 void bar(Args... args) {
288 bar($param1[[42]], $param2[[42]]);
290 void foo(int a, int b) {};
292 ExpectedHint{"a: ",
"param1"},
293 ExpectedHint{
"b: ",
"param2"});
296TEST(ParameterHints, NameMismatch) {
298 assertParameterHints(R
"cpp(
303 void foo(int bad) {};
305 ExpectedHint{"good: ",
"good"});
308TEST(ParameterHints, NameConstReference) {
310 assertParameterHints(R
"cpp(
311 void foo(const int& param);
316 ExpectedHint{"param: ",
"param"});
319TEST(ParameterHints, NameTypeAliasConstReference) {
321 assertParameterHints(R
"cpp(
322 using alias = const int&;
323 void foo(alias param);
329 ExpectedHint{"param: ",
"param"});
332TEST(ParameterHints, NameReference) {
334 assertParameterHints(R
"cpp(
335 void foo(int& param);
341 ExpectedHint{"¶m: ",
"param"});
344TEST(ParameterHints, NameTypeAliasReference) {
346 assertParameterHints(R
"cpp(
348 void foo(alias param);
354 ExpectedHint{"¶m: ",
"param"});
357TEST(ParameterHints, NameRValueReference) {
359 assertParameterHints(R
"cpp(
360 void foo(int&& param);
365 ExpectedHint{"param: ",
"param"});
368TEST(ParameterHints, VariadicForwardedConstructor) {
371 assertParameterHints(R
"cpp(
372 namespace std { template <typename T> T&& forward(T&); }
373 struct S { S(int a); };
374 template <typename T, typename... Args>
375 T bar(Args&&... args) { return T{std::forward<Args>(args)...}; }
381 ExpectedHint{"a: ",
"param"});
384TEST(ParameterHints, VariadicPlainConstructor) {
386 assertParameterHints(R
"cpp(
387 struct S { S(int a); };
388 template <typename T, typename... Args>
389 T bar(Args&&... args) { return T{args...}; }
395 ExpectedHint{"a: ",
"param"});
398TEST(ParameterHints, VariadicForwardedNewConstructor) {
401 assertParameterHints(R
"cpp(
402 namespace std { template <typename T> T&& forward(T&); }
403 struct S { S(int a); };
404 template <typename T, typename... Args>
405 T* bar(Args&&... args) { return new T{std::forward<Args>(args)...}; }
411 ExpectedHint{"a: ",
"param"});
414TEST(ParameterHints, VariadicPlainNewConstructor) {
416 assertParameterHints(R
"cpp(
417 struct S { S(int a); };
418 template <typename T, typename... Args>
419 T* bar(Args&&... args) { return new T{args...}; }
425 ExpectedHint{"a: ",
"param"});
428TEST(ParameterHints, VariadicForwarded) {
431 assertParameterHints(R
"cpp(
432 namespace std { template <typename T> T&& forward(T&); }
434 template <typename... Args>
435 void bar(Args&&... args) { return foo(std::forward<Args>(args)...); }
441 ExpectedHint{"a: ",
"param"});
444TEST(ParameterHints, VariadicPlain) {
446 assertParameterHints(R
"cpp(
448 template <typename... Args>
449 void bar(Args&&... args) { return foo(args...); }
454 ExpectedHint{"a: ",
"param"});
457TEST(ParameterHints, VariadicPlainWithPackFirst) {
460 assertParameterHints(R
"cpp(
462 template <typename... Args, typename Arg>
463 void bar(Arg, Args&&... args) { return foo(args...); }
465 bar(1, $param[[42]]);
468 ExpectedHint{"a: ",
"param"});
471TEST(ParameterHints, VariadicSplitTwolevel) {
475 assertParameterHints(R
"cpp(
476 namespace std { template <typename T> T&& forward(T&); }
477 void baz(int, int b, double);
478 template <typename... Args>
479 void foo(int a, Args&&... args) {
480 return baz(1, std::forward<Args>(args)..., 1.0);
482 template <typename... Args>
483 void bar(Args&&... args) { return foo(std::forward<Args>(args)...); }
485 bar($param1[[32]], $param2[[42]]);
488 ExpectedHint{"a: ",
"param1"},
489 ExpectedHint{
"b: ",
"param2"});
492TEST(ParameterHints, VariadicNameFromSpecialization) {
495 assertParameterHints(R
"cpp(
497 template <typename... Args>
498 void bar(Args... args) {
502 void bar<int>(int b);
507 ExpectedHint{"b: ",
"param"});
510TEST(ParameterHints, VariadicNameFromSpecializationRecursive) {
513 assertParameterHints(R
"cpp(
515 template <typename... Args>
516 void foo(Args... args) {
519 template <typename... Args>
520 void bar(Args... args) {
524 void foo<int>(int b);
529 ExpectedHint{"b: ",
"param"});
532TEST(ParameterHints, VariadicOverloaded) {
536 assertParameterHints(
538 namespace std { template <typename T> T&& forward(T&); }
539 void baz(int b, int c);
540 void baz(int bb, int cc, int dd);
541 template <typename... Args>
542 void foo(int a, Args&&... args) {
543 return baz(std::forward<Args>(args)...);
545 template <typename... Args>
546 void bar(Args&&... args) { return foo(std::forward<Args>(args)...); }
548 bar($param1[[32]], $param2[[42]], $param3[[52]]);
549 bar($param4[[1]], $param5[[2]], $param6[[3]], $param7[[4]]);
552 ExpectedHint{"a: ",
"param1"}, ExpectedHint{
"b: ",
"param2"},
553 ExpectedHint{
"c: ",
"param3"}, ExpectedHint{
"a: ",
"param4"},
554 ExpectedHint{
"bb: ",
"param5"}, ExpectedHint{
"cc: ",
"param6"},
555 ExpectedHint{
"dd: ",
"param7"});
558TEST(ParameterHints, VariadicRecursive) {
560 assertParameterHints(
564 template <typename Head, typename... Tail>
565 void foo(Head head, Tail... tail) {
569 template <typename... Args>
570 void bar(Args... args) {
580TEST(ParameterHints, VariadicVarargs) {
582 assertParameterHints(R
"cpp(
583 void foo(int fixed, ...);
584 template <typename... Args>
585 void bar(Args&&... args) {
590 bar($fixed[[41]], 42, 43);
595TEST(ParameterHints, VariadicTwolevelUnresolved) {
597 assertParameterHints(R
"cpp(
598 template <typename... Args>
599 void foo(int fixed, Args&& ... args);
600 template <typename... Args>
601 void bar(Args&&... args) {
606 bar($fixed[[41]], 42, 43);
609 ExpectedHint{"fixed: ",
"fixed"});
612TEST(ParameterHints, VariadicTwoCalls) {
614 assertParameterHints(
616 void f1(int a, int b);
617 void f2(int c, int d);
621 template <typename... Args>
622 void foo(Args... args) {
631 foo($param1[[1]], $param2[[2]]);
634 ExpectedHint{"a: ",
"param1"}, ExpectedHint{
"b: ",
"param2"});
637TEST(ParameterHints, VariadicInfinite) {
639 assertParameterHints(
641 template <typename... Args>
644 template <typename... Args>
645 void bar(Args... args) {
649 template <typename... Args>
650 void foo(Args... args) {
660TEST(ParameterHints, VariadicDuplicatePack) {
662 assertParameterHints(
664 void foo(int a, int b, int c, int);
666 template <typename... Args>
667 void bar(int, Args... args, int d) {
671 template <typename... Args>
672 void baz(Args... args, Args... args2) {
673 bar<Args..., int>(1, args..., args2...);
677 baz<int, int>($p1[[1]], $p2[[2]], $p3[[3]], $p4[[4]]);
680 ExpectedHint{"a: ",
"p1"}, ExpectedHint{
"b: ",
"p2"},
681 ExpectedHint{
"c: ",
"p3"}, ExpectedHint{
"d: ",
"p4"});
684TEST(ParameterHints, VariadicEmplace) {
687 assertParameterHints(
689 namespace std { template <typename T> T&& forward(T&); }
690 using size_t = decltype(sizeof(0));
691 void *operator new(size_t, void *);
697 template <typename T>
699 template <typename T, typename... Args>
700 void construct(T* ptr, Args&&... args) {
701 ::new ((void*)ptr) T{std::forward<Args>(args)...};
704 template <typename T>
706 template <typename... Args>
707 void emplace(Args&&... args) {
709 auto ptr = a.template allocate<T>();
710 a.construct(ptr, std::forward<Args>(args)...);
715 c.emplace($param1[[1]]);
716 c.emplace($param2[[2]], $param3[[3]]);
719 ExpectedHint{"A: ",
"param1"}, ExpectedHint{
"B: ",
"param2"},
720 ExpectedHint{
"C: ",
"param3"});
723TEST(ParameterHints, VariadicReferenceHint) {
724 assertParameterHints(R
"cpp(
726 template <typename... Args>
727 void bar(Args... args) { return foo(args...); }
736TEST(ParameterHints, VariadicReferenceHintForwardingRef) {
737 assertParameterHints(R"cpp(
739 template <typename... Args>
740 void bar(Args&&... args) { return foo(args...); }
747 ExpectedHint{"&: ",
"param"});
750TEST(ParameterHints, VariadicReferenceHintForwardingRefStdForward) {
751 assertParameterHints(R
"cpp(
752 namespace std { template <typename T> T&& forward(T&); }
754 template <typename... Args>
755 void bar(Args&&... args) { return foo(std::forward<Args>(args)...); }
761 ExpectedHint{"&: ",
"param"});
764TEST(ParameterHints, VariadicNoReferenceHintForwardingRefStdForward) {
765 assertParameterHints(R
"cpp(
766 namespace std { template <typename T> T&& forward(T&); }
768 template <typename... Args>
769 void bar(Args&&... args) { return foo(std::forward<Args>(args)...); }
778TEST(ParameterHints, VariadicNoReferenceHintUnresolvedForward) {
779 assertParameterHints(R"cpp(
780 template <typename... Args>
781 void foo(Args&&... args);
789TEST(ParameterHints, MatchingNameVariadicForwarded) {
792 assertParameterHints(R
"cpp(
793 namespace std { template <typename T> T&& forward(T&); }
795 template <typename... Args>
796 void bar(Args&&... args) { return foo(std::forward<Args>(args)...); }
804TEST(ParameterHints, MatchingNameVariadicPlain) {
806 assertParameterHints(R
"cpp(
808 template <typename... Args>
809 void bar(Args&&... args) { return foo(args...); }
819 assertParameterHints(R
"cpp(
821 void operator+(S lhs, S rhs);
829TEST(ParameterHints, FunctionCallOperator) {
830 assertParameterHints(R"cpp(
832 void operator()(int x);
836 static void operator()(int x, int y);
839 auto l1 = [](int x) {};
840 auto l2 = [](int x) static {};
844 s.operator()($2[[1]]);
845 s.operator()($3[[1]], $4[[2]]);
846 S::operator()($5[[1]], $6[[2]]);
849 l1.operator()($8[[1]]);
851 l2.operator()($10[[1]]);
853 void (*ptr)(int a, int b) = &S::operator();
854 ptr($11[[1]], $12[[2]]);
857 ExpectedHint{"x: ",
"1"}, ExpectedHint{
"x: ",
"2"},
858 ExpectedHint{
"x: ",
"3"}, ExpectedHint{
"y: ",
"4"},
859 ExpectedHint{
"x: ",
"5"}, ExpectedHint{
"y: ",
"6"},
860 ExpectedHint{
"x: ",
"7"}, ExpectedHint{
"x: ",
"8"},
861 ExpectedHint{
"x: ",
"9"}, ExpectedHint{
"x: ",
"10"},
862 ExpectedHint{
"a: ",
"11"}, ExpectedHint{
"b: ",
"12"});
865TEST(ParameterHints, DeducingThis) {
866 assertParameterHints(R
"cpp(
868 template <typename This>
869 auto operator()(this This &&Self, int Param) {
873 auto function(this auto &Self, int Param) {
880 s.function($2[[42]]);
882 auto lambda = [](this auto &Self, char C) -> void {
888 ExpectedHint{"Param: ",
"1"},
889 ExpectedHint{
"Param: ",
"2"},
890 ExpectedHint{
"Param: ",
"3"}, ExpectedHint{
"C: ",
"4"});
893TEST(ParameterHints, DependentDeducingThis) {
894 assertParameterHints(R
"cpp(
895 template <typename T>
897 void f1(this S& obj);
898 void f2(this S& obj, int x, int y);
901 s.f2($x[[42]], $y[[43]]);
905 ExpectedHint{"x: ",
"x"}, ExpectedHint{
"y: ",
"y"});
908TEST(ParameterHints, Macros) {
913 assertParameterHints(R
"cpp(
915 #define ExpandsToCall() foo(42)
923 assertParameterHints(R
"cpp(
925 void foo(double param);
930 ExpectedHint{"param: ",
"param"});
933 assertParameterHints(R
"cpp(
935 #define ASSERT(expr) if (!expr) abort()
938 ASSERT(foo($param[[42]]) == 0);
941 ExpectedHint{"param: ",
"param"});
944 assertParameterHints(R
"cpp(
945 void foo(double x, double y);
946 #define CONSTANTS 3.14, 2.72
953TEST(ParameterHints, ConstructorParens) {
954 assertParameterHints(R"cpp(
962 ExpectedHint{"param: ",
"param"});
965TEST(ParameterHints, ConstructorBraces) {
966 assertParameterHints(R
"cpp(
974 ExpectedHint{"param: ",
"param"});
977TEST(ParameterHints, ConstructorStdInitList) {
979 assertParameterHints(R
"cpp(
981 template <typename E> class initializer_list { const E *a, *b; };
984 S(std::initializer_list<int> param);
992TEST(ParameterHints, MemberInit) {
993 assertParameterHints(R"cpp(
999 T() : member($param[[42]]) {}
1002 ExpectedHint{"param: ",
"param"});
1005TEST(ParameterHints, ImplicitConstructor) {
1006 assertParameterHints(R
"cpp(
1012 // Do not show hint for implicit constructor call in argument.
1014 // Do not show hint for implicit constructor call in return.
1020TEST(ParameterHints, FunctionPointer) {
1021 assertParameterHints(
1023 void (*f1)(int param);
1024 void (__stdcall *f2)(int param);
1025 using f3_t = void(*)(int param);
1027 using f4_t = void(__stdcall *)(int param);
1029 __attribute__((noreturn)) f4_t f5;
1035 // This one runs into an edge case in clang's type model
1036 // and we can't extract the parameter name. But at least
1037 // we shouldn't crash.
1041 ExpectedHint{"param: ",
"f1"}, ExpectedHint{
"param: ",
"f2"},
1042 ExpectedHint{
"param: ",
"f3"}, ExpectedHint{
"param: ",
"f4"});
1045TEST(ParameterHints, ArgMatchesParam) {
1046 assertParameterHints(R
"cpp(
1047 void foo(int param);
1049 static const int param = 42;
1053 // Do not show redundant "param: param".
1055 // But show it if the argument is qualified.
1056 foo($param[[S::param]]);
1061 // Do not show "param: param" for member-expr.
1066 ExpectedHint{"param: ",
"param"});
1069TEST(ParameterHints, ArgMatchesParamReference) {
1070 assertParameterHints(R
"cpp(
1071 void foo(int& param);
1072 void foo2(const int& param);
1075 // show reference hint on mutable reference
1076 foo($param[[param]]);
1077 // but not on const reference
1081 ExpectedHint{"&: ",
"param"});
1084TEST(ParameterHints, LeadingUnderscore) {
1085 assertParameterHints(R
"cpp(
1086 void foo(int p1, int _p2, int __p3);
1088 foo($p1[[41]], $p2[[42]], $p3[[43]]);
1091 ExpectedHint{"p1: ",
"p1"}, ExpectedHint{
"p2: ",
"p2"},
1092 ExpectedHint{
"p3: ",
"p3"});
1095TEST(ParameterHints, DependentCalls) {
1096 assertParameterHints(R
"cpp(
1097 template <typename T>
1098 void nonmember(T par1);
1100 template <typename T>
1102 void member(T par2);
1103 static void static_member(T par3);
1106 void overload(int anInt);
1107 void overload(double aDouble);
1109 template <typename T>
1111 void bar(A<T> a, T t) {
1112 nonmember($par1[[t]]);
1113 a.member($par2[[t]]);
1114 A<T>::static_member($par3[[t]]);
1115 // We don't want to arbitrarily pick between
1116 // "anInt" or "aDouble", so just show no hint.
1121 ExpectedHint{"par1: ",
"par1"},
1122 ExpectedHint{
"par2: ",
"par2"},
1123 ExpectedHint{
"par3: ",
"par3"});
1126TEST(ParameterHints, VariadicFunction) {
1127 assertParameterHints(R
"cpp(
1128 template <typename... T>
1129 void foo(int fixed, T... variadic);
1132 foo($fixed[[41]], 42, 43);
1135 ExpectedHint{"fixed: ",
"fixed"});
1138TEST(ParameterHints, VarargsFunction) {
1139 assertParameterHints(R
"cpp(
1140 void foo(int fixed, ...);
1143 foo($fixed[[41]], 42, 43);
1146 ExpectedHint{"fixed: ",
"fixed"});
1149TEST(ParameterHints, CopyOrMoveConstructor) {
1151 assertParameterHints(R
"cpp(
1165TEST(ParameterHints, UserDefinedLiteral) {
1167 assertParameterHints(R
"cpp(
1168 long double operator"" _w(long double param);
1175TEST(ParameterHints, ParamNameComment) {
1178 assertParameterHints(R
"cpp(
1179 void foo(int param);
1182 foo( /* param = */ 42);
1186 foo(/*param=*/Z(a));
1187 foo($macro[[Z(a)]]);
1188 foo(/* the answer */$param[[42]]);
1191 ExpectedHint{"param: ",
"macro"},
1192 ExpectedHint{
"param: ",
"param"});
1195TEST(ParameterHints, SetterFunctions) {
1196 assertParameterHints(R
"cpp(
1198 void setParent(S* parent);
1199 void set_parent(S* parent);
1200 void setTimeout(int timeoutMillis);
1201 void setTimeoutMillis(int timeout_millis);
1205 // Parameter name matches setter name - omit hint.
1206 s.setParent(nullptr);
1207 // Support snake_case
1208 s.set_parent(nullptr);
1209 // Parameter name may contain extra info - show hint.
1210 s.setTimeout($timeoutMillis[[120]]);
1211 // FIXME: Ideally we'd want to omit this.
1212 s.setTimeoutMillis($timeout_millis[[120]]);
1215 ExpectedHint{"timeoutMillis: ",
"timeoutMillis"},
1216 ExpectedHint{
"timeout_millis: ",
"timeout_millis"});
1219TEST(ParameterHints, BuiltinFunctions) {
1221 assertParameterHints(R
"cpp(
1222 namespace std { template <typename T> T&& forward(T&); }
1230TEST(ParameterHints, IncludeAtNonGlobalScope) {
1232 void bar() { foo(42); }
1236 void foo(int param);
1242 Workspace.addSource("foo.inc", FooInc.code());
1243 Workspace.addMainFile(
"foo.cc", FooCC.code());
1245 auto AST = Workspace.openFile(
"foo.cc");
1246 ASSERT_TRUE(
bool(
AST));
1254TEST(TypeHints, Smoke) {
1255 assertTypeHints(R
"cpp(
1256 auto $waldo[[waldo]] = 42;
1258 ExpectedHint{": int",
"waldo"});
1261TEST(TypeHints, Decorations) {
1262 assertTypeHints(R
"cpp(
1264 auto* $var1[[var1]] = &x;
1265 auto&& $var2[[var2]] = x;
1266 const auto& $var3[[var3]] = x;
1268 ExpectedHint{": int *",
"var1"},
1269 ExpectedHint{
": int &",
"var2"},
1270 ExpectedHint{
": const int &",
"var3"});
1273TEST(TypeHints, DecltypeAuto) {
1274 assertTypeHints(R
"cpp(
1277 decltype(auto) $z[[z]] = y;
1279 ExpectedHint{": int &",
"z"});
1282TEST(TypeHints, NoQualifiers) {
1283 assertTypeHints(R
"cpp(
1288 auto $x[[x]] = foo();
1291 template <typename T>
1294 S2::Inner<int> bar();
1295 auto $y[[y]] = bar();
1299 ExpectedHint{": S1",
"x"}, ExpectedHint{
": Inner<int>",
"y"});
1302TEST(TypeHints, Lambda) {
1305 assertTypeHints(R
"cpp(
1308 auto $L[[L]] = [cap, $init[[init]] = 1 + 1](int a$ret[[)]] {
1309 return a + cap + init;
1313 ExpectedHint{": (lambda)",
"L"},
1314 ExpectedHint{
": int",
"init"}, ExpectedHint{
"-> int",
"ret"});
1318 assertTypeHints(
"auto $L[[x]] = <:$ret[[:>]]{return 42;};",
1319 ExpectedHint{
": (lambda)",
"L"},
1320 ExpectedHint{
"-> int",
"ret"});
1326TEST(TypeHints, StructuredBindings_PublicStruct) {
1327 assertTypeHints(R
"cpp(
1328 // Struct with public fields.
1334 auto [$x[[x]], $y[[y]]] = foo();
1336 ExpectedHint{": int",
"x"}, ExpectedHint{
": int",
"y"});
1339TEST(TypeHints, StructuredBindings_Array) {
1340 assertTypeHints(R
"cpp(
1342 auto [$x[[x]], $y[[y]]] = arr;
1344 ExpectedHint{": int",
"x"}, ExpectedHint{
": int",
"y"});
1347TEST(TypeHints, StructuredBindings_TupleLike) {
1348 assertTypeHints(R
"cpp(
1355 template <typename T>
1356 struct tuple_size {};
1358 struct tuple_size<IntPair> {
1359 constexpr static unsigned value = 2;
1361 template <unsigned I, typename T>
1362 struct tuple_element {};
1363 template <unsigned I>
1364 struct tuple_element<I, IntPair> {
1368 template <unsigned I>
1369 int get(const IntPair& p) {
1370 if constexpr (I == 0) {
1372 } else if constexpr (I == 1) {
1377 auto [$x[[x]], $y[[y]]] = bar();
1379 ExpectedHint{": int",
"x"}, ExpectedHint{
": int",
"y"});
1382TEST(TypeHints, StructuredBindings_NoInitializer) {
1383 assertTypeHints(R
"cpp(
1384 // No initializer (ill-formed).
1385 // Do not show useless "NULL TYPE" hint.
1386 auto [x, y]; /*error-ok*/
1390TEST(TypeHints, InvalidType) {
1391 assertTypeHints(R"cpp(
1392 auto x = (unknown_type)42; /*error-ok*/
1393 auto *y = (unknown_ptr)nullptr;
1397TEST(TypeHints, ReturnTypeDeduction) {
1400 auto f1(int x$ret1a[[)]]; // Hint forward declaration too
1401 auto f1(int x$ret1b[[)]] { return x + 1; }
1403 // Include pointer operators in hint
1405 auto& f2($ret2[[)]] { return s; }
1407 // Do not hint `auto` for trailing return type.
1410 // Do not hint when a trailing return type is specified.
1411 auto f4() -> auto* { return "foo"; }
1413 auto f5($noreturn[[)]] {}
1415 // `auto` conversion operator
1417 operator auto($retConv[[)]] { return 42; }
1420 // FIXME: Dependent types do not work yet.
1421 template <typename T>
1423 auto method() { return T(); }
1426 ExpectedHint{"-> int",
"ret1a"}, ExpectedHint{
"-> int",
"ret1b"},
1427 ExpectedHint{
"-> int &",
"ret2"}, ExpectedHint{
"-> void",
"noreturn"},
1428 ExpectedHint{
"-> int",
"retConv"});
1431TEST(TypeHints, DependentType) {
1432 assertTypeHints(R
"cpp(
1433 template <typename T>
1435 // The hint would just be "auto" and we can't do any better.
1436 auto var1 = arg.method();
1437 // FIXME: It would be nice to show "T" as the hint.
1438 auto $var2[[var2]] = arg;
1441 template <typename T>
1446 ExpectedHint{": T",
"var2"});
1449TEST(TypeHints, LongTypeName) {
1450 assertTypeHints(R
"cpp(
1451 template <typename, typename, typename>
1453 struct MultipleWords {};
1454 A<MultipleWords, MultipleWords, MultipleWords> foo();
1455 // Omit type hint past a certain length (currently 32)
1465 template <typename, typename, typename>
1467 struct MultipleWords {};
1468 A<MultipleWords, MultipleWords, MultipleWords> foo();
1469 // Should have type hint with TypeNameLimit = 0
1470 auto $var[[var]] = foo();
1472 ExpectedHint{": A<MultipleWords, MultipleWords, MultipleWords>",
"var"});
1475TEST(TypeHints, DefaultTemplateArgs) {
1476 assertTypeHints(R
"cpp(
1477 template <typename, typename = int>
1480 auto $var[[var]] = foo();
1482 auto [$binding[[value]]] = bar;
1484 ExpectedHint{": A<float>",
"var"},
1485 ExpectedHint{
": A<float>",
"binding"});
1488TEST(DefaultArguments, Smoke) {
1492 Cfg.InlayHints.DeducedTypes =
false;
1493 Cfg.InlayHints.Designators =
false;
1494 Cfg.InlayHints.BlockEnd =
false;
1496 Cfg.InlayHints.DefaultArguments =
true;
1499 const auto *Code = R
"cpp(
1500 int foo(int A = 4) { return A; }
1501 int bar(int A, int B = 1, bool C = foo($default1[[)]]) { return A; }
1502 int A = bar($explicit[[2]]$default2[[)]];
1504 void baz(int = 5) { if (false) baz($unnamed[[)]]; };
1508 ExpectedHint{"A: 4",
"default1",
Left},
1509 ExpectedHint{
", B: 1, C: foo()",
"default2",
Left},
1510 ExpectedHint{
"5",
"unnamed",
Left});
1513 ExpectedHint{
"A: ",
"explicit",
Left});
1516TEST(DefaultArguments, WithoutParameterNames) {
1519 Cfg.InlayHints.DeducedTypes =
false;
1520 Cfg.InlayHints.Designators =
false;
1521 Cfg.InlayHints.BlockEnd =
false;
1523 Cfg.InlayHints.DefaultArguments =
true;
1526 const auto *Code = R
"cpp(
1532 Foo(int, Baz baz = //
1533 Baz{$abbreviated[[}]]
1540 Foo foo1(1$paren[[)]];
1541 Foo foo2{2$brace1[[}]];
1542 Foo foo3 = {3$brace2[[}]];
1543 auto foo4 = Foo{4$brace3[[}]];
1548 ExpectedHint{"...",
"abbreviated",
Left},
1549 ExpectedHint{
", Baz{}",
"paren",
Left},
1550 ExpectedHint{
", Baz{}",
"brace1",
Left},
1551 ExpectedHint{
", Baz{}",
"brace2",
Left},
1552 ExpectedHint{
", Baz{}",
"brace3",
Left});
1557TEST(TypeHints, Deduplication) {
1558 assertTypeHints(R
"cpp(
1559 template <typename T>
1561 auto $var[[var]] = 42;
1563 template void foo<int>();
1564 template void foo<float>();
1566 ExpectedHint{": int",
"var"});
1569TEST(TypeHints, SinglyInstantiatedTemplate) {
1570 assertTypeHints(R
"cpp(
1571 auto $lambda[[x]] = [](auto *$param[[y]], auto) { return 42; };
1572 int m = x("foo", 3);
1574 ExpectedHint{": (lambda)",
"lambda"},
1575 ExpectedHint{
": const char *",
"param"});
1578 assertTypeHints(R
"cpp(
1579 int x(auto $a[[a]], auto... b, auto c) { return 42; }
1580 int m = x<void*, char, float>(nullptr, 'c', 2.0, 2);
1582 ExpectedHint{": void *",
"a"});
1585TEST(TypeHints, Aliased) {
1589 TU.ExtraArgs.push_back(
"-xc");
1590 auto AST = TU.build();
1596TEST(TypeHints, CallingConvention) {
1601 []($lambda[[)]]__cdecl {};
1605 TU.ExtraArgs.push_back("--target=x86_64-w64-mingw32");
1606 TU.PredefineMacros =
true;
1607 auto AST = TU.build();
1611 ElementsAre(HintMatcher(ExpectedHint{
"-> void",
"lambda"}, Source)));
1614TEST(TypeHints, Decltype) {
1615 assertTypeHints(R
"cpp(
1616 $a[[decltype(0)]] a;
1617 $b[[decltype(a)]] b;
1618 const $c[[decltype(0)]] &c = b;
1620 // Don't show for dependent type
1622 constexpr decltype(T{}) d;
1624 $e[[decltype(0)]] e();
1625 auto f() -> $f[[decltype(0)]];
1627 template <class, class> struct Foo;
1628 using G = Foo<$g[[decltype(0)]], float>;
1630 auto $h[[h]] = $i[[decltype(0)]]{};
1636 ExpectedHint{": int",
"a"}, ExpectedHint{
": int",
"b"},
1637 ExpectedHint{
": int",
"c"}, ExpectedHint{
": int",
"e"},
1638 ExpectedHint{
": int",
"f"}, ExpectedHint{
": int",
"g"},
1639 ExpectedHint{
": int",
"h"}, ExpectedHint{
": int",
"i"});
1642TEST(TypeHints, SubstTemplateParameterAliases) {
1643 llvm::StringRef Header = R
"cpp(
1644 template <class T> struct allocator {};
1646 template <class T, class A>
1647 struct vector_base {
1651 template <class T, class A>
1652 struct internal_iterator_type_template_we_dont_expect {};
1654 struct my_iterator {};
1656 template <class T, class A = allocator<T>>
1657 struct vector : vector_base<T, A> {
1658 using base = vector_base<T, A>;
1659 typedef T value_type;
1660 typedef base::pointer pointer;
1661 using allocator_type = A;
1662 using size_type = int;
1663 using iterator = internal_iterator_type_template_we_dont_expect<T, A>;
1664 using non_template_iterator = my_iterator;
1666 value_type& operator[](int index) { return elements[index]; }
1667 const value_type& at(int index) const { return elements[index]; }
1668 pointer data() { return &elements[0]; }
1669 allocator_type get_allocator() { return A(); }
1670 size_type size() const { return 10; }
1671 iterator begin() { return iterator(); }
1672 non_template_iterator end() { return non_template_iterator(); }
1678 llvm::StringRef VectorIntPtr = R"cpp(
1679 vector<int *> array;
1680 auto $no_modifier[[x]] = array[3];
1681 auto* $ptr_modifier[[ptr]] = &array[3];
1682 auto& $ref_modifier[[ref]] = array[3];
1683 auto& $at[[immutable]] = array.at(3);
1685 auto $data[[data]] = array.data();
1686 auto $allocator[[alloc]] = array.get_allocator();
1687 auto $size[[size]] = array.size();
1688 auto $begin[[begin]] = array.begin();
1689 auto $end[[end]] = array.end();
1692 assertHintsWithHeader(
1694 ExpectedHint{": int *",
"no_modifier"},
1695 ExpectedHint{
": int **",
"ptr_modifier"},
1696 ExpectedHint{
": int *&",
"ref_modifier"},
1697 ExpectedHint{
": int *const &",
"at"}, ExpectedHint{
": int **",
"data"},
1698 ExpectedHint{
": allocator<int *>",
"allocator"},
1699 ExpectedHint{
": size_type",
"size"}, ExpectedHint{
": iterator",
"begin"},
1700 ExpectedHint{
": non_template_iterator",
"end"});
1702 llvm::StringRef VectorInt = R
"cpp(
1704 auto $no_modifier[[by_value]] = array[3];
1705 auto* $ptr_modifier[[ptr]] = &array[3];
1706 auto& $ref_modifier[[ref]] = array[3];
1707 auto& $at[[immutable]] = array.at(3);
1709 auto $data[[data]] = array.data();
1710 auto $allocator[[alloc]] = array.get_allocator();
1711 auto $size[[size]] = array.size();
1712 auto $begin[[begin]] = array.begin();
1713 auto $end[[end]] = array.end();
1716 assertHintsWithHeader(
1718 ExpectedHint{": int",
"no_modifier"},
1719 ExpectedHint{
": int *",
"ptr_modifier"},
1720 ExpectedHint{
": int &",
"ref_modifier"},
1721 ExpectedHint{
": const int &",
"at"}, ExpectedHint{
": int *",
"data"},
1722 ExpectedHint{
": allocator<int>",
"allocator"},
1723 ExpectedHint{
": size_type",
"size"}, ExpectedHint{
": iterator",
"begin"},
1724 ExpectedHint{
": non_template_iterator",
"end"});
1726 llvm::StringRef TypeAlias = R
"cpp(
1727 // If the type alias is not of substituted template parameter type,
1728 // do not show desugared type.
1729 using VeryLongLongTypeName = my_iterator;
1730 using Short = VeryLongLongTypeName;
1732 auto $short_name[[my_value]] = Short();
1734 // Same applies with templates.
1735 template <typename T, typename A>
1736 using basic_static_vector = vector<T, A>;
1737 template <typename T>
1738 using static_vector = basic_static_vector<T, allocator<T>>;
1740 auto $vector_name[[vec]] = static_vector<int>();
1744 DefaultOptsForTests,
1745 ExpectedHint{": Short",
"short_name"},
1746 ExpectedHint{
": static_vector<int>",
"vector_name"});
1749TEST(DesignatorHints, Basic) {
1750 assertDesignatorHints(R
"cpp(
1751 struct S { int x, y, z; };
1752 S s {$x[[1]], $y[[2+2]]};
1754 int x[] = {$0[[0]], $1[[1]]};
1756 ExpectedHint{".x=",
"x"}, ExpectedHint{
".y=",
"y"},
1757 ExpectedHint{
"[0]=",
"0"}, ExpectedHint{
"[1]=",
"1"});
1760TEST(DesignatorHints, Nested) {
1761 assertDesignatorHints(R
"cpp(
1762 struct Inner { int x, y; };
1763 struct Outer { Inner a, b; };
1764 Outer o{ $a[[{ $x[[1]], $y[[2]] }]], $bx[[3]] };
1766 ExpectedHint{".a=",
"a"}, ExpectedHint{
".x=",
"x"},
1767 ExpectedHint{
".y=",
"y"}, ExpectedHint{
".b.x=",
"bx"});
1770TEST(DesignatorHints, AnonymousRecord) {
1771 assertDesignatorHints(R
"cpp(
1783 ExpectedHint{".x.y=",
"xy"});
1786TEST(DesignatorHints, Suppression) {
1787 assertDesignatorHints(R
"cpp(
1788 struct Point { int a, b, c, d, e, f, g, h; };
1789 Point p{/*a=*/1, .c=2, /* .d = */3, $e[[4]]};
1791 ExpectedHint{".e=",
"e"});
1794TEST(DesignatorHints, StdArray) {
1797 assertDesignatorHints(R
"cpp(
1798 template <typename T, int N> struct Array { T __elements[N]; };
1799 Array<int, 2> x = {$0[[0]], $1[[1]]};
1801 ExpectedHint{"[0]=",
"0"}, ExpectedHint{
"[1]=",
"1"});
1804TEST(DesignatorHints, OnlyAggregateInit) {
1805 assertDesignatorHints(R
"cpp(
1806 struct Copyable { int x; } c;
1809 struct Constructible { Constructible(int x); };
1810 Constructible x{42};
1814TEST(DesignatorHints, NoCrash) {
1815 assertDesignatorHints(R
"cpp(
1818 struct Foo {int a; int b;};
1820 Foo f{A(), $b[[1]]};
1823 ExpectedHint{".b=",
"b"});
1826TEST(DesignatorHints, ParenInit) {
1827 assertDesignatorHints(R
"cpp(
1833 S s ($x[[1]], $y[[2+2]], $z[[4]]);
1835 ExpectedHint{".x=",
"x"}, ExpectedHint{
".y=",
"y"},
1836 ExpectedHint{
".z=",
"z"});
1839TEST(DesignatorHints, ParenInitDerived) {
1840 assertDesignatorHints(R
"cpp(
1850 S2 s2 ({$a[[0]], $b[[0]]}, $c[[0]], $d[[0]]);
1853 ExpectedHint{
".a=",
"a"}, ExpectedHint{
".b=",
"b"},
1854 ExpectedHint{
".c=",
"c"}, ExpectedHint{
".d=",
"d"});
1857TEST(DesignatorHints, ParenInitTemplate) {
1858 assertDesignatorHints(R
"cpp(
1859 template <typename T>
1866 struct S2 : S1<S2> {
1873 S2 sa ({$a1[[0]], $b1[[0]]}, $c[[0]], $d[[0]], $mem[[S1<int>($a2[[1]], $b2[[2]], $ptr[[nullptr]])]]);
1876 ExpectedHint{".a=",
"a1"}, ExpectedHint{
".b=",
"b1"},
1877 ExpectedHint{
".c=",
"c"}, ExpectedHint{
".d=",
"d"},
1878 ExpectedHint{
".mem=",
"mem"}, ExpectedHint{
".a=",
"a2"},
1879 ExpectedHint{
".b=",
"b2"},
1880 ExpectedHint{
".ptr=",
"ptr"});
1883TEST(InlayHints, RestrictRange) {
1892 ElementsAre(labelIs(
": int"), labelIs(
": char")));
1895TEST(ParameterHints, PseudoObjectExpr) {
1898 __declspec(property(get=GetX, put=PutX)) int x[];
1899 int GetX(int y, int z) { return 42 + y; }
1902 // This is a PseudoObjectExpression whose syntactic form is a binary
1904 void Work(int y) { x = y; } // Not `x = y: y`.
1907 int printf(const char *Format, ...);
1911 __builtin_dump_struct(&s, printf); // Not `Format: __builtin_dump_struct()`
1912 printf($Param[["Hello, %d"]], 42); // Normal calls are not affected.
1913 // This builds a PseudoObjectExpr, but here it's useful for showing the
1914 // arguments from the semantic form.
1915 return s.x[ $one[[1]] ][ $two[[2]] ]; // `x[y: 1][z: 2]`
1919 TU.ExtraArgs.push_back(
"-fms-extensions");
1920 auto AST = TU.build();
1922 ElementsAre(HintMatcher(ExpectedHint{
"Format: ",
"Param"}, Code),
1923 HintMatcher(ExpectedHint{
"y: ",
"one"}, Code),
1924 HintMatcher(ExpectedHint{
"z: ",
"two"}, Code)));
1927TEST(ParameterHints, ArgPacksAndConstructors) {
1928 assertParameterHints(
1930 struct Foo{ Foo(); Foo(int x); };
1931 void foo(Foo a, int b);
1932 template <typename... Args>
1933 void bar(Args... args) {
1936 template <typename... Args>
1937 void baz(Args... args) { foo($param1[[Foo{args...}]], $param2[[1]]); }
1939 template <typename... Args>
1940 void bax(Args... args) { foo($param3[[{args...}]], args...); }
1943 bar($param4[[Foo{}]], $param5[[42]]);
1944 bar($param6[[42]], $param7[[42]]);
1949 ExpectedHint{"a: ",
"param1"}, ExpectedHint{
"b: ",
"param2"},
1950 ExpectedHint{
"a: ",
"param3"}, ExpectedHint{
"a: ",
"param4"},
1951 ExpectedHint{
"b: ",
"param5"}, ExpectedHint{
"a: ",
"param6"},
1952 ExpectedHint{
"b: ",
"param7"}, ExpectedHint{
"x: ",
"param8"},
1953 ExpectedHint{
"b: ",
"param9"});
1956TEST(ParameterHints, DoesntExpandAllArgs) {
1957 assertParameterHints(
1959 void foo(int x, int y);
1960 int id(int a, int b, int c);
1961 template <typename... Args>
1962 void bar(Args... args) {
1963 foo(id($param1[[args]], $param2[[1]], $param3[[args]])...);
1966 bar(1, 2); // FIXME: We could have `bar(a: 1, a: 2)` here.
1969 ExpectedHint{"a: ",
"param1"}, ExpectedHint{
"b: ",
"param2"},
1970 ExpectedHint{
"c: ",
"param3"});
1973TEST(BlockEndHints, Functions) {
1974 assertBlockEndHints(R
"cpp(
1981 // No hint for lambda for now
1988 // No hint because this isn't a definition
1992 bool operator==(S, S) {
1996 ExpectedHint{" // foo",
"foo"},
1997 ExpectedHint{
" // bar",
"bar"},
1998 ExpectedHint{
" // operator==",
"opEqual"});
2001TEST(BlockEndHints, Methods) {
2002 assertBlockEndHints(R
"cpp(
2004 // No hint because there's no function body
2013 // No hint because this isn't a definition
2016 template <typename T>
2020 // No hint because this isn't a definition
2021 template <typename T>
2024 Test operator+(int) const {
2028 operator bool() const {
2032 // No hint because there's no function body
2033 operator int() const = delete;
2036 void Test::method2() {
2039 template <typename T>
2040 void Test::method4() {
2043 ExpectedHint{" // ~Test",
"dtor"},
2044 ExpectedHint{
" // method1",
"method1"},
2045 ExpectedHint{
" // method3",
"method3"},
2046 ExpectedHint{
" // operator+",
"opIdentity"},
2047 ExpectedHint{
" // operator bool",
"opBool"},
2048 ExpectedHint{
" // Test::method2",
"method2"},
2049 ExpectedHint{
" // Test::method4",
"method4"});
2052TEST(BlockEndHints, Namespaces) {
2053 assertBlockEndHints(
2063 ExpectedHint{" // namespace",
"anon"},
2064 ExpectedHint{
" // namespace ns",
"ns"});
2067TEST(BlockEndHints, Types) {
2068 assertBlockEndHints(
2085 ExpectedHint{" // struct S",
"S"}, ExpectedHint{
" // class C",
"C"},
2086 ExpectedHint{
" // union U",
"U"}, ExpectedHint{
" // enum E1",
"E1"},
2087 ExpectedHint{
" // enum class E2",
"E2"});
2090TEST(BlockEndHints, If) {
2091 assertBlockEndHints(
2093 void foo(bool cond) {
2115 if (auto X = cond) {
2118 if (int i = 0; i > 10) {
2121 if (ptr != nullptr) {
2125 ExpectedHint{" // if cond",
"simple"},
2126 ExpectedHint{
" // if cond",
"ifelse"}, ExpectedHint{
" // if",
"elseif"},
2127 ExpectedHint{
" // if !cond",
"inner"},
2128 ExpectedHint{
" // if cond",
"outer"}, ExpectedHint{
" // if X",
"init"},
2129 ExpectedHint{
" // if i > 10",
"init_cond"},
2130 ExpectedHint{
" // if ptr != nullptr",
"null_check"});
2133TEST(BlockEndHints, Loops) {
2134 assertBlockEndHints(
2149 for (int I = 0; I < 10; ++I) {
2157 ExpectedHint{" // while true",
"while"},
2158 ExpectedHint{
" // for true",
"forcond"},
2159 ExpectedHint{
" // for I",
"forvar"},
2160 ExpectedHint{
" // for V",
"foreach"});
2163TEST(BlockEndHints, Switch) {
2164 assertBlockEndHints(
2172 ExpectedHint{" // switch I",
"switch"});
2175TEST(BlockEndHints, PrintLiterals) {
2176 assertBlockEndHints(
2182 while ("foo but this time it is very long") {
2195 ExpectedHint{" // while \"foo\"",
"string"},
2196 ExpectedHint{
" // while \"foo but...\"",
"string_long"},
2197 ExpectedHint{
" // while true",
"boolean"},
2198 ExpectedHint{
" // while 1",
"integer"},
2199 ExpectedHint{
" // while 1.5",
"float"});
2202TEST(BlockEndHints, PrintRefs) {
2203 assertBlockEndHints(
2208 int func2(int, int);
2211 int method1() const;
2212 int method2(int, int) const;
2220 while (ns::func1()) {
2223 while (ns::func2(int_a, int_a)) {
2226 while (ns::S{}.Field) {
2229 while (ns::S{}.method1()) {
2232 while (ns::S{}.method2(int_a, int_a)) {
2236 ExpectedHint{" // while Var",
"var"},
2237 ExpectedHint{
" // while func1()",
"func1"},
2238 ExpectedHint{
" // while func2(...)",
"func2"},
2239 ExpectedHint{
" // while Field",
"field"},
2240 ExpectedHint{
" // while method1()",
"method1"},
2241 ExpectedHint{
" // while method2(...)",
"method2"});
2244TEST(BlockEndHints, PrintConversions) {
2245 assertBlockEndHints(
2250 explicit operator bool();
2254 $convert_primitive[[}]]
2260 $construct_class[[}]]
2263 ExpectedHint{" // while float",
"convert_primitive"},
2264 ExpectedHint{
" // while S",
"convert_class"},
2265 ExpectedHint{
" // while S",
"construct_class"});
2268TEST(BlockEndHints, PrintOperators) {
2269 std::string AnnotatedCode = R
"cpp(
2270 void foo(Integer I) {
2289 while((I + I) < (I + I)){
2290 $binary_complex[[}]]
2295 auto AssertExpectedHints = [&](llvm::StringRef Code) {
2296 assertBlockEndHints(Code, ExpectedHint{
" // while ++I",
"preinc"},
2297 ExpectedHint{
" // while I++",
"postinc"},
2298 ExpectedHint{
" // while",
"unary_complex"},
2299 ExpectedHint{
" // while I < 0",
"compare"},
2300 ExpectedHint{
" // while ... < I",
"lhs_complex"},
2301 ExpectedHint{
" // while I < ...",
"rhs_complex"},
2302 ExpectedHint{
" // while",
"binary_complex"});
2306 AssertExpectedHints(
"using Integer = int;" + AnnotatedCode);
2308 AssertExpectedHints(R
"cpp(
2310 explicit operator bool();
2311 Integer operator++();
2312 Integer operator++(int);
2313 Integer operator+(Integer);
2314 Integer operator+();
2315 bool operator<(Integer);
2316 bool operator<(int);
2318 )cpp" + AnnotatedCode);
2321TEST(BlockEndHints, TrailingSemicolon) {
2322 assertBlockEndHints(R"cpp(
2323 // The hint is placed after the trailing ';'
2327 // The hint is always placed in the same line with the closing '}'.
2328 // So in this case where ';' is missing, it is attached to '}'.
2334 // No hint because only one trailing ';' is allowed
2338 // No hint because trailing ';' is only allowed for class/struct/union/enum
2342 // Rare case, but yes we'll have a hint here.
2349 ExpectedHint{" // struct S1",
"S1"},
2350 ExpectedHint{
" // struct S2",
"S2"},
2351 ExpectedHint{
" // struct",
"anon"});
2354TEST(BlockEndHints, TrailingText) {
2355 assertBlockEndHints(R
"cpp(
2359 // No hint for S2 because of the trailing comment
2361 }; /* Put anything here */
2364 // No hint for S4 because of the trailing source code
2368 // No hint for ns because of the trailing comment
2372 ExpectedHint{" // struct S1",
"S1"},
2373 ExpectedHint{
" // struct S3",
"S3"});
2377 assertBlockEndHints(R
"cpp(
2378 #define DECL_STRUCT(NAME) struct NAME {
2384 // No hint because we require a '}'
2388 ExpectedHint{" // struct S1",
"S1"});
2391TEST(BlockEndHints, PointerToMemberFunction) {
2393 assertBlockEndHints(R
"cpp(
2395 using Predicate = bool(A::*)();
2396 void foo(A* a, Predicate p) {
2401 ExpectedHint{" // if ()",
"ptrmem"});
2404TEST(BlockEndHints, MinLineLimit) {
2409 assertBlockEndHintsWithOpts(
2414 int func2(int, int);
2417 int method1() const;
2418 int method2(int, int) const;
2426 while (ns::func1()) {
2429 while (ns::func2(int_a, int_a)) {
2432 while (ns::S{}.Field) {
2435 while (ns::S{}.method1()) {
2438 while (ns::S{}.method2(int_a, int_a)) {
2442 Opts, ExpectedHint{" // namespace ns",
"namespace"},
2443 ExpectedHint{
" // foo",
"foo"});
Same as llvm::Annotations, but adjusts functions to LSP-specific types for positions and ranges.
void addSource(llvm::StringRef Filename, llvm::StringRef Code)
WithContextValue extends Context::current() with a single value.
FIXME: Skip testing on windows temporarily due to the different escaping code mode.
MATCHER_P2(hasFlag, Flag, Path, "")
llvm::raw_ostream & operator<<(llvm::raw_ostream &OS, const CodeCompletion &C)
TEST(BackgroundQueueTest, Priority)
InlayHintKind
Inlay hint kinds.
@ BlockEnd
A hint after function, type or namespace definition, indicating the defined symbol name of the defini...
@ DefaultArgument
An inlay hint that is for a default argument.
@ Parameter
An inlay hint that is for a parameter.
@ Type
An inlay hint that for a type annotation.
@ Designator
A hint before an element of an aggregate braced initializer list, indicating what it is initializing.
std::vector< InlayHint > inlayHints(ParsedAST &AST, std::optional< Range > RestrictRange, InlayHintOptions HintOptions)
Compute and return inlay hints for a file.
===– Representation.cpp - ClangDoc Representation --------—*- C++ -*-===//
Settings that express user/project preferences and control clangd behavior.
static clangd::Key< Config > Key
Context key which can be used to set the current Config.
struct clang::clangd::Config::@041344304366110202143331236314370324353035136032 InlayHints
std::string joinLabels() const
Join the label[].value together.
Range range
The range of source code to which the hint applies.
static TestTU withCode(llvm::StringRef Code)