Skip to content
This repository has been archived by the owner on Jun 19, 2024. It is now read-only.

Async runtime #5

Merged
merged 51 commits into from
Dec 22, 2023
Merged
Show file tree
Hide file tree
Changes from 26 commits
Commits
Show all changes
51 commits
Select commit Hold shift + click to select a range
61a3900
add: async runtime implementation
Gavin-Niederman Oct 3, 2023
4d8ddf4
fix: switch to concurrent_queue queues
Gavin-Niederman Oct 3, 2023
c3711d6
fix: add new function to Executor
Gavin-Niederman Oct 3, 2023
5677729
fix: switch to unsafecell instad of mutex
Gavin-Niederman Oct 5, 2023
d730f1a
update todo list
Gavin-Niederman Oct 5, 2023
d475594
start implementing final executor design
Gavin-Niederman Oct 6, 2023
1113e4c
fix: switch to storing futures in raw pointers
Gavin-Niederman Oct 6, 2023
b622dad
fix: task internal now can poll futures
Gavin-Niederman Oct 7, 2023
7b4be88
feat: executor tick and run functions
Gavin-Niederman Oct 7, 2023
840a144
feat: block_on function
Gavin-Niederman Oct 7, 2023
7503003
add: ready macro
Gavin-Niederman Oct 7, 2023
c33ddc9
feat: task_local! macro
Gavin-Niederman Oct 9, 2023
ac725ae
feat: new executor
Gavin-Niederman Oct 9, 2023
40b5f95
fix: add block_on function
Gavin-Niederman Oct 9, 2023
9b8d938
fix: block on requires future to be static
Gavin-Niederman Oct 9, 2023
a3a5ddb
Merge branch 'main' into async_runtime
doinkythederp Oct 9, 2023
d4e94ab
feat: make Robot trait functions async
Gavin-Niederman Oct 9, 2023
4affa3d
add: simple motor stopped future
Gavin-Niederman Oct 9, 2023
430ee75
chore: clippy
Gavin-Niederman Oct 9, 2023
9450fb6
add: wakers and async sleep function
Gavin-Niederman Oct 10, 2023
bff860e
fix: initialize main thread tls
Gavin-Niederman Oct 12, 2023
fa7a840
fix: call init main
Gavin-Niederman Oct 12, 2023
af43133
fix: change task local storage method
Gavin-Niederman Oct 13, 2023
b2fe5bd
add: reactor
Gavin-Niederman Oct 13, 2023
fdccd4c
fix: drive all futures to completion
Gavin-Niederman Oct 14, 2023
e5ec7a6
feat: future join handles
Gavin-Niederman Oct 14, 2023
6c85e72
docs: document async executor functions
Gavin-Niederman Oct 17, 2023
3635741
fix: remove unused marker field
Gavin-Niederman Oct 17, 2023
8d9b65a
docs(async_runtime): specify the benefits of spawn()
doinkythederp Oct 17, 2023
39c0fa4
fix: correct slightly incorrect docs on spawn
Gavin-Niederman Oct 17, 2023
3b73a5e
feat: support panic messages in simulator
doinkythederp Oct 17, 2023
6fd1957
fix: panic messages
doinkythederp Oct 17, 2023
b686f34
add: basic example
Gavin-Niederman Oct 18, 2023
6ca9686
feat: complete_timeout
Gavin-Niederman Oct 18, 2023
046da3c
refactor: rename task_local to os_task_local
Gavin-Niederman Oct 18, 2023
6b8ed48
refactor(async): simplify runtime using async-task
max-niederman Oct 21, 2023
2ab4ce2
fix(async): only poll task when woken
max-niederman Oct 21, 2023
ef4c69c
fix: use CString directly
doinkythederp Oct 21, 2023
e4a1c01
fix: remove std dependency from task init code
max-niederman Oct 21, 2023
7b1a86c
Merge branch 'async_runtime' into async-simplified
max-niederman Oct 21, 2023
16c290e
Merge pull request #6 from max-niederman/async-simplified
Gavin-Niederman Oct 21, 2023
e589833
Merge branch 'main' into async_runtime
Gavin-Niederman Oct 21, 2023
54ee3d2
feat: seperate async and sync robot traits
Gavin-Niederman Oct 22, 2023
72348f1
fix: get rid lockup in task locals
Gavin-Niederman Dec 9, 2023
5698cdf
refactor: change task module structure
Gavin-Niederman Dec 9, 2023
d556ce8
Merge branch 'main' into async_runtime
Gavin-Niederman Dec 11, 2023
731533b
update todo list
Gavin-Niederman Dec 11, 2023
5dd3021
fix: allow async task spawning in an async task
Gavin-Niederman Dec 12, 2023
f51fbdd
fix(examples): sleeps in accessories and typo in basic
Gavin-Niederman Dec 22, 2023
5b34c62
add: async_runtimecomplete_all function
Gavin-Niederman Dec 22, 2023
591fa84
chore: formatting
Gavin-Niederman Dec 22, 2023
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
7 changes: 7 additions & 0 deletions TODO.md
Original file line number Diff line number Diff line change
Expand Up @@ -32,6 +32,13 @@ This is the todo list for the eventual 1.0.0 release of pros-rs
* [X] Controller data
* [x] Controller printing
* [X] Link
* [ ] Async
* [X] Returning top level futures
* [ ] Future impls for everything
* [ ] Reactor
* [ ] User API (block_on, join!, etc.)
* [ ] MPSC
* [X] Task Locals

