#include #if CONFIG_HAL_BOARD == HAL_BOARD_PX4 #include #include #include #include #include #include #include #include #include "Storage.h" using namespace PX4; /* This stores eeprom data in the PX4 MTD interface with a 4k size, and a in-memory buffer. This keeps the latency and devices IOs down. */ // name the storage file after the sketch so you can use the same sd // card for ArduCopter and ArduPlane #define STORAGE_DIR "/fs/microsd/APM" #define OLD_STORAGE_FILE STORAGE_DIR "/" SKETCHNAME ".stg" #define OLD_STORAGE_FILE_BAK STORAGE_DIR "/" SKETCHNAME ".bak" //#define SAVE_STORAGE_FILE STORAGE_DIR "/" SKETCHNAME ".sav" #define MTD_PARAMS_FILE "/fs/mtd" #define MTD_SIGNATURE 0x14012014 #define MTD_SIGNATURE_OFFSET (8192-4) #define STORAGE_RENAME_OLD_FILE 0 extern const AP_HAL::HAL& hal; PX4Storage::PX4Storage(void) : _fd(-1), _dirty_mask(0), _perf_storage(perf_alloc(PC_ELAPSED, "APM_storage")), _perf_errors(perf_alloc(PC_COUNT, "APM_storage_errors")) { } /* get signature from bytes at offset MTD_SIGNATURE_OFFSET */ uint32_t PX4Storage::_mtd_signature(void) { int mtd_fd = open(MTD_PARAMS_FILE, O_RDONLY); if (mtd_fd == -1) { AP_HAL::panic("Failed to open " MTD_PARAMS_FILE); } uint32_t v; if (lseek(mtd_fd, MTD_SIGNATURE_OFFSET, SEEK_SET) != MTD_SIGNATURE_OFFSET) { AP_HAL::panic("Failed to seek in " MTD_PARAMS_FILE); } bus_lock(true); if (read(mtd_fd, &v, sizeof(v)) != sizeof(v)) { AP_HAL::panic("Failed to read signature from " MTD_PARAMS_FILE); } bus_lock(false); close(mtd_fd); return v; } /* put signature bytes at offset MTD_SIGNATURE_OFFSET */ void PX4Storage::_mtd_write_signature(void) { int mtd_fd = open(MTD_PARAMS_FILE, O_WRONLY); if (mtd_fd == -1) { AP_HAL::panic("Failed to open " MTD_PARAMS_FILE); } uint32_t v = MTD_SIGNATURE; if (lseek(mtd_fd, MTD_SIGNATURE_OFFSET, SEEK_SET) != MTD_SIGNATURE_OFFSET) { AP_HAL::panic("Failed to seek in " MTD_PARAMS_FILE); } bus_lock(true); if (write(mtd_fd, &v, sizeof(v)) != sizeof(v)) { AP_HAL::panic("Failed to write signature in " MTD_PARAMS_FILE); } bus_lock(false); close(mtd_fd); } /* upgrade from microSD to MTD (FRAM) */ void PX4Storage::_upgrade_to_mtd(void) { // the MTD is completely uninitialised - try to get a // copy from OLD_STORAGE_FILE int old_fd = open(OLD_STORAGE_FILE, O_RDONLY); if (old_fd == -1) { ::printf("Failed to open %s\n", OLD_STORAGE_FILE); return; } int mtd_fd = open(MTD_PARAMS_FILE, O_WRONLY); if (mtd_fd == -1) { AP_HAL::panic("Unable to open MTD for upgrade"); } if (::read(old_fd, _buffer, sizeof(_buffer)) != sizeof(_buffer)) { close(old_fd); close(mtd_fd); ::printf("Failed to read %s\n", OLD_STORAGE_FILE); return; } close(old_fd); ssize_t ret; bus_lock(true); if ((ret=::write(mtd_fd, _buffer, sizeof(_buffer))) != sizeof(_buffer)) { ::printf("mtd write of %u bytes returned %d errno=%d\n", sizeof(_buffer), ret, errno); AP_HAL::panic("Unable to write MTD for upgrade"); } bus_lock(false); close(mtd_fd); #if STORAGE_RENAME_OLD_FILE rename(OLD_STORAGE_FILE, OLD_STORAGE_FILE_BAK); #endif ::printf("Upgraded MTD from %s\n", OLD_STORAGE_FILE); } void PX4Storage::_storage_open(void) { if (_initialised) { return; } struct stat st; _have_mtd = (stat(MTD_PARAMS_FILE, &st) == 0); // PX4 should always have /fs/mtd_params if (!_have_mtd) { AP_HAL::panic("Failed to find " MTD_PARAMS_FILE); } /* cope with upgrading from OLD_STORAGE_FILE to MTD */ bool good_signature = (_mtd_signature() == MTD_SIGNATURE); if (stat(OLD_STORAGE_FILE, &st) == 0) { if (good_signature) { #if STORAGE_RENAME_OLD_FILE rename(OLD_STORAGE_FILE, OLD_STORAGE_FILE_BAK); #endif } else { _upgrade_to_mtd(); } } // we write the signature every time, even if it already is // good, as this gives us a way to detect if the MTD device is // functional. It is better to panic now than to fail to save // parameters in flight _mtd_write_signature(); _dirty_mask = 0; int fd = open(MTD_PARAMS_FILE, O_RDONLY); if (fd == -1) { AP_HAL::panic("Failed to open " MTD_PARAMS_FILE); } const uint16_t chunk_size = 128; for (uint16_t ofs=0; ofs>PX4_STORAGE_LINE_SHIFT; line <= end>>PX4_STORAGE_LINE_SHIFT; line++) { _dirty_mask |= 1U << line; } } void PX4Storage::read_block(void *dst, uint16_t loc, size_t n) { if (loc >= sizeof(_buffer)-(n-1)) { return; } _storage_open(); memcpy(dst, &_buffer[loc], n); } void PX4Storage::write_block(uint16_t loc, const void *src, size_t n) { if (loc >= sizeof(_buffer)-(n-1)) { return; } if (memcmp(src, &_buffer[loc], n) != 0) { _storage_open(); memcpy(&_buffer[loc], src, n); _mark_dirty(loc, n); } } void PX4Storage::bus_lock(bool lock) { #if defined (CONFIG_ARCH_BOARD_PX4FMU_V4) /* this is needed on Pixracer where the ms5611 may be on the same bus as FRAM, and the NuttX ramtron driver does not go via the PX4 spi bus abstraction. The ramtron driver relies on SPI_LOCK(). We need to prevent the ms5611 reads which happen in interrupt context from interfering with the FRAM operations. As the px4 spi bus abstraction just uses interrupt blocking as the locking mechanism we need to block interrupts here as well. */ if (lock) { irq_state = irqsave(); } else { irqrestore(irq_state); } #endif } void PX4Storage::_timer_tick(void) { if (!_initialised || _dirty_mask == 0) { return; } perf_begin(_perf_storage); if (_fd == -1) { _fd = open(MTD_PARAMS_FILE, O_WRONLY); if (_fd == -1) { perf_end(_perf_storage); perf_count(_perf_errors); return; } } // write out the first dirty set of lines. We don't write more // than one to keep the latency of this call to a minimum uint8_t i, n; for (i=0; i>PX4_STORAGE_LINE_SHIFT); n++) { if (!(_dirty_mask & (1<<(n+i)))) { break; } // mark that line clean write_mask |= (1<<(n+i)); } /* write the lines. This also updates _dirty_mask. Note that because this is a SCHED_FIFO thread it will not be preempted by the main task except during blocking calls. This means we don't need a semaphore around the _dirty_mask updates. */ if (lseek(_fd, i<