Adds functionality for getting semantic highlights
Details
Diff Detail
- Repository
- rG LLVM Github Monorepo
- Build Status
Buildable 33933 Build 33932: arc lint + arc unit
Event Timeline
clang-tools-extra/clangd/SemanticHighlight.cpp | ||
---|---|---|
1 ↗ | (On Diff #205608) | Header is missing. |
32 ↗ | (On Diff #205608) | Please don't use auto if type is not spelled explicitly. |
34 ↗ | (On Diff #205608) | Please don't use auto if type is not spelled explicitly. |
99 ↗ | (On Diff #205608) | May be std::array is better way to do this? Same below. |
135 ↗ | (On Diff #205608) | Please don't use auto if type is not spelled explicitly. |
clang-tools-extra/clangd/SemanticHighlight.h | ||
2 ↗ | (On Diff #205608) | Please merge with previous line. If it doesn't fit, remove some - and =. |
14 ↗ | (On Diff #205608) | Please separate with empty line. |
35 ↗ | (On Diff #205608) | Please use = default; |
clang-tools-extra/clangd/unittests/SemanticHighlightTests.cpp | ||
2 ↗ | (On Diff #205608) | Please merge with previous line. If it doesn't fit, remove some - and =. |
Thanks, I think we can simplify the interface further, see my comments inline.
The boundary of this patch seems unclear, currently it contains C++ APIs, and some implementations for semantic highlighting LSP. I think we should merely focus on the C++ APIs in this patch.
clang-tools-extra/clangd/SemanticHighlight.cpp | ||
---|---|---|
32 ↗ | (On Diff #205626) | Looks like the code doesn't handle the case where D is expanded from a macro (Loc is a macro location). |
129 ↗ | (On Diff #205626) | This is Textmate-specific, I think we should lift it to a new File (TextMate.cpp). We can do it in a separate patch. I'd use a map to explicitly express the relationship between SemanticScope and TextMate scope, the current implementation is not obvious. {SemanticScope::Function, "entity.name.function"}, {SemanticScope::Variable, "variable.other"}, |
133 ↗ | (On Diff #205626) | We should only collect the highlights from the main AST, I think we should set traversal scope only to localTopDecls (use ParsedAST as parameter would help here) ASTCtx.setTraversalScope(MainAST.getLocalTopLevelDecls()); |
clang-tools-extra/clangd/SemanticHighlight.h | ||
14 ↗ | (On Diff #205626) | nit: the name of header guard doesn't reflect the file name. |
21 ↗ | (On Diff #205626) | please clean-up your #includes:
|
26 ↗ | (On Diff #205626) | The comment seems not very related to this structure. We'd use this enum class to find a corresponding TextMate scope in the future, but this is implementation detail, the comment here should mention the high-level things about this structure. |
28 ↗ | (On Diff #205626) | TextMate is using the term scope for different tokens, but the scope has different meaning in C++ world (usually the namespace "ns::" of a symbol). I'd avoid using this word in the C++ interface. Just SemanticHighlightingKind? |
29 ↗ | (On Diff #205626) | not exactly sure whether we should distinguish these cases at the moment (function declaration vs function calls, variable declaration vs variable references). I think we could just use Variable, Function at the beginning, and add more kinds afterwards. |
39 ↗ | (On Diff #205626) | instead of using start position + length, I'd use Range R; |
61 ↗ | (On Diff #205626) | I'd drop the AST word, how about naming it "getSemanticHighlights"? I think we can just return "vector<HighlightingToken>", and we could we could group them by line when returning a LSP result. The function parameter should be ParsedAST. |
Removed LSP specific stuff. Changed a bunch of types and names. Added fixme for macro expansion.
clang-tools-extra/clangd/SemanticHighlight.cpp | ||
---|---|---|
129 ↗ | (On Diff #205626) | Will do this in CL when adding C++ ClangdServer api calls |
clang-tools-extra/clangd/SemanticHighlight.cpp | ||
---|---|---|
39 ↗ | (On Diff #206178) | We can check the name of NamedDecl is empty rather than detecting the length. Could you add a test for this? |
75 ↗ | (On Diff #206178) | let's move the above lines into SemanticSymbolASTCollector, we can define a new method "collectTokens()". |
clang-tools-extra/clangd/SemanticHighlight.h | ||
9 ↗ | (On Diff #206178) | this comment doesn't provide much information IMO, let's remove it. |
clang-tools-extra/clangd/SemanticHighlight.cpp | ||
---|---|---|
11 ↗ | (On Diff #206178) | The comment is out of date now. |
clang-tools-extra/clangd/SemanticHighlight.h | ||
29 ↗ | (On Diff #206178) | We can get rid of the default constructor and the constructor below. |
38 ↗ | (On Diff #206178) | The comment is out of date. |
clang-tools-extra/clangd/unittests/SemanticHighlightTests.cpp | ||
31 ↗ | (On Diff #206178) | Thinking more about the test, I think we could improve the test to make it more scalable and less verbose code. How about associating the range name (e.g. $vlan) in the annotation with the name in SemanticHighlightKind? For example, in this test case, we could use annotation like void $Function[foo](int $Variable[[a]]) { auto $Variable[[A]] = 222; } We have a common function like checkHighlights which verifies that all Functions and Variable ranges. void checkHighlights(... annotation) { auto ActualHighlights = getHighlights(annotation.code()); EXPECT_THAT(annotations.ranges('Function'), unorderedElemenetsAre(filter(ActualHighlights, SemanticHighlightKind::Function)); EXPECT_THAT(annotations.ranges('Variable'), unorderedElemenetsAre(filter(ActualHighlights, SemanticHighlightKind::Variable)) } |
clang-tools-extra/clangd/SemanticHighlight.cpp | ||
---|---|---|
75 ↗ | (On Diff #206178) | Should I expose the entire class or keep the getSemanticHighlights function? (I'm just thinking that RecursiveASTVisitor contains a lot of public functions which makes it not super obvious which function to call to get semantic highlight things when looking at autocomplete) |
clang-tools-extra/clangd/SemanticHighlight.cpp | ||
---|---|---|
75 ↗ | (On Diff #206178) | Actually I can just do private inheritance and declare RecursiveASTVisitor<SemanticSymbolASTCollector> to be a friend. |
clang-tools-extra/clangd/SemanticHighlight.cpp | ||
---|---|---|
43 ↗ | (On Diff #206376) | we can reuse the getTokenRange() function (in SourceCode.h), you need to sync your branch to master. |
46 ↗ | (On Diff #206376) | I'd move this line at the beginning of the method. |
clang-tools-extra/clangd/unittests/SemanticHighlightTests.cpp | ||
57 ↗ | (On Diff #206376) | checkHighlights(R"cpp(void $Function[[Foo]]))cpp"); checkHighlights(...) { ... ExpectedTokens = annotation.ranges(toString(SemanticHighlightKind::Variable)); ExpectedTokens = annotation.ranges(toString(SemanticHighlightKind::Function)); EXPECT_THAT(ExpectedTokens, UnorderedElementsAreArray(ActualTokens)); } |
Changed tests
clang-tools-extra/clangd/unittests/SemanticHighlightTests.cpp | ||
---|---|---|
57 ↗ | (On Diff #206376) | Had to add some extra logic to merge the different token types |
the test looks better now, another round of reviews, most are nits.
clang-tools-extra/clangd/SemanticHighlight.cpp | ||
---|---|---|
19 ↗ | (On Diff #206393) | nit: public. |
20 ↗ | (On Diff #206393) | do we need friend declaration now? |
28 ↗ | (On Diff #206393) | I'd move this line to collectTokens as they are related. As discussed, setTraversalScope doesn't guarantee that we wouldnot visit Decl* outside of the main file (some decls are still reachable), so we may get tokens outside of the main file. If you don't address it in this patch, that's fine, leave a FIXME here. |
32 ↗ | (On Diff #206393) | add assume(Tokens.empty())?, if we call this method twice, we will accumulate the Tokens. |
45 ↗ | (On Diff #206393) | addSymbol => addToken; |
47 ↗ | (On Diff #206393) | nit: FIXME: skip tokens inside macros for now. |
51 ↗ | (On Diff #206393) | use getDeclName().isEmpty(). |
56 ↗ | (On Diff #206393) | How about pulling out a function llvm::Optional<SemanticToken> makeSemanticToken(..)? |
59 ↗ | (On Diff #206393) | this would crash clangd if we meet this case, I think we could just emit a log. |
clang-tools-extra/clangd/SemanticHighlight.h | ||
14 ↗ | (On Diff #206393) | nit: remove the unused include. |
19 ↗ | (On Diff #206393) | Maybe just enum SemanticHighlightKind { Variable, Function }; |
39 ↗ | (On Diff #206393) | add a new line. |
clang-tools-extra/clangd/unittests/SemanticHighlightTests.cpp | ||
11 ↗ | (On Diff #206393) | not used #include |
15 ↗ | (On Diff #206393) | is this #include used? |
24 ↗ | (On Diff #206393) | we are passing a copy here, use llvm::ArrayRef. |
27 ↗ | (On Diff #206393) | nit: ++I |
35 ↗ | (On Diff #206393) | nit: llvm::StringRef |
39 ↗ | (On Diff #206393) | static const |
43 ↗ | (On Diff #206393) | nit: const auto& |
53 ↗ | (On Diff #206393) | nit: update the test name. |
65 ↗ | (On Diff #206393) | I think we can group these two testcases like const char* TestCases[] = { {}, {}, }; for (...) { } |
66 ↗ | (On Diff #206393) | maybe use another obvious case: struct { } name; |
clang-tools-extra/clangd/SemanticHighlight.cpp | ||
---|---|---|
28 ↗ | (On Diff #206393) | Will fix in a separate patch for topLevelDecls. Don't really know what FIXME to add here. Added one to getLocalTopLevelDecls. Don't really have a good understanding of the reason as to what happens yet so the FIXME is very general (as you can probably tell from it)... |
56 ↗ | (On Diff #206393) | Don't understand what you mean. |
clang-tools-extra/clangd/unittests/SemanticHighlightTests.cpp | ||
24 ↗ | (On Diff #206393) | I changed to pass a const vector & instead. Is that alright as well? |
much clearer now, a few more nits.
clang-tools-extra/clangd/ClangdUnit.cpp | ||
---|---|---|
69 ↗ | (On Diff #206593) | not need to add this in this patch as you are also working on a fix in a separate patch. |
clang-tools-extra/clangd/SemanticHighlight.cpp | ||
56 ↗ | (On Diff #206393) | I meant we could move lines 56 ~ 66 to a new function, the current implementation is fine (as addToken is small). |
51 ↗ | (On Diff #206593) | nit: remove the {} if the body contains only one statement, the same to other places. |
57 ↗ | (On Diff #206593) | nit: just "if (!R)" would work. |
64 ↗ | (On Diff #206593) | maybe just Tokens.emplace_back(Kind, *R). |
75 ↗ | (On Diff #206593) | we probably don't need the != operator in this patch. |
clang-tools-extra/clangd/SemanticHighlight.h | ||
13 ↗ | (On Diff #206593) | the RecursiveASTVisitor.h should be in the .cpp file. |
32 ↗ | (On Diff #206593) | needs high-level comments. |
clang-tools-extra/clangd/unittests/SemanticHighlightTests.cpp | ||
47 ↗ | (On Diff #206593) | the code seems not clang-format, could you run clang-format on your patch? |
24 ↗ | (On Diff #206393) | yeah, that works as well, llvm prefers llvm::ArrayRef, but up to you. |
- [clangd] Added functionality for getting semantic highlights for variable and function declarations
clang-tools-extra/clangd/unittests/SemanticHighlightTests.cpp | ||
---|---|---|
47 ↗ | (On Diff #206593) | It has been clang-formated. The spaces are part of the test (need to check that the range does not include spaces even if there are multiple ones) |
thanks mostly good. Thinking more about the name, we should align with the LSP proposal, see my comments inline.
clang-tools-extra/clangd/SemanticHighlight.cpp | ||
---|---|---|
20 ↗ | (On Diff #206618) | HighlightingTokenCollector. |
clang-tools-extra/clangd/SemanticHighlight.h | ||
17 ↗ | (On Diff #206618) | LSP proposal is using Highlighting rather than Highlight, let's align with the LSP proposal, using Highlighting in our names (comments, function, class, and files). The name seems too verbose, let's drop the Semantic, just use HighlightingKind. |
23 ↗ | (On Diff #206618) | here use HighlightingToken. |
32 ↗ | (On Diff #206618) | getSemanticHighlightings. |
clang-tools-extra/clangd/unittests/SemanticHighlightTests.cpp | ||
46 ↗ | (On Diff #206618) | nit: SemanticTokenCollector => SemanticHighlighting |
clang-tools-extra/clangd/SemanticHighlight.h | ||
---|---|---|
17 ↗ | (On Diff #206618) | There is also a DocumentHighlightingKind enum though. Is there not a possibility of confusing people with HighlightingKind and DocumentHighlightingKind? |
clang-tools-extra/clangd/SemanticHighlight.h | ||
---|---|---|
17 ↗ | (On Diff #206618) | We have a DocumentHighlightKind (without ing), but that is in the LSP layer. Yeah, it would not be super clear, but if we are in the related context, HighlightingKind is clear -- ClangdServer will use getSemanticHighlights to get all highlight tokens, so I'd prefer shorter names. |
the patch was landed in rL364421, not sure why it was not associated with this review.
nit: SemanticHighlightingTest.cpp => SemanticHighlightingTests.cpp