/* * Copyright (C) 2021 ML!PA Consulting GmbH * * 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 cpu_native * @ingroup drivers_periph_flashpage * @{ * * @file * @brief Low-level flashpage driver emulation * * @author Benjamin Valentin * @} */ #include #include #include "cpu.h" #include "periph/flashpage.h" #define ENABLE_DEBUG 0 #include "debug.h" __attribute__((aligned(FLASHPAGE_SIZE * FLASHPAGE_NUMOF))) char _native_flash[FLASHPAGE_SIZE * FLASHPAGE_NUMOF]; void flashpage_erase(unsigned page) { assert(page < FLASHPAGE_NUMOF); DEBUG("%p: erase %u bytes\n", flashpage_addr(page), FLASHPAGE_SIZE); memset(flashpage_addr(page), FLASHPAGE_ERASE_STATE, FLASHPAGE_SIZE); } static void _flash_write(uint8_t *dst, const char *src, size_t len) { while (len--) { #if FLASHPAGE_ERASE_STATE == 0x0 *dst++ |= *src++; #else *dst++ &= *src++; #endif } } void flashpage_write(void *target_addr, const void *data, size_t len) { assert((uintptr_t)target_addr >= (uintptr_t)_native_flash); assert((uintptr_t)target_addr + len <= (uintptr_t)_native_flash + sizeof(_native_flash)); assert(!(len % FLASHPAGE_WRITE_BLOCK_SIZE)); assert(!((uintptr_t)target_addr % FLASHPAGE_WRITE_BLOCK_ALIGNMENT)); DEBUG("%p: write %zu bytes\n", target_addr, len); _flash_write(target_addr, data, len); } unsigned flashpage_first_free(void) { return flashpage_page(&_native_flash[0]); } unsigned flashpage_last_free(void) { return flashpage_page(&_native_flash[FLASHPAGE_SIZE * FLASHPAGE_NUMOF - 1]); }