mirror of
https://github.com/RIOT-OS/RIOT.git
synced 2025-01-18 12:52:44 +01:00
0425325260
Just read the JEDEC ID with a single SPI transfer, there can only be 10 banks. This makes adaption to QSPI much simpler because now flash functions don't call the SPI API directly anymore.
176 lines
5.5 KiB
C
176 lines
5.5 KiB
C
/*
|
|
* 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.
|
|
*/
|
|
|
|
/**
|
|
* @defgroup drivers_mtd_spi_nor Serial NOR flash
|
|
* @ingroup drivers_storage
|
|
* @brief Driver for serial NOR flash memory technology devices attached via SPI
|
|
*
|
|
* @{
|
|
*
|
|
* @file
|
|
* @brief Interface definition for the serial flash memory driver
|
|
*
|
|
* @author Joakim Nohlgård <joakim.nohlgard@eistec.se>
|
|
* @author Vincent Dupont <vincent@otakeys.com>
|
|
*/
|
|
|
|
#ifndef MTD_SPI_NOR_H
|
|
#define MTD_SPI_NOR_H
|
|
|
|
#include <stdint.h>
|
|
|
|
#include "periph_conf.h"
|
|
#include "periph/spi.h"
|
|
#include "periph/gpio.h"
|
|
#include "mtd.h"
|
|
|
|
#ifdef __cplusplus
|
|
extern "C"
|
|
{
|
|
#endif
|
|
|
|
/**
|
|
* @brief SPI NOR flash opcode table
|
|
*/
|
|
typedef struct {
|
|
uint8_t rdid; /**< Read identification (JEDEC ID) */
|
|
uint8_t wren; /**< Write enable */
|
|
uint8_t rdsr; /**< Read status register */
|
|
uint8_t wrsr; /**< Write status register */
|
|
uint8_t read; /**< Read data bytes, 3 byte address */
|
|
uint8_t read_fast; /**< Read data bytes, 3 byte address, at higher speed */
|
|
uint8_t page_program; /**< Page program */
|
|
uint8_t sector_erase; /**< Block erase 4 KiB */
|
|
uint8_t block_erase_32k; /**< 32KiB block erase */
|
|
uint8_t block_erase; /**< Block erase (usually 64 KiB) */
|
|
uint8_t chip_erase; /**< Chip erase */
|
|
uint8_t sleep; /**< Deep power down */
|
|
uint8_t wake; /**< Release from deep power down */
|
|
/* TODO: enter 4 byte address mode for large memories */
|
|
} mtd_spi_nor_opcode_t;
|
|
|
|
/**
|
|
* @brief Internal representation of JEDEC memory ID codes.
|
|
*
|
|
* @see http://www.jedec.org/standards-documents/results/jep106
|
|
*/
|
|
typedef struct __attribute__((packed)) {
|
|
uint8_t bank; /**< Manufacturer ID bank number, 1 through 10, see JEP106 */
|
|
uint8_t manuf; /**< Manufacturer ID, 1 byte */
|
|
uint8_t device[2]; /**< Device ID, 2 bytes */
|
|
} mtd_jedec_id_t;
|
|
|
|
/**
|
|
* @brief Byte to signal increment bank number when reading manufacturer ID
|
|
*
|
|
* @see http://www.jedec.org/standards-documents/results/jep106
|
|
*/
|
|
#define JEDEC_NEXT_BANK (0x7f)
|
|
|
|
/**
|
|
* @brief The highest possible bank number when reading manufacturer ID
|
|
*
|
|
* @see http://www.jedec.org/standards-documents/results/jep106
|
|
*/
|
|
#define JEDEC_BANK_MAX (10)
|
|
|
|
/**
|
|
* @brief Flag to set when the device support 4KiB sector erase (sector_erase opcode)
|
|
*/
|
|
#define SPI_NOR_F_SECT_4K (1)
|
|
/**
|
|
* @brief Flag to set when the device support 32KiB block erase (block_erase_32k opcode)
|
|
*/
|
|
#define SPI_NOR_F_SECT_32K (2)
|
|
|
|
/**
|
|
* @brief Compile-time parameters for a serial flash device
|
|
*/
|
|
typedef struct {
|
|
const mtd_spi_nor_opcode_t *opcode; /**< Opcode table for the device */
|
|
uint32_t wait_chip_erase; /**< Full chip erase wait time in µs */
|
|
uint32_t wait_sector_erase; /**< Sector erase wait time in µs */
|
|
uint32_t wait_32k_erase; /**< 32KB page erase wait time in µs */
|
|
uint32_t wait_4k_erase; /**< 4KB page erase wait time in µs */
|
|
uint32_t wait_chip_wake_up; /**< Chip wake up time in µs */
|
|
spi_clk_t clk; /**< SPI clock */
|
|
uint16_t flag; /**< Config flags */
|
|
spi_t spi; /**< SPI bus the device is connected to */
|
|
spi_mode_t mode; /**< SPI mode */
|
|
gpio_t cs; /**< CS pin GPIO handle */
|
|
uint8_t addr_width; /**< Number of bytes in addresses, usually 3 for small devices */
|
|
} mtd_spi_nor_params_t;
|
|
|
|
/**
|
|
* @brief Device descriptor for serial flash memory devices
|
|
*
|
|
* This is an extension of the @c mtd_dev_t struct
|
|
*/
|
|
typedef struct {
|
|
mtd_dev_t base; /**< inherit from mtd_dev_t object */
|
|
const mtd_spi_nor_params_t *params; /**< SPI NOR params */
|
|
mtd_jedec_id_t jedec_id; /**< JEDEC ID of the chip */
|
|
|
|
/**
|
|
* @brief bitmask to corresponding to the page address
|
|
*
|
|
* Computed by mtd_spi_nor_init, no need to touch outside the driver.
|
|
*/
|
|
uint32_t page_addr_mask;
|
|
/**
|
|
* @brief bitmask to corresponding to the sector address
|
|
*
|
|
* Computed by mtd_spi_nor_init, no need to touch outside the driver.
|
|
*/
|
|
uint32_t sec_addr_mask;
|
|
/**
|
|
* @brief number of right shifts to get the address to the start of the page
|
|
*
|
|
* Computed by mtd_spi_nor_init, no need to touch outside the driver.
|
|
*/
|
|
uint8_t page_addr_shift;
|
|
/**
|
|
* @brief number of right shifts to get the address to the start of the sector
|
|
*
|
|
* Computed by mtd_spi_nor_init, no need to touch outside the driver.
|
|
*/
|
|
uint8_t sec_addr_shift;
|
|
} mtd_spi_nor_t;
|
|
|
|
/**
|
|
* @brief NOR flash SPI MTD device operations table
|
|
*/
|
|
extern const mtd_desc_t mtd_spi_nor_driver;
|
|
|
|
/* Available opcode tables for known devices */
|
|
/* Defined in mtd_spi_nor_configs.c */
|
|
/**
|
|
* @brief Default command opcodes
|
|
*
|
|
* The numbers were taken from Micron M25P16, but the same opcodes can
|
|
* be found in Macronix MX25L25735E, and multiple other data sheets for
|
|
* different devices, as well as in the Linux kernel, so they seem quite
|
|
* sensible for default values. */
|
|
extern const mtd_spi_nor_opcode_t mtd_spi_nor_opcode_default;
|
|
|
|
/**
|
|
* @brief Default 4-byte addresses opcodes
|
|
*
|
|
* Commands for 4-byte addresses chips (above 128Mb)
|
|
*/
|
|
extern const mtd_spi_nor_opcode_t mtd_spi_nor_opcode_default_4bytes;
|
|
|
|
#ifdef __cplusplus
|
|
}
|
|
#endif
|
|
|
|
#endif /* MTD_SPI_NOR_H */
|
|
/** @} */
|