mirror of
https://github.com/openwrt/openwrt.git
synced 2024-12-30 18:47:06 +00:00
b9d58f7e06
generic: Add/rename patches for upstream consistency ipq40xx: generic-level patch replaces same-source patches-4.19/ 082-v4.20-mtd-spinand-winbond-Add-support-for-W25N01GV.patch The SPI-NAND framework from Linux uses common driver code that is then "tuned" by a tiny struct of chip-specific data that describes available commands, timing, and layout (data and OOB data). Several manufacturers and chips have been added since 4.19, several of which are used in devices already supported by OpenWrt (typically with no or "legacy" access to their NAND memory). This commit catches up the supported-chip definitions through Linux 5.2-rc6 and linux/next. The driver is only compiled for platforms with CONFIG_MTD_SPI_NAND=y. This presently includes ipq40xx and pistachio, with the addition of ath79-nand in these commits (and not ath79-generic or ath79-tiny). Upstream patches refreshed against 4.19.75 Build-tested-on: ipq40xx Run-tested-on: ath79-nand Signed-off-by: Jeff Kletsky <git-commits@allycomm.com>
198 lines
6.3 KiB
Diff
198 lines
6.3 KiB
Diff
|
|
IMPORTANT NOTE
|
|
==============
|
|
|
|
The content of this patch has been adapted for Linux 4.19
|
|
|
|
Changes were made in Linux 5.x to add the bad-block limit
|
|
to the metadata available to the driver, adding a parameter
|
|
|
|
NAND_MEMORG(1, 2048, 128, 64, 1024, 20, 1, 1, 1),
|
|
^- New bad-block limit
|
|
|
|
This patch omits that parameter from the upstream patch
|
|
for compatibility with the Linux 4.19 driver.
|
|
|
|
=====
|
|
|
|
From 049df13c4e63884fe6634db5568e08f65922256e Mon Sep 17 00:00:00 2001
|
|
From: Jeff Kletsky <git-commits@allycomm.com>
|
|
Date: Wed, 22 May 2019 15:05:55 -0700
|
|
Subject: [PATCH 3/3] mtd: spinand: Add support for GigaDevice GD5F1GQ4UFxxG
|
|
|
|
The GigaDevice GD5F1GQ4UFxxG SPI NAND is in current production devices
|
|
and, while it has the same logical layout as the E-series devices,
|
|
it differs in the SPI interfacing in significant ways.
|
|
|
|
This support is contingent on previous commits to:
|
|
|
|
* Add support for two-byte device IDs
|
|
* Define macros for page-read ops with three-byte addresses
|
|
|
|
http://www.gigadevice.com/datasheet/gd5f1gq4xfxxg/
|
|
|
|
Signed-off-by: Jeff Kletsky <git-commits@allycomm.com>
|
|
Reviewed-by: Frieder Schrempf <frieder.schrempf@kontron.de>
|
|
Signed-off-by: Miquel Raynal <miquel.raynal@bootlin.com>
|
|
---
|
|
drivers/mtd/nand/spi/gigadevice.c | 79 +++++++++++++++++++++++++------
|
|
1 file changed, 64 insertions(+), 15 deletions(-)
|
|
|
|
--- a/drivers/mtd/nand/spi/gigadevice.c
|
|
+++ b/drivers/mtd/nand/spi/gigadevice.c
|
|
@@ -9,11 +9,17 @@
|
|
#include <linux/mtd/spinand.h>
|
|
|
|
#define SPINAND_MFR_GIGADEVICE 0xC8
|
|
+
|
|
#define GD5FXGQ4XA_STATUS_ECC_1_7_BITFLIPS (1 << 4)
|
|
#define GD5FXGQ4XA_STATUS_ECC_8_BITFLIPS (3 << 4)
|
|
|
|
#define GD5FXGQ4UEXXG_REG_STATUS2 0xf0
|
|
|
|
+#define GD5FXGQ4UXFXXG_STATUS_ECC_MASK (7 << 4)
|
|
+#define GD5FXGQ4UXFXXG_STATUS_ECC_NO_BITFLIPS (0 << 4)
|
|
+#define GD5FXGQ4UXFXXG_STATUS_ECC_1_3_BITFLIPS (1 << 4)
|
|
+#define GD5FXGQ4UXFXXG_STATUS_ECC_UNCOR_ERROR (7 << 4)
|
|
+
|
|
static SPINAND_OP_VARIANTS(read_cache_variants,
|
|
SPINAND_PAGE_READ_FROM_CACHE_QUADIO_OP(0, 2, NULL, 0),
|
|
SPINAND_PAGE_READ_FROM_CACHE_X4_OP(0, 1, NULL, 0),
|
|
@@ -22,6 +28,14 @@ static SPINAND_OP_VARIANTS(read_cache_va
|
|
SPINAND_PAGE_READ_FROM_CACHE_OP(true, 0, 1, NULL, 0),
|
|
SPINAND_PAGE_READ_FROM_CACHE_OP(false, 0, 1, NULL, 0));
|
|
|
|
+static SPINAND_OP_VARIANTS(read_cache_variants_f,
|
|
+ SPINAND_PAGE_READ_FROM_CACHE_QUADIO_OP(0, 2, NULL, 0),
|
|
+ SPINAND_PAGE_READ_FROM_CACHE_X4_OP_3A(0, 1, NULL, 0),
|
|
+ SPINAND_PAGE_READ_FROM_CACHE_DUALIO_OP(0, 1, NULL, 0),
|
|
+ SPINAND_PAGE_READ_FROM_CACHE_X2_OP_3A(0, 1, NULL, 0),
|
|
+ SPINAND_PAGE_READ_FROM_CACHE_OP_3A(true, 0, 1, NULL, 0),
|
|
+ SPINAND_PAGE_READ_FROM_CACHE_OP_3A(false, 0, 0, NULL, 0));
|
|
+
|
|
static SPINAND_OP_VARIANTS(write_cache_variants,
|
|
SPINAND_PROG_LOAD_X4(true, 0, NULL, 0),
|
|
SPINAND_PROG_LOAD(true, 0, NULL, 0));
|
|
@@ -59,6 +73,11 @@ static int gd5fxgq4xa_ooblayout_free(str
|
|
return 0;
|
|
}
|
|
|
|
+static const struct mtd_ooblayout_ops gd5fxgq4xa_ooblayout = {
|
|
+ .ecc = gd5fxgq4xa_ooblayout_ecc,
|
|
+ .free = gd5fxgq4xa_ooblayout_free,
|
|
+};
|
|
+
|
|
static int gd5fxgq4xa_ecc_get_status(struct spinand_device *spinand,
|
|
u8 status)
|
|
{
|
|
@@ -83,7 +102,7 @@ static int gd5fxgq4xa_ecc_get_status(str
|
|
return -EINVAL;
|
|
}
|
|
|
|
-static int gd5fxgq4uexxg_ooblayout_ecc(struct mtd_info *mtd, int section,
|
|
+static int gd5fxgq4_variant2_ooblayout_ecc(struct mtd_info *mtd, int section,
|
|
struct mtd_oob_region *region)
|
|
{
|
|
if (section)
|
|
@@ -95,7 +114,7 @@ static int gd5fxgq4uexxg_ooblayout_ecc(s
|
|
return 0;
|
|
}
|
|
|
|
-static int gd5fxgq4uexxg_ooblayout_free(struct mtd_info *mtd, int section,
|
|
+static int gd5fxgq4_variant2_ooblayout_free(struct mtd_info *mtd, int section,
|
|
struct mtd_oob_region *region)
|
|
{
|
|
if (section)
|
|
@@ -108,6 +127,11 @@ static int gd5fxgq4uexxg_ooblayout_free(
|
|
return 0;
|
|
}
|
|
|
|
+static const struct mtd_ooblayout_ops gd5fxgq4_variant2_ooblayout = {
|
|
+ .ecc = gd5fxgq4_variant2_ooblayout_ecc,
|
|
+ .free = gd5fxgq4_variant2_ooblayout_free,
|
|
+};
|
|
+
|
|
static int gd5fxgq4uexxg_ecc_get_status(struct spinand_device *spinand,
|
|
u8 status)
|
|
{
|
|
@@ -150,15 +174,25 @@ static int gd5fxgq4uexxg_ecc_get_status(
|
|
return -EINVAL;
|
|
}
|
|
|
|
-static const struct mtd_ooblayout_ops gd5fxgq4xa_ooblayout = {
|
|
- .ecc = gd5fxgq4xa_ooblayout_ecc,
|
|
- .free = gd5fxgq4xa_ooblayout_free,
|
|
-};
|
|
+static int gd5fxgq4ufxxg_ecc_get_status(struct spinand_device *spinand,
|
|
+ u8 status)
|
|
+{
|
|
+ switch (status & GD5FXGQ4UXFXXG_STATUS_ECC_MASK) {
|
|
+ case GD5FXGQ4UXFXXG_STATUS_ECC_NO_BITFLIPS:
|
|
+ return 0;
|
|
|
|
-static const struct mtd_ooblayout_ops gd5fxgq4uexxg_ooblayout = {
|
|
- .ecc = gd5fxgq4uexxg_ooblayout_ecc,
|
|
- .free = gd5fxgq4uexxg_ooblayout_free,
|
|
-};
|
|
+ case GD5FXGQ4UXFXXG_STATUS_ECC_1_3_BITFLIPS:
|
|
+ return 3;
|
|
+
|
|
+ case GD5FXGQ4UXFXXG_STATUS_ECC_UNCOR_ERROR:
|
|
+ return -EBADMSG;
|
|
+
|
|
+ default: /* (2 << 4) through (6 << 4) are 4-8 corrected errors */
|
|
+ return ((status & GD5FXGQ4UXFXXG_STATUS_ECC_MASK) >> 4) + 2;
|
|
+ }
|
|
+
|
|
+ return -EINVAL;
|
|
+}
|
|
|
|
static const struct spinand_info gigadevice_spinand_table[] = {
|
|
SPINAND_INFO("GD5F1GQ4xA", 0xF1,
|
|
@@ -195,25 +229,40 @@ static const struct spinand_info gigadev
|
|
&write_cache_variants,
|
|
&update_cache_variants),
|
|
0,
|
|
- SPINAND_ECCINFO(&gd5fxgq4uexxg_ooblayout,
|
|
+ SPINAND_ECCINFO(&gd5fxgq4_variant2_ooblayout,
|
|
gd5fxgq4uexxg_ecc_get_status)),
|
|
+ SPINAND_INFO("GD5F1GQ4UFxxG", 0xb148,
|
|
+ NAND_MEMORG(1, 2048, 128, 64, 1024, 1, 1, 1),
|
|
+ NAND_ECCREQ(8, 512),
|
|
+ SPINAND_INFO_OP_VARIANTS(&read_cache_variants_f,
|
|
+ &write_cache_variants,
|
|
+ &update_cache_variants),
|
|
+ 0,
|
|
+ SPINAND_ECCINFO(&gd5fxgq4_variant2_ooblayout,
|
|
+ gd5fxgq4ufxxg_ecc_get_status)),
|
|
};
|
|
|
|
static int gigadevice_spinand_detect(struct spinand_device *spinand)
|
|
{
|
|
u8 *id = spinand->id.data;
|
|
+ u16 did;
|
|
int ret;
|
|
|
|
/*
|
|
- * For GD NANDs, There is an address byte needed to shift in before IDs
|
|
- * are read out, so the first byte in raw_id is dummy.
|
|
+ * Earlier GDF5-series devices (A,E) return [0][MID][DID]
|
|
+ * Later (F) devices return [MID][DID1][DID2]
|
|
*/
|
|
- if (id[1] != SPINAND_MFR_GIGADEVICE)
|
|
+
|
|
+ if (id[0] == SPINAND_MFR_GIGADEVICE)
|
|
+ did = (id[1] << 8) + id[2];
|
|
+ else if (id[0] == 0 && id[1] == SPINAND_MFR_GIGADEVICE)
|
|
+ did = id[2];
|
|
+ else
|
|
return 0;
|
|
|
|
ret = spinand_match_and_init(spinand, gigadevice_spinand_table,
|
|
ARRAY_SIZE(gigadevice_spinand_table),
|
|
- id[2]);
|
|
+ did);
|
|
if (ret)
|
|
return ret;
|
|
|