diff options
| author | Brian Norris <computersforpeace@gmail.com> | 2012-09-24 23:40:49 -0400 |
|---|---|---|
| committer | David Woodhouse <David.Woodhouse@intel.com> | 2012-09-29 10:57:33 -0400 |
| commit | 7e74c2d7141e8929049233e28c74cd089f6ae962 (patch) | |
| tree | d811df16199367e4ebba1d6a6135cc9783e87c1e /drivers | |
| parent | 4aef9b78de057349ad9d620851b14800af0b962c (diff) | |
mtd: nand: split BB marker options decoding into its own function
When detecting NAND parameters, the code gets a little ugly so that the
logic is obscured. Try to remedy that by moving code to separate functions
that have well-defined purposes.
This patch splits the bad block marker options detection into its own function,
away from the other parameters (e.g., chip size, page size, etc.).
Signed-off-by: Brian Norris <computersforpeace@gmail.com>
Signed-off-by: Artem Bityutskiy <artem.bityutskiy@linux.intel.com>
Signed-off-by: David Woodhouse <David.Woodhouse@intel.com>
Diffstat (limited to 'drivers')
| -rw-r--r-- | drivers/mtd/nand/nand_base.c | 66 |
1 files changed, 39 insertions, 27 deletions
diff --git a/drivers/mtd/nand/nand_base.c b/drivers/mtd/nand/nand_base.c index 77b340068c6e..16bb17f93997 100644 --- a/drivers/mtd/nand/nand_base.c +++ b/drivers/mtd/nand/nand_base.c | |||
| @@ -2906,6 +2906,43 @@ static int nand_flash_detect_onfi(struct mtd_info *mtd, struct nand_chip *chip, | |||
| 2906 | } | 2906 | } |
| 2907 | 2907 | ||
| 2908 | /* | 2908 | /* |
| 2909 | * Set the bad block marker/indicator (BBM/BBI) patterns according to some | ||
| 2910 | * heuristic patterns using various detected parameters (e.g., manufacturer, | ||
| 2911 | * page size, cell-type information). | ||
| 2912 | */ | ||
| 2913 | static void nand_decode_bbm_options(struct mtd_info *mtd, | ||
| 2914 | struct nand_chip *chip, u8 id_data[8]) | ||
| 2915 | { | ||
| 2916 | int maf_id = id_data[0]; | ||
| 2917 | |||
| 2918 | /* Set the bad block position */ | ||
| 2919 | if (mtd->writesize > 512 || (chip->options & NAND_BUSWIDTH_16)) | ||
| 2920 | chip->badblockpos = NAND_LARGE_BADBLOCK_POS; | ||
| 2921 | else | ||
| 2922 | chip->badblockpos = NAND_SMALL_BADBLOCK_POS; | ||
| 2923 | |||
| 2924 | /* | ||
| 2925 | * Bad block marker is stored in the last page of each block on Samsung | ||
| 2926 | * and Hynix MLC devices; stored in first two pages of each block on | ||
| 2927 | * Micron devices with 2KiB pages and on SLC Samsung, Hynix, Toshiba, | ||
| 2928 | * AMD/Spansion, and Macronix. All others scan only the first page. | ||
| 2929 | */ | ||
| 2930 | if ((chip->cellinfo & NAND_CI_CELLTYPE_MSK) && | ||
| 2931 | (maf_id == NAND_MFR_SAMSUNG || | ||
| 2932 | maf_id == NAND_MFR_HYNIX)) | ||
| 2933 | chip->bbt_options |= NAND_BBT_SCANLASTPAGE; | ||
| 2934 | else if ((!(chip->cellinfo & NAND_CI_CELLTYPE_MSK) && | ||
| 2935 | (maf_id == NAND_MFR_SAMSUNG || | ||
| 2936 | maf_id == NAND_MFR_HYNIX || | ||
| 2937 | maf_id == NAND_MFR_TOSHIBA || | ||
| 2938 | maf_id == NAND_MFR_AMD || | ||
| 2939 | maf_id == NAND_MFR_MACRONIX)) || | ||
| 2940 | (mtd->writesize == 2048 && | ||
| 2941 | maf_id == NAND_MFR_MICRON)) | ||
| 2942 | chip->bbt_options |= NAND_BBT_SCAN2NDPAGE; | ||
| 2943 | } | ||
| 2944 | |||
| 2945 | /* | ||
| 2909 | * Get the flash and manufacturer id and lookup if the type is supported. | 2946 | * Get the flash and manufacturer id and lookup if the type is supported. |
| 2910 | */ | 2947 | */ |
| 2911 | static struct nand_flash_dev *nand_get_flash_type(struct mtd_info *mtd, | 2948 | static struct nand_flash_dev *nand_get_flash_type(struct mtd_info *mtd, |
| @@ -3087,6 +3124,8 @@ ident_done: | |||
| 3087 | return ERR_PTR(-EINVAL); | 3124 | return ERR_PTR(-EINVAL); |
| 3088 | } | 3125 | } |
| 3089 | 3126 | ||
| 3127 | nand_decode_bbm_options(mtd, chip, id_data); | ||
| 3128 | |||
| 3090 | /* Calculate the address shift from the page size */ | 3129 | /* Calculate the address shift from the page size */ |
| 3091 | chip->page_shift = ffs(mtd->writesize) - 1; | 3130 | chip->page_shift = ffs(mtd->writesize) - 1; |
| 3092 | /* Convert chipsize to number of pages per chip -1 */ | 3131 | /* Convert chipsize to number of pages per chip -1 */ |
| @@ -3103,33 +3142,6 @@ ident_done: | |||
| 3103 | 3142 | ||
| 3104 | chip->badblockbits = 8; | 3143 | chip->badblockbits = 8; |
| 3105 | 3144 | ||
| 3106 | /* Set the bad block position */ | ||
| 3107 | if (mtd->writesize > 512 || (busw & NAND_BUSWIDTH_16)) | ||
| 3108 | chip->badblockpos = NAND_LARGE_BADBLOCK_POS; | ||
| 3109 | else | ||
| 3110 | chip->badblockpos = NAND_SMALL_BADBLOCK_POS; | ||
| 3111 | |||
| 3112 | /* | ||
| 3113 | * Bad block marker is stored in the last page of each block | ||
| 3114 | * on Samsung and Hynix MLC devices; stored in first two pages | ||
| 3115 | * of each block on Micron devices with 2KiB pages and on | ||
| 3116 | * SLC Samsung, Hynix, Toshiba, AMD/Spansion, and Macronix. | ||
| 3117 | * All others scan only the first page. | ||
| 3118 | */ | ||
| 3119 | if ((chip->cellinfo & NAND_CI_CELLTYPE_MSK) && | ||
| 3120 | (*maf_id == NAND_MFR_SAMSUNG || | ||
| 3121 | *maf_id == NAND_MFR_HYNIX)) | ||
| 3122 | chip->bbt_options |= NAND_BBT_SCANLASTPAGE; | ||
| 3123 | else if ((!(chip->cellinfo & NAND_CI_CELLTYPE_MSK) && | ||
| 3124 | (*maf_id == NAND_MFR_SAMSUNG || | ||
| 3125 | *maf_id == NAND_MFR_HYNIX || | ||
| 3126 | *maf_id == NAND_MFR_TOSHIBA || | ||
| 3127 | *maf_id == NAND_MFR_AMD || | ||
| 3128 | *maf_id == NAND_MFR_MACRONIX)) || | ||
| 3129 | (mtd->writesize == 2048 && | ||
| 3130 | *maf_id == NAND_MFR_MICRON)) | ||
| 3131 | chip->bbt_options |= NAND_BBT_SCAN2NDPAGE; | ||
| 3132 | |||
| 3133 | /* Check for AND chips with 4 page planes */ | 3145 | /* Check for AND chips with 4 page planes */ |
| 3134 | if (chip->options & NAND_4PAGE_ARRAY) | 3146 | if (chip->options & NAND_4PAGE_ARRAY) |
| 3135 | chip->erase_cmd = multi_erase_cmd; | 3147 | chip->erase_cmd = multi_erase_cmd; |
