summaryrefslogtreecommitdiff
path: root/alacritty_terminal/src/tty/windows/child.rs
diff options
context:
space:
mode:
authorMaciej Makowski <maciejm.github@cfiet.net>2019-11-16 21:11:56 +0000
committerChristian Duerr <contact@christianduerr.com>2019-11-16 22:11:56 +0100
commit48861e463311145a653350688dc4bad83a528d91 (patch)
tree6d384990dde03d27eb83a89852e75aa275b9db0e /alacritty_terminal/src/tty/windows/child.rs
parentd741d3817debe9fdd4030bede3e4c8ca84ad078a (diff)
downloadalacritty-48861e463311145a653350688dc4bad83a528d91.tar.gz
alacritty-48861e463311145a653350688dc4bad83a528d91.zip
Fix WinPTY freeze on termination
Fixes #2889.
Diffstat (limited to 'alacritty_terminal/src/tty/windows/child.rs')
-rw-r--r--alacritty_terminal/src/tty/windows/child.rs115
1 files changed, 115 insertions, 0 deletions
diff --git a/alacritty_terminal/src/tty/windows/child.rs b/alacritty_terminal/src/tty/windows/child.rs
new file mode 100644
index 00000000..447b7fbf
--- /dev/null
+++ b/alacritty_terminal/src/tty/windows/child.rs
@@ -0,0 +1,115 @@
+// Licensed under the Apache License, Version 2.0 (the "License");
+// you may not use this file except in compliance with the License.
+// You may obtain a copy of the License at
+//
+// http://www.apache.org/licenses/LICENSE-2.0
+//
+// Unless required by applicable law or agreed to in writing, software
+// distributed under the License is distributed on an "AS IS" BASIS,
+// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+// See the License for the specific language governing permissions and
+// limitations under the License.
+
+use std::ffi::c_void;
+use std::io::Error;
+use std::sync::atomic::{AtomicPtr, Ordering};
+
+use mio_extras::channel::{channel, Receiver, Sender};
+
+use winapi::shared::ntdef::{BOOLEAN, HANDLE, PVOID};
+use winapi::um::winbase::{RegisterWaitForSingleObject, UnregisterWait, INFINITE};
+use winapi::um::winnt::{WT_EXECUTEINWAITTHREAD, WT_EXECUTEONLYONCE};
+
+use crate::tty::ChildEvent;
+
+/// WinAPI callback to run when child process exits.
+extern "system" fn child_exit_callback(ctx: PVOID, timed_out: BOOLEAN) {
+ if timed_out != 0 {
+ return;
+ }
+
+ let event_tx: Box<_> = unsafe { Box::from_raw(ctx as *mut Sender<ChildEvent>) };
+ let _ = event_tx.send(ChildEvent::Exited);
+}
+
+pub struct ChildExitWatcher {
+ wait_handle: AtomicPtr<c_void>,
+ event_rx: Receiver<ChildEvent>,
+}
+
+impl ChildExitWatcher {
+ pub fn new(child_handle: HANDLE) -> Result<ChildExitWatcher, Error> {
+ let (event_tx, event_rx) = channel::<ChildEvent>();
+
+ let mut wait_handle: HANDLE = 0 as HANDLE;
+ let sender_ref = Box::new(event_tx);
+
+ let success = unsafe {
+ RegisterWaitForSingleObject(
+ &mut wait_handle,
+ child_handle,
+ Some(child_exit_callback),
+ Box::into_raw(sender_ref) as PVOID,
+ INFINITE,
+ WT_EXECUTEINWAITTHREAD | WT_EXECUTEONLYONCE,
+ )
+ };
+
+ if success == 0 {
+ Err(Error::last_os_error())
+ } else {
+ Ok(ChildExitWatcher { wait_handle: AtomicPtr::from(wait_handle), event_rx })
+ }
+ }
+
+ pub fn event_rx(&self) -> &Receiver<ChildEvent> {
+ &self.event_rx
+ }
+}
+
+impl Drop for ChildExitWatcher {
+ fn drop(&mut self) {
+ unsafe {
+ UnregisterWait(self.wait_handle.load(Ordering::Relaxed));
+ }
+ }
+}
+
+#[cfg(test)]
+mod test {
+ use std::os::windows::io::AsRawHandle;
+ use std::process::Command;
+ use std::time::Duration;
+
+ use mio::{Events, Poll, PollOpt, Ready, Token};
+
+ use super::*;
+
+ #[test]
+ pub fn event_is_emitted_when_child_exits() {
+ const WAIT_TIMEOUT: Duration = Duration::from_millis(200);
+
+ let mut child = Command::new("cmd.exe").spawn().unwrap();
+ let child_exit_watcher = ChildExitWatcher::new(child.as_raw_handle()).unwrap();
+
+ let mut events = Events::with_capacity(1);
+ let poll = Poll::new().unwrap();
+ let child_events_token = Token::from(0usize);
+
+ poll.register(
+ child_exit_watcher.event_rx(),
+ child_events_token,
+ Ready::readable(),
+ PollOpt::oneshot(),
+ )
+ .unwrap();
+
+ child.kill().unwrap();
+
+ // Poll for the event or fail with timeout if nothing has been sent
+ poll.poll(&mut events, Some(WAIT_TIMEOUT)).unwrap();
+ assert_eq!(events.iter().next().unwrap().token(), child_events_token);
+ // Verify that at least one `ChildEvent::Exited` was received
+ assert_eq!(child_exit_watcher.event_rx().try_recv(), Ok(ChildEvent::Exited));
+ }
+}