mirror of
https://github.com/rtic-rs/rtic.git
synced 2024-11-29 15:04:32 +01:00
safe &'static mut
references via init.resources
This commit is contained in:
parent
a6dd004113
commit
d30bdcb096
6 changed files with 153 additions and 10 deletions
32
examples/safe-static-mut-ref.rs
Normal file
32
examples/safe-static-mut-ref.rs
Normal file
|
@ -0,0 +1,32 @@
|
||||||
|
//! Safe creation of `&'static mut` references
|
||||||
|
#![deny(unsafe_code)]
|
||||||
|
#![deny(warnings)]
|
||||||
|
#![feature(proc_macro)]
|
||||||
|
#![no_std]
|
||||||
|
|
||||||
|
extern crate cortex_m_rtfm as rtfm;
|
||||||
|
extern crate stm32f103xx;
|
||||||
|
|
||||||
|
use rtfm::app;
|
||||||
|
|
||||||
|
app! {
|
||||||
|
device: stm32f103xx,
|
||||||
|
|
||||||
|
resources: {
|
||||||
|
static BUFFER: [u8; 16] = [0; 16];
|
||||||
|
},
|
||||||
|
|
||||||
|
init: {
|
||||||
|
resources: [BUFFER],
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
fn init(_p: init::Peripherals, r: init::Resources) {
|
||||||
|
let _buf: &'static mut [u8] = r.BUFFER;
|
||||||
|
}
|
||||||
|
|
||||||
|
fn idle() -> ! {
|
||||||
|
loop {
|
||||||
|
rtfm::wfi();
|
||||||
|
}
|
||||||
|
}
|
|
@ -12,7 +12,8 @@ version = "0.2.1"
|
||||||
[dependencies]
|
[dependencies]
|
||||||
error-chain = "0.10.0"
|
error-chain = "0.10.0"
|
||||||
quote = "0.3.15"
|
quote = "0.3.15"
|
||||||
rtfm-syntax = "0.2.0"
|
# rtfm-syntax = "0.2.0"
|
||||||
|
rtfm-syntax = { git = "https://github.com/japaric/rtfm-syntax", branch = "init-resources" }
|
||||||
syn = "0.11.11"
|
syn = "0.11.11"
|
||||||
|
|
||||||
[lib]
|
[lib]
|
||||||
|
|
|
@ -77,7 +77,38 @@ pub fn app(app: check::App) -> Result<App> {
|
||||||
}
|
}
|
||||||
|
|
||||||
fn resources(app: &App) -> Result<()> {
|
fn resources(app: &App) -> Result<()> {
|
||||||
|
for name in &app.init.resources {
|
||||||
|
if let Some(resource) = app.resources.get(name) {
|
||||||
|
ensure!(
|
||||||
|
resource.expr.is_some(),
|
||||||
|
"resource `{}`, allocated to `init`, must have an initial value",
|
||||||
|
name
|
||||||
|
);
|
||||||
|
} else {
|
||||||
|
bail!(
|
||||||
|
"resource `{}`, allocated to `init`, must be a data resource",
|
||||||
|
name
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
ensure!(
|
||||||
|
!app.idle.resources.contains(name),
|
||||||
|
"resources assigned to `init` can't be shared with `idle`"
|
||||||
|
);
|
||||||
|
|
||||||
|
ensure!(
|
||||||
|
app.tasks
|
||||||
|
.iter()
|
||||||
|
.all(|(_, task)| !task.resources.contains(name)),
|
||||||
|
"resources assigned to `init` can't be shared with tasks"
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
for resource in app.resources.keys() {
|
for resource in app.resources.keys() {
|
||||||
|
if app.init.resources.contains(resource) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
if app.idle.resources.contains(resource) {
|
if app.idle.resources.contains(resource) {
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
|
@ -249,11 +249,22 @@ fn init(app: &App, main: &mut Vec<Tokens>, root: &mut Vec<Tokens>) {
|
||||||
let mut rexprs = vec![];
|
let mut rexprs = vec![];
|
||||||
|
|
||||||
for (name, resource) in init_resources {
|
for (name, resource) in init_resources {
|
||||||
|
let ty = &resource.ty;
|
||||||
|
|
||||||
|
if app.init.resources.contains(name) {
|
||||||
|
fields.push(quote! {
|
||||||
|
pub #name: &'static mut #ty,
|
||||||
|
});
|
||||||
|
|
||||||
|
let expr = &resource.expr;
|
||||||
|
rexprs.push(quote!(#name: {
|
||||||
|
static mut #name: #ty = #expr;
|
||||||
|
&mut #name
|
||||||
|
}));
|
||||||
|
} else {
|
||||||
let _name = Ident::new(format!("_{}", name.as_ref()));
|
let _name = Ident::new(format!("_{}", name.as_ref()));
|
||||||
lifetime = Some(quote!('a));
|
lifetime = Some(quote!('a));
|
||||||
|
|
||||||
let ty = &resource.ty;
|
|
||||||
|
|
||||||
fields.push(quote! {
|
fields.push(quote! {
|
||||||
pub #name: &'a mut #ty,
|
pub #name: &'a mut #ty,
|
||||||
});
|
});
|
||||||
|
@ -262,6 +273,7 @@ fn init(app: &App, main: &mut Vec<Tokens>, root: &mut Vec<Tokens>) {
|
||||||
#name: &mut ::#_name,
|
#name: &mut ::#_name,
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
root.push(quote! {
|
root.push(quote! {
|
||||||
#[allow(non_camel_case_types)]
|
#[allow(non_camel_case_types)]
|
||||||
|
|
31
tests/cfail/init-resource-share-idle.rs
Normal file
31
tests/cfail/init-resource-share-idle.rs
Normal file
|
@ -0,0 +1,31 @@
|
||||||
|
#![deny(warnings)]
|
||||||
|
#![feature(proc_macro)]
|
||||||
|
#![no_std]
|
||||||
|
|
||||||
|
extern crate cortex_m_rtfm as rtfm;
|
||||||
|
extern crate stm32f103xx;
|
||||||
|
|
||||||
|
use rtfm::app;
|
||||||
|
|
||||||
|
app! { //~ proc macro panicked
|
||||||
|
device: stm32f103xx,
|
||||||
|
|
||||||
|
resources: {
|
||||||
|
static BUFFER: [u8; 16] = [0; 16];
|
||||||
|
},
|
||||||
|
|
||||||
|
init: {
|
||||||
|
resources: [BUFFER],
|
||||||
|
},
|
||||||
|
|
||||||
|
idle: {
|
||||||
|
// ERROR resources assigned to `init` can't be shared with `idle`
|
||||||
|
resources: [BUFFER],
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
fn init(_p: init::Peripherals, _r: init::Resources) {}
|
||||||
|
|
||||||
|
fn idle(_r: init::Resources) -> ! {
|
||||||
|
loop {}
|
||||||
|
}
|
36
tests/cfail/init-resource-share-task.rs
Normal file
36
tests/cfail/init-resource-share-task.rs
Normal file
|
@ -0,0 +1,36 @@
|
||||||
|
#![deny(warnings)]
|
||||||
|
#![feature(proc_macro)]
|
||||||
|
#![no_std]
|
||||||
|
|
||||||
|
extern crate cortex_m_rtfm as rtfm;
|
||||||
|
extern crate stm32f103xx;
|
||||||
|
|
||||||
|
use rtfm::app;
|
||||||
|
|
||||||
|
app! { //~ proc macro panicked
|
||||||
|
device: stm32f103xx,
|
||||||
|
|
||||||
|
resources: {
|
||||||
|
static BUFFER: [u8; 16] = [0; 16];
|
||||||
|
},
|
||||||
|
|
||||||
|
init: {
|
||||||
|
resources: [BUFFER],
|
||||||
|
},
|
||||||
|
|
||||||
|
tasks: {
|
||||||
|
SYS_TICK: {
|
||||||
|
path: sys_tick,
|
||||||
|
// ERROR resources assigned to `init` can't be shared with tasks
|
||||||
|
resources: [BUFFER],
|
||||||
|
},
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
fn init(_p: init::Peripherals) {}
|
||||||
|
|
||||||
|
fn idle() -> ! {
|
||||||
|
loop {}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn sys_tick() {}
|
Loading…
Reference in a new issue