Commit 55750756 authored by Romain Izard's avatar Romain Izard Committed by Brian Norris

mtd: atmel_nand: Support PMECC on SAMA5D2

Starting with the SAMA5D2, there is a new revision of the Atmel PMECC
controller that can correct 32 bits in each sector. This controller is
not 100% compatible with the previous revision that corrected a maximum
of 24 bits by sector, as some register addresses overlap.

Using information from the device tree, we can configure the driver to
work with both versions.

For the binding:
Acked-by: default avatarRob Herring <robh@kernel.org>
Tested-by: default avatarWenyou Yang <wenyou.yang@atmel.com>
Reviewed-by: default avatarBoris Brezillon <boris.brezillon@free-electrons.com>
Signed-off-by: default avatarRomain Izard <romain.izard.pro@gmail.com>
Signed-off-by: default avatarBrian Norris <computersforpeace@gmail.com>
parent ec4ee5fb
Atmel NAND flash Atmel NAND flash
Required properties: Required properties:
- compatible : should be "atmel,at91rm9200-nand" or "atmel,sama5d4-nand". - compatible: The possible values are:
"atmel,at91rm9200-nand"
"atmel,sama5d2-nand"
"atmel,sama5d4-nand"
- reg : should specify localbus address and size used for the chip, - reg : should specify localbus address and size used for the chip,
and hardware ECC controller if available. and hardware ECC controller if available.
If the hardware ECC is PMECC, it should contain address and size for If the hardware ECC is PMECC, it should contain address and size for
......
...@@ -65,6 +65,7 @@ module_param(on_flash_bbt, int, 0); ...@@ -65,6 +65,7 @@ module_param(on_flash_bbt, int, 0);
struct atmel_nand_caps { struct atmel_nand_caps {
bool pmecc_correct_erase_page; bool pmecc_correct_erase_page;
uint8_t pmecc_max_correction;
}; };
struct atmel_nand_nfc_caps { struct atmel_nand_nfc_caps {
...@@ -145,6 +146,7 @@ struct atmel_nand_host { ...@@ -145,6 +146,7 @@ struct atmel_nand_host {
int pmecc_cw_len; /* Length of codeword */ int pmecc_cw_len; /* Length of codeword */
void __iomem *pmerrloc_base; void __iomem *pmerrloc_base;
void __iomem *pmerrloc_el_base;
void __iomem *pmecc_rom_base; void __iomem *pmecc_rom_base;
/* lookup table for alpha_to and index_of */ /* lookup table for alpha_to and index_of */
...@@ -818,7 +820,7 @@ static void pmecc_correct_data(struct mtd_info *mtd, uint8_t *buf, uint8_t *ecc, ...@@ -818,7 +820,7 @@ static void pmecc_correct_data(struct mtd_info *mtd, uint8_t *buf, uint8_t *ecc,
sector_size = host->pmecc_sector_size; sector_size = host->pmecc_sector_size;
while (err_nbr) { while (err_nbr) {
tmp = pmerrloc_readl_el_relaxed(host->pmerrloc_base, i) - 1; tmp = pmerrloc_readl_el_relaxed(host->pmerrloc_el_base, i) - 1;
byte_pos = tmp / 8; byte_pos = tmp / 8;
bit_pos = tmp % 8; bit_pos = tmp % 8;
...@@ -1210,6 +1212,8 @@ static int atmel_pmecc_nand_init_params(struct platform_device *pdev, ...@@ -1210,6 +1212,8 @@ static int atmel_pmecc_nand_init_params(struct platform_device *pdev,
err_no = PTR_ERR(host->pmerrloc_base); err_no = PTR_ERR(host->pmerrloc_base);
goto err; goto err;
} }
host->pmerrloc_el_base = host->pmerrloc_base + ATMEL_PMERRLOC_SIGMAx +
(host->caps->pmecc_max_correction + 1) * 4;
if (!host->has_no_lookup_table) { if (!host->has_no_lookup_table) {
regs_rom = platform_get_resource(pdev, IORESOURCE_MEM, 3); regs_rom = platform_get_resource(pdev, IORESOURCE_MEM, 3);
...@@ -2307,17 +2311,34 @@ static int atmel_nand_remove(struct platform_device *pdev) ...@@ -2307,17 +2311,34 @@ static int atmel_nand_remove(struct platform_device *pdev)
return 0; return 0;
} }
/*
* AT91RM9200 does not have PMECC or PMECC Errloc peripherals for
* BCH ECC. Combined with the "atmel,has-pmecc", it is used to describe
* devices from the SAM9 family that have those.
*/
static const struct atmel_nand_caps at91rm9200_caps = { static const struct atmel_nand_caps at91rm9200_caps = {
.pmecc_correct_erase_page = false, .pmecc_correct_erase_page = false,
.pmecc_max_correction = 24,
}; };
static const struct atmel_nand_caps sama5d4_caps = { static const struct atmel_nand_caps sama5d4_caps = {
.pmecc_correct_erase_page = true, .pmecc_correct_erase_page = true,
.pmecc_max_correction = 24,
};
/*
* The PMECC Errloc controller starting in SAMA5D2 is not compatible,
* as the increased correction strength requires more registers.
*/
static const struct atmel_nand_caps sama5d2_caps = {
.pmecc_correct_erase_page = true,
.pmecc_max_correction = 32,
}; };
static const struct of_device_id atmel_nand_dt_ids[] = { static const struct of_device_id atmel_nand_dt_ids[] = {
{ .compatible = "atmel,at91rm9200-nand", .data = &at91rm9200_caps }, { .compatible = "atmel,at91rm9200-nand", .data = &at91rm9200_caps },
{ .compatible = "atmel,sama5d4-nand", .data = &sama5d4_caps }, { .compatible = "atmel,sama5d4-nand", .data = &sama5d4_caps },
{ .compatible = "atmel,sama5d2-nand", .data = &sama5d2_caps },
{ /* sentinel */ } { /* sentinel */ }
}; };
......
...@@ -108,7 +108,11 @@ ...@@ -108,7 +108,11 @@
#define PMERRLOC_ERR_NUM_MASK (0x1f << 8) #define PMERRLOC_ERR_NUM_MASK (0x1f << 8)
#define PMERRLOC_CALC_DONE (1 << 0) #define PMERRLOC_CALC_DONE (1 << 0)
#define ATMEL_PMERRLOC_SIGMAx 0x028 /* Error location SIGMA x */ #define ATMEL_PMERRLOC_SIGMAx 0x028 /* Error location SIGMA x */
#define ATMEL_PMERRLOC_ELx 0x08c /* Error location x */
/*
* The ATMEL_PMERRLOC_ELx register location depends from the number of
* bits corrected by the PMECC controller. Do not use it.
*/
/* Register access macros for PMECC */ /* Register access macros for PMECC */
#define pmecc_readl_relaxed(addr, reg) \ #define pmecc_readl_relaxed(addr, reg) \
...@@ -136,7 +140,7 @@ ...@@ -136,7 +140,7 @@
readl_relaxed((addr) + ATMEL_PMERRLOC_SIGMAx + ((n) * 4)) readl_relaxed((addr) + ATMEL_PMERRLOC_SIGMAx + ((n) * 4))
#define pmerrloc_readl_el_relaxed(addr, n) \ #define pmerrloc_readl_el_relaxed(addr, n) \
readl_relaxed((addr) + ATMEL_PMERRLOC_ELx + ((n) * 4)) readl_relaxed((addr) + ((n) * 4))
/* Galois field dimension */ /* Galois field dimension */
#define PMECC_GF_DIMENSION_13 13 #define PMECC_GF_DIMENSION_13 13
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment