Skip to content

Commit 3989788

Browse files
committed
[clangd] Implement LSP 3.17 positionEncoding
1 parent 4e80a03 commit 3989788

File tree

4 files changed

+72
-6
lines changed

4 files changed

+72
-6
lines changed

clang-tools-extra/clangd/ClangdLSPServer.cpp

Lines changed: 28 additions & 6 deletions
Original file line numberDiff line numberDiff line change
@@ -494,13 +494,22 @@ static std::vector<llvm::StringRef> semanticTokenModifiers() {
494494
void ClangdLSPServer::onInitialize(const InitializeParams &Params,
495495
Callback<llvm::json::Value> Reply) {
496496
// Determine character encoding first as it affects constructed ClangdServer.
497-
if (Params.capabilities.offsetEncoding && !Opts.Encoding) {
498-
Opts.Encoding = OffsetEncoding::UTF16; // fallback
499-
for (OffsetEncoding Supported : *Params.capabilities.offsetEncoding)
500-
if (Supported != OffsetEncoding::UnsupportedEncoding) {
501-
Opts.Encoding = Supported;
502-
break;
497+
if (!Opts.Encoding) {
498+
if (Params.capabilities.PositionEncodings) {
499+
for (OffsetEncoding Supported : *Params.capabilities.PositionEncodings) {
500+
if (Supported != OffsetEncoding::UnsupportedEncoding) {
501+
Opts.Encoding = Supported;
502+
break;
503+
}
504+
}
505+
} else if (Params.capabilities.offsetEncoding) {
506+
for (OffsetEncoding Supported : *Params.capabilities.offsetEncoding) {
507+
if (Supported != OffsetEncoding::UnsupportedEncoding) {
508+
Opts.Encoding = Supported;
509+
break;
510+
}
503511
}
512+
}
504513
}
505514

506515
if (Params.capabilities.TheiaSemanticHighlighting &&
@@ -686,6 +695,9 @@ void ClangdLSPServer::onInitialize(const InitializeParams &Params,
686695
ServerCaps["executeCommandProvider"] =
687696
llvm::json::Object{{"commands", Commands}};
688697

698+
if (Opts.Encoding)
699+
ServerCaps["positionEncoding"] = *Opts.Encoding;
700+
689701
llvm::json::Object Result{
690702
{{"serverInfo",
691703
llvm::json::Object{
@@ -697,6 +709,16 @@ void ClangdLSPServer::onInitialize(const InitializeParams &Params,
697709
Result["offsetEncoding"] = *Opts.Encoding;
698710
Reply(std::move(Result));
699711

712+
if (Params.capabilities.offsetEncoding) {
713+
ShowMessageParams Msg;
714+
Msg.message =
715+
"offsetEncoding is deprecated in favor of general.positionEncodings. "
716+
"Migrate to standard positionEncodings request";
717+
Msg.type = MessageType::Warning;
718+
elog(Msg.message.c_str());
719+
ShowMessage(Msg);
720+
}
721+
700722
// Apply settings after we're fully initialized.
701723
// This can start background indexing and in turn trigger LSP notifications.
702724
applyConfiguration(Params.initializationOptions.ConfigSettings);

clang-tools-extra/clangd/Protocol.cpp

Lines changed: 6 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -497,6 +497,12 @@ bool fromJSON(const llvm::json::Value &Params, ClientCapabilities &R,
497497
if (auto Cancel = StaleRequestSupport->getBoolean("cancel"))
498498
R.CancelsStaleRequests = *Cancel;
499499
}
500+
if (auto *PositionEncodings = General->get("positionEncodings")) {
501+
R.PositionEncodings.emplace();
502+
if (!fromJSON(*PositionEncodings, *R.PositionEncodings,
503+
P.field("general").field("positionEncodings")))
504+
return false;
505+
}
500506
}
501507
if (auto *OffsetEncoding = O->get("offsetEncoding")) {
502508
R.offsetEncoding.emplace();

clang-tools-extra/clangd/Protocol.h

Lines changed: 6 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -529,8 +529,14 @@ struct ClientCapabilities {
529529
bool TheiaSemanticHighlighting = false;
530530

531531
/// Supported encodings for LSP character offsets. (clangd extension).
532+
///
533+
/// @deprecated in favour of PositionEncodings.
532534
std::optional<std::vector<OffsetEncoding>> offsetEncoding;
533535

536+
/// Supported encodings for LSP character offsets. (introduced by LSP 3.17).
537+
/// general.positionEncodings
538+
std::optional<std::vector<OffsetEncoding>> PositionEncodings;
539+
534540
/// The content format that should be used for Hover requests.
535541
/// textDocument.hover.contentEncoding
536542
MarkupKind HoverContentFormat = MarkupKind::PlainText;
Lines changed: 32 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,32 @@
1+
# RUN: clangd -lit-test < %s | FileCheck -strict-whitespace %s
2+
# This test verifies that we can negotiate UTF-8 offsets via the positionEncodings capability introduced in LSP 3.17.
3+
{"jsonrpc":"2.0","id":0,"method":"initialize","params":{"processId":123,"rootPath":"clangd","capabilities":{"general":{"positionEncodings":["utf-8","utf-16"]}},"trace":"off"}}
4+
# CHECK: "positionEncoding": "utf-8"
5+
---
6+
{"jsonrpc":"2.0","method":"textDocument/didOpen","params":{"textDocument":{"uri":"test:///main.cpp","languageId":"cpp","version":1,"text":"/*ö*/int x;\nint y=x;"}}}
7+
---
8+
{"jsonrpc":"2.0","id":1,"method":"textDocument/definition","params":{"textDocument":{"uri":"test:///main.cpp"},"position":{"line":1,"character":6}}}
9+
# /*ö*/int x;
10+
# 01234567890
11+
# x is character (and utf-16) range [9,10) but byte range [10,11).
12+
# CHECK: "id": 1,
13+
# CHECK-NEXT: "jsonrpc": "2.0",
14+
# CHECK-NEXT: "result": [
15+
# CHECK-NEXT: {
16+
# CHECK-NEXT: "range": {
17+
# CHECK-NEXT: "end": {
18+
# CHECK-NEXT: "character": 11,
19+
# CHECK-NEXT: "line": 0
20+
# CHECK-NEXT: },
21+
# CHECK-NEXT: "start": {
22+
# CHECK-NEXT: "character": 10,
23+
# CHECK-NEXT: "line": 0
24+
# CHECK-NEXT: }
25+
# CHECK-NEXT: },
26+
# CHECK-NEXT: "uri": "file://{{.*}}/main.cpp"
27+
# CHECK-NEXT: }
28+
# CHECK-NEXT: ]
29+
---
30+
{"jsonrpc":"2.0","id":10000,"method":"shutdown"}
31+
---
32+
{"jsonrpc":"2.0","method":"exit"}

0 commit comments

Comments
 (0)