-
Notifications
You must be signed in to change notification settings - Fork 4
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
- Loading branch information
0 parents
commit 836ca17
Showing
17 changed files
with
645 additions
and
0 deletions.
There are no files selected for viewing
Empty file.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,33 @@ | ||
[target.thumbv7m-none-eabi] | ||
# uncomment this to make `cargo run` execute programs on QEMU | ||
# runner = "qemu-system-arm -cpu cortex-m3 -machine lm3s6965evb -nographic -semihosting-config enable=on,target=native -kernel" | ||
|
||
[target.'cfg(all(target_arch = "arm", target_os = "none"))'] | ||
# uncomment ONE of these three option to make `cargo run` start a GDB session | ||
# which option to pick depends on your system | ||
runner = "arm-none-eabi-gdb -q -x openocd.gdb" | ||
# runner = "gdb-multiarch -q -x openocd.gdb" | ||
# runner = "gdb -q -x openocd.gdb" | ||
|
||
rustflags = [ | ||
# LLD (shipped with the Rust toolchain) is used as the default linker | ||
"-C", "link-arg=-Tlink.x", | ||
|
||
# if you run into problems with LLD switch to the GNU linker by commenting out | ||
# this line | ||
# "-C", "linker=arm-none-eabi-ld", | ||
|
||
# if you need to link to pre-compiled C libraries provided by a C toolchain | ||
# use GCC as the linker by commenting out both lines above and then | ||
# uncommenting the three lines below | ||
# "-C", "linker=arm-none-eabi-gcc", | ||
# "-C", "link-arg=-Wl,-Tlink.x", | ||
# "-C", "link-arg=-nostartfiles", | ||
] | ||
|
||
[build] | ||
# Pick ONE of these compilation targets | ||
# target = "thumbv6m-none-eabi" # Cortex-M0 and Cortex-M0+ | ||
# target = "thumbv7m-none-eabi" # Cortex-M3 | ||
# target = "thumbv7em-none-eabi" # Cortex-M4 and Cortex-M7 (no FPU) | ||
target = "thumbv7em-none-eabihf" # Cortex-M4F and Cortex-M7F (with FPU) |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,5 @@ | ||
**/*.rs.bk | ||
.#* | ||
.gdb_history | ||
Cargo.lock | ||
target/ |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,29 @@ | ||
[package] | ||
authors = ["Raphael Nestler <[email protected]>"] | ||
edition = "2018" | ||
readme = "README.md" | ||
name = "hello-rtfm" | ||
version = "0.1.0" | ||
|
||
[dependencies] | ||
cortex-m = "0.5.8" | ||
cortex-m-rt = "0.6.5" | ||
cortex-m-semihosting = "0.3.2" | ||
panic-semihosting = "0.5.1" | ||
cortex-m-rtfm = {git="https://github.com/japaric/cortex-m-rtfm.git"} | ||
|
||
|
||
[dependencies.stm32f30x] | ||
version = "0.8.0" | ||
features= ["rt"] | ||
|
||
# this lets you use `cargo fix`! | ||
[[bin]] | ||
name = "hello-rtfm" | ||
test = false | ||
bench = false | ||
|
||
[profile.release] | ||
codegen-units = 1 # better optimizations | ||
debug = true # symbols are nice and they don't increase the size on Flash | ||
lto = true # better optimizations |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,129 @@ | ||
# `cortex-m-quickstart` | ||
|
||
> A template for building applications for ARM Cortex-M microcontrollers | ||
This project is developed and maintained by the [Cortex-M team][team]. | ||
|
||
## Dependencies | ||
|
||
To build embedded programs using this template you'll need: | ||
|
||
- Rust 1.31, 1.30-beta, nightly-2018-09-13 or a newer toolchain. e.g. `rustup | ||
default beta` | ||
|
||
- The `cargo generate` subcommand. [Installation | ||
instructions](https://github.com/ashleygwilliams/cargo-generate#installation). | ||
|
||
- `rust-std` components (pre-compiled `core` crate) for the ARM Cortex-M | ||
targets. Run: | ||
|
||
``` console | ||
$ rustup target add thumbv6m-none-eabi thumbv7m-none-eabi thumbv7em-none-eabi thumbv7em-none-eabihf | ||
``` | ||
|
||
## Using this template | ||
|
||
**NOTE**: This is the very short version that only covers building programs. For | ||
the long version, which additionally covers flashing, running and debugging | ||
programs, check [the embedded Rust book][book]. | ||
|
||
[book]: https://rust-embedded.github.io/book | ||
|
||
0. Before we begin you need to identify some characteristics of the target | ||
device as these will be used to configure the project: | ||
|
||
- The ARM core. e.g. Cortex-M3. | ||
|
||
- Does the ARM core include an FPU? Cortex-M4**F** and Cortex-M7**F** cores do. | ||
|
||
- How much Flash memory and RAM does the target device has? e.g. 256 KiB of | ||
Flash and 32 KiB of RAM. | ||
|
||
- Where are Flash memory and RAM mapped in the address space? e.g. RAM is | ||
commonly located at address `0x2000_0000`. | ||
|
||
You can find this information in the data sheet or the reference manual of your | ||
device. | ||
|
||
In this example we'll be using the STM32F3DISCOVERY. This board contains an | ||
STM32F303VCT6 microcontroller. This microcontroller has: | ||
|
||
- A Cortex-M4F core that includes a single precision FPU | ||
|
||
- 256 KiB of Flash located at address 0x0800_0000. | ||
|
||
- 40 KiB of RAM located at address 0x2000_0000. (There's another RAM region but | ||
for simplicity we'll ignore it). | ||
|
||
1. Instantiate the template. | ||
|
||
``` console | ||
$ cargo generate --git https://github.com/rust-embedded/cortex-m-quickstart | ||
Project Name: app | ||
Creating project called `app`... | ||
Done! New project created /tmp/app | ||
|
||
$ cd app | ||
``` | ||
|
||
2. Set a default compilation target. There are four options as mentioned at the | ||
bottom of `.cargo/config`. For the STM32F303VCT6, which has a Cortex-M4F | ||
core, we'll pick the `thumbv7em-none-eabihf` target. | ||
|
||
``` console | ||
$ tail -n6 .cargo/config | ||
``` | ||
|
||
``` toml | ||
[build] | ||
# Pick ONE of these compilation targets | ||
# target = "thumbv6m-none-eabi" # Cortex-M0 and Cortex-M0+ | ||
# target = "thumbv7m-none-eabi" # Cortex-M3 | ||
# target = "thumbv7em-none-eabi" # Cortex-M4 and Cortex-M7 (no FPU) | ||
target = "thumbv7em-none-eabihf" # Cortex-M4F and Cortex-M7F (with FPU) | ||
``` | ||
|
||
3. Enter the memory region information into the `memory.x` file. | ||
|
||
``` console | ||
$ cat memory.x | ||
/* Linker script for the STM32F303VCT6 */ | ||
MEMORY | ||
{ | ||
/* NOTE 1 K = 1 KiBi = 1024 bytes */ | ||
FLASH : ORIGIN = 0x08000000, LENGTH = 256K | ||
RAM : ORIGIN = 0x20000000, LENGTH = 40K | ||
} | ||
``` | ||
|
||
4. Build the template application or one of the examples. | ||
|
||
``` console | ||
$ cargo build | ||
``` | ||
|
||
# License | ||
|
||
This template is licensed under either of | ||
|
||
- Apache License, Version 2.0 ([LICENSE-APACHE](LICENSE-APACHE) or | ||
http://www.apache.org/licenses/LICENSE-2.0) | ||
|
||
- MIT license ([LICENSE-MIT](LICENSE-MIT) or http://opensource.org/licenses/MIT) | ||
|
||
at your option. | ||
|
||
## Contribution | ||
|
||
Unless you explicitly state otherwise, any contribution intentionally submitted | ||
for inclusion in the work by you, as defined in the Apache-2.0 license, shall be | ||
dual licensed as above, without any additional terms or conditions. | ||
|
||
## Code of Conduct | ||
|
||
Contribution to this crate is organized under the terms of the [Rust Code of | ||
Conduct][CoC], the maintainer of this crate, the [Cortex-M team][team], promises | ||
to intervene to uphold that code of conduct. | ||
|
||
[CoC]: CODE_OF_CONDUCT.md | ||
[team]: https://github.com/rust-embedded/wg#the-cortex-m-team |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,18 @@ | ||
use std::env; | ||
use std::fs::File; | ||
use std::io::Write; | ||
use std::path::PathBuf; | ||
|
||
fn main() { | ||
// Put the linker script somewhere the linker can find it | ||
let out = &PathBuf::from(env::var_os("OUT_DIR").unwrap()); | ||
File::create(out.join("memory.x")) | ||
.unwrap() | ||
.write_all(include_bytes!("memory.x")) | ||
.unwrap(); | ||
println!("cargo:rustc-link-search={}", out.display()); | ||
|
||
// Only re-run the build script when memory.x is changed, | ||
// instead of when any part of the source code changes. | ||
println!("cargo:rerun-if-changed=memory.x"); | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,57 @@ | ||
//! How to use the heap and a dynamic memory allocator | ||
//! | ||
//! This example depends on the alloc-cortex-m crate so you'll have to add it to your Cargo.toml: | ||
//! | ||
//! ``` text | ||
//! # or edit the Cargo.toml file manually | ||
//! $ cargo add alloc-cortex-m | ||
//! ``` | ||
//! | ||
//! --- | ||
#![feature(alloc)] | ||
#![feature(alloc_error_handler)] | ||
#![no_main] | ||
#![no_std] | ||
|
||
extern crate alloc; | ||
extern crate panic_halt; | ||
|
||
use self::alloc::vec; | ||
use core::alloc::Layout; | ||
|
||
use alloc_cortex_m::CortexMHeap; | ||
use cortex_m::asm; | ||
use cortex_m_rt::entry; | ||
use cortex_m_semihosting::hprintln; | ||
|
||
// this is the allocator the application will use | ||
#[global_allocator] | ||
static ALLOCATOR: CortexMHeap = CortexMHeap::empty(); | ||
|
||
const HEAP_SIZE: usize = 1024; // in bytes | ||
|
||
#[entry] | ||
fn main() -> ! { | ||
// Initialize the allocator BEFORE you use it | ||
unsafe { ALLOCATOR.init(cortex_m_rt::heap_start() as usize, HEAP_SIZE) } | ||
|
||
// Growable array allocated on the heap | ||
let xs = vec![0, 1, 2]; | ||
|
||
hprintln!("{:?}", xs).unwrap(); | ||
|
||
// exit QEMU | ||
// NOTE do not run this on hardware; it can corrupt OpenOCD state | ||
debug::exit(debug::EXIT_SUCCESS); | ||
|
||
loop {} | ||
} | ||
|
||
// define what happens in an Out Of Memory (OOM) condition | ||
#[alloc_error_handler] | ||
fn alloc_error(_layout: Layout) -> ! { | ||
asm::bkpt(); | ||
|
||
loop {} | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,96 @@ | ||
//! Debugging a crash (exception) | ||
//! | ||
//! Most crash conditions trigger a hard fault exception, whose handler is defined via | ||
//! `exception!(HardFault, ..)`. The `HardFault` handler has access to the exception frame, a | ||
//! snapshot of the CPU registers at the moment of the exception. | ||
//! | ||
//! This program crashes and the `HardFault` handler prints to the console the contents of the | ||
//! `ExceptionFrame` and then triggers a breakpoint. From that breakpoint one can see the backtrace | ||
//! that led to the exception. | ||
//! | ||
//! ``` text | ||
//! (gdb) continue | ||
//! Program received signal SIGTRAP, Trace/breakpoint trap. | ||
//! __bkpt () at asm/bkpt.s:3 | ||
//! 3 bkpt | ||
//! | ||
//! (gdb) backtrace | ||
//! #0 __bkpt () at asm/bkpt.s:3 | ||
//! #1 0x080030b4 in cortex_m::asm::bkpt () at $$/cortex-m-0.5.0/src/asm.rs:19 | ||
//! #2 rust_begin_unwind (args=..., file=..., line=99, col=5) at $$/panic-semihosting-0.2.0/src/lib.rs:87 | ||
//! #3 0x08001d06 in core::panicking::panic_fmt () at libcore/panicking.rs:71 | ||
//! #4 0x080004a6 in crash::hard_fault (ef=0x20004fa0) at examples/crash.rs:99 | ||
//! #5 0x08000548 in UserHardFault (ef=0x20004fa0) at <exception macros>:10 | ||
//! #6 0x0800093a in HardFault () at asm.s:5 | ||
//! Backtrace stopped: previous frame identical to this frame (corrupt stack?) | ||
//! ``` | ||
//! | ||
//! In the console output one will find the state of the Program Counter (PC) register at the time | ||
//! of the exception. | ||
//! | ||
//! ``` text | ||
//! panicked at 'HardFault at ExceptionFrame { | ||
//! r0: 0x2fffffff, | ||
//! r1: 0x2fffffff, | ||
//! r2: 0x080051d4, | ||
//! r3: 0x080051d4, | ||
//! r12: 0x20000000, | ||
//! lr: 0x08000435, | ||
//! pc: 0x08000ab6, | ||
//! xpsr: 0x61000000 | ||
//! }', examples/crash.rs:106:5 | ||
//! ``` | ||
//! | ||
//! This register contains the address of the instruction that caused the exception. In GDB one can | ||
//! disassemble the program around this address to observe the instruction that caused the | ||
//! exception. | ||
//! | ||
//! ``` text | ||
//! (gdb) disassemble/m 0x08000ab6 | ||
//! Dump of assembler code for function core::ptr::read_volatile: | ||
//! 451 pub unsafe fn read_volatile<T>(src: *const T) -> T { | ||
//! 0x08000aae <+0>: sub sp, #16 | ||
//! 0x08000ab0 <+2>: mov r1, r0 | ||
//! 0x08000ab2 <+4>: str r0, [sp, #8] | ||
//! | ||
//! 452 intrinsics::volatile_load(src) | ||
//! 0x08000ab4 <+6>: ldr r0, [sp, #8] | ||
//! -> 0x08000ab6 <+8>: ldr r0, [r0, #0] | ||
//! 0x08000ab8 <+10>: str r0, [sp, #12] | ||
//! 0x08000aba <+12>: ldr r0, [sp, #12] | ||
//! 0x08000abc <+14>: str r1, [sp, #4] | ||
//! 0x08000abe <+16>: str r0, [sp, #0] | ||
//! 0x08000ac0 <+18>: b.n 0x8000ac2 <core::ptr::read_volatile+20> | ||
//! | ||
//! 453 } | ||
//! 0x08000ac2 <+20>: ldr r0, [sp, #0] | ||
//! 0x08000ac4 <+22>: add sp, #16 | ||
//! 0x08000ac6 <+24>: bx lr | ||
//! | ||
//! End of assembler dump. | ||
//! ``` | ||
//! | ||
//! `ldr r0, [r0, #0]` caused the exception. This instruction tried to load (read) a 32-bit word | ||
//! from the address stored in the register `r0`. Looking again at the contents of `ExceptionFrame` | ||
//! we see that the `r0` contained the address `0x2FFF_FFFF` when this instruction was executed. | ||
//! | ||
//! --- | ||
#![no_main] | ||
#![no_std] | ||
|
||
extern crate panic_halt; | ||
|
||
use core::ptr; | ||
|
||
use cortex_m_rt::entry; | ||
|
||
#[entry] | ||
fn main() -> ! { | ||
unsafe { | ||
// read an address outside of the RAM region; this causes a HardFault exception | ||
ptr::read_volatile(0x2FFF_FFFF as *const u32); | ||
} | ||
|
||
loop {} | ||
} |
Oops, something went wrong.