2016-04-20 10:23:12 +02:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2016 Freie Universität Berlin
|
2017-11-08 23:00:29 +01:00
|
|
|
* 2017 Inria
|
2016-04-20 10:23:12 +02:00
|
|
|
*
|
|
|
|
* 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!
|
|
|
|
*
|
2017-06-06 13:47:25 +02:00
|
|
|
* @{
|
|
|
|
* @file
|
|
|
|
* @brief Low-level flash page peripheral driver interface
|
|
|
|
*
|
2016-04-20 10:23:12 +02:00
|
|
|
* @author Hauke Petersen <hauke.petersen@fu-berlin.de>
|
2017-11-08 23:00:29 +01:00
|
|
|
* @author Kaspar Schleiser <kaspar@schleiser.de>
|
|
|
|
* @author Francisco Acosta <francisco.acosta@inria.fr>
|
|
|
|
*
|
2016-04-20 10:23:12 +02:00
|
|
|
*/
|
|
|
|
|
2017-05-23 18:19:52 +02:00
|
|
|
#ifndef PERIPH_FLASHPAGE_H
|
|
|
|
#define PERIPH_FLASHPAGE_H
|
2016-04-20 10:23:12 +02:00
|
|
|
|
|
|
|
#include <stdint.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
|
2017-11-08 23:00:29 +01:00
|
|
|
/**
|
|
|
|
* @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
|
2016-04-20 10:23:12 +02:00
|
|
|
|
|
|
|
/**
|
2017-11-08 23:00:29 +01:00
|
|
|
* @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
|
|
|
|
*
|
2016-04-20 10:23:12 +02:00
|
|
|
* @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));
|
|
|
|
}
|
|
|
|
|
2016-11-30 15:17:08 +01:00
|
|
|
/**
|
|
|
|
* @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);
|
|
|
|
}
|
|
|
|
|
2016-04-20 10:23:12 +02:00
|
|
|
/**
|
|
|
|
* @brief Write the given page with the given data
|
|
|
|
*
|
|
|
|
* @param[in] page page to write
|
2017-11-08 23:00:29 +01:00
|
|
|
* @param[in] data data to write to the page, MUST be FLASHPAGE_SIZE
|
2016-04-20 10:23:12 +02:00
|
|
|
* byte. Set to NULL for page erase only.
|
|
|
|
*/
|
|
|
|
void flashpage_write(int page, void *data);
|
|
|
|
|
2017-11-08 23:00:29 +01:00
|
|
|
/**
|
|
|
|
* @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, void *data, size_t len);
|
|
|
|
|
2016-04-20 10:23:12 +02:00
|
|
|
/**
|
|
|
|
* @brief Read the given page into the given memory location
|
|
|
|
*
|
|
|
|
* @param[in] page page to read
|
2017-11-08 23:00:29 +01:00
|
|
|
* @param[out] data memory to write the page to, MUST be FLASHPAGE_SIZE
|
2016-04-20 10:23:12 +02:00
|
|
|
* byte
|
|
|
|
*/
|
|
|
|
void flashpage_read(int page, void *data);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Verify the given page against the given data
|
|
|
|
*
|
|
|
|
* @param[in] page page to verify
|
2017-11-08 23:00:29 +01:00
|
|
|
* @param[in] data data to compare page against, MUST be FLASHPAGE_SIZE
|
2016-04-20 10:23:12 +02:00
|
|
|
* 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, 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
|
2017-11-08 23:00:29 +01:00
|
|
|
* @param[in] data data to write to the page, MUST be FLASHPAGE_SIZE
|
2016-04-20 10:23:12 +02:00
|
|
|
* byte.
|
|
|
|
*
|
|
|
|
* @return FLASHPAGE_OK on success
|
|
|
|
* @return FLASHPAGE_NOMATCH if data and page content diverge
|
|
|
|
*/
|
|
|
|
int flashpage_write_and_verify(int page, void *data);
|
|
|
|
|
|
|
|
#ifdef __cplusplus
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2017-05-23 18:19:52 +02:00
|
|
|
#endif /* PERIPH_FLASHPAGE_H */
|
2016-04-20 10:23:12 +02:00
|
|
|
/** @} */
|