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