## API

Expand Down
43 changes: 42 additions & 1 deletion pros-sys/src/rtos.rs
Original file line number Diff line number Diff line change
Expand Up @@ -25,7 +25,7 @@ pub type task_t = *const core::ffi::c_void;
pub type task_fn_t = Option<unsafe extern "C" fn(arg1: *mut ::core::ffi::c_void)>;
pub type mutex_t = *const core::ffi::c_void;

const CURRENT_TASK: task_t = 0 as task_t;
const CURRENT_TASK: task_t = core::ptr::null();

extern "C" {
/** Gets the number of milliseconds since PROS initialized.
Expand Down Expand Up @@ -280,4 +280,45 @@ extern "C" {
\param mutex
Mutex to unlock.*/
pub fn mutex_delete(mutex: mutex_t);

/** Sets a value in a task's thread local storage array.

This function is intended for advanced users only.

Parameters:
xTaskToSet The handle of the task to which the thread local data is being written. A task can write to its own thread local data by using NULL as the parameter value.
xIndex The index into the thread local storage array to which data is being written.

The number of available array indexes is set by the configNUM_THREAD_LOCAL_STORAGE_POINTERS compile time configuration constant in FreeRTOSConfig.h.
pvValue The value to write into the index specified by the xIndex parameter.

Example usage:

See the examples provided on the thread local storage array documentation page. */
pub fn vTaskSetThreadLocalStoragePointer(
xTaskToSet: task_t,
xIndex: i32,
pvValue: *const core::ffi::c_void,
);

/** Retrieves a value from a task's thread local storage array.

This function is intended for advanced users only.

Parameters:
xTaskToQuery The handle of the task from which the thread local data is being read. A task can read its own thread local data by using NULL as the parameter value.
xIndex The index into the thread local storage array from which data is being read.

The number of available array indexes is set by the configNUM_THREAD_LOCAL_STORAGE_POINTERS compile time configuration constant in FreeRTOSConfig.h.

Returns:
The values stored in index position xIndex of the thread local storage array of task xTaskToQuery.

Example usage:

See the examples provided on the thread local storage array documentation page. */
pub fn pvTaskGetThreadLocalStoragePointer(
xTaskToQuery: task_t,
xIndex: i32,
) -> *const core::ffi::c_void;
}
4 changes: 4 additions & 0 deletions pros/Cargo.toml
Original file line number Diff line number Diff line change
Expand Up @@ -19,6 +19,10 @@ snafu = { version = "0.7.5", default-features = false, features = [
"rust_1_61",
] }
no_std_io = { version = "0.6.0", features = ["alloc"] }
futures = { version = "0.3.28", default-features = false, features = ["alloc"] }
slab = { version = "0.4.9", default-features = false }
hashbrown = { version = "0.14.1", default-features = true }
async-trait = "0.1.73"

[features]
lvgl = ["pros-sys/xapi"]
13 changes: 12 additions & 1 deletion pros/examples/accessories.rs
Original file line number Diff line number Diff line change
Expand Up @@ -6,10 +6,21 @@ use pros::prelude::*;

