18d36fe1eSThomas Gleixner // SPDX-License-Identifier: GPL-2.0-only 2d90db4a0SLee Jones /* 3d90db4a0SLee Jones * st_spi_fsm.c - ST Fast Sequence Mode (FSM) Serial Flash Controller 4d90db4a0SLee Jones * 5d90db4a0SLee Jones * Author: Angus Clark <angus.clark@st.com> 6d90db4a0SLee Jones * 7f1919cb8SLee Jones * Copyright (C) 2010-2014 STMicroelectronics Limited 8d90db4a0SLee Jones * 9d90db4a0SLee Jones * JEDEC probe based on drivers/mtd/devices/m25p80.c 10d90db4a0SLee Jones */ 11d90db4a0SLee Jones #include <linux/kernel.h> 12d90db4a0SLee Jones #include <linux/module.h> 13a63984c1SLee Jones #include <linux/regmap.h> 14d90db4a0SLee Jones #include <linux/platform_device.h> 15a63984c1SLee Jones #include <linux/mfd/syscon.h> 16d90db4a0SLee Jones #include <linux/mtd/mtd.h> 17221cff13SLee Jones #include <linux/mtd/partitions.h> 186c8e1b33SBrian Norris #include <linux/mtd/spi-nor.h> 19d90db4a0SLee Jones #include <linux/sched.h> 20d90db4a0SLee Jones #include <linux/delay.h> 21d90db4a0SLee Jones #include <linux/io.h> 22d90db4a0SLee Jones #include <linux/of.h> 2369d5af8dSLee Jones #include <linux/clk.h> 24d90db4a0SLee Jones 255549fbd5SLee Jones #include "serial_flash_cmds.h" 265549fbd5SLee Jones 27bc09fb57SLee Jones /* 28bc09fb57SLee Jones * FSM SPI Controller Registers 29bc09fb57SLee Jones */ 30bc09fb57SLee Jones #define SPI_CLOCKDIV 0x0010 31bc09fb57SLee Jones #define SPI_MODESELECT 0x0018 32bc09fb57SLee Jones #define SPI_CONFIGDATA 0x0020 33bc09fb57SLee Jones #define SPI_STA_MODE_CHANGE 0x0028 34bc09fb57SLee Jones #define SPI_FAST_SEQ_TRANSFER_SIZE 0x0100 35bc09fb57SLee Jones #define SPI_FAST_SEQ_ADD1 0x0104 36bc09fb57SLee Jones #define SPI_FAST_SEQ_ADD2 0x0108 37bc09fb57SLee Jones #define SPI_FAST_SEQ_ADD_CFG 0x010c 38bc09fb57SLee Jones #define SPI_FAST_SEQ_OPC1 0x0110 39bc09fb57SLee Jones #define SPI_FAST_SEQ_OPC2 0x0114 40bc09fb57SLee Jones #define SPI_FAST_SEQ_OPC3 0x0118 41bc09fb57SLee Jones #define SPI_FAST_SEQ_OPC4 0x011c 42bc09fb57SLee Jones #define SPI_FAST_SEQ_OPC5 0x0120 43bc09fb57SLee Jones #define SPI_MODE_BITS 0x0124 44bc09fb57SLee Jones #define SPI_DUMMY_BITS 0x0128 45bc09fb57SLee Jones #define SPI_FAST_SEQ_FLASH_STA_DATA 0x012c 46bc09fb57SLee Jones #define SPI_FAST_SEQ_1 0x0130 47bc09fb57SLee Jones #define SPI_FAST_SEQ_2 0x0134 48bc09fb57SLee Jones #define SPI_FAST_SEQ_3 0x0138 49bc09fb57SLee Jones #define SPI_FAST_SEQ_4 0x013c 50bc09fb57SLee Jones #define SPI_FAST_SEQ_CFG 0x0140 51bc09fb57SLee Jones #define SPI_FAST_SEQ_STA 0x0144 52bc09fb57SLee Jones #define SPI_QUAD_BOOT_SEQ_INIT_1 0x0148 53bc09fb57SLee Jones #define SPI_QUAD_BOOT_SEQ_INIT_2 0x014c 54bc09fb57SLee Jones #define SPI_QUAD_BOOT_READ_SEQ_1 0x0150 55bc09fb57SLee Jones #define SPI_QUAD_BOOT_READ_SEQ_2 0x0154 56bc09fb57SLee Jones #define SPI_PROGRAM_ERASE_TIME 0x0158 57bc09fb57SLee Jones #define SPI_MULT_PAGE_REPEAT_SEQ_1 0x015c 58bc09fb57SLee Jones #define SPI_MULT_PAGE_REPEAT_SEQ_2 0x0160 59bc09fb57SLee Jones #define SPI_STATUS_WR_TIME_REG 0x0164 60bc09fb57SLee Jones #define SPI_FAST_SEQ_DATA_REG 0x0300 61bc09fb57SLee Jones 62bc09fb57SLee Jones /* 63bc09fb57SLee Jones * Register: SPI_MODESELECT 64bc09fb57SLee Jones */ 65bc09fb57SLee Jones #define SPI_MODESELECT_CONTIG 0x01 66bc09fb57SLee Jones #define SPI_MODESELECT_FASTREAD 0x02 67bc09fb57SLee Jones #define SPI_MODESELECT_DUALIO 0x04 68bc09fb57SLee Jones #define SPI_MODESELECT_FSM 0x08 69bc09fb57SLee Jones #define SPI_MODESELECT_QUADBOOT 0x10 70bc09fb57SLee Jones 71bc09fb57SLee Jones /* 72bc09fb57SLee Jones * Register: SPI_CONFIGDATA 73bc09fb57SLee Jones */ 74bc09fb57SLee Jones #define SPI_CFG_DEVICE_ST 0x1 75bc09fb57SLee Jones #define SPI_CFG_DEVICE_ATMEL 0x4 76bc09fb57SLee Jones #define SPI_CFG_MIN_CS_HIGH(x) (((x) & 0xfff) << 4) 77bc09fb57SLee Jones #define SPI_CFG_CS_SETUPHOLD(x) (((x) & 0xff) << 16) 78bc09fb57SLee Jones #define SPI_CFG_DATA_HOLD(x) (((x) & 0xff) << 24) 79bc09fb57SLee Jones 8086f309fdSLee Jones #define SPI_CFG_DEFAULT_MIN_CS_HIGH SPI_CFG_MIN_CS_HIGH(0x0AA) 8186f309fdSLee Jones #define SPI_CFG_DEFAULT_CS_SETUPHOLD SPI_CFG_CS_SETUPHOLD(0xA0) 8286f309fdSLee Jones #define SPI_CFG_DEFAULT_DATA_HOLD SPI_CFG_DATA_HOLD(0x00) 8386f309fdSLee Jones 84bc09fb57SLee Jones /* 85bc09fb57SLee Jones * Register: SPI_FAST_SEQ_TRANSFER_SIZE 86bc09fb57SLee Jones */ 87bc09fb57SLee Jones #define TRANSFER_SIZE(x) ((x) * 8) 88bc09fb57SLee Jones 89bc09fb57SLee Jones /* 90bc09fb57SLee Jones * Register: SPI_FAST_SEQ_ADD_CFG 91bc09fb57SLee Jones */ 92bc09fb57SLee Jones #define ADR_CFG_CYCLES_ADD1(x) ((x) << 0) 93bc09fb57SLee Jones #define ADR_CFG_PADS_1_ADD1 (0x0 << 6) 94bc09fb57SLee Jones #define ADR_CFG_PADS_2_ADD1 (0x1 << 6) 95bc09fb57SLee Jones #define ADR_CFG_PADS_4_ADD1 (0x3 << 6) 96bc09fb57SLee Jones #define ADR_CFG_CSDEASSERT_ADD1 (1 << 8) 97bc09fb57SLee Jones #define ADR_CFG_CYCLES_ADD2(x) ((x) << (0+16)) 98bc09fb57SLee Jones #define ADR_CFG_PADS_1_ADD2 (0x0 << (6+16)) 99bc09fb57SLee Jones #define ADR_CFG_PADS_2_ADD2 (0x1 << (6+16)) 100bc09fb57SLee Jones #define ADR_CFG_PADS_4_ADD2 (0x3 << (6+16)) 101bc09fb57SLee Jones #define ADR_CFG_CSDEASSERT_ADD2 (1 << (8+16)) 102bc09fb57SLee Jones 103bc09fb57SLee Jones /* 104bc09fb57SLee Jones * Register: SPI_FAST_SEQ_n 105bc09fb57SLee Jones */ 106bc09fb57SLee Jones #define SEQ_OPC_OPCODE(x) ((x) << 0) 107bc09fb57SLee Jones #define SEQ_OPC_CYCLES(x) ((x) << 8) 108bc09fb57SLee Jones #define SEQ_OPC_PADS_1 (0x0 << 14) 109bc09fb57SLee Jones #define SEQ_OPC_PADS_2 (0x1 << 14) 110bc09fb57SLee Jones #define SEQ_OPC_PADS_4 (0x3 << 14) 111bc09fb57SLee Jones #define SEQ_OPC_CSDEASSERT (1 << 16) 112bc09fb57SLee Jones 113bc09fb57SLee Jones /* 114bc09fb57SLee Jones * Register: SPI_FAST_SEQ_CFG 115bc09fb57SLee Jones */ 116bc09fb57SLee Jones #define SEQ_CFG_STARTSEQ (1 << 0) 117bc09fb57SLee Jones #define SEQ_CFG_SWRESET (1 << 5) 118bc09fb57SLee Jones #define SEQ_CFG_CSDEASSERT (1 << 6) 119bc09fb57SLee Jones #define SEQ_CFG_READNOTWRITE (1 << 7) 120bc09fb57SLee Jones #define SEQ_CFG_ERASE (1 << 8) 121bc09fb57SLee Jones #define SEQ_CFG_PADS_1 (0x0 << 16) 122bc09fb57SLee Jones #define SEQ_CFG_PADS_2 (0x1 << 16) 123bc09fb57SLee Jones #define SEQ_CFG_PADS_4 (0x3 << 16) 124bc09fb57SLee Jones 125bc09fb57SLee Jones /* 126bc09fb57SLee Jones * Register: SPI_MODE_BITS 127bc09fb57SLee Jones */ 128bc09fb57SLee Jones #define MODE_DATA(x) (x & 0xff) 129bc09fb57SLee Jones #define MODE_CYCLES(x) ((x & 0x3f) << 16) 130bc09fb57SLee Jones #define MODE_PADS_1 (0x0 << 22) 131bc09fb57SLee Jones #define MODE_PADS_2 (0x1 << 22) 132bc09fb57SLee Jones #define MODE_PADS_4 (0x3 << 22) 133bc09fb57SLee Jones #define DUMMY_CSDEASSERT (1 << 24) 134bc09fb57SLee Jones 135bc09fb57SLee Jones /* 136bc09fb57SLee Jones * Register: SPI_DUMMY_BITS 137bc09fb57SLee Jones */ 138bc09fb57SLee Jones #define DUMMY_CYCLES(x) ((x & 0x3f) << 16) 139bc09fb57SLee Jones #define DUMMY_PADS_1 (0x0 << 22) 140bc09fb57SLee Jones #define DUMMY_PADS_2 (0x1 << 22) 141bc09fb57SLee Jones #define DUMMY_PADS_4 (0x3 << 22) 142bc09fb57SLee Jones #define DUMMY_CSDEASSERT (1 << 24) 143bc09fb57SLee Jones 144bc09fb57SLee Jones /* 145bc09fb57SLee Jones * Register: SPI_FAST_SEQ_FLASH_STA_DATA 146bc09fb57SLee Jones */ 147bc09fb57SLee Jones #define STA_DATA_BYTE1(x) ((x & 0xff) << 0) 148bc09fb57SLee Jones #define STA_DATA_BYTE2(x) ((x & 0xff) << 8) 149bc09fb57SLee Jones #define STA_PADS_1 (0x0 << 16) 150bc09fb57SLee Jones #define STA_PADS_2 (0x1 << 16) 151bc09fb57SLee Jones #define STA_PADS_4 (0x3 << 16) 152bc09fb57SLee Jones #define STA_CSDEASSERT (0x1 << 20) 153bc09fb57SLee Jones #define STA_RDNOTWR (0x1 << 21) 154bc09fb57SLee Jones 155bc09fb57SLee Jones /* 156bc09fb57SLee Jones * FSM SPI Instruction Opcodes 157bc09fb57SLee Jones */ 158bc09fb57SLee Jones #define STFSM_OPC_CMD 0x1 159bc09fb57SLee Jones #define STFSM_OPC_ADD 0x2 160bc09fb57SLee Jones #define STFSM_OPC_STA 0x3 161bc09fb57SLee Jones #define STFSM_OPC_MODE 0x4 162bc09fb57SLee Jones #define STFSM_OPC_DUMMY 0x5 163bc09fb57SLee Jones #define STFSM_OPC_DATA 0x6 164bc09fb57SLee Jones #define STFSM_OPC_WAIT 0x7 165bc09fb57SLee Jones #define STFSM_OPC_JUMP 0x8 166bc09fb57SLee Jones #define STFSM_OPC_GOTO 0x9 167bc09fb57SLee Jones #define STFSM_OPC_STOP 0xF 168bc09fb57SLee Jones 169bc09fb57SLee Jones /* 170bc09fb57SLee Jones * FSM SPI Instructions (== opcode + operand). 171bc09fb57SLee Jones */ 172bc09fb57SLee Jones #define STFSM_INSTR(cmd, op) ((cmd) | ((op) << 4)) 173bc09fb57SLee Jones 174bc09fb57SLee Jones #define STFSM_INST_CMD1 STFSM_INSTR(STFSM_OPC_CMD, 1) 175bc09fb57SLee Jones #define STFSM_INST_CMD2 STFSM_INSTR(STFSM_OPC_CMD, 2) 176bc09fb57SLee Jones #define STFSM_INST_CMD3 STFSM_INSTR(STFSM_OPC_CMD, 3) 177bc09fb57SLee Jones #define STFSM_INST_CMD4 STFSM_INSTR(STFSM_OPC_CMD, 4) 178bc09fb57SLee Jones #define STFSM_INST_CMD5 STFSM_INSTR(STFSM_OPC_CMD, 5) 179bc09fb57SLee Jones #define STFSM_INST_ADD1 STFSM_INSTR(STFSM_OPC_ADD, 1) 180bc09fb57SLee Jones #define STFSM_INST_ADD2 STFSM_INSTR(STFSM_OPC_ADD, 2) 181bc09fb57SLee Jones 182bc09fb57SLee Jones #define STFSM_INST_DATA_WRITE STFSM_INSTR(STFSM_OPC_DATA, 1) 183bc09fb57SLee Jones #define STFSM_INST_DATA_READ STFSM_INSTR(STFSM_OPC_DATA, 2) 184bc09fb57SLee Jones 185bc09fb57SLee Jones #define STFSM_INST_STA_RD1 STFSM_INSTR(STFSM_OPC_STA, 0x1) 186bc09fb57SLee Jones #define STFSM_INST_STA_WR1 STFSM_INSTR(STFSM_OPC_STA, 0x1) 187bc09fb57SLee Jones #define STFSM_INST_STA_RD2 STFSM_INSTR(STFSM_OPC_STA, 0x2) 188bc09fb57SLee Jones #define STFSM_INST_STA_WR1_2 STFSM_INSTR(STFSM_OPC_STA, 0x3) 189bc09fb57SLee Jones 190bc09fb57SLee Jones #define STFSM_INST_MODE STFSM_INSTR(STFSM_OPC_MODE, 0) 191bc09fb57SLee Jones #define STFSM_INST_DUMMY STFSM_INSTR(STFSM_OPC_DUMMY, 0) 192bc09fb57SLee Jones #define STFSM_INST_WAIT STFSM_INSTR(STFSM_OPC_WAIT, 0) 193bc09fb57SLee Jones #define STFSM_INST_STOP STFSM_INSTR(STFSM_OPC_STOP, 0) 194bc09fb57SLee Jones 19586f309fdSLee Jones #define STFSM_DEFAULT_EMI_FREQ 100000000UL /* 100 MHz */ 19686f309fdSLee Jones #define STFSM_DEFAULT_WR_TIME (STFSM_DEFAULT_EMI_FREQ * (15/1000)) /* 15ms */ 19786f309fdSLee Jones 19886f309fdSLee Jones #define STFSM_FLASH_SAFE_FREQ 10000000UL /* 10 MHz */ 19986f309fdSLee Jones 2003c8b85b3SLee Jones #define STFSM_MAX_WAIT_SEQ_MS 1000 /* FSM execution time */ 2013c8b85b3SLee Jones 2025343a123SLee Jones /* S25FLxxxS commands */ 2035343a123SLee Jones #define S25FL_CMD_WRITE4_1_1_4 0x34 2045343a123SLee Jones #define S25FL_CMD_SE4 0xdc 2055343a123SLee Jones #define S25FL_CMD_CLSR 0x30 2065343a123SLee Jones #define S25FL_CMD_DYBWR 0xe1 2075343a123SLee Jones #define S25FL_CMD_DYBRD 0xe0 2085343a123SLee Jones #define S25FL_CMD_WRITE4 0x12 /* Note, opcode clashes with 20992d3af9aSBrian Norris * 'SPINOR_OP_WRITE_1_4_4' 2105343a123SLee Jones * as found on N25Qxxx devices! */ 2115343a123SLee Jones 212176b4377SLee Jones /* Status register */ 213176b4377SLee Jones #define FLASH_STATUS_BUSY 0x01 214176b4377SLee Jones #define FLASH_STATUS_WEL 0x02 215176b4377SLee Jones #define FLASH_STATUS_BP0 0x04 216176b4377SLee Jones #define FLASH_STATUS_BP1 0x08 217176b4377SLee Jones #define FLASH_STATUS_BP2 0x10 218176b4377SLee Jones #define FLASH_STATUS_SRWP0 0x80 219176b4377SLee Jones #define FLASH_STATUS_TIMEOUT 0xff 2205343a123SLee Jones /* S25FL Error Flags */ 2215343a123SLee Jones #define S25FL_STATUS_E_ERR 0x20 2225343a123SLee Jones #define S25FL_STATUS_P_ERR 0x40 223176b4377SLee Jones 2245d0bddabSAngus Clark #define N25Q_CMD_WRVCR 0x81 2255d0bddabSAngus Clark #define N25Q_CMD_RDVCR 0x85 2265d0bddabSAngus Clark #define N25Q_CMD_RDVECR 0x65 2275d0bddabSAngus Clark #define N25Q_CMD_RDNVCR 0xb5 2285d0bddabSAngus Clark #define N25Q_CMD_WRNVCR 0xb1 2295d0bddabSAngus Clark 230e514f105SLee Jones #define FLASH_PAGESIZE 256 /* In Bytes */ 231e514f105SLee Jones #define FLASH_PAGESIZE_32 (FLASH_PAGESIZE / 4) /* In uint32_t */ 232176b4377SLee Jones #define FLASH_MAX_BUSY_WAIT (300 * HZ) /* Maximum 'CHIPERASE' time */ 233e514f105SLee Jones 234e85a6196SLee Jones /* 235e85a6196SLee Jones * Flags to tweak operation of default read/write/erase routines 236e85a6196SLee Jones */ 237e85a6196SLee Jones #define CFG_READ_TOGGLE_32BIT_ADDR 0x00000001 238e85a6196SLee Jones #define CFG_WRITE_TOGGLE_32BIT_ADDR 0x00000002 239e85a6196SLee Jones #define CFG_ERASESEC_TOGGLE_32BIT_ADDR 0x00000008 240e85a6196SLee Jones #define CFG_S25FL_CHECK_ERROR_FLAGS 0x00000010 241e85a6196SLee Jones 242e6b1bb4eSLee Jones struct stfsm_seq { 243e6b1bb4eSLee Jones uint32_t data_size; 244e6b1bb4eSLee Jones uint32_t addr1; 245e6b1bb4eSLee Jones uint32_t addr2; 246e6b1bb4eSLee Jones uint32_t addr_cfg; 247e6b1bb4eSLee Jones uint32_t seq_opc[5]; 248e6b1bb4eSLee Jones uint32_t mode; 249e6b1bb4eSLee Jones uint32_t dummy; 250e6b1bb4eSLee Jones uint32_t status; 251e6b1bb4eSLee Jones uint8_t seq[16]; 252e6b1bb4eSLee Jones uint32_t seq_cfg; 253e6b1bb4eSLee Jones } __packed __aligned(4); 254e6b1bb4eSLee Jones 255d90db4a0SLee Jones struct stfsm { 256d90db4a0SLee Jones struct device *dev; 257d90db4a0SLee Jones void __iomem *base; 258d90db4a0SLee Jones struct mtd_info mtd; 259d90db4a0SLee Jones struct mutex lock; 26024fec651SLee Jones struct flash_info *info; 26169d5af8dSLee Jones struct clk *clk; 26286f309fdSLee Jones 263e85a6196SLee Jones uint32_t configuration; 26486f309fdSLee Jones uint32_t fifo_dir_delay; 265a63984c1SLee Jones bool booted_from_spi; 2660ea7d706SLee Jones bool reset_signal; 2670ea7d706SLee Jones bool reset_por; 268d90db4a0SLee Jones 269e6b1bb4eSLee Jones struct stfsm_seq stfsm_seq_read; 270e6b1bb4eSLee Jones struct stfsm_seq stfsm_seq_write; 271e6b1bb4eSLee Jones struct stfsm_seq stfsm_seq_en_32bit_addr; 272e6b1bb4eSLee Jones }; 2733c8b85b3SLee Jones 27408981274SLee Jones /* Parameters to configure a READ or WRITE FSM sequence */ 27508981274SLee Jones struct seq_rw_config { 27608981274SLee Jones uint32_t flags; /* flags to support config */ 27708981274SLee Jones uint8_t cmd; /* FLASH command */ 27808981274SLee Jones int write; /* Write Sequence */ 27908981274SLee Jones uint8_t addr_pads; /* No. of addr pads (MODE & DUMMY) */ 28008981274SLee Jones uint8_t data_pads; /* No. of data pads */ 28108981274SLee Jones uint8_t mode_data; /* MODE data */ 28208981274SLee Jones uint8_t mode_cycles; /* No. of MODE cycles */ 28308981274SLee Jones uint8_t dummy_cycles; /* No. of DUMMY cycles */ 28408981274SLee Jones }; 28508981274SLee Jones 28611d7f826SLee Jones /* SPI Flash Device Table */ 28711d7f826SLee Jones struct flash_info { 28811d7f826SLee Jones char *name; 28911d7f826SLee Jones /* 29011d7f826SLee Jones * JEDEC id zero means "no ID" (most older chips); otherwise it has 29111d7f826SLee Jones * a high byte of zero plus three data bytes: the manufacturer id, 29211d7f826SLee Jones * then a two byte device id. 29311d7f826SLee Jones */ 29411d7f826SLee Jones u32 jedec_id; 29511d7f826SLee Jones u16 ext_id; 29611d7f826SLee Jones /* 29792d3af9aSBrian Norris * The size listed here is what works with SPINOR_OP_SE, which isn't 29811d7f826SLee Jones * necessarily called a "sector" by the vendor. 29911d7f826SLee Jones */ 30011d7f826SLee Jones unsigned sector_size; 30111d7f826SLee Jones u16 n_sectors; 30211d7f826SLee Jones u32 flags; 30311d7f826SLee Jones /* 30411d7f826SLee Jones * Note, where FAST_READ is supported, freq_max specifies the 30511d7f826SLee Jones * FAST_READ frequency, not the READ frequency. 30611d7f826SLee Jones */ 30711d7f826SLee Jones u32 max_freq; 30811d7f826SLee Jones int (*config)(struct stfsm *); 30911d7f826SLee Jones }; 31011d7f826SLee Jones 311218b870fSLee Jones static int stfsm_n25q_config(struct stfsm *fsm); 31289818066SLee Jones static int stfsm_mx25_config(struct stfsm *fsm); 3135343a123SLee Jones static int stfsm_s25fl_config(struct stfsm *fsm); 314cd7cac9eSLee Jones static int stfsm_w25q_config(struct stfsm *fsm); 315218b870fSLee Jones 31611d7f826SLee Jones static struct flash_info flash_types[] = { 31711d7f826SLee Jones /* 31811d7f826SLee Jones * ST Microelectronics/Numonyx -- 31911d7f826SLee Jones * (newer production versions may have feature updates 32011d7f826SLee Jones * (eg faster operating frequency) 32111d7f826SLee Jones */ 32211d7f826SLee Jones #define M25P_FLAG (FLASH_FLAG_READ_WRITE | FLASH_FLAG_READ_FAST) 32311d7f826SLee Jones { "m25p40", 0x202013, 0, 64 * 1024, 8, M25P_FLAG, 25, NULL }, 32411d7f826SLee Jones { "m25p80", 0x202014, 0, 64 * 1024, 16, M25P_FLAG, 25, NULL }, 32511d7f826SLee Jones { "m25p16", 0x202015, 0, 64 * 1024, 32, M25P_FLAG, 25, NULL }, 32611d7f826SLee Jones { "m25p32", 0x202016, 0, 64 * 1024, 64, M25P_FLAG, 50, NULL }, 32711d7f826SLee Jones { "m25p64", 0x202017, 0, 64 * 1024, 128, M25P_FLAG, 50, NULL }, 32811d7f826SLee Jones { "m25p128", 0x202018, 0, 256 * 1024, 64, M25P_FLAG, 50, NULL }, 32911d7f826SLee Jones 33011d7f826SLee Jones #define M25PX_FLAG (FLASH_FLAG_READ_WRITE | \ 33111d7f826SLee Jones FLASH_FLAG_READ_FAST | \ 33211d7f826SLee Jones FLASH_FLAG_READ_1_1_2 | \ 33311d7f826SLee Jones FLASH_FLAG_WRITE_1_1_2) 33411d7f826SLee Jones { "m25px32", 0x207116, 0, 64 * 1024, 64, M25PX_FLAG, 75, NULL }, 33511d7f826SLee Jones { "m25px64", 0x207117, 0, 64 * 1024, 128, M25PX_FLAG, 75, NULL }, 33611d7f826SLee Jones 3376b6d3737SAngus Clark /* Macronix MX25xxx 3386b6d3737SAngus Clark * - Support for 'FLASH_FLAG_WRITE_1_4_4' is omitted for devices 3396b6d3737SAngus Clark * where operating frequency must be reduced. 3406b6d3737SAngus Clark */ 34111d7f826SLee Jones #define MX25_FLAG (FLASH_FLAG_READ_WRITE | \ 34211d7f826SLee Jones FLASH_FLAG_READ_FAST | \ 34311d7f826SLee Jones FLASH_FLAG_READ_1_1_2 | \ 34411d7f826SLee Jones FLASH_FLAG_READ_1_2_2 | \ 34511d7f826SLee Jones FLASH_FLAG_READ_1_1_4 | \ 34611d7f826SLee Jones FLASH_FLAG_SE_4K | \ 34711d7f826SLee Jones FLASH_FLAG_SE_32K) 3486b6d3737SAngus Clark { "mx25l3255e", 0xc29e16, 0, 64 * 1024, 64, 3496b6d3737SAngus Clark (MX25_FLAG | FLASH_FLAG_WRITE_1_4_4), 86, 3506b6d3737SAngus Clark stfsm_mx25_config}, 35111d7f826SLee Jones { "mx25l25635e", 0xc22019, 0, 64*1024, 512, 35289818066SLee Jones (MX25_FLAG | FLASH_FLAG_32BIT_ADDR | FLASH_FLAG_RESET), 70, 35389818066SLee Jones stfsm_mx25_config }, 3545fa98069SAngus Clark { "mx25l25655e", 0xc22619, 0, 64*1024, 512, 3555fa98069SAngus Clark (MX25_FLAG | FLASH_FLAG_32BIT_ADDR | FLASH_FLAG_RESET), 70, 3565fa98069SAngus Clark stfsm_mx25_config}, 35711d7f826SLee Jones 35811d7f826SLee Jones #define N25Q_FLAG (FLASH_FLAG_READ_WRITE | \ 35911d7f826SLee Jones FLASH_FLAG_READ_FAST | \ 36011d7f826SLee Jones FLASH_FLAG_READ_1_1_2 | \ 36111d7f826SLee Jones FLASH_FLAG_READ_1_2_2 | \ 36211d7f826SLee Jones FLASH_FLAG_READ_1_1_4 | \ 36311d7f826SLee Jones FLASH_FLAG_READ_1_4_4 | \ 36411d7f826SLee Jones FLASH_FLAG_WRITE_1_1_2 | \ 36511d7f826SLee Jones FLASH_FLAG_WRITE_1_2_2 | \ 36611d7f826SLee Jones FLASH_FLAG_WRITE_1_1_4 | \ 36711d7f826SLee Jones FLASH_FLAG_WRITE_1_4_4) 368218b870fSLee Jones { "n25q128", 0x20ba18, 0, 64 * 1024, 256, N25Q_FLAG, 108, 369218b870fSLee Jones stfsm_n25q_config }, 37011d7f826SLee Jones { "n25q256", 0x20ba19, 0, 64 * 1024, 512, 371218b870fSLee Jones N25Q_FLAG | FLASH_FLAG_32BIT_ADDR, 108, stfsm_n25q_config }, 37211d7f826SLee Jones 37311d7f826SLee Jones /* 37411d7f826SLee Jones * Spansion S25FLxxxP 37511d7f826SLee Jones * - 256KiB and 64KiB sector variants (identified by ext. JEDEC) 37611d7f826SLee Jones */ 37711d7f826SLee Jones #define S25FLXXXP_FLAG (FLASH_FLAG_READ_WRITE | \ 37811d7f826SLee Jones FLASH_FLAG_READ_1_1_2 | \ 37911d7f826SLee Jones FLASH_FLAG_READ_1_2_2 | \ 38011d7f826SLee Jones FLASH_FLAG_READ_1_1_4 | \ 38111d7f826SLee Jones FLASH_FLAG_READ_1_4_4 | \ 38211d7f826SLee Jones FLASH_FLAG_WRITE_1_1_4 | \ 38311d7f826SLee Jones FLASH_FLAG_READ_FAST) 38485bdcf6bSAngus Clark { "s25fl032p", 0x010215, 0x4d00, 64 * 1024, 64, S25FLXXXP_FLAG, 80, 38585bdcf6bSAngus Clark stfsm_s25fl_config}, 38611d7f826SLee Jones { "s25fl129p0", 0x012018, 0x4d00, 256 * 1024, 64, S25FLXXXP_FLAG, 80, 3875343a123SLee Jones stfsm_s25fl_config }, 38811d7f826SLee Jones { "s25fl129p1", 0x012018, 0x4d01, 64 * 1024, 256, S25FLXXXP_FLAG, 80, 3895343a123SLee Jones stfsm_s25fl_config }, 39011d7f826SLee Jones 39111d7f826SLee Jones /* 39211d7f826SLee Jones * Spansion S25FLxxxS 39311d7f826SLee Jones * - 256KiB and 64KiB sector variants (identified by ext. JEDEC) 39411d7f826SLee Jones * - RESET# signal supported by die but not bristled out on all 39511d7f826SLee Jones * package types. The package type is a function of board design, 39611d7f826SLee Jones * so this information is captured in the board's flags. 39711d7f826SLee Jones * - Supports 'DYB' sector protection. Depending on variant, sectors 39811d7f826SLee Jones * may default to locked state on power-on. 39911d7f826SLee Jones */ 40011d7f826SLee Jones #define S25FLXXXS_FLAG (S25FLXXXP_FLAG | \ 40111d7f826SLee Jones FLASH_FLAG_RESET | \ 40211d7f826SLee Jones FLASH_FLAG_DYB_LOCKING) 40311d7f826SLee Jones { "s25fl128s0", 0x012018, 0x0300, 256 * 1024, 64, S25FLXXXS_FLAG, 80, 4045343a123SLee Jones stfsm_s25fl_config }, 40511d7f826SLee Jones { "s25fl128s1", 0x012018, 0x0301, 64 * 1024, 256, S25FLXXXS_FLAG, 80, 4065343a123SLee Jones stfsm_s25fl_config }, 40711d7f826SLee Jones { "s25fl256s0", 0x010219, 0x4d00, 256 * 1024, 128, 4085343a123SLee Jones S25FLXXXS_FLAG | FLASH_FLAG_32BIT_ADDR, 80, stfsm_s25fl_config }, 40911d7f826SLee Jones { "s25fl256s1", 0x010219, 0x4d01, 64 * 1024, 512, 4105343a123SLee Jones S25FLXXXS_FLAG | FLASH_FLAG_32BIT_ADDR, 80, stfsm_s25fl_config }, 41111d7f826SLee Jones 41211d7f826SLee Jones /* Winbond -- w25x "blocks" are 64K, "sectors" are 4KiB */ 41311d7f826SLee Jones #define W25X_FLAG (FLASH_FLAG_READ_WRITE | \ 41411d7f826SLee Jones FLASH_FLAG_READ_FAST | \ 41511d7f826SLee Jones FLASH_FLAG_READ_1_1_2 | \ 41611d7f826SLee Jones FLASH_FLAG_WRITE_1_1_2) 41711d7f826SLee Jones { "w25x40", 0xef3013, 0, 64 * 1024, 8, W25X_FLAG, 75, NULL }, 41811d7f826SLee Jones { "w25x80", 0xef3014, 0, 64 * 1024, 16, W25X_FLAG, 75, NULL }, 41911d7f826SLee Jones { "w25x16", 0xef3015, 0, 64 * 1024, 32, W25X_FLAG, 75, NULL }, 42011d7f826SLee Jones { "w25x32", 0xef3016, 0, 64 * 1024, 64, W25X_FLAG, 75, NULL }, 42111d7f826SLee Jones { "w25x64", 0xef3017, 0, 64 * 1024, 128, W25X_FLAG, 75, NULL }, 42211d7f826SLee Jones 42311d7f826SLee Jones /* Winbond -- w25q "blocks" are 64K, "sectors" are 4KiB */ 42411d7f826SLee Jones #define W25Q_FLAG (FLASH_FLAG_READ_WRITE | \ 42511d7f826SLee Jones FLASH_FLAG_READ_FAST | \ 42611d7f826SLee Jones FLASH_FLAG_READ_1_1_2 | \ 42711d7f826SLee Jones FLASH_FLAG_READ_1_2_2 | \ 42811d7f826SLee Jones FLASH_FLAG_READ_1_1_4 | \ 42911d7f826SLee Jones FLASH_FLAG_READ_1_4_4 | \ 43011d7f826SLee Jones FLASH_FLAG_WRITE_1_1_4) 431cd7cac9eSLee Jones { "w25q80", 0xef4014, 0, 64 * 1024, 16, W25Q_FLAG, 80, 432cd7cac9eSLee Jones stfsm_w25q_config }, 433cd7cac9eSLee Jones { "w25q16", 0xef4015, 0, 64 * 1024, 32, W25Q_FLAG, 80, 434cd7cac9eSLee Jones stfsm_w25q_config }, 435cd7cac9eSLee Jones { "w25q32", 0xef4016, 0, 64 * 1024, 64, W25Q_FLAG, 80, 436cd7cac9eSLee Jones stfsm_w25q_config }, 437cd7cac9eSLee Jones { "w25q64", 0xef4017, 0, 64 * 1024, 128, W25Q_FLAG, 80, 438cd7cac9eSLee Jones stfsm_w25q_config }, 43911d7f826SLee Jones 44011d7f826SLee Jones /* Sentinel */ 44111d7f826SLee Jones { NULL, 0x000000, 0, 0, 0, 0, 0, NULL }, 44211d7f826SLee Jones }; 44311d7f826SLee Jones 444a37b2f5aSLee Jones /* 445a37b2f5aSLee Jones * FSM message sequence configurations: 446a37b2f5aSLee Jones * 447a37b2f5aSLee Jones * All configs are presented in order of preference 448a37b2f5aSLee Jones */ 449a37b2f5aSLee Jones 450a37b2f5aSLee Jones /* Default READ configurations, in order of preference */ 451a37b2f5aSLee Jones static struct seq_rw_config default_read_configs[] = { 45292d3af9aSBrian Norris {FLASH_FLAG_READ_1_4_4, SPINOR_OP_READ_1_4_4, 0, 4, 4, 0x00, 2, 4}, 45392d3af9aSBrian Norris {FLASH_FLAG_READ_1_1_4, SPINOR_OP_READ_1_1_4, 0, 1, 4, 0x00, 4, 0}, 45492d3af9aSBrian Norris {FLASH_FLAG_READ_1_2_2, SPINOR_OP_READ_1_2_2, 0, 2, 2, 0x00, 4, 0}, 45592d3af9aSBrian Norris {FLASH_FLAG_READ_1_1_2, SPINOR_OP_READ_1_1_2, 0, 1, 2, 0x00, 0, 8}, 45692d3af9aSBrian Norris {FLASH_FLAG_READ_FAST, SPINOR_OP_READ_FAST, 0, 1, 1, 0x00, 0, 8}, 45792d3af9aSBrian Norris {FLASH_FLAG_READ_WRITE, SPINOR_OP_READ, 0, 1, 1, 0x00, 0, 0}, 458a37b2f5aSLee Jones {0x00, 0, 0, 0, 0, 0x00, 0, 0}, 459a37b2f5aSLee Jones }; 460a37b2f5aSLee Jones 461a37b2f5aSLee Jones /* Default WRITE configurations */ 462a37b2f5aSLee Jones static struct seq_rw_config default_write_configs[] = { 46392d3af9aSBrian Norris {FLASH_FLAG_WRITE_1_4_4, SPINOR_OP_WRITE_1_4_4, 1, 4, 4, 0x00, 0, 0}, 46492d3af9aSBrian Norris {FLASH_FLAG_WRITE_1_1_4, SPINOR_OP_WRITE_1_1_4, 1, 1, 4, 0x00, 0, 0}, 46592d3af9aSBrian Norris {FLASH_FLAG_WRITE_1_2_2, SPINOR_OP_WRITE_1_2_2, 1, 2, 2, 0x00, 0, 0}, 46692d3af9aSBrian Norris {FLASH_FLAG_WRITE_1_1_2, SPINOR_OP_WRITE_1_1_2, 1, 1, 2, 0x00, 0, 0}, 46792d3af9aSBrian Norris {FLASH_FLAG_READ_WRITE, SPINOR_OP_WRITE, 1, 1, 1, 0x00, 0, 0}, 468a37b2f5aSLee Jones {0x00, 0, 0, 0, 0, 0x00, 0, 0}, 469a37b2f5aSLee Jones }; 470a37b2f5aSLee Jones 471e85a6196SLee Jones /* 472e85a6196SLee Jones * [N25Qxxx] Configuration 473e85a6196SLee Jones */ 474e85a6196SLee Jones #define N25Q_VCR_DUMMY_CYCLES(x) (((x) & 0xf) << 4) 475e85a6196SLee Jones #define N25Q_VCR_XIP_DISABLED ((uint8_t)0x1 << 3) 476e85a6196SLee Jones #define N25Q_VCR_WRAP_CONT 0x3 477e85a6196SLee Jones 478e85a6196SLee Jones /* N25Q 3-byte Address READ configurations 479e85a6196SLee Jones * - 'FAST' variants configured for 8 dummy cycles. 480e85a6196SLee Jones * 481e85a6196SLee Jones * Note, the number of dummy cycles used for 'FAST' READ operations is 482e85a6196SLee Jones * configurable and would normally be tuned according to the READ command and 483e85a6196SLee Jones * operating frequency. However, this applies universally to all 'FAST' READ 484e85a6196SLee Jones * commands, including those used by the SPIBoot controller, and remains in 485e85a6196SLee Jones * force until the device is power-cycled. Since the SPIBoot controller is 486e85a6196SLee Jones * hard-wired to use 8 dummy cycles, we must configure the device to also use 8 487e85a6196SLee Jones * cycles. 488e85a6196SLee Jones */ 489e85a6196SLee Jones static struct seq_rw_config n25q_read3_configs[] = { 49092d3af9aSBrian Norris {FLASH_FLAG_READ_1_4_4, SPINOR_OP_READ_1_4_4, 0, 4, 4, 0x00, 0, 8}, 49192d3af9aSBrian Norris {FLASH_FLAG_READ_1_1_4, SPINOR_OP_READ_1_1_4, 0, 1, 4, 0x00, 0, 8}, 49292d3af9aSBrian Norris {FLASH_FLAG_READ_1_2_2, SPINOR_OP_READ_1_2_2, 0, 2, 2, 0x00, 0, 8}, 49392d3af9aSBrian Norris {FLASH_FLAG_READ_1_1_2, SPINOR_OP_READ_1_1_2, 0, 1, 2, 0x00, 0, 8}, 49492d3af9aSBrian Norris {FLASH_FLAG_READ_FAST, SPINOR_OP_READ_FAST, 0, 1, 1, 0x00, 0, 8}, 49592d3af9aSBrian Norris {FLASH_FLAG_READ_WRITE, SPINOR_OP_READ, 0, 1, 1, 0x00, 0, 0}, 496e85a6196SLee Jones {0x00, 0, 0, 0, 0, 0x00, 0, 0}, 497e85a6196SLee Jones }; 498e85a6196SLee Jones 499e85a6196SLee Jones /* N25Q 4-byte Address READ configurations 500e85a6196SLee Jones * - use special 4-byte address READ commands (reduces overheads, and 501e85a6196SLee Jones * reduces risk of hitting watchdog reset issues). 502e85a6196SLee Jones * - 'FAST' variants configured for 8 dummy cycles (see note above.) 503e85a6196SLee Jones */ 504e85a6196SLee Jones static struct seq_rw_config n25q_read4_configs[] = { 505902cc69aSCyrille Pitchen {FLASH_FLAG_READ_1_4_4, SPINOR_OP_READ_1_4_4_4B, 0, 4, 4, 0x00, 0, 8}, 506902cc69aSCyrille Pitchen {FLASH_FLAG_READ_1_1_4, SPINOR_OP_READ_1_1_4_4B, 0, 1, 4, 0x00, 0, 8}, 507902cc69aSCyrille Pitchen {FLASH_FLAG_READ_1_2_2, SPINOR_OP_READ_1_2_2_4B, 0, 2, 2, 0x00, 0, 8}, 508902cc69aSCyrille Pitchen {FLASH_FLAG_READ_1_1_2, SPINOR_OP_READ_1_1_2_4B, 0, 1, 2, 0x00, 0, 8}, 509902cc69aSCyrille Pitchen {FLASH_FLAG_READ_FAST, SPINOR_OP_READ_FAST_4B, 0, 1, 1, 0x00, 0, 8}, 510902cc69aSCyrille Pitchen {FLASH_FLAG_READ_WRITE, SPINOR_OP_READ_4B, 0, 1, 1, 0x00, 0, 0}, 511e85a6196SLee Jones {0x00, 0, 0, 0, 0, 0x00, 0, 0}, 512e85a6196SLee Jones }; 513e85a6196SLee Jones 51489818066SLee Jones /* 51589818066SLee Jones * [MX25xxx] Configuration 51689818066SLee Jones */ 51789818066SLee Jones #define MX25_STATUS_QE (0x1 << 6) 51889818066SLee Jones 51989818066SLee Jones static int stfsm_mx25_en_32bit_addr_seq(struct stfsm_seq *seq) 52089818066SLee Jones { 52189818066SLee Jones seq->seq_opc[0] = (SEQ_OPC_PADS_1 | 52289818066SLee Jones SEQ_OPC_CYCLES(8) | 5236c8e1b33SBrian Norris SEQ_OPC_OPCODE(SPINOR_OP_EN4B) | 52489818066SLee Jones SEQ_OPC_CSDEASSERT); 52589818066SLee Jones 52689818066SLee Jones seq->seq[0] = STFSM_INST_CMD1; 52789818066SLee Jones seq->seq[1] = STFSM_INST_WAIT; 52889818066SLee Jones seq->seq[2] = STFSM_INST_STOP; 52989818066SLee Jones 53089818066SLee Jones seq->seq_cfg = (SEQ_CFG_PADS_1 | 53189818066SLee Jones SEQ_CFG_ERASE | 53289818066SLee Jones SEQ_CFG_READNOTWRITE | 53389818066SLee Jones SEQ_CFG_CSDEASSERT | 53489818066SLee Jones SEQ_CFG_STARTSEQ); 53589818066SLee Jones 53689818066SLee Jones return 0; 53789818066SLee Jones } 53889818066SLee Jones 5395343a123SLee Jones /* 5405343a123SLee Jones * [S25FLxxx] Configuration 5415343a123SLee Jones */ 5425343a123SLee Jones #define STFSM_S25FL_CONFIG_QE (0x1 << 1) 5435343a123SLee Jones 5445343a123SLee Jones /* 5455343a123SLee Jones * S25FLxxxS devices provide three ways of supporting 32-bit addressing: Bank 5465343a123SLee Jones * Register, Extended Address Modes, and a 32-bit address command set. The 5475343a123SLee Jones * 32-bit address command set is used here, since it avoids any problems with 5485343a123SLee Jones * entering a state that is incompatible with the SPIBoot Controller. 5495343a123SLee Jones */ 5505343a123SLee Jones static struct seq_rw_config stfsm_s25fl_read4_configs[] = { 551902cc69aSCyrille Pitchen {FLASH_FLAG_READ_1_4_4, SPINOR_OP_READ_1_4_4_4B, 0, 4, 4, 0x00, 2, 4}, 552902cc69aSCyrille Pitchen {FLASH_FLAG_READ_1_1_4, SPINOR_OP_READ_1_1_4_4B, 0, 1, 4, 0x00, 0, 8}, 553902cc69aSCyrille Pitchen {FLASH_FLAG_READ_1_2_2, SPINOR_OP_READ_1_2_2_4B, 0, 2, 2, 0x00, 4, 0}, 554902cc69aSCyrille Pitchen {FLASH_FLAG_READ_1_1_2, SPINOR_OP_READ_1_1_2_4B, 0, 1, 2, 0x00, 0, 8}, 555902cc69aSCyrille Pitchen {FLASH_FLAG_READ_FAST, SPINOR_OP_READ_FAST_4B, 0, 1, 1, 0x00, 0, 8}, 556902cc69aSCyrille Pitchen {FLASH_FLAG_READ_WRITE, SPINOR_OP_READ_4B, 0, 1, 1, 0x00, 0, 0}, 5575343a123SLee Jones {0x00, 0, 0, 0, 0, 0x00, 0, 0}, 5585343a123SLee Jones }; 5595343a123SLee Jones 5605343a123SLee Jones static struct seq_rw_config stfsm_s25fl_write4_configs[] = { 5615343a123SLee Jones {FLASH_FLAG_WRITE_1_1_4, S25FL_CMD_WRITE4_1_1_4, 1, 1, 4, 0x00, 0, 0}, 5625343a123SLee Jones {FLASH_FLAG_READ_WRITE, S25FL_CMD_WRITE4, 1, 1, 1, 0x00, 0, 0}, 5635343a123SLee Jones {0x00, 0, 0, 0, 0, 0x00, 0, 0}, 5645343a123SLee Jones }; 5655343a123SLee Jones 566cd7cac9eSLee Jones /* 567cd7cac9eSLee Jones * [W25Qxxx] Configuration 568cd7cac9eSLee Jones */ 5695d0bddabSAngus Clark #define W25Q_STATUS_QE (0x1 << 1) 570cd7cac9eSLee Jones 5711bd512b5SLee Jones static struct stfsm_seq stfsm_seq_read_jedec = { 5721bd512b5SLee Jones .data_size = TRANSFER_SIZE(8), 5731bd512b5SLee Jones .seq_opc[0] = (SEQ_OPC_PADS_1 | 5741bd512b5SLee Jones SEQ_OPC_CYCLES(8) | 57592d3af9aSBrian Norris SEQ_OPC_OPCODE(SPINOR_OP_RDID)), 5761bd512b5SLee Jones .seq = { 5771bd512b5SLee Jones STFSM_INST_CMD1, 5781bd512b5SLee Jones STFSM_INST_DATA_READ, 5791bd512b5SLee Jones STFSM_INST_STOP, 5801bd512b5SLee Jones }, 5811bd512b5SLee Jones .seq_cfg = (SEQ_CFG_PADS_1 | 5821bd512b5SLee Jones SEQ_CFG_READNOTWRITE | 5831bd512b5SLee Jones SEQ_CFG_CSDEASSERT | 5841bd512b5SLee Jones SEQ_CFG_STARTSEQ), 5851bd512b5SLee Jones }; 5861bd512b5SLee Jones 587176b4377SLee Jones static struct stfsm_seq stfsm_seq_read_status_fifo = { 588176b4377SLee Jones .data_size = TRANSFER_SIZE(4), 589176b4377SLee Jones .seq_opc[0] = (SEQ_OPC_PADS_1 | 590176b4377SLee Jones SEQ_OPC_CYCLES(8) | 59192d3af9aSBrian Norris SEQ_OPC_OPCODE(SPINOR_OP_RDSR)), 592176b4377SLee Jones .seq = { 593176b4377SLee Jones STFSM_INST_CMD1, 594176b4377SLee Jones STFSM_INST_DATA_READ, 595176b4377SLee Jones STFSM_INST_STOP, 596176b4377SLee Jones }, 597176b4377SLee Jones .seq_cfg = (SEQ_CFG_PADS_1 | 598176b4377SLee Jones SEQ_CFG_READNOTWRITE | 599176b4377SLee Jones SEQ_CFG_CSDEASSERT | 600176b4377SLee Jones SEQ_CFG_STARTSEQ), 601176b4377SLee Jones }; 602176b4377SLee Jones 603fa5ba3afSLee Jones static struct stfsm_seq stfsm_seq_erase_sector = { 604fa5ba3afSLee Jones /* 'addr_cfg' configured during initialisation */ 605fa5ba3afSLee Jones .seq_opc = { 606fa5ba3afSLee Jones (SEQ_OPC_PADS_1 | SEQ_OPC_CYCLES(8) | 60792d3af9aSBrian Norris SEQ_OPC_OPCODE(SPINOR_OP_WREN) | SEQ_OPC_CSDEASSERT), 608fa5ba3afSLee Jones 609fa5ba3afSLee Jones (SEQ_OPC_PADS_1 | SEQ_OPC_CYCLES(8) | 61092d3af9aSBrian Norris SEQ_OPC_OPCODE(SPINOR_OP_SE)), 611fa5ba3afSLee Jones }, 612fa5ba3afSLee Jones .seq = { 613fa5ba3afSLee Jones STFSM_INST_CMD1, 614fa5ba3afSLee Jones STFSM_INST_CMD2, 615fa5ba3afSLee Jones STFSM_INST_ADD1, 616fa5ba3afSLee Jones STFSM_INST_ADD2, 617fa5ba3afSLee Jones STFSM_INST_STOP, 618fa5ba3afSLee Jones }, 619fa5ba3afSLee Jones .seq_cfg = (SEQ_CFG_PADS_1 | 620fa5ba3afSLee Jones SEQ_CFG_READNOTWRITE | 621fa5ba3afSLee Jones SEQ_CFG_CSDEASSERT | 622fa5ba3afSLee Jones SEQ_CFG_STARTSEQ), 623fa5ba3afSLee Jones }; 624fa5ba3afSLee Jones 6254a341fe7SLee Jones static struct stfsm_seq stfsm_seq_erase_chip = { 6264a341fe7SLee Jones .seq_opc = { 6274a341fe7SLee Jones (SEQ_OPC_PADS_1 | SEQ_OPC_CYCLES(8) | 62892d3af9aSBrian Norris SEQ_OPC_OPCODE(SPINOR_OP_WREN) | SEQ_OPC_CSDEASSERT), 6294a341fe7SLee Jones 6304a341fe7SLee Jones (SEQ_OPC_PADS_1 | SEQ_OPC_CYCLES(8) | 6316c8e1b33SBrian Norris SEQ_OPC_OPCODE(SPINOR_OP_CHIP_ERASE) | SEQ_OPC_CSDEASSERT), 6324a341fe7SLee Jones }, 6334a341fe7SLee Jones .seq = { 6344a341fe7SLee Jones STFSM_INST_CMD1, 6354a341fe7SLee Jones STFSM_INST_CMD2, 6364a341fe7SLee Jones STFSM_INST_WAIT, 6374a341fe7SLee Jones STFSM_INST_STOP, 6384a341fe7SLee Jones }, 6394a341fe7SLee Jones .seq_cfg = (SEQ_CFG_PADS_1 | 6404a341fe7SLee Jones SEQ_CFG_ERASE | 6414a341fe7SLee Jones SEQ_CFG_READNOTWRITE | 6424a341fe7SLee Jones SEQ_CFG_CSDEASSERT | 6434a341fe7SLee Jones SEQ_CFG_STARTSEQ), 6444a341fe7SLee Jones }; 6454a341fe7SLee Jones 646150571b7SLee Jones static struct stfsm_seq stfsm_seq_write_status = { 647150571b7SLee Jones .seq_opc[0] = (SEQ_OPC_PADS_1 | SEQ_OPC_CYCLES(8) | 64892d3af9aSBrian Norris SEQ_OPC_OPCODE(SPINOR_OP_WREN) | SEQ_OPC_CSDEASSERT), 649150571b7SLee Jones .seq_opc[1] = (SEQ_OPC_PADS_1 | SEQ_OPC_CYCLES(8) | 65092d3af9aSBrian Norris SEQ_OPC_OPCODE(SPINOR_OP_WRSR)), 651150571b7SLee Jones .seq = { 652150571b7SLee Jones STFSM_INST_CMD1, 653150571b7SLee Jones STFSM_INST_CMD2, 654150571b7SLee Jones STFSM_INST_STA_WR1, 655150571b7SLee Jones STFSM_INST_STOP, 656150571b7SLee Jones }, 657150571b7SLee Jones .seq_cfg = (SEQ_CFG_PADS_1 | 658150571b7SLee Jones SEQ_CFG_READNOTWRITE | 659150571b7SLee Jones SEQ_CFG_CSDEASSERT | 660150571b7SLee Jones SEQ_CFG_STARTSEQ), 661150571b7SLee Jones }; 662150571b7SLee Jones 6635ecd3ea1SLee Jones /* Dummy sequence to read one byte of data from flash into the FIFO */ 6645ecd3ea1SLee Jones static const struct stfsm_seq stfsm_seq_load_fifo_byte = { 6655ecd3ea1SLee Jones .data_size = TRANSFER_SIZE(1), 6665ecd3ea1SLee Jones .seq_opc[0] = (SEQ_OPC_PADS_1 | 6675ecd3ea1SLee Jones SEQ_OPC_CYCLES(8) | 6685ecd3ea1SLee Jones SEQ_OPC_OPCODE(SPINOR_OP_RDID)), 6695ecd3ea1SLee Jones .seq = { 6705ecd3ea1SLee Jones STFSM_INST_CMD1, 6715ecd3ea1SLee Jones STFSM_INST_DATA_READ, 6725ecd3ea1SLee Jones STFSM_INST_STOP, 6735ecd3ea1SLee Jones }, 6745ecd3ea1SLee Jones .seq_cfg = (SEQ_CFG_PADS_1 | 6755ecd3ea1SLee Jones SEQ_CFG_READNOTWRITE | 6765ecd3ea1SLee Jones SEQ_CFG_CSDEASSERT | 6775ecd3ea1SLee Jones SEQ_CFG_STARTSEQ), 6785ecd3ea1SLee Jones }; 6795ecd3ea1SLee Jones 6806bd29600SLee Jones static int stfsm_n25q_en_32bit_addr_seq(struct stfsm_seq *seq) 6816bd29600SLee Jones { 6826bd29600SLee Jones seq->seq_opc[0] = (SEQ_OPC_PADS_1 | SEQ_OPC_CYCLES(8) | 6836c8e1b33SBrian Norris SEQ_OPC_OPCODE(SPINOR_OP_EN4B)); 6846bd29600SLee Jones seq->seq_opc[1] = (SEQ_OPC_PADS_1 | SEQ_OPC_CYCLES(8) | 68592d3af9aSBrian Norris SEQ_OPC_OPCODE(SPINOR_OP_WREN) | 6866bd29600SLee Jones SEQ_OPC_CSDEASSERT); 6876bd29600SLee Jones 6886bd29600SLee Jones seq->seq[0] = STFSM_INST_CMD2; 6896bd29600SLee Jones seq->seq[1] = STFSM_INST_CMD1; 6906bd29600SLee Jones seq->seq[2] = STFSM_INST_WAIT; 6916bd29600SLee Jones seq->seq[3] = STFSM_INST_STOP; 6926bd29600SLee Jones 6936bd29600SLee Jones seq->seq_cfg = (SEQ_CFG_PADS_1 | 6946bd29600SLee Jones SEQ_CFG_ERASE | 6956bd29600SLee Jones SEQ_CFG_READNOTWRITE | 6966bd29600SLee Jones SEQ_CFG_CSDEASSERT | 6976bd29600SLee Jones SEQ_CFG_STARTSEQ); 6986bd29600SLee Jones 6996bd29600SLee Jones return 0; 7006bd29600SLee Jones } 7016bd29600SLee Jones 7023c8b85b3SLee Jones static inline int stfsm_is_idle(struct stfsm *fsm) 7033c8b85b3SLee Jones { 7043c8b85b3SLee Jones return readl(fsm->base + SPI_FAST_SEQ_STA) & 0x10; 7053c8b85b3SLee Jones } 7063c8b85b3SLee Jones 70786f309fdSLee Jones static inline uint32_t stfsm_fifo_available(struct stfsm *fsm) 70886f309fdSLee Jones { 70986f309fdSLee Jones return (readl(fsm->base + SPI_FAST_SEQ_STA) >> 5) & 0x7f; 71086f309fdSLee Jones } 71186f309fdSLee Jones 7123c8b85b3SLee Jones static inline void stfsm_load_seq(struct stfsm *fsm, 7133c8b85b3SLee Jones const struct stfsm_seq *seq) 7143c8b85b3SLee Jones { 7153c8b85b3SLee Jones void __iomem *dst = fsm->base + SPI_FAST_SEQ_TRANSFER_SIZE; 7163c8b85b3SLee Jones const uint32_t *src = (const uint32_t *)seq; 7173c8b85b3SLee Jones int words = sizeof(*seq) / sizeof(*src); 7183c8b85b3SLee Jones 7193c8b85b3SLee Jones BUG_ON(!stfsm_is_idle(fsm)); 7203c8b85b3SLee Jones 7213c8b85b3SLee Jones while (words--) { 7223c8b85b3SLee Jones writel(*src, dst); 7233c8b85b3SLee Jones src++; 7243c8b85b3SLee Jones dst += 4; 7253c8b85b3SLee Jones } 7263c8b85b3SLee Jones } 7273c8b85b3SLee Jones 7283c8b85b3SLee Jones static void stfsm_wait_seq(struct stfsm *fsm) 7293c8b85b3SLee Jones { 7303c8b85b3SLee Jones unsigned long deadline; 7313c8b85b3SLee Jones int timeout = 0; 7323c8b85b3SLee Jones 7333c8b85b3SLee Jones deadline = jiffies + msecs_to_jiffies(STFSM_MAX_WAIT_SEQ_MS); 7343c8b85b3SLee Jones 7353c8b85b3SLee Jones while (!timeout) { 7363c8b85b3SLee Jones if (time_after_eq(jiffies, deadline)) 7373c8b85b3SLee Jones timeout = 1; 7383c8b85b3SLee Jones 7393c8b85b3SLee Jones if (stfsm_is_idle(fsm)) 7403c8b85b3SLee Jones return; 7413c8b85b3SLee Jones 7423c8b85b3SLee Jones cond_resched(); 7433c8b85b3SLee Jones } 7443c8b85b3SLee Jones 7453c8b85b3SLee Jones dev_err(fsm->dev, "timeout on sequence completion\n"); 7463c8b85b3SLee Jones } 7473c8b85b3SLee Jones 7483f9d720aSLee Jones static void stfsm_read_fifo(struct stfsm *fsm, uint32_t *buf, uint32_t size) 749030e82dcSLee Jones { 750030e82dcSLee Jones uint32_t remaining = size >> 2; 751030e82dcSLee Jones uint32_t avail; 752030e82dcSLee Jones uint32_t words; 753030e82dcSLee Jones 754030e82dcSLee Jones dev_dbg(fsm->dev, "Reading %d bytes from FIFO\n", size); 755030e82dcSLee Jones 75638e2eee9SBrian Norris BUG_ON((((uintptr_t)buf) & 0x3) || (size & 0x3)); 757030e82dcSLee Jones 758030e82dcSLee Jones while (remaining) { 759030e82dcSLee Jones for (;;) { 760030e82dcSLee Jones avail = stfsm_fifo_available(fsm); 761030e82dcSLee Jones if (avail) 762030e82dcSLee Jones break; 763030e82dcSLee Jones udelay(1); 764030e82dcSLee Jones } 765030e82dcSLee Jones words = min(avail, remaining); 766030e82dcSLee Jones remaining -= words; 767030e82dcSLee Jones 768030e82dcSLee Jones readsl(fsm->base + SPI_FAST_SEQ_DATA_REG, buf, words); 769030e82dcSLee Jones buf += words; 770030e82dcSLee Jones } 771030e82dcSLee Jones } 772030e82dcSLee Jones 7735ecd3ea1SLee Jones /* 7745ecd3ea1SLee Jones * Clear the data FIFO 7755ecd3ea1SLee Jones * 7765ecd3ea1SLee Jones * Typically, this is only required during driver initialisation, where no 7775ecd3ea1SLee Jones * assumptions can be made regarding the state of the FIFO. 7785ecd3ea1SLee Jones * 7795ecd3ea1SLee Jones * The process of clearing the FIFO is complicated by fact that while it is 7805ecd3ea1SLee Jones * possible for the FIFO to contain an arbitrary number of bytes [1], the 7815ecd3ea1SLee Jones * SPI_FAST_SEQ_STA register only reports the number of complete 32-bit words 7825ecd3ea1SLee Jones * present. Furthermore, data can only be drained from the FIFO by reading 7835ecd3ea1SLee Jones * complete 32-bit words. 7845ecd3ea1SLee Jones * 7855ecd3ea1SLee Jones * With this in mind, a two stage process is used to the clear the FIFO: 7865ecd3ea1SLee Jones * 7875ecd3ea1SLee Jones * 1. Read any complete 32-bit words from the FIFO, as reported by the 7885ecd3ea1SLee Jones * SPI_FAST_SEQ_STA register. 7895ecd3ea1SLee Jones * 7905ecd3ea1SLee Jones * 2. Mop up any remaining bytes. At this point, it is not known if there 7915ecd3ea1SLee Jones * are 0, 1, 2, or 3 bytes in the FIFO. To handle all cases, a dummy FSM 7925ecd3ea1SLee Jones * sequence is used to load one byte at a time, until a complete 32-bit 7935ecd3ea1SLee Jones * word is formed; at most, 4 bytes will need to be loaded. 7945ecd3ea1SLee Jones * 7955ecd3ea1SLee Jones * [1] It is theoretically possible for the FIFO to contain an arbitrary number 7965ecd3ea1SLee Jones * of bits. However, since there are no known use-cases that leave 7975ecd3ea1SLee Jones * incomplete bytes in the FIFO, only words and bytes are considered here. 7985ecd3ea1SLee Jones */ 7995ecd3ea1SLee Jones static void stfsm_clear_fifo(struct stfsm *fsm) 8005ecd3ea1SLee Jones { 8015ecd3ea1SLee Jones const struct stfsm_seq *seq = &stfsm_seq_load_fifo_byte; 8025ecd3ea1SLee Jones uint32_t words, i; 8035ecd3ea1SLee Jones 8045ecd3ea1SLee Jones /* 1. Clear any 32-bit words */ 8055ecd3ea1SLee Jones words = stfsm_fifo_available(fsm); 8065ecd3ea1SLee Jones if (words) { 8075ecd3ea1SLee Jones for (i = 0; i < words; i++) 8085ecd3ea1SLee Jones readl(fsm->base + SPI_FAST_SEQ_DATA_REG); 8095ecd3ea1SLee Jones dev_dbg(fsm->dev, "cleared %d words from FIFO\n", words); 8105ecd3ea1SLee Jones } 8115ecd3ea1SLee Jones 8125ecd3ea1SLee Jones /* 8135ecd3ea1SLee Jones * 2. Clear any remaining bytes 8145ecd3ea1SLee Jones * - Load the FIFO, one byte at a time, until a complete 32-bit word 8155ecd3ea1SLee Jones * is available. 8165ecd3ea1SLee Jones */ 8175ecd3ea1SLee Jones for (i = 0, words = 0; i < 4 && !words; i++) { 8185ecd3ea1SLee Jones stfsm_load_seq(fsm, seq); 8195ecd3ea1SLee Jones stfsm_wait_seq(fsm); 8205ecd3ea1SLee Jones words = stfsm_fifo_available(fsm); 8215ecd3ea1SLee Jones } 8225ecd3ea1SLee Jones 8235ecd3ea1SLee Jones /* - A single word must be available now */ 8245ecd3ea1SLee Jones if (words != 1) { 8255ecd3ea1SLee Jones dev_err(fsm->dev, "failed to clear bytes from the data FIFO\n"); 8265ecd3ea1SLee Jones return; 8275ecd3ea1SLee Jones } 8285ecd3ea1SLee Jones 8295ecd3ea1SLee Jones /* - Read the 32-bit word */ 8305ecd3ea1SLee Jones readl(fsm->base + SPI_FAST_SEQ_DATA_REG); 8315ecd3ea1SLee Jones 8325ecd3ea1SLee Jones dev_dbg(fsm->dev, "cleared %d byte(s) from the data FIFO\n", 4 - i); 8335ecd3ea1SLee Jones } 8345ecd3ea1SLee Jones 8353f9d720aSLee Jones static int stfsm_write_fifo(struct stfsm *fsm, const uint32_t *buf, 8363f9d720aSLee Jones uint32_t size) 83730ca64f9SLee Jones { 83830ca64f9SLee Jones uint32_t words = size >> 2; 83930ca64f9SLee Jones 84030ca64f9SLee Jones dev_dbg(fsm->dev, "writing %d bytes to FIFO\n", size); 84130ca64f9SLee Jones 84238e2eee9SBrian Norris BUG_ON((((uintptr_t)buf) & 0x3) || (size & 0x3)); 84330ca64f9SLee Jones 84430ca64f9SLee Jones writesl(fsm->base + SPI_FAST_SEQ_DATA_REG, buf, words); 84530ca64f9SLee Jones 84630ca64f9SLee Jones return size; 84730ca64f9SLee Jones } 84830ca64f9SLee Jones 8490de08e43SLee Jones static int stfsm_enter_32bit_addr(struct stfsm *fsm, int enter) 8500de08e43SLee Jones { 851e6b1bb4eSLee Jones struct stfsm_seq *seq = &fsm->stfsm_seq_en_32bit_addr; 8526c8e1b33SBrian Norris uint32_t cmd = enter ? SPINOR_OP_EN4B : SPINOR_OP_EX4B; 8530de08e43SLee Jones 8540de08e43SLee Jones seq->seq_opc[0] = (SEQ_OPC_PADS_1 | 8550de08e43SLee Jones SEQ_OPC_CYCLES(8) | 8560de08e43SLee Jones SEQ_OPC_OPCODE(cmd) | 8570de08e43SLee Jones SEQ_OPC_CSDEASSERT); 8580de08e43SLee Jones 8590de08e43SLee Jones stfsm_load_seq(fsm, seq); 8600de08e43SLee Jones 8610de08e43SLee Jones stfsm_wait_seq(fsm); 8620de08e43SLee Jones 8630de08e43SLee Jones return 0; 8640de08e43SLee Jones } 8650de08e43SLee Jones 866176b4377SLee Jones static uint8_t stfsm_wait_busy(struct stfsm *fsm) 867176b4377SLee Jones { 868176b4377SLee Jones struct stfsm_seq *seq = &stfsm_seq_read_status_fifo; 869176b4377SLee Jones unsigned long deadline; 870176b4377SLee Jones uint32_t status; 871176b4377SLee Jones int timeout = 0; 872176b4377SLee Jones 873176b4377SLee Jones /* Use RDRS1 */ 874176b4377SLee Jones seq->seq_opc[0] = (SEQ_OPC_PADS_1 | 875176b4377SLee Jones SEQ_OPC_CYCLES(8) | 87692d3af9aSBrian Norris SEQ_OPC_OPCODE(SPINOR_OP_RDSR)); 877176b4377SLee Jones 878176b4377SLee Jones /* Load read_status sequence */ 879176b4377SLee Jones stfsm_load_seq(fsm, seq); 880176b4377SLee Jones 881176b4377SLee Jones /* 882176b4377SLee Jones * Repeat until busy bit is deasserted, or timeout, or error (S25FLxxxS) 883176b4377SLee Jones */ 884176b4377SLee Jones deadline = jiffies + FLASH_MAX_BUSY_WAIT; 885176b4377SLee Jones while (!timeout) { 886176b4377SLee Jones if (time_after_eq(jiffies, deadline)) 887176b4377SLee Jones timeout = 1; 888176b4377SLee Jones 889176b4377SLee Jones stfsm_wait_seq(fsm); 890176b4377SLee Jones 891176b4377SLee Jones stfsm_read_fifo(fsm, &status, 4); 892176b4377SLee Jones 893176b4377SLee Jones if ((status & FLASH_STATUS_BUSY) == 0) 894176b4377SLee Jones return 0; 895176b4377SLee Jones 896176b4377SLee Jones if ((fsm->configuration & CFG_S25FL_CHECK_ERROR_FLAGS) && 897176b4377SLee Jones ((status & S25FL_STATUS_P_ERR) || 898176b4377SLee Jones (status & S25FL_STATUS_E_ERR))) 899176b4377SLee Jones return (uint8_t)(status & 0xff); 900176b4377SLee Jones 901176b4377SLee Jones if (!timeout) 902176b4377SLee Jones /* Restart */ 903176b4377SLee Jones writel(seq->seq_cfg, fsm->base + SPI_FAST_SEQ_CFG); 904ea7864bfSLee Jones 905ea7864bfSLee Jones cond_resched(); 906176b4377SLee Jones } 907176b4377SLee Jones 908176b4377SLee Jones dev_err(fsm->dev, "timeout on wait_busy\n"); 909176b4377SLee Jones 910176b4377SLee Jones return FLASH_STATUS_TIMEOUT; 911176b4377SLee Jones } 912176b4377SLee Jones 913ac94dbcbSLee Jones static int stfsm_read_status(struct stfsm *fsm, uint8_t cmd, 9145d0bddabSAngus Clark uint8_t *data, int bytes) 915ac94dbcbSLee Jones { 916ac94dbcbSLee Jones struct stfsm_seq *seq = &stfsm_seq_read_status_fifo; 917ac94dbcbSLee Jones uint32_t tmp; 9185d0bddabSAngus Clark uint8_t *t = (uint8_t *)&tmp; 9195d0bddabSAngus Clark int i; 920ac94dbcbSLee Jones 9215d0bddabSAngus Clark dev_dbg(fsm->dev, "read 'status' register [0x%02x], %d byte(s)\n", 9225d0bddabSAngus Clark cmd, bytes); 923ac94dbcbSLee Jones 9245d0bddabSAngus Clark BUG_ON(bytes != 1 && bytes != 2); 9255d0bddabSAngus Clark 9265d0bddabSAngus Clark seq->seq_opc[0] = (SEQ_OPC_PADS_1 | SEQ_OPC_CYCLES(8) | 927*106a3ec3SZheng Yongjun SEQ_OPC_OPCODE(cmd)); 928ac94dbcbSLee Jones 929ac94dbcbSLee Jones stfsm_load_seq(fsm, seq); 930ac94dbcbSLee Jones 931ac94dbcbSLee Jones stfsm_read_fifo(fsm, &tmp, 4); 932ac94dbcbSLee Jones 9335d0bddabSAngus Clark for (i = 0; i < bytes; i++) 9345d0bddabSAngus Clark data[i] = t[i]; 935ac94dbcbSLee Jones 936ac94dbcbSLee Jones stfsm_wait_seq(fsm); 937ac94dbcbSLee Jones 938ac94dbcbSLee Jones return 0; 939ac94dbcbSLee Jones } 940ac94dbcbSLee Jones 9415d0bddabSAngus Clark static int stfsm_write_status(struct stfsm *fsm, uint8_t cmd, 9425d0bddabSAngus Clark uint16_t data, int bytes, int wait_busy) 943150571b7SLee Jones { 944150571b7SLee Jones struct stfsm_seq *seq = &stfsm_seq_write_status; 945150571b7SLee Jones 9465d0bddabSAngus Clark dev_dbg(fsm->dev, 9475d0bddabSAngus Clark "write 'status' register [0x%02x], %d byte(s), 0x%04x\n" 9485d0bddabSAngus Clark " %s wait-busy\n", cmd, bytes, data, wait_busy ? "with" : "no"); 949150571b7SLee Jones 9505d0bddabSAngus Clark BUG_ON(bytes != 1 && bytes != 2); 9515d0bddabSAngus Clark 9525d0bddabSAngus Clark seq->seq_opc[1] = (SEQ_OPC_PADS_1 | SEQ_OPC_CYCLES(8) | 9535d0bddabSAngus Clark SEQ_OPC_OPCODE(cmd)); 9545d0bddabSAngus Clark 9555d0bddabSAngus Clark seq->status = (uint32_t)data | STA_PADS_1 | STA_CSDEASSERT; 9565d0bddabSAngus Clark seq->seq[2] = (bytes == 1) ? STFSM_INST_STA_WR1 : STFSM_INST_STA_WR1_2; 957150571b7SLee Jones 958150571b7SLee Jones stfsm_load_seq(fsm, seq); 959150571b7SLee Jones 960150571b7SLee Jones stfsm_wait_seq(fsm); 961150571b7SLee Jones 9625d0bddabSAngus Clark if (wait_busy) 9635d0bddabSAngus Clark stfsm_wait_busy(fsm); 964249516c9SLee Jones 965249516c9SLee Jones return 0; 966249516c9SLee Jones } 967249516c9SLee Jones 9680ea7d706SLee Jones /* 9690ea7d706SLee Jones * SoC reset on 'boot-from-spi' systems 9700ea7d706SLee Jones * 9710ea7d706SLee Jones * Certain modes of operation cause the Flash device to enter a particular state 9720ea7d706SLee Jones * for a period of time (e.g. 'Erase Sector', 'Quad Enable', and 'Enter 32-bit 9730ea7d706SLee Jones * Addr' commands). On boot-from-spi systems, it is important to consider what 9740ea7d706SLee Jones * happens if a warm reset occurs during this period. The SPIBoot controller 9750ea7d706SLee Jones * assumes that Flash device is in its default reset state, 24-bit address mode, 9760ea7d706SLee Jones * and ready to accept commands. This can be achieved using some form of 9770ea7d706SLee Jones * on-board logic/controller to force a device POR in response to a SoC-level 9780ea7d706SLee Jones * reset or by making use of the device reset signal if available (limited 9790ea7d706SLee Jones * number of devices only). 9800ea7d706SLee Jones * 9810ea7d706SLee Jones * Failure to take such precautions can cause problems following a warm reset. 9820ea7d706SLee Jones * For some operations (e.g. ERASE), there is little that can be done. For 9830ea7d706SLee Jones * other modes of operation (e.g. 32-bit addressing), options are often 9840ea7d706SLee Jones * available that can help minimise the window in which a reset could cause a 9850ea7d706SLee Jones * problem. 9860ea7d706SLee Jones * 9870ea7d706SLee Jones */ 9880ea7d706SLee Jones static bool stfsm_can_handle_soc_reset(struct stfsm *fsm) 9890ea7d706SLee Jones { 9900ea7d706SLee Jones /* Reset signal is available on the board and supported by the device */ 9910ea7d706SLee Jones if (fsm->reset_signal && fsm->info->flags & FLASH_FLAG_RESET) 9920ea7d706SLee Jones return true; 9930ea7d706SLee Jones 9940ea7d706SLee Jones /* Board-level logic forces a power-on-reset */ 9950ea7d706SLee Jones if (fsm->reset_por) 9960ea7d706SLee Jones return true; 9970ea7d706SLee Jones 9980ea7d706SLee Jones /* Reset is not properly handled and may result in failure to reboot */ 9990ea7d706SLee Jones return false; 10000ea7d706SLee Jones } 10010ea7d706SLee Jones 1002fa5ba3afSLee Jones /* Configure 'addr_cfg' according to addressing mode */ 1003fa5ba3afSLee Jones static void stfsm_prepare_erasesec_seq(struct stfsm *fsm, 1004fa5ba3afSLee Jones struct stfsm_seq *seq) 1005fa5ba3afSLee Jones { 1006fa5ba3afSLee Jones int addr1_cycles = fsm->info->flags & FLASH_FLAG_32BIT_ADDR ? 16 : 8; 1007fa5ba3afSLee Jones 1008fa5ba3afSLee Jones seq->addr_cfg = (ADR_CFG_CYCLES_ADD1(addr1_cycles) | 1009fa5ba3afSLee Jones ADR_CFG_PADS_1_ADD1 | 1010fa5ba3afSLee Jones ADR_CFG_CYCLES_ADD2(16) | 1011fa5ba3afSLee Jones ADR_CFG_PADS_1_ADD2 | 1012fa5ba3afSLee Jones ADR_CFG_CSDEASSERT_ADD2); 1013fa5ba3afSLee Jones } 1014fa5ba3afSLee Jones 101508981274SLee Jones /* Search for preferred configuration based on available flags */ 101608981274SLee Jones static struct seq_rw_config * 101708981274SLee Jones stfsm_search_seq_rw_configs(struct stfsm *fsm, 101808981274SLee Jones struct seq_rw_config cfgs[]) 101908981274SLee Jones { 102008981274SLee Jones struct seq_rw_config *config; 102108981274SLee Jones int flags = fsm->info->flags; 102208981274SLee Jones 102308981274SLee Jones for (config = cfgs; config->cmd != 0; config++) 102408981274SLee Jones if ((config->flags & flags) == config->flags) 102508981274SLee Jones return config; 102608981274SLee Jones 102708981274SLee Jones return NULL; 102808981274SLee Jones } 102908981274SLee Jones 103097ccf2d2SLee Jones /* Prepare a READ/WRITE sequence according to configuration parameters */ 103197ccf2d2SLee Jones static void stfsm_prepare_rw_seq(struct stfsm *fsm, 103297ccf2d2SLee Jones struct stfsm_seq *seq, 103397ccf2d2SLee Jones struct seq_rw_config *cfg) 103497ccf2d2SLee Jones { 103597ccf2d2SLee Jones int addr1_cycles, addr2_cycles; 103697ccf2d2SLee Jones int i = 0; 103797ccf2d2SLee Jones 103897ccf2d2SLee Jones memset(seq, 0, sizeof(*seq)); 103997ccf2d2SLee Jones 104097ccf2d2SLee Jones /* Add READ/WRITE OPC */ 104197ccf2d2SLee Jones seq->seq_opc[i++] = (SEQ_OPC_PADS_1 | 104297ccf2d2SLee Jones SEQ_OPC_CYCLES(8) | 104397ccf2d2SLee Jones SEQ_OPC_OPCODE(cfg->cmd)); 104497ccf2d2SLee Jones 104597ccf2d2SLee Jones /* Add WREN OPC for a WRITE sequence */ 104697ccf2d2SLee Jones if (cfg->write) 104797ccf2d2SLee Jones seq->seq_opc[i++] = (SEQ_OPC_PADS_1 | 104897ccf2d2SLee Jones SEQ_OPC_CYCLES(8) | 104992d3af9aSBrian Norris SEQ_OPC_OPCODE(SPINOR_OP_WREN) | 105097ccf2d2SLee Jones SEQ_OPC_CSDEASSERT); 105197ccf2d2SLee Jones 105297ccf2d2SLee Jones /* Address configuration (24 or 32-bit addresses) */ 105397ccf2d2SLee Jones addr1_cycles = (fsm->info->flags & FLASH_FLAG_32BIT_ADDR) ? 16 : 8; 105497ccf2d2SLee Jones addr1_cycles /= cfg->addr_pads; 105597ccf2d2SLee Jones addr2_cycles = 16 / cfg->addr_pads; 105697ccf2d2SLee Jones seq->addr_cfg = ((addr1_cycles & 0x3f) << 0 | /* ADD1 cycles */ 105797ccf2d2SLee Jones (cfg->addr_pads - 1) << 6 | /* ADD1 pads */ 105897ccf2d2SLee Jones (addr2_cycles & 0x3f) << 16 | /* ADD2 cycles */ 105997ccf2d2SLee Jones ((cfg->addr_pads - 1) << 22)); /* ADD2 pads */ 106097ccf2d2SLee Jones 106197ccf2d2SLee Jones /* Data/Sequence configuration */ 106297ccf2d2SLee Jones seq->seq_cfg = ((cfg->data_pads - 1) << 16 | 106397ccf2d2SLee Jones SEQ_CFG_STARTSEQ | 106497ccf2d2SLee Jones SEQ_CFG_CSDEASSERT); 106597ccf2d2SLee Jones if (!cfg->write) 106697ccf2d2SLee Jones seq->seq_cfg |= SEQ_CFG_READNOTWRITE; 106797ccf2d2SLee Jones 106897ccf2d2SLee Jones /* Mode configuration (no. of pads taken from addr cfg) */ 106997ccf2d2SLee Jones seq->mode = ((cfg->mode_data & 0xff) << 0 | /* data */ 107097ccf2d2SLee Jones (cfg->mode_cycles & 0x3f) << 16 | /* cycles */ 107197ccf2d2SLee Jones (cfg->addr_pads - 1) << 22); /* pads */ 107297ccf2d2SLee Jones 107397ccf2d2SLee Jones /* Dummy configuration (no. of pads taken from addr cfg) */ 107497ccf2d2SLee Jones seq->dummy = ((cfg->dummy_cycles & 0x3f) << 16 | /* cycles */ 107597ccf2d2SLee Jones (cfg->addr_pads - 1) << 22); /* pads */ 107697ccf2d2SLee Jones 107797ccf2d2SLee Jones 107897ccf2d2SLee Jones /* Instruction sequence */ 107997ccf2d2SLee Jones i = 0; 108097ccf2d2SLee Jones if (cfg->write) 108197ccf2d2SLee Jones seq->seq[i++] = STFSM_INST_CMD2; 108297ccf2d2SLee Jones 108397ccf2d2SLee Jones seq->seq[i++] = STFSM_INST_CMD1; 108497ccf2d2SLee Jones 108597ccf2d2SLee Jones seq->seq[i++] = STFSM_INST_ADD1; 108697ccf2d2SLee Jones seq->seq[i++] = STFSM_INST_ADD2; 108797ccf2d2SLee Jones 108897ccf2d2SLee Jones if (cfg->mode_cycles) 108997ccf2d2SLee Jones seq->seq[i++] = STFSM_INST_MODE; 109097ccf2d2SLee Jones 109197ccf2d2SLee Jones if (cfg->dummy_cycles) 109297ccf2d2SLee Jones seq->seq[i++] = STFSM_INST_DUMMY; 109397ccf2d2SLee Jones 109497ccf2d2SLee Jones seq->seq[i++] = 109597ccf2d2SLee Jones cfg->write ? STFSM_INST_DATA_WRITE : STFSM_INST_DATA_READ; 109697ccf2d2SLee Jones seq->seq[i++] = STFSM_INST_STOP; 109797ccf2d2SLee Jones } 109897ccf2d2SLee Jones 109988cccb89SLee Jones static int stfsm_search_prepare_rw_seq(struct stfsm *fsm, 110088cccb89SLee Jones struct stfsm_seq *seq, 110188cccb89SLee Jones struct seq_rw_config *cfgs) 110288cccb89SLee Jones { 110388cccb89SLee Jones struct seq_rw_config *config; 110488cccb89SLee Jones 110588cccb89SLee Jones config = stfsm_search_seq_rw_configs(fsm, cfgs); 110688cccb89SLee Jones if (!config) { 110788cccb89SLee Jones dev_err(fsm->dev, "failed to find suitable config\n"); 110888cccb89SLee Jones return -EINVAL; 110988cccb89SLee Jones } 111088cccb89SLee Jones 111188cccb89SLee Jones stfsm_prepare_rw_seq(fsm, seq, config); 111288cccb89SLee Jones 111388cccb89SLee Jones return 0; 111488cccb89SLee Jones } 111588cccb89SLee Jones 11164eb3f0d8SLee Jones /* Prepare a READ/WRITE/ERASE 'default' sequences */ 11174eb3f0d8SLee Jones static int stfsm_prepare_rwe_seqs_default(struct stfsm *fsm) 11184eb3f0d8SLee Jones { 11194eb3f0d8SLee Jones uint32_t flags = fsm->info->flags; 11204eb3f0d8SLee Jones int ret; 11214eb3f0d8SLee Jones 11224eb3f0d8SLee Jones /* Configure 'READ' sequence */ 1123e6b1bb4eSLee Jones ret = stfsm_search_prepare_rw_seq(fsm, &fsm->stfsm_seq_read, 11244eb3f0d8SLee Jones default_read_configs); 11254eb3f0d8SLee Jones if (ret) { 11264eb3f0d8SLee Jones dev_err(fsm->dev, 11274eb3f0d8SLee Jones "failed to prep READ sequence with flags [0x%08x]\n", 11284eb3f0d8SLee Jones flags); 11294eb3f0d8SLee Jones return ret; 11304eb3f0d8SLee Jones } 11314eb3f0d8SLee Jones 11324eb3f0d8SLee Jones /* Configure 'WRITE' sequence */ 1133e6b1bb4eSLee Jones ret = stfsm_search_prepare_rw_seq(fsm, &fsm->stfsm_seq_write, 11344eb3f0d8SLee Jones default_write_configs); 11354eb3f0d8SLee Jones if (ret) { 11364eb3f0d8SLee Jones dev_err(fsm->dev, 11374eb3f0d8SLee Jones "failed to prep WRITE sequence with flags [0x%08x]\n", 11384eb3f0d8SLee Jones flags); 11394eb3f0d8SLee Jones return ret; 11404eb3f0d8SLee Jones } 11414eb3f0d8SLee Jones 11424eb3f0d8SLee Jones /* Configure 'ERASE_SECTOR' sequence */ 11434eb3f0d8SLee Jones stfsm_prepare_erasesec_seq(fsm, &stfsm_seq_erase_sector); 11444eb3f0d8SLee Jones 11454eb3f0d8SLee Jones return 0; 11464eb3f0d8SLee Jones } 11474eb3f0d8SLee Jones 114889818066SLee Jones static int stfsm_mx25_config(struct stfsm *fsm) 114989818066SLee Jones { 115089818066SLee Jones uint32_t flags = fsm->info->flags; 115189818066SLee Jones uint32_t data_pads; 115289818066SLee Jones uint8_t sta; 115389818066SLee Jones int ret; 115489818066SLee Jones bool soc_reset; 115589818066SLee Jones 115689818066SLee Jones /* 115789818066SLee Jones * Use default READ/WRITE sequences 115889818066SLee Jones */ 115989818066SLee Jones ret = stfsm_prepare_rwe_seqs_default(fsm); 116089818066SLee Jones if (ret) 116189818066SLee Jones return ret; 116289818066SLee Jones 116389818066SLee Jones /* 116489818066SLee Jones * Configure 32-bit Address Support 116589818066SLee Jones */ 116689818066SLee Jones if (flags & FLASH_FLAG_32BIT_ADDR) { 116789818066SLee Jones /* Configure 'enter_32bitaddr' FSM sequence */ 1168e6b1bb4eSLee Jones stfsm_mx25_en_32bit_addr_seq(&fsm->stfsm_seq_en_32bit_addr); 116989818066SLee Jones 117089818066SLee Jones soc_reset = stfsm_can_handle_soc_reset(fsm); 1171009e7e61SAngus Clark if (soc_reset || !fsm->booted_from_spi) 117289818066SLee Jones /* If we can handle SoC resets, we enable 32-bit address 117389818066SLee Jones * mode pervasively */ 117489818066SLee Jones stfsm_enter_32bit_addr(fsm, 1); 117589818066SLee Jones 1176009e7e61SAngus Clark else 117789818066SLee Jones /* Else, enable/disable 32-bit addressing before/after 117889818066SLee Jones * each operation */ 117989818066SLee Jones fsm->configuration = (CFG_READ_TOGGLE_32BIT_ADDR | 118089818066SLee Jones CFG_WRITE_TOGGLE_32BIT_ADDR | 118189818066SLee Jones CFG_ERASESEC_TOGGLE_32BIT_ADDR); 118289818066SLee Jones } 118389818066SLee Jones 11845d0bddabSAngus Clark /* Check status of 'QE' bit, update if required. */ 118592d3af9aSBrian Norris stfsm_read_status(fsm, SPINOR_OP_RDSR, &sta, 1); 1186e6b1bb4eSLee Jones data_pads = ((fsm->stfsm_seq_read.seq_cfg >> 16) & 0x3) + 1; 118789818066SLee Jones if (data_pads == 4) { 1188cc666863SAngus Clark if (!(sta & MX25_STATUS_QE)) { 1189cc666863SAngus Clark /* Set 'QE' */ 119089818066SLee Jones sta |= MX25_STATUS_QE; 1191cc666863SAngus Clark 119292d3af9aSBrian Norris stfsm_write_status(fsm, SPINOR_OP_WRSR, sta, 1, 1); 1193cc666863SAngus Clark } 1194cc666863SAngus Clark } else { 1195cc666863SAngus Clark if (sta & MX25_STATUS_QE) { 1196cc666863SAngus Clark /* Clear 'QE' */ 1197cc666863SAngus Clark sta &= ~MX25_STATUS_QE; 1198cc666863SAngus Clark 119992d3af9aSBrian Norris stfsm_write_status(fsm, SPINOR_OP_WRSR, sta, 1, 1); 1200cc666863SAngus Clark } 120189818066SLee Jones } 120289818066SLee Jones 120389818066SLee Jones return 0; 120489818066SLee Jones } 120589818066SLee Jones 1206218b870fSLee Jones static int stfsm_n25q_config(struct stfsm *fsm) 1207218b870fSLee Jones { 1208218b870fSLee Jones uint32_t flags = fsm->info->flags; 1209218b870fSLee Jones uint8_t vcr; 1210218b870fSLee Jones int ret = 0; 1211218b870fSLee Jones bool soc_reset; 1212218b870fSLee Jones 1213218b870fSLee Jones /* Configure 'READ' sequence */ 1214218b870fSLee Jones if (flags & FLASH_FLAG_32BIT_ADDR) 1215e6b1bb4eSLee Jones ret = stfsm_search_prepare_rw_seq(fsm, &fsm->stfsm_seq_read, 1216218b870fSLee Jones n25q_read4_configs); 1217218b870fSLee Jones else 1218e6b1bb4eSLee Jones ret = stfsm_search_prepare_rw_seq(fsm, &fsm->stfsm_seq_read, 1219218b870fSLee Jones n25q_read3_configs); 1220218b870fSLee Jones if (ret) { 1221218b870fSLee Jones dev_err(fsm->dev, 1222218b870fSLee Jones "failed to prepare READ sequence with flags [0x%08x]\n", 1223218b870fSLee Jones flags); 1224218b870fSLee Jones return ret; 1225218b870fSLee Jones } 1226218b870fSLee Jones 1227218b870fSLee Jones /* Configure 'WRITE' sequence (default configs) */ 1228e6b1bb4eSLee Jones ret = stfsm_search_prepare_rw_seq(fsm, &fsm->stfsm_seq_write, 1229218b870fSLee Jones default_write_configs); 1230218b870fSLee Jones if (ret) { 1231218b870fSLee Jones dev_err(fsm->dev, 1232218b870fSLee Jones "preparing WRITE sequence using flags [0x%08x] failed\n", 1233218b870fSLee Jones flags); 1234218b870fSLee Jones return ret; 1235218b870fSLee Jones } 1236218b870fSLee Jones 1237218b870fSLee Jones /* * Configure 'ERASE_SECTOR' sequence */ 1238218b870fSLee Jones stfsm_prepare_erasesec_seq(fsm, &stfsm_seq_erase_sector); 1239218b870fSLee Jones 1240218b870fSLee Jones /* Configure 32-bit address support */ 1241218b870fSLee Jones if (flags & FLASH_FLAG_32BIT_ADDR) { 1242e6b1bb4eSLee Jones stfsm_n25q_en_32bit_addr_seq(&fsm->stfsm_seq_en_32bit_addr); 1243218b870fSLee Jones 1244218b870fSLee Jones soc_reset = stfsm_can_handle_soc_reset(fsm); 1245218b870fSLee Jones if (soc_reset || !fsm->booted_from_spi) { 1246218b870fSLee Jones /* 1247218b870fSLee Jones * If we can handle SoC resets, we enable 32-bit 1248218b870fSLee Jones * address mode pervasively 1249218b870fSLee Jones */ 1250218b870fSLee Jones stfsm_enter_32bit_addr(fsm, 1); 1251218b870fSLee Jones } else { 1252218b870fSLee Jones /* 1253218b870fSLee Jones * If not, enable/disable for WRITE and ERASE 1254218b870fSLee Jones * operations (READ uses special commands) 1255218b870fSLee Jones */ 1256218b870fSLee Jones fsm->configuration = (CFG_WRITE_TOGGLE_32BIT_ADDR | 1257218b870fSLee Jones CFG_ERASESEC_TOGGLE_32BIT_ADDR); 1258218b870fSLee Jones } 1259218b870fSLee Jones } 1260218b870fSLee Jones 1261218b870fSLee Jones /* 1262218b870fSLee Jones * Configure device to use 8 dummy cycles 1263218b870fSLee Jones */ 1264218b870fSLee Jones vcr = (N25Q_VCR_DUMMY_CYCLES(8) | N25Q_VCR_XIP_DISABLED | 1265218b870fSLee Jones N25Q_VCR_WRAP_CONT); 12665d0bddabSAngus Clark stfsm_write_status(fsm, N25Q_CMD_WRVCR, vcr, 1, 0); 1267218b870fSLee Jones 1268218b870fSLee Jones return 0; 1269218b870fSLee Jones } 1270218b870fSLee Jones 12715343a123SLee Jones static void stfsm_s25fl_prepare_erasesec_seq_32(struct stfsm_seq *seq) 12725343a123SLee Jones { 12735343a123SLee Jones seq->seq_opc[1] = (SEQ_OPC_PADS_1 | 12745343a123SLee Jones SEQ_OPC_CYCLES(8) | 12755343a123SLee Jones SEQ_OPC_OPCODE(S25FL_CMD_SE4)); 12765343a123SLee Jones 12775343a123SLee Jones seq->addr_cfg = (ADR_CFG_CYCLES_ADD1(16) | 12785343a123SLee Jones ADR_CFG_PADS_1_ADD1 | 12795343a123SLee Jones ADR_CFG_CYCLES_ADD2(16) | 12805343a123SLee Jones ADR_CFG_PADS_1_ADD2 | 12815343a123SLee Jones ADR_CFG_CSDEASSERT_ADD2); 12825343a123SLee Jones } 12835343a123SLee Jones 12845343a123SLee Jones static void stfsm_s25fl_read_dyb(struct stfsm *fsm, uint32_t offs, uint8_t *dby) 12855343a123SLee Jones { 12865343a123SLee Jones uint32_t tmp; 12875343a123SLee Jones struct stfsm_seq seq = { 12885343a123SLee Jones .data_size = TRANSFER_SIZE(4), 12895343a123SLee Jones .seq_opc[0] = (SEQ_OPC_PADS_1 | 12905343a123SLee Jones SEQ_OPC_CYCLES(8) | 12915343a123SLee Jones SEQ_OPC_OPCODE(S25FL_CMD_DYBRD)), 12925343a123SLee Jones .addr_cfg = (ADR_CFG_CYCLES_ADD1(16) | 12935343a123SLee Jones ADR_CFG_PADS_1_ADD1 | 12945343a123SLee Jones ADR_CFG_CYCLES_ADD2(16) | 12955343a123SLee Jones ADR_CFG_PADS_1_ADD2), 12965343a123SLee Jones .addr1 = (offs >> 16) & 0xffff, 12975343a123SLee Jones .addr2 = offs & 0xffff, 12985343a123SLee Jones .seq = { 12995343a123SLee Jones STFSM_INST_CMD1, 13005343a123SLee Jones STFSM_INST_ADD1, 13015343a123SLee Jones STFSM_INST_ADD2, 13025343a123SLee Jones STFSM_INST_DATA_READ, 13035343a123SLee Jones STFSM_INST_STOP, 13045343a123SLee Jones }, 13055343a123SLee Jones .seq_cfg = (SEQ_CFG_PADS_1 | 13065343a123SLee Jones SEQ_CFG_READNOTWRITE | 13075343a123SLee Jones SEQ_CFG_CSDEASSERT | 13085343a123SLee Jones SEQ_CFG_STARTSEQ), 13095343a123SLee Jones }; 13105343a123SLee Jones 13115343a123SLee Jones stfsm_load_seq(fsm, &seq); 13125343a123SLee Jones 13135343a123SLee Jones stfsm_read_fifo(fsm, &tmp, 4); 13145343a123SLee Jones 13155343a123SLee Jones *dby = (uint8_t)(tmp >> 24); 13165343a123SLee Jones 13175343a123SLee Jones stfsm_wait_seq(fsm); 13185343a123SLee Jones } 13195343a123SLee Jones 13205343a123SLee Jones static void stfsm_s25fl_write_dyb(struct stfsm *fsm, uint32_t offs, uint8_t dby) 13215343a123SLee Jones { 13225343a123SLee Jones struct stfsm_seq seq = { 13235343a123SLee Jones .seq_opc[0] = (SEQ_OPC_PADS_1 | SEQ_OPC_CYCLES(8) | 132492d3af9aSBrian Norris SEQ_OPC_OPCODE(SPINOR_OP_WREN) | 13255343a123SLee Jones SEQ_OPC_CSDEASSERT), 13265343a123SLee Jones .seq_opc[1] = (SEQ_OPC_PADS_1 | SEQ_OPC_CYCLES(8) | 13275343a123SLee Jones SEQ_OPC_OPCODE(S25FL_CMD_DYBWR)), 13285343a123SLee Jones .addr_cfg = (ADR_CFG_CYCLES_ADD1(16) | 13295343a123SLee Jones ADR_CFG_PADS_1_ADD1 | 13305343a123SLee Jones ADR_CFG_CYCLES_ADD2(16) | 13315343a123SLee Jones ADR_CFG_PADS_1_ADD2), 13325343a123SLee Jones .status = (uint32_t)dby | STA_PADS_1 | STA_CSDEASSERT, 13335343a123SLee Jones .addr1 = (offs >> 16) & 0xffff, 13345343a123SLee Jones .addr2 = offs & 0xffff, 13355343a123SLee Jones .seq = { 13365343a123SLee Jones STFSM_INST_CMD1, 13375343a123SLee Jones STFSM_INST_CMD2, 13385343a123SLee Jones STFSM_INST_ADD1, 13395343a123SLee Jones STFSM_INST_ADD2, 13405343a123SLee Jones STFSM_INST_STA_WR1, 13415343a123SLee Jones STFSM_INST_STOP, 13425343a123SLee Jones }, 13435343a123SLee Jones .seq_cfg = (SEQ_CFG_PADS_1 | 13445343a123SLee Jones SEQ_CFG_READNOTWRITE | 13455343a123SLee Jones SEQ_CFG_CSDEASSERT | 13465343a123SLee Jones SEQ_CFG_STARTSEQ), 13475343a123SLee Jones }; 13485343a123SLee Jones 13495343a123SLee Jones stfsm_load_seq(fsm, &seq); 13505343a123SLee Jones stfsm_wait_seq(fsm); 13515343a123SLee Jones 13525343a123SLee Jones stfsm_wait_busy(fsm); 13535343a123SLee Jones } 13545343a123SLee Jones 13555343a123SLee Jones static int stfsm_s25fl_clear_status_reg(struct stfsm *fsm) 13565343a123SLee Jones { 13575343a123SLee Jones struct stfsm_seq seq = { 13585343a123SLee Jones .seq_opc[0] = (SEQ_OPC_PADS_1 | 13595343a123SLee Jones SEQ_OPC_CYCLES(8) | 13605343a123SLee Jones SEQ_OPC_OPCODE(S25FL_CMD_CLSR) | 13615343a123SLee Jones SEQ_OPC_CSDEASSERT), 13625343a123SLee Jones .seq_opc[1] = (SEQ_OPC_PADS_1 | 13635343a123SLee Jones SEQ_OPC_CYCLES(8) | 136492d3af9aSBrian Norris SEQ_OPC_OPCODE(SPINOR_OP_WRDI) | 13655343a123SLee Jones SEQ_OPC_CSDEASSERT), 13665343a123SLee Jones .seq = { 13675343a123SLee Jones STFSM_INST_CMD1, 13685343a123SLee Jones STFSM_INST_CMD2, 13695343a123SLee Jones STFSM_INST_WAIT, 13705343a123SLee Jones STFSM_INST_STOP, 13715343a123SLee Jones }, 13725343a123SLee Jones .seq_cfg = (SEQ_CFG_PADS_1 | 13735343a123SLee Jones SEQ_CFG_ERASE | 13745343a123SLee Jones SEQ_CFG_READNOTWRITE | 13755343a123SLee Jones SEQ_CFG_CSDEASSERT | 13765343a123SLee Jones SEQ_CFG_STARTSEQ), 13775343a123SLee Jones }; 13785343a123SLee Jones 13795343a123SLee Jones stfsm_load_seq(fsm, &seq); 13805343a123SLee Jones 13815343a123SLee Jones stfsm_wait_seq(fsm); 13825343a123SLee Jones 13835343a123SLee Jones return 0; 13845343a123SLee Jones } 13855343a123SLee Jones 13865343a123SLee Jones static int stfsm_s25fl_config(struct stfsm *fsm) 13875343a123SLee Jones { 13885343a123SLee Jones struct flash_info *info = fsm->info; 13895343a123SLee Jones uint32_t flags = info->flags; 13905343a123SLee Jones uint32_t data_pads; 13915343a123SLee Jones uint32_t offs; 13925343a123SLee Jones uint16_t sta_wr; 13935343a123SLee Jones uint8_t sr1, cr1, dyb; 13945d0bddabSAngus Clark int update_sr = 0; 13955343a123SLee Jones int ret; 13965343a123SLee Jones 13975343a123SLee Jones if (flags & FLASH_FLAG_32BIT_ADDR) { 13985343a123SLee Jones /* 13995343a123SLee Jones * Prepare Read/Write/Erase sequences according to S25FLxxx 14005343a123SLee Jones * 32-bit address command set 14015343a123SLee Jones */ 1402e6b1bb4eSLee Jones ret = stfsm_search_prepare_rw_seq(fsm, &fsm->stfsm_seq_read, 14035343a123SLee Jones stfsm_s25fl_read4_configs); 14045343a123SLee Jones if (ret) 14055343a123SLee Jones return ret; 14065343a123SLee Jones 1407e6b1bb4eSLee Jones ret = stfsm_search_prepare_rw_seq(fsm, &fsm->stfsm_seq_write, 14085343a123SLee Jones stfsm_s25fl_write4_configs); 14095343a123SLee Jones if (ret) 14105343a123SLee Jones return ret; 14115343a123SLee Jones 14125343a123SLee Jones stfsm_s25fl_prepare_erasesec_seq_32(&stfsm_seq_erase_sector); 14135343a123SLee Jones 14145343a123SLee Jones } else { 14155343a123SLee Jones /* Use default configurations for 24-bit addressing */ 14165343a123SLee Jones ret = stfsm_prepare_rwe_seqs_default(fsm); 14175343a123SLee Jones if (ret) 14185343a123SLee Jones return ret; 14195343a123SLee Jones } 14205343a123SLee Jones 14215343a123SLee Jones /* 14225343a123SLee Jones * For devices that support 'DYB' sector locking, check lock status and 14235343a123SLee Jones * unlock sectors if necessary (some variants power-on with sectors 14245343a123SLee Jones * locked by default) 14255343a123SLee Jones */ 14265343a123SLee Jones if (flags & FLASH_FLAG_DYB_LOCKING) { 14275343a123SLee Jones offs = 0; 14285343a123SLee Jones for (offs = 0; offs < info->sector_size * info->n_sectors;) { 14295343a123SLee Jones stfsm_s25fl_read_dyb(fsm, offs, &dyb); 14305343a123SLee Jones if (dyb == 0x00) 14315343a123SLee Jones stfsm_s25fl_write_dyb(fsm, offs, 0xff); 14325343a123SLee Jones 14335343a123SLee Jones /* Handle bottom/top 4KiB parameter sectors */ 14345343a123SLee Jones if ((offs < info->sector_size * 2) || 14355343a123SLee Jones (offs >= (info->sector_size - info->n_sectors * 4))) 14365343a123SLee Jones offs += 0x1000; 14375343a123SLee Jones else 14385343a123SLee Jones offs += 0x10000; 14395343a123SLee Jones } 14405343a123SLee Jones } 14415343a123SLee Jones 14425d0bddabSAngus Clark /* Check status of 'QE' bit, update if required. */ 14439447332fSCyrille Pitchen stfsm_read_status(fsm, SPINOR_OP_RDCR, &cr1, 1); 1444e6b1bb4eSLee Jones data_pads = ((fsm->stfsm_seq_read.seq_cfg >> 16) & 0x3) + 1; 14455343a123SLee Jones if (data_pads == 4) { 14465343a123SLee Jones if (!(cr1 & STFSM_S25FL_CONFIG_QE)) { 14475343a123SLee Jones /* Set 'QE' */ 14485343a123SLee Jones cr1 |= STFSM_S25FL_CONFIG_QE; 14495343a123SLee Jones 14505d0bddabSAngus Clark update_sr = 1; 14515343a123SLee Jones } 14525343a123SLee Jones } else { 14535d0bddabSAngus Clark if (cr1 & STFSM_S25FL_CONFIG_QE) { 14545343a123SLee Jones /* Clear 'QE' */ 14555343a123SLee Jones cr1 &= ~STFSM_S25FL_CONFIG_QE; 14565343a123SLee Jones 14575d0bddabSAngus Clark update_sr = 1; 14585343a123SLee Jones } 14595d0bddabSAngus Clark } 14605d0bddabSAngus Clark if (update_sr) { 146192d3af9aSBrian Norris stfsm_read_status(fsm, SPINOR_OP_RDSR, &sr1, 1); 14625d0bddabSAngus Clark sta_wr = ((uint16_t)cr1 << 8) | sr1; 146392d3af9aSBrian Norris stfsm_write_status(fsm, SPINOR_OP_WRSR, sta_wr, 2, 1); 14645343a123SLee Jones } 14655343a123SLee Jones 14665343a123SLee Jones /* 14675343a123SLee Jones * S25FLxxx devices support Program and Error error flags. 14685343a123SLee Jones * Configure driver to check flags and clear if necessary. 14695343a123SLee Jones */ 14705343a123SLee Jones fsm->configuration |= CFG_S25FL_CHECK_ERROR_FLAGS; 14715343a123SLee Jones 14725343a123SLee Jones return 0; 14735343a123SLee Jones } 14745343a123SLee Jones 1475cd7cac9eSLee Jones static int stfsm_w25q_config(struct stfsm *fsm) 1476cd7cac9eSLee Jones { 1477cd7cac9eSLee Jones uint32_t data_pads; 14785d0bddabSAngus Clark uint8_t sr1, sr2; 14795d0bddabSAngus Clark uint16_t sr_wr; 14805d0bddabSAngus Clark int update_sr = 0; 1481cd7cac9eSLee Jones int ret; 1482cd7cac9eSLee Jones 1483cd7cac9eSLee Jones ret = stfsm_prepare_rwe_seqs_default(fsm); 1484cd7cac9eSLee Jones if (ret) 1485cd7cac9eSLee Jones return ret; 1486cd7cac9eSLee Jones 14875d0bddabSAngus Clark /* Check status of 'QE' bit, update if required. */ 14889447332fSCyrille Pitchen stfsm_read_status(fsm, SPINOR_OP_RDCR, &sr2, 1); 1489e6b1bb4eSLee Jones data_pads = ((fsm->stfsm_seq_read.seq_cfg >> 16) & 0x3) + 1; 1490cd7cac9eSLee Jones if (data_pads == 4) { 14915d0bddabSAngus Clark if (!(sr2 & W25Q_STATUS_QE)) { 14925d0bddabSAngus Clark /* Set 'QE' */ 14935d0bddabSAngus Clark sr2 |= W25Q_STATUS_QE; 14945d0bddabSAngus Clark update_sr = 1; 14955d0bddabSAngus Clark } 14965d0bddabSAngus Clark } else { 14975d0bddabSAngus Clark if (sr2 & W25Q_STATUS_QE) { 14985d0bddabSAngus Clark /* Clear 'QE' */ 14995d0bddabSAngus Clark sr2 &= ~W25Q_STATUS_QE; 15005d0bddabSAngus Clark update_sr = 1; 15015d0bddabSAngus Clark } 15025d0bddabSAngus Clark } 15035d0bddabSAngus Clark if (update_sr) { 15045d0bddabSAngus Clark /* Write status register */ 150592d3af9aSBrian Norris stfsm_read_status(fsm, SPINOR_OP_RDSR, &sr1, 1); 15065d0bddabSAngus Clark sr_wr = ((uint16_t)sr2 << 8) | sr1; 150792d3af9aSBrian Norris stfsm_write_status(fsm, SPINOR_OP_WRSR, sr_wr, 2, 1); 1508cd7cac9eSLee Jones } 1509cd7cac9eSLee Jones 1510cd7cac9eSLee Jones return 0; 1511cd7cac9eSLee Jones } 1512cd7cac9eSLee Jones 1513e514f105SLee Jones static int stfsm_read(struct stfsm *fsm, uint8_t *buf, uint32_t size, 1514e514f105SLee Jones uint32_t offset) 1515e514f105SLee Jones { 1516e6b1bb4eSLee Jones struct stfsm_seq *seq = &fsm->stfsm_seq_read; 1517e514f105SLee Jones uint32_t data_pads; 1518e514f105SLee Jones uint32_t read_mask; 1519e514f105SLee Jones uint32_t size_ub; 1520e514f105SLee Jones uint32_t size_lb; 1521e514f105SLee Jones uint32_t size_mop; 1522e514f105SLee Jones uint32_t tmp[4]; 1523e514f105SLee Jones uint32_t page_buf[FLASH_PAGESIZE_32]; 1524e514f105SLee Jones uint8_t *p; 1525e514f105SLee Jones 1526e514f105SLee Jones dev_dbg(fsm->dev, "reading %d bytes from 0x%08x\n", size, offset); 1527e514f105SLee Jones 1528e514f105SLee Jones /* Enter 32-bit address mode, if required */ 1529e514f105SLee Jones if (fsm->configuration & CFG_READ_TOGGLE_32BIT_ADDR) 1530e514f105SLee Jones stfsm_enter_32bit_addr(fsm, 1); 1531e514f105SLee Jones 1532e514f105SLee Jones /* Must read in multiples of 32 cycles (or 32*pads/8 Bytes) */ 1533e514f105SLee Jones data_pads = ((seq->seq_cfg >> 16) & 0x3) + 1; 1534e514f105SLee Jones read_mask = (data_pads << 2) - 1; 1535e514f105SLee Jones 1536e514f105SLee Jones /* Handle non-aligned buf */ 153738e2eee9SBrian Norris p = ((uintptr_t)buf & 0x3) ? (uint8_t *)page_buf : buf; 1538e514f105SLee Jones 1539e514f105SLee Jones /* Handle non-aligned size */ 1540e514f105SLee Jones size_ub = (size + read_mask) & ~read_mask; 1541e514f105SLee Jones size_lb = size & ~read_mask; 1542e514f105SLee Jones size_mop = size & read_mask; 1543e514f105SLee Jones 1544e514f105SLee Jones seq->data_size = TRANSFER_SIZE(size_ub); 1545e514f105SLee Jones seq->addr1 = (offset >> 16) & 0xffff; 1546e514f105SLee Jones seq->addr2 = offset & 0xffff; 1547e514f105SLee Jones 1548e514f105SLee Jones stfsm_load_seq(fsm, seq); 1549e514f105SLee Jones 1550e514f105SLee Jones if (size_lb) 1551e514f105SLee Jones stfsm_read_fifo(fsm, (uint32_t *)p, size_lb); 1552e514f105SLee Jones 1553e514f105SLee Jones if (size_mop) { 1554e514f105SLee Jones stfsm_read_fifo(fsm, tmp, read_mask + 1); 1555e514f105SLee Jones memcpy(p + size_lb, &tmp, size_mop); 1556e514f105SLee Jones } 1557e514f105SLee Jones 1558e514f105SLee Jones /* Handle non-aligned buf */ 155938e2eee9SBrian Norris if ((uintptr_t)buf & 0x3) 1560e514f105SLee Jones memcpy(buf, page_buf, size); 1561e514f105SLee Jones 1562e514f105SLee Jones /* Wait for sequence to finish */ 1563e514f105SLee Jones stfsm_wait_seq(fsm); 1564e514f105SLee Jones 1565e514f105SLee Jones stfsm_clear_fifo(fsm); 1566e514f105SLee Jones 1567e514f105SLee Jones /* Exit 32-bit address mode, if required */ 1568e514f105SLee Jones if (fsm->configuration & CFG_READ_TOGGLE_32BIT_ADDR) 1569e514f105SLee Jones stfsm_enter_32bit_addr(fsm, 0); 1570e514f105SLee Jones 1571e514f105SLee Jones return 0; 1572e514f105SLee Jones } 1573e514f105SLee Jones 15743f9d720aSLee Jones static int stfsm_write(struct stfsm *fsm, const uint8_t *buf, 15753f9d720aSLee Jones uint32_t size, uint32_t offset) 1576176b4377SLee Jones { 1577e6b1bb4eSLee Jones struct stfsm_seq *seq = &fsm->stfsm_seq_write; 1578176b4377SLee Jones uint32_t data_pads; 1579176b4377SLee Jones uint32_t write_mask; 1580176b4377SLee Jones uint32_t size_ub; 1581176b4377SLee Jones uint32_t size_lb; 1582176b4377SLee Jones uint32_t size_mop; 1583176b4377SLee Jones uint32_t tmp[4]; 1584a9b679bfSLee Jones uint32_t i; 1585176b4377SLee Jones uint32_t page_buf[FLASH_PAGESIZE_32]; 1586176b4377SLee Jones uint8_t *t = (uint8_t *)&tmp; 1587176b4377SLee Jones const uint8_t *p; 1588176b4377SLee Jones int ret; 1589176b4377SLee Jones 1590176b4377SLee Jones dev_dbg(fsm->dev, "writing %d bytes to 0x%08x\n", size, offset); 1591176b4377SLee Jones 1592176b4377SLee Jones /* Enter 32-bit address mode, if required */ 1593176b4377SLee Jones if (fsm->configuration & CFG_WRITE_TOGGLE_32BIT_ADDR) 1594176b4377SLee Jones stfsm_enter_32bit_addr(fsm, 1); 1595176b4377SLee Jones 1596176b4377SLee Jones /* Must write in multiples of 32 cycles (or 32*pads/8 bytes) */ 1597176b4377SLee Jones data_pads = ((seq->seq_cfg >> 16) & 0x3) + 1; 1598176b4377SLee Jones write_mask = (data_pads << 2) - 1; 1599176b4377SLee Jones 1600176b4377SLee Jones /* Handle non-aligned buf */ 160138e2eee9SBrian Norris if ((uintptr_t)buf & 0x3) { 1602176b4377SLee Jones memcpy(page_buf, buf, size); 1603176b4377SLee Jones p = (uint8_t *)page_buf; 1604176b4377SLee Jones } else { 1605176b4377SLee Jones p = buf; 1606176b4377SLee Jones } 1607176b4377SLee Jones 1608176b4377SLee Jones /* Handle non-aligned size */ 1609176b4377SLee Jones size_ub = (size + write_mask) & ~write_mask; 1610176b4377SLee Jones size_lb = size & ~write_mask; 1611176b4377SLee Jones size_mop = size & write_mask; 1612176b4377SLee Jones 1613176b4377SLee Jones seq->data_size = TRANSFER_SIZE(size_ub); 1614176b4377SLee Jones seq->addr1 = (offset >> 16) & 0xffff; 1615176b4377SLee Jones seq->addr2 = offset & 0xffff; 1616176b4377SLee Jones 1617176b4377SLee Jones /* Need to set FIFO to write mode, before writing data to FIFO (see 1618176b4377SLee Jones * GNBvb79594) 1619176b4377SLee Jones */ 1620176b4377SLee Jones writel(0x00040000, fsm->base + SPI_FAST_SEQ_CFG); 1621176b4377SLee Jones 1622176b4377SLee Jones /* 1623176b4377SLee Jones * Before writing data to the FIFO, apply a small delay to allow a 1624176b4377SLee Jones * potential change of FIFO direction to complete. 1625176b4377SLee Jones */ 1626176b4377SLee Jones if (fsm->fifo_dir_delay == 0) 1627176b4377SLee Jones readl(fsm->base + SPI_FAST_SEQ_CFG); 1628176b4377SLee Jones else 1629176b4377SLee Jones udelay(fsm->fifo_dir_delay); 1630176b4377SLee Jones 1631176b4377SLee Jones 1632176b4377SLee Jones /* Write data to FIFO, before starting sequence (see GNBvd79593) */ 1633176b4377SLee Jones if (size_lb) { 1634176b4377SLee Jones stfsm_write_fifo(fsm, (uint32_t *)p, size_lb); 1635176b4377SLee Jones p += size_lb; 1636176b4377SLee Jones } 1637176b4377SLee Jones 1638176b4377SLee Jones /* Handle non-aligned size */ 1639176b4377SLee Jones if (size_mop) { 1640176b4377SLee Jones memset(t, 0xff, write_mask + 1); /* fill with 0xff's */ 1641176b4377SLee Jones for (i = 0; i < size_mop; i++) 1642176b4377SLee Jones t[i] = *p++; 1643176b4377SLee Jones 1644176b4377SLee Jones stfsm_write_fifo(fsm, tmp, write_mask + 1); 1645176b4377SLee Jones } 1646176b4377SLee Jones 1647176b4377SLee Jones /* Start sequence */ 1648176b4377SLee Jones stfsm_load_seq(fsm, seq); 1649176b4377SLee Jones 1650176b4377SLee Jones /* Wait for sequence to finish */ 1651176b4377SLee Jones stfsm_wait_seq(fsm); 1652176b4377SLee Jones 1653176b4377SLee Jones /* Wait for completion */ 1654176b4377SLee Jones ret = stfsm_wait_busy(fsm); 16555343a123SLee Jones if (ret && fsm->configuration & CFG_S25FL_CHECK_ERROR_FLAGS) 16565343a123SLee Jones stfsm_s25fl_clear_status_reg(fsm); 1657176b4377SLee Jones 1658176b4377SLee Jones /* Exit 32-bit address mode, if required */ 1659009e7e61SAngus Clark if (fsm->configuration & CFG_WRITE_TOGGLE_32BIT_ADDR) 1660176b4377SLee Jones stfsm_enter_32bit_addr(fsm, 0); 1661176b4377SLee Jones 1662176b4377SLee Jones return 0; 1663176b4377SLee Jones } 1664176b4377SLee Jones 1665e514f105SLee Jones /* 1666e514f105SLee Jones * Read an address range from the flash chip. The address range 1667e514f105SLee Jones * may be any size provided it is within the physical boundaries. 1668e514f105SLee Jones */ 1669e514f105SLee Jones static int stfsm_mtd_read(struct mtd_info *mtd, loff_t from, size_t len, 1670e514f105SLee Jones size_t *retlen, u_char *buf) 1671e514f105SLee Jones { 1672e514f105SLee Jones struct stfsm *fsm = dev_get_drvdata(mtd->dev.parent); 1673e514f105SLee Jones uint32_t bytes; 1674e514f105SLee Jones 1675e514f105SLee Jones dev_dbg(fsm->dev, "%s from 0x%08x, len %zd\n", 1676e514f105SLee Jones __func__, (u32)from, len); 1677e514f105SLee Jones 1678e514f105SLee Jones mutex_lock(&fsm->lock); 1679e514f105SLee Jones 1680e514f105SLee Jones while (len > 0) { 1681e514f105SLee Jones bytes = min_t(size_t, len, FLASH_PAGESIZE); 1682e514f105SLee Jones 1683e514f105SLee Jones stfsm_read(fsm, buf, bytes, from); 1684e514f105SLee Jones 1685e514f105SLee Jones buf += bytes; 1686e514f105SLee Jones from += bytes; 1687e514f105SLee Jones len -= bytes; 1688e514f105SLee Jones 1689e514f105SLee Jones *retlen += bytes; 1690e514f105SLee Jones } 1691e514f105SLee Jones 1692e514f105SLee Jones mutex_unlock(&fsm->lock); 1693e514f105SLee Jones 1694e514f105SLee Jones return 0; 1695e514f105SLee Jones } 1696e514f105SLee Jones 16973f9d720aSLee Jones static int stfsm_erase_sector(struct stfsm *fsm, uint32_t offset) 16984a341fe7SLee Jones { 16994a341fe7SLee Jones struct stfsm_seq *seq = &stfsm_seq_erase_sector; 17004a341fe7SLee Jones int ret; 17014a341fe7SLee Jones 17024a341fe7SLee Jones dev_dbg(fsm->dev, "erasing sector at 0x%08x\n", offset); 17034a341fe7SLee Jones 17044a341fe7SLee Jones /* Enter 32-bit address mode, if required */ 17054a341fe7SLee Jones if (fsm->configuration & CFG_ERASESEC_TOGGLE_32BIT_ADDR) 17064a341fe7SLee Jones stfsm_enter_32bit_addr(fsm, 1); 17074a341fe7SLee Jones 17084a341fe7SLee Jones seq->addr1 = (offset >> 16) & 0xffff; 17094a341fe7SLee Jones seq->addr2 = offset & 0xffff; 17104a341fe7SLee Jones 17114a341fe7SLee Jones stfsm_load_seq(fsm, seq); 17124a341fe7SLee Jones 17134a341fe7SLee Jones stfsm_wait_seq(fsm); 17144a341fe7SLee Jones 17154a341fe7SLee Jones /* Wait for completion */ 17164a341fe7SLee Jones ret = stfsm_wait_busy(fsm); 17175343a123SLee Jones if (ret && fsm->configuration & CFG_S25FL_CHECK_ERROR_FLAGS) 17185343a123SLee Jones stfsm_s25fl_clear_status_reg(fsm); 17194a341fe7SLee Jones 17204a341fe7SLee Jones /* Exit 32-bit address mode, if required */ 17214a341fe7SLee Jones if (fsm->configuration & CFG_ERASESEC_TOGGLE_32BIT_ADDR) 17224a341fe7SLee Jones stfsm_enter_32bit_addr(fsm, 0); 17234a341fe7SLee Jones 17244a341fe7SLee Jones return ret; 17254a341fe7SLee Jones } 17264a341fe7SLee Jones 17274a341fe7SLee Jones static int stfsm_erase_chip(struct stfsm *fsm) 17284a341fe7SLee Jones { 17294a341fe7SLee Jones const struct stfsm_seq *seq = &stfsm_seq_erase_chip; 17304a341fe7SLee Jones 17314a341fe7SLee Jones dev_dbg(fsm->dev, "erasing chip\n"); 17324a341fe7SLee Jones 17334a341fe7SLee Jones stfsm_load_seq(fsm, seq); 17344a341fe7SLee Jones 17354a341fe7SLee Jones stfsm_wait_seq(fsm); 17364a341fe7SLee Jones 17374a341fe7SLee Jones return stfsm_wait_busy(fsm); 17384a341fe7SLee Jones } 17394a341fe7SLee Jones 1740176b4377SLee Jones /* 1741176b4377SLee Jones * Write an address range to the flash chip. Data must be written in 1742176b4377SLee Jones * FLASH_PAGESIZE chunks. The address range may be any size provided 1743176b4377SLee Jones * it is within the physical boundaries. 1744176b4377SLee Jones */ 1745176b4377SLee Jones static int stfsm_mtd_write(struct mtd_info *mtd, loff_t to, size_t len, 1746176b4377SLee Jones size_t *retlen, const u_char *buf) 1747176b4377SLee Jones { 1748176b4377SLee Jones struct stfsm *fsm = dev_get_drvdata(mtd->dev.parent); 1749176b4377SLee Jones 1750176b4377SLee Jones u32 page_offs; 1751176b4377SLee Jones u32 bytes; 1752176b4377SLee Jones uint8_t *b = (uint8_t *)buf; 1753176b4377SLee Jones int ret = 0; 1754176b4377SLee Jones 1755176b4377SLee Jones dev_dbg(fsm->dev, "%s to 0x%08x, len %zd\n", __func__, (u32)to, len); 1756176b4377SLee Jones 1757176b4377SLee Jones /* Offset within page */ 1758176b4377SLee Jones page_offs = to % FLASH_PAGESIZE; 1759176b4377SLee Jones 1760176b4377SLee Jones mutex_lock(&fsm->lock); 1761176b4377SLee Jones 1762176b4377SLee Jones while (len) { 1763176b4377SLee Jones /* Write up to page boundary */ 176438e2eee9SBrian Norris bytes = min_t(size_t, FLASH_PAGESIZE - page_offs, len); 1765176b4377SLee Jones 1766176b4377SLee Jones ret = stfsm_write(fsm, b, bytes, to); 1767176b4377SLee Jones if (ret) 1768176b4377SLee Jones goto out1; 1769176b4377SLee Jones 1770176b4377SLee Jones b += bytes; 1771176b4377SLee Jones len -= bytes; 1772176b4377SLee Jones to += bytes; 1773176b4377SLee Jones 1774176b4377SLee Jones /* We are now page-aligned */ 1775176b4377SLee Jones page_offs = 0; 1776176b4377SLee Jones 1777176b4377SLee Jones *retlen += bytes; 1778176b4377SLee Jones 1779176b4377SLee Jones } 1780176b4377SLee Jones 1781176b4377SLee Jones out1: 1782176b4377SLee Jones mutex_unlock(&fsm->lock); 1783176b4377SLee Jones 1784176b4377SLee Jones return ret; 1785176b4377SLee Jones } 1786176b4377SLee Jones 17874a341fe7SLee Jones /* 17884a341fe7SLee Jones * Erase an address range on the flash chip. The address range may extend 17894a341fe7SLee Jones * one or more erase sectors. Return an error is there is a problem erasing. 17904a341fe7SLee Jones */ 17914a341fe7SLee Jones static int stfsm_mtd_erase(struct mtd_info *mtd, struct erase_info *instr) 17924a341fe7SLee Jones { 17934a341fe7SLee Jones struct stfsm *fsm = dev_get_drvdata(mtd->dev.parent); 17944a341fe7SLee Jones u32 addr, len; 17954a341fe7SLee Jones int ret; 17964a341fe7SLee Jones 17974a341fe7SLee Jones dev_dbg(fsm->dev, "%s at 0x%llx, len %lld\n", __func__, 17984a341fe7SLee Jones (long long)instr->addr, (long long)instr->len); 17994a341fe7SLee Jones 18004a341fe7SLee Jones addr = instr->addr; 18014a341fe7SLee Jones len = instr->len; 18024a341fe7SLee Jones 18034a341fe7SLee Jones mutex_lock(&fsm->lock); 18044a341fe7SLee Jones 18054a341fe7SLee Jones /* Whole-chip erase? */ 18064a341fe7SLee Jones if (len == mtd->size) { 18074a341fe7SLee Jones ret = stfsm_erase_chip(fsm); 18084a341fe7SLee Jones if (ret) 18094a341fe7SLee Jones goto out1; 18104a341fe7SLee Jones } else { 18114a341fe7SLee Jones while (len) { 18124a341fe7SLee Jones ret = stfsm_erase_sector(fsm, addr); 18134a341fe7SLee Jones if (ret) 18144a341fe7SLee Jones goto out1; 18154a341fe7SLee Jones 18164a341fe7SLee Jones addr += mtd->erasesize; 18174a341fe7SLee Jones len -= mtd->erasesize; 18184a341fe7SLee Jones } 18194a341fe7SLee Jones } 18204a341fe7SLee Jones 18214a341fe7SLee Jones mutex_unlock(&fsm->lock); 18224a341fe7SLee Jones 18234a341fe7SLee Jones return 0; 18244a341fe7SLee Jones 18254a341fe7SLee Jones out1: 18264a341fe7SLee Jones mutex_unlock(&fsm->lock); 18274a341fe7SLee Jones 18284a341fe7SLee Jones return ret; 18294a341fe7SLee Jones } 18304a341fe7SLee Jones 18313f9d720aSLee Jones static void stfsm_read_jedec(struct stfsm *fsm, uint8_t *jedec) 18321bd512b5SLee Jones { 18331bd512b5SLee Jones const struct stfsm_seq *seq = &stfsm_seq_read_jedec; 18341bd512b5SLee Jones uint32_t tmp[2]; 18351bd512b5SLee Jones 18361bd512b5SLee Jones stfsm_load_seq(fsm, seq); 18371bd512b5SLee Jones 18381bd512b5SLee Jones stfsm_read_fifo(fsm, tmp, 8); 18391bd512b5SLee Jones 18401bd512b5SLee Jones memcpy(jedec, tmp, 5); 18411bd512b5SLee Jones 18421bd512b5SLee Jones stfsm_wait_seq(fsm); 18431bd512b5SLee Jones } 18441bd512b5SLee Jones 18451bd512b5SLee Jones static struct flash_info *stfsm_jedec_probe(struct stfsm *fsm) 18461bd512b5SLee Jones { 184724fec651SLee Jones struct flash_info *info; 18481bd512b5SLee Jones u16 ext_jedec; 18491bd512b5SLee Jones u32 jedec; 18501bd512b5SLee Jones u8 id[5]; 18511bd512b5SLee Jones 18521bd512b5SLee Jones stfsm_read_jedec(fsm, id); 18531bd512b5SLee Jones 18541bd512b5SLee Jones jedec = id[0] << 16 | id[1] << 8 | id[2]; 18551bd512b5SLee Jones /* 18561bd512b5SLee Jones * JEDEC also defines an optional "extended device information" 18571bd512b5SLee Jones * string for after vendor-specific data, after the three bytes 18581bd512b5SLee Jones * we use here. Supporting some chips might require using it. 18591bd512b5SLee Jones */ 18601bd512b5SLee Jones ext_jedec = id[3] << 8 | id[4]; 18611bd512b5SLee Jones 1862f0a37a8dSAntonio Cardace dev_dbg(fsm->dev, "JEDEC = 0x%08x [%5ph]\n", jedec, id); 18631bd512b5SLee Jones 186424fec651SLee Jones for (info = flash_types; info->name; info++) { 186524fec651SLee Jones if (info->jedec_id == jedec) { 186624fec651SLee Jones if (info->ext_id && info->ext_id != ext_jedec) 186724fec651SLee Jones continue; 186824fec651SLee Jones return info; 186924fec651SLee Jones } 187024fec651SLee Jones } 187124fec651SLee Jones dev_err(fsm->dev, "Unrecognized JEDEC id %06x\n", jedec); 187224fec651SLee Jones 18731bd512b5SLee Jones return NULL; 18741bd512b5SLee Jones } 18751bd512b5SLee Jones 187686f309fdSLee Jones static int stfsm_set_mode(struct stfsm *fsm, uint32_t mode) 187786f309fdSLee Jones { 187886f309fdSLee Jones int ret, timeout = 10; 187986f309fdSLee Jones 188086f309fdSLee Jones /* Wait for controller to accept mode change */ 188186f309fdSLee Jones while (--timeout) { 188286f309fdSLee Jones ret = readl(fsm->base + SPI_STA_MODE_CHANGE); 188386f309fdSLee Jones if (ret & 0x1) 188486f309fdSLee Jones break; 188586f309fdSLee Jones udelay(1); 188686f309fdSLee Jones } 188786f309fdSLee Jones 188886f309fdSLee Jones if (!timeout) 188986f309fdSLee Jones return -EBUSY; 189086f309fdSLee Jones 189186f309fdSLee Jones writel(mode, fsm->base + SPI_MODESELECT); 189286f309fdSLee Jones 189386f309fdSLee Jones return 0; 189486f309fdSLee Jones } 189586f309fdSLee Jones 189686f309fdSLee Jones static void stfsm_set_freq(struct stfsm *fsm, uint32_t spi_freq) 189786f309fdSLee Jones { 189886f309fdSLee Jones uint32_t emi_freq; 189986f309fdSLee Jones uint32_t clk_div; 190086f309fdSLee Jones 190169d5af8dSLee Jones emi_freq = clk_get_rate(fsm->clk); 190286f309fdSLee Jones 190386f309fdSLee Jones /* 190486f309fdSLee Jones * Calculate clk_div - values between 2 and 128 190586f309fdSLee Jones * Multiple of 2, rounded up 190686f309fdSLee Jones */ 190786f309fdSLee Jones clk_div = 2 * DIV_ROUND_UP(emi_freq, 2 * spi_freq); 190886f309fdSLee Jones if (clk_div < 2) 190986f309fdSLee Jones clk_div = 2; 191086f309fdSLee Jones else if (clk_div > 128) 191186f309fdSLee Jones clk_div = 128; 191286f309fdSLee Jones 191386f309fdSLee Jones /* 191486f309fdSLee Jones * Determine a suitable delay for the IP to complete a change of 191586f309fdSLee Jones * direction of the FIFO. The required delay is related to the clock 191686f309fdSLee Jones * divider used. The following heuristics are based on empirical tests, 191786f309fdSLee Jones * using a 100MHz EMI clock. 191886f309fdSLee Jones */ 191986f309fdSLee Jones if (clk_div <= 4) 192086f309fdSLee Jones fsm->fifo_dir_delay = 0; 192186f309fdSLee Jones else if (clk_div <= 10) 192286f309fdSLee Jones fsm->fifo_dir_delay = 1; 192386f309fdSLee Jones else 192486f309fdSLee Jones fsm->fifo_dir_delay = DIV_ROUND_UP(clk_div, 10); 192586f309fdSLee Jones 192686f309fdSLee Jones dev_dbg(fsm->dev, "emi_clk = %uHZ, spi_freq = %uHZ, clk_div = %u\n", 192786f309fdSLee Jones emi_freq, spi_freq, clk_div); 192886f309fdSLee Jones 192986f309fdSLee Jones writel(clk_div, fsm->base + SPI_CLOCKDIV); 193086f309fdSLee Jones } 193186f309fdSLee Jones 193286f309fdSLee Jones static int stfsm_init(struct stfsm *fsm) 193386f309fdSLee Jones { 193486f309fdSLee Jones int ret; 193586f309fdSLee Jones 193686f309fdSLee Jones /* Perform a soft reset of the FSM controller */ 193786f309fdSLee Jones writel(SEQ_CFG_SWRESET, fsm->base + SPI_FAST_SEQ_CFG); 193886f309fdSLee Jones udelay(1); 193986f309fdSLee Jones writel(0, fsm->base + SPI_FAST_SEQ_CFG); 194086f309fdSLee Jones 194186f309fdSLee Jones /* Set clock to 'safe' frequency initially */ 194286f309fdSLee Jones stfsm_set_freq(fsm, STFSM_FLASH_SAFE_FREQ); 194386f309fdSLee Jones 194486f309fdSLee Jones /* Switch to FSM */ 194586f309fdSLee Jones ret = stfsm_set_mode(fsm, SPI_MODESELECT_FSM); 194686f309fdSLee Jones if (ret) 194786f309fdSLee Jones return ret; 194886f309fdSLee Jones 194986f309fdSLee Jones /* Set timing parameters */ 195086f309fdSLee Jones writel(SPI_CFG_DEVICE_ST | 195186f309fdSLee Jones SPI_CFG_DEFAULT_MIN_CS_HIGH | 195286f309fdSLee Jones SPI_CFG_DEFAULT_CS_SETUPHOLD | 195386f309fdSLee Jones SPI_CFG_DEFAULT_DATA_HOLD, 195486f309fdSLee Jones fsm->base + SPI_CONFIGDATA); 195586f309fdSLee Jones writel(STFSM_DEFAULT_WR_TIME, fsm->base + SPI_STATUS_WR_TIME_REG); 195686f309fdSLee Jones 1957009e7e61SAngus Clark /* 1958009e7e61SAngus Clark * Set the FSM 'WAIT' delay to the minimum workable value. Note, for 1959009e7e61SAngus Clark * our purposes, the WAIT instruction is used purely to achieve 1960009e7e61SAngus Clark * "sequence validity" rather than actually implement a delay. 1961009e7e61SAngus Clark */ 1962009e7e61SAngus Clark writel(0x00000001, fsm->base + SPI_PROGRAM_ERASE_TIME); 1963009e7e61SAngus Clark 196486f309fdSLee Jones /* Clear FIFO, just in case */ 196586f309fdSLee Jones stfsm_clear_fifo(fsm); 196686f309fdSLee Jones 196786f309fdSLee Jones return 0; 196886f309fdSLee Jones } 196986f309fdSLee Jones 1970a63984c1SLee Jones static void stfsm_fetch_platform_configs(struct platform_device *pdev) 1971a63984c1SLee Jones { 1972a63984c1SLee Jones struct stfsm *fsm = platform_get_drvdata(pdev); 1973a63984c1SLee Jones struct device_node *np = pdev->dev.of_node; 1974a63984c1SLee Jones struct regmap *regmap; 1975a63984c1SLee Jones uint32_t boot_device_reg; 1976a63984c1SLee Jones uint32_t boot_device_spi; 1977a63984c1SLee Jones uint32_t boot_device; /* Value we read from *boot_device_reg */ 1978a63984c1SLee Jones int ret; 1979a63984c1SLee Jones 1980a63984c1SLee Jones /* Booting from SPI NOR Flash is the default */ 1981a63984c1SLee Jones fsm->booted_from_spi = true; 1982a63984c1SLee Jones 1983a63984c1SLee Jones regmap = syscon_regmap_lookup_by_phandle(np, "st,syscfg"); 1984a63984c1SLee Jones if (IS_ERR(regmap)) 1985a63984c1SLee Jones goto boot_device_fail; 1986a63984c1SLee Jones 19870ea7d706SLee Jones fsm->reset_signal = of_property_read_bool(np, "st,reset-signal"); 19880ea7d706SLee Jones 19890ea7d706SLee Jones fsm->reset_por = of_property_read_bool(np, "st,reset-por"); 19900ea7d706SLee Jones 1991a63984c1SLee Jones /* Where in the syscon the boot device information lives */ 1992a63984c1SLee Jones ret = of_property_read_u32(np, "st,boot-device-reg", &boot_device_reg); 1993a63984c1SLee Jones if (ret) 1994a63984c1SLee Jones goto boot_device_fail; 1995a63984c1SLee Jones 1996a63984c1SLee Jones /* Boot device value when booted from SPI NOR */ 1997a63984c1SLee Jones ret = of_property_read_u32(np, "st,boot-device-spi", &boot_device_spi); 1998a63984c1SLee Jones if (ret) 1999a63984c1SLee Jones goto boot_device_fail; 2000a63984c1SLee Jones 2001a63984c1SLee Jones ret = regmap_read(regmap, boot_device_reg, &boot_device); 2002a63984c1SLee Jones if (ret) 2003a63984c1SLee Jones goto boot_device_fail; 2004a63984c1SLee Jones 2005a63984c1SLee Jones if (boot_device != boot_device_spi) 2006a63984c1SLee Jones fsm->booted_from_spi = false; 2007a63984c1SLee Jones 2008a63984c1SLee Jones return; 2009a63984c1SLee Jones 2010a63984c1SLee Jones boot_device_fail: 2011a63984c1SLee Jones dev_warn(&pdev->dev, 2012a63984c1SLee Jones "failed to fetch boot device, assuming boot from SPI\n"); 2013a63984c1SLee Jones } 2014a63984c1SLee Jones 2015d90db4a0SLee Jones static int stfsm_probe(struct platform_device *pdev) 2016d90db4a0SLee Jones { 2017d90db4a0SLee Jones struct device_node *np = pdev->dev.of_node; 201824fec651SLee Jones struct flash_info *info; 2019d90db4a0SLee Jones struct resource *res; 2020d90db4a0SLee Jones struct stfsm *fsm; 202186f309fdSLee Jones int ret; 2022d90db4a0SLee Jones 2023d90db4a0SLee Jones if (!np) { 2024d90db4a0SLee Jones dev_err(&pdev->dev, "No DT found\n"); 2025d90db4a0SLee Jones return -EINVAL; 2026d90db4a0SLee Jones } 2027d90db4a0SLee Jones 2028d90db4a0SLee Jones fsm = devm_kzalloc(&pdev->dev, sizeof(*fsm), GFP_KERNEL); 2029d90db4a0SLee Jones if (!fsm) 2030d90db4a0SLee Jones return -ENOMEM; 2031d90db4a0SLee Jones 2032d90db4a0SLee Jones fsm->dev = &pdev->dev; 2033d90db4a0SLee Jones 2034d90db4a0SLee Jones platform_set_drvdata(pdev, fsm); 2035d90db4a0SLee Jones 2036d90db4a0SLee Jones res = platform_get_resource(pdev, IORESOURCE_MEM, 0); 2037d90db4a0SLee Jones if (!res) { 2038d90db4a0SLee Jones dev_err(&pdev->dev, "Resource not found\n"); 2039d90db4a0SLee Jones return -ENODEV; 2040d90db4a0SLee Jones } 2041d90db4a0SLee Jones 2042d90db4a0SLee Jones fsm->base = devm_ioremap_resource(&pdev->dev, res); 2043d90db4a0SLee Jones if (IS_ERR(fsm->base)) { 2044d90db4a0SLee Jones dev_err(&pdev->dev, 2045d90db4a0SLee Jones "Failed to reserve memory region %pR\n", res); 2046d90db4a0SLee Jones return PTR_ERR(fsm->base); 2047d90db4a0SLee Jones } 2048d90db4a0SLee Jones 204969d5af8dSLee Jones fsm->clk = devm_clk_get(&pdev->dev, NULL); 205069d5af8dSLee Jones if (IS_ERR(fsm->clk)) { 205169d5af8dSLee Jones dev_err(fsm->dev, "Couldn't find EMI clock.\n"); 205269d5af8dSLee Jones return PTR_ERR(fsm->clk); 205369d5af8dSLee Jones } 205469d5af8dSLee Jones 205569d5af8dSLee Jones ret = clk_prepare_enable(fsm->clk); 205669d5af8dSLee Jones if (ret) { 205769d5af8dSLee Jones dev_err(fsm->dev, "Failed to enable EMI clock.\n"); 205869d5af8dSLee Jones return ret; 205969d5af8dSLee Jones } 206069d5af8dSLee Jones 2061d90db4a0SLee Jones mutex_init(&fsm->lock); 2062d90db4a0SLee Jones 206386f309fdSLee Jones ret = stfsm_init(fsm); 206486f309fdSLee Jones if (ret) { 206586f309fdSLee Jones dev_err(&pdev->dev, "Failed to initialise FSM Controller\n"); 2066481815a6SArvind Yadav goto err_clk_unprepare; 206786f309fdSLee Jones } 206886f309fdSLee Jones 2069a63984c1SLee Jones stfsm_fetch_platform_configs(pdev); 2070a63984c1SLee Jones 20711bd512b5SLee Jones /* Detect SPI FLASH device */ 207224fec651SLee Jones info = stfsm_jedec_probe(fsm); 2073481815a6SArvind Yadav if (!info) { 2074481815a6SArvind Yadav ret = -ENODEV; 2075481815a6SArvind Yadav goto err_clk_unprepare; 2076481815a6SArvind Yadav } 207724fec651SLee Jones fsm->info = info; 20781bd512b5SLee Jones 20793b5d1981SLee Jones /* Use device size to determine address width */ 20803b5d1981SLee Jones if (info->sector_size * info->n_sectors > 0x1000000) 20813b5d1981SLee Jones info->flags |= FLASH_FLAG_32BIT_ADDR; 20823b5d1981SLee Jones 2083218b870fSLee Jones /* 2084218b870fSLee Jones * Configure READ/WRITE/ERASE sequences according to platform and 2085218b870fSLee Jones * device flags. 2086218b870fSLee Jones */ 2087218b870fSLee Jones if (info->config) { 2088218b870fSLee Jones ret = info->config(fsm); 2089218b870fSLee Jones if (ret) 2090481815a6SArvind Yadav goto err_clk_unprepare; 20914eb3f0d8SLee Jones } else { 20924eb3f0d8SLee Jones ret = stfsm_prepare_rwe_seqs_default(fsm); 20934eb3f0d8SLee Jones if (ret) 2094481815a6SArvind Yadav goto err_clk_unprepare; 2095218b870fSLee Jones } 2096218b870fSLee Jones 2097221cff13SLee Jones fsm->mtd.name = info->name; 2098d90db4a0SLee Jones fsm->mtd.dev.parent = &pdev->dev; 2099004b5e60SBrian Norris mtd_set_of_node(&fsm->mtd, np); 2100d90db4a0SLee Jones fsm->mtd.type = MTD_NORFLASH; 2101d90db4a0SLee Jones fsm->mtd.writesize = 4; 2102d90db4a0SLee Jones fsm->mtd.writebufsize = fsm->mtd.writesize; 2103d90db4a0SLee Jones fsm->mtd.flags = MTD_CAP_NORFLASH; 210424fec651SLee Jones fsm->mtd.size = info->sector_size * info->n_sectors; 210524fec651SLee Jones fsm->mtd.erasesize = info->sector_size; 210624fec651SLee Jones 2107e514f105SLee Jones fsm->mtd._read = stfsm_mtd_read; 2108176b4377SLee Jones fsm->mtd._write = stfsm_mtd_write; 21094a341fe7SLee Jones fsm->mtd._erase = stfsm_mtd_erase; 2110e514f105SLee Jones 21114a341fe7SLee Jones dev_info(&pdev->dev, 211224fec651SLee Jones "Found serial flash device: %s\n" 211324fec651SLee Jones " size = %llx (%lldMiB) erasesize = 0x%08x (%uKiB)\n", 211424fec651SLee Jones info->name, 211524fec651SLee Jones (long long)fsm->mtd.size, (long long)(fsm->mtd.size >> 20), 211624fec651SLee Jones fsm->mtd.erasesize, (fsm->mtd.erasesize >> 10)); 2117d90db4a0SLee Jones 2118004b5e60SBrian Norris return mtd_device_register(&fsm->mtd, NULL, 0); 2119481815a6SArvind Yadav 2120481815a6SArvind Yadav err_clk_unprepare: 2121481815a6SArvind Yadav clk_disable_unprepare(fsm->clk); 2122481815a6SArvind Yadav return ret; 2123d90db4a0SLee Jones } 2124d90db4a0SLee Jones 2125d90db4a0SLee Jones static int stfsm_remove(struct platform_device *pdev) 2126d90db4a0SLee Jones { 2127d90db4a0SLee Jones struct stfsm *fsm = platform_get_drvdata(pdev); 2128d90db4a0SLee Jones 2129d9ba46d3SLee Jones return mtd_device_unregister(&fsm->mtd); 2130d90db4a0SLee Jones } 2131d90db4a0SLee Jones 213269d5af8dSLee Jones #ifdef CONFIG_PM_SLEEP 213369d5af8dSLee Jones static int stfsmfsm_suspend(struct device *dev) 213469d5af8dSLee Jones { 213569d5af8dSLee Jones struct stfsm *fsm = dev_get_drvdata(dev); 213669d5af8dSLee Jones 213769d5af8dSLee Jones clk_disable_unprepare(fsm->clk); 213869d5af8dSLee Jones 213969d5af8dSLee Jones return 0; 214069d5af8dSLee Jones } 214169d5af8dSLee Jones 214269d5af8dSLee Jones static int stfsmfsm_resume(struct device *dev) 214369d5af8dSLee Jones { 214469d5af8dSLee Jones struct stfsm *fsm = dev_get_drvdata(dev); 214569d5af8dSLee Jones 2146481815a6SArvind Yadav return clk_prepare_enable(fsm->clk); 214769d5af8dSLee Jones } 214869d5af8dSLee Jones #endif 214969d5af8dSLee Jones 215069d5af8dSLee Jones static SIMPLE_DEV_PM_OPS(stfsm_pm_ops, stfsmfsm_suspend, stfsmfsm_resume); 215169d5af8dSLee Jones 2152aeea6eb4SJingoo Han static const struct of_device_id stfsm_match[] = { 2153d90db4a0SLee Jones { .compatible = "st,spi-fsm", }, 2154d90db4a0SLee Jones {}, 2155d90db4a0SLee Jones }; 2156d90db4a0SLee Jones MODULE_DEVICE_TABLE(of, stfsm_match); 2157d90db4a0SLee Jones 2158d90db4a0SLee Jones static struct platform_driver stfsm_driver = { 2159d90db4a0SLee Jones .probe = stfsm_probe, 2160d90db4a0SLee Jones .remove = stfsm_remove, 2161d90db4a0SLee Jones .driver = { 2162d90db4a0SLee Jones .name = "st-spi-fsm", 2163d90db4a0SLee Jones .of_match_table = stfsm_match, 216469d5af8dSLee Jones .pm = &stfsm_pm_ops, 2165d90db4a0SLee Jones }, 2166d90db4a0SLee Jones }; 2167d90db4a0SLee Jones module_platform_driver(stfsm_driver); 2168d90db4a0SLee Jones 2169d90db4a0SLee Jones MODULE_AUTHOR("Angus Clark <angus.clark@st.com>"); 2170d90db4a0SLee Jones MODULE_DESCRIPTION("ST SPI FSM driver"); 2171d90db4a0SLee Jones MODULE_LICENSE("GPL"); 2172