123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150 |
- /*****************************************************************************
- * Copyright 2004 - 2009 Broadcom Corporation. All rights reserved.
- *
- * Unless you and Broadcom execute a separate written software license
- * agreement governing use of this software, this software is licensed to you
- * under the terms of the GNU General Public License version 2, available at
- * http://www.broadcom.com/licenses/GPLv2.php (the "GPL").
- *
- * Notwithstanding the above, under no circumstances may you combine this
- * software in any way with any other Broadcom software provided under a
- * license other than the GPL, without Broadcom's express prior written
- * consent.
- *****************************************************************************/
- /* ---- Include Files ---------------------------------------------------- */
- #include <mach/reg_umi.h>
- #include "nand_bcm_umi.h"
- #ifdef BOOT0_BUILD
- #include <uart.h>
- #endif
- /* ---- External Variable Declarations ----------------------------------- */
- /* ---- External Function Prototypes ------------------------------------- */
- /* ---- Public Variables ------------------------------------------------- */
- /* ---- Private Constants and Types -------------------------------------- */
- /* ---- Private Function Prototypes -------------------------------------- */
- /* ---- Private Variables ------------------------------------------------ */
- /* ---- Private Functions ------------------------------------------------ */
- #if NAND_ECC_BCH
- /****************************************************************************
- * nand_bch_ecc_flip_bit - Routine to flip an errored bit
- *
- * PURPOSE:
- * This is a helper routine that flips the bit (0 -> 1 or 1 -> 0) of the
- * errored bit specified
- *
- * PARAMETERS:
- * datap - Container that holds the 512 byte data
- * errorLocation - Location of the bit that needs to be flipped
- *
- * RETURNS:
- * None
- ****************************************************************************/
- static void nand_bcm_umi_bch_ecc_flip_bit(uint8_t *datap, int errorLocation)
- {
- int locWithinAByte = (errorLocation & REG_UMI_BCH_ERR_LOC_BYTE) >> 0;
- int locWithinAWord = (errorLocation & REG_UMI_BCH_ERR_LOC_WORD) >> 3;
- int locWithinAPage = (errorLocation & REG_UMI_BCH_ERR_LOC_PAGE) >> 5;
- uint8_t errorByte = 0;
- uint8_t byteMask = 1 << locWithinAByte;
- /* BCH uses big endian, need to change the location
- * bits to little endian */
- locWithinAWord = 3 - locWithinAWord;
- errorByte = datap[locWithinAPage * sizeof(uint32_t) + locWithinAWord];
- #ifdef BOOT0_BUILD
- puthexs("\nECC Correct Offset: ",
- locWithinAPage * sizeof(uint32_t) + locWithinAWord);
- puthexs(" errorByte:", errorByte);
- puthex8(" Bit: ", locWithinAByte);
- #endif
- if (errorByte & byteMask) {
- /* bit needs to be cleared */
- errorByte &= ~byteMask;
- } else {
- /* bit needs to be set */
- errorByte |= byteMask;
- }
- /* write back the value with the fixed bit */
- datap[locWithinAPage * sizeof(uint32_t) + locWithinAWord] = errorByte;
- }
- /****************************************************************************
- * nand_correct_page_bch - Routine to correct bit errors when reading NAND
- *
- * PURPOSE:
- * This routine reads the BCH registers to determine if there are any bit
- * errors during the read of the last 512 bytes of data + ECC bytes. If
- * errors exists, the routine fixes it.
- *
- * PARAMETERS:
- * datap - Container that holds the 512 byte data
- *
- * RETURNS:
- * 0 or greater = Number of errors corrected
- * (No errors are found or errors have been fixed)
- * -1 = Error(s) cannot be fixed
- ****************************************************************************/
- int nand_bcm_umi_bch_correct_page(uint8_t *datap, uint8_t *readEccData,
- int numEccBytes)
- {
- int numErrors;
- int errorLocation;
- int idx;
- uint32_t regValue;
- /* wait for read ECC to be valid */
- regValue = nand_bcm_umi_bch_poll_read_ecc_calc();
- /*
- * read the control status register to determine if there
- * are error'ed bits
- * see if errors are correctible
- */
- if ((regValue & REG_UMI_BCH_CTRL_STATUS_UNCORR_ERR) > 0) {
- int i;
- for (i = 0; i < numEccBytes; i++) {
- if (readEccData[i] != 0xff) {
- /* errors cannot be fixed, return -1 */
- return -1;
- }
- }
- /* If ECC is unprogrammed then we can't correct,
- * assume everything OK */
- return 0;
- }
- if ((regValue & REG_UMI_BCH_CTRL_STATUS_CORR_ERR) == 0) {
- /* no errors */
- return 0;
- }
- /*
- * Fix errored bits by doing the following:
- * 1. Read the number of errors in the control and status register
- * 2. Read the error location registers that corresponds to the number
- * of errors reported
- * 3. Invert the bit in the data
- */
- numErrors = (regValue & REG_UMI_BCH_CTRL_STATUS_NB_CORR_ERROR) >> 20;
- for (idx = 0; idx < numErrors; idx++) {
- errorLocation =
- REG_UMI_BCH_ERR_LOC_ADDR(idx) & REG_UMI_BCH_ERR_LOC_MASK;
- /* Flip bit */
- nand_bcm_umi_bch_ecc_flip_bit(datap, errorLocation);
- }
- /* Errors corrected */
- return numErrors;
- }
- #endif
|