Skip to content

Commit b795b7b

Browse files
authored
Merge pull request #18972 from osiewicz/drop-outgoing-messages-on-background-thread
lsp-server: Drop outgoing messages on background thread
2 parents 40ba51c + b57157e commit b795b7b

File tree

3 files changed

+42
-12
lines changed

3 files changed

+42
-12
lines changed

lib/lsp-server/src/msg.rs

+4-4
Original file line numberDiff line numberDiff line change
@@ -181,15 +181,15 @@ impl Message {
181181

182182
Ok(Some(msg))
183183
}
184-
pub fn write(self, w: &mut impl Write) -> io::Result<()> {
184+
pub fn write(&self, w: &mut impl Write) -> io::Result<()> {
185185
self._write(w)
186186
}
187-
fn _write(self, w: &mut dyn Write) -> io::Result<()> {
187+
fn _write(&self, w: &mut dyn Write) -> io::Result<()> {
188188
#[derive(Serialize)]
189-
struct JsonRpc {
189+
struct JsonRpc<'a> {
190190
jsonrpc: &'static str,
191191
#[serde(flatten)]
192-
msg: Message,
192+
msg: &'a Message,
193193
}
194194
let text = serde_json::to_string(&JsonRpc { jsonrpc: "2.0", msg: self })?;
195195
write_msg_text(w, &text)

lib/lsp-server/src/socket.rs

+18-5
Original file line numberDiff line numberDiff line change
@@ -15,8 +15,11 @@ pub(crate) fn socket_transport(
1515
stream: TcpStream,
1616
) -> (Sender<Message>, Receiver<Message>, IoThreads) {
1717
let (reader_receiver, reader) = make_reader(stream.try_clone().unwrap());
18-
let (writer_sender, writer) = make_write(stream);
19-
let io_threads = make_io_threads(reader, writer);
18+
let (writer_sender, writer, messages_to_drop) = make_write(stream);
19+
let dropper = std::thread::spawn(move || {
20+
messages_to_drop.into_iter().for_each(drop);
21+
});
22+
let io_threads = make_io_threads(reader, writer, dropper);
2023
(writer_sender, reader_receiver, io_threads)
2124
}
2225

@@ -36,11 +39,21 @@ fn make_reader(stream: TcpStream) -> (Receiver<Message>, thread::JoinHandle<io::
3639
(reader_receiver, reader)
3740
}
3841

39-
fn make_write(mut stream: TcpStream) -> (Sender<Message>, thread::JoinHandle<io::Result<()>>) {
42+
fn make_write(
43+
mut stream: TcpStream,
44+
) -> (Sender<Message>, thread::JoinHandle<io::Result<()>>, Receiver<Message>) {
4045
let (writer_sender, writer_receiver) = bounded::<Message>(0);
46+
let (drop_sender, drop_receiver) = bounded::<Message>(0);
4147
let writer = thread::spawn(move || {
42-
writer_receiver.into_iter().try_for_each(|it| it.write(&mut stream)).unwrap();
48+
writer_receiver
49+
.into_iter()
50+
.try_for_each(|it| {
51+
let result = it.write(&mut stream);
52+
let _ = drop_sender.send(it);
53+
result
54+
})
55+
.unwrap();
4356
Ok(())
4457
});
45-
(writer_sender, writer)
58+
(writer_sender, writer, drop_receiver)
4659
}

lib/lsp-server/src/stdio.rs

+20-3
Original file line numberDiff line numberDiff line change
@@ -11,15 +11,24 @@ use crate::Message;
1111

1212
/// Creates an LSP connection via stdio.
1313
pub(crate) fn stdio_transport() -> (Sender<Message>, Receiver<Message>, IoThreads) {
14+
let (drop_sender, drop_receiver) = bounded::<Message>(0);
1415
let (writer_sender, writer_receiver) = bounded::<Message>(0);
1516
let writer = thread::Builder::new()
1617
.name("LspServerWriter".to_owned())
1718
.spawn(move || {
1819
let stdout = stdout();
1920
let mut stdout = stdout.lock();
20-
writer_receiver.into_iter().try_for_each(|it| it.write(&mut stdout))
21+
writer_receiver.into_iter().try_for_each(|it| {
22+
let result = it.write(&mut stdout);
23+
let _ = drop_sender.send(it);
24+
result
25+
})
2126
})
2227
.unwrap();
28+
let dropper = thread::Builder::new()
29+
.name("LspMessageDropper".to_owned())
30+
.spawn(move || drop_receiver.into_iter().for_each(drop))
31+
.unwrap();
2332
let (reader_sender, reader_receiver) = bounded::<Message>(0);
2433
let reader = thread::Builder::new()
2534
.name("LspServerReader".to_owned())
@@ -41,21 +50,23 @@ pub(crate) fn stdio_transport() -> (Sender<Message>, Receiver<Message>, IoThread
4150
Ok(())
4251
})
4352
.unwrap();
44-
let threads = IoThreads { reader, writer };
53+
let threads = IoThreads { reader, writer, dropper };
4554
(writer_sender, reader_receiver, threads)
4655
}
4756

4857
// Creates an IoThreads
4958
pub(crate) fn make_io_threads(
5059
reader: thread::JoinHandle<io::Result<()>>,
5160
writer: thread::JoinHandle<io::Result<()>>,
61+
dropper: thread::JoinHandle<()>,
5262
) -> IoThreads {
53-
IoThreads { reader, writer }
63+
IoThreads { reader, writer, dropper }
5464
}
5565

5666
pub struct IoThreads {
5767
reader: thread::JoinHandle<io::Result<()>>,
5868
writer: thread::JoinHandle<io::Result<()>>,
69+
dropper: thread::JoinHandle<()>,
5970
}
6071

6172
impl IoThreads {
@@ -64,6 +75,12 @@ impl IoThreads {
6475
Ok(r) => r?,
6576
Err(err) => std::panic::panic_any(err),
6677
}
78+
match self.dropper.join() {
79+
Ok(_) => (),
80+
Err(err) => {
81+
std::panic::panic_any(err);
82+
}
83+
}
6784
match self.writer.join() {
6885
Ok(r) => r,
6986
Err(err) => {

0 commit comments

Comments
 (0)