2019-06-13 23:56:59 +02:00
|
|
|
use core::ops;
|
|
|
|
use std::collections::{BTreeMap, BTreeSet};
|
2017-07-15 01:54:54 +02:00
|
|
|
|
2019-06-13 23:56:59 +02:00
|
|
|
use rtfm_syntax::{
|
|
|
|
analyze::{self, Priority},
|
|
|
|
ast::App,
|
|
|
|
Core, P,
|
|
|
|
};
|
|
|
|
use syn::Ident;
|
2017-07-15 01:54:54 +02:00
|
|
|
|
2019-06-13 23:56:59 +02:00
|
|
|
/// Extend the upstream `Analysis` struct with our field
|
2018-11-03 17:02:41 +01:00
|
|
|
pub struct Analysis {
|
2019-06-13 23:56:59 +02:00
|
|
|
parent: P<analyze::Analysis>,
|
|
|
|
pub interrupts: BTreeMap<Core, BTreeMap<Priority, Ident>>,
|
2018-11-03 17:02:41 +01:00
|
|
|
}
|
|
|
|
|
2019-06-13 23:56:59 +02:00
|
|
|
impl ops::Deref for Analysis {
|
|
|
|
type Target = analyze::Analysis;
|
2018-11-04 18:50:42 +01:00
|
|
|
|
2019-06-13 23:56:59 +02:00
|
|
|
fn deref(&self) -> &Self::Target {
|
|
|
|
&self.parent
|
2018-11-04 18:50:42 +01:00
|
|
|
}
|
2017-07-15 01:54:54 +02:00
|
|
|
}
|
|
|
|
|
2019-06-13 23:56:59 +02:00
|
|
|
// Assign an `extern` interrupt to each priority level
|
|
|
|
pub fn app(analysis: P<analyze::Analysis>, app: &App) -> P<Analysis> {
|
|
|
|
let mut interrupts = BTreeMap::new();
|
|
|
|
for core in 0..app.args.cores {
|
|
|
|
let priorities = app
|
|
|
|
.software_tasks
|
|
|
|
.values()
|
|
|
|
.filter_map(|task| {
|
|
|
|
if task.args.core == core {
|
|
|
|
Some(task.args.priority)
|
|
|
|
} else {
|
|
|
|
None
|
2017-07-15 01:54:54 +02:00
|
|
|
}
|
2019-06-13 23:56:59 +02:00
|
|
|
})
|
|
|
|
.chain(analysis.timer_queues.get(&core).map(|tq| tq.priority))
|
|
|
|
.collect::<BTreeSet<_>>();
|
|
|
|
|
|
|
|
if !priorities.is_empty() {
|
|
|
|
interrupts.insert(
|
|
|
|
core,
|
|
|
|
priorities
|
|
|
|
.iter()
|
|
|
|
.cloned()
|
|
|
|
.rev()
|
|
|
|
.zip(app.extern_interrupts[&core].keys().cloned())
|
|
|
|
.collect(),
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
P::new(Analysis {
|
|
|
|
parent: analysis,
|
|
|
|
interrupts,
|
|
|
|
})
|
2017-07-15 01:54:54 +02:00
|
|
|
}
|