blob: fe7e963bd870736ca982b7b5a7b59cebf94e34d2 [file] [log] [blame]
#ifndef _SPI_FLASH_H
#define _SPI_FLASH_H
#include "Typedef.h"
#include "Flash.h"
#include "predefines.h"
#define false 0
#define true 1
#ifndef min
#define min(a, b) ((a) < (b) ? (a) : (b))
#endif
#ifndef max
#define max(a, b) ((a) > (b) ? (a) : (b))
#endif
#define _readb(p) (*(volatile uint8_t *)(p))
#define _readw(p) (*(volatile uint16_t *)(p))
#define _readl(p) (*(volatile uint32_t *)(p))
#define _writeb(v,p) (*(volatile uint8_t *)(p) = ((uint8_t)v))
#define _writew(v,p) (*(volatile uint16_t *)(p) = ((uint16_t)v))
#define _writel(v,p) (*(volatile uint32_t *)(p) = (v))
/* Vendor Manufacture ID */
#define SPIFLASH_MFR_MICRON 0x2C
#define SPIFLASH_MFR_GIGADEVICE 0xC8
#define SPIFLASH_MFR_WINBOND 0xEF
#define SPIFLASH_MFR_MXIC 0xC2
#define SPIFLASH_MFR_DOSILICON 0xE5
#define SPIFLASH_MFR_ZETTADEVICE 0xBA
#define SPIFLASH_MFR_DOUQI 0x54
#define SPIFLASH_MFR_DOSILICON_NOR 0xF8
#define SPIFLASH_MFR_PUYA 0x85
#define SPIFLASH_MFR_XM 0x20
#define SPIFLASH_MFR_FMSH 0xA1
struct spi_flash_chip;
struct spi_flash_ops {
int (*reset)(struct spi_flash_chip *chip);
int (*read_id)(struct spi_flash_chip *chip, uint8_t *buf);
int (*read)(struct spi_flash_chip *chip, int addr, int size, uint8_t *rbuf);
int (*write)(struct spi_flash_chip *chip, int addr, int size, uint8_t *wbuf);
int (*erase)(struct spi_flash_chip *chip, int addr, int len);
};
struct micron_onfi_specific {
uint8_t two_plane_page_read; /*166*/
uint8_t reserved0[8]; /*167-174*/
uint8_t otp_mode; /*175*/
uint8_t otp_page_start; /*176*/
uint8_t otp_data_protect_addr; /*177*/
uint8_t otp_page_number; /*178*/
uint8_t otp_feature_addr; /*179*/
uint8_t reserved1[68]; /*180-247*/
uint8_t ecc_ability; /*248*/
uint8_t die_selection; /*249*/
uint8_t reserved2[3]; /*250-252*/
uint8_t parameter_page_version; /*253*/
} __attribute__((packed));
union spi_nand_vendor_specific {
uint8_t vendor_specific[88];
struct micron_onfi_specific micron_sepcific;
};
struct spi_nand_onfi_params {
/* rev info and features block */
/* 'O' 'N' 'F' 'I' */
uint8_t sig[4]; /*0-3*/
uint16_t revision; /*4-5*/
uint16_t features; /*6-7*/
uint16_t opt_cmd; /*8-9*/
uint8_t reserved0[22]; /*10-31*/
/* manufacturer information block */
char manufacturer[12]; /*32-43*/
char model[20]; /*44-63*/
uint8_t mfr_id; /*64*/
uint16_t date_code; /*65-66*/
uint8_t reserved1[13]; /*67-79*/
/* memory organization block */
uint32_t byte_per_page; /*80-83*/
uint16_t spare_bytes_per_page; /*84*85*/
uint32_t data_bytes_per_ppage; /*86-89*/
uint16_t spare_bytes_per_ppage; /*90-91*/
uint32_t pages_per_block; /*92-95*/
uint32_t blocks_per_lun; /*96-99*/
uint8_t lun_count; /*100*/
uint8_t addr_cycles; /*101*/
uint8_t bits_per_cell; /*102*/
uint16_t bb_per_lun; /*103-104*/
uint16_t block_endurance; /*105-106*/
uint8_t guaranteed_good_blocks; /*107*/
uint16_t guaranteed_block_endurance; /*108-109*/
uint8_t programs_per_page; /*110*/
uint8_t ppage_attr; /*111*/
uint8_t ecc_bits; /*112*/
uint8_t interleaved_bits; /*113*/
uint8_t interleaved_ops; /*114*/
uint8_t reserved2[13]; /*115-127*/
/* electrical parameter block */
uint8_t io_pin_capacitance_max; /*128*/
uint16_t timing_mode; /*129-130*/
uint16_t program_cache_timing_mode; /*131-132*/
uint16_t t_prog; /*133-134*/
uint16_t t_bers; /*135-136*/
uint16_t t_r; /*137-138*/
uint16_t t_ccs; /*139-140*/
uint8_t reserved3[23]; /*141-163*/
/* vendor */
uint16_t vendor_specific_revision; /*164-165*/
union spi_nand_vendor_specific vendor; /*166-253*/
uint16_t crc; /*254-255*/
} __attribute__((packed));
#define ONFI_CRC_BASE 0x4F4E
/**
* struct spi_flash_chip - SPI-NAND Private Flash Chip Data
* @name: name of the chip
* @spi: [INTERN] point to spi device structure
* @mfr_id: [BOARDSPECIFIC] manufacture id
* @dev_id: [BOARDSPECIFIC] device id
* @read_cache_op: [REPLACEABLE] Opcode of read from cache
* @write_cache_op: [REPLACEABLE] Opcode of program load
* @write_cache_rdm_op: [REPLACEABLE] Opcode of program load random
* @oobbuf: [INTERN] buffer for read/write oob
* @size: [INTERN] the size of chip
* @block_size: [INTERN] the size of eraseblock
* @page_size: [INTERN] the size of page
* @oob_size: [INTERN] the size of page oob size
* @block_shift: [INTERN] number of address bits in a eraseblock
* @page_shift: [INTERN] number of address bits in a page (column
* address bits).
* @pagemask: [INTERN] page number mask = number of (pages / chip) - 1
* @options: [BOARDSPECIFIC] various chip options. They can partly
* be set to inform nand_scan about special functionality.
* @ecc_strength: [INTERN] ECC correctability
* @refresh_threshold: [INTERN] Bitflip threshold to return -EUCLEAN
* @ecclayout: [BOARDSPECIFIC] ECC layout control structure
* See the defines for further explanation.
* @onfi_params: [INTERN] holds the ONFI page parameter
*/
struct spi_flash_chip {
char *name;
struct spi_flash_cmd_cfg *table;
struct spi_flash_ops *ops;
struct qspi_host *host;
uint8_t cs;
uint8_t mfr_id;
uint16_t dev_id;
uint32_t tx_max_len;
uint32_t rx_max_len;
uint8_t gd_ver_c;
uint8_t read_op;
uint8_t read_cache_op;
uint8_t write_op;
uint8_t write_cache_op;
uint8_t write_cache_rdm_op;
int (*enable_ecc)(struct spi_flash_chip *chip);
int (*disable_ecc)(struct spi_flash_chip *chip);
void (*get_ecc_status)(struct spi_flash_chip *chip, uint8_t status,
uint32_t *corrected, uint32_t *ecc_error);
uint8_t en_addr_4byte;
uint8_t qpi_enabled;
uint8_t qpi_dummy;
uint8_t *oobbuf;
uint64_t size;
uint32_t block_size;
uint16_t page_size;
uint16_t oob_size;
uint8_t lun_shift;
uint8_t block_shift;
uint8_t page_shift;
uint16_t page_mask;
uint32_t options;
uint32_t ecc_strength;
uint8_t refresh_threshold;
uint8_t lun;
uint32_t max_mhz;
void * flash_info;
//struct nand_ecclayout *ecclayout;
struct spi_nand_onfi_params onfi_params;
};
enum {
CMD_W_NO_DATA = 0,
CMD_W_RX_DATA = 1,
CMD_W_TX_DATA = 2,
};
struct spi_flash_cmd_cfg {
uint8_t opcode;
uint8_t cmd_dtr;
uint8_t addr_bytes;
uint8_t addr_pins;
uint8_t addr_dtr;
uint8_t mode_bits;
uint8_t mode_pins;
uint8_t mode_dtr;
uint8_t dummy_cycles;
uint8_t dummy_pins;
uint8_t data_pins;
uint8_t data_dtr;
int8_t seq_id;
int8_t type;
};
#define SPI_CMD_DTR(_opcode,_cmd_dtr,_addr_bytes,_addr_pins,_addr_dtr, \
_mode_bits,_mode_pins,_mode_dtr,_dummy_cycles,_dummy_pins, \
_data_pins,_data_dtr, _seq_id, _type) \
{ \
.opcode = _opcode, \
.cmd_dtr = _cmd_dtr, \
.addr_bytes = _addr_bytes, \
.addr_pins = _addr_pins, \
.addr_dtr = _addr_dtr, \
.mode_bits = _mode_bits, \
.mode_pins = _mode_pins, \
.mode_dtr = _mode_dtr, \
.dummy_cycles = _dummy_cycles, \
.dummy_pins = _dummy_pins, \
.data_pins = _data_pins, \
.data_dtr = _data_dtr, \
.seq_id = _seq_id, \
.type = _type, \
}
#define SPI_CMD(_opcode,_addr_bytes,_addr_pins,_mode_bits, \
_mode_pins,_dummy_cycles,_dummy_pins,_data_pins, \
_seq_id, _type) \
SPI_CMD_DTR(_opcode,0,_addr_bytes,_addr_pins,0, \
_mode_bits,_mode_pins,0,_dummy_cycles,_dummy_pins, \
_data_pins,0, _seq_id, _type) \
enum {
RST_AHB_DOMAIN = 1 << 0,
AHB_MAP_SIZE_PAGE = 1 << 1,
};
#define SPINAND_MAX_ADDR_LEN 4
struct spi_flash_cmd {
struct spi_flash_cmd_cfg *cmd_cfg;
uint8_t n_addr; /* Number of address */
uint8_t addr[SPINAND_MAX_ADDR_LEN]; /* Reg Offset */
uint32_t n_tx; /* Number of tx bytes */
const uint8_t *tx_buf; /* Tx buf */
uint32_t n_rx; /* Number of rx bytes */
uint8_t *rx_buf; /* Rx buf */
uint8_t mode;
uint8_t flag;
int error;
};
/* feature registers */
#define REG_BLOCK_LOCK 0xa0
#define REG_CFG 0xb0
#define REG_STATUS 0xc0
#define REG_DIE_SELECT 0xd0
/* status */
#define STATUS_OIP_MASK 0x01
#define STATUS_CRBSY_MASK 0x80
#define STATUS_READY (0 << 0)
#define STATUS_BUSY (1 << 0)
#define STATUS_E_FAIL_MASK 0x04
#define STATUS_E_FAIL (1 << 2)
#define STATUS_P_FAIL_MASK 0x08
#define STATUS_P_FAIL (1 << 3)
enum {
NORMAL_MODE,
OTP_MODE,
OTP_PROTECT_MODE,
SNOR_READ_ENABLE_MODE,
};
/* Ability of SPI Controllers */
#define SPI_OPM_RX_QUAD 1 << 2 /* Support 4 pin RX */
#define SPI_OPM_RX_DUAL 1 << 1 /* Support 2 pin RX */
#define SPI_OPM_RX 1 << 0 /* Support 1 pin RX */
#define SPI_OPM_TX_QUAD 1 << 2 /* Support 4 pin TX */
#define SPI_OPM_TX 1 << 0 /* Support 1 pin TX */
static inline uint8_t ilog2(unsigned int v)
{
uint8_t l = 0;
while ((1UL << l) < v)
l++;
return l;
}
#endif