- nodes can have special-cased hit ranges including "holes" (FunctionTypeLoc in void foo())
- token conflicts between siblings (int a,b;) are resolved in favor of left sibling
- parent/child overlap is handled statefully rather than explicitly by comparing parent/child ranges (this lets us share a mechanism with sibling conflicts)
Details
Diff Detail
- Repository
- rCTE Clang Tools Extra
- Build Status
Buildable 33873 Build 33872: arc lint + arc unit
Event Timeline
Also there were some offline discussions around, handling of "invisible nodes"(e.g, ExprWithCleanups) and other types of typelocs like ParenTypeLocs and owner of '=' sign in copy/move assignment constructors
clangd/Selection.cpp | ||
---|---|---|
54 | assume we have inserted the range [1,5] and then tried inserting {[1,5], [2,3]}. Also if we are going to store possibly-overlapping OffsetRanges why are we trying to remove those? | |
136 | what about moving this to top of the file? | |
137 | already defined at top | |
221 | what happens to parentheses in this case? |
Thanks for the comments here and the offline discussion.
The problems identified were:
- when a node's range is adjusted, then invisible parents (like ExprWithCleanups) don't adjust to match and end up owning tokens.
- it would be nice if CXXConstructExpr claimed the = when copy-assigning, so it would be a go-to-definition target.
- the FunctionTypeLoc handling is naive, and won't treat void (*s)() correctly where the identifier is wrapped in a further typeloc
The best solution to the latter appears to be to lean even more on traversal order and the "already claimed" mechanism: claim the identifier for the VarDecl on the way down, before the children are traversed.
I believe this will eliminate the last case where CXXConstructExpr actually needs special bounds: Type() will have Type claimed by a sibling typeloc, Type t() will have t claimed by the parent vardecl, etc.
Removing the special bounds will make = owned by the CXXConstructExpr, and sidestep the invisible-nodes problem.
And obviously it will simplify the code to have a single range for nodes (once again). So I'll fold it into this patch.
clangd/Selection.cpp | ||
---|---|---|
54 | I forgot to mention the NewRanges must be disjoint, which (I think) makes this always correct. Why remove the covered ranges? Just to avoid growing Ranges, since the algorithm is linear in that. Not important, but we're doing the check anyway. There's no good reason to copy, remove in place, and copy again though, I'll fix that. | |
221 | There are none; we handled the parens case two lines up :-) |
Revert multi-range support. Add early hit detection (before children) instead.
Add more tests.
LGTM, thanks!
clangd/Selection.cpp | ||
---|---|---|
84 | I suppose comment should be [Begin, R.first). Could you also change the order within condition? | |
87 | nit: maybe move this check into previous condition? i.e: if (Begin < R.second) { Begin = R.second; // Prefix is covered, truncate the range. if (Begin >= End) return true; } | |
267–268 | also called from push() now | |
clangd/unittests/SelectionTests.cpp | ||
170 | could you also add a test case with cursor on identifier(i.e, s) |
assume we have inserted the range [1,5] and then tried inserting {[1,5], [2,3]}.
In that case IsCovered would return false for [2,3]. And add would return true, but all of the newranges were contained in the RangeSet
Also if we are going to store possibly-overlapping OffsetRanges why are we trying to remove those?