rtic/rtic-macros/src/analyze.rs
Henri Lunnikivi 9bfd5a3bb2
Improve error output for prios > dispatchers (#943)
* Improve error output for prios > dispatchers

* Update changelog
2024-05-28 06:01:53 +00:00

72 lines
1.9 KiB
Rust

use core::ops;
use std::collections::{BTreeMap, BTreeSet};
use crate::syntax::{
analyze::{self, Priority},
ast::{App, Dispatcher},
};
use syn::Ident;
/// Extend the upstream `Analysis` struct with our field
pub struct Analysis {
parent: analyze::Analysis,
pub interrupts: BTreeMap<Priority, (Ident, Dispatcher)>,
pub max_async_prio: Option<u8>,
}
impl ops::Deref for Analysis {
type Target = analyze::Analysis;
fn deref(&self) -> &Self::Target {
&self.parent
}
}
// Assign an interrupt to each priority level
pub fn app(analysis: analyze::Analysis, app: &App) -> Analysis {
let mut available_dispatchers = app.args.dispatchers.clone();
// the set of priorities (each priority only once)
let priorities = app
.software_tasks
.values()
.map(|task| task.args.priority)
.collect::<BTreeSet<_>>();
// map from priorities to interrupts (holding name and attributes)
let nonzero_priorities = priorities
.iter()
// 0 prio tasks are run in main
.filter(|prio| **prio > 0);
assert!(
available_dispatchers.len() >= nonzero_priorities.clone().count(),
"The number of dispatchers must be equal to or greater than the number of distinct task priorities."
);
let interrupts: BTreeMap<Priority, _> = nonzero_priorities
.copied()
.rev()
.map(|p| {
(
p,
available_dispatchers
.pop()
// EXPECT: covered by above assertion
.expect("UNREACHABLE"),
)
})
.collect();
let max_async_prio = app
.hardware_tasks
.iter()
.map(|(_, task)| task.args.priority)
.min()
.map(|v| v - 1); // One less than the smallest HW task
Analysis {
parent: analysis,
interrupts,
max_async_prio,
}
}