3754: Use automatic thread joining for cargo-watch r=matklad a=matklad

r? @kiljacken 

Co-authored-by: Aleksey Kladov <aleksey.kladov@gmail.com>
This commit is contained in:
bors[bot] 2020-03-30 09:33:43 +00:00 committed by GitHub
commit df8752bf3f
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
5 changed files with 23 additions and 44 deletions

5
Cargo.lock generated
View file

@ -563,9 +563,9 @@ dependencies = [
[[package]] [[package]]
name = "jod-thread" name = "jod-thread"
version = "0.1.0" version = "0.1.1"
source = "registry+https://github.com/rust-lang/crates.io-index" source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "2f52a11f73b88fab829a0e4d9e13ea5982c7ac457c72eb3541d82a4afdfce4ff" checksum = "4022656272c3e564a7cdebcaaba6518d844b0d0c1836597196efb5bfeb98bb49"
[[package]] [[package]]
name = "kernel32-sys" name = "kernel32-sys"
@ -893,6 +893,7 @@ dependencies = [
"cargo_metadata", "cargo_metadata",
"crossbeam-channel", "crossbeam-channel",
"insta", "insta",
"jod-thread",
"log", "log",
"lsp-types", "lsp-types",
"serde_json", "serde_json",

View file

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

View file

@ -12,7 +12,6 @@ use std::{
io::{BufRead, BufReader}, io::{BufRead, BufReader},
path::{Path, PathBuf}, path::{Path, PathBuf},
process::{Command, Stdio}, process::{Command, Stdio},
thread::JoinHandle,
time::Instant, time::Instant,
}; };
@ -36,9 +35,10 @@ pub struct CheckOptions {
/// The spawned thread is shut down when this struct is dropped. /// The spawned thread is shut down when this struct is dropped.
#[derive(Debug)] #[derive(Debug)]
pub struct CheckWatcher { pub struct CheckWatcher {
pub task_recv: Receiver<CheckTask>, // XXX: drop order is significant
cmd_send: Option<Sender<CheckCommand>>, cmd_send: Option<Sender<CheckCommand>>,
handle: Option<JoinHandle<()>>, handle: Option<jod_thread::JoinHandle<()>>,
pub task_recv: Receiver<CheckTask>,
} }
impl CheckWatcher { impl CheckWatcher {
@ -47,7 +47,7 @@ impl CheckWatcher {
let (task_send, task_recv) = unbounded::<CheckTask>(); let (task_send, task_recv) = unbounded::<CheckTask>();
let (cmd_send, cmd_recv) = unbounded::<CheckCommand>(); 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); let mut check = CheckWatcherThread::new(options, workspace_root);
check.run(&task_send, &cmd_recv); 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)] #[derive(Debug)]
pub enum CheckTask { pub enum CheckTask {
/// Request a clearing of all cached diagnostics from the check watcher /// 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 /// 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. /// sub-process will be killed, and the thread will be joined.
struct WatchThread { struct WatchThread {
handle: Option<JoinHandle<()>>, // XXX: drop order is significant
message_recv: Receiver<CheckEvent>, message_recv: Receiver<CheckEvent>,
_handle: Option<jod_thread::JoinHandle<()>>,
} }
enum CheckEvent { enum CheckEvent {
@ -333,7 +318,7 @@ pub fn run_cargo(
impl WatchThread { impl WatchThread {
fn dummy() -> WatchThread { fn dummy() -> WatchThread {
WatchThread { handle: None, message_recv: never() } WatchThread { message_recv: never(), _handle: None }
} }
fn new(options: &CheckOptions, workspace_root: &Path) -> WatchThread { fn new(options: &CheckOptions, workspace_root: &Path) -> WatchThread {
@ -352,7 +337,7 @@ impl WatchThread {
let (message_send, message_recv) = unbounded(); let (message_send, message_recv) = unbounded();
let workspace_root = workspace_root.to_owned(); let workspace_root = workspace_root.to_owned();
let handle = if options.enable { 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, // 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 // which will break out of the loop, and continue the shutdown
let _ = message_send.send(CheckEvent::Begin); let _ = message_send.send(CheckEvent::Begin);
@ -383,23 +368,6 @@ impl WatchThread {
} else { } else {
None None
}; };
WatchThread { handle, message_recv } WatchThread { message_recv, _handle: handle }
}
}
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();
}
} }
} }

View file

@ -339,6 +339,14 @@ pub fn print_backtrace() {
let bt = backtrace::Backtrace::new(); let bt = backtrace::Backtrace::new();
eprintln!("{:?}", bt); eprintln!("{:?}", bt);
} }
#[cfg(not(feature = "backtrace"))]
pub fn print_backtrace() {
eprintln!(
r#"enable the backtrace feature:
ra_prof = {{ path = "../ra_prof", features = [ "backtrace"] }}
"#
);
}
thread_local!(static IN_SCOPE: RefCell<bool> = RefCell::new(false)); thread_local!(static IN_SCOPE: RefCell<bool> = RefCell::new(false));

View file

@ -83,9 +83,10 @@ pub fn project(fixture: &str) -> Server {
pub struct Server { pub struct Server {
req_id: Cell<u64>, req_id: Cell<u64>,
messages: RefCell<Vec<Message>>, messages: RefCell<Vec<Message>>,
dir: TempDir,
_thread: jod_thread::JoinHandle<()>, _thread: jod_thread::JoinHandle<()>,
client: Connection, client: Connection,
/// XXX: remove the tempdir last
dir: TempDir,
} }
impl Server { impl Server {