Skip to content

Commit

Permalink
write a test that proves the problem is there
Browse files Browse the repository at this point in the history
This is just a prototype to communicate that I didn't get the loom test
to work and to see if this testing approach would be accepted. It's not
great, because the test implementation is very coupled to the blocking
pool implementation (f.i, spawning a previous blocking task and awaiting
on it before launching the offending task). However, it takes 150ms
to run on my machine when it succeeds, and fails in the first few
attempts when using `spawn_blocking` instead of
`spawn_mandatory_blocking`, which is a good sign
  • Loading branch information
BraulioVM committed Dec 27, 2021
1 parent 155d2bd commit dfe6b0d
Showing 1 changed file with 55 additions and 0 deletions.
55 changes: 55 additions & 0 deletions tokio/src/runtime/tests/queue.rs
Original file line number Diff line number Diff line change
Expand Up @@ -189,6 +189,61 @@ fn stress2() {
}
}

#[test]
fn mandatory_blocking_tasks_get_executed() {
use crate::runtime;
use std::sync::{Arc, atomic::AtomicBool};

// We need to execute the test a few times because the failure is
// non-deterministic. I tried to write a loom that would prove the
// bug is there (when using spawn_blocking) but failed. The test
// I wrote didn't fail, even when running >30 minutes.
for i in 1..1000 {
let rt = runtime::Builder::new_multi_thread()
.max_blocking_threads(1)
.worker_threads(1)
.build().unwrap();

let did_blocking_task_execute = Arc::new(
AtomicBool::new(false)
);

{
let did_blocking_task_execute = did_blocking_task_execute.clone();
let _enter = rt.enter();

rt.block_on(async move {
// We need to have spawned a previous blocking task
// so that the thread in the blocking pool gets to
// the state where it's waiting either for a shutdown
// or another task.
runtime::spawn_blocking(move || {

}).await.unwrap();

// Changing this spawn_mandatory_blocking to
// spawn_mandatory makes the test fail in a few
// iterations
runtime::spawn_mandatory_blocking(move || {
did_blocking_task_execute.store(
true,
std::sync::atomic::Ordering::Release
);
})
});
drop(rt);
}

assert!(
did_blocking_task_execute.load(
std::sync::atomic::Ordering::Acquire
),
"Failed at iteration {:?}", i
);
}
}


struct Runtime;

impl Schedule for Runtime {
Expand Down

0 comments on commit dfe6b0d

Please sign in to comment.