Skip to content

Commit 0668e53

Browse files
authored
Rollup merge of rust-lang#90439 - m-ou-se:thread-is-running, r=Mark-Simulacrum
Add JoinHandle::is_running. This adds: ```rust impl<T> JoinHandle<T> { /// Checks if the the associated thread is still running its main function. /// /// This might return `false` for a brief moment after the thread's main /// function has returned, but before the thread itself has stopped running. pub fn is_running(&self) -> bool; } ``` The usual way to check if a background thread is still running is to set some atomic flag at the end of its main function. We already do that, in the form of dropping an Arc which will reduce the reference counter. So we might as well expose that information. This is useful in applications with a main loop (e.g. a game, gui, control system, ..) where you spawn some background task, and check every frame/iteration whether the background task is finished to .join() it in that frame/iteration while keeping the program responsive.
2 parents 35f74b2 + 978ebd9 commit 0668e53

File tree

2 files changed

+44
-1
lines changed

2 files changed

+44
-1
lines changed

library/std/src/thread/mod.rs

+9
Original file line numberDiff line numberDiff line change
@@ -1407,6 +1407,15 @@ impl<T> JoinHandle<T> {
14071407
pub fn join(mut self) -> Result<T> {
14081408
self.0.join()
14091409
}
1410+
1411+
/// Checks if the the associated thread is still running its main function.
1412+
///
1413+
/// This might return `false` for a brief moment after the thread's main
1414+
/// function has returned, but before the thread itself has stopped running.
1415+
#[unstable(feature = "thread_is_running", issue = "90470")]
1416+
pub fn is_running(&self) -> bool {
1417+
Arc::strong_count(&self.0.packet.0) > 1
1418+
}
14101419
}
14111420

14121421
impl<T> AsInner<imp::Thread> for JoinHandle<T> {

library/std/src/thread/tests.rs

+35-1
Original file line numberDiff line numberDiff line change
@@ -2,9 +2,13 @@ use super::Builder;
22
use crate::any::Any;
33
use crate::mem;
44
use crate::result;
5-
use crate::sync::mpsc::{channel, Sender};
5+
use crate::sync::{
6+
mpsc::{channel, Sender},
7+
Arc, Barrier,
8+
};
69
use crate::thread::{self, ThreadId};
710
use crate::time::Duration;
11+
use crate::time::Instant;
812

913
// !!! These tests are dangerous. If something is buggy, they will hang, !!!
1014
// !!! instead of exiting cleanly. This might wedge the buildbots. !!!
@@ -46,6 +50,36 @@ fn test_run_basic() {
4650
rx.recv().unwrap();
4751
}
4852

53+
#[test]
54+
fn test_is_running() {
55+
let b = Arc::new(Barrier::new(2));
56+
let t = thread::spawn({
57+
let b = b.clone();
58+
move || {
59+
b.wait();
60+
1234
61+
}
62+
});
63+
64+
// Thread is definitely running here, since it's still waiting for the barrier.
65+
assert_eq!(t.is_running(), true);
66+
67+
// Unblock the barrier.
68+
b.wait();
69+
70+
// Now check that t.is_running() becomes false within a reasonable time.
71+
let start = Instant::now();
72+
while t.is_running() {
73+
assert!(start.elapsed() < Duration::from_secs(2));
74+
thread::sleep(Duration::from_millis(15));
75+
}
76+
77+
// Joining the thread should not block for a significant time now.
78+
let join_time = Instant::now();
79+
assert_eq!(t.join().unwrap(), 1234);
80+
assert!(join_time.elapsed() < Duration::from_secs(2));
81+
}
82+
4983
#[test]
5084
fn test_join_panic() {
5185
match thread::spawn(move || panic!()).join() {

0 commit comments

Comments
 (0)