2016-07-15 10:40:56 +02:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2016 Eistec AB
|
|
|
|
* 2017 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_spi_nor
|
|
|
|
* @{
|
|
|
|
*
|
|
|
|
* @file
|
|
|
|
* @brief Driver for serial flash memory attached to SPI
|
|
|
|
*
|
|
|
|
* @author Joakim Nohlgård <joakim.nohlgard@eistec.se>
|
|
|
|
* @author Vincent Dupont <vincent@otakeys.com>
|
|
|
|
*
|
|
|
|
* @}
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <stdint.h>
|
2020-10-25 20:28:16 +01:00
|
|
|
#include <string.h>
|
2016-07-15 10:40:56 +02:00
|
|
|
#include <errno.h>
|
|
|
|
|
2024-01-15 14:37:46 +01:00
|
|
|
#include "busy_wait.h"
|
2023-01-07 01:02:12 +01:00
|
|
|
#include "byteorder.h"
|
2022-03-22 08:29:25 +01:00
|
|
|
#include "kernel_defines.h"
|
2023-08-31 14:32:56 +02:00
|
|
|
#include "macros/math.h"
|
2023-01-07 01:02:12 +01:00
|
|
|
#include "macros/utils.h"
|
2016-07-15 10:40:56 +02:00
|
|
|
#include "mtd.h"
|
2023-01-07 01:02:12 +01:00
|
|
|
#include "mtd_spi_nor.h"
|
2023-08-31 14:32:56 +02:00
|
|
|
#include "time_units.h"
|
2023-01-07 01:02:12 +01:00
|
|
|
#include "thread.h"
|
|
|
|
|
2023-08-31 14:32:56 +02:00
|
|
|
#if IS_USED(MODULE_ZTIMER)
|
2022-03-22 08:29:25 +01:00
|
|
|
#include "ztimer.h"
|
|
|
|
#elif IS_USED(MODULE_XTIMER)
|
2016-07-15 10:40:56 +02:00
|
|
|
#include "xtimer.h"
|
2022-03-22 08:29:25 +01:00
|
|
|
#endif
|
2016-07-15 10:40:56 +02:00
|
|
|
|
2020-10-22 11:34:31 +02:00
|
|
|
#define ENABLE_DEBUG 0
|
2016-07-15 10:40:56 +02:00
|
|
|
#include "debug.h"
|
|
|
|
|
2020-10-22 11:34:31 +02:00
|
|
|
#define ENABLE_TRACE 0
|
|
|
|
#define TRACE(...) DEBUG(__VA_ARGS__)
|
2016-07-15 10:40:56 +02:00
|
|
|
|
2020-03-05 10:09:13 +01:00
|
|
|
/* after power up, on an invalid JEDEC ID, wait and read N times */
|
|
|
|
#ifndef MTD_POWER_UP_WAIT_FOR_ID
|
|
|
|
#define MTD_POWER_UP_WAIT_FOR_ID (0x0F)
|
|
|
|
#endif
|
|
|
|
|
2020-11-02 00:01:00 +01:00
|
|
|
#define SFLASH_CMD_4_BYTE_ADDR (0xB7) /**< enable 32 bit addressing */
|
|
|
|
#define SFLASH_CMD_3_BYTE_ADDR (0xE9) /**< enable 24 bit addressing */
|
|
|
|
|
2022-02-03 21:30:46 +01:00
|
|
|
#define SFLASH_CMD_ULBPR (0x98) /**< Global Block Protection Unlock */
|
|
|
|
|
2020-10-25 23:23:48 +01:00
|
|
|
#define MTD_64K (65536ul)
|
|
|
|
#define MTD_64K_ADDR_MASK (0xFFFF)
|
2018-01-22 14:04:35 +01:00
|
|
|
#define MTD_32K (32768ul)
|
|
|
|
#define MTD_32K_ADDR_MASK (0x7FFF)
|
|
|
|
#define MTD_4K (4096ul)
|
|
|
|
#define MTD_4K_ADDR_MASK (0xFFF)
|
|
|
|
|
2020-05-14 22:06:22 +02:00
|
|
|
#define MBIT_AS_BYTES ((1024 * 1024) / 8)
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief JEDEC memory manufacturer ID codes.
|
|
|
|
*
|
|
|
|
* see http://www.softnology.biz/pdf/JEP106AV.pdf
|
|
|
|
* @{
|
|
|
|
*/
|
|
|
|
#define JEDEC_BANK(n) ((n) << 8)
|
|
|
|
|
|
|
|
typedef enum {
|
2020-10-25 20:51:11 +01:00
|
|
|
SPI_NOR_JEDEC_ATMEL = 0x1F | JEDEC_BANK(1),
|
2022-02-03 21:30:46 +01:00
|
|
|
SPI_NOR_JEDEC_MICROCHIP = 0xBF | JEDEC_BANK(1),
|
2020-05-14 22:06:22 +02:00
|
|
|
} jedec_manuf_t;
|
|
|
|
/** @} */
|
|
|
|
|
2020-10-25 20:51:11 +01:00
|
|
|
static inline spi_t _get_spi(const mtd_spi_nor_t *dev)
|
|
|
|
{
|
|
|
|
return dev->params->spi;
|
|
|
|
}
|
|
|
|
|
2020-02-10 20:39:49 +01:00
|
|
|
static void mtd_spi_acquire(const mtd_spi_nor_t *dev)
|
|
|
|
{
|
2020-10-25 20:51:11 +01:00
|
|
|
spi_acquire(_get_spi(dev), dev->params->cs,
|
2020-02-10 20:39:49 +01:00
|
|
|
dev->params->mode, dev->params->clk);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void mtd_spi_release(const mtd_spi_nor_t *dev)
|
|
|
|
{
|
2020-10-25 20:51:11 +01:00
|
|
|
spi_release(_get_spi(dev));
|
2020-05-14 22:06:22 +02:00
|
|
|
}
|
|
|
|
|
2020-10-25 21:41:33 +01:00
|
|
|
static inline uint8_t* _be_addr(const mtd_spi_nor_t *dev, uint32_t *addr)
|
|
|
|
{
|
|
|
|
*addr = htonl(*addr);
|
2022-02-03 21:52:19 +01:00
|
|
|
return &((uint8_t*)addr)[4 - dev->addr_width];
|
2020-10-25 21:41:33 +01:00
|
|
|
}
|
|
|
|
|
2016-07-15 10:40:56 +02:00
|
|
|
/**
|
|
|
|
* @internal
|
|
|
|
* @brief Send command opcode followed by address, followed by a read to buffer
|
|
|
|
*
|
|
|
|
* @param[in] dev pointer to device descriptor
|
|
|
|
* @param[in] opcode command opcode
|
|
|
|
* @param[in] addr address (big endian)
|
|
|
|
* @param[out] dest read buffer
|
|
|
|
* @param[in] count number of bytes to read after the address has been sent
|
|
|
|
*/
|
2017-06-29 22:46:16 +02:00
|
|
|
static void mtd_spi_cmd_addr_read(const mtd_spi_nor_t *dev, uint8_t opcode,
|
2020-10-25 21:41:33 +01:00
|
|
|
uint32_t addr, void *dest, uint32_t count)
|
2016-07-15 10:40:56 +02:00
|
|
|
{
|
2020-10-25 21:41:33 +01:00
|
|
|
TRACE("mtd_spi_cmd_addr_read: %p, %02x, (%06"PRIx32"), %p, %" PRIu32 "\n",
|
|
|
|
(void *)dev, (unsigned int)opcode, addr, dest, count);
|
|
|
|
|
|
|
|
uint8_t *addr_buf = _be_addr(dev, &addr);
|
2016-07-15 10:40:56 +02:00
|
|
|
|
2020-10-23 00:40:33 +02:00
|
|
|
if (IS_ACTIVE(ENABLE_TRACE)) {
|
2016-07-15 10:40:56 +02:00
|
|
|
TRACE("mtd_spi_cmd_addr_read: addr:");
|
2022-02-03 21:52:19 +01:00
|
|
|
for (unsigned int i = 0; i < dev->addr_width; ++i) {
|
2016-07-15 10:40:56 +02:00
|
|
|
TRACE(" %02x", addr_buf[i]);
|
|
|
|
}
|
|
|
|
TRACE("\n");
|
|
|
|
}
|
|
|
|
|
2020-10-25 19:27:29 +01:00
|
|
|
/* Send opcode followed by address */
|
2020-10-25 20:51:11 +01:00
|
|
|
spi_transfer_byte(_get_spi(dev), dev->params->cs, true, opcode);
|
|
|
|
spi_transfer_bytes(_get_spi(dev), dev->params->cs, true,
|
2022-02-03 21:52:19 +01:00
|
|
|
(char *)addr_buf, NULL, dev->addr_width);
|
2020-10-25 19:27:29 +01:00
|
|
|
|
|
|
|
/* Read data */
|
2020-10-25 20:51:11 +01:00
|
|
|
spi_transfer_bytes(_get_spi(dev), dev->params->cs, false,
|
2020-10-25 19:27:29 +01:00
|
|
|
NULL, dest, count);
|
2016-07-15 10:40:56 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @internal
|
|
|
|
* @brief Send command opcode followed by address, followed by a write from buffer
|
|
|
|
*
|
|
|
|
* @param[in] dev pointer to device descriptor
|
|
|
|
* @param[in] opcode command opcode
|
|
|
|
* @param[in] addr address (big endian)
|
|
|
|
* @param[out] src write buffer
|
|
|
|
* @param[in] count number of bytes to write after the opcode has been sent
|
|
|
|
*/
|
2017-06-29 22:46:16 +02:00
|
|
|
static void mtd_spi_cmd_addr_write(const mtd_spi_nor_t *dev, uint8_t opcode,
|
2020-10-25 21:41:33 +01:00
|
|
|
uint32_t addr, const void *src, uint32_t count)
|
2016-07-15 10:40:56 +02:00
|
|
|
{
|
2020-10-25 21:41:33 +01:00
|
|
|
TRACE("mtd_spi_cmd_addr_write: %p, %02x, (%06"PRIx32"), %p, %" PRIu32 "\n",
|
|
|
|
(void *)dev, (unsigned int)opcode, addr, src, count);
|
|
|
|
|
|
|
|
uint8_t *addr_buf = _be_addr(dev, &addr);
|
2016-07-15 10:40:56 +02:00
|
|
|
|
2020-10-23 00:40:33 +02:00
|
|
|
if (IS_ACTIVE(ENABLE_TRACE)) {
|
2016-07-15 10:40:56 +02:00
|
|
|
TRACE("mtd_spi_cmd_addr_write: addr:");
|
2022-02-03 21:52:19 +01:00
|
|
|
for (unsigned int i = 0; i < dev->addr_width; ++i) {
|
2016-07-15 10:40:56 +02:00
|
|
|
TRACE(" %02x", addr_buf[i]);
|
|
|
|
}
|
|
|
|
TRACE("\n");
|
|
|
|
}
|
|
|
|
|
2020-10-25 19:27:29 +01:00
|
|
|
/* Send opcode followed by address */
|
2020-10-25 20:51:11 +01:00
|
|
|
spi_transfer_byte(_get_spi(dev), dev->params->cs, true, opcode);
|
2020-10-25 19:27:29 +01:00
|
|
|
|
|
|
|
/* only keep CS asserted when there is data that follows */
|
|
|
|
bool cont = (count > 0);
|
2020-10-25 20:51:11 +01:00
|
|
|
spi_transfer_bytes(_get_spi(dev), dev->params->cs, cont,
|
2022-02-03 21:52:19 +01:00
|
|
|
(char *)addr_buf, NULL, dev->addr_width);
|
2020-10-25 19:27:29 +01:00
|
|
|
|
|
|
|
/* Write data */
|
|
|
|
if (cont) {
|
2020-10-25 20:51:11 +01:00
|
|
|
spi_transfer_bytes(_get_spi(dev), dev->params->cs,
|
2020-10-25 19:27:29 +01:00
|
|
|
false, (void *)src, NULL, count);
|
|
|
|
}
|
2016-07-15 10:40:56 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @internal
|
|
|
|
* @brief Send command opcode followed by a read to buffer
|
|
|
|
*
|
|
|
|
* @param[in] dev pointer to device descriptor
|
|
|
|
* @param[in] opcode command opcode
|
|
|
|
* @param[out] dest read buffer
|
|
|
|
* @param[in] count number of bytes to write after the opcode has been sent
|
|
|
|
*/
|
2018-03-05 15:54:08 +01:00
|
|
|
static void mtd_spi_cmd_read(const mtd_spi_nor_t *dev, uint8_t opcode, void *dest, uint32_t count)
|
2016-07-15 10:40:56 +02:00
|
|
|
{
|
|
|
|
TRACE("mtd_spi_cmd_read: %p, %02x, %p, %" PRIu32 "\n",
|
2018-03-05 15:54:08 +01:00
|
|
|
(void *)dev, (unsigned int)opcode, dest, count);
|
2016-07-15 10:40:56 +02:00
|
|
|
|
2020-10-25 20:51:11 +01:00
|
|
|
spi_transfer_regs(_get_spi(dev), dev->params->cs, opcode, NULL, dest, count);
|
2016-07-15 10:40:56 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @internal
|
|
|
|
* @brief Send command opcode followed by a write from buffer
|
|
|
|
*
|
|
|
|
* @param[in] dev pointer to device descriptor
|
|
|
|
* @param[in] opcode command opcode
|
|
|
|
* @param[out] src write buffer
|
|
|
|
* @param[in] count number of bytes to write after the opcode has been sent
|
|
|
|
*/
|
2018-03-05 15:54:08 +01:00
|
|
|
static void __attribute__((unused)) mtd_spi_cmd_write(const mtd_spi_nor_t *dev, uint8_t opcode, const void *src, uint32_t count)
|
2016-07-15 10:40:56 +02:00
|
|
|
{
|
|
|
|
TRACE("mtd_spi_cmd_write: %p, %02x, %p, %" PRIu32 "\n",
|
2018-03-05 15:54:08 +01:00
|
|
|
(void *)dev, (unsigned int)opcode, src, count);
|
2016-07-15 10:40:56 +02:00
|
|
|
|
2020-10-25 20:51:11 +01:00
|
|
|
spi_transfer_regs(_get_spi(dev), dev->params->cs, opcode,
|
2020-02-10 20:39:49 +01:00
|
|
|
(void *)src, NULL, count);
|
2016-07-15 10:40:56 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @internal
|
|
|
|
* @brief Send command opcode
|
|
|
|
*
|
|
|
|
* @param[in] dev pointer to device descriptor
|
|
|
|
* @param[in] opcode command opcode
|
|
|
|
*/
|
2017-06-29 22:46:16 +02:00
|
|
|
static void mtd_spi_cmd(const mtd_spi_nor_t *dev, uint8_t opcode)
|
2016-07-15 10:40:56 +02:00
|
|
|
{
|
|
|
|
TRACE("mtd_spi_cmd: %p, %02x\n",
|
2018-03-05 15:54:08 +01:00
|
|
|
(void *)dev, (unsigned int)opcode);
|
2016-07-15 10:40:56 +02:00
|
|
|
|
2020-10-25 20:51:11 +01:00
|
|
|
spi_transfer_byte(_get_spi(dev), dev->params->cs, false, opcode);
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool mtd_spi_manuf_match(const mtd_jedec_id_t *id, jedec_manuf_t manuf)
|
|
|
|
{
|
|
|
|
return manuf == ((id->bank << 8) | id->manuf);
|
2016-07-15 10:40:56 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @internal
|
|
|
|
* @brief Compute 8 bit parity
|
|
|
|
*/
|
|
|
|
static inline uint8_t parity8(uint8_t x)
|
|
|
|
{
|
|
|
|
/* Taken from http://stackoverflow.com/a/21618038/1805713 */
|
|
|
|
x ^= x >> 4;
|
|
|
|
x ^= x >> 2;
|
|
|
|
x ^= x >> 1;
|
|
|
|
return (x & 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @internal
|
|
|
|
* @brief Read JEDEC ID
|
|
|
|
*/
|
2017-06-29 22:46:16 +02:00
|
|
|
static int mtd_spi_read_jedec_id(const mtd_spi_nor_t *dev, mtd_jedec_id_t *out)
|
2016-07-15 10:40:56 +02:00
|
|
|
{
|
2020-10-25 20:28:16 +01:00
|
|
|
uint8_t buffer[JEDEC_BANK_MAX + sizeof(mtd_jedec_id_t) - 1];
|
2016-07-15 10:40:56 +02:00
|
|
|
|
2020-02-10 20:39:49 +01:00
|
|
|
DEBUG("mtd_spi_read_jedec_id: rdid=0x%02x\n",
|
|
|
|
(unsigned int)dev->params->opcode->rdid);
|
2016-07-15 10:40:56 +02:00
|
|
|
|
|
|
|
/* Send opcode */
|
2020-10-25 20:28:16 +01:00
|
|
|
mtd_spi_cmd_read(dev, dev->params->opcode->rdid, buffer, sizeof(buffer));
|
|
|
|
|
|
|
|
/* Manufacturer IDs are organized in 'banks'.
|
|
|
|
* If we read the 'next bank' instead of manufacturer ID, skip
|
|
|
|
* the byte and increment the bank counter.
|
|
|
|
*/
|
|
|
|
uint8_t bank = 0;
|
|
|
|
while (buffer[bank] == JEDEC_NEXT_BANK) {
|
|
|
|
if (++bank == JEDEC_BANK_MAX) {
|
2021-12-08 18:30:43 +01:00
|
|
|
DEBUG_PUTS("mtd_spi_read_jedec_id: bank out of bounds\n");
|
2020-10-25 20:28:16 +01:00
|
|
|
return -1;
|
2016-07-15 10:40:56 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-10-25 20:28:16 +01:00
|
|
|
if (parity8(buffer[bank]) == 0) {
|
|
|
|
/* saw even parity, we expected odd parity => parity error */
|
|
|
|
DEBUG("mtd_spi_read_jedec_id: Parity error (0x%02x)\n", buffer[bank]);
|
|
|
|
return -2;
|
2016-07-15 10:40:56 +02:00
|
|
|
}
|
|
|
|
|
2020-10-25 20:28:16 +01:00
|
|
|
if (buffer[bank] == 0xFF || buffer[bank] == 0x00) {
|
|
|
|
DEBUG_PUTS("mtd_spi_read_jedec_id: failed to read manufacturer ID");
|
|
|
|
return -3;
|
2016-07-15 10:40:56 +02:00
|
|
|
}
|
|
|
|
|
2020-10-25 20:28:16 +01:00
|
|
|
/* Copy manufacturer ID */
|
|
|
|
out->bank = bank + 1;
|
|
|
|
memcpy((uint8_t*)out + 1, &buffer[bank], 3);
|
|
|
|
|
|
|
|
DEBUG("mtd_spi_read_jedec_id: bank=%u manuf=0x%02x\n", (unsigned int)out->bank,
|
|
|
|
(unsigned int)out->manuf);
|
|
|
|
|
|
|
|
DEBUG("mtd_spi_read_jedec_id: device=0x%02x, 0x%02x\n",
|
|
|
|
(unsigned int)out->device[0], (unsigned int)out->device[1]);
|
|
|
|
|
|
|
|
return 0;
|
2016-07-15 10:40:56 +02:00
|
|
|
}
|
|
|
|
|
2020-05-14 22:06:22 +02:00
|
|
|
/**
|
|
|
|
* @internal
|
|
|
|
* @brief Get Flash capacity based on JEDEC ID
|
|
|
|
*
|
|
|
|
* @note The way the capacity is encoded differs between vendors.
|
|
|
|
* This formula has been tested with flash chips from Adesto,
|
|
|
|
* ISSI, Micron and Spansion, but it might not cover all cases.
|
|
|
|
* Please extend the function if necessary.
|
|
|
|
*/
|
|
|
|
static uint32_t mtd_spi_nor_get_size(const mtd_jedec_id_t *id)
|
|
|
|
{
|
|
|
|
/* old Atmel (now Adesto) parts use 5 lower bits of device ID 1 for density */
|
|
|
|
if (mtd_spi_manuf_match(id, SPI_NOR_JEDEC_ATMEL) &&
|
|
|
|
/* ID 2 is used to encode the product version, usually 1 or 2 */
|
|
|
|
(id->device[1] & ~0x3) == 0) {
|
2022-02-23 18:10:55 +01:00
|
|
|
/* capacity encoded as power of 32k sectors */
|
|
|
|
return (32 * 1024) << (0x1F & id->device[0]);
|
2020-05-14 22:06:22 +02:00
|
|
|
}
|
2022-02-03 21:30:46 +01:00
|
|
|
if (mtd_spi_manuf_match(id, SPI_NOR_JEDEC_MICROCHIP)) {
|
|
|
|
switch (id->device[1]) {
|
|
|
|
case 0x12: /* SST26VF020A */
|
|
|
|
case 0x8c: /* SST25VF020B */
|
|
|
|
return 2 * MBIT_AS_BYTES;
|
|
|
|
case 0x54: /* SST26WF040B */
|
|
|
|
case 0x8d: /* SST25VF040B */
|
|
|
|
return 4 * MBIT_AS_BYTES;
|
|
|
|
case 0x58: /* SST26WF080B */
|
|
|
|
case 0x8e: /* SST25VF080B */
|
|
|
|
return 8 * MBIT_AS_BYTES;
|
|
|
|
case 0x1: /* SST26VF016 */
|
|
|
|
case 0x41: /* SST26VF016B */
|
|
|
|
return 16 * MBIT_AS_BYTES;
|
|
|
|
case 0x2: /* SST26VF032 */
|
|
|
|
case 0x42: /* SST26VF032B */
|
|
|
|
return 32 * MBIT_AS_BYTES;
|
|
|
|
case 0x43: /* SST26VF064B */
|
|
|
|
case 0x53: /* SST26WF064C */
|
|
|
|
return 64 * MBIT_AS_BYTES;
|
|
|
|
}
|
|
|
|
}
|
2020-05-14 22:06:22 +02:00
|
|
|
|
|
|
|
/* everyone else seems to use device ID 2 for density */
|
|
|
|
return 1 << id->device[1];
|
|
|
|
}
|
|
|
|
|
2024-01-15 14:37:46 +01:00
|
|
|
static void delay_us(unsigned us)
|
|
|
|
{
|
|
|
|
#if defined(MODULE_ZTIMER_USEC)
|
|
|
|
ztimer_sleep(ZTIMER_USEC, us);
|
|
|
|
#elif defined(MODULE_ZTIMER_MSEC)
|
|
|
|
ztimer_sleep(ZTIMER_MSEC, DIV_ROUND_UP(us, US_PER_MS));
|
|
|
|
#else
|
|
|
|
busy_wait_us(us);
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2018-06-14 14:15:11 +02:00
|
|
|
static inline void wait_for_write_complete(const mtd_spi_nor_t *dev, uint32_t us)
|
2016-07-15 10:40:56 +02:00
|
|
|
{
|
2018-06-14 14:15:11 +02:00
|
|
|
unsigned i = 0, j = 0;
|
2022-03-29 13:32:35 +02:00
|
|
|
uint32_t div = 1; /* first wait one full interval */
|
2022-03-22 08:29:25 +01:00
|
|
|
#if IS_ACTIVE(ENABLE_DEBUG)
|
2020-10-23 00:40:33 +02:00
|
|
|
uint32_t diff = 0;
|
2022-03-22 08:29:25 +01:00
|
|
|
#endif
|
|
|
|
#if IS_ACTIVE(ENABLE_DEBUG) && IS_USED(MODULE_ZTIMER_USEC)
|
|
|
|
diff = ztimer_now(ZTIMER_USEC);
|
|
|
|
#elif IS_ACTIVE(ENABLE_DEBUG) && IS_USED(MODULE_XTIMER)
|
|
|
|
diff = xtimer_now_usec();
|
|
|
|
#endif
|
2016-07-15 10:40:56 +02:00
|
|
|
do {
|
|
|
|
uint8_t status;
|
2020-02-10 20:39:49 +01:00
|
|
|
mtd_spi_cmd_read(dev, dev->params->opcode->rdsr, &status, sizeof(status));
|
2016-07-15 10:40:56 +02:00
|
|
|
|
|
|
|
TRACE("mtd_spi_nor: wait device status = 0x%02x\n", (unsigned int)status);
|
|
|
|
if ((status & 1) == 0) { /* TODO magic number */
|
|
|
|
break;
|
|
|
|
}
|
2018-06-14 14:15:11 +02:00
|
|
|
i++;
|
2022-03-22 08:29:25 +01:00
|
|
|
if (us) {
|
2022-03-29 13:32:35 +02:00
|
|
|
uint32_t wait_us = us / div;
|
|
|
|
uint32_t wait_min = 2;
|
|
|
|
|
|
|
|
wait_us = wait_us > wait_min ? wait_us : wait_min;
|
|
|
|
|
2024-01-15 14:37:46 +01:00
|
|
|
delay_us(wait_us);
|
2022-03-22 08:29:25 +01:00
|
|
|
/* reduce the waiting time quickly if the estimate was too short,
|
|
|
|
* but still avoid busy (yield) waiting */
|
2022-03-29 13:32:35 +02:00
|
|
|
div++;
|
2022-03-22 08:29:25 +01:00
|
|
|
}
|
|
|
|
else {
|
|
|
|
j++;
|
|
|
|
thread_yield();
|
|
|
|
}
|
2016-07-15 10:40:56 +02:00
|
|
|
} while (1);
|
2020-02-10 22:26:19 +01:00
|
|
|
DEBUG("wait loop %u times, yield %u times", i, j);
|
2022-03-22 08:29:25 +01:00
|
|
|
#if IS_ACTIVE(ENABLE_DEBUG)
|
|
|
|
#if IS_USED(MODULE_ZTIMER_USEC)
|
|
|
|
diff = ztimer_now(ZTIMER_USEC) - diff;
|
|
|
|
#elif IS_USED(MODULE_XTIMER)
|
|
|
|
diff = xtimer_now_usec() - diff;
|
|
|
|
#endif
|
|
|
|
DEBUG(", total wait %"PRIu32"us", diff);
|
|
|
|
#endif
|
2020-02-10 22:26:19 +01:00
|
|
|
DEBUG("\n");
|
2016-07-15 10:40:56 +02:00
|
|
|
}
|
|
|
|
|
2020-11-02 02:41:37 +01:00
|
|
|
static void _init_pins(mtd_spi_nor_t *dev)
|
|
|
|
{
|
|
|
|
DEBUG("mtd_spi_nor_init: init pins\n");
|
|
|
|
|
|
|
|
/* CS */
|
|
|
|
spi_init_cs(_get_spi(dev), dev->params->cs);
|
|
|
|
|
|
|
|
/* Write Protect - not used by the driver */
|
|
|
|
if (gpio_is_valid(dev->params->wp)) {
|
|
|
|
gpio_init(dev->params->wp, GPIO_OUT);
|
|
|
|
gpio_set(dev->params->wp);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Hold - not used by the driver */
|
|
|
|
if (gpio_is_valid(dev->params->hold)) {
|
|
|
|
gpio_init(dev->params->hold, GPIO_OUT);
|
|
|
|
gpio_set(dev->params->hold);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-02-03 21:52:19 +01:00
|
|
|
static void _enable_32bit_addr(mtd_spi_nor_t *dev)
|
|
|
|
{
|
|
|
|
mtd_spi_cmd(dev, dev->params->opcode->wren);
|
|
|
|
mtd_spi_cmd(dev, SFLASH_CMD_4_BYTE_ADDR);
|
|
|
|
}
|
|
|
|
|
2020-10-25 19:27:29 +01:00
|
|
|
static int mtd_spi_nor_power(mtd_dev_t *mtd, enum mtd_power_state power)
|
|
|
|
{
|
|
|
|
mtd_spi_nor_t *dev = (mtd_spi_nor_t *)mtd;
|
|
|
|
|
|
|
|
mtd_spi_acquire(dev);
|
|
|
|
switch (power) {
|
|
|
|
case MTD_POWER_UP:
|
|
|
|
mtd_spi_cmd(dev, dev->params->opcode->wake);
|
2023-08-31 15:22:33 +02:00
|
|
|
|
|
|
|
/* fall back to polling if no timer is used */
|
|
|
|
unsigned retries = MTD_POWER_UP_WAIT_FOR_ID;
|
|
|
|
if (!IS_USED(MODULE_ZTIMER) && !IS_USED(MODULE_XTIMER)) {
|
|
|
|
retries *= dev->params->wait_chip_wake_up * 1000;
|
|
|
|
}
|
|
|
|
|
2020-10-25 19:27:29 +01:00
|
|
|
int res = 0;
|
|
|
|
do {
|
2024-01-15 14:37:46 +01:00
|
|
|
delay_us(dev->params->wait_chip_wake_up);
|
2020-10-25 19:27:29 +01:00
|
|
|
res = mtd_spi_read_jedec_id(dev, &dev->jedec_id);
|
2023-08-31 15:22:33 +02:00
|
|
|
} while (res < 0 && --retries);
|
2020-10-25 19:27:29 +01:00
|
|
|
if (res < 0) {
|
2023-05-22 18:16:01 +02:00
|
|
|
mtd_spi_release(dev);
|
2020-10-25 19:27:29 +01:00
|
|
|
return -EIO;
|
|
|
|
}
|
2020-11-02 00:01:00 +01:00
|
|
|
/* enable 32 bit address mode */
|
2022-02-03 21:52:19 +01:00
|
|
|
if (dev->addr_width == 4) {
|
|
|
|
_enable_32bit_addr(dev);
|
2020-11-02 00:01:00 +01:00
|
|
|
}
|
|
|
|
|
2020-10-25 19:27:29 +01:00
|
|
|
break;
|
|
|
|
case MTD_POWER_DOWN:
|
|
|
|
mtd_spi_cmd(dev, dev->params->opcode->sleep);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
mtd_spi_release(dev);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2022-02-03 21:58:34 +01:00
|
|
|
static void _set_addr_width(mtd_dev_t *mtd)
|
|
|
|
{
|
|
|
|
mtd_spi_nor_t *dev = (mtd_spi_nor_t *)mtd;
|
|
|
|
|
|
|
|
uint32_t flash_size = mtd->pages_per_sector * mtd->page_size
|
|
|
|
* mtd->sector_count;
|
|
|
|
|
2022-04-13 17:24:39 +02:00
|
|
|
if (flash_size > (0x1UL << 24)) {
|
2022-02-03 21:58:34 +01:00
|
|
|
dev->addr_width = 4;
|
|
|
|
} else {
|
|
|
|
dev->addr_width = 3;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-07-15 10:40:56 +02:00
|
|
|
static int mtd_spi_nor_init(mtd_dev_t *mtd)
|
|
|
|
{
|
|
|
|
DEBUG("mtd_spi_nor_init: %p\n", (void *)mtd);
|
|
|
|
mtd_spi_nor_t *dev = (mtd_spi_nor_t *)mtd;
|
|
|
|
|
|
|
|
DEBUG("mtd_spi_nor_init: -> spi: %lx, cs: %lx, opcodes: %p\n",
|
2020-10-25 20:51:11 +01:00
|
|
|
(unsigned long)_get_spi(dev), (unsigned long)dev->params->cs, (void *)dev->params->opcode);
|
2016-07-15 10:40:56 +02:00
|
|
|
|
2020-11-02 02:41:37 +01:00
|
|
|
/* CS, WP, Hold */
|
|
|
|
_init_pins(dev);
|
2016-07-15 10:40:56 +02:00
|
|
|
|
2020-03-05 10:09:13 +01:00
|
|
|
/* power up the MTD device*/
|
2023-05-22 18:13:54 +02:00
|
|
|
DEBUG_PUTS("mtd_spi_nor_init: power up MTD device");
|
2020-03-05 10:09:13 +01:00
|
|
|
if (mtd_spi_nor_power(mtd, MTD_POWER_UP)) {
|
2023-05-22 18:13:54 +02:00
|
|
|
DEBUG_PUTS("mtd_spi_nor_init: failed to power up MTD device");
|
2020-03-05 10:09:13 +01:00
|
|
|
return -EIO;
|
|
|
|
}
|
|
|
|
|
2020-02-10 20:39:49 +01:00
|
|
|
mtd_spi_acquire(dev);
|
2016-07-15 10:40:56 +02:00
|
|
|
int res = mtd_spi_read_jedec_id(dev, &dev->jedec_id);
|
|
|
|
if (res < 0) {
|
2020-02-10 20:39:49 +01:00
|
|
|
mtd_spi_release(dev);
|
2016-07-15 10:40:56 +02:00
|
|
|
return -EIO;
|
|
|
|
}
|
|
|
|
DEBUG("mtd_spi_nor_init: Found chip with ID: (%d, 0x%02x, 0x%02x, 0x%02x)\n",
|
2018-03-05 15:54:08 +01:00
|
|
|
dev->jedec_id.bank, dev->jedec_id.manuf, dev->jedec_id.device[0], dev->jedec_id.device[1]);
|
2016-07-15 10:40:56 +02:00
|
|
|
|
2020-05-14 22:06:22 +02:00
|
|
|
/* derive density from JEDEC ID */
|
|
|
|
if (mtd->sector_count == 0) {
|
2022-02-03 21:58:34 +01:00
|
|
|
mtd->sector_count = mtd_spi_nor_get_size(&dev->jedec_id)
|
2020-05-14 22:06:22 +02:00
|
|
|
/ (mtd->pages_per_sector * mtd->page_size);
|
|
|
|
}
|
2022-02-27 00:09:23 +01:00
|
|
|
/* SPI NOR is byte addressable; instances don't need to configure that */
|
|
|
|
assert(mtd->write_size <= 1);
|
|
|
|
mtd->write_size = 1;
|
2022-02-03 21:58:34 +01:00
|
|
|
_set_addr_width(mtd);
|
2020-05-14 22:06:22 +02:00
|
|
|
|
2020-05-14 22:07:08 +02:00
|
|
|
DEBUG("mtd_spi_nor_init: %" PRIu32 " bytes "
|
|
|
|
"(%" PRIu32 " sectors, %" PRIu32 " bytes/sector, "
|
|
|
|
"%" PRIu32 " pages, "
|
|
|
|
"%" PRIu32 " pages/sector, %" PRIu32 " bytes/page)\n",
|
|
|
|
mtd->pages_per_sector * mtd->sector_count * mtd->page_size,
|
|
|
|
mtd->sector_count, mtd->pages_per_sector * mtd->page_size,
|
|
|
|
mtd->pages_per_sector * mtd->sector_count,
|
|
|
|
mtd->pages_per_sector, mtd->page_size);
|
2022-02-03 21:58:34 +01:00
|
|
|
DEBUG("mtd_spi_nor_init: Using %u byte addresses\n", dev->addr_width);
|
2020-05-14 22:07:08 +02:00
|
|
|
|
2016-07-15 10:40:56 +02:00
|
|
|
uint8_t status;
|
2020-02-10 20:39:49 +01:00
|
|
|
mtd_spi_cmd_read(dev, dev->params->opcode->rdsr, &status, sizeof(status));
|
2016-07-15 10:40:56 +02:00
|
|
|
DEBUG("mtd_spi_nor_init: device status = 0x%02x\n", (unsigned int)status);
|
|
|
|
|
2022-02-03 21:52:19 +01:00
|
|
|
/* enable 32 bit address mode */
|
|
|
|
if (dev->addr_width == 4) {
|
|
|
|
_enable_32bit_addr(dev);
|
|
|
|
}
|
|
|
|
|
2022-02-03 21:30:46 +01:00
|
|
|
/* Global Block-Protection Unlock */
|
|
|
|
mtd_spi_cmd(dev, dev->params->opcode->wren);
|
|
|
|
mtd_spi_cmd(dev, SFLASH_CMD_ULBPR);
|
|
|
|
|
|
|
|
mtd_spi_release(dev);
|
|
|
|
|
2016-07-15 10:40:56 +02:00
|
|
|
/* check whether page size and sector size are powers of two (most chips' are)
|
|
|
|
* and compute the number of shifts needed to get the page and sector addresses
|
|
|
|
* from a byte address */
|
|
|
|
uint8_t shift = 0;
|
|
|
|
uint32_t page_size = mtd->page_size;
|
|
|
|
uint32_t mask = 0;
|
2020-10-25 19:27:29 +01:00
|
|
|
|
2016-07-15 10:40:56 +02:00
|
|
|
if ((page_size & (page_size - 1)) == 0) {
|
|
|
|
while ((page_size >> shift) > 1) {
|
|
|
|
++shift;
|
|
|
|
}
|
|
|
|
mask = (UINT32_MAX << shift);
|
|
|
|
}
|
2020-10-25 19:27:29 +01:00
|
|
|
|
2016-07-15 10:40:56 +02:00
|
|
|
dev->page_addr_mask = mask;
|
|
|
|
dev->page_addr_shift = shift;
|
|
|
|
DEBUG("mtd_spi_nor_init: page_addr_mask = 0x%08" PRIx32 ", page_addr_shift = %u\n",
|
2018-03-05 15:54:08 +01:00
|
|
|
mask, (unsigned int)shift);
|
2016-07-15 10:40:56 +02:00
|
|
|
|
|
|
|
mask = 0;
|
|
|
|
shift = 0;
|
|
|
|
uint32_t sector_size = mtd->page_size * mtd->pages_per_sector;
|
|
|
|
if ((sector_size & (sector_size - 1)) == 0) {
|
|
|
|
while ((sector_size >> shift) > 1) {
|
|
|
|
++shift;
|
|
|
|
}
|
|
|
|
mask = (UINT32_MAX << shift);
|
|
|
|
}
|
|
|
|
dev->sec_addr_mask = mask;
|
|
|
|
dev->sec_addr_shift = shift;
|
|
|
|
|
|
|
|
DEBUG("mtd_spi_nor_init: sec_addr_mask = 0x%08" PRIx32 ", sec_addr_shift = %u\n",
|
2018-03-05 15:54:08 +01:00
|
|
|
mask, (unsigned int)shift);
|
2016-07-15 10:40:56 +02:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int mtd_spi_nor_read(mtd_dev_t *mtd, void *dest, uint32_t addr, uint32_t size)
|
|
|
|
{
|
|
|
|
DEBUG("mtd_spi_nor_read: %p, %p, 0x%" PRIx32 ", 0x%" PRIx32 "\n",
|
2018-03-05 15:54:08 +01:00
|
|
|
(void *)mtd, dest, addr, size);
|
2017-06-29 22:46:16 +02:00
|
|
|
const mtd_spi_nor_t *dev = (mtd_spi_nor_t *)mtd;
|
2020-05-07 15:07:47 +02:00
|
|
|
uint32_t chipsize = mtd->page_size * mtd->pages_per_sector * mtd->sector_count;
|
2020-10-25 19:27:29 +01:00
|
|
|
|
2016-07-15 10:40:56 +02:00
|
|
|
if (addr > chipsize) {
|
|
|
|
return -EOVERFLOW;
|
|
|
|
}
|
|
|
|
if ((addr + size) > chipsize) {
|
|
|
|
size = chipsize - addr;
|
|
|
|
}
|
|
|
|
if (size == 0) {
|
|
|
|
return 0;
|
|
|
|
}
|
2020-10-25 19:27:29 +01:00
|
|
|
|
2020-02-10 20:39:49 +01:00
|
|
|
mtd_spi_acquire(dev);
|
2020-10-25 21:41:33 +01:00
|
|
|
mtd_spi_cmd_addr_read(dev, dev->params->opcode->read, addr, dest, size);
|
2020-02-10 20:39:49 +01:00
|
|
|
mtd_spi_release(dev);
|
2016-07-15 10:40:56 +02:00
|
|
|
|
2020-04-17 19:41:43 +02:00
|
|
|
return 0;
|
2016-07-15 10:40:56 +02:00
|
|
|
}
|
|
|
|
|
2020-05-07 15:07:47 +02:00
|
|
|
static int mtd_spi_nor_write_page(mtd_dev_t *mtd, const void *src, uint32_t page, uint32_t offset,
|
|
|
|
uint32_t size)
|
|
|
|
{
|
|
|
|
const mtd_spi_nor_t *dev = (mtd_spi_nor_t *)mtd;
|
|
|
|
|
|
|
|
DEBUG("mtd_spi_nor_write_page: %p, %p, 0x%" PRIx32 ", 0x%" PRIx32 ", 0x%" PRIx32 "\n",
|
|
|
|
(void *)mtd, src, page, offset, size);
|
|
|
|
|
|
|
|
uint32_t remaining = mtd->page_size - offset;
|
|
|
|
size = MIN(remaining, size);
|
|
|
|
|
2020-10-25 21:41:33 +01:00
|
|
|
uint32_t addr = page * mtd->page_size + offset;
|
2020-05-07 15:07:47 +02:00
|
|
|
|
|
|
|
mtd_spi_acquire(dev);
|
|
|
|
|
|
|
|
/* write enable */
|
|
|
|
mtd_spi_cmd(dev, dev->params->opcode->wren);
|
|
|
|
|
|
|
|
/* Page program */
|
2020-10-25 21:41:33 +01:00
|
|
|
mtd_spi_cmd_addr_write(dev, dev->params->opcode->page_program, addr, src, size);
|
2020-05-07 15:07:47 +02:00
|
|
|
|
|
|
|
/* waiting for the command to complete before returning */
|
|
|
|
wait_for_write_complete(dev, 0);
|
|
|
|
|
|
|
|
mtd_spi_release(dev);
|
|
|
|
|
|
|
|
return size;
|
|
|
|
}
|
|
|
|
|
2016-07-15 10:40:56 +02:00
|
|
|
static int mtd_spi_nor_erase(mtd_dev_t *mtd, uint32_t addr, uint32_t size)
|
|
|
|
{
|
|
|
|
DEBUG("mtd_spi_nor_erase: %p, 0x%" PRIx32 ", 0x%" PRIx32 "\n",
|
2018-03-05 15:54:08 +01:00
|
|
|
(void *)mtd, addr, size);
|
2016-07-15 10:40:56 +02:00
|
|
|
mtd_spi_nor_t *dev = (mtd_spi_nor_t *)mtd;
|
|
|
|
uint32_t sector_size = mtd->page_size * mtd->pages_per_sector;
|
|
|
|
uint32_t total_size = sector_size * mtd->sector_count;
|
|
|
|
|
|
|
|
if (dev->sec_addr_mask &&
|
|
|
|
((addr & ~dev->sec_addr_mask) != 0)) {
|
|
|
|
/* This is not a requirement in hardware, but it helps in catching
|
|
|
|
* software bugs (the erase-all-your-files kind) */
|
|
|
|
DEBUG("addr = %" PRIx32 " ~dev->erase_addr_mask = %" PRIx32 "", addr, ~dev->sec_addr_mask);
|
|
|
|
DEBUG("mtd_spi_nor_erase: ERR: erase addr not aligned on %" PRIu32 " byte boundary.\n",
|
2018-03-05 15:54:08 +01:00
|
|
|
sector_size);
|
2016-07-15 10:40:56 +02:00
|
|
|
return -EOVERFLOW;
|
|
|
|
}
|
|
|
|
if (addr + size > total_size) {
|
|
|
|
return -EOVERFLOW;
|
|
|
|
}
|
2018-01-22 14:04:35 +01:00
|
|
|
if (size % sector_size != 0) {
|
2016-07-15 10:40:56 +02:00
|
|
|
return -EOVERFLOW;
|
|
|
|
}
|
2018-01-22 14:04:35 +01:00
|
|
|
|
2020-02-10 20:39:49 +01:00
|
|
|
mtd_spi_acquire(dev);
|
2018-01-22 14:04:35 +01:00
|
|
|
while (size) {
|
2018-06-14 14:15:11 +02:00
|
|
|
uint32_t us;
|
2020-10-25 21:41:33 +01:00
|
|
|
|
2018-01-22 14:04:35 +01:00
|
|
|
/* write enable */
|
2020-02-10 20:39:49 +01:00
|
|
|
mtd_spi_cmd(dev, dev->params->opcode->wren);
|
2018-01-22 14:04:35 +01:00
|
|
|
|
|
|
|
if (size == total_size) {
|
2020-02-10 20:39:49 +01:00
|
|
|
mtd_spi_cmd(dev, dev->params->opcode->chip_erase);
|
2018-01-22 14:04:35 +01:00
|
|
|
size -= total_size;
|
2020-02-10 22:28:42 +01:00
|
|
|
us = dev->params->wait_chip_erase;
|
2018-01-22 14:04:35 +01:00
|
|
|
}
|
2020-10-25 23:23:48 +01:00
|
|
|
else if ((dev->params->flag & SPI_NOR_F_SECT_64K) && (size >= MTD_64K) &&
|
|
|
|
((addr & MTD_64K_ADDR_MASK) == 0)) {
|
|
|
|
/* 64 KiB blocks can be erased with block erase command */
|
|
|
|
mtd_spi_cmd_addr_write(dev, dev->params->opcode->block_erase_64k, addr, NULL, 0);
|
|
|
|
addr += MTD_64K;
|
|
|
|
size -= MTD_64K;
|
|
|
|
us = dev->params->wait_64k_erase;
|
|
|
|
}
|
2020-02-10 20:39:49 +01:00
|
|
|
else if ((dev->params->flag & SPI_NOR_F_SECT_32K) && (size >= MTD_32K) &&
|
2018-01-22 14:04:35 +01:00
|
|
|
((addr & MTD_32K_ADDR_MASK) == 0)) {
|
|
|
|
/* 32 KiB blocks can be erased with block erase command */
|
2020-10-25 21:41:33 +01:00
|
|
|
mtd_spi_cmd_addr_write(dev, dev->params->opcode->block_erase_32k, addr, NULL, 0);
|
2018-01-22 14:04:35 +01:00
|
|
|
addr += MTD_32K;
|
|
|
|
size -= MTD_32K;
|
2020-02-10 22:28:42 +01:00
|
|
|
us = dev->params->wait_32k_erase;
|
2018-01-22 14:04:35 +01:00
|
|
|
}
|
2020-02-10 20:39:49 +01:00
|
|
|
else if ((dev->params->flag & SPI_NOR_F_SECT_4K) && (size >= MTD_4K) &&
|
2018-01-22 14:04:35 +01:00
|
|
|
((addr & MTD_4K_ADDR_MASK) == 0)) {
|
|
|
|
/* 4 KiB sectors can be erased with sector erase command */
|
2020-10-25 21:41:33 +01:00
|
|
|
mtd_spi_cmd_addr_write(dev, dev->params->opcode->sector_erase, addr, NULL, 0);
|
2018-01-22 14:04:35 +01:00
|
|
|
addr += MTD_4K;
|
|
|
|
size -= MTD_4K;
|
2020-10-25 23:23:48 +01:00
|
|
|
us = dev->params->wait_sector_erase;
|
2018-01-22 14:04:35 +01:00
|
|
|
}
|
|
|
|
else {
|
2020-10-25 23:23:48 +01:00
|
|
|
/* no suitable erase block found */
|
|
|
|
assert(0);
|
|
|
|
|
|
|
|
mtd_spi_release(dev);
|
|
|
|
return -EINVAL;
|
2016-07-15 10:40:56 +02:00
|
|
|
}
|
2018-01-22 14:04:35 +01:00
|
|
|
|
|
|
|
/* waiting for the command to complete before continuing */
|
2018-06-14 14:15:11 +02:00
|
|
|
wait_for_write_complete(dev, us);
|
2016-07-15 10:40:56 +02:00
|
|
|
}
|
2020-02-10 20:39:49 +01:00
|
|
|
mtd_spi_release(dev);
|
2016-07-15 10:40:56 +02:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2020-05-07 15:07:47 +02:00
|
|
|
const mtd_desc_t mtd_spi_nor_driver = {
|
|
|
|
.init = mtd_spi_nor_init,
|
|
|
|
.read = mtd_spi_nor_read,
|
|
|
|
.write_page = mtd_spi_nor_write_page,
|
|
|
|
.erase = mtd_spi_nor_erase,
|
|
|
|
.power = mtd_spi_nor_power,
|
|
|
|
};
|