2018-03-13 14:11:12 +01:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2018 OTA keys S.A.
|
|
|
|
*
|
|
|
|
* 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_mtd_flashpage
|
|
|
|
* @brief Driver for internal flash devices implementing flashpage interface
|
|
|
|
*
|
|
|
|
* @{
|
|
|
|
*
|
|
|
|
* @file
|
|
|
|
* @brief Implementation for the flashpage memory driver
|
|
|
|
*
|
|
|
|
* @author Vincent Dupont <vincent@otakeys.com>
|
|
|
|
* @}
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <string.h>
|
|
|
|
#include <errno.h>
|
|
|
|
#include <assert.h>
|
|
|
|
|
2021-02-16 18:20:20 +01:00
|
|
|
#include "architecture.h"
|
2021-02-22 10:24:39 +01:00
|
|
|
#include "cpu.h"
|
2018-03-13 14:11:12 +01:00
|
|
|
#include "cpu_conf.h"
|
|
|
|
#include "mtd_flashpage.h"
|
|
|
|
#include "periph/flashpage.h"
|
|
|
|
|
2019-10-03 23:49:40 +02:00
|
|
|
#define MTD_FLASHPAGE_END_ADDR ((uint32_t) CPU_FLASH_BASE + (FLASHPAGE_NUMOF * FLASHPAGE_SIZE))
|
2018-03-13 14:11:12 +01:00
|
|
|
|
|
|
|
static int _init(mtd_dev_t *dev)
|
|
|
|
{
|
|
|
|
(void)dev;
|
|
|
|
assert(dev->pages_per_sector * dev->page_size == FLASHPAGE_SIZE);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int _read(mtd_dev_t *dev, void *buf, uint32_t addr, uint32_t size)
|
|
|
|
{
|
|
|
|
assert(addr < MTD_FLASHPAGE_END_ADDR);
|
2019-10-03 23:49:40 +02:00
|
|
|
|
2018-03-13 14:11:12 +01:00
|
|
|
(void)dev;
|
|
|
|
|
2021-02-23 09:47:58 +01:00
|
|
|
#ifndef CPU_HAS_UNALIGNED_ACCESS
|
|
|
|
if (addr % sizeof(uword_t)) {
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2021-02-16 18:20:20 +01:00
|
|
|
uword_t dst_addr = addr;
|
2019-10-03 23:49:40 +02:00
|
|
|
memcpy(buf, (void *)dst_addr, size);
|
2018-03-13 14:11:12 +01:00
|
|
|
|
2020-04-17 19:41:43 +02:00
|
|
|
return 0;
|
2018-03-13 14:11:12 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
static int _write(mtd_dev_t *dev, const void *buf, uint32_t addr, uint32_t size)
|
|
|
|
{
|
|
|
|
(void)dev;
|
2019-10-03 23:49:40 +02:00
|
|
|
|
2021-02-22 10:24:39 +01:00
|
|
|
#ifndef CPU_HAS_UNALIGNED_ACCESS
|
|
|
|
if ((uintptr_t)buf % sizeof(uword_t)) {
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
#endif
|
2020-11-09 16:45:35 +01:00
|
|
|
if (addr % FLASHPAGE_WRITE_BLOCK_ALIGNMENT) {
|
2018-03-13 14:11:12 +01:00
|
|
|
return -EINVAL;
|
|
|
|
}
|
2020-11-09 16:45:35 +01:00
|
|
|
if (size % FLASHPAGE_WRITE_BLOCK_SIZE) {
|
2018-03-13 14:11:12 +01:00
|
|
|
return -EOVERFLOW;
|
|
|
|
}
|
|
|
|
if (addr + size > MTD_FLASHPAGE_END_ADDR) {
|
|
|
|
return -EOVERFLOW;
|
|
|
|
}
|
2019-10-03 23:49:40 +02:00
|
|
|
|
2021-02-16 18:20:20 +01:00
|
|
|
uword_t dst_addr = addr;
|
2020-11-09 16:45:35 +01:00
|
|
|
flashpage_write((void *)dst_addr, buf, size);
|
2018-03-13 14:11:12 +01:00
|
|
|
|
2020-04-17 19:41:43 +02:00
|
|
|
return 0;
|
2018-03-13 14:11:12 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
int _erase(mtd_dev_t *dev, uint32_t addr, uint32_t size)
|
|
|
|
{
|
|
|
|
size_t sector_size = dev->page_size * dev->pages_per_sector;
|
|
|
|
|
|
|
|
if (size % sector_size) {
|
|
|
|
return -EOVERFLOW;
|
|
|
|
}
|
|
|
|
if (addr + size > MTD_FLASHPAGE_END_ADDR) {
|
2019-10-03 23:49:40 +02:00
|
|
|
return -EOVERFLOW;
|
2018-03-13 14:11:12 +01:00
|
|
|
}
|
|
|
|
if (addr % sector_size) {
|
2019-10-03 23:49:40 +02:00
|
|
|
return -EOVERFLOW;
|
2018-03-13 14:11:12 +01:00
|
|
|
}
|
2019-10-03 23:49:40 +02:00
|
|
|
|
2021-02-16 18:20:20 +01:00
|
|
|
uword_t dst_addr = addr;
|
2019-10-03 23:49:40 +02:00
|
|
|
|
2018-03-13 14:11:12 +01:00
|
|
|
for (size_t i = 0; i < size; i += sector_size) {
|
2020-11-09 16:45:35 +01:00
|
|
|
flashpage_erase(flashpage_page((void *)(dst_addr + i)));
|
2018-03-13 14:11:12 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
const mtd_desc_t mtd_flashpage_driver = {
|
|
|
|
.init = _init,
|
|
|
|
.read = _read,
|
|
|
|
.write = _write,
|
|
|
|
.erase = _erase,
|
|
|
|
};
|