/* * 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! * * # (Low-) Power Implications * * The flashpage driver implementation **should** make sure, that the CPU uses * no additional energy while the flashpage driver is inactive. This means, that * any particular CPU peripherals used for reading and writing flash pages * **should** be disabled before the read and write functions return. * * If an implementation puts the calling thread to sleep for a duration of time, * the implementation might need to take care of blocking certain power modes. * * @{ * @file * @brief Low-level flash page peripheral driver interface * * @author Hauke Petersen * @author Kaspar Schleiser * @author Francisco Acosta * @author Federico Pellegrin * */ #ifndef PERIPH_FLASHPAGE_H #define PERIPH_FLASHPAGE_H #include #include #include "cpu_conf.h" #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_WRITE_BLOCK_SIZE * * @brief For raw writings to the flash, this constant must define the * minimum write length allowed by the MCU. */ #ifdef DOXYGEN #define FLASHPAGE_WRITE_BLOCK_SIZE #endif /** * @def FLASHPAGE_WRITE_BLOCK_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_WRITE_BLOCK_ALIGNMENT #endif /** * @def FLASHPAGE_ERASE_STATE * * @brief State of an erased byte in memory */ #if defined(DOXYGEN) || !defined(FLASHPAGE_ERASE_STATE) #define FLASHPAGE_ERASE_STATE (0xFFU) #endif /** * @def PERIPH_FLASHPAGE_CUSTOM_PAGESIZES * * @brief Defined to signal that the peripheral has non-uniform flash page * sizes. These devices do not define FLASHPAGE_SIZE and do not * implement the pagewise api. */ #ifdef DOXYGEN #define PERIPH_FLASHPAGE_CUSTOM_PAGESIZES #endif /** * @def PERIPH_FLASHPAGE_NEEDS_FLASHPAGE_ADDR * * @brief If non-uniform flash page sizes are required, defined to signal * that the peripheral does not implement a custom @ref flashpage_addr * function and instead relies on the generic helper function that * relies on @ref flashpage_size. */ #ifdef DOXYGEN #define PERIPH_FLASHPAGE_NEEDS_FLASHPAGE_ADDR #endif /** * @def PERIPH_FLASHPAGE_NEEDS_FLASHPAGE_PAGE * * @brief If non-uniform flash page sizes are required, defined to signal * that the peripheral does not implement a custom @ref flashpage_page * function and instead relies on the generic helper function that * relies on @ref flashpage_size. */ #ifdef DOXYGEN #define PERIPH_FLASHPAGE_NEEDS_FLASHPAGE_PAGE #endif /** * @brief Return values used in this interface */ enum { FLASHPAGE_OK = 0, /**< everything succeeded */ FLASHPAGE_NOMATCH = -1 /**< page differs from target data */ }; #if !defined(PERIPH_FLASHPAGE_CUSTOM_PAGESIZES) || defined(DOXYGEN) /** * @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 #ifdef MODULE_PERIPH_FLASHPAGE_IN_ADDRESS_SPACE /** * @def FLASH_WRITABLE_INIT(name, size) * @brief Define an array in flash memory * * This macro defines an array stored in the ".flash_writable" section * which is part of flash memory. With this macro it is possible to * reserve flash memory at build time. * * E.g. FLASH_WRITABLE_INIT(a, 2); will create a array with name 'a' * of size 2 * @ref FLASHPAGE_SIZE which is stored in flash memory and takes up * 2 flash pages. * * Symbols created by using this macro are sorted in ascending order by name. * Therefore, &a < &b where a and b are arrays created using this macro. * * @param[in] name name of the array * @param[in] size size of the array in unit of @ref FLASHPAGE_SIZE */ #define FLASH_WRITABLE_INIT(name, size) \ __attribute__((aligned(FLASHPAGE_SIZE))) \ __attribute__((section(".flash_writable." #name))) \ static const uint8_t name [size * FLASHPAGE_SIZE] #endif /** * @brief Get the page size of the given page number * * @param[in] page page number to get the size for * * @return Page size of the given page */ static inline size_t flashpage_size(unsigned page) { (void)page; return FLASHPAGE_SIZE; } /** * @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(unsigned 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 unsigned flashpage_page(const void *addr) { return (((intptr_t)addr - CPU_FLASH_BASE) / FLASHPAGE_SIZE); } #else /* Bare prototypes for the above functions. See above for the documentation */ size_t flashpage_size(unsigned page); void *flashpage_addr(unsigned page); unsigned flashpage_page(const void *addr); #endif /** * @brief Erase the given page * * @param[in] page Page to erase */ void flashpage_erase(unsigned page); /** * @brief Get number of first free flashpage * * If riotboot is used in two slot mode, this number will change across * firmware updates as the firmware slots alternate. */ unsigned flashpage_first_free(void); /** * @brief Get number of last free flashpage * * If riotboot is used in two slot mode, this number will change across * firmware updates as the firmware slots alternate. */ unsigned flashpage_last_free(void); /** * @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_page(unsigned 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_BLOCK_ALIGN. @p len must be a multiple of FLASHPAGE_WRITE_BLOCK_SIZE. * 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_WRITE_BLOCK_ALIGNMENT. * @param[in] data data to write to the address. MUST be aligned * to FLASHPAGE_WRITE_BLOCK_ALIGNMENT. * @param[in] len length of the data to be written. It MUST be * multiple of FLASHPAGE_WRITE_BLOCK_SIZE. Also, * ensure it doesn't exceed the actual flash * memory size. */ void flashpage_write(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(unsigned 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(unsigned 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(unsigned page, const void *data); /** * @brief Functions to support additional RWWEE flash section * * Some CPUs (ie. SAML21, SAML11) have an additional separated flash section * named RWWEE (Read While Write EEPROM Emulation) that can be accessed from * RIOT in a similar fashion but with a set of functions ending with a * _rwwee name variant. * * The support of this feature is determined by the FLASHPAGE_RWWEE_NUMOF * define that, if defined, contains the number of RWWEE sectors in the CPU. * This is defined in the CPU specific headers (ie. for the ATMEL SAM family * in cpu/sam0_common/include/cpu_conf.h) */ #ifdef FLASHPAGE_RWWEE_NUMOF /** * @brief Translate the given RWEE 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 RWWEE page number to get the address of * * @return starting memory address of the given RWWEE page */ static inline void *flashpage_rwwee_addr(unsigned page) { return (void *)(CPU_FLASH_RWWEE_BASE + (page * FLASHPAGE_SIZE)); } /** * @brief Translate the given address into the corresponding RWWEE page number * * The given address can be any address inside a RWWEE page. * * @note The given address MUST be a valid RWWEE flash address! * * @param[in] addr address inside the targeted RWWEE page * * @return RWWEE page containing the given address */ static inline int flashpage_rwwee_page(const void *addr) { return (int)(((int)addr - CPU_FLASH_RWWEE_BASE) / FLASHPAGE_SIZE); } /** * @brief Write the given RWWEE page with the given data * * @param[in] page RWWEE page to write * @param[in] data data to write to the RWWEE page, MUST be FLASHPAGE_SIZE * byte. Set to NULL for RWWEE page erase only. */ void flashpage_rwwee_write_page(unsigned page, const void *data); /** * @brief Write any number of data bytes to a given location in the * RWWEE 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_BLOCK_ALIGN. @p len must be a multiple of FLASHPAGE_WRITE_BLOCK_SIZE. * 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 RWWEE address in flash to write to. MUST be aligned * to FLASHPAGE_WRITE_BLOCK_ALIGNMENT. * @param[in] data data to write to the address. MUST be aligned * to FLASHPAGE_WRITE_BLOCK_ALIGNMENT. * @param[in] len length of the data to be written. It MUST be * multiple of FLASHPAGE_WRITE_BLOCK_SIZE. Also, * ensure it doesn't exceed the actual RWWEE flash * memory size. */ void flashpage_rwwee_write(void *target_addr, const void *data, size_t len); /** * @brief Read the given RWWEE page into the given memory location * * @param[in] page RWWEE page to read * @param[out] data memory to write the RWWEE page to, MUST be FLASHPAGE_SIZE * byte */ void flashpage_rwwee_read(unsigned page, void *data); /** * @brief Verify the given RWWEE page against the given data * * @param[in] page RWWEE page to verify * @param[in] data data to compare RWWEE page against, MUST be FLASHPAGE_SIZE * byte of data * * @return FLASHPAGE_OK if data in the RWWEE page is identical to @p data * @return FLASHPAGE_NOMATCH if data and RWWEE page content diverge */ int flashpage_rwwee_verify(unsigned page, const void *data); /** * @brief Write the given RWWEE page and verify the results * * This is a convenience function wrapping flashpage_rwwee_write and * flashpage_rwwee_verify. * * @param[in] page RWWEE page to write * @param[in] data data to write to the RWWEE page, MUST be FLASHPAGE_SIZE * byte. * * @return FLASHPAGE_OK on success * @return FLASHPAGE_NOMATCH if data and RWWEE page content diverge */ int flashpage_rwwee_write_and_verify(unsigned page, const void *data); #endif /* FLASHPAGE_RWWEE_NUMOF */ #ifdef __cplusplus } #endif #endif /* PERIPH_FLASHPAGE_H */ /** @} */