blob: d90186684db85ff18a99fbbb95f93a466824e37b [file] [log] [blame]
Leo (Hao) Chen266dead2009-10-09 19:13:08 -07001/*****************************************************************************
2* Copyright 2003 - 2009 Broadcom Corporation. All rights reserved.
3*
4* Unless you and Broadcom execute a separate written software license
5* agreement governing use of this software, this software is licensed to you
6* under the terms of the GNU General Public License version 2, available at
7* http://www.broadcom.com/licenses/GPLv2.php (the "GPL").
8*
9* Notwithstanding the above, under no circumstances may you combine this
10* software in any way with any other Broadcom software provided under a
11* license other than the GPL, without Broadcom's express prior written
12* consent.
13*****************************************************************************/
14#ifndef NAND_BCM_UMI_H
15#define NAND_BCM_UMI_H
16
17/* ---- Include Files ---------------------------------------------------- */
18#include <mach/reg_umi.h>
19#include <mach/reg_nand.h>
Arnd Bergmann651ef182012-04-23 15:42:54 +000020#include <mach/cfg_global.h>
Leo (Hao) Chen266dead2009-10-09 19:13:08 -070021
22/* ---- Constants and Types ---------------------------------------------- */
23#if (CFG_GLOBAL_CHIP_FAMILY == CFG_GLOBAL_CHIP_FAMILY_BCMRING)
24#define NAND_ECC_BCH (CFG_GLOBAL_CHIP_REV > 0xA0)
25#else
26#define NAND_ECC_BCH 0
27#endif
28
29#define CFG_GLOBAL_NAND_ECC_BCH_NUM_BYTES 13
30
31#if NAND_ECC_BCH
32#ifdef BOOT0_BUILD
33#define NAND_ECC_NUM_BYTES 13
34#else
35#define NAND_ECC_NUM_BYTES CFG_GLOBAL_NAND_ECC_BCH_NUM_BYTES
36#endif
37#else
38#define NAND_ECC_NUM_BYTES 3
39#endif
40
41#define NAND_DATA_ACCESS_SIZE 512
42
43/* ---- Variable Externs ------------------------------------------ */
44/* ---- Function Prototypes --------------------------------------- */
45int nand_bcm_umi_bch_correct_page(uint8_t *datap, uint8_t *readEccData,
46 int numEccBytes);
47
48/* Check in device is ready */
49static inline int nand_bcm_umi_dev_ready(void)
50{
Arnd Bergmann878040e2012-04-25 16:44:23 +000051 return readl(&REG_UMI_NAND_RCSR) & REG_UMI_NAND_RCSR_RDY;
Leo (Hao) Chen266dead2009-10-09 19:13:08 -070052}
53
54/* Wait until device is ready */
55static inline void nand_bcm_umi_wait_till_ready(void)
56{
57 while (nand_bcm_umi_dev_ready() == 0)
58 ;
59}
60
61/* Enable Hamming ECC */
62static inline void nand_bcm_umi_hamming_enable_hwecc(void)
63{
64 /* disable and reset ECC, 512 byte page */
Arnd Bergmann878040e2012-04-25 16:44:23 +000065 writel(readl(&REG_UMI_NAND_ECC_CSR) & ~(REG_UMI_NAND_ECC_CSR_ECC_ENABLE |
66 REG_UMI_NAND_ECC_CSR_256BYTE), &REG_UMI_NAND_ECC_CSR);
Leo (Hao) Chen266dead2009-10-09 19:13:08 -070067 /* enable ECC */
Arnd Bergmann878040e2012-04-25 16:44:23 +000068 writel(readl(&REG_UMI_NAND_ECC_CSR) | REG_UMI_NAND_ECC_CSR_ECC_ENABLE,
69 &REG_UMI_NAND_ECC_CSR);
Leo (Hao) Chen266dead2009-10-09 19:13:08 -070070}
71
72#if NAND_ECC_BCH
73/* BCH ECC specifics */
74#define ECC_BITS_PER_CORRECTABLE_BIT 13
75
76/* Enable BCH Read ECC */
77static inline void nand_bcm_umi_bch_enable_read_hwecc(void)
78{
79 /* disable and reset ECC */
Arnd Bergmann878040e2012-04-25 16:44:23 +000080 writel(REG_UMI_BCH_CTRL_STATUS_RD_ECC_VALID, &REG_UMI_BCH_CTRL_STATUS);
Leo (Hao) Chen266dead2009-10-09 19:13:08 -070081 /* Turn on ECC */
Arnd Bergmann878040e2012-04-25 16:44:23 +000082 writel(REG_UMI_BCH_CTRL_STATUS_ECC_RD_EN, &REG_UMI_BCH_CTRL_STATUS);
Leo (Hao) Chen266dead2009-10-09 19:13:08 -070083}
84
85/* Enable BCH Write ECC */
86static inline void nand_bcm_umi_bch_enable_write_hwecc(void)
87{
88 /* disable and reset ECC */
Arnd Bergmann878040e2012-04-25 16:44:23 +000089 writel(REG_UMI_BCH_CTRL_STATUS_WR_ECC_VALID, &REG_UMI_BCH_CTRL_STATUS);
Leo (Hao) Chen266dead2009-10-09 19:13:08 -070090 /* Turn on ECC */
Arnd Bergmann878040e2012-04-25 16:44:23 +000091 writel(REG_UMI_BCH_CTRL_STATUS_ECC_WR_EN, &REG_UMI_BCH_CTRL_STATUS);
Leo (Hao) Chen266dead2009-10-09 19:13:08 -070092}
93
94/* Config number of BCH ECC bytes */
95static inline void nand_bcm_umi_bch_config_ecc(uint8_t numEccBytes)
96{
97 uint32_t nValue;
98 uint32_t tValue;
99 uint32_t kValue;
100 uint32_t numBits = numEccBytes * 8;
101
102 /* disable and reset ECC */
Arnd Bergmann878040e2012-04-25 16:44:23 +0000103 writel(REG_UMI_BCH_CTRL_STATUS_WR_ECC_VALID |
104 REG_UMI_BCH_CTRL_STATUS_RD_ECC_VALID,
105 &REG_UMI_BCH_CTRL_STATUS);
Leo (Hao) Chen266dead2009-10-09 19:13:08 -0700106
107 /* Every correctible bit requires 13 ECC bits */
108 tValue = (uint32_t) (numBits / ECC_BITS_PER_CORRECTABLE_BIT);
109
110 /* Total data in number of bits for generating and computing BCH ECC */
111 nValue = (NAND_DATA_ACCESS_SIZE + numEccBytes) * 8;
112
113 /* K parameter is used internally. K = N - (T * 13) */
114 kValue = nValue - (tValue * ECC_BITS_PER_CORRECTABLE_BIT);
115
116 /* Write the settings */
Arnd Bergmann878040e2012-04-25 16:44:23 +0000117 writel(nValue, &REG_UMI_BCH_N);
118 writel(tValue, &REG_UMI_BCH_T);
119 writel(kValue, &REG_UMI_BCH_K);
Leo (Hao) Chen266dead2009-10-09 19:13:08 -0700120}
121
122/* Pause during ECC read calculation to skip bytes in OOB */
123static inline void nand_bcm_umi_bch_pause_read_ecc_calc(void)
124{
Arnd Bergmann878040e2012-04-25 16:44:23 +0000125 writel(REG_UMI_BCH_CTRL_STATUS_ECC_RD_EN | REG_UMI_BCH_CTRL_STATUS_PAUSE_ECC_DEC, &REG_UMI_BCH_CTRL_STATUS);
Leo (Hao) Chen266dead2009-10-09 19:13:08 -0700126}
127
128/* Resume during ECC read calculation after skipping bytes in OOB */
129static inline void nand_bcm_umi_bch_resume_read_ecc_calc(void)
130{
Arnd Bergmann878040e2012-04-25 16:44:23 +0000131 writel(REG_UMI_BCH_CTRL_STATUS_ECC_RD_EN, &REG_UMI_BCH_CTRL_STATUS);
Leo (Hao) Chen266dead2009-10-09 19:13:08 -0700132}
133
134/* Poll read ECC calc to check when hardware completes */
135static inline uint32_t nand_bcm_umi_bch_poll_read_ecc_calc(void)
136{
137 uint32_t regVal;
138
139 do {
140 /* wait for ECC to be valid */
Arnd Bergmann878040e2012-04-25 16:44:23 +0000141 regVal = readl(&REG_UMI_BCH_CTRL_STATUS);
Leo (Hao) Chen266dead2009-10-09 19:13:08 -0700142 } while ((regVal & REG_UMI_BCH_CTRL_STATUS_RD_ECC_VALID) == 0);
143
144 return regVal;
145}
146
147/* Poll write ECC calc to check when hardware completes */
148static inline void nand_bcm_umi_bch_poll_write_ecc_calc(void)
149{
150 /* wait for ECC to be valid */
Arnd Bergmann878040e2012-04-25 16:44:23 +0000151 while ((readl(&REG_UMI_BCH_CTRL_STATUS) & REG_UMI_BCH_CTRL_STATUS_WR_ECC_VALID)
Leo (Hao) Chen266dead2009-10-09 19:13:08 -0700152 == 0)
153 ;
154}
155
156/* Read the OOB and ECC, for kernel write OOB to a buffer */
157#if defined(__KERNEL__) && !defined(STANDALONE)
158static inline void nand_bcm_umi_bch_read_oobEcc(uint32_t pageSize,
159 uint8_t *eccCalc, int numEccBytes, uint8_t *oobp)
160#else
161static inline void nand_bcm_umi_bch_read_oobEcc(uint32_t pageSize,
162 uint8_t *eccCalc, int numEccBytes)
163#endif
164{
165 int eccPos = 0;
166 int numToRead = 16; /* There are 16 bytes per sector in the OOB */
167
168 /* ECC is already paused when this function is called */
Roel Kluin13858582010-01-22 22:22:52 +0100169 if (pageSize != NAND_DATA_ACCESS_SIZE) {
170 /* skip BI */
171#if defined(__KERNEL__) && !defined(STANDALONE)
Arnd Bergmann878040e2012-04-25 16:44:23 +0000172 *oobp++ = readb(&REG_NAND_DATA8);
Roel Kluin13858582010-01-22 22:22:52 +0100173#else
Arnd Bergmann878040e2012-04-25 16:44:23 +0000174 readb(&REG_NAND_DATA8);
Roel Kluin13858582010-01-22 22:22:52 +0100175#endif
176 numToRead--;
177 }
178
179 while (numToRead > numEccBytes) {
180 /* skip free oob region */
181#if defined(__KERNEL__) && !defined(STANDALONE)
Arnd Bergmann878040e2012-04-25 16:44:23 +0000182 *oobp++ = readb(&REG_NAND_DATA8);
Roel Kluin13858582010-01-22 22:22:52 +0100183#else
Arnd Bergmann878040e2012-04-25 16:44:23 +0000184 readb(&REG_NAND_DATA8);
Roel Kluin13858582010-01-22 22:22:52 +0100185#endif
186 numToRead--;
187 }
Leo (Hao) Chen266dead2009-10-09 19:13:08 -0700188
189 if (pageSize == NAND_DATA_ACCESS_SIZE) {
Leo (Hao) Chen266dead2009-10-09 19:13:08 -0700190 /* read ECC bytes before BI */
191 nand_bcm_umi_bch_resume_read_ecc_calc();
192
193 while (numToRead > 11) {
194#if defined(__KERNEL__) && !defined(STANDALONE)
Arnd Bergmann878040e2012-04-25 16:44:23 +0000195 *oobp = readb(&REG_NAND_DATA8);
Leo (Hao) Chen266dead2009-10-09 19:13:08 -0700196 eccCalc[eccPos++] = *oobp;
197 oobp++;
198#else
Arnd Bergmann878040e2012-04-25 16:44:23 +0000199 eccCalc[eccPos++] = readb(&REG_NAND_DATA8);
Leo (Hao) Chen266dead2009-10-09 19:13:08 -0700200#endif
Roel Kluin13858582010-01-22 22:22:52 +0100201 numToRead--;
Leo (Hao) Chen266dead2009-10-09 19:13:08 -0700202 }
203
204 nand_bcm_umi_bch_pause_read_ecc_calc();
205
206 if (numToRead == 11) {
207 /* read BI */
208#if defined(__KERNEL__) && !defined(STANDALONE)
Arnd Bergmann878040e2012-04-25 16:44:23 +0000209 *oobp++ = readb(&REG_NAND_DATA8);
Leo (Hao) Chen266dead2009-10-09 19:13:08 -0700210#else
Arnd Bergmann878040e2012-04-25 16:44:23 +0000211 readb(&REG_NAND_DATA8);
Leo (Hao) Chen266dead2009-10-09 19:13:08 -0700212#endif
213 numToRead--;
214 }
215
Roel Kluin13858582010-01-22 22:22:52 +0100216 }
217 /* read ECC bytes */
218 nand_bcm_umi_bch_resume_read_ecc_calc();
219 while (numToRead) {
Leo (Hao) Chen266dead2009-10-09 19:13:08 -0700220#if defined(__KERNEL__) && !defined(STANDALONE)
Arnd Bergmann878040e2012-04-25 16:44:23 +0000221 *oobp = readb(&REG_NAND_DATA8);
Roel Kluin13858582010-01-22 22:22:52 +0100222 eccCalc[eccPos++] = *oobp;
223 oobp++;
Leo (Hao) Chen266dead2009-10-09 19:13:08 -0700224#else
Arnd Bergmann878040e2012-04-25 16:44:23 +0000225 eccCalc[eccPos++] = readb(&REG_NAND_DATA8);
Leo (Hao) Chen266dead2009-10-09 19:13:08 -0700226#endif
227 numToRead--;
Leo (Hao) Chen266dead2009-10-09 19:13:08 -0700228 }
229}
230
231/* Helper function to write ECC */
232static inline void NAND_BCM_UMI_ECC_WRITE(int numEccBytes, int eccBytePos,
233 uint8_t *oobp, uint8_t eccVal)
234{
235 if (eccBytePos <= numEccBytes)
236 *oobp = eccVal;
237}
238
239/* Write OOB with ECC */
240static inline void nand_bcm_umi_bch_write_oobEcc(uint32_t pageSize,
241 uint8_t *oobp, int numEccBytes)
242{
243 uint32_t eccVal = 0xffffffff;
244
245 /* wait for write ECC to be valid */
246 nand_bcm_umi_bch_poll_write_ecc_calc();
247
248 /*
249 ** Get the hardware ecc from the 32-bit result registers.
250 ** Read after 512 byte accesses. Format B3B2B1B0
251 ** where B3 = ecc3, etc.
252 */
253
254 if (pageSize == NAND_DATA_ACCESS_SIZE) {
255 /* Now fill in the ECC bytes */
256 if (numEccBytes >= 13)
Arnd Bergmann878040e2012-04-25 16:44:23 +0000257 eccVal = readl(&REG_UMI_BCH_WR_ECC_3);
Leo (Hao) Chen266dead2009-10-09 19:13:08 -0700258
259 /* Usually we skip CM in oob[0,1] */
260 NAND_BCM_UMI_ECC_WRITE(numEccBytes, 15, &oobp[0],
261 (eccVal >> 16) & 0xff);
262 NAND_BCM_UMI_ECC_WRITE(numEccBytes, 14, &oobp[1],
263 (eccVal >> 8) & 0xff);
264
265 /* Write ECC in oob[2,3,4] */
266 NAND_BCM_UMI_ECC_WRITE(numEccBytes, 13, &oobp[2],
267 eccVal & 0xff); /* ECC 12 */
268
269 if (numEccBytes >= 9)
Arnd Bergmann878040e2012-04-25 16:44:23 +0000270 eccVal = readl(&REG_UMI_BCH_WR_ECC_2);
Leo (Hao) Chen266dead2009-10-09 19:13:08 -0700271
272 NAND_BCM_UMI_ECC_WRITE(numEccBytes, 12, &oobp[3],
273 (eccVal >> 24) & 0xff); /* ECC11 */
274 NAND_BCM_UMI_ECC_WRITE(numEccBytes, 11, &oobp[4],
275 (eccVal >> 16) & 0xff); /* ECC10 */
276
277 /* Always Skip BI in oob[5] */
278 } else {
279 /* Always Skip BI in oob[0] */
280
281 /* Now fill in the ECC bytes */
282 if (numEccBytes >= 13)
Arnd Bergmann878040e2012-04-25 16:44:23 +0000283 eccVal = readl(&REG_UMI_BCH_WR_ECC_3);
Leo (Hao) Chen266dead2009-10-09 19:13:08 -0700284
285 /* Usually skip CM in oob[1,2] */
286 NAND_BCM_UMI_ECC_WRITE(numEccBytes, 15, &oobp[1],
287 (eccVal >> 16) & 0xff);
288 NAND_BCM_UMI_ECC_WRITE(numEccBytes, 14, &oobp[2],
289 (eccVal >> 8) & 0xff);
290
291 /* Write ECC in oob[3-15] */
292 NAND_BCM_UMI_ECC_WRITE(numEccBytes, 13, &oobp[3],
293 eccVal & 0xff); /* ECC12 */
294
295 if (numEccBytes >= 9)
Arnd Bergmann878040e2012-04-25 16:44:23 +0000296 eccVal = readl(&REG_UMI_BCH_WR_ECC_2);
Leo (Hao) Chen266dead2009-10-09 19:13:08 -0700297
298 NAND_BCM_UMI_ECC_WRITE(numEccBytes, 12, &oobp[4],
299 (eccVal >> 24) & 0xff); /* ECC11 */
300 NAND_BCM_UMI_ECC_WRITE(numEccBytes, 11, &oobp[5],
301 (eccVal >> 16) & 0xff); /* ECC10 */
302 }
303
304 /* Fill in the remainder of ECC locations */
305 NAND_BCM_UMI_ECC_WRITE(numEccBytes, 10, &oobp[6],
306 (eccVal >> 8) & 0xff); /* ECC9 */
307 NAND_BCM_UMI_ECC_WRITE(numEccBytes, 9, &oobp[7],
308 eccVal & 0xff); /* ECC8 */
309
310 if (numEccBytes >= 5)
Arnd Bergmann878040e2012-04-25 16:44:23 +0000311 eccVal = readl(&REG_UMI_BCH_WR_ECC_1);
Leo (Hao) Chen266dead2009-10-09 19:13:08 -0700312
313 NAND_BCM_UMI_ECC_WRITE(numEccBytes, 8, &oobp[8],
314 (eccVal >> 24) & 0xff); /* ECC7 */
315 NAND_BCM_UMI_ECC_WRITE(numEccBytes, 7, &oobp[9],
316 (eccVal >> 16) & 0xff); /* ECC6 */
317 NAND_BCM_UMI_ECC_WRITE(numEccBytes, 6, &oobp[10],
318 (eccVal >> 8) & 0xff); /* ECC5 */
319 NAND_BCM_UMI_ECC_WRITE(numEccBytes, 5, &oobp[11],
320 eccVal & 0xff); /* ECC4 */
321
322 if (numEccBytes >= 1)
Arnd Bergmann878040e2012-04-25 16:44:23 +0000323 eccVal = readl(&REG_UMI_BCH_WR_ECC_0);
Leo (Hao) Chen266dead2009-10-09 19:13:08 -0700324
325 NAND_BCM_UMI_ECC_WRITE(numEccBytes, 4, &oobp[12],
326 (eccVal >> 24) & 0xff); /* ECC3 */
327 NAND_BCM_UMI_ECC_WRITE(numEccBytes, 3, &oobp[13],
328 (eccVal >> 16) & 0xff); /* ECC2 */
329 NAND_BCM_UMI_ECC_WRITE(numEccBytes, 2, &oobp[14],
330 (eccVal >> 8) & 0xff); /* ECC1 */
331 NAND_BCM_UMI_ECC_WRITE(numEccBytes, 1, &oobp[15],
332 eccVal & 0xff); /* ECC0 */
333}
334#endif
335
336#endif /* NAND_BCM_UMI_H */