blob: 2cd3f64dc63c9c7cfcf0e6b4afa07bb5080a8e27 [file] [log] [blame]
Tom Rini10e47792018-05-06 17:58:06 -04001// SPDX-License-Identifier: GPL-2.0+
Siva Durga Prasad Paladuguc7b0bc72015-11-17 14:30:10 +05302/*
3 * Arasan NAND Flash Controller Driver
4 *
5 * Copyright (C) 2014 - 2015 Xilinx, Inc.
Siva Durga Prasad Paladuguc7b0bc72015-11-17 14:30:10 +05306 */
7
8#include <common.h>
9#include <malloc.h>
10#include <asm/io.h>
Masahiro Yamada56a931c2016-09-21 11:28:55 +090011#include <linux/errno.h>
Siva Durga Prasad Paladuguc7b0bc72015-11-17 14:30:10 +053012#include <linux/mtd/mtd.h>
Masahiro Yamada2b7a8732017-11-30 13:45:24 +090013#include <linux/mtd/rawnand.h>
Siva Durga Prasad Paladuguc7b0bc72015-11-17 14:30:10 +053014#include <linux/mtd/partitions.h>
15#include <linux/mtd/nand_ecc.h>
16#include <asm/arch/hardware.h>
17#include <asm/arch/sys_proto.h>
18#include <nand.h>
19
20struct arasan_nand_info {
21 void __iomem *nand_base;
22 u32 page;
Siva Durga Prasad Paladugu45e512c2018-01-04 16:04:21 +053023 bool on_die_ecc_enabled;
Siva Durga Prasad Paladuguc7b0bc72015-11-17 14:30:10 +053024};
25
26struct nand_regs {
27 u32 pkt_reg;
28 u32 memadr_reg1;
29 u32 memadr_reg2;
30 u32 cmd_reg;
31 u32 pgm_reg;
32 u32 intsts_enr;
33 u32 intsig_enr;
34 u32 intsts_reg;
35 u32 rdy_busy;
36 u32 cms_sysadr_reg;
37 u32 flash_sts_reg;
38 u32 tmg_reg;
39 u32 buf_dataport;
40 u32 ecc_reg;
41 u32 ecc_errcnt_reg;
42 u32 ecc_sprcmd_reg;
43 u32 errcnt_1bitreg;
44 u32 errcnt_2bitreg;
45 u32 errcnt_3bitreg;
46 u32 errcnt_4bitreg;
47 u32 dma_sysadr0_reg;
48 u32 dma_bufbdry_reg;
49 u32 cpu_rls_reg;
50 u32 errcnt_5bitreg;
51 u32 errcnt_6bitreg;
52 u32 errcnt_7bitreg;
53 u32 errcnt_8bitreg;
54 u32 data_if_reg;
55};
56
57#define arasan_nand_base ((struct nand_regs __iomem *)ARASAN_NAND_BASEADDR)
58
59struct arasan_nand_command_format {
60 u8 cmd1;
61 u8 cmd2;
62 u8 addr_cycles;
63 u32 pgm;
64};
65
66#define ONDIE_ECC_FEATURE_ADDR 0x90
Siva Durga Prasad Paladugu45e512c2018-01-04 16:04:21 +053067#define ENABLE_ONDIE_ECC 0x08
Siva Durga Prasad Paladuguc7b0bc72015-11-17 14:30:10 +053068
69#define ARASAN_PROG_RD_MASK 0x00000001
70#define ARASAN_PROG_BLK_ERS_MASK 0x00000004
71#define ARASAN_PROG_RD_ID_MASK 0x00000040
72#define ARASAN_PROG_RD_STS_MASK 0x00000008
73#define ARASAN_PROG_PG_PROG_MASK 0x00000010
74#define ARASAN_PROG_RD_PARAM_PG_MASK 0x00000080
75#define ARASAN_PROG_RST_MASK 0x00000100
76#define ARASAN_PROG_GET_FTRS_MASK 0x00000200
77#define ARASAN_PROG_SET_FTRS_MASK 0x00000400
78#define ARASAN_PROG_CHNG_ROWADR_END_MASK 0x00400000
79
80#define ARASAN_NAND_CMD_ECC_ON_MASK 0x80000000
81#define ARASAN_NAND_CMD_CMD12_MASK 0xFFFF
82#define ARASAN_NAND_CMD_PG_SIZE_MASK 0x3800000
83#define ARASAN_NAND_CMD_PG_SIZE_SHIFT 23
84#define ARASAN_NAND_CMD_CMD2_SHIFT 8
85#define ARASAN_NAND_CMD_ADDR_CYCL_MASK 0x70000000
86#define ARASAN_NAND_CMD_ADDR_CYCL_SHIFT 28
87
Vipul Kumar9d9b99b2018-03-10 17:52:23 +053088#define ARASAN_NAND_MEM_ADDR1_PAGE_MASK 0xFFFF0000
Siva Durga Prasad Paladuguc7b0bc72015-11-17 14:30:10 +053089#define ARASAN_NAND_MEM_ADDR1_COL_MASK 0xFFFF
90#define ARASAN_NAND_MEM_ADDR1_PAGE_SHIFT 16
91#define ARASAN_NAND_MEM_ADDR2_PAGE_MASK 0xFF
92#define ARASAN_NAND_MEM_ADDR2_CS_MASK 0xC0000000
T Karthik Reddy7cd85222018-12-03 16:11:58 +053093#define ARASAN_NAND_MEM_ADDR2_CS0_MASK (0x3 << 30)
94#define ARASAN_NAND_MEM_ADDR2_CS1_MASK (0x1 << 30)
Siva Durga Prasad Paladuguc7b0bc72015-11-17 14:30:10 +053095#define ARASAN_NAND_MEM_ADDR2_BCH_MASK 0xE000000
96#define ARASAN_NAND_MEM_ADDR2_BCH_SHIFT 25
97
98#define ARASAN_NAND_INT_STS_ERR_EN_MASK 0x10
99#define ARASAN_NAND_INT_STS_MUL_BIT_ERR_MASK 0x08
100#define ARASAN_NAND_INT_STS_BUF_RD_RDY_MASK 0x02
101#define ARASAN_NAND_INT_STS_BUF_WR_RDY_MASK 0x01
102#define ARASAN_NAND_INT_STS_XFR_CMPLT_MASK 0x04
103
104#define ARASAN_NAND_PKT_REG_PKT_CNT_MASK 0xFFF000
105#define ARASAN_NAND_PKT_REG_PKT_SIZE_MASK 0x7FF
106#define ARASAN_NAND_PKT_REG_PKT_CNT_SHFT 12
107
108#define ARASAN_NAND_ROW_ADDR_CYCL_MASK 0x0F
109#define ARASAN_NAND_COL_ADDR_CYCL_MASK 0xF0
110#define ARASAN_NAND_COL_ADDR_CYCL_SHIFT 4
111
112#define ARASAN_NAND_ECC_SIZE_SHIFT 16
113#define ARASAN_NAND_ECC_BCH_SHIFT 27
114
115#define ARASAN_NAND_PKTSIZE_1K 1024
116#define ARASAN_NAND_PKTSIZE_512 512
117
118#define ARASAN_NAND_POLL_TIMEOUT 1000000
119#define ARASAN_NAND_INVALID_ADDR_CYCL 0xFF
120
121#define ERR_ADDR_CYCLE -1
122#define READ_BUFF_SIZE 0x4000
123
124static struct arasan_nand_command_format *curr_cmd;
125
126enum addr_cycles {
127 NAND_ADDR_CYCL_NONE,
128 NAND_ADDR_CYCL_ONE,
129 NAND_ADDR_CYCL_ROW,
130 NAND_ADDR_CYCL_COL,
131 NAND_ADDR_CYCL_BOTH,
132};
133
134static struct arasan_nand_command_format arasan_nand_commands[] = {
135 {NAND_CMD_READ0, NAND_CMD_READSTART, NAND_ADDR_CYCL_BOTH,
136 ARASAN_PROG_RD_MASK},
137 {NAND_CMD_RNDOUT, NAND_CMD_RNDOUTSTART, NAND_ADDR_CYCL_COL,
138 ARASAN_PROG_RD_MASK},
139 {NAND_CMD_READID, NAND_CMD_NONE, NAND_ADDR_CYCL_ONE,
140 ARASAN_PROG_RD_ID_MASK},
141 {NAND_CMD_STATUS, NAND_CMD_NONE, NAND_ADDR_CYCL_NONE,
142 ARASAN_PROG_RD_STS_MASK},
143 {NAND_CMD_SEQIN, NAND_CMD_PAGEPROG, NAND_ADDR_CYCL_BOTH,
144 ARASAN_PROG_PG_PROG_MASK},
145 {NAND_CMD_RNDIN, NAND_CMD_NONE, NAND_ADDR_CYCL_COL,
146 ARASAN_PROG_CHNG_ROWADR_END_MASK},
147 {NAND_CMD_ERASE1, NAND_CMD_ERASE2, NAND_ADDR_CYCL_ROW,
148 ARASAN_PROG_BLK_ERS_MASK},
149 {NAND_CMD_RESET, NAND_CMD_NONE, NAND_ADDR_CYCL_NONE,
150 ARASAN_PROG_RST_MASK},
151 {NAND_CMD_PARAM, NAND_CMD_NONE, NAND_ADDR_CYCL_ONE,
152 ARASAN_PROG_RD_PARAM_PG_MASK},
153 {NAND_CMD_GET_FEATURES, NAND_CMD_NONE, NAND_ADDR_CYCL_ONE,
154 ARASAN_PROG_GET_FTRS_MASK},
155 {NAND_CMD_SET_FEATURES, NAND_CMD_NONE, NAND_ADDR_CYCL_ONE,
156 ARASAN_PROG_SET_FTRS_MASK},
157 {NAND_CMD_NONE, NAND_CMD_NONE, NAND_ADDR_CYCL_NONE, 0},
158};
159
160struct arasan_ecc_matrix {
161 u32 pagesize;
162 u32 ecc_codeword_size;
163 u8 eccbits;
164 u8 bch;
165 u8 bchval;
166 u16 eccaddr;
167 u16 eccsize;
168};
169
170static const struct arasan_ecc_matrix ecc_matrix[] = {
171 {512, 512, 1, 0, 0, 0x20D, 0x3},
172 {512, 512, 4, 1, 3, 0x209, 0x7},
173 {512, 512, 8, 1, 2, 0x203, 0xD},
174 /*
175 * 2K byte page
176 */
177 {2048, 512, 1, 0, 0, 0x834, 0xC},
178 {2048, 512, 4, 1, 3, 0x826, 0x1A},
179 {2048, 512, 8, 1, 2, 0x80c, 0x34},
180 {2048, 512, 12, 1, 1, 0x822, 0x4E},
181 {2048, 512, 16, 1, 0, 0x808, 0x68},
182 {2048, 1024, 24, 1, 4, 0x81c, 0x54},
183 /*
184 * 4K byte page
185 */
186 {4096, 512, 1, 0, 0, 0x1068, 0x18},
187 {4096, 512, 4, 1, 3, 0x104c, 0x34},
188 {4096, 512, 8, 1, 2, 0x1018, 0x68},
189 {4096, 512, 12, 1, 1, 0x1044, 0x9C},
190 {4096, 512, 16, 1, 0, 0x1010, 0xD0},
191 {4096, 1024, 24, 1, 4, 0x1038, 0xA8},
192 /*
193 * 8K byte page
194 */
195 {8192, 512, 1, 0, 0, 0x20d0, 0x30},
196 {8192, 512, 4, 1, 3, 0x2098, 0x68},
197 {8192, 512, 8, 1, 2, 0x2030, 0xD0},
198 {8192, 512, 12, 1, 1, 0x2088, 0x138},
199 {8192, 512, 16, 1, 0, 0x2020, 0x1A0},
200 {8192, 1024, 24, 1, 4, 0x2070, 0x150},
201 /*
202 * 16K byte page
203 */
204 {16384, 512, 1, 0, 0, 0x4460, 0x60},
205 {16384, 512, 4, 1, 3, 0x43f0, 0xD0},
206 {16384, 512, 8, 1, 2, 0x4320, 0x1A0},
207 {16384, 512, 12, 1, 1, 0x4250, 0x270},
208 {16384, 512, 16, 1, 0, 0x4180, 0x340},
209 {16384, 1024, 24, 1, 4, 0x4220, 0x2A0}
210};
211
Siva Durga Prasad Paladugu45e512c2018-01-04 16:04:21 +0530212static struct nand_ecclayout ondie_nand_oob_64 = {
213 .eccbytes = 32,
214
215 .eccpos = {
216 8, 9, 10, 11, 12, 13, 14, 15,
217 24, 25, 26, 27, 28, 29, 30, 31,
218 40, 41, 42, 43, 44, 45, 46, 47,
219 56, 57, 58, 59, 60, 61, 62, 63
220 },
221
222 .oobfree = {
223 { .offset = 4, .length = 4 },
224 { .offset = 20, .length = 4 },
225 { .offset = 36, .length = 4 },
226 { .offset = 52, .length = 4 }
227 }
228};
229
230/*
231 * bbt decriptors for chips with on-die ECC and
232 * chips with 64-byte OOB
233 */
234static u8 bbt_pattern[] = {'B', 'b', 't', '0' };
235static u8 mirror_pattern[] = {'1', 't', 'b', 'B' };
236
237static struct nand_bbt_descr bbt_main_descr = {
238 .options = NAND_BBT_LASTBLOCK | NAND_BBT_CREATE | NAND_BBT_WRITE |
239 NAND_BBT_2BIT | NAND_BBT_VERSION | NAND_BBT_PERCHIP,
240 .offs = 4,
241 .len = 4,
242 .veroffs = 20,
243 .maxblocks = 4,
244 .pattern = bbt_pattern
245};
246
247static struct nand_bbt_descr bbt_mirror_descr = {
248 .options = NAND_BBT_LASTBLOCK | NAND_BBT_CREATE | NAND_BBT_WRITE |
249 NAND_BBT_2BIT | NAND_BBT_VERSION | NAND_BBT_PERCHIP,
250 .offs = 4,
251 .len = 4,
252 .veroffs = 20,
253 .maxblocks = 4,
254 .pattern = mirror_pattern
255};
256
Siva Durga Prasad Paladuguc7b0bc72015-11-17 14:30:10 +0530257static u8 buf_data[READ_BUFF_SIZE];
258static u32 buf_index;
259
260static struct nand_ecclayout nand_oob;
261
262static struct nand_chip nand_chip[CONFIG_SYS_MAX_NAND_DEVICE];
263
264static void arasan_nand_select_chip(struct mtd_info *mtd, int chip)
265{
T Karthik Reddy7cd85222018-12-03 16:11:58 +0530266 u32 reg_val;
267
268 reg_val = readl(&arasan_nand_base->memadr_reg2);
269 if (chip == 0) {
270 reg_val &= ~ARASAN_NAND_MEM_ADDR2_CS0_MASK;
271 writel(reg_val, &arasan_nand_base->memadr_reg2);
272 } else if (chip == 1) {
273 reg_val |= ARASAN_NAND_MEM_ADDR2_CS1_MASK;
274 writel(reg_val, &arasan_nand_base->memadr_reg2);
275 }
Siva Durga Prasad Paladuguc7b0bc72015-11-17 14:30:10 +0530276}
277
278static void arasan_nand_enable_ecc(void)
279{
280 u32 reg_val;
281
282 reg_val = readl(&arasan_nand_base->cmd_reg);
283 reg_val |= ARASAN_NAND_CMD_ECC_ON_MASK;
284
285 writel(reg_val, &arasan_nand_base->cmd_reg);
286}
287
288static u8 arasan_nand_get_addrcycle(struct mtd_info *mtd)
289{
290 u8 addrcycles;
Scott Wood17fed142016-05-30 13:57:56 -0500291 struct nand_chip *chip = mtd_to_nand(mtd);
Siva Durga Prasad Paladuguc7b0bc72015-11-17 14:30:10 +0530292
293 switch (curr_cmd->addr_cycles) {
294 case NAND_ADDR_CYCL_NONE:
295 addrcycles = 0;
296 break;
297 case NAND_ADDR_CYCL_ONE:
298 addrcycles = 1;
299 break;
300 case NAND_ADDR_CYCL_ROW:
301 addrcycles = chip->onfi_params.addr_cycles &
302 ARASAN_NAND_ROW_ADDR_CYCL_MASK;
303 break;
304 case NAND_ADDR_CYCL_COL:
305 addrcycles = (chip->onfi_params.addr_cycles &
306 ARASAN_NAND_COL_ADDR_CYCL_MASK) >>
307 ARASAN_NAND_COL_ADDR_CYCL_SHIFT;
308 break;
309 case NAND_ADDR_CYCL_BOTH:
310 addrcycles = chip->onfi_params.addr_cycles &
311 ARASAN_NAND_ROW_ADDR_CYCL_MASK;
312 addrcycles += (chip->onfi_params.addr_cycles &
313 ARASAN_NAND_COL_ADDR_CYCL_MASK) >>
314 ARASAN_NAND_COL_ADDR_CYCL_SHIFT;
315 break;
316 default:
317 addrcycles = ARASAN_NAND_INVALID_ADDR_CYCL;
318 break;
319 }
320 return addrcycles;
321}
322
323static int arasan_nand_read_page(struct mtd_info *mtd, u8 *buf, u32 size)
324{
Scott Wood17fed142016-05-30 13:57:56 -0500325 struct nand_chip *chip = mtd_to_nand(mtd);
Siva Durga Prasad Paladugu45e512c2018-01-04 16:04:21 +0530326 struct arasan_nand_info *nand = nand_get_controller_data(chip);
Siva Durga Prasad Paladuguc7b0bc72015-11-17 14:30:10 +0530327 u32 reg_val, i, pktsize, pktnum;
328 u32 *bufptr = (u32 *)buf;
329 u32 timeout;
330 u32 rdcount = 0;
331 u8 addr_cycles;
332
333 if (chip->ecc_step_ds >= ARASAN_NAND_PKTSIZE_1K)
334 pktsize = ARASAN_NAND_PKTSIZE_1K;
335 else
336 pktsize = ARASAN_NAND_PKTSIZE_512;
337
338 if (size % pktsize)
339 pktnum = size/pktsize + 1;
340 else
341 pktnum = size/pktsize;
342
343 reg_val = readl(&arasan_nand_base->intsts_enr);
344 reg_val |= ARASAN_NAND_INT_STS_ERR_EN_MASK |
345 ARASAN_NAND_INT_STS_MUL_BIT_ERR_MASK;
346 writel(reg_val, &arasan_nand_base->intsts_enr);
347
348 reg_val = readl(&arasan_nand_base->pkt_reg);
349 reg_val &= ~(ARASAN_NAND_PKT_REG_PKT_CNT_MASK |
350 ARASAN_NAND_PKT_REG_PKT_SIZE_MASK);
351 reg_val |= (pktnum << ARASAN_NAND_PKT_REG_PKT_CNT_SHFT) |
352 pktsize;
353 writel(reg_val, &arasan_nand_base->pkt_reg);
354
Siva Durga Prasad Paladugu45e512c2018-01-04 16:04:21 +0530355 if (!nand->on_die_ecc_enabled) {
356 arasan_nand_enable_ecc();
357 addr_cycles = arasan_nand_get_addrcycle(mtd);
358 if (addr_cycles == ARASAN_NAND_INVALID_ADDR_CYCL)
359 return ERR_ADDR_CYCLE;
Siva Durga Prasad Paladuguc7b0bc72015-11-17 14:30:10 +0530360
Siva Durga Prasad Paladugu45e512c2018-01-04 16:04:21 +0530361 writel((NAND_CMD_RNDOUTSTART << ARASAN_NAND_CMD_CMD2_SHIFT) |
362 NAND_CMD_RNDOUT | (addr_cycles <<
363 ARASAN_NAND_CMD_ADDR_CYCL_SHIFT),
364 &arasan_nand_base->ecc_sprcmd_reg);
365 }
Siva Durga Prasad Paladuguc7b0bc72015-11-17 14:30:10 +0530366 writel(curr_cmd->pgm, &arasan_nand_base->pgm_reg);
367
368 while (rdcount < pktnum) {
369 timeout = ARASAN_NAND_POLL_TIMEOUT;
370 while (!(readl(&arasan_nand_base->intsts_reg) &
371 ARASAN_NAND_INT_STS_BUF_RD_RDY_MASK) && timeout) {
372 udelay(1);
373 timeout--;
374 }
375 if (!timeout) {
376 puts("arasan_read_page: timedout:Buff RDY\n");
377 return -ETIMEDOUT;
378 }
379
380 rdcount++;
381
382 if (pktnum == rdcount) {
383 reg_val = readl(&arasan_nand_base->intsts_enr);
384 reg_val |= ARASAN_NAND_INT_STS_XFR_CMPLT_MASK;
385 writel(reg_val, &arasan_nand_base->intsts_enr);
386 } else {
387 reg_val = readl(&arasan_nand_base->intsts_enr);
388 writel(reg_val | ARASAN_NAND_INT_STS_BUF_RD_RDY_MASK,
389 &arasan_nand_base->intsts_enr);
390 }
391 reg_val = readl(&arasan_nand_base->intsts_reg);
392 writel(reg_val | ARASAN_NAND_INT_STS_BUF_RD_RDY_MASK,
393 &arasan_nand_base->intsts_reg);
394
395 for (i = 0; i < pktsize/4; i++)
396 bufptr[i] = readl(&arasan_nand_base->buf_dataport);
397
398
399 bufptr += pktsize/4;
400
401 if (rdcount >= pktnum)
402 break;
403
404 writel(ARASAN_NAND_INT_STS_BUF_RD_RDY_MASK,
405 &arasan_nand_base->intsts_enr);
406 }
407
408 timeout = ARASAN_NAND_POLL_TIMEOUT;
409
410 while (!(readl(&arasan_nand_base->intsts_reg) &
411 ARASAN_NAND_INT_STS_XFR_CMPLT_MASK) && timeout) {
412 udelay(1);
413 timeout--;
414 }
415 if (!timeout) {
416 puts("arasan rd_page timedout:Xfer CMPLT\n");
417 return -ETIMEDOUT;
418 }
419
420 reg_val = readl(&arasan_nand_base->intsts_enr);
421 writel(reg_val | ARASAN_NAND_INT_STS_XFR_CMPLT_MASK,
422 &arasan_nand_base->intsts_enr);
423 reg_val = readl(&arasan_nand_base->intsts_reg);
424 writel(reg_val | ARASAN_NAND_INT_STS_XFR_CMPLT_MASK,
425 &arasan_nand_base->intsts_reg);
426
Siva Durga Prasad Paladugu45e512c2018-01-04 16:04:21 +0530427 if (!nand->on_die_ecc_enabled) {
428 if (readl(&arasan_nand_base->intsts_reg) &
429 ARASAN_NAND_INT_STS_MUL_BIT_ERR_MASK) {
430 printf("arasan rd_page:sbiterror\n");
431 return -1;
432 }
Siva Durga Prasad Paladuguc7b0bc72015-11-17 14:30:10 +0530433
Siva Durga Prasad Paladugu45e512c2018-01-04 16:04:21 +0530434 if (readl(&arasan_nand_base->intsts_reg) &
435 ARASAN_NAND_INT_STS_ERR_EN_MASK) {
436 mtd->ecc_stats.failed++;
437 printf("arasan rd_page:multibiterror\n");
438 return -1;
439 }
Siva Durga Prasad Paladuguc7b0bc72015-11-17 14:30:10 +0530440 }
441
442 return 0;
443}
444
445static int arasan_nand_read_page_hwecc(struct mtd_info *mtd,
446 struct nand_chip *chip, u8 *buf, int oob_required, int page)
447{
448 int status;
449
450 status = arasan_nand_read_page(mtd, buf, (mtd->writesize));
451
452 if (oob_required)
453 chip->ecc.read_oob(mtd, chip, page);
454
455 return status;
456}
457
458static void arasan_nand_fill_tx(const u8 *buf, int len)
459{
460 u32 __iomem *nand = &arasan_nand_base->buf_dataport;
461
462 if (((unsigned long)buf & 0x3) != 0) {
463 if (((unsigned long)buf & 0x1) != 0) {
464 if (len) {
465 writeb(*buf, nand);
466 buf += 1;
467 len--;
468 }
469 }
470
471 if (((unsigned long)buf & 0x3) != 0) {
472 if (len >= 2) {
473 writew(*(u16 *)buf, nand);
474 buf += 2;
475 len -= 2;
476 }
477 }
478 }
479
480 while (len >= 4) {
481 writel(*(u32 *)buf, nand);
482 buf += 4;
483 len -= 4;
484 }
485
486 if (len) {
487 if (len >= 2) {
488 writew(*(u16 *)buf, nand);
489 buf += 2;
490 len -= 2;
491 }
492
493 if (len)
494 writeb(*buf, nand);
495 }
496}
497
498static int arasan_nand_write_page_hwecc(struct mtd_info *mtd,
Scott Wood46e13102016-05-30 13:57:57 -0500499 struct nand_chip *chip, const u8 *buf, int oob_required,
500 int page)
Siva Durga Prasad Paladuguc7b0bc72015-11-17 14:30:10 +0530501{
502 u32 reg_val, i, pktsize, pktnum;
503 const u32 *bufptr = (const u32 *)buf;
504 u32 timeout = ARASAN_NAND_POLL_TIMEOUT;
505 u32 size = mtd->writesize;
506 u32 rdcount = 0;
507 u8 column_addr_cycles;
Scott Wood17fed142016-05-30 13:57:56 -0500508 struct arasan_nand_info *nand = nand_get_controller_data(chip);
Siva Durga Prasad Paladuguc7b0bc72015-11-17 14:30:10 +0530509
510 if (chip->ecc_step_ds >= ARASAN_NAND_PKTSIZE_1K)
511 pktsize = ARASAN_NAND_PKTSIZE_1K;
512 else
513 pktsize = ARASAN_NAND_PKTSIZE_512;
514
515 if (size % pktsize)
516 pktnum = size/pktsize + 1;
517 else
518 pktnum = size/pktsize;
519
520 reg_val = readl(&arasan_nand_base->pkt_reg);
521 reg_val &= ~(ARASAN_NAND_PKT_REG_PKT_CNT_MASK |
522 ARASAN_NAND_PKT_REG_PKT_SIZE_MASK);
523 reg_val |= (pktnum << ARASAN_NAND_PKT_REG_PKT_CNT_SHFT) | pktsize;
524 writel(reg_val, &arasan_nand_base->pkt_reg);
525
Siva Durga Prasad Paladugu45e512c2018-01-04 16:04:21 +0530526 if (!nand->on_die_ecc_enabled) {
527 arasan_nand_enable_ecc();
528 column_addr_cycles = (chip->onfi_params.addr_cycles &
529 ARASAN_NAND_COL_ADDR_CYCL_MASK) >>
530 ARASAN_NAND_COL_ADDR_CYCL_SHIFT;
531 writel((NAND_CMD_RNDIN | (column_addr_cycles << 28)),
532 &arasan_nand_base->ecc_sprcmd_reg);
533 }
Siva Durga Prasad Paladuguc7b0bc72015-11-17 14:30:10 +0530534 writel(curr_cmd->pgm, &arasan_nand_base->pgm_reg);
535
536 while (rdcount < pktnum) {
537 timeout = ARASAN_NAND_POLL_TIMEOUT;
538 while (!(readl(&arasan_nand_base->intsts_reg) &
539 ARASAN_NAND_INT_STS_BUF_WR_RDY_MASK) && timeout) {
540 udelay(1);
541 timeout--;
542 }
543
544 if (!timeout) {
545 puts("arasan_write_page: timedout:Buff RDY\n");
546 return -ETIMEDOUT;
547 }
548
549 rdcount++;
550
551 if (pktnum == rdcount) {
552 reg_val = readl(&arasan_nand_base->intsts_enr);
553 reg_val |= ARASAN_NAND_INT_STS_XFR_CMPLT_MASK;
554 writel(reg_val, &arasan_nand_base->intsts_enr);
555 } else {
556 reg_val = readl(&arasan_nand_base->intsts_enr);
557 writel(reg_val | ARASAN_NAND_INT_STS_BUF_WR_RDY_MASK,
558 &arasan_nand_base->intsts_enr);
559 }
560
561 reg_val = readl(&arasan_nand_base->intsts_reg);
562 writel(reg_val | ARASAN_NAND_INT_STS_BUF_WR_RDY_MASK,
563 &arasan_nand_base->intsts_reg);
564
565 for (i = 0; i < pktsize/4; i++)
566 writel(bufptr[i], &arasan_nand_base->buf_dataport);
567
568 bufptr += pktsize/4;
569
570 if (rdcount >= pktnum)
571 break;
572
573 writel(ARASAN_NAND_INT_STS_BUF_WR_RDY_MASK,
574 &arasan_nand_base->intsts_enr);
575 }
576
577 timeout = ARASAN_NAND_POLL_TIMEOUT;
578
579 while (!(readl(&arasan_nand_base->intsts_reg) &
580 ARASAN_NAND_INT_STS_XFR_CMPLT_MASK) && timeout) {
581 udelay(1);
582 timeout--;
583 }
584 if (!timeout) {
585 puts("arasan write_page timedout:Xfer CMPLT\n");
586 return -ETIMEDOUT;
587 }
588
589 reg_val = readl(&arasan_nand_base->intsts_enr);
590 writel(reg_val | ARASAN_NAND_INT_STS_XFR_CMPLT_MASK,
591 &arasan_nand_base->intsts_enr);
592 reg_val = readl(&arasan_nand_base->intsts_reg);
593 writel(reg_val | ARASAN_NAND_INT_STS_XFR_CMPLT_MASK,
594 &arasan_nand_base->intsts_reg);
595
596 if (oob_required)
597 chip->ecc.write_oob(mtd, chip, nand->page);
598
599 return 0;
600}
601
602static int arasan_nand_read_oob(struct mtd_info *mtd, struct nand_chip *chip,
603 int page)
604{
605 chip->cmdfunc(mtd, NAND_CMD_READOOB, 0, page);
606 chip->read_buf(mtd, chip->oob_poi, (mtd->oobsize));
607
608 return 0;
609}
610
611static int arasan_nand_write_oob(struct mtd_info *mtd, struct nand_chip *chip,
612 int page)
613{
614 int status = 0;
615 const u8 *buf = chip->oob_poi;
616
617 chip->cmdfunc(mtd, NAND_CMD_SEQIN, mtd->writesize, page);
618 chip->write_buf(mtd, buf, mtd->oobsize);
619
620 return status;
621}
622
623static int arasan_nand_reset(struct arasan_nand_command_format *curr_cmd)
624{
625 u32 timeout = ARASAN_NAND_POLL_TIMEOUT;
626 u32 cmd_reg = 0;
627
628 writel(ARASAN_NAND_INT_STS_XFR_CMPLT_MASK,
629 &arasan_nand_base->intsts_enr);
630 cmd_reg = readl(&arasan_nand_base->cmd_reg);
631 cmd_reg &= ~ARASAN_NAND_CMD_CMD12_MASK;
632
633 cmd_reg |= curr_cmd->cmd1 |
634 (curr_cmd->cmd2 << ARASAN_NAND_CMD_CMD2_SHIFT);
635 writel(cmd_reg, &arasan_nand_base->cmd_reg);
636 writel(curr_cmd->pgm, &arasan_nand_base->pgm_reg);
637
638 while (!(readl(&arasan_nand_base->intsts_reg) &
639 ARASAN_NAND_INT_STS_XFR_CMPLT_MASK) && timeout) {
640 udelay(1);
641 timeout--;
642 }
643 if (!timeout) {
644 printf("ERROR:%s timedout\n", __func__);
645 return -ETIMEDOUT;
646 }
647
648 writel(ARASAN_NAND_INT_STS_XFR_CMPLT_MASK,
649 &arasan_nand_base->intsts_enr);
650
651 writel(ARASAN_NAND_INT_STS_XFR_CMPLT_MASK,
652 &arasan_nand_base->intsts_reg);
653
654 return 0;
655}
656
657static u8 arasan_nand_page(struct mtd_info *mtd)
658{
659 u8 page_val = 0;
660
661 switch (mtd->writesize) {
662 case 512:
663 page_val = 0;
664 break;
665 case 2048:
666 page_val = 1;
667 break;
668 case 4096:
669 page_val = 2;
670 break;
671 case 8192:
672 page_val = 3;
673 break;
674 case 16384:
675 page_val = 4;
676 break;
677 case 1024:
678 page_val = 5;
679 break;
680 default:
681 printf("%s:Pagesize>16K\n", __func__);
682 break;
683 }
684
685 return page_val;
686}
687
688static int arasan_nand_send_wrcmd(struct arasan_nand_command_format *curr_cmd,
689 int column, int page_addr, struct mtd_info *mtd)
690{
691 u32 reg_val, page;
692 u8 page_val, addr_cycles;
693
694 writel(ARASAN_NAND_INT_STS_BUF_WR_RDY_MASK,
695 &arasan_nand_base->intsts_enr);
696 reg_val = readl(&arasan_nand_base->cmd_reg);
697 reg_val &= ~ARASAN_NAND_CMD_CMD12_MASK;
698 reg_val |= curr_cmd->cmd1 |
699 (curr_cmd->cmd2 << ARASAN_NAND_CMD_CMD2_SHIFT);
700 if (curr_cmd->cmd1 == NAND_CMD_SEQIN) {
701 reg_val &= ~ARASAN_NAND_CMD_PG_SIZE_MASK;
702 page_val = arasan_nand_page(mtd);
703 reg_val |= (page_val << ARASAN_NAND_CMD_PG_SIZE_SHIFT);
704 }
705
706 reg_val &= ~ARASAN_NAND_CMD_ADDR_CYCL_MASK;
707 addr_cycles = arasan_nand_get_addrcycle(mtd);
708
709 if (addr_cycles == ARASAN_NAND_INVALID_ADDR_CYCL)
710 return ERR_ADDR_CYCLE;
711
712 reg_val |= (addr_cycles <<
713 ARASAN_NAND_CMD_ADDR_CYCL_SHIFT);
714 writel(reg_val, &arasan_nand_base->cmd_reg);
715
716 if (page_addr == -1)
717 page_addr = 0;
718
719 page = (page_addr << ARASAN_NAND_MEM_ADDR1_PAGE_SHIFT) &
720 ARASAN_NAND_MEM_ADDR1_PAGE_MASK;
721 column &= ARASAN_NAND_MEM_ADDR1_COL_MASK;
722 writel(page|column, &arasan_nand_base->memadr_reg1);
723
724 reg_val = readl(&arasan_nand_base->memadr_reg2);
725 reg_val &= ~ARASAN_NAND_MEM_ADDR2_PAGE_MASK;
726 reg_val |= (page_addr >> ARASAN_NAND_MEM_ADDR1_PAGE_SHIFT);
727 writel(reg_val, &arasan_nand_base->memadr_reg2);
Siva Durga Prasad Paladuguc7b0bc72015-11-17 14:30:10 +0530728
729 return 0;
730}
731
732static void arasan_nand_write_buf(struct mtd_info *mtd, const u8 *buf, int len)
733{
734 u32 reg_val;
735 u32 timeout = ARASAN_NAND_POLL_TIMEOUT;
736
737 reg_val = readl(&arasan_nand_base->pkt_reg);
738 reg_val &= ~(ARASAN_NAND_PKT_REG_PKT_CNT_MASK |
739 ARASAN_NAND_PKT_REG_PKT_SIZE_MASK);
740
741 reg_val |= (1 << ARASAN_NAND_PKT_REG_PKT_CNT_SHFT) | len;
742 writel(reg_val, &arasan_nand_base->pkt_reg);
743 writel(curr_cmd->pgm, &arasan_nand_base->pgm_reg);
744
745 while (!(readl(&arasan_nand_base->intsts_reg) &
746 ARASAN_NAND_INT_STS_BUF_WR_RDY_MASK) && timeout) {
747 udelay(1);
748 timeout--;
749 }
750
751 if (!timeout)
752 puts("ERROR:arasan_nand_write_buf timedout:Buff RDY\n");
753
754 reg_val = readl(&arasan_nand_base->intsts_enr);
755 reg_val |= ARASAN_NAND_INT_STS_XFR_CMPLT_MASK;
756 writel(reg_val, &arasan_nand_base->intsts_enr);
757 writel(reg_val | ARASAN_NAND_INT_STS_BUF_WR_RDY_MASK,
758 &arasan_nand_base->intsts_enr);
759 reg_val = readl(&arasan_nand_base->intsts_reg);
760 writel(reg_val | ARASAN_NAND_INT_STS_BUF_WR_RDY_MASK,
761 &arasan_nand_base->intsts_reg);
762
763 arasan_nand_fill_tx(buf, len);
764
765 timeout = ARASAN_NAND_POLL_TIMEOUT;
766 while (!(readl(&arasan_nand_base->intsts_reg) &
767 ARASAN_NAND_INT_STS_XFR_CMPLT_MASK) && timeout) {
768 udelay(1);
769 timeout--;
770 }
771 if (!timeout)
772 puts("ERROR:arasan_nand_write_buf timedout:Xfer CMPLT\n");
773
774 writel(readl(&arasan_nand_base->intsts_enr) |
775 ARASAN_NAND_INT_STS_XFR_CMPLT_MASK,
776 &arasan_nand_base->intsts_enr);
777 writel(readl(&arasan_nand_base->intsts_reg) |
778 ARASAN_NAND_INT_STS_XFR_CMPLT_MASK,
779 &arasan_nand_base->intsts_reg);
780}
781
782static int arasan_nand_erase(struct arasan_nand_command_format *curr_cmd,
783 int column, int page_addr, struct mtd_info *mtd)
784{
785 u32 reg_val, page;
786 u32 timeout = ARASAN_NAND_POLL_TIMEOUT;
787 u8 row_addr_cycles;
788
789 writel(ARASAN_NAND_INT_STS_XFR_CMPLT_MASK,
790 &arasan_nand_base->intsts_enr);
791 reg_val = readl(&arasan_nand_base->cmd_reg);
792 reg_val &= ~ARASAN_NAND_CMD_CMD12_MASK;
793 reg_val |= curr_cmd->cmd1 |
794 (curr_cmd->cmd2 << ARASAN_NAND_CMD_CMD2_SHIFT);
795 row_addr_cycles = arasan_nand_get_addrcycle(mtd);
796
797 if (row_addr_cycles == ARASAN_NAND_INVALID_ADDR_CYCL)
798 return ERR_ADDR_CYCLE;
799
800 reg_val &= ~ARASAN_NAND_CMD_ADDR_CYCL_MASK;
801 reg_val |= (row_addr_cycles <<
802 ARASAN_NAND_CMD_ADDR_CYCL_SHIFT);
803
804 writel(reg_val, &arasan_nand_base->cmd_reg);
805
Vipul Kumar673a5c22018-03-05 15:24:59 +0530806 page = (page_addr >> ARASAN_NAND_MEM_ADDR1_PAGE_SHIFT) &
Vipul Kumar9d9b99b2018-03-10 17:52:23 +0530807 ARASAN_NAND_MEM_ADDR1_COL_MASK;
Siva Durga Prasad Paladuguc7b0bc72015-11-17 14:30:10 +0530808 column = page_addr & ARASAN_NAND_MEM_ADDR1_COL_MASK;
Vipul Kumar673a5c22018-03-05 15:24:59 +0530809 writel(column | (page << ARASAN_NAND_MEM_ADDR1_PAGE_SHIFT),
810 &arasan_nand_base->memadr_reg1);
Siva Durga Prasad Paladuguc7b0bc72015-11-17 14:30:10 +0530811
812 reg_val = readl(&arasan_nand_base->memadr_reg2);
813 reg_val &= ~ARASAN_NAND_MEM_ADDR2_PAGE_MASK;
814 reg_val |= (page_addr >> ARASAN_NAND_MEM_ADDR1_PAGE_SHIFT);
815 writel(reg_val, &arasan_nand_base->memadr_reg2);
Siva Durga Prasad Paladuguc7b0bc72015-11-17 14:30:10 +0530816 writel(curr_cmd->pgm, &arasan_nand_base->pgm_reg);
817
818 while (!(readl(&arasan_nand_base->intsts_reg) &
819 ARASAN_NAND_INT_STS_XFR_CMPLT_MASK) && timeout) {
820 udelay(1);
821 timeout--;
822 }
823 if (!timeout) {
824 printf("ERROR:%s timedout:Xfer CMPLT\n", __func__);
825 return -ETIMEDOUT;
826 }
827
828 reg_val = readl(&arasan_nand_base->intsts_enr);
829 writel(reg_val | ARASAN_NAND_INT_STS_XFR_CMPLT_MASK,
830 &arasan_nand_base->intsts_enr);
831 reg_val = readl(&arasan_nand_base->intsts_reg);
832 writel(reg_val | ARASAN_NAND_INT_STS_XFR_CMPLT_MASK,
833 &arasan_nand_base->intsts_reg);
834
835 return 0;
836}
837
838static int arasan_nand_read_status(struct arasan_nand_command_format *curr_cmd,
839 int column, int page_addr, struct mtd_info *mtd)
840{
841 u32 reg_val;
842 u32 timeout = ARASAN_NAND_POLL_TIMEOUT;
843 u8 addr_cycles;
844
845 writel(ARASAN_NAND_INT_STS_XFR_CMPLT_MASK,
846 &arasan_nand_base->intsts_enr);
847 reg_val = readl(&arasan_nand_base->cmd_reg);
848 reg_val &= ~ARASAN_NAND_CMD_CMD12_MASK;
849 reg_val |= curr_cmd->cmd1 |
850 (curr_cmd->cmd2 << ARASAN_NAND_CMD_CMD2_SHIFT);
851 addr_cycles = arasan_nand_get_addrcycle(mtd);
852
853 if (addr_cycles == ARASAN_NAND_INVALID_ADDR_CYCL)
854 return ERR_ADDR_CYCLE;
855
856 reg_val &= ~ARASAN_NAND_CMD_ADDR_CYCL_MASK;
857 reg_val |= (addr_cycles <<
858 ARASAN_NAND_CMD_ADDR_CYCL_SHIFT);
859
860 writel(reg_val, &arasan_nand_base->cmd_reg);
861
862 reg_val = readl(&arasan_nand_base->pkt_reg);
863 reg_val &= ~(ARASAN_NAND_PKT_REG_PKT_CNT_MASK |
864 ARASAN_NAND_PKT_REG_PKT_SIZE_MASK);
865 reg_val |= (1 << ARASAN_NAND_PKT_REG_PKT_CNT_SHFT) | 1;
866 writel(reg_val, &arasan_nand_base->pkt_reg);
867
Siva Durga Prasad Paladuguc7b0bc72015-11-17 14:30:10 +0530868 writel(curr_cmd->pgm, &arasan_nand_base->pgm_reg);
869 while (!(readl(&arasan_nand_base->intsts_reg) &
870 ARASAN_NAND_INT_STS_XFR_CMPLT_MASK) && timeout) {
871 udelay(1);
872 timeout--;
873 }
874
875 if (!timeout) {
876 printf("ERROR:%s: timedout:Xfer CMPLT\n", __func__);
877 return -ETIMEDOUT;
878 }
879
880 reg_val = readl(&arasan_nand_base->intsts_enr);
881 writel(reg_val | ARASAN_NAND_INT_STS_XFR_CMPLT_MASK,
882 &arasan_nand_base->intsts_enr);
883 reg_val = readl(&arasan_nand_base->intsts_reg);
884 writel(reg_val | ARASAN_NAND_INT_STS_XFR_CMPLT_MASK,
885 &arasan_nand_base->intsts_reg);
886
887 return 0;
888}
889
890static int arasan_nand_send_rdcmd(struct arasan_nand_command_format *curr_cmd,
891 int column, int page_addr, struct mtd_info *mtd)
892{
893 u32 reg_val, addr_cycles, page;
894 u8 page_val;
895
896 reg_val = readl(&arasan_nand_base->intsts_enr);
897 writel(reg_val | ARASAN_NAND_INT_STS_BUF_RD_RDY_MASK,
898 &arasan_nand_base->intsts_enr);
899
900 reg_val = readl(&arasan_nand_base->cmd_reg);
901 reg_val &= ~ARASAN_NAND_CMD_CMD12_MASK;
902 reg_val |= curr_cmd->cmd1 |
903 (curr_cmd->cmd2 << ARASAN_NAND_CMD_CMD2_SHIFT);
904
905 if (curr_cmd->cmd1 == NAND_CMD_RNDOUT ||
906 curr_cmd->cmd1 == NAND_CMD_READ0) {
907 reg_val &= ~ARASAN_NAND_CMD_PG_SIZE_MASK;
908 page_val = arasan_nand_page(mtd);
909 reg_val |= (page_val << ARASAN_NAND_CMD_PG_SIZE_SHIFT);
910 }
911
Siva Durga Prasad Paladugu99459c22016-08-25 16:00:04 +0530912 reg_val &= ~ARASAN_NAND_CMD_ECC_ON_MASK;
913
Siva Durga Prasad Paladuguc7b0bc72015-11-17 14:30:10 +0530914 reg_val &= ~ARASAN_NAND_CMD_ADDR_CYCL_MASK;
915
916 addr_cycles = arasan_nand_get_addrcycle(mtd);
917
918 if (addr_cycles == ARASAN_NAND_INVALID_ADDR_CYCL)
919 return ERR_ADDR_CYCLE;
920
921 reg_val |= (addr_cycles << 28);
922 writel(reg_val, &arasan_nand_base->cmd_reg);
923
924 if (page_addr == -1)
925 page_addr = 0;
926
927 page = (page_addr << ARASAN_NAND_MEM_ADDR1_PAGE_SHIFT) &
928 ARASAN_NAND_MEM_ADDR1_PAGE_MASK;
929 column &= ARASAN_NAND_MEM_ADDR1_COL_MASK;
930 writel(page | column, &arasan_nand_base->memadr_reg1);
931
932 reg_val = readl(&arasan_nand_base->memadr_reg2);
933 reg_val &= ~ARASAN_NAND_MEM_ADDR2_PAGE_MASK;
934 reg_val |= (page_addr >> ARASAN_NAND_MEM_ADDR1_PAGE_SHIFT);
935 writel(reg_val, &arasan_nand_base->memadr_reg2);
936
Siva Durga Prasad Paladuguc7b0bc72015-11-17 14:30:10 +0530937 buf_index = 0;
938
939 return 0;
940}
941
942static void arasan_nand_read_buf(struct mtd_info *mtd, u8 *buf, int size)
943{
944 u32 reg_val, i;
945 u32 *bufptr = (u32 *)buf;
946 u32 timeout = ARASAN_NAND_POLL_TIMEOUT;
947
948 reg_val = readl(&arasan_nand_base->pkt_reg);
949 reg_val &= ~(ARASAN_NAND_PKT_REG_PKT_CNT_MASK |
950 ARASAN_NAND_PKT_REG_PKT_SIZE_MASK);
951 reg_val |= (1 << ARASAN_NAND_PKT_REG_PKT_CNT_SHFT) | size;
952 writel(reg_val, &arasan_nand_base->pkt_reg);
953
954 writel(curr_cmd->pgm, &arasan_nand_base->pgm_reg);
955
956 while (!(readl(&arasan_nand_base->intsts_reg) &
957 ARASAN_NAND_INT_STS_BUF_RD_RDY_MASK) && timeout) {
958 udelay(1);
959 timeout--;
960 }
961
962 if (!timeout)
963 puts("ERROR:arasan_nand_read_buf timedout:Buff RDY\n");
964
965 reg_val = readl(&arasan_nand_base->intsts_enr);
966 reg_val |= ARASAN_NAND_INT_STS_XFR_CMPLT_MASK;
967 writel(reg_val, &arasan_nand_base->intsts_enr);
968
969 writel(reg_val | ARASAN_NAND_INT_STS_BUF_RD_RDY_MASK,
970 &arasan_nand_base->intsts_enr);
971 reg_val = readl(&arasan_nand_base->intsts_reg);
972 writel(reg_val | ARASAN_NAND_INT_STS_BUF_RD_RDY_MASK,
973 &arasan_nand_base->intsts_reg);
974
975 buf_index = 0;
976 for (i = 0; i < size / 4; i++)
977 bufptr[i] = readl(&arasan_nand_base->buf_dataport);
978
979 if (size & 0x03)
980 bufptr[i] = readl(&arasan_nand_base->buf_dataport);
981
982 timeout = ARASAN_NAND_POLL_TIMEOUT;
983
984 while (!(readl(&arasan_nand_base->intsts_reg) &
985 ARASAN_NAND_INT_STS_XFR_CMPLT_MASK) && timeout) {
986 udelay(1);
987 timeout--;
988 }
989
990 if (!timeout)
991 puts("ERROR:arasan_nand_read_buf timedout:Xfer CMPLT\n");
992
993 reg_val = readl(&arasan_nand_base->intsts_enr);
994 writel(reg_val | ARASAN_NAND_INT_STS_XFR_CMPLT_MASK,
995 &arasan_nand_base->intsts_enr);
996 reg_val = readl(&arasan_nand_base->intsts_reg);
997 writel(reg_val | ARASAN_NAND_INT_STS_XFR_CMPLT_MASK,
998 &arasan_nand_base->intsts_reg);
999}
1000
1001static u8 arasan_nand_read_byte(struct mtd_info *mtd)
1002{
Scott Wood17fed142016-05-30 13:57:56 -05001003 struct nand_chip *chip = mtd_to_nand(mtd);
Siva Durga Prasad Paladuguc7b0bc72015-11-17 14:30:10 +05301004 u32 size;
1005 u8 val;
1006 struct nand_onfi_params *p;
1007
1008 if (buf_index == 0) {
1009 p = &chip->onfi_params;
1010 if (curr_cmd->cmd1 == NAND_CMD_READID)
1011 size = 4;
1012 else if (curr_cmd->cmd1 == NAND_CMD_PARAM)
1013 size = sizeof(struct nand_onfi_params);
1014 else if (curr_cmd->cmd1 == NAND_CMD_RNDOUT)
1015 size = le16_to_cpu(p->ext_param_page_length) * 16;
1016 else if (curr_cmd->cmd1 == NAND_CMD_GET_FEATURES)
1017 size = 4;
1018 else if (curr_cmd->cmd1 == NAND_CMD_STATUS)
1019 return readb(&arasan_nand_base->flash_sts_reg);
1020 else
1021 size = 8;
1022 chip->read_buf(mtd, &buf_data[0], size);
1023 }
1024
1025 val = *(&buf_data[0] + buf_index);
1026 buf_index++;
1027
1028 return val;
1029}
1030
1031static void arasan_nand_cmd_function(struct mtd_info *mtd, unsigned int command,
1032 int column, int page_addr)
1033{
1034 u32 i, ret = 0;
Scott Wood17fed142016-05-30 13:57:56 -05001035 struct nand_chip *chip = mtd_to_nand(mtd);
1036 struct arasan_nand_info *nand = nand_get_controller_data(chip);
Siva Durga Prasad Paladuguc7b0bc72015-11-17 14:30:10 +05301037
1038 curr_cmd = NULL;
1039 writel(ARASAN_NAND_INT_STS_XFR_CMPLT_MASK,
1040 &arasan_nand_base->intsts_enr);
1041
1042 if ((command == NAND_CMD_READOOB) &&
1043 (mtd->writesize > 512)) {
1044 column += mtd->writesize;
1045 command = NAND_CMD_READ0;
1046 }
1047
1048 /* Get the command format */
1049 for (i = 0; (arasan_nand_commands[i].cmd1 != NAND_CMD_NONE ||
1050 arasan_nand_commands[i].cmd2 != NAND_CMD_NONE); i++) {
1051 if (command == arasan_nand_commands[i].cmd1) {
1052 curr_cmd = &arasan_nand_commands[i];
1053 break;
1054 }
1055 }
1056
1057 if (curr_cmd == NULL) {
1058 printf("Unsupported Command; 0x%x\n", command);
1059 return;
1060 }
1061
1062 if (curr_cmd->cmd1 == NAND_CMD_RESET)
1063 ret = arasan_nand_reset(curr_cmd);
1064
1065 if ((curr_cmd->cmd1 == NAND_CMD_READID) ||
1066 (curr_cmd->cmd1 == NAND_CMD_PARAM) ||
1067 (curr_cmd->cmd1 == NAND_CMD_RNDOUT) ||
1068 (curr_cmd->cmd1 == NAND_CMD_GET_FEATURES) ||
1069 (curr_cmd->cmd1 == NAND_CMD_READ0))
1070 ret = arasan_nand_send_rdcmd(curr_cmd, column, page_addr, mtd);
1071
1072 if ((curr_cmd->cmd1 == NAND_CMD_SET_FEATURES) ||
1073 (curr_cmd->cmd1 == NAND_CMD_SEQIN)) {
1074 nand->page = page_addr;
1075 ret = arasan_nand_send_wrcmd(curr_cmd, column, page_addr, mtd);
1076 }
1077
1078 if (curr_cmd->cmd1 == NAND_CMD_ERASE1)
1079 ret = arasan_nand_erase(curr_cmd, column, page_addr, mtd);
1080
1081 if (curr_cmd->cmd1 == NAND_CMD_STATUS)
1082 ret = arasan_nand_read_status(curr_cmd, column, page_addr, mtd);
1083
1084 if (ret != 0)
1085 printf("ERROR:%s:command:0x%x\n", __func__, curr_cmd->cmd1);
1086}
1087
Siva Durga Prasad Paladugu45e512c2018-01-04 16:04:21 +05301088static void arasan_check_ondie(struct mtd_info *mtd)
1089{
1090 struct nand_chip *nand_chip = mtd_to_nand(mtd);
1091 struct arasan_nand_info *nand = nand_get_controller_data(nand_chip);
1092 u8 maf_id, dev_id;
1093 u8 get_feature[4];
1094 u8 set_feature[4] = {ENABLE_ONDIE_ECC, 0x00, 0x00, 0x00};
1095 u32 i;
1096
1097 /* Send the command for reading device ID */
1098 nand_chip->cmdfunc(mtd, NAND_CMD_RESET, -1, -1);
1099 nand_chip->cmdfunc(mtd, NAND_CMD_READID, 0, -1);
1100
1101 /* Read manufacturer and device IDs */
1102 maf_id = nand_chip->read_byte(mtd);
1103 dev_id = nand_chip->read_byte(mtd);
1104
1105 if ((maf_id == NAND_MFR_MICRON) &&
1106 ((dev_id == 0xf1) || (dev_id == 0xa1) || (dev_id == 0xb1) ||
1107 (dev_id == 0xaa) || (dev_id == 0xba) || (dev_id == 0xda) ||
1108 (dev_id == 0xca) || (dev_id == 0xac) || (dev_id == 0xbc) ||
1109 (dev_id == 0xdc) || (dev_id == 0xcc) || (dev_id == 0xa3) ||
1110 (dev_id == 0xb3) || (dev_id == 0xd3) || (dev_id == 0xc3))) {
1111 nand_chip->cmdfunc(mtd, NAND_CMD_SET_FEATURES,
1112 ONDIE_ECC_FEATURE_ADDR, -1);
1113
1114 nand_chip->write_buf(mtd, &set_feature[0], 4);
1115 nand_chip->cmdfunc(mtd, NAND_CMD_GET_FEATURES,
1116 ONDIE_ECC_FEATURE_ADDR, -1);
1117
1118 for (i = 0; i < 4; i++)
1119 get_feature[i] = nand_chip->read_byte(mtd);
1120
1121 if (get_feature[0] & ENABLE_ONDIE_ECC)
1122 nand->on_die_ecc_enabled = true;
1123 else
1124 printf("%s: Unable to enable OnDie ECC\n", __func__);
1125
1126 /* Use the BBT pattern descriptors */
1127 nand_chip->bbt_td = &bbt_main_descr;
1128 nand_chip->bbt_md = &bbt_mirror_descr;
1129 }
1130}
1131
Siva Durga Prasad Paladuguc7b0bc72015-11-17 14:30:10 +05301132static int arasan_nand_ecc_init(struct mtd_info *mtd)
1133{
1134 int found = -1;
Siva Durga Prasad Paladugub0a4f132018-01-04 16:04:22 +05301135 u32 regval, eccpos_start, i, eccaddr;
Scott Wood17fed142016-05-30 13:57:56 -05001136 struct nand_chip *nand_chip = mtd_to_nand(mtd);
Siva Durga Prasad Paladuguc7b0bc72015-11-17 14:30:10 +05301137
Siva Durga Prasad Paladuguc7b0bc72015-11-17 14:30:10 +05301138 for (i = 0; i < ARRAY_SIZE(ecc_matrix); i++) {
1139 if ((ecc_matrix[i].pagesize == mtd->writesize) &&
1140 (ecc_matrix[i].ecc_codeword_size >=
1141 nand_chip->ecc_step_ds)) {
1142 if (ecc_matrix[i].eccbits >=
1143 nand_chip->ecc_strength_ds) {
1144 found = i;
1145 break;
1146 }
1147 found = i;
1148 }
1149 }
1150
1151 if (found < 0)
1152 return 1;
1153
Siva Durga Prasad Paladugub0a4f132018-01-04 16:04:22 +05301154 eccaddr = mtd->writesize + mtd->oobsize -
1155 ecc_matrix[found].eccsize;
1156
1157 regval = eccaddr |
Siva Durga Prasad Paladugudb796632016-05-25 15:20:38 +05301158 (ecc_matrix[found].eccsize << ARASAN_NAND_ECC_SIZE_SHIFT) |
1159 (ecc_matrix[found].bch << ARASAN_NAND_ECC_BCH_SHIFT);
Siva Durga Prasad Paladuguc7b0bc72015-11-17 14:30:10 +05301160 writel(regval, &arasan_nand_base->ecc_reg);
1161
Siva Durga Prasad Paladugudb796632016-05-25 15:20:38 +05301162 if (ecc_matrix[found].bch) {
Siva Durga Prasad Paladuguc7b0bc72015-11-17 14:30:10 +05301163 regval = readl(&arasan_nand_base->memadr_reg2);
1164 regval &= ~ARASAN_NAND_MEM_ADDR2_BCH_MASK;
Siva Durga Prasad Paladugudb796632016-05-25 15:20:38 +05301165 regval |= (ecc_matrix[found].bchval <<
Siva Durga Prasad Paladuguc7b0bc72015-11-17 14:30:10 +05301166 ARASAN_NAND_MEM_ADDR2_BCH_SHIFT);
1167 writel(regval, &arasan_nand_base->memadr_reg2);
1168 }
1169
Siva Durga Prasad Paladugudb796632016-05-25 15:20:38 +05301170 nand_oob.eccbytes = ecc_matrix[found].eccsize;
Siva Durga Prasad Paladuguc7b0bc72015-11-17 14:30:10 +05301171 eccpos_start = mtd->oobsize - nand_oob.eccbytes;
1172
1173 for (i = 0; i < nand_oob.eccbytes; i++)
1174 nand_oob.eccpos[i] = eccpos_start + i;
1175
1176 nand_oob.oobfree[0].offset = 2;
1177 nand_oob.oobfree[0].length = eccpos_start - 2;
1178
Siva Durga Prasad Paladugudb796632016-05-25 15:20:38 +05301179 nand_chip->ecc.size = ecc_matrix[found].ecc_codeword_size;
1180 nand_chip->ecc.strength = ecc_matrix[found].eccbits;
1181 nand_chip->ecc.bytes = ecc_matrix[found].eccsize;
Siva Durga Prasad Paladuguc7b0bc72015-11-17 14:30:10 +05301182 nand_chip->ecc.layout = &nand_oob;
1183
1184 return 0;
1185}
1186
1187static int arasan_nand_init(struct nand_chip *nand_chip, int devnum)
1188{
1189 struct arasan_nand_info *nand;
1190 struct mtd_info *mtd;
1191 int err = -1;
1192
1193 nand = calloc(1, sizeof(struct arasan_nand_info));
1194 if (!nand) {
1195 printf("%s: failed to allocate\n", __func__);
1196 return err;
1197 }
1198
1199 nand->nand_base = arasan_nand_base;
Scott Wood17fed142016-05-30 13:57:56 -05001200 mtd = nand_to_mtd(nand_chip);
1201 nand_set_controller_data(nand_chip, nand);
Siva Durga Prasad Paladuguc7b0bc72015-11-17 14:30:10 +05301202
Martin Lund5f9ae0e2018-10-30 14:16:19 +01001203#ifdef CONFIG_SYS_NAND_NO_SUBPAGE_WRITE
1204 nand_chip->options |= NAND_NO_SUBPAGE_WRITE;
1205#endif
1206
Siva Durga Prasad Paladuguc7b0bc72015-11-17 14:30:10 +05301207 /* Set the driver entry points for MTD */
1208 nand_chip->cmdfunc = arasan_nand_cmd_function;
1209 nand_chip->select_chip = arasan_nand_select_chip;
1210 nand_chip->read_byte = arasan_nand_read_byte;
1211
1212 /* Buffer read/write routines */
1213 nand_chip->read_buf = arasan_nand_read_buf;
1214 nand_chip->write_buf = arasan_nand_write_buf;
1215 nand_chip->bbt_options = NAND_BBT_USE_FLASH;
1216
1217 writel(0x0, &arasan_nand_base->cmd_reg);
1218 writel(0x0, &arasan_nand_base->pgm_reg);
1219
1220 /* first scan to find the device and get the page size */
T Karthik Reddy7cd85222018-12-03 16:11:58 +05301221 if (nand_scan_ident(mtd, CONFIG_SYS_NAND_MAX_CHIPS, NULL)) {
Siva Durga Prasad Paladuguc7b0bc72015-11-17 14:30:10 +05301222 printf("%s: nand_scan_ident failed\n", __func__);
1223 goto fail;
1224 }
1225
Siva Durga Prasad Paladugu80c889c2018-01-04 16:04:20 +05301226 nand_chip->ecc.mode = NAND_ECC_HW;
1227 nand_chip->ecc.hwctl = NULL;
1228 nand_chip->ecc.read_page = arasan_nand_read_page_hwecc;
1229 nand_chip->ecc.write_page = arasan_nand_write_page_hwecc;
1230 nand_chip->ecc.read_oob = arasan_nand_read_oob;
1231 nand_chip->ecc.write_oob = arasan_nand_write_oob;
1232
Siva Durga Prasad Paladugu45e512c2018-01-04 16:04:21 +05301233 arasan_check_ondie(mtd);
1234
1235 /*
1236 * If on die supported, then give priority to on-die ecc and use
1237 * it instead of controller ecc.
1238 */
1239 if (nand->on_die_ecc_enabled) {
1240 nand_chip->ecc.strength = 1;
1241 nand_chip->ecc.size = mtd->writesize;
1242 nand_chip->ecc.bytes = 0;
1243 nand_chip->ecc.layout = &ondie_nand_oob_64;
1244 } else {
1245 if (arasan_nand_ecc_init(mtd)) {
1246 printf("%s: nand_ecc_init failed\n", __func__);
1247 goto fail;
1248 }
Siva Durga Prasad Paladuguc7b0bc72015-11-17 14:30:10 +05301249 }
1250
1251 if (nand_scan_tail(mtd)) {
1252 printf("%s: nand_scan_tail failed\n", __func__);
1253 goto fail;
1254 }
1255
Scott Wood2c1b7e12016-05-30 13:57:55 -05001256 if (nand_register(devnum, mtd)) {
Siva Durga Prasad Paladuguc7b0bc72015-11-17 14:30:10 +05301257 printf("Nand Register Fail\n");
1258 goto fail;
1259 }
1260
1261 return 0;
1262fail:
1263 free(nand);
1264 return err;
1265}
1266
1267void board_nand_init(void)
1268{
1269 struct nand_chip *nand = &nand_chip[0];
1270
1271 if (arasan_nand_init(nand, 0))
1272 puts("NAND init failed\n");
1273}