1*54fd6939SJiyong Park /*
2*54fd6939SJiyong Park * Copyright (c) 2019-2021, STMicroelectronics - All Rights Reserved
3*54fd6939SJiyong Park *
4*54fd6939SJiyong Park * SPDX-License-Identifier: BSD-3-Clause
5*54fd6939SJiyong Park */
6*54fd6939SJiyong Park
7*54fd6939SJiyong Park #include <assert.h>
8*54fd6939SJiyong Park #include <errno.h>
9*54fd6939SJiyong Park #include <stddef.h>
10*54fd6939SJiyong Park
11*54fd6939SJiyong Park #include <platform_def.h>
12*54fd6939SJiyong Park
13*54fd6939SJiyong Park #include <common/debug.h>
14*54fd6939SJiyong Park #include <drivers/delay_timer.h>
15*54fd6939SJiyong Park #include <drivers/spi_nand.h>
16*54fd6939SJiyong Park #include <lib/utils.h>
17*54fd6939SJiyong Park
18*54fd6939SJiyong Park #define SPI_NAND_MAX_ID_LEN 4U
19*54fd6939SJiyong Park #define DELAY_US_400MS 400000U
20*54fd6939SJiyong Park #define MACRONIX_ID 0xC2U
21*54fd6939SJiyong Park
22*54fd6939SJiyong Park static struct spinand_device spinand_dev;
23*54fd6939SJiyong Park
24*54fd6939SJiyong Park #pragma weak plat_get_spi_nand_data
plat_get_spi_nand_data(struct spinand_device * device)25*54fd6939SJiyong Park int plat_get_spi_nand_data(struct spinand_device *device)
26*54fd6939SJiyong Park {
27*54fd6939SJiyong Park return 0;
28*54fd6939SJiyong Park }
29*54fd6939SJiyong Park
spi_nand_reg(bool read_reg,uint8_t reg,uint8_t * val,enum spi_mem_data_dir dir)30*54fd6939SJiyong Park static int spi_nand_reg(bool read_reg, uint8_t reg, uint8_t *val,
31*54fd6939SJiyong Park enum spi_mem_data_dir dir)
32*54fd6939SJiyong Park {
33*54fd6939SJiyong Park struct spi_mem_op op;
34*54fd6939SJiyong Park
35*54fd6939SJiyong Park zeromem(&op, sizeof(struct spi_mem_op));
36*54fd6939SJiyong Park if (read_reg) {
37*54fd6939SJiyong Park op.cmd.opcode = SPI_NAND_OP_GET_FEATURE;
38*54fd6939SJiyong Park } else {
39*54fd6939SJiyong Park op.cmd.opcode = SPI_NAND_OP_SET_FEATURE;
40*54fd6939SJiyong Park }
41*54fd6939SJiyong Park
42*54fd6939SJiyong Park op.cmd.buswidth = SPI_MEM_BUSWIDTH_1_LINE;
43*54fd6939SJiyong Park op.addr.val = reg;
44*54fd6939SJiyong Park op.addr.nbytes = 1U;
45*54fd6939SJiyong Park op.addr.buswidth = SPI_MEM_BUSWIDTH_1_LINE;
46*54fd6939SJiyong Park op.data.buswidth = SPI_MEM_BUSWIDTH_1_LINE;
47*54fd6939SJiyong Park op.data.dir = dir;
48*54fd6939SJiyong Park op.data.nbytes = 1U;
49*54fd6939SJiyong Park op.data.buf = val;
50*54fd6939SJiyong Park
51*54fd6939SJiyong Park return spi_mem_exec_op(&op);
52*54fd6939SJiyong Park }
53*54fd6939SJiyong Park
spi_nand_read_reg(uint8_t reg,uint8_t * val)54*54fd6939SJiyong Park static int spi_nand_read_reg(uint8_t reg, uint8_t *val)
55*54fd6939SJiyong Park {
56*54fd6939SJiyong Park return spi_nand_reg(true, reg, val, SPI_MEM_DATA_IN);
57*54fd6939SJiyong Park }
58*54fd6939SJiyong Park
spi_nand_write_reg(uint8_t reg,uint8_t val)59*54fd6939SJiyong Park static int spi_nand_write_reg(uint8_t reg, uint8_t val)
60*54fd6939SJiyong Park {
61*54fd6939SJiyong Park return spi_nand_reg(false, reg, &val, SPI_MEM_DATA_OUT);
62*54fd6939SJiyong Park }
63*54fd6939SJiyong Park
spi_nand_update_cfg(uint8_t mask,uint8_t val)64*54fd6939SJiyong Park static int spi_nand_update_cfg(uint8_t mask, uint8_t val)
65*54fd6939SJiyong Park {
66*54fd6939SJiyong Park int ret;
67*54fd6939SJiyong Park uint8_t cfg = spinand_dev.cfg_cache;
68*54fd6939SJiyong Park
69*54fd6939SJiyong Park cfg &= ~mask;
70*54fd6939SJiyong Park cfg |= val;
71*54fd6939SJiyong Park
72*54fd6939SJiyong Park if (cfg == spinand_dev.cfg_cache) {
73*54fd6939SJiyong Park return 0;
74*54fd6939SJiyong Park }
75*54fd6939SJiyong Park
76*54fd6939SJiyong Park ret = spi_nand_write_reg(SPI_NAND_REG_CFG, cfg);
77*54fd6939SJiyong Park if (ret == 0) {
78*54fd6939SJiyong Park spinand_dev.cfg_cache = cfg;
79*54fd6939SJiyong Park }
80*54fd6939SJiyong Park
81*54fd6939SJiyong Park return ret;
82*54fd6939SJiyong Park }
83*54fd6939SJiyong Park
spi_nand_ecc_enable(bool enable)84*54fd6939SJiyong Park static int spi_nand_ecc_enable(bool enable)
85*54fd6939SJiyong Park {
86*54fd6939SJiyong Park return spi_nand_update_cfg(SPI_NAND_CFG_ECC_EN,
87*54fd6939SJiyong Park enable ? SPI_NAND_CFG_ECC_EN : 0U);
88*54fd6939SJiyong Park }
89*54fd6939SJiyong Park
spi_nand_quad_enable(uint8_t manufacturer_id)90*54fd6939SJiyong Park static int spi_nand_quad_enable(uint8_t manufacturer_id)
91*54fd6939SJiyong Park {
92*54fd6939SJiyong Park bool enable = false;
93*54fd6939SJiyong Park
94*54fd6939SJiyong Park if (manufacturer_id != MACRONIX_ID) {
95*54fd6939SJiyong Park return 0;
96*54fd6939SJiyong Park }
97*54fd6939SJiyong Park
98*54fd6939SJiyong Park if (spinand_dev.spi_read_cache_op.data.buswidth ==
99*54fd6939SJiyong Park SPI_MEM_BUSWIDTH_4_LINE) {
100*54fd6939SJiyong Park enable = true;
101*54fd6939SJiyong Park }
102*54fd6939SJiyong Park
103*54fd6939SJiyong Park return spi_nand_update_cfg(SPI_NAND_CFG_QE,
104*54fd6939SJiyong Park enable ? SPI_NAND_CFG_QE : 0U);
105*54fd6939SJiyong Park }
106*54fd6939SJiyong Park
spi_nand_wait_ready(uint8_t * status)107*54fd6939SJiyong Park static int spi_nand_wait_ready(uint8_t *status)
108*54fd6939SJiyong Park {
109*54fd6939SJiyong Park int ret;
110*54fd6939SJiyong Park uint64_t timeout = timeout_init_us(DELAY_US_400MS);
111*54fd6939SJiyong Park
112*54fd6939SJiyong Park while (!timeout_elapsed(timeout)) {
113*54fd6939SJiyong Park ret = spi_nand_read_reg(SPI_NAND_REG_STATUS, status);
114*54fd6939SJiyong Park if (ret != 0) {
115*54fd6939SJiyong Park return ret;
116*54fd6939SJiyong Park }
117*54fd6939SJiyong Park
118*54fd6939SJiyong Park VERBOSE("%s Status %x\n", __func__, *status);
119*54fd6939SJiyong Park if ((*status & SPI_NAND_STATUS_BUSY) == 0U) {
120*54fd6939SJiyong Park return 0;
121*54fd6939SJiyong Park }
122*54fd6939SJiyong Park }
123*54fd6939SJiyong Park
124*54fd6939SJiyong Park return -ETIMEDOUT;
125*54fd6939SJiyong Park }
126*54fd6939SJiyong Park
spi_nand_reset(void)127*54fd6939SJiyong Park static int spi_nand_reset(void)
128*54fd6939SJiyong Park {
129*54fd6939SJiyong Park struct spi_mem_op op;
130*54fd6939SJiyong Park uint8_t status;
131*54fd6939SJiyong Park int ret;
132*54fd6939SJiyong Park
133*54fd6939SJiyong Park zeromem(&op, sizeof(struct spi_mem_op));
134*54fd6939SJiyong Park op.cmd.opcode = SPI_NAND_OP_RESET;
135*54fd6939SJiyong Park op.cmd.buswidth = SPI_MEM_BUSWIDTH_1_LINE;
136*54fd6939SJiyong Park
137*54fd6939SJiyong Park ret = spi_mem_exec_op(&op);
138*54fd6939SJiyong Park if (ret != 0) {
139*54fd6939SJiyong Park return ret;
140*54fd6939SJiyong Park }
141*54fd6939SJiyong Park
142*54fd6939SJiyong Park return spi_nand_wait_ready(&status);
143*54fd6939SJiyong Park }
144*54fd6939SJiyong Park
spi_nand_read_id(uint8_t * id)145*54fd6939SJiyong Park static int spi_nand_read_id(uint8_t *id)
146*54fd6939SJiyong Park {
147*54fd6939SJiyong Park struct spi_mem_op op;
148*54fd6939SJiyong Park
149*54fd6939SJiyong Park zeromem(&op, sizeof(struct spi_mem_op));
150*54fd6939SJiyong Park op.cmd.opcode = SPI_NAND_OP_READ_ID;
151*54fd6939SJiyong Park op.cmd.buswidth = SPI_MEM_BUSWIDTH_1_LINE;
152*54fd6939SJiyong Park op.data.dir = SPI_MEM_DATA_IN;
153*54fd6939SJiyong Park op.data.nbytes = SPI_NAND_MAX_ID_LEN;
154*54fd6939SJiyong Park op.data.buf = id;
155*54fd6939SJiyong Park op.data.buswidth = SPI_MEM_BUSWIDTH_1_LINE;
156*54fd6939SJiyong Park
157*54fd6939SJiyong Park return spi_mem_exec_op(&op);
158*54fd6939SJiyong Park }
159*54fd6939SJiyong Park
spi_nand_load_page(unsigned int page)160*54fd6939SJiyong Park static int spi_nand_load_page(unsigned int page)
161*54fd6939SJiyong Park {
162*54fd6939SJiyong Park struct spi_mem_op op;
163*54fd6939SJiyong Park uint32_t block_nb = page / spinand_dev.nand_dev->block_size;
164*54fd6939SJiyong Park uint32_t page_nb = page - (block_nb * spinand_dev.nand_dev->page_size);
165*54fd6939SJiyong Park uint32_t nbpages_per_block = spinand_dev.nand_dev->block_size /
166*54fd6939SJiyong Park spinand_dev.nand_dev->page_size;
167*54fd6939SJiyong Park uint32_t block_sh = __builtin_ctz(nbpages_per_block) + 1U;
168*54fd6939SJiyong Park
169*54fd6939SJiyong Park zeromem(&op, sizeof(struct spi_mem_op));
170*54fd6939SJiyong Park op.cmd.opcode = SPI_NAND_OP_LOAD_PAGE;
171*54fd6939SJiyong Park op.cmd.buswidth = SPI_MEM_BUSWIDTH_1_LINE;
172*54fd6939SJiyong Park op.addr.val = (block_nb << block_sh) | page_nb;
173*54fd6939SJiyong Park op.addr.nbytes = 3U;
174*54fd6939SJiyong Park op.addr.buswidth = SPI_MEM_BUSWIDTH_1_LINE;
175*54fd6939SJiyong Park
176*54fd6939SJiyong Park return spi_mem_exec_op(&op);
177*54fd6939SJiyong Park }
178*54fd6939SJiyong Park
spi_nand_read_from_cache(unsigned int page,unsigned int offset,uint8_t * buffer,unsigned int len)179*54fd6939SJiyong Park static int spi_nand_read_from_cache(unsigned int page, unsigned int offset,
180*54fd6939SJiyong Park uint8_t *buffer, unsigned int len)
181*54fd6939SJiyong Park {
182*54fd6939SJiyong Park uint32_t nbpages_per_block = spinand_dev.nand_dev->block_size /
183*54fd6939SJiyong Park spinand_dev.nand_dev->page_size;
184*54fd6939SJiyong Park uint32_t block_nb = page / nbpages_per_block;
185*54fd6939SJiyong Park uint32_t page_sh = __builtin_ctz(spinand_dev.nand_dev->page_size) + 1U;
186*54fd6939SJiyong Park
187*54fd6939SJiyong Park spinand_dev.spi_read_cache_op.addr.val = offset;
188*54fd6939SJiyong Park
189*54fd6939SJiyong Park if ((spinand_dev.nand_dev->nb_planes > 1U) && ((block_nb % 2U) == 1U)) {
190*54fd6939SJiyong Park spinand_dev.spi_read_cache_op.addr.val |= 1U << page_sh;
191*54fd6939SJiyong Park }
192*54fd6939SJiyong Park
193*54fd6939SJiyong Park spinand_dev.spi_read_cache_op.data.buf = buffer;
194*54fd6939SJiyong Park spinand_dev.spi_read_cache_op.data.nbytes = len;
195*54fd6939SJiyong Park
196*54fd6939SJiyong Park return spi_mem_exec_op(&spinand_dev.spi_read_cache_op);
197*54fd6939SJiyong Park }
198*54fd6939SJiyong Park
spi_nand_read_page(unsigned int page,unsigned int offset,uint8_t * buffer,unsigned int len,bool ecc_enabled)199*54fd6939SJiyong Park static int spi_nand_read_page(unsigned int page, unsigned int offset,
200*54fd6939SJiyong Park uint8_t *buffer, unsigned int len,
201*54fd6939SJiyong Park bool ecc_enabled)
202*54fd6939SJiyong Park {
203*54fd6939SJiyong Park uint8_t status;
204*54fd6939SJiyong Park int ret;
205*54fd6939SJiyong Park
206*54fd6939SJiyong Park ret = spi_nand_ecc_enable(ecc_enabled);
207*54fd6939SJiyong Park if (ret != 0) {
208*54fd6939SJiyong Park return ret;
209*54fd6939SJiyong Park }
210*54fd6939SJiyong Park
211*54fd6939SJiyong Park ret = spi_nand_load_page(page);
212*54fd6939SJiyong Park if (ret != 0) {
213*54fd6939SJiyong Park return ret;
214*54fd6939SJiyong Park }
215*54fd6939SJiyong Park
216*54fd6939SJiyong Park ret = spi_nand_wait_ready(&status);
217*54fd6939SJiyong Park if (ret != 0) {
218*54fd6939SJiyong Park return ret;
219*54fd6939SJiyong Park }
220*54fd6939SJiyong Park
221*54fd6939SJiyong Park ret = spi_nand_read_from_cache(page, offset, buffer, len);
222*54fd6939SJiyong Park if (ret != 0) {
223*54fd6939SJiyong Park return ret;
224*54fd6939SJiyong Park }
225*54fd6939SJiyong Park
226*54fd6939SJiyong Park if (ecc_enabled && ((status & SPI_NAND_STATUS_ECC_UNCOR) != 0U)) {
227*54fd6939SJiyong Park return -EBADMSG;
228*54fd6939SJiyong Park }
229*54fd6939SJiyong Park
230*54fd6939SJiyong Park return 0;
231*54fd6939SJiyong Park }
232*54fd6939SJiyong Park
spi_nand_mtd_block_is_bad(unsigned int block)233*54fd6939SJiyong Park static int spi_nand_mtd_block_is_bad(unsigned int block)
234*54fd6939SJiyong Park {
235*54fd6939SJiyong Park unsigned int nbpages_per_block = spinand_dev.nand_dev->block_size /
236*54fd6939SJiyong Park spinand_dev.nand_dev->page_size;
237*54fd6939SJiyong Park uint8_t bbm_marker[2];
238*54fd6939SJiyong Park int ret;
239*54fd6939SJiyong Park
240*54fd6939SJiyong Park ret = spi_nand_read_page(block * nbpages_per_block,
241*54fd6939SJiyong Park spinand_dev.nand_dev->page_size,
242*54fd6939SJiyong Park bbm_marker, sizeof(bbm_marker), false);
243*54fd6939SJiyong Park if (ret != 0) {
244*54fd6939SJiyong Park return ret;
245*54fd6939SJiyong Park }
246*54fd6939SJiyong Park
247*54fd6939SJiyong Park if ((bbm_marker[0] != GENMASK_32(7, 0)) ||
248*54fd6939SJiyong Park (bbm_marker[1] != GENMASK_32(7, 0))) {
249*54fd6939SJiyong Park WARN("Block %i is bad\n", block);
250*54fd6939SJiyong Park return 1;
251*54fd6939SJiyong Park }
252*54fd6939SJiyong Park
253*54fd6939SJiyong Park return 0;
254*54fd6939SJiyong Park }
255*54fd6939SJiyong Park
spi_nand_mtd_read_page(struct nand_device * nand,unsigned int page,uintptr_t buffer)256*54fd6939SJiyong Park static int spi_nand_mtd_read_page(struct nand_device *nand, unsigned int page,
257*54fd6939SJiyong Park uintptr_t buffer)
258*54fd6939SJiyong Park {
259*54fd6939SJiyong Park return spi_nand_read_page(page, 0, (uint8_t *)buffer,
260*54fd6939SJiyong Park spinand_dev.nand_dev->page_size, true);
261*54fd6939SJiyong Park }
262*54fd6939SJiyong Park
spi_nand_init(unsigned long long * size,unsigned int * erase_size)263*54fd6939SJiyong Park int spi_nand_init(unsigned long long *size, unsigned int *erase_size)
264*54fd6939SJiyong Park {
265*54fd6939SJiyong Park uint8_t id[SPI_NAND_MAX_ID_LEN];
266*54fd6939SJiyong Park int ret;
267*54fd6939SJiyong Park
268*54fd6939SJiyong Park spinand_dev.nand_dev = get_nand_device();
269*54fd6939SJiyong Park if (spinand_dev.nand_dev == NULL) {
270*54fd6939SJiyong Park return -EINVAL;
271*54fd6939SJiyong Park }
272*54fd6939SJiyong Park
273*54fd6939SJiyong Park spinand_dev.nand_dev->mtd_block_is_bad = spi_nand_mtd_block_is_bad;
274*54fd6939SJiyong Park spinand_dev.nand_dev->mtd_read_page = spi_nand_mtd_read_page;
275*54fd6939SJiyong Park spinand_dev.nand_dev->nb_planes = 1;
276*54fd6939SJiyong Park
277*54fd6939SJiyong Park spinand_dev.spi_read_cache_op.cmd.opcode = SPI_NAND_OP_READ_FROM_CACHE;
278*54fd6939SJiyong Park spinand_dev.spi_read_cache_op.cmd.buswidth = SPI_MEM_BUSWIDTH_1_LINE;
279*54fd6939SJiyong Park spinand_dev.spi_read_cache_op.addr.nbytes = 2U;
280*54fd6939SJiyong Park spinand_dev.spi_read_cache_op.addr.buswidth = SPI_MEM_BUSWIDTH_1_LINE;
281*54fd6939SJiyong Park spinand_dev.spi_read_cache_op.dummy.nbytes = 1U;
282*54fd6939SJiyong Park spinand_dev.spi_read_cache_op.dummy.buswidth = SPI_MEM_BUSWIDTH_1_LINE;
283*54fd6939SJiyong Park spinand_dev.spi_read_cache_op.data.buswidth = SPI_MEM_BUSWIDTH_1_LINE;
284*54fd6939SJiyong Park
285*54fd6939SJiyong Park if (plat_get_spi_nand_data(&spinand_dev) != 0) {
286*54fd6939SJiyong Park return -EINVAL;
287*54fd6939SJiyong Park }
288*54fd6939SJiyong Park
289*54fd6939SJiyong Park assert((spinand_dev.nand_dev->page_size != 0U) &&
290*54fd6939SJiyong Park (spinand_dev.nand_dev->block_size != 0U) &&
291*54fd6939SJiyong Park (spinand_dev.nand_dev->size != 0U));
292*54fd6939SJiyong Park
293*54fd6939SJiyong Park ret = spi_nand_reset();
294*54fd6939SJiyong Park if (ret != 0) {
295*54fd6939SJiyong Park return ret;
296*54fd6939SJiyong Park }
297*54fd6939SJiyong Park
298*54fd6939SJiyong Park ret = spi_nand_read_id(id);
299*54fd6939SJiyong Park if (ret != 0) {
300*54fd6939SJiyong Park return ret;
301*54fd6939SJiyong Park }
302*54fd6939SJiyong Park
303*54fd6939SJiyong Park ret = spi_nand_read_reg(SPI_NAND_REG_CFG, &spinand_dev.cfg_cache);
304*54fd6939SJiyong Park if (ret != 0) {
305*54fd6939SJiyong Park return ret;
306*54fd6939SJiyong Park }
307*54fd6939SJiyong Park
308*54fd6939SJiyong Park ret = spi_nand_quad_enable(id[1]);
309*54fd6939SJiyong Park if (ret != 0) {
310*54fd6939SJiyong Park return ret;
311*54fd6939SJiyong Park }
312*54fd6939SJiyong Park
313*54fd6939SJiyong Park VERBOSE("SPI_NAND Detected ID 0x%x\n", id[1]);
314*54fd6939SJiyong Park
315*54fd6939SJiyong Park VERBOSE("Page size %i, Block size %i, size %lli\n",
316*54fd6939SJiyong Park spinand_dev.nand_dev->page_size,
317*54fd6939SJiyong Park spinand_dev.nand_dev->block_size,
318*54fd6939SJiyong Park spinand_dev.nand_dev->size);
319*54fd6939SJiyong Park
320*54fd6939SJiyong Park *size = spinand_dev.nand_dev->size;
321*54fd6939SJiyong Park *erase_size = spinand_dev.nand_dev->block_size;
322*54fd6939SJiyong Park
323*54fd6939SJiyong Park return 0;
324*54fd6939SJiyong Park }
325