PACs and svd2rust

Introduction

The Peripheral Access Crate crate sits near the bottom of the 'stack'. It provides access to the memory-mapped peripherals in your MCU.

Memory Mapped Peripherals

  • e.g. a UART peripheral
  • Has registers, represented by a memory address
  • Registers are usually consecutive in memory (not always)
  • Peripherals can have instances (same layout of registers, different start address)
    • UART0, UART1, etc

Note:

The Universal Asynchronous Receiver Transmitter is an IP block implementing a logic-level RS-232 interface, and one is fitted to basically every microcontroller. Also known as a serial port.

Nordic calls their peripheral UARTE, with the E standing for Easy DMA.

Registers

  • Registers are comprised of one or more bitfields.
  • Each bitfield is at least 1 bit in length.
  • Sometimes bitfields can only take from a limited set of values
  • This is all in your datasheet!

C Code

Embedded Code in C often uses shifts and bitwise-AND to extract bitfields from registers.

#define UARTE_INTEN_CTS_SHIFT (0)
#define UARTE_INTEN_CTS_MASK (0x00000001)
#define UARTE_INTEN_RXRDY_SHIFT (2)
#define UARTE_INTEN_RXRDY_MASK (0x00000001)

// The other nine fields are skipped for brevity
uint32_t cts = 0;
uint32_t rxrdy = 1;

uint32_t inten_value = ((cts & UARTE_INTEN_CTS_MASK) << UARTE_INTEN_CTS_SHIFT)
    | ((rxrdy & UARTE_INTEN_RXRDY_MASK) << UARTE_INTEN_RXRDY_SHIFT);

*((volatile uint32_t*) 0x40002300) = inten_value;

Rust Code

You could do this in Rust if you wanted...

const UARTE0_INTEN: *mut u32 = 0x4000_2300 as *mut u32;
unsafe { UARTE0_INTEN.write_volatile(0x0000_0003); }

But this still seems very error-prone. Nothing stops you putting the wrong value at the wrong address.

Adding structure

In C, the various registers for a peripheral can also be grouped into a struct:

typedef volatile struct uart0_reg_t {
    uint32_t tasks_startrx; // @ 0x000
    uint32_t tasks_stoprx; // @ 0x004
    // ...
    uint32_t inten; // @ 0x300
    uint32_t _padding[79]; 
    uint32_t baudrate; // @ 0x500
} uart0_reg_t

uart0_reg_t* const p_uart = (uart0_reg_t*) 0x40002000;

Structures in Rust

#[repr(C)]
pub struct Uart0 {
    pub tasks_startrx: VolatileCell<u32>, // @ 0x000
    pub tasks_stoprx: VolatileCell<u32>, // @ 0x004
    // ...
    pub inten: VolatileCell<u32>, // @ 0x300
    _reserved12: [u32; 79],
    pub baudrate: VolatileCell<u32>, // @ 0x500
}

let p_uart: &Uart0 = unsafe { &*(0x40002000 as *const Uart0) };    

The vcell::VolatileCell type ensures the compiler emits volatile pointer read/writes.

Note:

There is some discussion about whether VolatileCell technically breaks Rust's rules around references. It works in practice, but it might be technically unsound.

Other approaches

#![allow(unused)]
fn main() {
pub struct Uart { base: *mut u32 } // now has no fields

impl Uart {
    fn write_tasks_stoprx(&mut self, value: u32) {
        unsafe {
            let ptr = self.base.offset(1);
            ptr.write_volatile(value)
        }
    }

    fn read_baudrate(&self) -> u32 {
        unsafe {
            let ptr = self.base.offset(0x140);
            ptr.read_volatile()
        }
    }
}

let uart = Uart { base: unsafe { 0x40002000 as *mut u32 } };
}

Note:

The pointer is a *mut u32 so the offsets are all in 32-bit words, not bytes.

Zero Sized Types

We could handle the address as part of the type instead...

