Implementing FOTA on nRF52840 DK Using External QSPI Flash (MX25R64)

Hi Nordic community,

I'm currently working on a project where I'm aiming to implement FOTA updates on the nRF52840 DK, with the application images stored on the MX25R64 external flash.

To summarize: My application requires around 600KB of storage. When I enable CONFIG_MCUBOOT_BOOTLOADER, a significant portion of the 1MB internal flash is consumed, resulting in a flash overflow when attempting to flash the program.

I'm trying to expand the flash region to utilize the 64MB available on the MX25R64 external flash. However, I'm encountering difficulties with configuring the necessary config, yaml, and dts files.

From my research, it seems I need to use LittleFS to access the external flash over QSPI. I've attached my current prj.conf and related configuration files. I would greatly appreciate any guidance or example code that demonstrates how to perform FOTA updates using the external flash.

Thank you in advance for your help!

proj.conf:

Fullscreen
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
# Deactivate DEBUG/LOG for production builds
# Logging
CONFIG_LOG=y
#### BLE ####
# Activate the BLE layer, set as peripheral
CONFIG_BT=y
CONFIG_BT_SMP=y
CONFIG_BT_PERIPHERAL=y
CONFIG_BT_CTLR=y
CONFIG_BT_CTLR_PHY_2M=y
CONFIG_BT_GATT_CLIENT=y
# Set up the Services
CONFIG_BT_DIS=y
CONFIG_BT_DIS_PNP=n
# Allow another device that unpaired to pair again
CONFIG_BT_SMP_ALLOW_UNAUTH_OVERWRITE=y
#### FLASH ####
# Configure stack size for process
XXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXX

pm.static:

Fullscreen
1
2
3
4
5
6
7
8
9
10
external_flash:
address: 0x0
end_address: 0x400000 # 64mb
region: external_flash
size: 0x400000
littlefs_storage:
address: 0x0
device: MX25R64_SPI
region: external_flash
size: 0x400000
XXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXX

nrf52840dk_nrf52840.dts

