-
Notifications
You must be signed in to change notification settings - Fork 7
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
This new feature is gated behind the `async` feature flag. It allows to asynchronously await for a pool slot to become available, making it easier to share constrained resources between multiple tasks. It requires the `AtomicWaker` functionality from the `embassy-sync` crate, which in turn requires a critical section implementation.
- Loading branch information
1 parent
d93c469
commit 8a7ca7f
Showing
8 changed files
with
300 additions
and
14 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -20,4 +20,4 @@ jobs: | |
override: true | ||
components: miri | ||
- name: Test | ||
run: cargo miri test | ||
run: cargo miri test --all-features |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,79 @@ | ||
use embassy_executor::Spawner; | ||
use embassy_futures::join::join5; | ||
use embassy_time::Timer; | ||
|
||
use std::{mem, process}; | ||
|
||
use atomic_pool::{pool, Box}; | ||
|
||
#[derive(Debug)] | ||
#[allow(dead_code)] | ||
struct Packet(u32); | ||
|
||
// A maximum of 2 Packet instances can be allocated at a time. | ||
// A maximum of 1 future can be waiting at a time. | ||
pool!(PacketPool: [Packet; 2], 1); | ||
|
||
#[embassy_executor::task] | ||
async fn run() { | ||
// Allocate non-blocking | ||
let fut1 = async { | ||
println!("1 - allocating async..."); | ||
let box1 = Box::<PacketPool>::new(Packet(1)); | ||
println!("1 - allocated: {:?}", box1); | ||
Timer::after_millis(100).await; | ||
println!("1 - dropping allocation..."); | ||
mem::drop(box1); | ||
}; | ||
|
||
// Allocate asynchronously | ||
let fut2 = async { | ||
Timer::after_millis(5).await; | ||
println!("2 - allocating sync..."); | ||
let box2 = Box::<PacketPool>::new_async(Packet(2)).await; | ||
println!("2 - allocated: {:?}", box2); | ||
Timer::after_millis(150).await; | ||
println!("2 - dropping allocation..."); | ||
mem::drop(box2); | ||
}; | ||
|
||
// Allocate non-blocking (fails, data pool is full) | ||
let fut3 = async { | ||
Timer::after_millis(10).await; | ||
println!("3 - allocating sync..."); | ||
let box3 = Box::<PacketPool>::new(Packet(3)); | ||
println!( | ||
"3 - allocation fails because the data pool is full: {:?}", | ||
box3 | ||
); | ||
}; | ||
|
||
// Allocate asynchronously (waits for a deallocation) | ||
let fut4 = async { | ||
Timer::after_millis(15).await; | ||
println!("4 - allocating async..."); | ||
let box4 = Box::<PacketPool>::new_async(Packet(4)).await; | ||
println!("4 - allocated: {:?}", box4); | ||
Timer::after_millis(100).await; | ||
println!("4 - dropping allocation..."); | ||
}; | ||
|
||
// Allocate asynchronously (fails, waker pool is full) | ||
let fut5 = async { | ||
Timer::after_millis(20).await; | ||
println!("5 - allocating async..."); | ||
let box5 = Box::<PacketPool>::new_async(Packet(5)).await; | ||
println!( | ||
"5 - allocation fails because the waker pool is full: {:?}", | ||
box5 | ||
); | ||
}; | ||
|
||
join5(fut1, fut2, fut3, fut4, fut5).await; | ||
process::exit(0); | ||
} | ||
|
||
#[embassy_executor::main] | ||
async fn main(spawner: Spawner) { | ||
spawner.spawn(run()).unwrap(); | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,52 @@ | ||
use super::AtomicBitset; | ||
|
||
/// Automatically frees the Bitset slot when DroppableBit is dropped | ||
/// Useful for async environments where the future might be dropped before it completes | ||
pub struct DroppableBit<'a, const N: usize, const K: usize> { | ||
bitset: &'a AtomicBitset<N, K>, | ||
inner: usize, | ||
} | ||
|
||
impl<'a, const N: usize, const K: usize> DroppableBit<'a, N, K> { | ||
/// Only a single instance of DroppableBit should be created for each slot | ||
/// Restrict it to only be created by AtomicBitset `alloc_droppable` method | ||
pub(super) fn new(bitset: &'a AtomicBitset<N, K>, inner: usize) -> Self { | ||
Self { bitset, inner } | ||
} | ||
|
||
pub fn inner(&self) -> usize { | ||
self.inner | ||
} | ||
} | ||
|
||
impl<const N: usize, const K: usize> Drop for DroppableBit<'_, N, K> { | ||
fn drop(&mut self) { | ||
self.bitset.free(self.inner); | ||
} | ||
} | ||
|
||
#[cfg(test)] | ||
mod test { | ||
use super::*; | ||
|
||
#[test] | ||
fn test_16() { | ||
let s = AtomicBitset::<16, 1>::new(); | ||
let mut v = vec![]; | ||
|
||
for _ in 0..16 { | ||
let bit = s.alloc().map(|i| DroppableBit::new(&s, i)); | ||
assert!(bit.is_some()); | ||
|
||
v.push(bit.unwrap()); | ||
} | ||
assert_eq!(s.alloc(), None); | ||
v.pop(); | ||
v.pop(); | ||
assert!(s.alloc().is_some()); | ||
assert!(s.alloc().is_some()); | ||
assert_eq!(s.alloc(), None); | ||
v.pop(); | ||
assert!(s.alloc().is_some()); | ||
} | ||
} |
Oops, something went wrong.