- Look up in parent directories for a .clang-format file. Use "LLVM" as fallback style.
- Move cached fixits to ClangdServer, remove fixits once the client closes the document.
- Fix translations between LSP rows/cols (0-based) and clang row/cols (1-based).
Details
Diff Detail
- Build Status
Buildable 11629 Build 11629: arc lint + arc unit
Event Timeline
Does anyone have an idea how I would write a test for this?
Also, I should have commit access after approval now. So that's less work for you guys ;)
Thanks for this!
clangd/ClangdLSPServer.cpp | ||
---|---|---|
93 | This function is pretty hard to understand from a high level - it's a bit of an odd split. Is the duplication really so bad? The straight-line code could be easy to read: auto File = Params.textDocument.uri.file; auto Edits = Server.formatRange(File, Params.range); if (Edits) C.reply(replacementsToEdits(Server.getDocument(File), Edits.get())); else C.replyError(Edits.takeError()); (The "[" won't be needed after D39435, and we should add a replyError() overload that takes an Error) | |
110 | It doesn't seem reasonable to assume the error is always .clang-format. Can we just include the message in the error object? | |
clangd/ClangdServer.cpp | ||
41 | This needs to use the VFS I think. (I'm not familiar with the details of the VFS usage I'm afraid) |
clangd/ClangdLSPServer.cpp | ||
---|---|---|
93 | This is a very nice idea! I'll probably update the diff tomorrow. |
Use llvm::Error for beautiful handling of errors.
It's perhaps an idea to write a replyExpected(llvm::Expected<T>), that will
transparently handle llvm::Errors.
I'll write a unit test for these changes next.
It's probably best to wait for https://reviews.llvm.org/D39435 to land, and then rebase.
clangd/ClangdServer.cpp | ||
---|---|---|
41 | Haven't tried implementing this yet, but from some simple experiments it doesn't seem necessary. |
- Rebase to latest SVN revision.
- Use void reply(llvm:Expected<Twine> Result) for transparent error handling.
- Use FSProvider from ClangdServer to provide a filesystem for format::getStyle.
- Organize things around llvm::Expected, I've become a fan of that class :-)
Thanks! Mostly just comments around the IO (which I'm still learning about myself).
Otherwise this LG.
I'm afraid you're going to run into some merge conflicts with rL317486 - I think/hope not too bad, though.
clangd/ClangdServer.cpp | ||
---|---|---|
41 | Thanks for doing this. The reason it's needed is we want to run clangd on systems where the source tree is not a real filesystem. Looking on the real filesystem for .clang-format files would be incorrect. | |
308 | As I understand, we tag the return values to ensure we get consistent reads, and describe the consistency of those reads, on systems that support that (e.g. snapshot filesystems). I don't think we need to do that here, because:
| |
441 | getStyle is going to stat several files, in the usual case. We should really cache this, and I think "when the file is opened" is a reasonable tradeoff between simplicity and producing sensible user-facing behavior. Unfortunately I'm not sure there's a sensible place to do things at present: addDocument() is fast and doing blocking IO there might be bad. Doing it asynchronously also seems bad (what happens if style's not ready?) Next best thing would be "first format request when the file is opened". If you feel like doing that here (only computing the style if it's not already known for this file), then feel free, but a FIXME is fine too. (FWIW, Google has a slow FS that's important to us, but we disable .clang-format scanning on that FS in other ways because it only holds code in one style. Other users might care about this, though) | |
clangd/ClangdServer.h | ||
312 | I don't understand this comment - I think we require and use Code. If you mean we can just read Code from disk, that won't work: we want to format the dirty buffer in memory, not the last saved version. |
- Merge with upstream, fix merge conflicts
- Add a FIXME for a caching strategy for .clang-format files
Is this ready to merge? I'd like to implement tests in another differential, I'm having trouble referencing a temp dir from lit in a JSON request to clangd.
clangd/ClangdServer.cpp | ||
---|---|---|
441 |
I've added a FIXME note. I think implementing a caching strategy is worthy of its own differential revision. |
clangd/ClangdLSPServer.cpp | ||
---|---|---|
118 | NIT: remove braces from single-statement branches | |
126 | NIT: remove braces from single-statement branches | |
clangd/ClangdServer.h | ||
297 | Why do we accept Code as a parameter here instead of getting it internally? Maybe we should consider moving this method out of ClangdServer? Its signature looks pretty self-contained now. |
clangd/ClangdServer.h | ||
---|---|---|
297 | There are a couple intertwined problems:
So yes, in principle you can call getDocument when you need it for replacementsToEdits, and you can let formatCode itself call getDocument for clang::format::getStyle. But then we create two copies of the document contents for one LSP request. If getDocument returned an llvm::StringRef, I'd probably vote for removing the Code argument everywhere and call getDocument as needed. |
clangd/ClangdServer.h | ||
---|---|---|
297 | Oh, I see. Calling getDocument twice does not really make sense. We could've made getDocument return StringRef, but we'd have to be more careful to ensure it's actually copied when we're scheduling async operations, worth a separate review. |
clangd/ClangdServer.h | ||
---|---|---|
297 |
I disagree, this will bring LSP-specific protocols into ClangdServer. The translation from tooling::Replacement to clangd::TextEdit should remain in ClangdLSPServer. |
clangd/ClangdServer.h | ||
---|---|---|
297 | I tried this anyway, but ClangdLSPServer::getFixIts also uses replacementsToEdits. And that maintains a map with tooling::Replacements. So, if we want to move replacementsToEdits into ClangdServer, we would have to move that map into ClangdServer too I think. |
clangd/ClangdServer.h | ||
---|---|---|
297 |
+1 to this, I'd also not make ClangdServer LSP-specific. However we already use some LSP structs from Protocol.h there to avoid code duplication (i.e., CompletionItem, though LSP-specific, suits ClangdServer pretty well). In principle, Protocol.h should not be a dependency, though, but we try to manage what we take from there instead. clangd::TextEdit seems to be in line with other things we take from there now (that is, Ranges, Locations, etc)
Could we change the map to store clangd::TextEdit instead? It's a bit unfortunate that we'll be making unnecessary tooling::Replacement -> clangd::TextEdit conversions and those aren't free. But honestly, I don't think this should bite us performance-wise. |
- Rebase to latest upstream revision.
- Go all-in with TextEdit, even down to ClangdUnit.cpp.
- Move FixItsMap to ClangdServer. ClangdLSPServer is much cleaner now.
- Remove the cached FixIts when the client closes the document.
- Fix (or introduce bug?) wrong conversion between clang line/columns and LSP line/columns. From what I understand, clang line/cols are 1-based, and LSP line/cols are 0-based. The line numbers were already converted correctly (subtracting 1 when going from clang line numbers to LSP line numbers), but the column numbers were not converted correctly. Consequently, a bunch of tests had to be adjusted for the correct (or wrong?) column numbers.
clangd/ClangdUnit.cpp | ||
---|---|---|
168 ↗ | (On Diff #124307) | Why subtract 1 from the line here, but not from the column? When subtracting 1 from both, diagnostics appear correct in my client. I have moved this logic into SourceLocToPosition. |
@ilya-biryukov We should get this landed.
I'm happy (my comments were addressed a few rounds ago) - any concerns?
- Merge with upstream revision 319613
- Fix FixItHints not getting applied correctly in all cases. The problem is that clang::CharSourceRange can either be a TokenRange or a CharRange. In the case of a TokenRange, we have to determine the end of the token ourselves (this would occur, for example, in things like if (i=1) { ... }; the fixit hint to replace = with == would be a TokenRange for some reason).
Consequently, a tiny change had to be made to the fixits.test file, where the fixit hint to replace = with == would previously have an empty replacement range, whereas now it has a replacement range of length 1, exactly removing the = and inserting == in its place.
After playing around with these changes more I'm fairly certain I'm on the right
track for converting clang ranges/positions to clangd ranges/positions.
- Look up in parent directories for a .clang-format file. Use "LLVM" as fallback style.
- Move cached fixits to ClangdServer, remove fixits once the client closes the document.
- Fix translations between LSP rows/cols (0-based) and clang row/cols (1-based).
I'd rather keep one commit per fix or feature. I would think the clang-format part should be one commit/review on its own. But I'm not very familiar with the practices here so I'll let others comment.
+1, please create a separate review per fix or feature.
clangd/ClangdServer.h | ||
---|---|---|
296 | There are two ways to return fixits from ClangdServer now, one is DiagnosticsConsumer, the other is the new getFixIts method. I'd also argue the fact that we can't attach fixits to diagnostics is LSP-specific and therefore it should be handled by ClangdLSPServer. I think we should not put this method into ClangdServer. |
This differential is now split into https://reviews.llvm.org/D41031 and https://reviews.llvm.org/D40860.
This function is pretty hard to understand from a high level - it's a bit of an odd split.
Is the duplication really so bad? The straight-line code could be easy to read:
(The "[" won't be needed after D39435, and we should add a replyError() overload that takes an Error)