Use automatic thread joining for cargo-watch

This commit is contained in:
Aleksey Kladov 2020-03-28 11:31:10 +01:00
parent 8bea5ace7e
commit f7df0b56a7
3 changed files with 10 additions and 40 deletions

1
Cargo.lock generated
View file

@ -893,6 +893,7 @@ dependencies = [
"cargo_metadata",
"crossbeam-channel",
"insta",
"jod-thread",
"log",
"lsp-types",
"serde_json",

View file

@ -10,6 +10,7 @@ lsp-types = { version = "0.73.0", features = ["proposed"] }
log = "0.4.8"
cargo_metadata = "0.9.1"
serde_json = "1.0.48"
jod-thread = "0.1.1"
[dev-dependencies]
insta = "0.15.0"

View file

@ -12,7 +12,6 @@ use std::{
io::{BufRead, BufReader},
path::{Path, PathBuf},
process::{Command, Stdio},
thread::JoinHandle,
time::Instant,
};
@ -37,8 +36,9 @@ pub struct CheckOptions {
#[derive(Debug)]
pub struct CheckWatcher {
pub task_recv: Receiver<CheckTask>,
// XXX: drop order is significant
cmd_send: Option<Sender<CheckCommand>>,
handle: Option<JoinHandle<()>>,
handle: Option<jod_thread::JoinHandle<()>>,
}
impl CheckWatcher {
@ -47,7 +47,7 @@ impl CheckWatcher {
let (task_send, task_recv) = unbounded::<CheckTask>();
let (cmd_send, cmd_recv) = unbounded::<CheckCommand>();
let handle = std::thread::spawn(move || {
let handle = jod_thread::spawn(move || {
let mut check = CheckWatcherThread::new(options, workspace_root);
check.run(&task_send, &cmd_recv);
});
@ -67,22 +67,6 @@ impl CheckWatcher {
}
}
impl std::ops::Drop for CheckWatcher {
fn drop(&mut self) {
if let Some(handle) = self.handle.take() {
// Take the sender out of the option
let cmd_send = self.cmd_send.take();
// Dropping the sender finishes the thread loop
drop(cmd_send);
// Join the thread, it should finish shortly. We don't really care
// whether it panicked, so it is safe to ignore the result
let _ = handle.join();
}
}
}
#[derive(Debug)]
pub enum CheckTask {
/// Request a clearing of all cached diagnostics from the check watcher
@ -237,8 +221,9 @@ pub struct DiagnosticWithFixes {
/// The correct way to dispose of the thread is to drop it, on which the
/// sub-process will be killed, and the thread will be joined.
struct WatchThread {
handle: Option<JoinHandle<()>>,
// XXX: drop order is significant
message_recv: Receiver<CheckEvent>,
_handle: Option<jod_thread::JoinHandle<()>>,
}
enum CheckEvent {
@ -333,7 +318,7 @@ pub fn run_cargo(
impl WatchThread {
fn dummy() -> WatchThread {
WatchThread { handle: None, message_recv: never() }
WatchThread { message_recv: never(), _handle: None }
}
fn new(options: &CheckOptions, workspace_root: &Path) -> WatchThread {
@ -352,7 +337,7 @@ impl WatchThread {
let (message_send, message_recv) = unbounded();
let workspace_root = workspace_root.to_owned();
let handle = if options.enable {
Some(std::thread::spawn(move || {
Some(jod_thread::spawn(move || {
// If we trigger an error here, we will do so in the loop instead,
// which will break out of the loop, and continue the shutdown
let _ = message_send.send(CheckEvent::Begin);
@ -383,23 +368,6 @@ impl WatchThread {
} else {
None
};
WatchThread { handle, message_recv }
}
}
impl std::ops::Drop for WatchThread {
fn drop(&mut self) {
if let Some(handle) = self.handle.take() {
// Replace our reciever with dummy one, so we can drop and close the
// one actually communicating with the thread
let recv = std::mem::replace(&mut self.message_recv, never());
// Dropping the original reciever initiates thread sub-process shutdown
drop(recv);
// Join the thread, it should finish shortly. We don't really care
// whether it panicked, so it is safe to ignore the result
let _ = handle.join();
}
WatchThread { message_recv, _handle: handle }
}
}