/* * Copyright (C) 2016 Freie Universität Berlin * 2017 Inria * * This file is subject to the terms and conditions of the GNU Lesser * General Public License v2.1. See the file LICENSE in the top level * directory for more details. */ /** * @defgroup drivers_periph_flashpage Flash page driver * @ingroup drivers_periph * @brief Low-level flash page interface * * This interface provides a very simple and straight forward way for writing * a MCU's internal flash. This interface is only capable of reading, verifying, * and writing complete flash pages, it has no support for partial flash access. * This enables for very slim and efficient implementations. * * A module for more fine-grained access of memory locations can easily be * programmed on top of this interface. * * @note Flash memory has only a limited amount of erase cycles (mostly * around 10K times), so using this interface in some kind of loops * can damage you MCU! * * @{ * @file * @brief Low-level flash page peripheral driver interface * * @author Hauke Petersen * @author Kaspar Schleiser * @author Francisco Acosta * */ #ifndef PERIPH_FLASHPAGE_H #define PERIPH_FLASHPAGE_H #include #include "periph_cpu.h" #ifdef __cplusplus extern "C" { #endif /** * @brief Per default, we expect the internal flash to start at address 0 */ #ifndef CPU_FLASH_BASE #define CPU_FLASH_BASE (0) #endif /** * @def FLASHPAGE_RAW_BLOCKSIZE * * @brief For raw writings to the flash, this constant must define the * minimum write length allowed by the MCU. */ #ifdef DOXYGEN #define FLASHPAGE_RAW_BLOCKSIZE #endif /** * @def FLASHPAGE_RAW_ALIGNMENT * * @brief The buffers to be written to the flash MUST be aligned, as well as * the address on which the buffer is written to the flash. This variable * must be defined for that purpose, according to the MCU align * requirements. */ #ifdef DOXYGEN #define FLASHPAGE_RAW_ALIGNMENT #endif /** * @def FLASHPAGE_SIZE * * @brief Make sure the page size and the number of pages is defined */ #ifndef FLASHPAGE_SIZE #error "periph/flashpage: FLASHPAGE_SIZE not defined" #endif #ifndef FLASHPAGE_NUMOF #error "periph/flashpage: FLASHPAGE_NUMOF not defined" #endif /** * @brief Return values used in this interface */ enum { FLASHPAGE_OK = 0, /**< everything succeeded */ FLASHPAGE_NOMATCH = -1 /**< page differs from target data */ }; /** * @brief Translate the given page number into the page's starting address * * @note The given page MUST be valid, otherwise the returned address points * to an undefined memory location! * * @param[in] page page number to get the address of * * @return starting memory address of the given page */ static inline void *flashpage_addr(int page) { return (void *)(CPU_FLASH_BASE + (page * FLASHPAGE_SIZE)); } /** * @brief Translate the given address into the corresponding page number * * The given address can be any address inside a page. * * @note The given address MUST be a valid flash address! * * @param[in] addr address inside the targeted page * * @return page containing the given address */ static inline int flashpage_page(void *addr) { return (int)(((int)addr - CPU_FLASH_BASE) / FLASHPAGE_SIZE); } /** * @brief Write the given page with the given data * * @param[in] page page to write * @param[in] data data to write to the page, MUST be FLASHPAGE_SIZE * byte. Set to NULL for page erase only. */ void flashpage_write(int page, const void *data); /** * @brief Write any number of data bytes to a given location in the * flash memory * * @warning Make sure the targeted memory area is erased before calling * this function * * Both target address and data address must be aligned to * FLASHPAGE_RAW_ALIGN. @p len must be a multiple of FLASHPAGE_RAW_BLOCKSIZE. * This function doesn't erase any area in flash, thus be sure the targeted * memory area is erased before writing on it (using the flashpage_write function). * * @param[in] target_addr address in flash to write to. MUST be aligned * to FLASHPAGE_RAW_ALIGNMENT. * @param[in] data data to write to the address. MUST be aligned * to FLASHPAGE_RAW_ALIGNMENT. * @param[in] len length of the data to be written. It MUST be * multiple of FLASHPAGE_RAW_BLOCKSIZE. Also, * ensure it doesn't exceed the actual flash * memory size. */ void flashpage_write_raw(void *target_addr, const void *data, size_t len); /** * @brief Read the given page into the given memory location * * @param[in] page page to read * @param[out] data memory to write the page to, MUST be FLASHPAGE_SIZE * byte */ void flashpage_read(int page, void *data); /** * @brief Verify the given page against the given data * * @param[in] page page to verify * @param[in] data data to compare page against, MUST be FLASHPAGE_SIZE * byte of data * * @return FLASHPAGE_OK if data in the page is identical to @p data * @return FLASHPAGE_NOMATCH if data and page content diverge */ int flashpage_verify(int page, const void *data); /** * @brief Write the given page and verify the results * * This is a convenience function wrapping flashpage_write and flashpage_verify. * * @param[in] page page to write * @param[in] data data to write to the page, MUST be FLASHPAGE_SIZE * byte. * * @return FLASHPAGE_OK on success * @return FLASHPAGE_NOMATCH if data and page content diverge */ int flashpage_write_and_verify(int page, const void *data); #ifdef __cplusplus } #endif #endif /* PERIPH_FLASHPAGE_H */ /** @} */