2018-05-24 23:15:20 +10:00
|
|
|
Mboot - MicroPython boot loader
|
|
|
|
===============================
|
|
|
|
|
|
|
|
Mboot is a custom bootloader for STM32 MCUs, and currently supports the
|
2020-06-26 21:26:06 +10:00
|
|
|
STM32F4xx, STM32F7xx and STM32WBxx families. It can provide a standard USB DFU
|
|
|
|
interface on either the FS or HS peripherals, as well as a sophisticated, custom I2C
|
2019-02-15 14:31:48 +11:00
|
|
|
interface. It can also load and program firmware in .dfu.gz format from a
|
2020-06-26 21:26:06 +10:00
|
|
|
filesystem, either FAT, littlefs 1 or littlfs 2.
|
|
|
|
It can fit in 16k of flash space, but all features enabled requires 32k.
|
2018-05-24 23:15:20 +10:00
|
|
|
|
|
|
|
How to use
|
|
|
|
----------
|
|
|
|
|
|
|
|
1. Configure your board to use a boot loader by editing the mpconfigboard.mk
|
|
|
|
and mpconfigboard.h files. For example, for an F767 be sure to have these
|
|
|
|
lines in mpconfigboard.mk:
|
|
|
|
|
|
|
|
LD_FILES = boards/stm32f767.ld boards/common_bl.ld
|
|
|
|
TEXT0_ADDR = 0x08008000
|
|
|
|
|
|
|
|
And this in mpconfigboard.h (recommended to put at the end of the file):
|
|
|
|
|
|
|
|
// Bootloader configuration
|
|
|
|
#define MBOOT_I2C_PERIPH_ID 1
|
|
|
|
#define MBOOT_I2C_SCL (pin_B8)
|
|
|
|
#define MBOOT_I2C_SDA (pin_B9)
|
|
|
|
#define MBOOT_I2C_ALTFUNC (4)
|
|
|
|
|
|
|
|
To configure a pin to force entry into the boot loader the following
|
|
|
|
options can be used (with example configuration):
|
|
|
|
|
|
|
|
#define MBOOT_BOOTPIN_PIN (pin_A0)
|
|
|
|
#define MBOOT_BOOTPIN_PULL (MP_HAL_PIN_PULL_UP)
|
|
|
|
#define MBOOT_BOOTPIN_ACTIVE (0)
|
|
|
|
|
2018-06-22 15:30:34 +10:00
|
|
|
Mboot supports programming external SPI flash via the DFU and I2C
|
|
|
|
interfaces. SPI flash will be mapped to an address range. To
|
|
|
|
configure it use the following options (edit as needed):
|
|
|
|
|
|
|
|
#define MBOOT_SPIFLASH_ADDR (0x80000000)
|
|
|
|
#define MBOOT_SPIFLASH_BYTE_SIZE (2 * 1024 * 1024)
|
|
|
|
#define MBOOT_SPIFLASH_LAYOUT "/0x80000000/64*32Kg"
|
|
|
|
#define MBOOT_SPIFLASH_ERASE_BLOCKS_PER_PAGE (32 / 4)
|
|
|
|
#define MBOOT_SPIFLASH_SPIFLASH (&spi_bdev.spiflash)
|
|
|
|
#define MBOOT_SPIFLASH_CONFIG (&spiflash_config)
|
|
|
|
|
|
|
|
This assumes that the board declares and defines the relevant SPI flash
|
|
|
|
configuration structs, eg in the board-specific bdev.c file. The
|
2019-05-29 16:26:02 +10:00
|
|
|
`MBOOT_SPIFLASH_LAYOUT` string will be seen by the USB DFU utility and
|
2018-06-22 15:30:34 +10:00
|
|
|
must describe the SPI flash layout. Note that the number of pages in
|
|
|
|
this layout description (the `64` above) cannot be larger than 99 (it
|
|
|
|
must fit in two digits) so the reported page size (the `32Kg` above)
|
|
|
|
must be made large enough so the number of pages fits in two digits.
|
|
|
|
Alternatively the layout can specify multiple sections like
|
|
|
|
`32*16Kg,32*16Kg`, in which case `MBOOT_SPIFLASH_ERASE_BLOCKS_PER_PAGE`
|
2019-05-29 16:26:02 +10:00
|
|
|
must be changed to `16 / 4` to match the `16Kg` value.
|
2018-06-22 15:30:34 +10:00
|
|
|
|
|
|
|
Mboot supports up to two external SPI flash devices. To configure the
|
|
|
|
second one use the same configuration names as above but with
|
|
|
|
`SPIFLASH2`, ie `MBOOT_SPIFLASH2_ADDR` etc.
|
|
|
|
|
2022-03-17 17:11:44 +11:00
|
|
|
SD card support (read-only, useful in combination with `MBOOT_FSLOAD`)
|
|
|
|
can be enabled with the following options:
|
|
|
|
|
|
|
|
#define MBOOT_ADDRESS_SPACE_64BIT (1)
|
|
|
|
#define MBOOT_SDCARD_ADDR (0x100000000ULL)
|
|
|
|
#define MBOOT_SDCARD_BYTE_SIZE (0x400000000ULL)
|
|
|
|
|
2019-02-15 14:31:48 +11:00
|
|
|
To enable loading firmware from a filesystem use:
|
|
|
|
|
|
|
|
#define MBOOT_FSLOAD (1)
|
|
|
|
|
2020-06-26 21:26:06 +10:00
|
|
|
and then enable one or more of the following depending on what filesystem
|
|
|
|
support is required in Mboot (note that the FAT driver is read-only and
|
|
|
|
quite compact, but littlefs supports both read and write so is rather
|
|
|
|
large):
|
|
|
|
|
|
|
|
#define MBOOT_VFS_FAT (1)
|
|
|
|
#define MBOOT_VFS_LFS1 (1)
|
|
|
|
#define MBOOT_VFS_LFS2 (1)
|
|
|
|
|
2018-05-24 23:15:20 +10:00
|
|
|
2. Build the board's main application firmware as usual.
|
|
|
|
|
|
|
|
3. Build mboot via:
|
|
|
|
|
|
|
|
$ cd mboot
|
|
|
|
$ make BOARD=<board-id>
|
|
|
|
|
|
|
|
That should produce a DFU file for mboot. It can be deployed using
|
|
|
|
USB DFU programming via (it will be placed at location 0x08000000):
|
|
|
|
|
|
|
|
$ make BOARD=<board-id> deploy
|
|
|
|
|
|
|
|
4. Reset the board while holding USR until all 3 LEDs are lit (the 4th option in
|
|
|
|
the cycle) and then release USR. LED0 will then blink once per second to
|
|
|
|
indicate that it's in mboot
|
|
|
|
|
|
|
|
5. Use either USB DFU or I2C to download firmware. The script mboot.py shows how
|
|
|
|
to communicate with the I2C boot loader interface. It should be run on a
|
|
|
|
pyboard connected via I2C to the target board.
|
2018-12-04 23:48:47 +11:00
|
|
|
|
2019-02-15 14:31:48 +11:00
|
|
|
Entering Mboot from application code
|
|
|
|
------------------------------------
|
|
|
|
|
|
|
|
To enter Mboot from a running application do the following:
|
|
|
|
|
|
|
|
1. Make sure I and D caches are disabled.
|
|
|
|
|
|
|
|
2. Load register r0 with the value 0x70ad0000. The lower 7 bits can be
|
|
|
|
optionally or'd with the desired I2C address.
|
|
|
|
|
|
|
|
3. Load the MSP with the value held at 0x08000000.
|
|
|
|
|
|
|
|
4. Jump to the value held at 0x08000004.
|
|
|
|
|
|
|
|
Additional data can be passed to Mboot from application code by storing this
|
|
|
|
data in a special region of RAM. This region begins at the address held at
|
|
|
|
location 0x08000000 (which will point to just after Mboot's stack). A
|
|
|
|
maximum of 1024 bytes can be stored here. To indicate to Mboot that this
|
|
|
|
region is valid load register r0 with 0x70ad0080 (instead of step 2 above),
|
|
|
|
optionally or'd with the desired I2C address.
|
|
|
|
|
|
|
|
Data in this region is a sequence of elements. Each element has the form:
|
|
|
|
|
|
|
|
<type:u8> <len:u8> <payload...>
|
|
|
|
|
|
|
|
where `type` and `len` are bytes (designated by `u8`) and `payload` is 0 or
|
|
|
|
more bytes. `len` must be the number of bytes in `payload`.
|
|
|
|
|
|
|
|
The last element in the data sequence must be the end element:
|
|
|
|
|
|
|
|
* END: type=1, len=0
|
|
|
|
|
2019-05-29 16:26:02 +10:00
|
|
|
Note: MicroPython's `machine.bootloader()` function performs steps 1-4
|
|
|
|
above, and also accepts an optional bytes argument as additional data to
|
|
|
|
pass through to Mboot.
|
|
|
|
|
2019-02-15 14:31:48 +11:00
|
|
|
Loading firmware from a filesystem
|
|
|
|
----------------------------------
|
|
|
|
|
|
|
|
To get Mboot to load firmware from a filesystem and automatically program it
|
|
|
|
requires passing data elements (see above) which tell where the filesystems
|
|
|
|
are located and what filename to program. The elements to use are:
|
|
|
|
|
|
|
|
* MOUNT: type=2, len=10, payload=(<mount-point:u8> <fs-type:u8> <base-addr:u32> <byte-len:u32>)
|
|
|
|
|
2022-03-18 16:22:26 +11:00
|
|
|
* MOUNT: type=2, len=14, payload=(<mount-point:u8> <fs-type:u8> <base-addr:u32> <byte-len:u32> <block-size:u32>)
|
|
|
|
|
|
|
|
* MOUNT: type=2, len=22, payload=(<mount-point:u8> <fs-type:u8> <base-addr:u64> <byte-len:u64> <block-size:u32>)
|
|
|
|
|
2019-02-15 14:31:48 +11:00
|
|
|
* FSLOAD: type=3, len=1+n, payload=(<mount-point:u8> <filename...>)
|
|
|
|
|
2022-03-18 16:22:26 +11:00
|
|
|
`u32`/`u64` mean unsigned 32-bit/64-bit little-endian integers.
|
2019-02-15 14:31:48 +11:00
|
|
|
|
2020-04-13 23:58:29 +02:00
|
|
|
The firmware to load must be a gzip'd DfuSe file (.dfu.gz) and stored within a
|
2020-06-26 21:26:06 +10:00
|
|
|
FAT or littlefs formatted partition.
|
2019-02-15 14:31:48 +11:00
|
|
|
|
2019-05-29 16:26:02 +10:00
|
|
|
The provided fwupdate.py script contains helper functions to call into Mboot
|
2020-06-26 21:26:06 +10:00
|
|
|
with the correct data, and also to update Mboot itself. For example on PYBD
|
|
|
|
the following will update the main MicroPython firmware from the file
|
|
|
|
firmware.dfu.gz stored on the default FAT filesystem:
|
|
|
|
|
|
|
|
import fwupdate
|
|
|
|
fwupdate.update_mpy('firmware.dfu.gz', 0x80000000, 2 * 1024 * 1024)
|
|
|
|
|
|
|
|
The 0x80000000 value is the address understood by Mboot as the location of
|
|
|
|
the external SPI flash, configured via `MBOOT_SPIFLASH_ADDR`.
|
2019-05-29 16:26:02 +10:00
|
|
|
|
2022-03-17 17:11:44 +11:00
|
|
|
To load a file from the SD card (see `MBOOT_SDCARD_ADDR`), assuming it is a
|
|
|
|
16GiB card, use:
|
|
|
|
|
|
|
|
fwupdate.update_mpy('firmware.dfu.gz', 0x1_00000000, 0x4_00000000, addr_64bit=True)
|
|
|
|
|
2021-01-15 03:00:40 +11:00
|
|
|
Signed and encrypted DFU support
|
|
|
|
--------------------------------
|
|
|
|
|
|
|
|
Mboot optionally supports signing and encrypting the binary firmware in the DFU file.
|
2021-04-25 01:38:45 -05:00
|
|
|
In general this is referred to as a packed DFU file. This requires additional settings
|
2021-01-15 03:00:40 +11:00
|
|
|
in the board config and requires the `pyhy` Python module to be installed for `python3`
|
|
|
|
to be used when building packed firmware, eg:
|
|
|
|
|
|
|
|
$ pip3 install pyhy
|
|
|
|
|
|
|
|
In addition to the changes made to mpconfigboard.mk earlier, for encrypted
|
|
|
|
support you also need to add:
|
|
|
|
|
|
|
|
MBOOT_ENABLE_PACKING = 1
|
|
|
|
|
|
|
|
You will also need to generate signing and encryption keys which will be built into
|
|
|
|
mboot and used for all subsequent installations of firmware. This can be done via:
|
|
|
|
|
|
|
|
$ python3 ports/stm32/mboot/mboot_pack_dfu.py generate-keys
|
|
|
|
|
|
|
|
This command generates a `mboot_keys.h` file which should be stored in the board
|
|
|
|
definition folder (next to mpconfigboard.mk).
|
|
|
|
|
|
|
|
Once you build the firmware, the `firmware.pack.dfu` file will contain the encrypted
|
|
|
|
and signed firmware, and can be deployed via USB DFU, or by copying it to the device's
|
|
|
|
internal filesystem (if `MBOOT_FSLOAD` is enabled). `firmware.dfu` is still unencrypted
|
|
|
|
and can be directly flashed with jtag etc.
|
|
|
|
|
2018-12-04 23:48:47 +11:00
|
|
|
Example: Mboot on PYBv1.x
|
|
|
|
-------------------------
|
|
|
|
|
|
|
|
By default mboot is not used on PYBv1.x, but full mboot configuration is provided
|
|
|
|
for these boards to demonstrate how it works and for testing. To build and
|
|
|
|
deploy mboot on these pyboards the only difference from the normal build process
|
|
|
|
is to pass `USE_MBOOT=1` to make, so that the mboot configuration is used instead
|
|
|
|
of the non-mboot configuration.
|
|
|
|
|
|
|
|
In detail for PYBv1.0 (for PYBv1.1 use PYBV11 instead of PYBV10):
|
|
|
|
|
|
|
|
1. Make sure the pyboard is in factory DFU mode (power up with BOOT0 connected to
|
|
|
|
3V3), then build mboot and deploy it (from the stm32/mboot/ directory):
|
|
|
|
|
|
|
|
$ make BOARD=PYBV10 USE_MBOOT=1 clean all deploy
|
|
|
|
|
|
|
|
This will put mboot on the pyboard.
|
|
|
|
|
|
|
|
2. Now put the pyboard in mboot mode by holding down USR, pressing RST, and
|
|
|
|
continue to hold down USR until the blue LED is lit (the 4th option in the
|
|
|
|
cycle) and then release USR. The red LED will blink once per second to
|
|
|
|
indicate that it's in mboot. Then build the MicroPython firmware and deploy
|
|
|
|
it (from the stm32/ directory):
|
|
|
|
|
|
|
|
$ make BOARD=PYBV10 USE_MBOOT=1 clean all deploy
|
|
|
|
|
|
|
|
MicroPython will now be on the device and should boot straightaway.
|
|
|
|
|
|
|
|
On PYBv1.x without mboot the flash layout is as follows:
|
|
|
|
|
|
|
|
0x08000000 0x08004000 0x08020000
|
|
|
|
| ISR text | filesystem | rest of MicroPython firmware
|
|
|
|
|
|
|
|
On PYBv1.x with mboot the flash layout is as follows:
|
|
|
|
|
|
|
|
0x08000000 0x08004000 0x08020000
|
|
|
|
| mboot | filesystem | ISR and full MicroPython firmware
|
|
|
|
|
|
|
|
Note that the filesystem remains intact when going to/from an mboot configuration
|
|
|
|
so its contents will be preserved.
|