Details
Diff Detail
- Repository
- rG LLVM Github Monorepo
Event Timeline
Thanks -- this patch is looking great so far!
clang-tools-extra/clangd/SemanticHighlighting.cpp | ||
---|---|---|
312 | We should consider the case where a dependent name is passed by non-const reference, for example: void increment_counter(int&); template <typename T> void bar() { increment_counter(T::static_counter); } This case does not work yet with the current patch (the dependent name is a DependentScopeDeclRefExpr rather than a DeclRefExpr), but we'll want to make it work in the future. With the conflict resolution logic in this patch, the Unknown token kind from highlightPassedByNonConstReference() will be chosen over the dependent token kind. As it happens, the dependent token kind for expressions is also Unknown so it doesn't matter, but perhaps we shouldn't be relying on this. Perhaps the following would make more sense:
| |
534 | nit: sense | |
549 | We can avoid the callback by constructing an ArrayRef of the arguments, see a similar case here: https://searchfox.org/llvm/source/clang-tools-extra/clangd/InlayHints.cpp#56 | |
561 | I think there are some edge cases where isConst() doesn't do what we want. For example, I think for a parameter of type const int*&, it will return true (and thus we will not highlight the corresponding argument), even thus this is actually a non-const reference. So, we may want to use a dedicated function that specifically handles reference-related logic only. (This probably also makes a good test case.) | |
566 | For a qualified name (e.g. A::B), I think this is going to return the beginning of the qualifier, whereas we only want to highlight the last name (otherwise there won't be a matching token from the first pass). So I think we want getLocation() instead. (Also makes a good test case.) |
Agree this is nice, well done! A few more notes for consideration...
clang-tools-extra/clangd/SemanticHighlighting.cpp | ||
---|---|---|
312 | The conflict-resolution idea is subtle (and IME hard to debug). I'm wary of overloading it by deliberately introducing "conflicts" that should actually be merged. Did you consider the idea of tracking extra modifiers separately and merging them in at the end? BTW: we're stretching the meaning of Unknown here. There are two subtly different concepts:
Call me weird, but I have "Unknown" highlighted in bright orange in my editor, because I want to know about the second case :-) | |
533 | CXXOperatorCallExpr seems to make sense to me for the most part:
There are some exceptions though:
I think these can be handled by choosing a minimum argument index to highlight based on the operator type. I think it makes sense to leave operators out of scope for this patch, but IMO should be a "FIXME" rather than a "let's never do this" :-) | |
547 | nit: this function name is a bit hard to parse, "highlight" is a transitive verb but the rest of the name isn't its object. Maybe highlightMutableReferenceArguments? | |
557 | I feel like you'd be better off using the FunctionProtoType and iterating over argument types, rather than the argument declarations on a particular declaration of the function. e.g. this code is legal in C: int x(); // i suspect this is the canonical decl int x(int); // but this one provides the type We don't have references in C of course!, but maybe similar issues lurking... | |
561 | Yeah this is the core of this modifier, worth being precise and explicit here. I think we want to match only reference types where the inner type is not top-level const. I think we should also conservatively forbid the inner type from being *dependent*. Consider the following function: template <typename X> void foo(X& x) { foo(x); // this call } Locally, the recursive call looks like "mutable reference to dependent type". But when instantiated, X may be const int and is in fact very likely to be deduced that way in practice. | |
565 | You may want to add an "unwrapping" step so that e.g. ArraySubscriptExpr a[i] can be unwrapped to a and then possibly highlighted (and its operator-overloaded form). Wouldn't suggest doing that in this patch, but you could leave a note if you like. | |
566 | And getLocation() will do the right thing for DeclRefExpr, MemberExpr, and others, so this can just be isa<DeclRefExpr, MemberExpr> with no need for dyn_cast. | |
clang-tools-extra/clangd/SemanticHighlighting.h | ||
73 | nit: no line break here. The line breaks separate normal modifiers vs scope modifiers (which are mutually exclusive) vs sentinels | |
74 | nit: "mutable" is more natural than "non-const", and modifiers don't use heavily C++-specific names anyway Please spell out "reference", we don't use other abbreviations. I'd consider "UsedAsMutableReference" rather than "passed" as this seems to be the deeper semantic idea we're expressing, and we may want to generalize this in some way |
clang-tools-extra/clangd/SemanticHighlighting.cpp | ||
---|---|---|
312 | I don't have a strong opinion on the options here, just wanted to chime in and say I also highlight Unknown prominently for similar reasons :) |
clang-tools-extra/clangd/SemanticHighlighting.cpp | ||
---|---|---|
566 | I'm not sure which getLocation() you're talking about here. There's DeclRefExpr::getLocation(), but neither Expr::getLocation() nor MemberExpr::getLocation(). Am I missing something? |
clang-tools-extra/clangd/SemanticHighlighting.cpp | ||
---|---|---|
566 | No, I think I'm just going mad (I was thinking of Decl::getLocation I guess). |
clang-tools-extra/clangd/SemanticHighlighting.cpp | ||
---|---|---|
557 | I think FD->getType()->getAs<FunctionProtoType>() should do it |
clang-tools-extra/clangd/SemanticHighlighting.cpp | ||
---|---|---|
312 | I switched to @sammccall's idea of having a map<Range, Modifier>, I agree that it feels more natural than the previous version | |
533 | Agreed, added a FIXME | |
561 | I adjusted the condition accordingly and added a few more test cases (including stuff like const int*&). | |
565 | Added a FIXME. | |
566 | Switched to getLocation() and added a test case. |
- Apply suggested renaming and fix nits
- Use a map for extra modifiers instead of abusing conflict resolution
- Use ArrayRef instead of callback
- Add FIXME for handling CXXOperatorCallExpr
- Use getLocation() instead of getBeginLoc()
- Add FIXME for unwrapping operators
- Iterate over FunctionProtoType instead of FunctionDecl
- Adjust highlighting condition and revise test cases
@sammccall @nridge Thanks for the positive feedback, let me know if I missed any of your suggestions!
Thanks for the update! Looks pretty good, just a few minor comments from me. Will leave approval to Sam.
clang-tools-extra/clangd/SemanticHighlighting.cpp | ||
---|---|---|
469 | Looking at existing usage of associative containers in clangd code, llvm::DenseMap and llvm::DenseSet (which are hashtable-based) are more commonly used than std::map and std::set, though it may require some boilerplate to teach them to use new key types (example) We could also consider making the value just a vector instead of a set (maybe even an llvm::SmallVector<HighlightingModifier, 1> given its anticipated usage), since even in the unlikely case that we get duplicates, addModifier() will handle them correctly | |
552 | nit: declarations --> types | |
553 | nit: parameters | |
559 | I think we can relax the dependent type check here a bit, for example std::vector<T>& is a dependent type but it's definitely a mutable reference. (Feel free to leave that for later and just add a FIXME.) | |
565 | Maybe also add a // FIXME: Handle dependent expression types just so we don't forget | |
clang-tools-extra/clangd/unittests/SemanticHighlightingTests.cpp | ||
773 | (test case needs uncommenting) |
clang-tools-extra/clangd/SemanticHighlighting.cpp | ||
---|---|---|
469 | Agreed, changed std::set to llvm::SmallVector. I can't really judge if it's worth using llvm::DenseMap here though. | |
clang-tools-extra/clangd/unittests/SemanticHighlightingTests.cpp | ||
773 | Actually it needed to be removed, has been replaced by the test above |
Thanks! This looks good to me to land as is.
I'd like to see the std::map eliminated, but if you don't want to do this I'm happy to take a stab at it after.
Want someone to commit this for you?
clang-tools-extra/clangd/SemanticHighlighting.cpp | ||
---|---|---|
469 | We do try to avoid std::map/std::set as they're pretty gratuitous performance sinks. And the map here may not be tiny. I'd suggest either:
We don't really need the full Range in the key, as the start defines it.
In practice these sets of modifiers end up being a bitmask, so if using a map, you could just store a uint32_t of 1<<Modifier and add an addModifiers(uint32_t) to HighlightingToken. I'm too really worried about preserving that encapsulation. |
Yeah, I‘d need someone to commit this for me.
About the std::map thing: let’s proceed as suggested and have @sammccall take look at it himself after :)
I'm happy to commit it. Did you want to make the SmallVector<1> --> uint32_t change, or should I just commit it as is?
I’d say commit as is and have a look at the SmallVector -> unit32_t thing together with the std::map issue afterwards
nit: no line break here. The line breaks separate normal modifiers vs scope modifiers (which are mutually exclusive) vs sentinels