HAL_PX4: implement optional flash backend for storage
this allows the last two flash sectors to be used as storage for parameters, waypoints etc. A change to the bootloader to not erase the last two sectors is needed
This commit is contained in:
parent
7e0d904657
commit
a66924422f
@ -9,6 +9,7 @@
|
|||||||
#include <errno.h>
|
#include <errno.h>
|
||||||
#include <stdio.h>
|
#include <stdio.h>
|
||||||
#include <ctype.h>
|
#include <ctype.h>
|
||||||
|
#include <nuttx/progmem.h>
|
||||||
|
|
||||||
#include "Storage.h"
|
#include "Storage.h"
|
||||||
using namespace PX4;
|
using namespace PX4;
|
||||||
@ -40,8 +41,12 @@ void PX4Storage::_storage_open(void)
|
|||||||
|
|
||||||
_dirty_mask.clearall();
|
_dirty_mask.clearall();
|
||||||
|
|
||||||
// load from mtd
|
// load from storage backend
|
||||||
|
#if USE_FLASH_STORAGE
|
||||||
|
_flash_load();
|
||||||
|
#else
|
||||||
_mtd_load();
|
_mtd_load();
|
||||||
|
#endif
|
||||||
|
|
||||||
#ifdef SAVE_STORAGE_FILE
|
#ifdef SAVE_STORAGE_FILE
|
||||||
fd = open(SAVE_STORAGE_FILE, O_WRONLY|O_CREAT|O_TRUNC, 0644);
|
fd = open(SAVE_STORAGE_FILE, O_WRONLY|O_CREAT|O_TRUNC, 0644);
|
||||||
@ -143,8 +148,12 @@ void PX4Storage::_timer_tick(void)
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
// write the line
|
// save to storage backend
|
||||||
|
#if USE_FLASH_STORAGE
|
||||||
|
_flash_write(i);
|
||||||
|
#else
|
||||||
_mtd_write(i);
|
_mtd_write(i);
|
||||||
|
#endif
|
||||||
|
|
||||||
perf_end(_perf_storage);
|
perf_end(_perf_storage);
|
||||||
}
|
}
|
||||||
@ -198,4 +207,65 @@ void PX4Storage::_mtd_load(void)
|
|||||||
close(fd);
|
close(fd);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
load all data from flash
|
||||||
|
*/
|
||||||
|
void PX4Storage::_flash_load(void)
|
||||||
|
{
|
||||||
|
_flash_page = up_progmem_npages() - 2;
|
||||||
|
if (up_progmem_pagesize(_flash_page) != 128*1024U ||
|
||||||
|
up_progmem_pagesize(_flash_page+1) != 128*1024U) {
|
||||||
|
AP_HAL::panic("Bad flash page sizes %u %u",
|
||||||
|
up_progmem_pagesize(_flash_page),
|
||||||
|
up_progmem_pagesize(_flash_page+1));
|
||||||
|
}
|
||||||
|
|
||||||
|
printf("Storage: Using flash pages %u and %u\n", _flash_page, _flash_page+1);
|
||||||
|
|
||||||
|
if (!_flash.init()) {
|
||||||
|
AP_HAL::panic("unable to init flash storage");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
write one storage line. This also updates _dirty_mask.
|
||||||
|
*/
|
||||||
|
void PX4Storage::_flash_write(uint16_t line)
|
||||||
|
{
|
||||||
|
if (_flash.write(line*PX4_STORAGE_LINE_SIZE, PX4_STORAGE_LINE_SIZE)) {
|
||||||
|
// mark the line clean
|
||||||
|
_dirty_mask.clear(line);
|
||||||
|
} else {
|
||||||
|
perf_count(_perf_errors);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
callback to write data to flash
|
||||||
|
*/
|
||||||
|
bool PX4Storage::_flash_write_data(uint8_t sector, uint32_t offset, const uint8_t *data, uint16_t length)
|
||||||
|
{
|
||||||
|
size_t base_address = up_progmem_getaddress(_flash_page+sector);
|
||||||
|
return up_progmem_write(base_address+offset, data, length) == length;
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
callback to read data from flash
|
||||||
|
*/
|
||||||
|
bool PX4Storage::_flash_read_data(uint8_t sector, uint32_t offset, uint8_t *data, uint16_t length)
|
||||||
|
{
|
||||||
|
size_t base_address = up_progmem_getaddress(_flash_page+sector);
|
||||||
|
const uint8_t *b = ((const uint8_t *)base_address)+offset;
|
||||||
|
memcpy(data, b, length);
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
callback to erase flash sector
|
||||||
|
*/
|
||||||
|
bool PX4Storage::_flash_erase_sector(uint8_t sector)
|
||||||
|
{
|
||||||
|
return up_progmem_erasepage(_flash_page+sector) > 0;
|
||||||
|
}
|
||||||
|
|
||||||
#endif // CONFIG_HAL_BOARD
|
#endif // CONFIG_HAL_BOARD
|
||||||
|
@ -1,12 +1,23 @@
|
|||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
|
#define USE_FLASH_STORAGE 1
|
||||||
|
|
||||||
#include <AP_HAL/AP_HAL.h>
|
#include <AP_HAL/AP_HAL.h>
|
||||||
#include "AP_HAL_PX4_Namespace.h"
|
#include "AP_HAL_PX4_Namespace.h"
|
||||||
#include <systemlib/perf_counter.h>
|
#include <systemlib/perf_counter.h>
|
||||||
#include <AP_Common/Bitmask.h>
|
#include <AP_Common/Bitmask.h>
|
||||||
|
#include <AP_FlashStorage/AP_FlashStorage.h>
|
||||||
|
|
||||||
#define PX4_STORAGE_SIZE HAL_STORAGE_SIZE
|
#define PX4_STORAGE_SIZE HAL_STORAGE_SIZE
|
||||||
|
|
||||||
|
#if USE_FLASH_STORAGE
|
||||||
|
// when using flash storage we use a small line size to make storage
|
||||||
|
// compact and minimise the number of erase cycles needed
|
||||||
|
#define PX4_STORAGE_LINE_SHIFT 3
|
||||||
|
#else
|
||||||
#define PX4_STORAGE_LINE_SHIFT 9
|
#define PX4_STORAGE_LINE_SHIFT 9
|
||||||
|
#endif
|
||||||
|
|
||||||
#define PX4_STORAGE_LINE_SIZE (1<<PX4_STORAGE_LINE_SHIFT)
|
#define PX4_STORAGE_LINE_SIZE (1<<PX4_STORAGE_LINE_SHIFT)
|
||||||
#define PX4_STORAGE_NUM_LINES (PX4_STORAGE_SIZE/PX4_STORAGE_LINE_SIZE)
|
#define PX4_STORAGE_NUM_LINES (PX4_STORAGE_SIZE/PX4_STORAGE_LINE_SIZE)
|
||||||
|
|
||||||
@ -33,9 +44,22 @@ private:
|
|||||||
int _fd = -1;
|
int _fd = -1;
|
||||||
void _mtd_load(void);
|
void _mtd_load(void);
|
||||||
void _mtd_write(uint16_t line);
|
void _mtd_write(uint16_t line);
|
||||||
|
|
||||||
#if defined (CONFIG_ARCH_BOARD_PX4FMU_V4)
|
#if defined (CONFIG_ARCH_BOARD_PX4FMU_V4)
|
||||||
irqstate_t irq_state;
|
irqstate_t irq_state;
|
||||||
#endif
|
#endif
|
||||||
void bus_lock(bool lock);
|
void bus_lock(bool lock);
|
||||||
|
|
||||||
|
bool _flash_write_data(uint8_t sector, uint32_t offset, const uint8_t *data, uint16_t length);
|
||||||
|
bool _flash_read_data(uint8_t sector, uint32_t offset, uint8_t *data, uint16_t length);
|
||||||
|
bool _flash_erase_sector(uint8_t sector);
|
||||||
|
uint8_t _flash_page;
|
||||||
|
|
||||||
|
AP_FlashStorage _flash{_buffer,
|
||||||
|
128*1024U,
|
||||||
|
FUNCTOR_BIND_MEMBER(&PX4Storage::_flash_write_data, bool, uint8_t, uint32_t, const uint8_t *, uint16_t),
|
||||||
|
FUNCTOR_BIND_MEMBER(&PX4Storage::_flash_read_data, bool, uint8_t, uint32_t, uint8_t *, uint16_t),
|
||||||
|
FUNCTOR_BIND_MEMBER(&PX4Storage::_flash_erase_sector, bool, uint8_t)};
|
||||||
|
|
||||||
|
void _flash_load(void);
|
||||||
|
void _flash_write(uint16_t line);
|
||||||
};
|
};
|
||||||
|
Loading…
Reference in New Issue
Block a user