Fullscreen
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
/*
* Copyright (c) 2017 Linaro Limited
*
* SPDX-License-Identifier: Apache-2.0
*/
/dts-v1/;
#include <nordic/nrf52840_qiaa.dtsi>
#include "nrf52840dk_nrf52840-pinctrl.dtsi"
#include <zephyr/dt-bindings/input/input-event-codes.h>
/ {
model = "Nordic nRF52840 DK NRF52840";
compatible = "nordic,nrf52840-dk-nrf52840";
chosen {
zephyr,console = &uart0;
zephyr,shell-uart = &uart0;
zephyr,uart-mcumgr = &uart0;
zephyr,bt-mon-uart = &uart0;
zephyr,bt-c2h-uart = &uart0;
XXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXX

nrf52840dk_nrf52840.overlay

Fullscreen
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
/ {
chosen {
nordic,pm-ext-flash = &mx25r64;
};
partitions {
compatible = "fixed-partitions";
#address-cells = <1>;
#size-cells = <1>;
littlefs_storage: partition@0 {
label = "littlefs_storage";
reg = <0x00000000 0x00400000>; // Adjust the size as needed
};
};
};
&spi2 {
status = "okay";
mx25r64: mx25r64@0 {
XXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXX

Parents
  • Hi

    What is the app_update you refer to here exactly? Are you sure the start address of the secondary (external flash slot) is set to the start of the external flash device and not on the internal flash. Please try debugging and/or comparing your application to the one in the DevAcademy as there must be something you're missing here.

    Best regards,

    Simon

  • Hi,

    From the memory report I see that my flash_primary and external_flash address are the same as the picture included in this course:

    https://academy.nordicsemi.com/courses/nrf-connect-sdk-intermediate/lessons/lesson-8-bootloaders-and-dfu-fota/topic/exercise-2-dfu-over-usb-adding-external-flash/

    Also on my partitions.yml file my external flash’s address comes after my internal flash.

    The app_update I refer to is what is generated in build → zephyr → app_update.bin

    I have attached it below for your consideration

    Also, Could you please link the tutorial you refer to if its different from what I mentioned above.

    Thank you

    5444.app_update.bin

  • Aha, I see.

    So you are able to do DFU first, but then when you upload hello_world, and use test&confirm, the image runs hello world only once and then changes back to peripheral LBS?

    If so, are you able to DFU to a new version of peripheral LBS?

  • I was able to do DFU with the internal flash (existing example of peripheral_lbs) but when I added the overlay for the external flash even with a new version of the same project (modified main.c) it doesn't update. 

    This is my terminal output

    Fullscreen
    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
    21
    *** Booting nRF Connect SDK v2.5.1 ***
    Starting Bluetooth Peripheral LBS example
    I: 2 Sectors of 4096 bytes
    I: alloc wra: 0, fd0
    I: data wra: 0, 1c
    I: SoftDevice Controller build revision:
    I: c5 93 ba a9 14 4d 8d 05 |.....M..
    I: 30 4e 9b 92 d7 71 1e e8 |0N...q..
    I: aa 02 50 3c |..P<
    I: HW Platform: Nordic Semiconductor (0x0002)
    I: HW Variant: nRF52x (0x0002)
    I: Firmware: Standard Bluetooth controller (0x00) Version 197.47763 Build 2370639017
    I: No ID address. App must call settings_load()
    Bluetooth initialized
    I: Identity: E9:8B:53:AC:D8:80 (random)
    I: HCI: version 5.4 (0x0d) revision 0x1102, manufacturer 0x0059
    I: LMP: version 5.4 (0x0d) subver 0x1102
    Flash erased successfully
    Advertising successfully started
    W: Ignoring data for unknown channel ID 0x003a
    I: Image index: 0, Swap type: none
    XXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXX

  • Byt then why are you talking about hello world?

    I am a bit inquisitive here because if I get the whole picture I can help you the best

  • To learn about DFU over the air, I started with the `peripheral_lbs` and `hello_world` examples. First, I flashed the `peripheral_lbs` example onto the board. Then, I used FOTA to update the board with the `hello_world` example, which worked as expected—the board rebooted and displayed "hello world."

    Next, I added external flash support to the `peripheral_lbs` example code, incorporating child_image overlays and configurations, and flashed this updated code onto the board. When I tried to use FOTA to flash the `hello_world` example again, the board did not accept the update and reverted to the previously flashed code.

    I then made changes to `main.c` in my modified `peripheral_lbs` code (with external flash support) but did not flash it directly. Instead, I built the project and attempted to send the DFU update via my phone. However, the board still did not accept the FOTA update and reverted to the existing code that I had previously flashed.

    Please let me know if you need further explanation or additional details.

  • Thanks for the explanation, things are more clear to me now.

    I have two comments:

    First: When using test&confirm, the app assumes that the new app will have BLE so it can send a confirm command to it. This will be an issue when updating from peripheral LBS to hello world, as hello world does not have BLE.

    Second: When you DFU the Peripheral LBS sample, did you change anything in the code before the DFU?
    If the DFU is exactly the same as the running code, the DFU will be ignored. No need to update to the same after all.

Reply
  • Thanks for the explanation, things are more clear to me now.

    I have two comments:

    First: When using test&confirm, the app assumes that the new app will have BLE so it can send a confirm command to it. This will be an issue when updating from peripheral LBS to hello world, as hello world does not have BLE.

    Second: When you DFU the Peripheral LBS sample, did you change anything in the code before the DFU?
    If the DFU is exactly the same as the running code, the DFU will be ignored. No need to update to the same after all.

Children
  • Thanks for your response

    1. I have added BLE configs to the hello_world example, (I used the same project that successfully did the FOTA in my first step as explained above (didn't change anything))

    2. I changed the main.c to additionally printk a message ("hello") to notice the difference but it didn't show up as it rebooted (meaning it has went back to the previous code on the board)

    Let me know if you want me to upload any of the mentioned files

  • Saman Niksiar said:
    but it didn't show up as it rebooted

    Do you get logs from MCUboot?
    If something goes wrong in the DFU, MCUboot logs usually explain what.

  • This is the logs I get from flashing and trying the FOTA. I also get Slot image has no hash TLV error on the nrf connect app side as shown below.

  • Seems you are not getting logs for MCUboot.
    I recommend that you add an overlay with logging configuration to the MCUboot image.