[clang-tools-extra] 0aaa2ac - [clangd] Compute the inactive code range for semantic highlighting.
Haojian Wu via cfe-commits
cfe-commits at lists.llvm.org
Wed Aug 26 01:51:24 PDT 2020
Author: Haojian Wu
Date: 2020-08-26T10:50:31+02:00
New Revision: 0aaa2acc4ca0389a79a46bca3dc5b9365c641d77
URL: https://github.com/llvm/llvm-project/commit/0aaa2acc4ca0389a79a46bca3dc5b9365c641d77
DIFF: https://github.com/llvm/llvm-project/commit/0aaa2acc4ca0389a79a46bca3dc5b9365c641d77.diff
LOG: [clangd] Compute the inactive code range for semantic highlighting.
Differential Revision: https://reviews.llvm.org/D85635
Added:
Modified:
clang-tools-extra/clangd/SemanticHighlighting.cpp
clang-tools-extra/clangd/unittests/SemanticHighlightingTests.cpp
Removed:
################################################################################
diff --git a/clang-tools-extra/clangd/SemanticHighlighting.cpp b/clang-tools-extra/clangd/SemanticHighlighting.cpp
index fb1ef1e326b41..4e66a9bb4e857 100644
--- a/clang-tools-extra/clangd/SemanticHighlighting.cpp
+++ b/clang-tools-extra/clangd/SemanticHighlighting.cpp
@@ -221,23 +221,51 @@ class HighlightingsBuilder {
// the end of the Tokens).
TokRef = TokRef.drop_front(Conflicting.size());
}
- // Add tokens indicating lines skipped by the preprocessor.
- for (const Range &R : AST.getMacros().SkippedRanges) {
+ const auto &SM = AST.getSourceManager();
+ StringRef MainCode = SM.getBuffer(SM.getMainFileID())->getBuffer();
+
+ // Merge token stream with "inactive line" markers.
+ std::vector<HighlightingToken> WithInactiveLines;
+ auto SortedSkippedRanges = AST.getMacros().SkippedRanges;
+ llvm::sort(SortedSkippedRanges);
+ auto It = NonConflicting.begin();
+ for (const Range &R : SortedSkippedRanges) {
// Create one token for each line in the skipped range, so it works
// with line-based
diff ing.
assert(R.start.line <= R.end.line);
for (int Line = R.start.line; Line <= R.end.line; ++Line) {
- // Don't bother computing the offset for the end of the line, just use
- // zero. The client will treat this highlighting kind specially, and
- // highlight the entire line visually (i.e. not just to where the text
- // on the line ends, but to the end of the screen).
- NonConflicting.push_back({HighlightingKind::InactiveCode,
- {Position{Line, 0}, Position{Line, 0}}});
+ // Copy tokens before the inactive line
+ for (; It != NonConflicting.end() && It->R.start.line < Line; ++It)
+ WithInactiveLines.push_back(std::move(*It));
+ // Add a token for the inactive line itself.
+ auto StartOfLine = positionToOffset(MainCode, Position{Line, 0});
+ if (StartOfLine) {
+ StringRef LineText =
+ MainCode.drop_front(*StartOfLine).take_until([](char C) {
+ return C == '\n';
+ });
+ WithInactiveLines.push_back(
+ {HighlightingKind::InactiveCode,
+ {Position{Line, 0},
+ Position{Line, static_cast<int>(lspLength(LineText))}}});
+ } else {
+ elog("Failed to convert position to offset: {0}",
+ StartOfLine.takeError());
+ }
+
+ // Skip any other tokens on the inactive line. e.g.
+ // `#ifndef Foo` is considered as part of an inactive region when Foo is
+ // defined, and there is a Foo macro token.
+ // FIXME: we should reduce the scope of the inactive region to not
+ // include the directive itself.
+ while (It != NonConflicting.end() && It->R.start.line == Line)
+ ++It;
}
}
- // Re-sort the tokens because that's what the
diff ing expects.
- llvm::sort(NonConflicting);
- return NonConflicting;
+ // Copy tokens after the last inactive line
+ for (; It != NonConflicting.end(); ++It)
+ WithInactiveLines.push_back(std::move(*It));
+ return WithInactiveLines;
}
private:
@@ -493,9 +521,6 @@ toSemanticTokens(llvm::ArrayRef<HighlightingToken> Tokens) {
std::vector<SemanticToken> Result;
const HighlightingToken *Last = nullptr;
for (const HighlightingToken &Tok : Tokens) {
- // FIXME: support inactive code - we need to provide the actual bounds.
- if (Tok.Kind == HighlightingKind::InactiveCode)
- continue;
Result.emplace_back();
SemanticToken &Out = Result.back();
// deltaStart/deltaLine are relative if possible.
diff --git a/clang-tools-extra/clangd/unittests/SemanticHighlightingTests.cpp b/clang-tools-extra/clangd/unittests/SemanticHighlightingTests.cpp
index 06743080166b4..232be6a783803 100644
--- a/clang-tools-extra/clangd/unittests/SemanticHighlightingTests.cpp
+++ b/clang-tools-extra/clangd/unittests/SemanticHighlightingTests.cpp
@@ -503,11 +503,11 @@ TEST(SemanticHighlighting, GetsCorrectTokens) {
#define $Macro[[test]]
#undef $Macro[[test]]
-$InactiveCode[[]] #ifdef $Macro[[test]]
-$InactiveCode[[]] #endif
+$InactiveCode[[#ifdef test]]
+$InactiveCode[[#endif]]
-$InactiveCode[[]] #if defined($Macro[[test]])
-$InactiveCode[[]] #endif
+$InactiveCode[[#if defined(test)]]
+$InactiveCode[[#endif]]
)cpp",
R"cpp(
struct $Class[[S]] {
@@ -614,8 +614,8 @@ TEST(SemanticHighlighting, GetsCorrectTokens) {
R"cpp(
// Code in the preamble.
// Inactive lines get an empty InactiveCode token at the beginning.
-$InactiveCode[[]] #ifdef $Macro[[test]]
-$InactiveCode[[]] #endif
+$InactiveCode[[#ifdef test]]
+$InactiveCode[[#endif]]
// A declaration to cause the preamble to end.
int $Variable[[EndPreamble]];
@@ -623,17 +623,17 @@ TEST(SemanticHighlighting, GetsCorrectTokens) {
// Code after the preamble.
// Code inside inactive blocks does not get regular highlightings
// because it's not part of the AST.
-$InactiveCode[[]] #ifdef $Macro[[test]]
-$InactiveCode[[]] int Inactive2;
-$InactiveCode[[]] #endif
+$InactiveCode[[#ifdef test]]
+$InactiveCode[[int Inactive2;]]
+$InactiveCode[[#endif]]
#ifndef $Macro[[test]]
int $Variable[[Active1]];
#endif
-$InactiveCode[[]] #ifdef $Macro[[test]]
-$InactiveCode[[]] int Inactive3;
-$InactiveCode[[]] #else
+$InactiveCode[[#ifdef test]]
+$InactiveCode[[int Inactive3;]]
+$InactiveCode[[#else]]
int $Variable[[Active2]];
#endif
)cpp",
More information about the cfe-commits
mailing list