diff --git a/clang-tools-extra/clangd/SemanticHighlighting.h b/clang-tools-extra/clangd/SemanticHighlighting.h --- a/clang-tools-extra/clangd/SemanticHighlighting.h +++ b/clang-tools-extra/clangd/SemanticHighlighting.h @@ -38,6 +38,7 @@ Namespace, TemplateParameter, Primitive, + MacroExpansion, NumKinds, }; diff --git a/clang-tools-extra/clangd/SemanticHighlighting.cpp b/clang-tools-extra/clangd/SemanticHighlighting.cpp --- a/clang-tools-extra/clangd/SemanticHighlighting.cpp +++ b/clang-tools-extra/clangd/SemanticHighlighting.cpp @@ -24,16 +24,20 @@ class HighlightingTokenCollector : public RecursiveASTVisitor { std::vector Tokens; - ASTContext &Ctx; + ParsedAST * const SourceManager &SM; public: HighlightingTokenCollector(ParsedAST &AST) - : Ctx(AST.getASTContext()), SM(AST.getSourceManager()) {} + : AST(AST), SM(AST.getSourceManager()) {} std::vector collectTokens() { Tokens.clear(); - TraverseAST(Ctx); + TraverseAST(AST.getASTContext()); + // Add highlightings for Macro Expansions as they are not traversed by the + // visitor. + for (const SourceLocation &L : AST.getMainFileExpansions()) + addToken(L, HighlightingKind::MacroExpansion); // Initializer lists can give duplicates of tokens, therefore all tokens // must be deduplicated. llvm::sort(Tokens); @@ -275,7 +279,7 @@ if (!isInsideMainFile(Loc, SM)) return; - auto R = getTokenRange(SM, Ctx.getLangOpts(), Loc); + auto R = getTokenRange(SM, AST.getASTContext().getLangOpts(), Loc); if (!R) { // R should always have a value, if it doesn't something is very wrong. elog("Tried to add semantic token with an invalid range"); @@ -462,6 +466,8 @@ return "entity.name.type.template.cpp"; case HighlightingKind::Primitive: return "storage.type.primitive.cpp"; + case HighlightingKind::MacroExpansion: + return "entity.name.function.preprocessor.expansion.cpp"; case HighlightingKind::NumKinds: llvm_unreachable("must not pass NumKinds to the function"); } diff --git a/clang-tools-extra/clangd/test/semantic-highlighting.test b/clang-tools-extra/clangd/test/semantic-highlighting.test --- a/clang-tools-extra/clangd/test/semantic-highlighting.test +++ b/clang-tools-extra/clangd/test/semantic-highlighting.test @@ -45,6 +45,9 @@ # CHECK-NEXT: ], # CHECK-NEXT: [ # CHECK-NEXT: "storage.type.primitive.cpp" +# CHECK-NEXT: ], +# CHECK-NEXT: [ +# CHECK-NEXT: "entity.name.function.preprocessor.expansion.cpp" # CHECK-NEXT: ] # CHECK-NEXT: ] # CHECK-NEXT: }, diff --git a/clang-tools-extra/clangd/unittests/SemanticHighlightingTests.cpp b/clang-tools-extra/clangd/unittests/SemanticHighlightingTests.cpp --- a/clang-tools-extra/clangd/unittests/SemanticHighlightingTests.cpp +++ b/clang-tools-extra/clangd/unittests/SemanticHighlightingTests.cpp @@ -47,7 +47,8 @@ {HighlightingKind::Method, "Method"}, {HighlightingKind::StaticMethod, "StaticMethod"}, {HighlightingKind::TemplateParameter, "TemplateParameter"}, - {HighlightingKind::Primitive, "Primitive"}}; + {HighlightingKind::Primitive, "Primitive"}, + {HighlightingKind::MacroExpansion, "MacroExpansion"}}; std::vector ExpectedTokens; for (const auto &KindString : KindToString) { std::vector Toks = makeHighlightingTokens( @@ -391,9 +392,9 @@ R"cpp( #define DEF_MULTIPLE(X) namespace X { class X { int X; }; } #define DEF_CLASS(T) class T {}; - DEF_MULTIPLE(XYZ); - DEF_MULTIPLE(XYZW); - DEF_CLASS($Class[[A]]) + $MacroExpansion[[DEF_MULTIPLE]](XYZ); + $MacroExpansion[[DEF_MULTIPLE]](XYZW); + $MacroExpansion[[DEF_CLASS]]($Class[[A]]) #define MACRO_CONCAT(X, V, T) T foo##X = V #define DEF_VAR(X, V) int X = V #define DEF_VAR_T(T, X, V) T X = V @@ -404,26 +405,27 @@ #define SOME_NAME_SET variable2 = 123 #define INC_VAR(X) X += 2 $Primitive[[void]] $Function[[foo]]() { - DEF_VAR($LocalVariable[[X]], 123); - DEF_VAR_REV(908, $LocalVariable[[XY]]); - $Primitive[[int]] CPY( $LocalVariable[[XX]] ); - DEF_VAR_TYPE($Class[[A]], $LocalVariable[[AA]]); - $Primitive[[double]] SOME_NAME; - $Primitive[[int]] SOME_NAME_SET; + $MacroExpansion[[DEF_VAR]]($LocalVariable[[X]], 123); + $MacroExpansion[[DEF_VAR_REV]](908, $LocalVariable[[XY]]); + $Primitive[[int]] $MacroExpansion[[CPY]]( $LocalVariable[[XX]] ); + $MacroExpansion[[DEF_VAR_TYPE]]($Class[[A]], $LocalVariable[[AA]]); + $Primitive[[double]] $MacroExpansion[[SOME_NAME]]; + $Primitive[[int]] $MacroExpansion[[SOME_NAME_SET]]; $LocalVariable[[variable]] = 20.1; - MACRO_CONCAT(var, 2, $Primitive[[float]]); - DEF_VAR_T($Class[[A]], CPY(CPY($LocalVariable[[Nested]])), - CPY($Class[[A]]())); - INC_VAR($LocalVariable[[variable]]); + $MacroExpansion[[MACRO_CONCAT]](var, 2, $Primitive[[float]]); + $MacroExpansion[[DEF_VAR_T]]($Class[[A]], $MacroExpansion[[CPY]]( + $MacroExpansion[[CPY]]($LocalVariable[[Nested]])), + $MacroExpansion[[CPY]]($Class[[A]]())); + $MacroExpansion[[INC_VAR]]($LocalVariable[[variable]]); } - $Primitive[[void]] SOME_NAME(); - DEF_VAR($Variable[[XYZ]], 567); - DEF_VAR_REV(756, $Variable[[AB]]); + $Primitive[[void]] $MacroExpansion[[SOME_NAME]](); + $MacroExpansion[[DEF_VAR]]($Variable[[XYZ]], 567); + $MacroExpansion[[DEF_VAR_REV]](756, $Variable[[AB]]); #define CALL_FN(F) F(); #define DEF_FN(F) void F () - DEF_FN($Function[[g]]) { - CALL_FN($Function[[foo]]); + $MacroExpansion[[DEF_FN]]($Function[[g]]) { + $MacroExpansion[[CALL_FN]]($Function[[foo]]); } )cpp", R"cpp( @@ -433,8 +435,8 @@ $Primitive[[int]] $Variable[[y]]; $Primitive[[int]] $Function[[f]](); $Primitive[[void]] $Function[[foo]]() { - assert($Variable[[x]] != $Variable[[y]]); - assert($Variable[[x]] != $Function[[f]]()); + $MacroExpansion[[assert]]($Variable[[x]] != $Variable[[y]]); + $MacroExpansion[[assert]]($Variable[[x]] != $Function[[f]]()); } )cpp"}; for (const auto &TestCase : TestCases) { @@ -455,8 +457,8 @@ // A separate test for macros in headers. checkHighlightings(R"cpp( #include "imp.h" - DEFINE_Y - DXYZ_Y(A); + $MacroExpansion[[DEFINE_Y]] + $MacroExpansion[[DXYZ_Y]](A); )cpp", {{"imp.h", R"cpp( #define DXYZ(X) class X {};