-
Notifications
You must be signed in to change notification settings - Fork 78
Conversation
this adds an entry2 attribute in the vein of cortex_m_rt::entry. This attribute lets you safely use `static mut` variables -- they get transformed into `&'static mut` references. Example: ``` rust #[entry2] fn main() -> ! { static mut COUNT: u32 = 0; // user code let count: &'static mut = COUNT; } ``` I did something different in this version: instead of using random identifiers I wrapped the user entry point into a `const` item to make it impossible to invoke it from software (if that were allowed using the `static mut` would result in UB). Just for reference the above code expands into this: ``` rust const main: () = { #[no_mangle] fn main() -> ! { let COUNT: &'static mut u32 = { static COUNT: u32 = 0; &mut COUNT }; // user code let count: &'static mut = COUNT; } }; ``` Not using random identifiers means that we avoid a (host) dependency on the rand crate. This change is a breaking change because it bumps the Minimum Supported Rust Version (MSRV) to 1.31.0 -- this crate current MSRV is 1.30.0. The MSRV needs to be bumped because `#[no_mangle]` / `#[link_section]` items inside private items (like the `const` item above) only get the right symbol visibility in Rust 1.31.0 (the visibility rules around `#[no_mangle]` / `#[link_section]` got changed in that version). Instead of naming this attribute: `entry2`, we could replace the existing `entry!` macro but that would be another breaking change.
macros/Cargo.toml
Outdated
@@ -0,0 +1,17 @@ | |||
[package] | |||
authors = [ | |||
# "The RISCV Team <[email protected]>", # uncomment when mailing list is up |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
It's up, but the address is risc-v@
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Nice! I'll send a PR to add it the /wg README.
Thanks! |
Argh, I just realized that you can recursively call yourself and that still results in UB (see example below) so we'll need to switch back to random identifiers. #[entry2]
fn main() -> ! {
static mut COUNT: u32 = 0;
// user code
let count: &'static mut = COUNT;
// unsound: creates another mutable reference to COUNT
main();
} |
@japaric What's wrong with your example? Infinite recursion is UB anyway, you can do it even without |
It creates two (or more) mutable references to the
Sure, but infinite recursion is not needed to break Rust aliasing rules in this case. Here's a more obvious example that has no infinite recursion: use core::sync::atomic::{AtomicBool, Ordering};
use spin::Mutex;
static CHANNEL: Mutex<Option<&'static mut u32>> = Mutex::new(None);
static ONCE: AtomicBool = AtomicBool::new(true);
#[entry]
fn main() -> ! {
static mut X: u32 = 0;
if ONCE.load(Ordering::Relaxed) {
// first invocation
ONCE.store(false, Ordering::Relaxed);
*CHANNEL.lock() = Some(X);
main()
} else {
// second invocation
let x: &'static mut u32 = X;
let y: &'static mut u32 = CHANNEL.lock().take().unwrap();
// `x` and `y` point to the same memory location; this is UB
// ..
}
loop {}
}
It depends on the exact semantics. If you allow the Since this is mainly waiting on a decision from the RISCV team on landing this as it is or replacing the existing |
Oh, sorry, I'm not attentive enough. I assumed
However I still can't understand why not using random names is bad: forbidding |
@Disasm maybe you are not familiar with the cortex-m-rt API? Because we use #[entry]
fn main() -> ! {
// main(); // ~ERROR: cannot find function `main` in this scope
loop {}
}
#[exception]
fn SysTick() {
// main(); // ~ERROR: cannot find function `main` in this scope
} As the function can't be (safely) called from software it being non-reentrant (due
I can't see why being reentrant-safe would be hard requirement for multi-core Ultimately, making |
I'm not familiar with Cortex-M so it looks strange for me in general. For example, if exception handlers are reentrant, you will have the same problem... But I don't know if they can be. Yet I understand that random identifiers is some sort of protection for the entry point and handlers.
It's not a hard requirement, but it's an easy way to implement it. At the moment I have no idea how to use non-reentrant code for this. Could you point to some examples of multi-core applications? |
(here I meant to say "not being reentrant safe")
I thought we were only talking about the entry point (reset handler), which only gets called once by the hardware during the lifetime of the program. Making local
Undocumented code / examples in https://github.com/japaric/ultrascale-plus ; |
@japaric Thanks! Now it's clear.
This reminds me one misaligned read exception caused by 2-byte (also misaligned) instruction on the RISC-V. There was an exception handler for this, which should hide this "misaligned access is forbidden" nature, calculating proper values for such accesses. Handler starts processing the instruction, which caused the exception, reads it and we get one more misaligned read exception (now in exception handler code), which is processed in the same way. So yes, if you can process an exception while processing the same exception, using static mut local variables is doubtful. I will try to understand the examples. Thanks again! |
27: Add 'entry' and 'pre_init' attributes r=dvc94ch a=Disasm Implementation is based on [`cortex-m-rt-macros`](https://github.com/rust-embedded/cortex-m-rt/tree/master/macros) code. This implementation has been changed to make `static mut` unsafe inside entry point and different handlers. Related: #20 Co-authored-by: Vadim Kaushan <[email protected]>
27: Add 'entry' and 'pre_init' attributes r=dvc94ch a=Disasm Implementation is based on [`cortex-m-rt-macros`](https://github.com/rust-embedded/cortex-m-rt/tree/master/macros) code. This implementation has been changed to make `static mut` unsafe inside entry point and different handlers. Related: rust-embedded/riscv-rt#20 Co-authored-by: Vadim Kaushan <[email protected]>
this adds an entry2 attribute in the vein of cortex_m_rt::entry. This attribute
lets you safely use
static mut
variables -- they get transformed into&'static mut
references. Example:I did something different in this version: instead of using random identifiers I
wrapped the user entry point into a
const
item to make it impossible to invokeit from software (if that were allowed using the
static mut
would result inUB).
Just for reference the above code expands into this:
Not using random identifiers means that we avoid a (host) dependency on the rand
crate.
This change is a breaking change because it bumps the Minimum Supported Rust
Version (MSRV) to 1.31.0 -- this crate current MSRV is 1.30.0. The MSRV needs to
be bumped because
#[no_mangle]
/#[link_section]
items inside privateitems (like the
const
item above) only get the right symbol visibility in Rust1.31.0 (the visibility rules around
#[no_mangle]
/#[link_section]
gotchanged in that version).
Instead of naming this attribute:
entry2
, we could replace the existingentry!
macro but that would be another breaking change.I didn't add the interrupt / exception attributes because I couldn't find an
example that makes use of interrupts / exceptions