1
0
mirror of https://github.com/RIOT-OS/RIOT.git synced 2025-01-18 06:52:44 +01:00
RIOT/drivers/include/periph/flashpage.h

444 lines
14 KiB
C
Raw Normal View History

/*
* 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 <hauke.petersen@fu-berlin.de>
* @author Kaspar Schleiser <kaspar@schleiser.de>
* @author Francisco Acosta <francisco.acosta@inria.fr>
2019-01-27 11:29:35 +01:00
* @author Federico Pellegrin <fede@evolware.org>
*
*/
#ifndef PERIPH_FLASHPAGE_H
#define PERIPH_FLASHPAGE_H
2021-08-08 13:34:57 +02:00
#include <stddef.h>
#include <stdint.h>
2019-01-27 11:29:35 +01:00
#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);
2021-10-19 22:33:09 +02:00
/**
* @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.
2021-10-19 22:33:09 +02:00
*/
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.
2021-10-19 22:33:09 +02:00
*/
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);
2019-01-27 11:29:35 +01:00
/**
* @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)
2019-01-27 11:29:35 +01:00
{
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)
2019-01-27 11:29:35 +01:00
{
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);
2019-01-27 11:29:35 +01:00
/**
* @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.
2019-01-27 11:29:35 +01:00
* 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.
2019-01-27 11:29:35 +01:00
* @param[in] data data to write to the address. MUST be aligned
* to FLASHPAGE_WRITE_BLOCK_ALIGNMENT.
2019-01-27 11:29:35 +01:00
* @param[in] len length of the data to be written. It MUST be
* multiple of FLASHPAGE_WRITE_BLOCK_SIZE. Also,
2019-01-27 11:29:35 +01:00
* ensure it doesn't exceed the actual RWWEE flash
* memory size.
*/
void flashpage_rwwee_write(void *target_addr, const void *data, size_t len);
2019-01-27 11:29:35 +01:00
/**
* @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);
2019-01-27 11:29:35 +01:00
/**
* @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);
2019-01-27 11:29:35 +01:00
/**
* @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);
2019-01-27 11:29:35 +01:00
#endif /* FLASHPAGE_RWWEE_NUMOF */
#ifdef __cplusplus
}
#endif
#endif /* PERIPH_FLASHPAGE_H */
/** @} */