Bare-Metal Firmware on Cortex-R52 - Writing a UART Driver
We have supplied a small Rust no-std application, which is designed to run
inside a QEMU emulation of an Armv8-R Cortex-R52 system. We build the code using
the armv8r-none-eabihf
target.
The application lives in
./qemu-code/uart-driver
.
The application talks to the outside world through a UART driver. We have provided two - a working one, and a template one that doesn't work which you need to fix.
Task 1 - Get UART TX working
Modify the template driver and complete the missing code sections as commented. You can peek at the complete driver if you really need to!
This will involve reading and writing to the given registers. You have been
given an MMIO handle for the UART peripheral as self.registers
. This is of
type MmioUart
, which was automatically generated using the derive_mmio::Mmio
derive-macro based on the register definition at the bottom of the file.
You'll want to read the derive-mmio
documentation, or run cargo doc --open
on your
project to see the API available.
Task 2 - Get UART RX working (Optional)
Continue modifying the UART driver so that you can read data. You'll need to
enable RX in the configuration register, and add an appropriate method
to read a single byte, returning Option<u8>
. Now modify the main loop to
echo back received characters.
You can look in the CMSDK UART documentation to see which bit in the status
register indicates that the 1-byte long RX FIFO has data in it.
Task 3 - Make the UART a global
Creating a UART on the stack is fine, but we would now like to use our UART from anywhere in our program.
To do that, we need to do a few things:
Step 1
Create a static variable, like static UART: T = something()
. Work out how to initialise that variable.
Answer
static UART: Uart = unsafe { Uart::new_uart0() };
Step 2
Use the critical_section::Mutex
type to solve the Sync
error that appears.
Answer
static UART: critical_section::Mutex<Uart> = critical_section::Mutex::new(unsafe { Uart::new_uart0() });
// Now every time you touch the UART you must lock the Mutex first
critical_section::with(|cs| {
UART.borrow(cs).enable(115200, PERIPHERAL_CLOCK);
});
Step 3
That's not enough. You only have &self
to the Uart
and you need &mut self
. Try using a RefCell
!
Answer
use core::cell::RefCell;
static UART: critical_section::Mutex<RefCell<Uart>> = critical_section::Mutex::new(RefCell::new(unsafe { Uart::new_uart0() }));
// Now every time you touch the UART you must lock the Mutex and borrow the RefCell first
critical_section::with(|cs| {
UART.borrow_ref_mut(cs).enable(115200, PERIPHERAL_CLOCK);
});
critical_section::with(|cs| {
let mut uart = UART.borrow_ref_mut(cs);
_ = writeln!(uart, "Hello, this is Rust!");
});
Step 4
That's a lot of locking. Make a wrapper type called GlobalUart
which hides this mess. Implement core::fmt::Write
for it so can just writeln!(&UART, "Hello")
.
Answer
struct GlobalUart {
inner: critical_section::Mutex<RefCell<Uart>>
}
impl GlobalUart {
const fn new() -> GlobalUart {
GlobalUart { inner: critical_section::Mutex::new(RefCell::new(unsafe { Uart::new_uart0() })) }
}
fn enable(&self, baudrate: u32, periph_clk: u32) {
critical_section::with(|cs| {
let mut uart = self.inner.borrow_ref_mut(cs);
uart.enable(baudrate, periph_clk);
});
}
}
static UART: GlobalUart = GlobalUart::new();
// Note that we are implementing the trait for reference-to-GlobalUart because
// we don't have mutable access to our static variable.
impl core::fmt::Write for &GlobalUart {
fn write_str(&mut self, s: &str) -> core::fmt::Result {
for b in s.bytes() {
critical_section::with(|cs| {
let mut uart = self.inner.borrow_ref_mut(cs);
uart.write(b);
});
}
Ok(())
}
}
// Note we are writing into a reference-to-GlobalUart
writeln!(&UART, "Hello, this is Rust!")?;
Running the code
You will need QEMU 9 installed and in your $PATH
for cargo run
to work. This
was the first version with Arm Cortex-R52 emulation.
Running the project gives:
$ cargo run
Compiling uart-exercise v0.1.0 (/Users/jonathan/Documents/ferrous-systems/rust-exercises/qemu-code/uart-driver)
warning: field `registers` is never read
--> src/uart_driver.rs:9:5
|
8 | pub struct Uart {
| ---- field in this struct
9 | registers: MmioRegisters,
| ^^^^^^^^^
|
= note: `#[warn(dead_code)]` on by default
warning: `uart-exercise` (lib) generated 1 warning
Finished `dev` profile [unoptimized + debuginfo] target(s) in 0.87s
Running `qemu-system-arm -machine mps3-an536 -cpu cortex-r52 -semihosting -nographic -kernel target/armv8r-none-eabihf/debug/uart-exercise`
PANIC: PanicInfo { message: I am a panic, location: Location { file: "src/main.rs", line: 45, col: 5 }, can_unwind: true, force_no_backtrace: false }
No UART output (the panic comes on semihosting). But also no-one is using that registers
field. You should fix that.
With the Task 1 completed (or using the solution file) you will get:
$ cargo run
Compiling uart-exercise v0.1.0 (/Users/jonathan/Documents/ferrous-systems/rust-exercises/qemu-code/uart-driver)
Finished `dev` profile [unoptimized + debuginfo] target(s) in 0.14s
Running `qemu-system-arm -machine mps3-an536 -cpu cortex-r52 -semihosting -nographic -kernel target/armv8r-none-eabihf/debug/uart-exercise`
Hello, this is Rust!
1.00 2.00 3.00 4.00 5.00 6.00 7.00 8.00 9.00 10.00
2.00 4.00 6.00 8.00 10.00 12.00 14.00 16.00 18.00 20.00
3.00 6.00 9.00 12.00 15.00 18.00 21.00 24.00 27.00 30.00
4.00 8.00 12.00 16.00 20.00 24.00 28.00 32.00 36.00 40.00
5.00 10.00 15.00 20.00 25.00 30.00 35.00 40.00 45.00 50.00
6.00 12.00 18.00 24.00 30.00 36.00 42.00 48.00 54.00 60.00
7.00 14.00 21.00 28.00 35.00 42.00 49.00 56.00 63.00 70.00
8.00 16.00 24.00 32.00 40.00 48.00 56.00 64.00 72.00 80.00
9.00 18.00 27.00 36.00 45.00 54.00 63.00 72.00 81.00 90.00
10.00 20.00 30.00 40.00 50.00 60.00 70.00 80.00 90.00 100.00
PANIC: PanicInfo { payload: Any { .. }, message: Some(I am a panic), location: Location { file: "src/main.rs", line: 43, col: 5 }, can_unwind: true, force_no_backtrace: false }