#![allow(unused)]
fn main() {
pub struct Uart<const ADDR: usize> {}

impl<const ADDR: usize> Uart<ADDR> {
    fn write_tasks_stoprx(&mut self, value: u32) {
        unsafe {
            let ptr = (ADDR as *mut u32).offset(1);
            ptr.write_volatile(value)
        }
    }

    fn read_baudrate(&self) -> u32 {
        unsafe {
            let ptr = (ADDR as *mut u32).offset(0x140);
            ptr.read_volatile()
        }
    }
}


let uart: Uart::<0x40002000> = Uart {};
}

Note:

By itself this seems a small change, but imagine a struct which represents 75 individual peripherals. That's not impossible for a modern microcontroller. Holding one word for each now takes up valuable RAM!

CMSIS-SVD Files

A CMSIS-SVD (or just SVD) file is an XML description of all the peripherals, registers and fields on an MCU.

We can use svd2rust to turn this into a Peripheral Access Crate.


%3svdSVD XMLsvd2rustsvd2rustsvd->svd2rustrustRust Sourcesvd2rust->rust

Note:

Although it is an Arm standard, there are examples of RISC-V based microcontrollers which use the same format SVD files and hence can use svd2rust.

Also be aware that manufacturers often assume you will only use the SVD file to inspect the microcontrollers state whilst debugging, and so accuracy has been known to vary somewhat. Rust groups often have to maintain a set of patches to fix known bugs in the SVD files.

The svd2rust generated API
%3PeripheralsPeripheralsuarte1.UARTE1: UARTE1Peripherals->uarte1uarte2.UARTE2: UARTE2Peripherals->uarte2uarte1_baudrate.baudrate: BAUDATEuarte1->uarte1_baudrateuarte1_inten.inten: INTENuarte1->uarte1_intenuarte2_baudrate.baudrate: BAUDATEuarte2->uarte2_baudrateuarte2_inten.inten: INTENuarte2->uarte2_inten


  • The crate has a top-level struct Peripherals with members for each Peripheral
  • Each Peripheral gets a struct, like UARTE0, SPI1, etc.
  • Each Peripheral struct has members for each Register
  • Each Register gets a struct, like BAUDRATE, INTEN, etc.
  • Each Register struct has read(), write() and modify() methods
  • Each Register also has a Read Type (R) and a Write Type (W)
    • Those Read/Write Types give you access to the Bitfields

The svd2rust generated API (2)

  • The read() method returns a special proxy object, with methods for each Field
  • The write() method takes a closure, which is given a special 'proxy' object, with methods for each Field
    • All the Field changes are batched together and written in one go
    • Any un-written Fields are set to a default value
  • The modify() method gives you both
    • Any un-written Fields are left alone

Using a PAC

let p = nrf52840_pac::Peripherals::take().unwrap();
// Reading the 'baudrate' field
let contents = p.UARTE1.baudrate.read();
let current_baud_rate = contents.baudrate();
// Modifying multiple fields in one go
p.UARTE1.inten.modify(|_r, w| {
    w.cts().enabled();
    w.ncts().enabled();
    w.rxrdy().enabled();
    w    
});

Wait, what's a closure?

  • It's an anonymous function, declared in-line with your other code
  • It can 'capture' local variables (although we don't use that feature here)
  • It enables a very powerful Rust idiom, that you can't easily do in C...

Let's take it in turns

  • I, the callee, need to set some stuff up
  • You, the caller, need to do a bit of work
  • I, the callee, need to clean everything up

We can use a closure to insert the caller-provided code in the middle of our function. We see this used all (1) over (2) the (3) Rust standard library!

Quiz time

What are the three steps here?

p.UARTE1.inten.modify(|_r, w| {
    w.cts().enabled();
    w.ncts().enabled();
    w.rxrdy().enabled();
    w    
});

Note:

  1. Read the peripheral MMIO register contents as an integer
  2. Call the closure to modify the integer
  3. Write the integer back to the peripheral MMIO register

Documentation

Docs can be generated from the source code.

See https://docs.rs/nrf52840-pac

Note that uarte0 is a module and UARTE0 could mean either a struct type, or a field on the Peripherals struct.

UPPER_CASE and TitleCase