Start-up

Overview

All current Cortex-M processors get out of reset as follows:

  • the initial stack pointer value is at address 0 of the vector table;
  • the initial program counter value is at address 4 of the vector table.

The processor’s hardware will initialise the stack pointer register and the program counter register using these values.

Astrobe puts the two values at the beginning of the binary file produced by the linker. However, the RPx MCUs don’t not use these values for the stack pointer and the program counter right after reset – they first execute start-up code in the boot rom.

Stepping over the initial steps after reset, the boot rom code will eventually execute our program code uploaded as UF2 file, as created from Astrobe’s .bin file, and use the aforementioned initial values. This is done differently on the RP2040 and the RP2350.

RP2040

On the RP2040, the boot code loads the first 256 bytes in flash memory from address 010000000H into SRAM.1 This code’s responsibility is to configure and enable Execute In Place (XIP). It must be prepended to Astrobe program code.2

This prepended code also must, as its last instructions, load the actual program’s initial stack pointer value and program counter value into their respective registers. The easiest way is to load the program code in Astrobe’s .bin file at flash address 010000100H, so that address 010000100H contains the initial stack pointer value, and address 010000104H the initial program counter value. That’s where the prepended code finds them.

From there on, the initialisation sequence of the Oberon program starts to execute.

RP2350

On the RP2350, the boot code in the ROM takes care of configuring and enabling Execute In Place (XIP), so there is no need to prepend any initialisation code. However, the RP2350 has a plethora of different possible modes and memory layouts to load and execute code, including code signing and address translation. Therefore, the boot code needs to get the corresponding configuration data, which is provided via meta data blocks in the UF2 file, which must be added to Astrobe’s program code.

These meta data blocks can be prepended or inserted in different ways, as described in chapter 5 of the datasheet. Astrobe for RP2350 uses an insertion technique, while makeuf2 prepends the meta data.

Let’s look at the two techniques.

Astrobe for RP2350

Module ImageDef contains a what the datasheet calls Minimum ARM IMAGE_DEF meta data block (section 5.9.5.1) in its body as data (using SYSTEM.DATA). That is, the linker will put the meta data block embedded into the binary, where the bootrom code will find and interpret it (the first meta data block must be in the first 4k of flash memory).

This IMAGE_DEF item does not contain any information about the vector table location, and with it the initial stack pointer value and code entry address as explained above. Therefore, the vector table is assumed to be at address 010000000H.

The bootrom code will find the two values at 010000000H (stack pointer) and 010000004H (entry address), load them into their corresponding registers, and from there on the initialisation sequence of the Oberon program starts to execute. The program runs in Secure, privileged thread mode.

makeuf2

Utility makeuf2 prepends the meta data block in the first 256 data bytes (a UF2 data block) at address 010000000, and adds the program code in the subsequent UF2 blocks, starting from address 010000100H, which is the address of the vector table with the initial values of the stack pointer and program counter.

To inform the bootrom code about this location, makeuf2 inserts the same IMAGE_DEF item as Astrobe, plus an additional item that defines the location of the vector table to be at 010000100H – unsurprisingly called a VECTOR_TABLE item.

Now the bootrom code finds the initial stack pointer and program counter values at 010000100H and 010000104H, respectively, loads them into their corresponding registers, and from there on the initialisation sequence of the Oberon program starts to execute. The program runs in Secure, privileged thread mode.

Quick Preliminary Assessment

For the very basic use case, both techniques work equally well, and are indistinguishable from a practical point of view.3

More advanced scenarios will require exploration and evaluation first with regard their practical value. makeuf2 can place meta data block anywhere, while meta data inserted via ImageDef cannot. So we’ll cross that bridge when we get there, I guess. :)


  1. Obviously, this code cannot run from flash, as XIP is not working yet. ↩︎

  2. The utility abin2uf2 does this, or Astrobe for RP2040 does it directly. ↩︎

  3. Just don’t forget to use the appropriate config.ini file for each case – see section Astrobe Configuration at the bottom of this document↩︎