/* * Copyright (C) 2016 Freie Universität Berlin * * 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. */ /** * @ingroup drivers_periph_flashpage * @{ * * @file * @brief Common flash page functions * * @author Hauke Petersen * * @} */ #include #include "cpu.h" #include "assert.h" /* guard this file, must be done before including periph/flashpage.h * TODO: remove as soon as periph drivers can be build selectively */ #if defined(FLASHPAGE_SIZE) || defined(PERIPH_FLASHPAGE_CUSTOM_PAGESIZES) #include "periph/flashpage.h" #endif #ifdef MODULE_PERIPH_FLASHPAGE_PAGEWISE void flashpage_read(unsigned page, void *data) { assert(page < FLASHPAGE_NUMOF); #if defined(CPU_FAM_STM32WB) || (defined(CPU_FAM_STM32WL) && \ !defined(CPU_LINE_STM32WLE5xx)) assert(page < (FLASH->SFR & FLASH_SFR_SFSA)); #endif memcpy(data, flashpage_addr(page), flashpage_size(page)); } int flashpage_verify(unsigned page, const void *data) { assert(page < (int)FLASHPAGE_NUMOF); #if defined(CPU_FAM_STM32WB) || (defined(CPU_FAM_STM32WL) && \ !defined(CPU_LINE_STM32WLE5xx)) assert(page < (int)(FLASH->SFR & FLASH_SFR_SFSA)); #endif if (memcmp(flashpage_addr(page), data, flashpage_size(page)) == 0) { return FLASHPAGE_OK; } else { return FLASHPAGE_NOMATCH; } } int flashpage_write_and_verify(unsigned page, const void *data) { flashpage_write_page(page, data); return flashpage_verify(page, data); } void flashpage_write_page(unsigned page, const void *data) { assert((unsigned) page < FLASHPAGE_NUMOF); flashpage_erase(page); /* write page */ if (data != NULL) { flashpage_write(flashpage_addr(page), data, FLASHPAGE_SIZE); } } #endif /* MODULE_PERIPH_FLASHPAGE_PAGEWISE */ #if defined(FLASHPAGE_RWWEE_NUMOF) void flashpage_rwwee_read(unsigned page, void *data) { assert(page < (int)FLASHPAGE_RWWEE_NUMOF); memcpy(data, flashpage_rwwee_addr(page), FLASHPAGE_SIZE); } int flashpage_rwwee_verify(unsigned page, const void *data) { assert(page < (int)FLASHPAGE_RWWEE_NUMOF); if (memcmp(flashpage_rwwee_addr(page), data, FLASHPAGE_SIZE) == 0) { return FLASHPAGE_OK; } else { return FLASHPAGE_NOMATCH; } } int flashpage_rwwee_write_and_verify(unsigned page, const void *data) { flashpage_rwwee_write_page(page, data); return flashpage_rwwee_verify(page, data); } #endif /* FLASHPAGE_RWWEE_NUMOF */ #ifdef PERIPH_FLASHPAGE_NEEDS_FLASHPAGE_ADDR void *flashpage_addr(unsigned page) { uintptr_t addr = CPU_FLASH_BASE; while (page) { addr += flashpage_size(--page); } return (void*)addr; } #endif /* PERIPH_FLASHPAGE_NEEDS_FLASHPAGE_ADDR */ #ifdef PERIPH_FLASHPAGE_NEEDS_FLASHPAGE_PAGE unsigned flashpage_page(void *addr) { unsigned page = 0; for (uintptr_t pos = CPU_FLASH_BASE; (uintptr_t)addr >= pos; ++page) { pos += flashpage_size(page); } return page - 1; } #endif /* PERIPH_FLASHPAGE_NEEDS_FLASHPAGE_PAGE */