This post is older than 2 years and might not be relevant anymore
More Info: Consider searching for newer posts

Update process in FLASH memory (BANK1 to BANK0) from application

Hi,

We use the nRF51 (nRF51822) on our products with the S130 SDK 2.0.1. In order to update the BLE application that we developed, we use the bootloader (secure bootloader DFU : SDK 12.2.0). Currently the update is done via the BLE in DFU mode.

We now want to be able to update the Nordic application via the UART TTL link with our communication protocol that we have integrated into our application. The update would be done via the application by loading the new firmware in the memory area BANK1. Once loading is complete and checked, we would like to reboot on the bootloader (secure bootloader DFU) by indicating that an update is to make from the BANK1 to the BANK0.

Would you have already implemented update process? Do you have any examples that come close to this principle? Which libraries are needed to do that? How to tell the bootloader that an update is to be done (registry, ...)?

Regards,

Parents
  • Hi Xavier, 

    What you described is similar to the background DFU that we implemented in our IOT libraries. 

    It's not implemented in SDK v12.x but it should be possible to follow what we did in newer SDK. Basically, you need to do something similar to nrf_dfu_postvalidate() function, where we write to the bootloader setting to bank 1 as NRF_DFU_BANK_VALID_APP. After the setting is written in flash, we will trigger a reset. When the bootloader boot up, it will check the setting and then swap the image when bank 1 is valid app. You can find the code doing that in nrf_dfu_continue() the call for it is inside nrf_dfu_init().

    So what you do in the application is to receive image and then write to the bootloader setting as described then trigger a reset. 

Reply
  • Hi Xavier, 

    What you described is similar to the background DFU that we implemented in our IOT libraries. 

    It's not implemented in SDK v12.x but it should be possible to follow what we did in newer SDK. Basically, you need to do something similar to nrf_dfu_postvalidate() function, where we write to the bootloader setting to bank 1 as NRF_DFU_BANK_VALID_APP. After the setting is written in flash, we will trigger a reset. When the bootloader boot up, it will check the setting and then swap the image when bank 1 is valid app. You can find the code doing that in nrf_dfu_continue() the call for it is inside nrf_dfu_init().

    So what you do in the application is to receive image and then write to the bootloader setting as described then trigger a reset. 

Children
  • Hi Hung,

    Thanks for your answer. Could you send me a link to download this sample code?

    Regards,

  • You can have a look here: https://infocenter.nordicsemi.com/topic/sdk_nrf5_v16.0.0/iot_sdk_app_tftp_dfu.html?cp=6_1_4_2_4_9_2

    The example is located at \examples\iot\tftp\background_dfu in SDK v16. 

    Vidar has made an example for UART background DFU, it could be more simple. See here: https://devzone.nordicsemi.com/f/nordic-q-a/46336/bootloader-no-transport-dual-bank-dfu-and-dependencies

  • Thanks for your answer.

    I am trying with this example.

    Just a question. With DFU process in BLE communication, the file downloaded is the zip file.

    In bank1, is it only the bin file that is downloaded ?

    Regards,

  • The data transfer via BLE is actually the binary image. The  DFU master's job (the phone) is to unzip it and get the init packet, binary image and send to the bootloader.

  • You were in contact with Xavier about updating nRF51 application via the UART TTL link.
    Unfortunatly, Xavier leaft our company last week, so I have to continue it's work about this subjet.

    I saw several examples but my code seems to have a problem (besauce it is not working); here are the different steps I do :

    1) calling to nrf_dfu_settings_init (in main.c) to initialize s_dfu_settings structure

    2) receiving new firmware and copying at adress calculated by :
        align_to_page(BANK0_ADDRESS + (uint32_t)s_dfu_settings.bank_0.image_size)
        with BANK0_ADDRESS =0x1B000

    3) registering bank 0 as invalide :
        // Set the bank-code to invalid, and reset size/CRC
        memset(&s_dfu_settings.bank_0, 0, sizeof(nrf_dfu_bank_t));

        // Reset write pointer after completed operation
        s_dfu_settings.write_offset = 0;

        // Reset SD size
        //s_dfu_settings.sd_size = 0; Not needed ????

        // Promote dual bank layout
        s_dfu_settings.bank_layout = NRF_DFU_BANK_LAYOUT_DUAL;

        // Signify that bank 0 is empty
        s_dfu_settings.bank_current = NRF_DFU_CURRENT_BANK_1;

        // s_dfu_settings.bank_1.image_size = (uint32_t)ul_LngApp; // the application length
        // s_dfu_settings.bank_1.image_crc = (uint32_t)crc32_compute((uint8_t *)fs_config.p_start_addr, ul_LngApp, NULL); // the application CRC
        s_dfu_settings.bank_1.bank_code = NRF_DFU_BANK_VALID_APP;

        ret_code_t ret_code = nrf_dfu_settings_write(NULL);



    4) rebooting the nRF51 :
        sd_power_gpregret_set(GPREGRET_BOOTLOADER_DFU_START);
        // Timer for delayed reboot (at least to let the answer be sent)
        app_timer_create(&m_reboot_timer_id, APP_TIMER_MODE_SINGLE_SHOT, (app_timer_timeout_handler_t)NVIC_SystemReset);
        app_timer_start(m_reboot_timer_id, APP_TIMER_TICKS(DELAYED_REBOOT_TIME, APP_TIMER_PRESCALER), NULL);

    This methods does not work properly (the firmware is not updating after reboot).
    So can you help me and tell me if I forgot something ?
    Do I need to calculate CRC and size of bank1, and store them into s_dfu_settings.bank_1.image_crc and s_dfu_settings.bank_1.image_size ?

    The problem also is that I have to deactivate NRF_LOG because of the the application code size (so I'm "blind" and it is very difficult to see what does not work).

Related