Nowadays many MCUs have an on-board flash, but not an eeprom. While it's very common for an application to store and load various configuration settings (eg. networking parameters), doing so using the onboard flash can be tricky.
That's because flash can be only erased in blocks, and can sustain only a certain amount of erase cycles - so one should try to minimize the number of erases. Also writes are not atomic, so a power-fail could happen in the middle of a write - in such a case one should recover the last valid configuration.
TxFlash is designed exactly to address these problems!
TxFlash is a C++11 header-only library designed to provide a simple way to store and load an opaque configuration using two flash banks.
The library includes also flash bank implementations for STM32F4 and STM32F7 families, but can easily be ported to any mcu.
TxFlash' key characteristics:
- Store configurations sequentially on the same bank, retrieving the latest one on load,
- When a bank is completely used, erase to the other and switch them,
- Implement a simple transaction management to protect against mid-write reset or other partial flushes
You can install TxFlash via conan, add the following dependency to your conanfile:
TxFlash/0.1@aleofreddi/testing
NOTE: The package has been submitted to conan central but it's not there yet - so you need to clone this repository and install it from the sources using conan export . aleofreddi/testing && conan install TxFlash/0.1@aleofreddi/testing --build
.
Then setup the flash banks to use, in this case we are dealing with STM32F4:
#include <txflash.hh>
#include <txflash_stm32f4.hh>
...
// Initialize the flash using flash sectors 1 & 2 with a default configuration
const char initial_conf[] = "default configuration";
auto flash = txflash::make_txflash(
txflash::Stm32f4FlashBank<FLASH_SECTOR_1, 0x08008000, 0x8000>(),
txflash::Stm32f4FlashBank<FLASH_SECTOR_2, 0x08010000, 0x8000>()
initial_conf,
sizeof(initial_conf)
);
// Read back the default value
assert(flash.length() == sizeof(initial_conf));
char tmp[100];
flash.read(tmp);
assert(std::string(tmp) == initial_conf);
// Now write something new and read it back
char new_conf[] = "another configuration";
flash.write(new_conf, sizeof(new_conf));
assert(flash.length() == sizeof(new_conf));
flash.read(tmp);
assert(std::string(tmp) == new_conf);
Since you are now using flash banks to store data, you need to ensure that the linker won't place code over there. Follows an example for GNU (arm) ld to allocate the first two bank sectors for TxFlash on STM32:
/* Define output sections */
SECTIONS
{
.isr_vector :
{
. = ALIGN(4);
KEEP(*(.isr_vector))
. = ALIGN(4);
} >FLASH
.flash_bank_1 :
{
. = ALIGN(0x8000);
KEEP(*(.flash_bank_1)) /* <-- Flash bank 1, reserved for TxFlash */
. += 0x8000;
} >FLASH
.flash_bank_2 :
{
KEEP(*(.flash_bank_2)) /* <-- Flash bank 2, reserved for TxFlash */
. += 0x8000;
} >FLASH
.text :
{
. = ALIGN(4); /* Text section: start after flash bank 2 */
//-- cut --//