From f461dc48d116a97188ae6d9934c8e3b421c335ac Mon Sep 17 00:00:00 2001 From: Edwin Cheng Date: Tue, 31 Mar 2020 21:51:43 +0800 Subject: [PATCH] Use a weak ptr to hold the send end of channel --- crates/ra_proc_macro/src/process.rs | 46 +++++++++++------------------ 1 file changed, 18 insertions(+), 28 deletions(-) diff --git a/crates/ra_proc_macro/src/process.rs b/crates/ra_proc_macro/src/process.rs index 07dc1bf9ef0..b44d6763c01 100644 --- a/crates/ra_proc_macro/src/process.rs +++ b/crates/ra_proc_macro/src/process.rs @@ -12,32 +12,24 @@ use std::{ io::{self, Write}, path::{Path, PathBuf}, process::{Child, Command, Stdio}, + sync::{Arc, Weak}, }; #[derive(Debug, Default)] pub(crate) struct ProcMacroProcessSrv { - inner: Option>, + inner: Option>>, } #[derive(Debug)] pub(crate) struct ProcMacroProcessThread { // XXX: drop order is significant - sender: SenderGuard, + sender: Arc>, handle: jod_thread::JoinHandle<()>, } -#[derive(Debug)] -struct SenderGuard(pub Sender); - -impl std::ops::Drop for SenderGuard { - fn drop(&mut self) { - self.0.send(Task::Close).unwrap(); - } -} - -enum Task { - Request { req: Request, result_tx: Sender }, - Close, +struct Task { + req: Request, + result_tx: Sender, } struct Process { @@ -88,8 +80,9 @@ impl ProcMacroProcessSrv { client_loop(task_rx, process); }); - let srv = ProcMacroProcessSrv { inner: Some(task_tx.clone()) }; - let thread = ProcMacroProcessThread { handle, sender: SenderGuard(task_tx) }; + let task_tx = Arc::new(task_tx); + let srv = ProcMacroProcessSrv { inner: Some(Arc::downgrade(&task_tx)) }; + let thread = ProcMacroProcessThread { handle, sender: task_tx }; Ok((thread, srv)) } @@ -131,8 +124,13 @@ impl ProcMacroProcessSrv { }; let (result_tx, result_rx) = bounded(0); - - sender.send(Task::Request { req: req.into(), result_tx }).unwrap(); + let sender = match sender.upgrade() { + None => { + return Err(ra_tt::ExpansionError::Unknown("Proc macro process is closed.".into())) + } + Some(it) => it, + }; + sender.send(Task { req: req.into(), result_tx }).unwrap(); let res = result_rx.recv().unwrap(); match res { @@ -155,16 +153,8 @@ fn client_loop(task_rx: Receiver, mut process: Process) { Some(it) => it, }; - loop { - let task = match task_rx.recv() { - Ok(task) => task, - Err(_) => break, - }; - - let (req, result_tx) = match task { - Task::Request { req, result_tx } => (req, result_tx), - Task::Close => break, - }; + for task in task_rx { + let Task { req, result_tx } = task; let res = match send_request(&mut stdin, &mut stdout, req) { Ok(res) => res,