#[derive(Debug, Default)]
struct ExampleRobot;
#[async_trait]
doinkythederp marked this conversation as resolved.
Show resolved Hide resolved
impl Robot for ExampleRobot {
fn opcontrol(&mut self) -> pros::Result {
async fn opcontrol(&mut self) -> pros::Result {
let handle = pros::async_runtime::spawn(async {
for _ in 0..5 {
println!("Hello from async!");
sleep(Duration::from_millis(1000));
Gavin-Niederman marked this conversation as resolved.
Show resolved Hide resolved
}
});

handle.join();

// Create a new motor plugged into port 2. The motor will brake when not moving.
let motor = Motor::new(2, BrakeMode::Brake)?;
motor.wait_until_stopped().await?;
// Create a controller, specifically controller 1.
let controller = Controller::Master;

Expand Down
121 changes: 121 additions & 0 deletions pros/src/async_runtime/executor.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,121 @@
use core::{
cell::RefCell,
future::Future,
pin::Pin,
task::{Context, Waker},
};

use alloc::{boxed::Box, collections::VecDeque, rc::Rc, task::Wake};

use crate::task_local;

use super::{reactor::Reactor, JoinHandle};

task_local! {
Gavin-Niederman marked this conversation as resolved.
Show resolved Hide resolved
pub(crate) static EXECUTOR: Rc<Executor> = Rc::new(Executor::new())
doinkythederp marked this conversation as resolved.
Show resolved Hide resolved
}

pub struct Task {
future: Pin<Box<dyn Future<Output = ()>>>,
}

impl Task {
pub(crate) fn wrap<F: Future + 'static>(
future: F,
output: Rc<RefCell<Option<F::Output>>>,
) -> Self {
Self {
future: Box::pin({
let output = output.clone();

async move {
let future_output = future.await;
*core::cell::RefCell::<_>::borrow_mut(&output) =
Some(future_output);
}
}),
}
}
}

pub(crate) struct Executor {
queue: RefCell<VecDeque<Task>>,
pub(crate) reactor: Reactor,
}
impl Executor {
pub fn new() -> Self {
Self {
queue: RefCell::new(VecDeque::new()),
reactor: Reactor::new(),
}
}

pub fn spawn<T>(&self, future: impl Future<Output = T> + Send + 'static) -> JoinHandle<T> {
let output = Rc::new(RefCell::new(None));
self.queue
.borrow_mut()
.push_back(Task::wrap(future, output.clone()));

JoinHandle {
output,
_marker: core::marker::PhantomData,
}
}

pub(crate) fn tick(&self) -> bool {
self.reactor.tick();

let mut task = match self.queue.borrow_mut().pop_front() {
Some(task) => task,
None => return false,
};

let task_waker = alloc::sync::Arc::new(TaskWaker {
task: RefCell::new(None),
});
let waker = Waker::from(task_waker.clone());

let cx = &mut Context::from_waker(&waker);
if task.future.as_mut().poll(cx).is_pending() {
task_waker.task.borrow_mut().replace(task);
}

true
}

pub fn block_on<F: Future + 'static>(&self, future: F) -> F::Output {
let output = Rc::new(RefCell::new(None));

self.queue
.borrow_mut()
.push_back(Task::wrap(future, output.clone()));

loop {
self.tick();

if let Some(output) = (*output).borrow_mut().take() {
break output;
}
}
}

pub fn complete(&self) {
Gavin-Niederman marked this conversation as resolved.
Show resolved Hide resolved
while self.tick() {}
}
}

pub struct TaskWaker {
task: RefCell<Option<Task>>,
}
// These are here to apease the waker struct.
// The executor is single threaded and this waker will never be passed around threads or shared between threads.
unsafe impl Send for TaskWaker {}
doinkythederp marked this conversation as resolved.
Show resolved Hide resolved
unsafe impl Sync for TaskWaker {}

impl Wake for TaskWaker {
fn wake(self: alloc::sync::Arc<Self>) {
if let Some(task) = self.task.borrow_mut().take() {
EXECUTOR.with(|e| e.queue.borrow_mut().push_back(task))
}
}
}
51 changes: 51 additions & 0 deletions pros/src/async_runtime/mod.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,51 @@
use core::{cell::RefCell, future::Future};

use alloc::rc::Rc;

pub(crate) mod executor;
pub(crate) mod reactor;

pub struct JoinHandle<T> {
output: Rc<RefCell<Option<T>>>,
_marker: core::marker::PhantomData<T>,
Gavin-Niederman marked this conversation as resolved.
Show resolved Hide resolved
}
impl<T> JoinHandle<T> {
pub fn join(self) -> T {
loop {
if let Some(output) = self.output.borrow_mut().take() {
break output;
}

executor::EXECUTOR.with(|e| (*e).tick());
}
}
}

pub trait FutureExt: Future + 'static + Sized {
fn block_on(self) -> Self::Output {
block_on(self)
}
}
impl<F> FutureExt for F where F: Future + Send + 'static {}

pub fn spawn<T>(future: impl Future<Output = T> + Send + 'static) -> JoinHandle<T> {
executor::EXECUTOR.with(|e| e.spawn(future))
}

pub fn block_on<F: Future + 'static>(future: F) -> F::Output {
Gavin-Niederman marked this conversation as resolved.
Show resolved Hide resolved
executor::EXECUTOR.with(|e| e.block_on(future))
}

pub fn complete() {
executor::EXECUTOR.with(|e| e.complete());
}
Gavin-Niederman marked this conversation as resolved.
Show resolved Hide resolved

#[macro_export]
macro_rules! ready {
($e:expr) => {
match $e {
core::task::Poll::Ready(val) => val,
core::task::Poll::Pending => return core::task::Poll::Pending,
}
};
}
39 changes: 39 additions & 0 deletions pros/src/async_runtime/reactor.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,39 @@
use core::{cell::RefCell, cmp::Reverse, task::Waker};

use alloc::vec::Vec;

pub struct Sleepers {
sleepers: Vec<(Waker, u32)>,
}

impl Sleepers {
pub fn push(&mut self, waker: Waker, target: u32) {
self.sleepers.push((waker, target));

self.sleepers.sort_by_key(|(_, target)| Reverse(*target));
}

pub fn pop(&mut self) -> Option<Waker> {
self.sleepers.pop().map(|(waker, _)| waker)
}
}

pub struct Reactor {
pub(crate) sleepers: RefCell<Sleepers>,
}

impl Reactor {
pub fn new() -> Self {
Self {
sleepers: RefCell::new(Sleepers {
sleepers: Vec::new(),
}),
}
}

pub fn tick(&self) {
if let Some(sleeper) = self.sleepers.borrow_mut().pop() {
sleeper.wake()
}
}
}
Loading