1 /*****************************************************************************
2 * Copyright 2003 - 2009 Broadcom Corporation. All rights reserved.
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").
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
13 *****************************************************************************/
14 #ifndef NAND_BCM_UMI_H
15 #define NAND_BCM_UMI_H
17 /* ---- Include Files ---------------------------------------------------- */
18 #include <mach/reg_umi.h>
19 #include <mach/reg_nand.h>
20 #include <cfg_global.h>
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)
26 #define NAND_ECC_BCH 0
29 #define CFG_GLOBAL_NAND_ECC_BCH_NUM_BYTES 13
33 #define NAND_ECC_NUM_BYTES 13
35 #define NAND_ECC_NUM_BYTES CFG_GLOBAL_NAND_ECC_BCH_NUM_BYTES
38 #define NAND_ECC_NUM_BYTES 3
41 #define NAND_DATA_ACCESS_SIZE 512
43 /* ---- Variable Externs ------------------------------------------ */
44 /* ---- Function Prototypes --------------------------------------- */
45 int nand_bcm_umi_bch_correct_page(uint8_t *datap
, uint8_t *readEccData
,
48 /* Check in device is ready */
49 static inline int nand_bcm_umi_dev_ready(void)
51 return REG_UMI_NAND_RCSR
& REG_UMI_NAND_RCSR_RDY
;
54 /* Wait until device is ready */
55 static inline void nand_bcm_umi_wait_till_ready(void)
57 while (nand_bcm_umi_dev_ready() == 0)
61 /* Enable Hamming ECC */
62 static inline void nand_bcm_umi_hamming_enable_hwecc(void)
64 /* disable and reset ECC, 512 byte page */
65 REG_UMI_NAND_ECC_CSR
&= ~(REG_UMI_NAND_ECC_CSR_ECC_ENABLE
|
66 REG_UMI_NAND_ECC_CSR_256BYTE
);
68 REG_UMI_NAND_ECC_CSR
|= REG_UMI_NAND_ECC_CSR_ECC_ENABLE
;
72 /* BCH ECC specifics */
73 #define ECC_BITS_PER_CORRECTABLE_BIT 13
75 /* Enable BCH Read ECC */
76 static inline void nand_bcm_umi_bch_enable_read_hwecc(void)
78 /* disable and reset ECC */
79 REG_UMI_BCH_CTRL_STATUS
= REG_UMI_BCH_CTRL_STATUS_RD_ECC_VALID
;
81 REG_UMI_BCH_CTRL_STATUS
= REG_UMI_BCH_CTRL_STATUS_ECC_RD_EN
;
84 /* Enable BCH Write ECC */
85 static inline void nand_bcm_umi_bch_enable_write_hwecc(void)
87 /* disable and reset ECC */
88 REG_UMI_BCH_CTRL_STATUS
= REG_UMI_BCH_CTRL_STATUS_WR_ECC_VALID
;
90 REG_UMI_BCH_CTRL_STATUS
= REG_UMI_BCH_CTRL_STATUS_ECC_WR_EN
;
93 /* Config number of BCH ECC bytes */
94 static inline void nand_bcm_umi_bch_config_ecc(uint8_t numEccBytes
)
99 uint32_t numBits
= numEccBytes
* 8;
101 /* disable and reset ECC */
102 REG_UMI_BCH_CTRL_STATUS
=
103 REG_UMI_BCH_CTRL_STATUS_WR_ECC_VALID
|
104 REG_UMI_BCH_CTRL_STATUS_RD_ECC_VALID
;
106 /* Every correctible bit requires 13 ECC bits */
107 tValue
= (uint32_t) (numBits
/ ECC_BITS_PER_CORRECTABLE_BIT
);
109 /* Total data in number of bits for generating and computing BCH ECC */
110 nValue
= (NAND_DATA_ACCESS_SIZE
+ numEccBytes
) * 8;
112 /* K parameter is used internally. K = N - (T * 13) */
113 kValue
= nValue
- (tValue
* ECC_BITS_PER_CORRECTABLE_BIT
);
115 /* Write the settings */
116 REG_UMI_BCH_N
= nValue
;
117 REG_UMI_BCH_T
= tValue
;
118 REG_UMI_BCH_K
= kValue
;
121 /* Pause during ECC read calculation to skip bytes in OOB */
122 static inline void nand_bcm_umi_bch_pause_read_ecc_calc(void)
124 REG_UMI_BCH_CTRL_STATUS
=
125 REG_UMI_BCH_CTRL_STATUS_ECC_RD_EN
|
126 REG_UMI_BCH_CTRL_STATUS_PAUSE_ECC_DEC
;
129 /* Resume during ECC read calculation after skipping bytes in OOB */
130 static inline void nand_bcm_umi_bch_resume_read_ecc_calc(void)
132 REG_UMI_BCH_CTRL_STATUS
= REG_UMI_BCH_CTRL_STATUS_ECC_RD_EN
;
135 /* Poll read ECC calc to check when hardware completes */
136 static inline uint32_t nand_bcm_umi_bch_poll_read_ecc_calc(void)
141 /* wait for ECC to be valid */
142 regVal
= REG_UMI_BCH_CTRL_STATUS
;
143 } while ((regVal
& REG_UMI_BCH_CTRL_STATUS_RD_ECC_VALID
) == 0);
148 /* Poll write ECC calc to check when hardware completes */
149 static inline void nand_bcm_umi_bch_poll_write_ecc_calc(void)
151 /* wait for ECC to be valid */
152 while ((REG_UMI_BCH_CTRL_STATUS
& REG_UMI_BCH_CTRL_STATUS_WR_ECC_VALID
)
157 /* Read the OOB and ECC, for kernel write OOB to a buffer */
158 #if defined(__KERNEL__) && !defined(STANDALONE)
159 static inline void nand_bcm_umi_bch_read_oobEcc(uint32_t pageSize
,
160 uint8_t *eccCalc
, int numEccBytes
, uint8_t *oobp
)
162 static inline void nand_bcm_umi_bch_read_oobEcc(uint32_t pageSize
,
163 uint8_t *eccCalc
, int numEccBytes
)
167 int numToRead
= 16; /* There are 16 bytes per sector in the OOB */
169 /* ECC is already paused when this function is called */
171 if (pageSize
== NAND_DATA_ACCESS_SIZE
) {
172 while (numToRead
> numEccBytes
) {
173 /* skip free oob region */
174 #if defined(__KERNEL__) && !defined(STANDALONE)
175 *oobp
++ = REG_NAND_DATA8
;
182 /* read ECC bytes before BI */
183 nand_bcm_umi_bch_resume_read_ecc_calc();
185 while (numToRead
> 11) {
186 #if defined(__KERNEL__) && !defined(STANDALONE)
187 *oobp
= REG_NAND_DATA8
;
188 eccCalc
[eccPos
++] = *oobp
;
191 eccCalc
[eccPos
++] = REG_NAND_DATA8
;
195 nand_bcm_umi_bch_pause_read_ecc_calc();
197 if (numToRead
== 11) {
199 #if defined(__KERNEL__) && !defined(STANDALONE)
200 *oobp
++ = REG_NAND_DATA8
;
208 nand_bcm_umi_bch_resume_read_ecc_calc();
210 #if defined(__KERNEL__) && !defined(STANDALONE)
211 *oobp
= REG_NAND_DATA8
;
212 eccCalc
[eccPos
++] = *oobp
;
215 eccCalc
[eccPos
++] = REG_NAND_DATA8
;
221 #if defined(__KERNEL__) && !defined(STANDALONE)
222 *oobp
++ = REG_NAND_DATA8
;
228 while (numToRead
> numEccBytes
) {
229 /* skip free oob region */
230 #if defined(__KERNEL__) && !defined(STANDALONE)
231 *oobp
++ = REG_NAND_DATA8
;
239 nand_bcm_umi_bch_resume_read_ecc_calc();
241 #if defined(__KERNEL__) && !defined(STANDALONE)
242 *oobp
= REG_NAND_DATA8
;
243 eccCalc
[eccPos
++] = *oobp
;
246 eccCalc
[eccPos
++] = REG_NAND_DATA8
;
253 /* Helper function to write ECC */
254 static inline void NAND_BCM_UMI_ECC_WRITE(int numEccBytes
, int eccBytePos
,
255 uint8_t *oobp
, uint8_t eccVal
)
257 if (eccBytePos
<= numEccBytes
)
261 /* Write OOB with ECC */
262 static inline void nand_bcm_umi_bch_write_oobEcc(uint32_t pageSize
,
263 uint8_t *oobp
, int numEccBytes
)
265 uint32_t eccVal
= 0xffffffff;
267 /* wait for write ECC to be valid */
268 nand_bcm_umi_bch_poll_write_ecc_calc();
271 ** Get the hardware ecc from the 32-bit result registers.
272 ** Read after 512 byte accesses. Format B3B2B1B0
273 ** where B3 = ecc3, etc.
276 if (pageSize
== NAND_DATA_ACCESS_SIZE
) {
277 /* Now fill in the ECC bytes */
278 if (numEccBytes
>= 13)
279 eccVal
= REG_UMI_BCH_WR_ECC_3
;
281 /* Usually we skip CM in oob[0,1] */
282 NAND_BCM_UMI_ECC_WRITE(numEccBytes
, 15, &oobp
[0],
283 (eccVal
>> 16) & 0xff);
284 NAND_BCM_UMI_ECC_WRITE(numEccBytes
, 14, &oobp
[1],
285 (eccVal
>> 8) & 0xff);
287 /* Write ECC in oob[2,3,4] */
288 NAND_BCM_UMI_ECC_WRITE(numEccBytes
, 13, &oobp
[2],
289 eccVal
& 0xff); /* ECC 12 */
291 if (numEccBytes
>= 9)
292 eccVal
= REG_UMI_BCH_WR_ECC_2
;
294 NAND_BCM_UMI_ECC_WRITE(numEccBytes
, 12, &oobp
[3],
295 (eccVal
>> 24) & 0xff); /* ECC11 */
296 NAND_BCM_UMI_ECC_WRITE(numEccBytes
, 11, &oobp
[4],
297 (eccVal
>> 16) & 0xff); /* ECC10 */
299 /* Always Skip BI in oob[5] */
301 /* Always Skip BI in oob[0] */
303 /* Now fill in the ECC bytes */
304 if (numEccBytes
>= 13)
305 eccVal
= REG_UMI_BCH_WR_ECC_3
;
307 /* Usually skip CM in oob[1,2] */
308 NAND_BCM_UMI_ECC_WRITE(numEccBytes
, 15, &oobp
[1],
309 (eccVal
>> 16) & 0xff);
310 NAND_BCM_UMI_ECC_WRITE(numEccBytes
, 14, &oobp
[2],
311 (eccVal
>> 8) & 0xff);
313 /* Write ECC in oob[3-15] */
314 NAND_BCM_UMI_ECC_WRITE(numEccBytes
, 13, &oobp
[3],
315 eccVal
& 0xff); /* ECC12 */
317 if (numEccBytes
>= 9)
318 eccVal
= REG_UMI_BCH_WR_ECC_2
;
320 NAND_BCM_UMI_ECC_WRITE(numEccBytes
, 12, &oobp
[4],
321 (eccVal
>> 24) & 0xff); /* ECC11 */
322 NAND_BCM_UMI_ECC_WRITE(numEccBytes
, 11, &oobp
[5],
323 (eccVal
>> 16) & 0xff); /* ECC10 */
326 /* Fill in the remainder of ECC locations */
327 NAND_BCM_UMI_ECC_WRITE(numEccBytes
, 10, &oobp
[6],
328 (eccVal
>> 8) & 0xff); /* ECC9 */
329 NAND_BCM_UMI_ECC_WRITE(numEccBytes
, 9, &oobp
[7],
330 eccVal
& 0xff); /* ECC8 */
332 if (numEccBytes
>= 5)
333 eccVal
= REG_UMI_BCH_WR_ECC_1
;
335 NAND_BCM_UMI_ECC_WRITE(numEccBytes
, 8, &oobp
[8],
336 (eccVal
>> 24) & 0xff); /* ECC7 */
337 NAND_BCM_UMI_ECC_WRITE(numEccBytes
, 7, &oobp
[9],
338 (eccVal
>> 16) & 0xff); /* ECC6 */
339 NAND_BCM_UMI_ECC_WRITE(numEccBytes
, 6, &oobp
[10],
340 (eccVal
>> 8) & 0xff); /* ECC5 */
341 NAND_BCM_UMI_ECC_WRITE(numEccBytes
, 5, &oobp
[11],
342 eccVal
& 0xff); /* ECC4 */
344 if (numEccBytes
>= 1)
345 eccVal
= REG_UMI_BCH_WR_ECC_0
;
347 NAND_BCM_UMI_ECC_WRITE(numEccBytes
, 4, &oobp
[12],
348 (eccVal
>> 24) & 0xff); /* ECC3 */
349 NAND_BCM_UMI_ECC_WRITE(numEccBytes
, 3, &oobp
[13],
350 (eccVal
>> 16) & 0xff); /* ECC2 */
351 NAND_BCM_UMI_ECC_WRITE(numEccBytes
, 2, &oobp
[14],
352 (eccVal
>> 8) & 0xff); /* ECC1 */
353 NAND_BCM_UMI_ECC_WRITE(numEccBytes
, 1, &oobp
[15],
354 eccVal
& 0xff); /* ECC0 */
358 #endif /* NAND_BCM_UMI_H */