* [PATCH v4] mtd: nand: automate NAND timings selection
@ 2016-09-15  8:32 Sascha Hauer
  2016-09-15  8:32 ` [PATCH 1/9] mtd: nand: Create a NAND reset function Sascha Hauer
                   ` (9 more replies)
  0 siblings, 10 replies; 19+ messages in thread
From: Sascha Hauer @ 2016-09-15  8:32 UTC (permalink / raw)
  To: linux-arm-kernel
This series aims at automating the NAND timings selection which is
currently supposed to be done in each NAND controller driver, thus
simplifying drivers implementation.
As suggested by Boris this version of the series introduces a nand_reset()
function which replaces the several open coded NAND_CMD_RESET commands
in the code. This makes sure we can apply the timing each time after
after reset.
Also I have brought back the conversion patch for teh sunxi driver whic
was part of Boris initial posting. It's untested due to the lack of hardware,
so please test before applying.
Sascha
Changes since v4:
- Change onfi_init_data_interface() prototype to be more future proof as
  requested by Boris
Changes since v3:
- Bring back patch dropped in v3
- Use statically allocated default timing for all chips and store one
  optimized timing in struct nand_chip
Changes since v2:
- Add accessor function to get the SDR timing from struct nand_data_interface
- Change nand_reset() argument to struct nand_chip
- Drop conversion of nand_timing array to struct nand_data_interface
- Recalculate timing whenever needed instead of storing a pointer in struct
  nand_chip
- some more refactoring
Changes since v1:
- create a nand_reset() function to create a single place to reset NAND
  chips and to apply timings
- Add patch to convert sunxi driver for automated timing setup
- split into more patches
Changes since the initial posting from Boris:
- Integrate Feedback from Ezequiel Garcia
- When iterating over the chips calling onfi_set_features() for each
  bail out when any of the calls fail, not only the last one.
- When one of the onfi_set_features() calls fail then reset the chipi
  afterwards.
- Drop Sunxi example, add patch for the mxc_nand controller instead.
----------------------------------------------------------------
Boris Brezillon (1):
      mtd: nand: automate NAND timings selection
Sascha Hauer (8):
      mtd: nand: Create a NAND reset function
      mtd: nand: Introduce nand_data_interface
      mtd: nand: convert ONFI mode into data interface
      mtd: nand: Add function to convert ONFI mode to data_interface
      mtd: nand: Expose data interface for ONFI mode 0
      mtd: nand: sunxi: switch from manual to automated timing config
      mtd: nand: mxc: implement onfi get/set features
      mtd: nand: mxc: Add timing setup for v2 controllers
 drivers/mtd/nand/mxc_nand.c     | 133 ++++++++++++
 drivers/mtd/nand/nand_base.c    | 179 ++++++++++++++-
 drivers/mtd/nand/nand_timings.c | 469 ++++++++++++++++++++++------------------
 drivers/mtd/nand/sunxi_nand.c   |  76 ++-----
 include/linux/mtd/nand.h        | 190 +++++++++++-----
 5 files changed, 721 insertions(+), 326 deletions(-)
^ permalink raw reply	[flat|nested] 19+ messages in thread* [PATCH 1/9] mtd: nand: Create a NAND reset function 2016-09-15 8:32 [PATCH v4] mtd: nand: automate NAND timings selection Sascha Hauer @ 2016-09-15 8:32 ` Sascha Hauer 2016-09-15 8:32 ` [PATCH 2/9] mtd: nand: Introduce nand_data_interface Sascha Hauer ` (8 subsequent siblings) 9 siblings, 0 replies; 19+ messages in thread From: Sascha Hauer @ 2016-09-15 8:32 UTC (permalink / raw) To: linux-arm-kernel When NAND devices are resetted some initialization may have to be done, like for example they have to be configured for the timing mode that shall be used. To get a common place where this initialization can be implemented create a nand_reset() function. This currently only issues a NAND_CMD_RESET to the NAND device. The places issuing this command manually are replaced with a call to nand_reset(). Signed-off-by: Sascha Hauer <s.hauer@pengutronix.de> --- drivers/mtd/nand/nand_base.c | 23 ++++++++++++++++++----- include/linux/mtd/nand.h | 4 ++++ 2 files changed, 22 insertions(+), 5 deletions(-) diff --git a/drivers/mtd/nand/nand_base.c b/drivers/mtd/nand/nand_base.c index 77533f7..1f704cc 100644 --- a/drivers/mtd/nand/nand_base.c +++ b/drivers/mtd/nand/nand_base.c @@ -948,6 +948,19 @@ static int nand_wait(struct mtd_info *mtd, struct nand_chip *chip) } /** + * nand_reset - Reset and initialize a NAND device + * @chip: The NAND chip + * + * Returns 0 for success or negative error code otherwise + */ +int nand_reset(struct nand_chip *chip) +{ + chip->cmdfunc(&chip->mtd, NAND_CMD_RESET, -1, -1); + + return 0; +} + +/** * __nand_unlock - [REPLACEABLE] unlocks specified locked blocks * @mtd: mtd info * @ofs: offset to start unlock from @@ -1025,7 +1038,7 @@ int nand_unlock(struct mtd_info *mtd, loff_t ofs, uint64_t len) * some operation can also clear the bit 7 of status register * eg. erase/program a locked block */ - chip->cmdfunc(mtd, NAND_CMD_RESET, -1, -1); + nand_reset(chip); /* Check, if it is write protected */ if (nand_check_wp(mtd)) { @@ -1084,7 +1097,7 @@ int nand_lock(struct mtd_info *mtd, loff_t ofs, uint64_t len) * some operation can also clear the bit 7 of status register * eg. erase/program a locked block */ - chip->cmdfunc(mtd, NAND_CMD_RESET, -1, -1); + nand_reset(chip); /* Check, if it is write protected */ if (nand_check_wp(mtd)) { @@ -2788,7 +2801,7 @@ static int nand_do_write_oob(struct mtd_info *mtd, loff_t to, * if we don't do this. I have no clue why, but I seem to have 'fixed' * it in the doc2000 driver in August 1999. dwmw2. */ - chip->cmdfunc(mtd, NAND_CMD_RESET, -1, -1); + nand_reset(chip); /* Check, if it is write protected */ if (nand_check_wp(mtd)) { @@ -3829,7 +3842,7 @@ static struct nand_flash_dev *nand_get_flash_type(struct mtd_info *mtd, * Reset the chip, required by some chips (e.g. Micron MT29FxGxxxxx) * after power-up. */ - chip->cmdfunc(mtd, NAND_CMD_RESET, -1, -1); + nand_reset(chip); /* Send the command for reading device ID */ chip->cmdfunc(mtd, NAND_CMD_READID, 0x00, -1); @@ -4161,7 +4174,7 @@ int nand_scan_ident(struct mtd_info *mtd, int maxchips, for (i = 1; i < maxchips; i++) { chip->select_chip(mtd, i); /* See comment in nand_get_flash_type for reset */ - chip->cmdfunc(mtd, NAND_CMD_RESET, -1, -1); + nand_reset(chip); /* Send the command for reading device ID */ chip->cmdfunc(mtd, NAND_CMD_READID, 0x00, -1); /* Read manufacturer and device IDs */ diff --git a/include/linux/mtd/nand.h b/include/linux/mtd/nand.h index 8dd6e01..9890df2 100644 --- a/include/linux/mtd/nand.h +++ b/include/linux/mtd/nand.h @@ -1093,4 +1093,8 @@ int nand_read_oob_std(struct mtd_info *mtd, struct nand_chip *chip, int page); /* Default read_oob syndrome implementation */ int nand_read_oob_syndrome(struct mtd_info *mtd, struct nand_chip *chip, int page); + +/* Reset and initialize a NAND device */ +int nand_reset(struct nand_chip *chip); + #endif /* __LINUX_MTD_NAND_H */ -- 2.8.1 ^ permalink raw reply related [flat|nested] 19+ messages in thread
* [PATCH 2/9] mtd: nand: Introduce nand_data_interface 2016-09-15 8:32 [PATCH v4] mtd: nand: automate NAND timings selection Sascha Hauer 2016-09-15 8:32 ` [PATCH 1/9] mtd: nand: Create a NAND reset function Sascha Hauer @ 2016-09-15 8:32 ` Sascha Hauer 2016-09-15 8:32 ` [PATCH 3/9] mtd: nand: convert ONFI mode into data interface Sascha Hauer ` (7 subsequent siblings) 9 siblings, 0 replies; 19+ messages in thread From: Sascha Hauer @ 2016-09-15 8:32 UTC (permalink / raw) To: linux-arm-kernel Currently we have no data structure to fully describe a NAND timing. We only have struct nand_sdr_timings for NAND timings in SDR mode, but nothing for DDR mode and also no container to store both types of timing. This patch adds struct nand_data_interface which stores the timing type and a union of different timings. This can be used to pass to drivers in order to configure the timing. Add kerneldoc for struct nand_sdr_timings while touching it anyway. Signed-off-by: Sascha Hauer <s.hauer@pengutronix.de> --- include/linux/mtd/nand.h | 165 +++++++++++++++++++++++++++++++++-------------- 1 file changed, 116 insertions(+), 49 deletions(-) diff --git a/include/linux/mtd/nand.h b/include/linux/mtd/nand.h index 9890df2..f305bed 100644 --- a/include/linux/mtd/nand.h +++ b/include/linux/mtd/nand.h @@ -566,6 +566,122 @@ struct nand_buffers { }; /** + * struct nand_sdr_timings - SDR NAND chip timings + * + * This struct defines the timing requirements of a SDR NAND chip. + * These information can be found in every NAND datasheets and the timings + * meaning are described in the ONFI specifications: + * www.onfi.org/~/media/ONFI/specs/onfi_3_1_spec.pdf (chapter 4.15 Timing + * Parameters) + * + * All these timings are expressed in picoseconds. + * + * @tALH_min: ALE hold time + * @tADL_min: ALE to data loading time + * @tALS_min: ALE setup time + * @tAR_min: ALE to RE# delay + * @tCEA_max: CE# access time + * @tCEH_min: + * @tCH_min: CE# hold time + * @tCHZ_max: CE# high to output hi-Z + * @tCLH_min: CLE hold time + * @tCLR_min: CLE to RE# delay + * @tCLS_min: CLE setup time + * @tCOH_min: CE# high to output hold + * @tCS_min: CE# setup time + * @tDH_min: Data hold time + * @tDS_min: Data setup time + * @tFEAT_max: Busy time for Set Features and Get Features + * @tIR_min: Output hi-Z to RE# low + * @tITC_max: Interface and Timing Mode Change time + * @tRC_min: RE# cycle time + * @tREA_max: RE# access time + * @tREH_min: RE# high hold time + * @tRHOH_min: RE# high to output hold + * @tRHW_min: RE# high to WE# low + * @tRHZ_max: RE# high to output hi-Z + * @tRLOH_min: RE# low to output hold + * @tRP_min: RE# pulse width + * @tRR_min: Ready to RE# low (data only) + * @tRST_max: Device reset time, measured from the falling edge of R/B# to the rising edge of R/B#. + * @tWB_max: WE# high to SR[6] low + * @tWC_min: WE# cycle time + * @tWH_min: WE# high hold time + * @tWHR_min: WE# high to RE# low + * @tWP_min: WE# pulse width + * @tWW_min: WP# transition to WE# low + */ +struct nand_sdr_timings { + u32 tALH_min; + u32 tADL_min; + u32 tALS_min; + u32 tAR_min; + u32 tCEA_max; + u32 tCEH_min; + u32 tCH_min; + u32 tCHZ_max; + u32 tCLH_min; + u32 tCLR_min; + u32 tCLS_min; + u32 tCOH_min; + u32 tCS_min; + u32 tDH_min; + u32 tDS_min; + u32 tFEAT_max; + u32 tIR_min; + u32 tITC_max; + u32 tRC_min; + u32 tREA_max; + u32 tREH_min; + u32 tRHOH_min; + u32 tRHW_min; + u32 tRHZ_max; + u32 tRLOH_min; + u32 tRP_min; + u32 tRR_min; + u64 tRST_max; + u32 tWB_max; + u32 tWC_min; + u32 tWH_min; + u32 tWHR_min; + u32 tWP_min; + u32 tWW_min; +}; + +/** + * enum nand_data_interface_type - NAND interface timing type + * @NAND_SDR_IFACE: Single Data Rate interface + */ +enum nand_data_interface_type { + NAND_SDR_IFACE, +}; + +/** + * struct nand_data_interface - NAND interface timing + * @type: type of the timing + * @timings: The timing, type according to @type + */ +struct nand_data_interface { + enum nand_data_interface_type type; + union { + struct nand_sdr_timings sdr; + } timings; +}; + +/** + * nand_get_sdr_timings - get SDR timing from data interface + * @conf: The data interface + */ +static inline const struct nand_sdr_timings * +nand_get_sdr_timings(const struct nand_data_interface *conf) +{ + if (conf->type != NAND_SDR_IFACE) + return ERR_PTR(-EINVAL); + + return &conf->timings.sdr; +} + +/** * struct nand_chip - NAND Private Flash Chip Data * @mtd: MTD device registered to the MTD framework * @IO_ADDR_R: [BOARDSPECIFIC] address to read the 8 I/O lines of the @@ -1023,55 +1139,6 @@ static inline int jedec_feature(struct nand_chip *chip) : 0; } -/* - * struct nand_sdr_timings - SDR NAND chip timings - * - * This struct defines the timing requirements of a SDR NAND chip. - * These informations can be found in every NAND datasheets and the timings - * meaning are described in the ONFI specifications: - * www.onfi.org/~/media/ONFI/specs/onfi_3_1_spec.pdf (chapter 4.15 Timing - * Parameters) - * - * All these timings are expressed in picoseconds. - */ - -struct nand_sdr_timings { - u32 tALH_min; - u32 tADL_min; - u32 tALS_min; - u32 tAR_min; - u32 tCEA_max; - u32 tCEH_min; - u32 tCH_min; - u32 tCHZ_max; - u32 tCLH_min; - u32 tCLR_min; - u32 tCLS_min; - u32 tCOH_min; - u32 tCS_min; - u32 tDH_min; - u32 tDS_min; - u32 tFEAT_max; - u32 tIR_min; - u32 tITC_max; - u32 tRC_min; - u32 tREA_max; - u32 tREH_min; - u32 tRHOH_min; - u32 tRHW_min; - u32 tRHZ_max; - u32 tRLOH_min; - u32 tRP_min; - u32 tRR_min; - u64 tRST_max; - u32 tWB_max; - u32 tWC_min; - u32 tWH_min; - u32 tWHR_min; - u32 tWP_min; - u32 tWW_min; -}; - /* get timing characteristics from ONFI timing mode. */ const struct nand_sdr_timings *onfi_async_timing_mode_to_sdr_timings(int mode); -- 2.8.1 ^ permalink raw reply related [flat|nested] 19+ messages in thread
* [PATCH 3/9] mtd: nand: convert ONFI mode into data interface 2016-09-15 8:32 [PATCH v4] mtd: nand: automate NAND timings selection Sascha Hauer 2016-09-15 8:32 ` [PATCH 1/9] mtd: nand: Create a NAND reset function Sascha Hauer 2016-09-15 8:32 ` [PATCH 2/9] mtd: nand: Introduce nand_data_interface Sascha Hauer @ 2016-09-15 8:32 ` Sascha Hauer 2016-09-15 8:32 ` [PATCH 4/9] mtd: nand: Add function to convert ONFI mode to data_interface Sascha Hauer ` (6 subsequent siblings) 9 siblings, 0 replies; 19+ messages in thread From: Sascha Hauer @ 2016-09-15 8:32 UTC (permalink / raw) To: linux-arm-kernel struct nand_data_interface is the designated type to pass to the NAND drivers to configure the timing. To simplify further patches convert the onfi_sdr_timings array from type struct nand_sdr_timings nand_data_interface. Signed-off-by: Sascha Hauer <s.hauer@pengutronix.de> --- drivers/mtd/nand/nand_timings.c | 430 +++++++++++++++++++++------------------- 1 file changed, 224 insertions(+), 206 deletions(-) diff --git a/drivers/mtd/nand/nand_timings.c b/drivers/mtd/nand/nand_timings.c index e81470a..9af2ebc 100644 --- a/drivers/mtd/nand/nand_timings.c +++ b/drivers/mtd/nand/nand_timings.c @@ -13,228 +13,246 @@ #include <linux/export.h> #include <linux/mtd/nand.h> -static const struct nand_sdr_timings onfi_sdr_timings[] = { +static const struct nand_data_interface onfi_sdr_timings[] = { /* Mode 0 */ { - .tADL_min = 200000, - .tALH_min = 20000, - .tALS_min = 50000, - .tAR_min = 25000, - .tCEA_max = 100000, - .tCEH_min = 20000, - .tCH_min = 20000, - .tCHZ_max = 100000, - .tCLH_min = 20000, - .tCLR_min = 20000, - .tCLS_min = 50000, - .tCOH_min = 0, - .tCS_min = 70000, - .tDH_min = 20000, - .tDS_min = 40000, - .tFEAT_max = 1000000, - .tIR_min = 10000, - .tITC_max = 1000000, - .tRC_min = 100000, - .tREA_max = 40000, - .tREH_min = 30000, - .tRHOH_min = 0, - .tRHW_min = 200000, - .tRHZ_max = 200000, - .tRLOH_min = 0, - .tRP_min = 50000, - .tRST_max = 250000000000ULL, - .tWB_max = 200000, - .tRR_min = 40000, - .tWC_min = 100000, - .tWH_min = 30000, - .tWHR_min = 120000, - .tWP_min = 50000, - .tWW_min = 100000, + .type = NAND_SDR_IFACE, + .timings.sdr = { + .tADL_min = 200000, + .tALH_min = 20000, + .tALS_min = 50000, + .tAR_min = 25000, + .tCEA_max = 100000, + .tCEH_min = 20000, + .tCH_min = 20000, + .tCHZ_max = 100000, + .tCLH_min = 20000, + .tCLR_min = 20000, + .tCLS_min = 50000, + .tCOH_min = 0, + .tCS_min = 70000, + .tDH_min = 20000, + .tDS_min = 40000, + .tFEAT_max = 1000000, + .tIR_min = 10000, + .tITC_max = 1000000, + .tRC_min = 100000, + .tREA_max = 40000, + .tREH_min = 30000, + .tRHOH_min = 0, + .tRHW_min = 200000, + .tRHZ_max = 200000, + .tRLOH_min = 0, + .tRP_min = 50000, + .tRST_max = 250000000000ULL, + .tWB_max = 200000, + .tRR_min = 40000, + .tWC_min = 100000, + .tWH_min = 30000, + .tWHR_min = 120000, + .tWP_min = 50000, + .tWW_min = 100000, + }, }, /* Mode 1 */ { - .tADL_min = 100000, - .tALH_min = 10000, - .tALS_min = 25000, - .tAR_min = 10000, - .tCEA_max = 45000, - .tCEH_min = 20000, - .tCH_min = 10000, - .tCHZ_max = 50000, - .tCLH_min = 10000, - .tCLR_min = 10000, - .tCLS_min = 25000, - .tCOH_min = 15000, - .tCS_min = 35000, - .tDH_min = 10000, - .tDS_min = 20000, - .tFEAT_max = 1000000, - .tIR_min = 0, - .tITC_max = 1000000, - .tRC_min = 50000, - .tREA_max = 30000, - .tREH_min = 15000, - .tRHOH_min = 15000, - .tRHW_min = 100000, - .tRHZ_max = 100000, - .tRLOH_min = 0, - .tRP_min = 25000, - .tRR_min = 20000, - .tRST_max = 500000000, - .tWB_max = 100000, - .tWC_min = 45000, - .tWH_min = 15000, - .tWHR_min = 80000, - .tWP_min = 25000, - .tWW_min = 100000, + .type = NAND_SDR_IFACE, + .timings.sdr = { + .tADL_min = 100000, + .tALH_min = 10000, + .tALS_min = 25000, + .tAR_min = 10000, + .tCEA_max = 45000, + .tCEH_min = 20000, + .tCH_min = 10000, + .tCHZ_max = 50000, + .tCLH_min = 10000, + .tCLR_min = 10000, + .tCLS_min = 25000, + .tCOH_min = 15000, + .tCS_min = 35000, + .tDH_min = 10000, + .tDS_min = 20000, + .tFEAT_max = 1000000, + .tIR_min = 0, + .tITC_max = 1000000, + .tRC_min = 50000, + .tREA_max = 30000, + .tREH_min = 15000, + .tRHOH_min = 15000, + .tRHW_min = 100000, + .tRHZ_max = 100000, + .tRLOH_min = 0, + .tRP_min = 25000, + .tRR_min = 20000, + .tRST_max = 500000000, + .tWB_max = 100000, + .tWC_min = 45000, + .tWH_min = 15000, + .tWHR_min = 80000, + .tWP_min = 25000, + .tWW_min = 100000, + }, }, /* Mode 2 */ { - .tADL_min = 100000, - .tALH_min = 10000, - .tALS_min = 15000, - .tAR_min = 10000, - .tCEA_max = 30000, - .tCEH_min = 20000, - .tCH_min = 10000, - .tCHZ_max = 50000, - .tCLH_min = 10000, - .tCLR_min = 10000, - .tCLS_min = 15000, - .tCOH_min = 15000, - .tCS_min = 25000, - .tDH_min = 5000, - .tDS_min = 15000, - .tFEAT_max = 1000000, - .tIR_min = 0, - .tITC_max = 1000000, - .tRC_min = 35000, - .tREA_max = 25000, - .tREH_min = 15000, - .tRHOH_min = 15000, - .tRHW_min = 100000, - .tRHZ_max = 100000, - .tRLOH_min = 0, - .tRR_min = 20000, - .tRST_max = 500000000, - .tWB_max = 100000, - .tRP_min = 17000, - .tWC_min = 35000, - .tWH_min = 15000, - .tWHR_min = 80000, - .tWP_min = 17000, - .tWW_min = 100000, + .type = NAND_SDR_IFACE, + .timings.sdr = { + .tADL_min = 100000, + .tALH_min = 10000, + .tALS_min = 15000, + .tAR_min = 10000, + .tCEA_max = 30000, + .tCEH_min = 20000, + .tCH_min = 10000, + .tCHZ_max = 50000, + .tCLH_min = 10000, + .tCLR_min = 10000, + .tCLS_min = 15000, + .tCOH_min = 15000, + .tCS_min = 25000, + .tDH_min = 5000, + .tDS_min = 15000, + .tFEAT_max = 1000000, + .tIR_min = 0, + .tITC_max = 1000000, + .tRC_min = 35000, + .tREA_max = 25000, + .tREH_min = 15000, + .tRHOH_min = 15000, + .tRHW_min = 100000, + .tRHZ_max = 100000, + .tRLOH_min = 0, + .tRR_min = 20000, + .tRST_max = 500000000, + .tWB_max = 100000, + .tRP_min = 17000, + .tWC_min = 35000, + .tWH_min = 15000, + .tWHR_min = 80000, + .tWP_min = 17000, + .tWW_min = 100000, + }, }, /* Mode 3 */ { - .tADL_min = 100000, - .tALH_min = 5000, - .tALS_min = 10000, - .tAR_min = 10000, - .tCEA_max = 25000, - .tCEH_min = 20000, - .tCH_min = 5000, - .tCHZ_max = 50000, - .tCLH_min = 5000, - .tCLR_min = 10000, - .tCLS_min = 10000, - .tCOH_min = 15000, - .tCS_min = 25000, - .tDH_min = 5000, - .tDS_min = 10000, - .tFEAT_max = 1000000, - .tIR_min = 0, - .tITC_max = 1000000, - .tRC_min = 30000, - .tREA_max = 20000, - .tREH_min = 10000, - .tRHOH_min = 15000, - .tRHW_min = 100000, - .tRHZ_max = 100000, - .tRLOH_min = 0, - .tRP_min = 15000, - .tRR_min = 20000, - .tRST_max = 500000000, - .tWB_max = 100000, - .tWC_min = 30000, - .tWH_min = 10000, - .tWHR_min = 80000, - .tWP_min = 15000, - .tWW_min = 100000, + .type = NAND_SDR_IFACE, + .timings.sdr = { + .tADL_min = 100000, + .tALH_min = 5000, + .tALS_min = 10000, + .tAR_min = 10000, + .tCEA_max = 25000, + .tCEH_min = 20000, + .tCH_min = 5000, + .tCHZ_max = 50000, + .tCLH_min = 5000, + .tCLR_min = 10000, + .tCLS_min = 10000, + .tCOH_min = 15000, + .tCS_min = 25000, + .tDH_min = 5000, + .tDS_min = 10000, + .tFEAT_max = 1000000, + .tIR_min = 0, + .tITC_max = 1000000, + .tRC_min = 30000, + .tREA_max = 20000, + .tREH_min = 10000, + .tRHOH_min = 15000, + .tRHW_min = 100000, + .tRHZ_max = 100000, + .tRLOH_min = 0, + .tRP_min = 15000, + .tRR_min = 20000, + .tRST_max = 500000000, + .tWB_max = 100000, + .tWC_min = 30000, + .tWH_min = 10000, + .tWHR_min = 80000, + .tWP_min = 15000, + .tWW_min = 100000, + }, }, /* Mode 4 */ { - .tADL_min = 70000, - .tALH_min = 5000, - .tALS_min = 10000, - .tAR_min = 10000, - .tCEA_max = 25000, - .tCEH_min = 20000, - .tCH_min = 5000, - .tCHZ_max = 30000, - .tCLH_min = 5000, - .tCLR_min = 10000, - .tCLS_min = 10000, - .tCOH_min = 15000, - .tCS_min = 20000, - .tDH_min = 5000, - .tDS_min = 10000, - .tFEAT_max = 1000000, - .tIR_min = 0, - .tITC_max = 1000000, - .tRC_min = 25000, - .tREA_max = 20000, - .tREH_min = 10000, - .tRHOH_min = 15000, - .tRHW_min = 100000, - .tRHZ_max = 100000, - .tRLOH_min = 5000, - .tRP_min = 12000, - .tRR_min = 20000, - .tRST_max = 500000000, - .tWB_max = 100000, - .tWC_min = 25000, - .tWH_min = 10000, - .tWHR_min = 80000, - .tWP_min = 12000, - .tWW_min = 100000, + .type = NAND_SDR_IFACE, + .timings.sdr = { + .tADL_min = 70000, + .tALH_min = 5000, + .tALS_min = 10000, + .tAR_min = 10000, + .tCEA_max = 25000, + .tCEH_min = 20000, + .tCH_min = 5000, + .tCHZ_max = 30000, + .tCLH_min = 5000, + .tCLR_min = 10000, + .tCLS_min = 10000, + .tCOH_min = 15000, + .tCS_min = 20000, + .tDH_min = 5000, + .tDS_min = 10000, + .tFEAT_max = 1000000, + .tIR_min = 0, + .tITC_max = 1000000, + .tRC_min = 25000, + .tREA_max = 20000, + .tREH_min = 10000, + .tRHOH_min = 15000, + .tRHW_min = 100000, + .tRHZ_max = 100000, + .tRLOH_min = 5000, + .tRP_min = 12000, + .tRR_min = 20000, + .tRST_max = 500000000, + .tWB_max = 100000, + .tWC_min = 25000, + .tWH_min = 10000, + .tWHR_min = 80000, + .tWP_min = 12000, + .tWW_min = 100000, + }, }, /* Mode 5 */ { - .tADL_min = 70000, - .tALH_min = 5000, - .tALS_min = 10000, - .tAR_min = 10000, - .tCEA_max = 25000, - .tCEH_min = 20000, - .tCH_min = 5000, - .tCHZ_max = 30000, - .tCLH_min = 5000, - .tCLR_min = 10000, - .tCLS_min = 10000, - .tCOH_min = 15000, - .tCS_min = 15000, - .tDH_min = 5000, - .tDS_min = 7000, - .tFEAT_max = 1000000, - .tIR_min = 0, - .tITC_max = 1000000, - .tRC_min = 20000, - .tREA_max = 16000, - .tREH_min = 7000, - .tRHOH_min = 15000, - .tRHW_min = 100000, - .tRHZ_max = 100000, - .tRLOH_min = 5000, - .tRP_min = 10000, - .tRR_min = 20000, - .tRST_max = 500000000, - .tWB_max = 100000, - .tWC_min = 20000, - .tWH_min = 7000, - .tWHR_min = 80000, - .tWP_min = 10000, - .tWW_min = 100000, + .type = NAND_SDR_IFACE, + .timings.sdr = { + .tADL_min = 70000, + .tALH_min = 5000, + .tALS_min = 10000, + .tAR_min = 10000, + .tCEA_max = 25000, + .tCEH_min = 20000, + .tCH_min = 5000, + .tCHZ_max = 30000, + .tCLH_min = 5000, + .tCLR_min = 10000, + .tCLS_min = 10000, + .tCOH_min = 15000, + .tCS_min = 15000, + .tDH_min = 5000, + .tDS_min = 7000, + .tFEAT_max = 1000000, + .tIR_min = 0, + .tITC_max = 1000000, + .tRC_min = 20000, + .tREA_max = 16000, + .tREH_min = 7000, + .tRHOH_min = 15000, + .tRHW_min = 100000, + .tRHZ_max = 100000, + .tRLOH_min = 5000, + .tRP_min = 10000, + .tRR_min = 20000, + .tRST_max = 500000000, + .tWB_max = 100000, + .tWC_min = 20000, + .tWH_min = 7000, + .tWHR_min = 80000, + .tWP_min = 10000, + .tWW_min = 100000, + }, }, }; @@ -248,6 +266,6 @@ const struct nand_sdr_timings *onfi_async_timing_mode_to_sdr_timings(int mode) if (mode < 0 || mode >= ARRAY_SIZE(onfi_sdr_timings)) return ERR_PTR(-EINVAL); - return &onfi_sdr_timings[mode]; + return &onfi_sdr_timings[mode].timings.sdr; } EXPORT_SYMBOL(onfi_async_timing_mode_to_sdr_timings); -- 2.8.1 ^ permalink raw reply related [flat|nested] 19+ messages in thread
* [PATCH 4/9] mtd: nand: Add function to convert ONFI mode to data_interface 2016-09-15 8:32 [PATCH v4] mtd: nand: automate NAND timings selection Sascha Hauer ` (2 preceding siblings ...) 2016-09-15 8:32 ` [PATCH 3/9] mtd: nand: convert ONFI mode into data interface Sascha Hauer @ 2016-09-15 8:32 ` Sascha Hauer 2016-09-15 8:32 ` [PATCH 5/9] mtd: nand: Expose data interface for ONFI mode 0 Sascha Hauer ` (5 subsequent siblings) 9 siblings, 0 replies; 19+ messages in thread From: Sascha Hauer @ 2016-09-15 8:32 UTC (permalink / raw) To: linux-arm-kernel onfi_init_data_interface() initializes a data interface with values from a given ONFI mode. Signed-off-by: Sascha Hauer <s.hauer@pengutronix.de> --- drivers/mtd/nand/nand_timings.c | 28 ++++++++++++++++++++++++++++ include/linux/mtd/nand.h | 5 +++++ 2 files changed, 33 insertions(+) diff --git a/drivers/mtd/nand/nand_timings.c b/drivers/mtd/nand/nand_timings.c index 9af2ebc..608d098 100644 --- a/drivers/mtd/nand/nand_timings.c +++ b/drivers/mtd/nand/nand_timings.c @@ -269,3 +269,31 @@ const struct nand_sdr_timings *onfi_async_timing_mode_to_sdr_timings(int mode) return &onfi_sdr_timings[mode].timings.sdr; } EXPORT_SYMBOL(onfi_async_timing_mode_to_sdr_timings); + +/** + * onfi_init_data_interface - [NAND Interface] Initialize a data interface from + * given ONFI mode + * @iface: The data interface to be initialized + * @mode: The ONFI timing mode + */ +int onfi_init_data_interface(struct nand_chip *chip, + struct nand_data_interface *iface, + enum nand_data_interface_type type, + int timing_mode) +{ + if (type != NAND_SDR_IFACE) + return -EINVAL; + + if (timing_mode < 0 || timing_mode >= ARRAY_SIZE(onfi_sdr_timings)) + return -EINVAL; + + *iface = onfi_sdr_timings[timing_mode]; + + /* + * TODO: initialize timings that cannot be deduced from timing mode: + * tR, tPROG, tCCS, ... + * These information are part of the ONFI parameter page. + */ + + return 0; +} diff --git a/include/linux/mtd/nand.h b/include/linux/mtd/nand.h index f305bed..7216f54 100644 --- a/include/linux/mtd/nand.h +++ b/include/linux/mtd/nand.h @@ -1104,6 +1104,11 @@ static inline int onfi_get_sync_timing_mode(struct nand_chip *chip) return le16_to_cpu(chip->onfi_params.src_sync_timing_mode); } +int onfi_init_data_interface(struct nand_chip *chip, + struct nand_data_interface *iface, + enum nand_data_interface_type type, + int timing_mode); + /* * Check if it is a SLC nand. * The !nand_is_slc() can be used to check the MLC/TLC nand chips. -- 2.8.1 ^ permalink raw reply related [flat|nested] 19+ messages in thread
* [PATCH 5/9] mtd: nand: Expose data interface for ONFI mode 0 2016-09-15 8:32 [PATCH v4] mtd: nand: automate NAND timings selection Sascha Hauer ` (3 preceding siblings ...) 2016-09-15 8:32 ` [PATCH 4/9] mtd: nand: Add function to convert ONFI mode to data_interface Sascha Hauer @ 2016-09-15 8:32 ` Sascha Hauer 2016-09-15 12:56 ` Boris Brezillon 2016-09-15 8:32 ` [PATCH 6/9] mtd: nand: automate NAND timings selection Sascha Hauer ` (4 subsequent siblings) 9 siblings, 1 reply; 19+ messages in thread From: Sascha Hauer @ 2016-09-15 8:32 UTC (permalink / raw) To: linux-arm-kernel The nand layer will need ONFI mode 0 to use it as timing mode before and right after reset. Signed-off-by: Sascha Hauer <s.hauer@pengutronix.de> --- drivers/mtd/nand/nand_timings.c | 11 +++++++++++ include/linux/mtd/nand.h | 2 ++ 2 files changed, 13 insertions(+) diff --git a/drivers/mtd/nand/nand_timings.c b/drivers/mtd/nand/nand_timings.c index 608d098..9cdbc16 100644 --- a/drivers/mtd/nand/nand_timings.c +++ b/drivers/mtd/nand/nand_timings.c @@ -297,3 +297,14 @@ int onfi_init_data_interface(struct nand_chip *chip, return 0; } + +/** + * nand_get_default_data_interface - [NAND Interface] Retrieve NAND + * data interface for mode 0. This is used as default timing after + * reset. + */ +const struct nand_data_interface *nand_get_default_data_interface(void) +{ + return &onfi_sdr_timings[0]; +} +EXPORT_SYMBOL(nand_get_default_data_interface); diff --git a/include/linux/mtd/nand.h b/include/linux/mtd/nand.h index 7216f54..29ae324 100644 --- a/include/linux/mtd/nand.h +++ b/include/linux/mtd/nand.h @@ -1146,6 +1146,8 @@ static inline int jedec_feature(struct nand_chip *chip) /* get timing characteristics from ONFI timing mode. */ const struct nand_sdr_timings *onfi_async_timing_mode_to_sdr_timings(int mode); +/* get data interface from ONFI timing mode 0, used after reset. */ +const struct nand_data_interface *nand_get_default_data_interface(void); int nand_check_erased_ecc_chunk(void *data, int datalen, void *ecc, int ecclen, -- 2.8.1 ^ permalink raw reply related [flat|nested] 19+ messages in thread
* [PATCH 5/9] mtd: nand: Expose data interface for ONFI mode 0 2016-09-15 8:32 ` [PATCH 5/9] mtd: nand: Expose data interface for ONFI mode 0 Sascha Hauer @ 2016-09-15 12:56 ` Boris Brezillon 2016-10-09 5:12 ` Brian Norris 0 siblings, 1 reply; 19+ messages in thread From: Boris Brezillon @ 2016-09-15 12:56 UTC (permalink / raw) To: linux-arm-kernel On Thu, 15 Sep 2016 10:32:49 +0200 Sascha Hauer <s.hauer@pengutronix.de> wrote: > The nand layer will need ONFI mode 0 to use it as timing mode > before and right after reset. > > Signed-off-by: Sascha Hauer <s.hauer@pengutronix.de> > --- > drivers/mtd/nand/nand_timings.c | 11 +++++++++++ > include/linux/mtd/nand.h | 2 ++ > 2 files changed, 13 insertions(+) > > diff --git a/drivers/mtd/nand/nand_timings.c b/drivers/mtd/nand/nand_timings.c > index 608d098..9cdbc16 100644 > --- a/drivers/mtd/nand/nand_timings.c > +++ b/drivers/mtd/nand/nand_timings.c > @@ -297,3 +297,14 @@ int onfi_init_data_interface(struct nand_chip *chip, > > return 0; > } > + > +/** > + * nand_get_default_data_interface - [NAND Interface] Retrieve NAND > + * data interface for mode 0. This is used as default timing after > + * reset. > + */ > +const struct nand_data_interface *nand_get_default_data_interface(void) > +{ > + return &onfi_sdr_timings[0]; > +} > +EXPORT_SYMBOL(nand_get_default_data_interface); You export nand_get_default_data_interface() here, but you don't export onfi_init_data_interface(). None of these functions should be called from NAND controller drivers, so they don't need to be exported. ITOH, the prototypes are public (defined in nand.h), so nothing prevents a driver from calling these functions. I don't know what's the best solution here: 1/ Export both nand_get_default_data_interface() and onfi_init_data_interface() 2/ Do not export them, but keep their prototypes in nand.h with a comment saying that they should not be directly called by NAND controller drivers 3/ Removing the prototypes from nand.h and defining them in nand_base.c (or in a private header). Brian, Richard, Sascha, any comments? > diff --git a/include/linux/mtd/nand.h b/include/linux/mtd/nand.h > index 7216f54..29ae324 100644 > --- a/include/linux/mtd/nand.h > +++ b/include/linux/mtd/nand.h > @@ -1146,6 +1146,8 @@ static inline int jedec_feature(struct nand_chip *chip) > > /* get timing characteristics from ONFI timing mode. */ > const struct nand_sdr_timings *onfi_async_timing_mode_to_sdr_timings(int mode); > +/* get data interface from ONFI timing mode 0, used after reset. */ > +const struct nand_data_interface *nand_get_default_data_interface(void); > > int nand_check_erased_ecc_chunk(void *data, int datalen, > void *ecc, int ecclen, ^ permalink raw reply [flat|nested] 19+ messages in thread
* [PATCH 5/9] mtd: nand: Expose data interface for ONFI mode 0 2016-09-15 12:56 ` Boris Brezillon @ 2016-10-09 5:12 ` Brian Norris 2016-10-09 11:58 ` Boris Brezillon 0 siblings, 1 reply; 19+ messages in thread From: Brian Norris @ 2016-10-09 5:12 UTC (permalink / raw) To: linux-arm-kernel On Thu, Sep 15, 2016 at 02:56:25PM +0200, Boris Brezillon wrote: > On Thu, 15 Sep 2016 10:32:49 +0200 > Sascha Hauer <s.hauer@pengutronix.de> wrote: > > > The nand layer will need ONFI mode 0 to use it as timing mode > > before and right after reset. > > > > Signed-off-by: Sascha Hauer <s.hauer@pengutronix.de> > > --- > > drivers/mtd/nand/nand_timings.c | 11 +++++++++++ > > include/linux/mtd/nand.h | 2 ++ > > 2 files changed, 13 insertions(+) > > > > diff --git a/drivers/mtd/nand/nand_timings.c b/drivers/mtd/nand/nand_timings.c > > index 608d098..9cdbc16 100644 > > --- a/drivers/mtd/nand/nand_timings.c > > +++ b/drivers/mtd/nand/nand_timings.c > > @@ -297,3 +297,14 @@ int onfi_init_data_interface(struct nand_chip *chip, > > > > return 0; > > } > > + > > +/** > > + * nand_get_default_data_interface - [NAND Interface] Retrieve NAND > > + * data interface for mode 0. This is used as default timing after > > + * reset. > > + */ > > +const struct nand_data_interface *nand_get_default_data_interface(void) > > +{ > > + return &onfi_sdr_timings[0]; > > +} > > +EXPORT_SYMBOL(nand_get_default_data_interface); > > You export nand_get_default_data_interface() here, but you don't export > onfi_init_data_interface(). > None of these functions should be called from NAND controller drivers, > so they don't need to be exported. ITOH, the prototypes are public > (defined in nand.h), so nothing prevents a driver from calling these > functions. A bit late, but we can still change things for the next cycle of course... > I don't know what's the best solution here: > 1/ Export both nand_get_default_data_interface() and > onfi_init_data_interface() I don't really like this one. We shouldn't export them if we don't want people to use them in modules. And I'm sure the various compile-test builders out there would complain eventually if someone did. > 2/ Do not export them, but keep their prototypes in nand.h with a > comment saying that they should not be directly called by NAND > controller drivers Could be OK. > 3/ Removing the prototypes from nand.h and defining them in nand_base.c > (or in a private header). I like this one. Why not add a drivers/mtd/nand/nand.h that's intended for the nand.{o,ko} module only? > Brian, Richard, Sascha, any comments? Brian ^ permalink raw reply [flat|nested] 19+ messages in thread
* [PATCH 5/9] mtd: nand: Expose data interface for ONFI mode 0 2016-10-09 5:12 ` Brian Norris @ 2016-10-09 11:58 ` Boris Brezillon 0 siblings, 0 replies; 19+ messages in thread From: Boris Brezillon @ 2016-10-09 11:58 UTC (permalink / raw) To: linux-arm-kernel Hi Brian, On Sat, 8 Oct 2016 22:12:48 -0700 Brian Norris <computersforpeace@gmail.com> wrote: > On Thu, Sep 15, 2016 at 02:56:25PM +0200, Boris Brezillon wrote: > > On Thu, 15 Sep 2016 10:32:49 +0200 > > Sascha Hauer <s.hauer@pengutronix.de> wrote: > > > > > The nand layer will need ONFI mode 0 to use it as timing mode > > > before and right after reset. > > > > > > Signed-off-by: Sascha Hauer <s.hauer@pengutronix.de> > > > --- > > > drivers/mtd/nand/nand_timings.c | 11 +++++++++++ > > > include/linux/mtd/nand.h | 2 ++ > > > 2 files changed, 13 insertions(+) > > > > > > diff --git a/drivers/mtd/nand/nand_timings.c b/drivers/mtd/nand/nand_timings.c > > > index 608d098..9cdbc16 100644 > > > --- a/drivers/mtd/nand/nand_timings.c > > > +++ b/drivers/mtd/nand/nand_timings.c > > > @@ -297,3 +297,14 @@ int onfi_init_data_interface(struct nand_chip *chip, > > > > > > return 0; > > > } > > > + > > > +/** > > > + * nand_get_default_data_interface - [NAND Interface] Retrieve NAND > > > + * data interface for mode 0. This is used as default timing after > > > + * reset. > > > + */ > > > +const struct nand_data_interface *nand_get_default_data_interface(void) > > > +{ > > > + return &onfi_sdr_timings[0]; > > > +} > > > +EXPORT_SYMBOL(nand_get_default_data_interface); > > > > You export nand_get_default_data_interface() here, but you don't export > > onfi_init_data_interface(). > > None of these functions should be called from NAND controller drivers, > > so they don't need to be exported. ITOH, the prototypes are public > > (defined in nand.h), so nothing prevents a driver from calling these > > functions. > > A bit late, but we can still change things for the next cycle of > course... > > > I don't know what's the best solution here: > > 1/ Export both nand_get_default_data_interface() and > > onfi_init_data_interface() > > I don't really like this one. We shouldn't export them if we don't want > people to use them in modules. And I'm sure the various compile-test > builders out there would complain eventually if someone did. Yes, I had the feeling it was the wrong choice, but it was also the less invasive one ;-). > > > 2/ Do not export them, but keep their prototypes in nand.h with a > > comment saying that they should not be directly called by NAND > > controller drivers > > Could be OK. I don't like that one. It just bloats nand.h with things that are not supposed to be exported to NAND controller drivers, which just adds more confusion (the NAND framework is already hard to follow/understand, let's try not to make it worse). > > > 3/ Removing the prototypes from nand.h and defining them in nand_base.c > > (or in a private header). > > I like this one. Why not add a drivers/mtd/nand/nand.h that's intended > for the nand.{o,ko} module only? I like this one too. I'll try to come up with something (maybe with a different name, like core.h, to prevent developers from including this file in their NAND controller drivers). Thanks, Boris ^ permalink raw reply [flat|nested] 19+ messages in thread
* [PATCH 6/9] mtd: nand: automate NAND timings selection 2016-09-15 8:32 [PATCH v4] mtd: nand: automate NAND timings selection Sascha Hauer ` (4 preceding siblings ...) 2016-09-15 8:32 ` [PATCH 5/9] mtd: nand: Expose data interface for ONFI mode 0 Sascha Hauer @ 2016-09-15 8:32 ` Sascha Hauer 2016-09-15 8:32 ` [PATCH 7/9] mtd: nand: sunxi: switch from manual to automated timing config Sascha Hauer ` (3 subsequent siblings) 9 siblings, 0 replies; 19+ messages in thread From: Sascha Hauer @ 2016-09-15 8:32 UTC (permalink / raw) To: linux-arm-kernel From: Boris Brezillon <boris.brezillon@free-electrons.com> The NAND framework provides several helpers to query timing modes supported by a NAND chip, but this implies that all NAND controller drivers have to implement the same timings selection dance. Also currently NAND devices can be resetted at arbitrary places which also resets the timing for ONFI chips to timing mode 0. Provide a common logic to select the best timings based on ONFI or ->onfi_timing_mode_default information. Hook this into nand_reset() to make sure the new timing is applied each time during a reset. NAND controller willing to support timings adjustment should just implement the ->setup_data_interface() method. Signed-off-by: Boris Brezillon <boris.brezillon@free-electrons.com> Signed-off-by: Sascha Hauer <s.hauer@pengutronix.de> --- drivers/mtd/nand/nand_base.c | 156 +++++++++++++++++++++++++++++++++++++++++++ include/linux/mtd/nand.h | 14 ++-- 2 files changed, 166 insertions(+), 4 deletions(-) diff --git a/drivers/mtd/nand/nand_base.c b/drivers/mtd/nand/nand_base.c index 1f704cc..c7d500ea 100644 --- a/drivers/mtd/nand/nand_base.c +++ b/drivers/mtd/nand/nand_base.c @@ -948,6 +948,146 @@ static int nand_wait(struct mtd_info *mtd, struct nand_chip *chip) } /** + * nand_reset_data_interface - Reset data interface and timings + * @chip: The NAND chip + * + * Reset the Data interface and timings to ONFI mode 0. + * + * Returns 0 for success or negative error code otherwise. + */ +static int nand_reset_data_interface(struct nand_chip *chip) +{ + struct mtd_info *mtd = &chip->mtd; + const struct nand_data_interface *conf; + int ret; + + if (!chip->setup_data_interface) + return 0; + + /* + * The ONFI specification says: + * " + * To transition from NV-DDR or NV-DDR2 to the SDR data + * interface, the host shall use the Reset (FFh) command + * using SDR timing mode 0. A device in any timing mode is + * required to recognize Reset (FFh) command issued in SDR + * timing mode 0. + * " + * + * Configure the data interface in SDR mode and set the + * timings to timing mode 0. + */ + + conf = nand_get_default_data_interface(); + ret = chip->setup_data_interface(mtd, conf, false); + if (ret) + pr_err("Failed to configure data interface to SDR timing mode 0\n"); + + return ret; +} + +/** + * nand_setup_data_interface - Setup the best data interface and timings + * @chip: The NAND chip + * + * Find and configure the best data interface and NAND timings supported by + * the chip and the driver. + * First tries to retrieve supported timing modes from ONFI information, + * and if the NAND chip does not support ONFI, relies on the + * ->onfi_timing_mode_default specified in the nand_ids table. + * + * Returns 0 for success or negative error code otherwise. + */ +static int nand_setup_data_interface(struct nand_chip *chip) +{ + struct mtd_info *mtd = &chip->mtd; + int ret; + + if (!chip->setup_data_interface || !chip->data_interface) + return 0; + + /* + * Ensure the timing mode has been changed on the chip side + * before changing timings on the controller side. + */ + if (chip->onfi_version) { + uint8_t tmode_param[ONFI_SUBFEATURE_PARAM_LEN] = { + chip->onfi_timing_mode_default, + }; + + ret = chip->onfi_set_features(mtd, chip, + ONFI_FEATURE_ADDR_TIMING_MODE, + tmode_param); + if (ret) + goto err; + } + + ret = chip->setup_data_interface(mtd, chip->data_interface, false); +err: + return ret; +} + +/** + * nand_init_data_interface - find the best data interface and timings + * @chip: The NAND chip + * + * Find the best data interface and NAND timings supported by the chip + * and the driver. + * First tries to retrieve supported timing modes from ONFI information, + * and if the NAND chip does not support ONFI, relies on the + * ->onfi_timing_mode_default specified in the nand_ids table. After this + * function nand_chip->data_interface is initialized with the best timing mode + * available. + * + * Returns 0 for success or negative error code otherwise. + */ +static int nand_init_data_interface(struct nand_chip *chip) +{ + struct mtd_info *mtd = &chip->mtd; + int modes, mode, ret; + + if (!chip->setup_data_interface) + return 0; + + /* + * First try to identify the best timings from ONFI parameters and + * if the NAND does not support ONFI, fallback to the default ONFI + * timing mode. + */ + modes = onfi_get_async_timing_mode(chip); + if (modes == ONFI_TIMING_MODE_UNKNOWN) { + if (!chip->onfi_timing_mode_default) + return 0; + + modes = GENMASK(chip->onfi_timing_mode_default, 0); + } + + chip->data_interface = kzalloc(sizeof(*chip->data_interface), GFP_KERNEL); + if (!chip->data_interface) + return -ENOMEM; + + for (mode = fls(modes) - 1; mode >= 0; mode--) { + ret = onfi_init_data_interface(chip, chip->data_interface, + NAND_SDR_IFACE, mode); + if (ret) + continue; + + ret = chip->setup_data_interface(mtd, chip->data_interface, true); + if (!ret) { + chip->onfi_timing_mode_default = mode; + break; + } + } + + return 0; +} + +static void nand_release_data_interface(struct nand_chip *chip) +{ + kfree(chip->data_interface); +} + +/** * nand_reset - Reset and initialize a NAND device * @chip: The NAND chip * @@ -955,8 +1095,18 @@ static int nand_wait(struct mtd_info *mtd, struct nand_chip *chip) */ int nand_reset(struct nand_chip *chip) { + int ret; + + ret = nand_reset_data_interface(chip); + if (ret) + return ret; + chip->cmdfunc(&chip->mtd, NAND_CMD_RESET, -1, -1); + ret = nand_setup_data_interface(chip); + if (ret) + return ret; + return 0; } @@ -4168,6 +4318,10 @@ int nand_scan_ident(struct mtd_info *mtd, int maxchips, return PTR_ERR(type); } + ret = nand_init_data_interface(chip); + if (ret) + return ret; + chip->select_chip(mtd, -1); /* Check for a chip array */ @@ -4627,6 +4781,8 @@ void nand_release(struct mtd_info *mtd) mtd_device_unregister(mtd); + nand_release_data_interface(chip); + /* Free bad block table memory */ kfree(chip->bbt); if (!(chip->options & NAND_OWN_BUFFERS)) diff --git a/include/linux/mtd/nand.h b/include/linux/mtd/nand.h index 29ae324..24b4190 100644 --- a/include/linux/mtd/nand.h +++ b/include/linux/mtd/nand.h @@ -743,10 +743,9 @@ nand_get_sdr_timings(const struct nand_data_interface *conf) * also from the datasheet. It is the recommended ECC step * size, if known; if unknown, set to zero. * @onfi_timing_mode_default: [INTERN] default ONFI timing mode. This field is - * either deduced from the datasheet if the NAND - * chip is not ONFI compliant or set to 0 if it is - * (an ONFI chip is always configured in mode 0 - * after a NAND reset) + * set to the actually used ONFI mode if the chip is + * ONFI compliant or deduced from the datasheet if + * the NAND chip is not ONFI compliant. * @numchips: [INTERN] number of physical chips * @chipsize: [INTERN] the size of one chip for multichip arrays * @pagemask: [INTERN] page number mask = number of (pages / chip) - 1 @@ -766,6 +765,7 @@ nand_get_sdr_timings(const struct nand_data_interface *conf) * @read_retries: [INTERN] the number of read retry modes supported * @onfi_set_features: [REPLACEABLE] set the features for ONFI nand * @onfi_get_features: [REPLACEABLE] get the features for ONFI nand + * @setup_data_interface: [OPTIONAL] setup the data interface and timing * @bbt: [INTERN] bad block table pointer * @bbt_td: [REPLACEABLE] bad block table descriptor for flash * lookup. @@ -812,6 +812,10 @@ struct nand_chip { int (*onfi_get_features)(struct mtd_info *mtd, struct nand_chip *chip, int feature_addr, uint8_t *subfeature_para); int (*setup_read_retry)(struct mtd_info *mtd, int retry_mode); + int (*setup_data_interface)(struct mtd_info *mtd, + const struct nand_data_interface *conf, + bool check_only); + int chip_delay; unsigned int options; @@ -841,6 +845,8 @@ struct nand_chip { struct nand_jedec_params jedec_params; }; + struct nand_data_interface *data_interface; + int read_retries; flstate_t state; -- 2.8.1 ^ permalink raw reply related [flat|nested] 19+ messages in thread
* [PATCH 7/9] mtd: nand: sunxi: switch from manual to automated timing config 2016-09-15 8:32 [PATCH v4] mtd: nand: automate NAND timings selection Sascha Hauer ` (5 preceding siblings ...) 2016-09-15 8:32 ` [PATCH 6/9] mtd: nand: automate NAND timings selection Sascha Hauer @ 2016-09-15 8:32 ` Sascha Hauer 2016-09-15 8:32 ` [PATCH 8/9] mtd: nand: mxc: implement onfi get/set features Sascha Hauer ` (2 subsequent siblings) 9 siblings, 0 replies; 19+ messages in thread From: Sascha Hauer @ 2016-09-15 8:32 UTC (permalink / raw) To: linux-arm-kernel The NAND framework is now able to select the best NAND timings for us. All we have to do is implement a ->setup_data_interface() function to apply those timings and remove the timing selection code from the sunxi driver. Signed-off-by: Sascha Hauer <s.hauer@pengutronix.de> --- drivers/mtd/nand/sunxi_nand.c | 76 ++++++++----------------------------------- 1 file changed, 14 insertions(+), 62 deletions(-) diff --git a/drivers/mtd/nand/sunxi_nand.c b/drivers/mtd/nand/sunxi_nand.c index e414b31..8c59a10 100644 --- a/drivers/mtd/nand/sunxi_nand.c +++ b/drivers/mtd/nand/sunxi_nand.c @@ -1572,14 +1572,22 @@ static int _sunxi_nand_lookup_timing(const s32 *lut, int lut_size, u32 duration, #define sunxi_nand_lookup_timing(l, p, c) \ _sunxi_nand_lookup_timing(l, ARRAY_SIZE(l), p, c) -static int sunxi_nand_chip_set_timings(struct sunxi_nand_chip *chip, - const struct nand_sdr_timings *timings) +static int sunxi_nfc_setup_data_interface(struct mtd_info *mtd, + const struct nand_data_interface *conf, + bool check_only) { + struct nand_chip *nand = mtd_to_nand(mtd); + struct sunxi_nand_chip *chip = to_sunxi_nand(nand); struct sunxi_nfc *nfc = to_sunxi_nfc(chip->nand.controller); + const struct nand_sdr_timings *timings; u32 min_clk_period = 0; s32 tWB, tADL, tWHR, tRHW, tCAD; long real_clk_rate; + timings = nand_get_sdr_timings(conf); + if (IS_ERR(timings)) + return -ENOTSUPP; + /* T1 <=> tCLS */ if (timings->tCLS_min > min_clk_period) min_clk_period = timings->tCLS_min; @@ -1679,6 +1687,9 @@ static int sunxi_nand_chip_set_timings(struct sunxi_nand_chip *chip, return tRHW; } + if (check_only) + return 0; + /* * TODO: according to ONFI specs this value only applies for DDR NAND, * but Allwinner seems to set this to 0x7. Mimic them for now. @@ -1712,44 +1723,6 @@ static int sunxi_nand_chip_set_timings(struct sunxi_nand_chip *chip, return 0; } -static int sunxi_nand_chip_init_timings(struct sunxi_nand_chip *chip, - struct device_node *np) -{ - struct mtd_info *mtd = nand_to_mtd(&chip->nand); - const struct nand_sdr_timings *timings; - int ret; - int mode; - - mode = onfi_get_async_timing_mode(&chip->nand); - if (mode == ONFI_TIMING_MODE_UNKNOWN) { - mode = chip->nand.onfi_timing_mode_default; - } else { - uint8_t feature[ONFI_SUBFEATURE_PARAM_LEN] = {}; - int i; - - mode = fls(mode) - 1; - if (mode < 0) - mode = 0; - - feature[0] = mode; - for (i = 0; i < chip->nsels; i++) { - chip->nand.select_chip(mtd, i); - ret = chip->nand.onfi_set_features(mtd, &chip->nand, - ONFI_FEATURE_ADDR_TIMING_MODE, - feature); - chip->nand.select_chip(mtd, -1); - if (ret) - return ret; - } - } - - timings = onfi_async_timing_mode_to_sdr_timings(mode); - if (IS_ERR(timings)) - return PTR_ERR(timings); - - return sunxi_nand_chip_set_timings(chip, timings); -} - static int sunxi_nand_ooblayout_ecc(struct mtd_info *mtd, int section, struct mtd_oob_region *oobregion) { @@ -1975,7 +1948,6 @@ static int sunxi_nand_ecc_init(struct mtd_info *mtd, struct nand_ecc_ctrl *ecc, static int sunxi_nand_chip_init(struct device *dev, struct sunxi_nfc *nfc, struct device_node *np) { - const struct nand_sdr_timings *timings; struct sunxi_nand_chip *chip; struct mtd_info *mtd; struct nand_chip *nand; @@ -2065,25 +2037,11 @@ static int sunxi_nand_chip_init(struct device *dev, struct sunxi_nfc *nfc, nand->read_buf = sunxi_nfc_read_buf; nand->write_buf = sunxi_nfc_write_buf; nand->read_byte = sunxi_nfc_read_byte; + nand->setup_data_interface = sunxi_nfc_setup_data_interface; mtd = nand_to_mtd(nand); mtd->dev.parent = dev; - timings = onfi_async_timing_mode_to_sdr_timings(0); - if (IS_ERR(timings)) { - ret = PTR_ERR(timings); - dev_err(dev, - "could not retrieve timings for ONFI mode 0: %d\n", - ret); - return ret; - } - - ret = sunxi_nand_chip_set_timings(chip, timings); - if (ret) { - dev_err(dev, "could not configure chip timings: %d\n", ret); - return ret; - } - ret = nand_scan_ident(mtd, nsels, NULL); if (ret) return ret; @@ -2096,12 +2054,6 @@ static int sunxi_nand_chip_init(struct device *dev, struct sunxi_nfc *nfc, nand->options |= NAND_SUBPAGE_READ; - ret = sunxi_nand_chip_init_timings(chip, np); - if (ret) { - dev_err(dev, "could not configure chip timings: %d\n", ret); - return ret; - } - ret = sunxi_nand_ecc_init(mtd, &nand->ecc, np); if (ret) { dev_err(dev, "ECC init failed: %d\n", ret); -- 2.8.1 ^ permalink raw reply related [flat|nested] 19+ messages in thread
* [PATCH 8/9] mtd: nand: mxc: implement onfi get/set features 2016-09-15 8:32 [PATCH v4] mtd: nand: automate NAND timings selection Sascha Hauer ` (6 preceding siblings ...) 2016-09-15 8:32 ` [PATCH 7/9] mtd: nand: sunxi: switch from manual to automated timing config Sascha Hauer @ 2016-09-15 8:32 ` Sascha Hauer 2016-11-21 13:56 ` Boris Brezillon 2016-09-15 8:32 ` [PATCH 9/9] mtd: nand: mxc: Add timing setup for v2 controllers Sascha Hauer 2016-09-16 12:19 ` [PATCH v4] mtd: nand: automate NAND timings selection Boris Brezillon 9 siblings, 1 reply; 19+ messages in thread From: Sascha Hauer @ 2016-09-15 8:32 UTC (permalink / raw) To: linux-arm-kernel To be able to support different ONFI timing modes we have to implement the onfi_set_features and onfi_get_features. Tested on an i.MX25 SoC. Signed-off-by: Sascha Hauer <s.hauer@pengutronix.de> --- drivers/mtd/nand/mxc_nand.c | 53 +++++++++++++++++++++++++++++++++++++++++++++ 1 file changed, 53 insertions(+) diff --git a/drivers/mtd/nand/mxc_nand.c b/drivers/mtd/nand/mxc_nand.c index 5173fad..1db8299 100644 --- a/drivers/mtd/nand/mxc_nand.c +++ b/drivers/mtd/nand/mxc_nand.c @@ -1239,6 +1239,57 @@ static void mxc_nand_command(struct mtd_info *mtd, unsigned command, } } +static int mxc_nand_onfi_set_features(struct mtd_info *mtd, struct nand_chip *chip, + int addr, uint8_t *subfeature_param) +{ + struct nand_chip *nand_chip = mtd_to_nand(mtd); + struct mxc_nand_host *host = nand_get_controller_data(nand_chip); + int i; + + if (!chip->onfi_version || + !(le16_to_cpu(chip->onfi_params.opt_cmd) + & ONFI_OPT_CMD_SET_GET_FEATURES)) + return -EINVAL; + + host->buf_start = 0; + + for (i = 0; i < ONFI_SUBFEATURE_PARAM_LEN; ++i) + chip->write_byte(mtd, subfeature_param[i]); + + memcpy32_toio(host->main_area0, host->data_buf, mtd->writesize); + host->devtype_data->send_cmd(host, NAND_CMD_SET_FEATURES, false); + mxc_do_addr_cycle(mtd, addr, -1); + host->devtype_data->send_page(mtd, NFC_INPUT); + + return 0; +} + +static int mxc_nand_onfi_get_features(struct mtd_info *mtd, struct nand_chip *chip, + int addr, uint8_t *subfeature_param) +{ + struct nand_chip *nand_chip = mtd_to_nand(mtd); + struct mxc_nand_host *host = nand_get_controller_data(nand_chip); + int i; + + if (!chip->onfi_version || + !(le16_to_cpu(chip->onfi_params.opt_cmd) + & ONFI_OPT_CMD_SET_GET_FEATURES)) + return -EINVAL; + + *(uint32_t *)host->main_area0 = 0xdeadbeef; + + host->devtype_data->send_cmd(host, NAND_CMD_GET_FEATURES, false); + mxc_do_addr_cycle(mtd, addr, -1); + host->devtype_data->send_page(mtd, NFC_OUTPUT); + memcpy32_fromio(host->data_buf, host->main_area0, 512); + host->buf_start = 0; + + for (i = 0; i < ONFI_SUBFEATURE_PARAM_LEN; ++i) + *subfeature_param++ = chip->read_byte(mtd); + + return 0; +} + /* * The generic flash bbt decriptors overlap with our ecc * hardware, so define some i.MX specific ones. @@ -1513,6 +1564,8 @@ static int mxcnd_probe(struct platform_device *pdev) this->read_word = mxc_nand_read_word; this->write_buf = mxc_nand_write_buf; this->read_buf = mxc_nand_read_buf; + this->onfi_set_features = mxc_nand_onfi_set_features; + this->onfi_get_features = mxc_nand_onfi_get_features; host->clk = devm_clk_get(&pdev->dev, NULL); if (IS_ERR(host->clk)) -- 2.8.1 ^ permalink raw reply related [flat|nested] 19+ messages in thread
* [PATCH 8/9] mtd: nand: mxc: implement onfi get/set features 2016-09-15 8:32 ` [PATCH 8/9] mtd: nand: mxc: implement onfi get/set features Sascha Hauer @ 2016-11-21 13:56 ` Boris Brezillon 2016-11-22 10:52 ` Sascha Hauer 0 siblings, 1 reply; 19+ messages in thread From: Boris Brezillon @ 2016-11-21 13:56 UTC (permalink / raw) To: linux-arm-kernel Hi Sascha, On Thu, 15 Sep 2016 10:32:52 +0200 Sascha Hauer <s.hauer@pengutronix.de> wrote: > To be able to support different ONFI timing modes we have to implement > the onfi_set_features and onfi_get_features. Tested on an i.MX25 SoC. > > Signed-off-by: Sascha Hauer <s.hauer@pengutronix.de> > --- > drivers/mtd/nand/mxc_nand.c | 53 +++++++++++++++++++++++++++++++++++++++++++++ > 1 file changed, 53 insertions(+) > > diff --git a/drivers/mtd/nand/mxc_nand.c b/drivers/mtd/nand/mxc_nand.c > index 5173fad..1db8299 100644 > --- a/drivers/mtd/nand/mxc_nand.c > +++ b/drivers/mtd/nand/mxc_nand.c > @@ -1239,6 +1239,57 @@ static void mxc_nand_command(struct mtd_info *mtd, unsigned command, > } > } > > +static int mxc_nand_onfi_set_features(struct mtd_info *mtd, struct nand_chip *chip, > + int addr, uint8_t *subfeature_param) > +{ > + struct nand_chip *nand_chip = mtd_to_nand(mtd); > + struct mxc_nand_host *host = nand_get_controller_data(nand_chip); > + int i; > + > + if (!chip->onfi_version || > + !(le16_to_cpu(chip->onfi_params.opt_cmd) > + & ONFI_OPT_CMD_SET_GET_FEATURES)) > + return -EINVAL; > + > + host->buf_start = 0; > + > + for (i = 0; i < ONFI_SUBFEATURE_PARAM_LEN; ++i) > + chip->write_byte(mtd, subfeature_param[i]); > + > + memcpy32_toio(host->main_area0, host->data_buf, mtd->writesize); > + host->devtype_data->send_cmd(host, NAND_CMD_SET_FEATURES, false); > + mxc_do_addr_cycle(mtd, addr, -1); > + host->devtype_data->send_page(mtd, NFC_INPUT); I've been working with an mx27 board embedding a NAND device lately, and had a closer look at the NAND controller IP. With this IP, you're not able to send only 4 bytes of data, and I'm sure sure what you're doing here (sending a full page of data) works for a SET_FEATURE command. Do you have a way to test it (my NAND is not ONFI compliant)? By test it, I mean, set a timing mode using SET_FEATURE and check if the new mode has been applied using GET_FEATURE. > + > + return 0; > +} > + > +static int mxc_nand_onfi_get_features(struct mtd_info *mtd, struct nand_chip *chip, > + int addr, uint8_t *subfeature_param) > +{ > + struct nand_chip *nand_chip = mtd_to_nand(mtd); > + struct mxc_nand_host *host = nand_get_controller_data(nand_chip); > + int i; > + > + if (!chip->onfi_version || > + !(le16_to_cpu(chip->onfi_params.opt_cmd) > + & ONFI_OPT_CMD_SET_GET_FEATURES)) > + return -EINVAL; > + > + *(uint32_t *)host->main_area0 = 0xdeadbeef; > + > + host->devtype_data->send_cmd(host, NAND_CMD_GET_FEATURES, false); > + mxc_do_addr_cycle(mtd, addr, -1); > + host->devtype_data->send_page(mtd, NFC_OUTPUT); > + memcpy32_fromio(host->data_buf, host->main_area0, 512); > + host->buf_start = 0; > + > + for (i = 0; i < ONFI_SUBFEATURE_PARAM_LEN; ++i) > + *subfeature_param++ = chip->read_byte(mtd); > + > + return 0; > +} > + > /* > * The generic flash bbt decriptors overlap with our ecc > * hardware, so define some i.MX specific ones. > @@ -1513,6 +1564,8 @@ static int mxcnd_probe(struct platform_device *pdev) > this->read_word = mxc_nand_read_word; > this->write_buf = mxc_nand_write_buf; > this->read_buf = mxc_nand_read_buf; > + this->onfi_set_features = mxc_nand_onfi_set_features; > + this->onfi_get_features = mxc_nand_onfi_get_features; > > host->clk = devm_clk_get(&pdev->dev, NULL); > if (IS_ERR(host->clk)) ^ permalink raw reply [flat|nested] 19+ messages in thread
* [PATCH 8/9] mtd: nand: mxc: implement onfi get/set features 2016-11-21 13:56 ` Boris Brezillon @ 2016-11-22 10:52 ` Sascha Hauer 2016-11-22 11:03 ` Boris Brezillon 0 siblings, 1 reply; 19+ messages in thread From: Sascha Hauer @ 2016-11-22 10:52 UTC (permalink / raw) To: linux-arm-kernel Hi Boris, On Mon, Nov 21, 2016 at 02:56:43PM +0100, Boris Brezillon wrote: > Hi Sascha, > > On Thu, 15 Sep 2016 10:32:52 +0200 > Sascha Hauer <s.hauer@pengutronix.de> wrote: > > > To be able to support different ONFI timing modes we have to implement > > the onfi_set_features and onfi_get_features. Tested on an i.MX25 SoC. > > > > Signed-off-by: Sascha Hauer <s.hauer@pengutronix.de> > > --- > > drivers/mtd/nand/mxc_nand.c | 53 +++++++++++++++++++++++++++++++++++++++++++++ > > 1 file changed, 53 insertions(+) > > > > diff --git a/drivers/mtd/nand/mxc_nand.c b/drivers/mtd/nand/mxc_nand.c > > index 5173fad..1db8299 100644 > > --- a/drivers/mtd/nand/mxc_nand.c > > +++ b/drivers/mtd/nand/mxc_nand.c > > @@ -1239,6 +1239,57 @@ static void mxc_nand_command(struct mtd_info *mtd, unsigned command, > > } > > } > > > > +static int mxc_nand_onfi_set_features(struct mtd_info *mtd, struct nand_chip *chip, > > + int addr, uint8_t *subfeature_param) > > +{ > > + struct nand_chip *nand_chip = mtd_to_nand(mtd); > > + struct mxc_nand_host *host = nand_get_controller_data(nand_chip); > > + int i; > > + > > + if (!chip->onfi_version || > > + !(le16_to_cpu(chip->onfi_params.opt_cmd) > > + & ONFI_OPT_CMD_SET_GET_FEATURES)) > > + return -EINVAL; > > + > > + host->buf_start = 0; > > + > > + for (i = 0; i < ONFI_SUBFEATURE_PARAM_LEN; ++i) > > + chip->write_byte(mtd, subfeature_param[i]); > > + > > + memcpy32_toio(host->main_area0, host->data_buf, mtd->writesize); > > + host->devtype_data->send_cmd(host, NAND_CMD_SET_FEATURES, false); > > + mxc_do_addr_cycle(mtd, addr, -1); > > + host->devtype_data->send_page(mtd, NFC_INPUT); > > I've been working with an mx27 board embedding a NAND device lately, > and had a closer look at the NAND controller IP. > With this IP, you're not able to send only 4 bytes of data, and I'm > sure sure what you're doing here (sending a full page of data) works > for a SET_FEATURE command. > > Do you have a way to test it (my NAND is not ONFI compliant)? By test > it, I mean, set a timing mode using SET_FEATURE and check if the new > mode has been applied using GET_FEATURE. I have an i.MX27 board with ONFI flash, but this one does not have the ONFI_OPT_CMD_SET_GET_FEATURES bit set, so I can't test it there. However, I can confirm that it works on an i.MX25. With the attached patch applied on vanilla v4.9-rc5 I get: GET FEATURES. chip->onfi_timing_mode_default: 4 timing before: 0x00 timing after: 0x04 Sascha -------------------------8<------------------------------- >From 38e45851a796ba47e25c72ebc58e358c70e18275 Mon Sep 17 00:00:00 2001 From: Sascha Hauer <s.hauer@pengutronix.de> Date: Tue, 22 Nov 2016 11:48:36 +0100 Subject: [PATCH] wip Signed-off-by: Sascha Hauer <s.hauer@pengutronix.de> --- drivers/mtd/nand/nand_base.c | 12 ++++++++++++ 1 file changed, 12 insertions(+) diff --git a/drivers/mtd/nand/nand_base.c b/drivers/mtd/nand/nand_base.c index 3bde96a..5b5be2b 100644 --- a/drivers/mtd/nand/nand_base.c +++ b/drivers/mtd/nand/nand_base.c @@ -1017,12 +1017,24 @@ static int nand_setup_data_interface(struct nand_chip *chip) u8 tmode_param[ONFI_SUBFEATURE_PARAM_LEN] = { chip->onfi_timing_mode_default, }; + u8 rmode_param[ONFI_SUBFEATURE_PARAM_LEN] = {}; + + printk("GET FEATURES. chip->onfi_timing_mode_default: %d\n", chip->onfi_timing_mode_default); + ret = chip->onfi_get_features(mtd, chip, ONFI_FEATURE_ADDR_TIMING_MODE, rmode_param); + if (ret) + goto err; + printk("timing before: 0x%02x\n", rmode_param[0]); ret = chip->onfi_set_features(mtd, chip, ONFI_FEATURE_ADDR_TIMING_MODE, tmode_param); if (ret) goto err; + + ret = chip->onfi_get_features(mtd, chip, ONFI_FEATURE_ADDR_TIMING_MODE, rmode_param); + if (ret) + goto err; + printk("timing after: 0x%02x\n", rmode_param[0]); } ret = chip->setup_data_interface(mtd, chip->data_interface, false); -- 2.10.2 -- Pengutronix e.K. | | Industrial Linux Solutions | http://www.pengutronix.de/ | Peiner Str. 6-8, 31137 Hildesheim, Germany | Phone: +49-5121-206917-0 | Amtsgericht Hildesheim, HRA 2686 | Fax: +49-5121-206917-5555 | ^ permalink raw reply related [flat|nested] 19+ messages in thread
* [PATCH 8/9] mtd: nand: mxc: implement onfi get/set features 2016-11-22 10:52 ` Sascha Hauer @ 2016-11-22 11:03 ` Boris Brezillon 0 siblings, 0 replies; 19+ messages in thread From: Boris Brezillon @ 2016-11-22 11:03 UTC (permalink / raw) To: linux-arm-kernel On Tue, 22 Nov 2016 11:52:53 +0100 Sascha Hauer <s.hauer@pengutronix.de> wrote: > Hi Boris, > > On Mon, Nov 21, 2016 at 02:56:43PM +0100, Boris Brezillon wrote: > > Hi Sascha, > > > > On Thu, 15 Sep 2016 10:32:52 +0200 > > Sascha Hauer <s.hauer@pengutronix.de> wrote: > > > > > To be able to support different ONFI timing modes we have to implement > > > the onfi_set_features and onfi_get_features. Tested on an i.MX25 SoC. > > > > > > Signed-off-by: Sascha Hauer <s.hauer@pengutronix.de> > > > --- > > > drivers/mtd/nand/mxc_nand.c | 53 +++++++++++++++++++++++++++++++++++++++++++++ > > > 1 file changed, 53 insertions(+) > > > > > > diff --git a/drivers/mtd/nand/mxc_nand.c b/drivers/mtd/nand/mxc_nand.c > > > index 5173fad..1db8299 100644 > > > --- a/drivers/mtd/nand/mxc_nand.c > > > +++ b/drivers/mtd/nand/mxc_nand.c > > > @@ -1239,6 +1239,57 @@ static void mxc_nand_command(struct mtd_info *mtd, unsigned command, > > > } > > > } > > > > > > +static int mxc_nand_onfi_set_features(struct mtd_info *mtd, struct nand_chip *chip, > > > + int addr, uint8_t *subfeature_param) > > > +{ > > > + struct nand_chip *nand_chip = mtd_to_nand(mtd); > > > + struct mxc_nand_host *host = nand_get_controller_data(nand_chip); > > > + int i; > > > + > > > + if (!chip->onfi_version || > > > + !(le16_to_cpu(chip->onfi_params.opt_cmd) > > > + & ONFI_OPT_CMD_SET_GET_FEATURES)) > > > + return -EINVAL; > > > + > > > + host->buf_start = 0; > > > + > > > + for (i = 0; i < ONFI_SUBFEATURE_PARAM_LEN; ++i) > > > + chip->write_byte(mtd, subfeature_param[i]); > > > + > > > + memcpy32_toio(host->main_area0, host->data_buf, mtd->writesize); > > > + host->devtype_data->send_cmd(host, NAND_CMD_SET_FEATURES, false); > > > + mxc_do_addr_cycle(mtd, addr, -1); > > > + host->devtype_data->send_page(mtd, NFC_INPUT); > > > > I've been working with an mx27 board embedding a NAND device lately, > > and had a closer look at the NAND controller IP. > > With this IP, you're not able to send only 4 bytes of data, and I'm > > sure sure what you're doing here (sending a full page of data) works > > for a SET_FEATURE command. > > > > Do you have a way to test it (my NAND is not ONFI compliant)? By test > > it, I mean, set a timing mode using SET_FEATURE and check if the new > > mode has been applied using GET_FEATURE. > > I have an i.MX27 board with ONFI flash, but this one does not have the > ONFI_OPT_CMD_SET_GET_FEATURES bit set, so I can't test it there. > However, I can confirm that it works on an i.MX25. With the attached > patch applied on vanilla v4.9-rc5 I get: > > GET FEATURES. chip->onfi_timing_mode_default: 4 > timing before: 0x00 > timing after: 0x04 Okay, cool. I guess most NANDs are more tolerant than what's strictly required in the ONFI spec. Thanks for testing. Boris ^ permalink raw reply [flat|nested] 19+ messages in thread
* [PATCH 9/9] mtd: nand: mxc: Add timing setup for v2 controllers 2016-09-15 8:32 [PATCH v4] mtd: nand: automate NAND timings selection Sascha Hauer ` (7 preceding siblings ...) 2016-09-15 8:32 ` [PATCH 8/9] mtd: nand: mxc: implement onfi get/set features Sascha Hauer @ 2016-09-15 8:32 ` Sascha Hauer 2016-09-16 12:19 ` [PATCH v4] mtd: nand: automate NAND timings selection Boris Brezillon 9 siblings, 0 replies; 19+ messages in thread From: Sascha Hauer @ 2016-09-15 8:32 UTC (permalink / raw) To: linux-arm-kernel So far we relied on reset default or the bootloader to configure a suitable clk rate for the Nand controller. This works but we can optimize the timing for better performance. This sets the clk rate for v2 controllers (i.MX25/35) based on the timing mode read from the ONFI parameter page. This may also enable the symmetric mode (aks EDO mode) if necessary which reads one word per clock cycle. Tested on an i.MX25 with a Micron MT29F4G08ABBDAHC attached. Signed-off-by: Sascha Hauer <s.hauer@pengutronix.de> --- drivers/mtd/nand/mxc_nand.c | 84 +++++++++++++++++++++++++++++++++++++++++++-- 1 file changed, 82 insertions(+), 2 deletions(-) diff --git a/drivers/mtd/nand/mxc_nand.c b/drivers/mtd/nand/mxc_nand.c index 1db8299..0d34d72 100644 --- a/drivers/mtd/nand/mxc_nand.c +++ b/drivers/mtd/nand/mxc_nand.c @@ -152,6 +152,9 @@ struct mxc_nand_devtype_data { void (*select_chip)(struct mtd_info *mtd, int chip); int (*correct_data)(struct mtd_info *mtd, u_char *dat, u_char *read_ecc, u_char *calc_ecc); + int (*setup_data_interface)(struct mtd_info *mtd, + const struct nand_data_interface *conf, + bool check_only); /* * On i.MX21 the CONFIG2:INT bit cannot be read if interrupts are masked @@ -1012,6 +1015,82 @@ static void preset_v1(struct mtd_info *mtd) writew(0x4, NFC_V1_V2_WRPROT); } +static int mxc_nand_v2_setup_data_interface(struct mtd_info *mtd, + const struct nand_data_interface *conf, + bool check_only) +{ + struct nand_chip *nand_chip = mtd_to_nand(mtd); + struct mxc_nand_host *host = nand_get_controller_data(nand_chip); + int tRC_min_ns, tRC_ps, ret; + unsigned long rate, rate_round; + const struct nand_sdr_timings *timings; + uint16_t config1; + + timings = nand_get_sdr_timings(conf); + if (IS_ERR(timings)) + return -ENOTSUPP; + + config1 = readw(NFC_V1_V2_CONFIG1); + + tRC_min_ns = timings->tRC_min / 1000; + rate = 1000000000 / tRC_min_ns; + + /* + * For tRC < 30ns we have to use EDO mode. In this case the controller + * does one access per clock cycle. Otherwise the controller does one + * access in two clock cycles, thus we have to double the rate to the + * controller. + */ + if (tRC_min_ns < 30) { + rate_round = clk_round_rate(host->clk, rate); + config1 |= NFC_V2_CONFIG1_ONE_CYCLE; + tRC_ps = 1000000000 / (rate_round / 1000); + } else { + rate *= 2; + rate_round = clk_round_rate(host->clk, rate); + config1 &= ~NFC_V2_CONFIG1_ONE_CYCLE; + tRC_ps = 1000000000 / (rate_round / 1000 / 2); + } + + /* + * The timing values compared against are from the i.MX25 Automotive + * datasheet, Table 50. NFC Timing Parameters + */ + if (timings->tCLS_min > tRC_ps - 1000 || + timings->tCLH_min > tRC_ps - 2000 || + timings->tCS_min > tRC_ps - 1000 || + timings->tCH_min > tRC_ps - 2000 || + timings->tWP_min > tRC_ps - 1500 || + timings->tALS_min > tRC_ps || + timings->tALH_min > tRC_ps - 3000 || + timings->tDS_min > tRC_ps || + timings->tDH_min > tRC_ps - 5000 || + timings->tWC_min > 2 * tRC_ps || + timings->tWH_min > tRC_ps - 2500 || + timings->tRR_min > 6 * tRC_ps || + timings->tRP_min > 3 * tRC_ps / 2 || + timings->tRC_min > 2 * tRC_ps || + timings->tREH_min > (tRC_ps / 2) - 2500) { + dev_dbg(host->dev, "Timing out of bounds\n"); + return -EINVAL; + } + + if (check_only) + return 0; + + ret = clk_set_rate(host->clk, rate); + if (ret) + return ret; + + writew(config1, NFC_V1_V2_CONFIG1); + + dev_dbg(host->dev, "Setting rate to %ldHz, %s mode\n", rate_round, + config1 & NFC_V2_CONFIG1_ONE_CYCLE ? "One cycle (EDO)" : + "normal"); + + return 0; +} + static void preset_v2(struct mtd_info *mtd) { struct nand_chip *nand_chip = mtd_to_nand(mtd); @@ -1276,8 +1355,6 @@ static int mxc_nand_onfi_get_features(struct mtd_info *mtd, struct nand_chip *ch & ONFI_OPT_CMD_SET_GET_FEATURES)) return -EINVAL; - *(uint32_t *)host->main_area0 = 0xdeadbeef; - host->devtype_data->send_cmd(host, NAND_CMD_GET_FEATURES, false); mxc_do_addr_cycle(mtd, addr, -1); host->devtype_data->send_page(mtd, NFC_OUTPUT); @@ -1378,6 +1455,7 @@ static const struct mxc_nand_devtype_data imx25_nand_devtype_data = { .ooblayout = &mxc_v2_ooblayout_ops, .select_chip = mxc_nand_select_chip_v2, .correct_data = mxc_nand_correct_data_v2_v3, + .setup_data_interface = mxc_nand_v2_setup_data_interface, .irqpending_quirk = 0, .needs_ip = 0, .regs_offset = 0x1e00, @@ -1586,6 +1664,8 @@ static int mxcnd_probe(struct platform_device *pdev) if (err < 0) return err; + this->setup_data_interface = host->devtype_data->setup_data_interface; + if (host->devtype_data->needs_ip) { res = platform_get_resource(pdev, IORESOURCE_MEM, 0); host->regs_ip = devm_ioremap_resource(&pdev->dev, res); -- 2.8.1 ^ permalink raw reply related [flat|nested] 19+ messages in thread
* [PATCH v4] mtd: nand: automate NAND timings selection 2016-09-15 8:32 [PATCH v4] mtd: nand: automate NAND timings selection Sascha Hauer ` (8 preceding siblings ...) 2016-09-15 8:32 ` [PATCH 9/9] mtd: nand: mxc: Add timing setup for v2 controllers Sascha Hauer @ 2016-09-16 12:19 ` Boris Brezillon 2016-09-19 6:43 ` Sascha Hauer 9 siblings, 1 reply; 19+ messages in thread From: Boris Brezillon @ 2016-09-16 12:19 UTC (permalink / raw) To: linux-arm-kernel On Thu, 15 Sep 2016 10:32:44 +0200 Sascha Hauer <s.hauer@pengutronix.de> wrote: > This series aims at automating the NAND timings selection which is > currently supposed to be done in each NAND controller driver, thus > simplifying drivers implementation. > > As suggested by Boris this version of the series introduces a nand_reset() > function which replaces the several open coded NAND_CMD_RESET commands > in the code. This makes sure we can apply the timing each time after > after reset. > > Also I have brought back the conversion patch for teh sunxi driver whic > was part of Boris initial posting. It's untested due to the lack of hardware, > so please test before applying. Applied after fixing a few coding style issues (to make checkpatch happy), and exporting the onfi_init_data_interface() function (not sure it's the best solution, but at least, it's consistent with the other functions defined in nand_timings.c). Thanks, Boris > > Sascha > > Changes since v4: > - Change onfi_init_data_interface() prototype to be more future proof as > requested by Boris > > Changes since v3: > - Bring back patch dropped in v3 > - Use statically allocated default timing for all chips and store one > optimized timing in struct nand_chip > > Changes since v2: > - Add accessor function to get the SDR timing from struct nand_data_interface > - Change nand_reset() argument to struct nand_chip > - Drop conversion of nand_timing array to struct nand_data_interface > - Recalculate timing whenever needed instead of storing a pointer in struct > nand_chip > - some more refactoring > > Changes since v1: > - create a nand_reset() function to create a single place to reset NAND > chips and to apply timings > - Add patch to convert sunxi driver for automated timing setup > - split into more patches > > Changes since the initial posting from Boris: > > - Integrate Feedback from Ezequiel Garcia > - When iterating over the chips calling onfi_set_features() for each > bail out when any of the calls fail, not only the last one. > - When one of the onfi_set_features() calls fail then reset the chipi > afterwards. > - Drop Sunxi example, add patch for the mxc_nand controller instead. > > ---------------------------------------------------------------- > > Boris Brezillon (1): > mtd: nand: automate NAND timings selection > > Sascha Hauer (8): > mtd: nand: Create a NAND reset function > mtd: nand: Introduce nand_data_interface > mtd: nand: convert ONFI mode into data interface > mtd: nand: Add function to convert ONFI mode to data_interface > mtd: nand: Expose data interface for ONFI mode 0 > mtd: nand: sunxi: switch from manual to automated timing config > mtd: nand: mxc: implement onfi get/set features > mtd: nand: mxc: Add timing setup for v2 controllers > > drivers/mtd/nand/mxc_nand.c | 133 ++++++++++++ > drivers/mtd/nand/nand_base.c | 179 ++++++++++++++- > drivers/mtd/nand/nand_timings.c | 469 ++++++++++++++++++++++------------------ > drivers/mtd/nand/sunxi_nand.c | 76 ++----- > include/linux/mtd/nand.h | 190 +++++++++++----- > 5 files changed, 721 insertions(+), 326 deletions(-) ^ permalink raw reply [flat|nested] 19+ messages in thread
* [PATCH v4] mtd: nand: automate NAND timings selection 2016-09-16 12:19 ` [PATCH v4] mtd: nand: automate NAND timings selection Boris Brezillon @ 2016-09-19 6:43 ` Sascha Hauer 0 siblings, 0 replies; 19+ messages in thread From: Sascha Hauer @ 2016-09-19 6:43 UTC (permalink / raw) To: linux-arm-kernel On Fri, Sep 16, 2016 at 02:19:18PM +0200, Boris Brezillon wrote: > On Thu, 15 Sep 2016 10:32:44 +0200 > Sascha Hauer <s.hauer@pengutronix.de> wrote: > > > This series aims at automating the NAND timings selection which is > > currently supposed to be done in each NAND controller driver, thus > > simplifying drivers implementation. > > > > As suggested by Boris this version of the series introduces a nand_reset() > > function which replaces the several open coded NAND_CMD_RESET commands > > in the code. This makes sure we can apply the timing each time after > > after reset. > > > > Also I have brought back the conversion patch for teh sunxi driver whic > > was part of Boris initial posting. It's untested due to the lack of hardware, > > so please test before applying. > > Applied after fixing a few coding style issues (to make checkpatch > happy), and exporting the onfi_init_data_interface() function (not sure > it's the best solution, but at least, it's consistent with the other > functions defined in nand_timings.c). Thanks Boris. Sascha -- Pengutronix e.K. | | Industrial Linux Solutions | http://www.pengutronix.de/ | Peiner Str. 6-8, 31137 Hildesheim, Germany | Phone: +49-5121-206917-0 | Amtsgericht Hildesheim, HRA 2686 | Fax: +49-5121-206917-5555 | ^ permalink raw reply [flat|nested] 19+ messages in thread
* [PATCH v4] mtd: nand: automate NAND timings selection
@ 2016-09-09 12:05 Sascha Hauer
  0 siblings, 0 replies; 19+ messages in thread
From: Sascha Hauer @ 2016-09-09 12:05 UTC (permalink / raw)
  To: linux-arm-kernel
This series aims at automating the NAND timings selection which is
currently supposed to be done in each NAND controller driver, thus
simplifying drivers implementation.
As suggested by Boris this version of the series introduces a nand_reset()
function which replaces the several open coded NAND_CMD_RESET commands
in the code. This makes sure we can apply the timing each time after
after reset.
Also I have brought back the conversion patch for teh sunxi driver whic
was part of Boris initial posting. It's untested due to the lack of hardware,
so please test before applying.
Sascha
Changes since v3:
- Bring back patch dropped in v3
- Use statically allocated default timing for all chips and store one
  optimized timing in struct nand_chip
Changes since v2:
- Add accessor function to get the SDR timing from struct nand_data_interface
- Change nand_reset() argument to struct nand_chip
- Drop conversion of nand_timing array to struct nand_data_interface
- Recalculate timing whenever needed instead of storing a pointer in struct
  nand_chip
- some more refactoring
Changes since v1:
- create a nand_reset() function to create a single place to reset NAND
  chips and to apply timings
- Add patch to convert sunxi driver for automated timing setup
- split into more patches
Changes since the initial posting from Boris:
- Integrate Feedback from Ezequiel Garcia
- When iterating over the chips calling onfi_set_features() for each
  bail out when any of the calls fail, not only the last one.
- When one of the onfi_set_features() calls fail then reset the chipi
  afterwards.
- Drop Sunxi example, add patch for the mxc_nand controller instead.
----------------------------------------------------------------
Boris Brezillon (1):
      mtd: nand: automate NAND timings selection
Sascha Hauer (8):
      mtd: nand: Create a NAND reset function
      mtd: nand: Introduce nand_data_interface
      mtd: nand: convert ONFI mode into data interface
      mtd: nand: Add function to convert ONFI mode to data_interface
      mtd: nand: Expose data interface for ONFI mode 0
      mtd: nand: sunxi: switch from manual to automated timing config
      mtd: nand: mxc: implement onfi get/set features
      mtd: nand: mxc: Add timing setup for v2 controllers
 drivers/mtd/nand/mxc_nand.c     | 133 ++++++++++++
 drivers/mtd/nand/nand_base.c    | 178 +++++++++++++++-
 drivers/mtd/nand/nand_timings.c | 457 ++++++++++++++++++++++------------------
 drivers/mtd/nand/sunxi_nand.c   |  76 ++-----
 include/linux/mtd/nand.h        | 187 +++++++++++-----
 5 files changed, 705 insertions(+), 326 deletions(-)
^ permalink raw reply	[flat|nested] 19+ messages in threadend of thread, other threads:[~2016-11-22 11:03 UTC | newest] Thread overview: 19+ messages (download: mbox.gz follow: Atom feed -- links below jump to the message on this page -- 2016-09-15 8:32 [PATCH v4] mtd: nand: automate NAND timings selection Sascha Hauer 2016-09-15 8:32 ` [PATCH 1/9] mtd: nand: Create a NAND reset function Sascha Hauer 2016-09-15 8:32 ` [PATCH 2/9] mtd: nand: Introduce nand_data_interface Sascha Hauer 2016-09-15 8:32 ` [PATCH 3/9] mtd: nand: convert ONFI mode into data interface Sascha Hauer 2016-09-15 8:32 ` [PATCH 4/9] mtd: nand: Add function to convert ONFI mode to data_interface Sascha Hauer 2016-09-15 8:32 ` [PATCH 5/9] mtd: nand: Expose data interface for ONFI mode 0 Sascha Hauer 2016-09-15 12:56 ` Boris Brezillon 2016-10-09 5:12 ` Brian Norris 2016-10-09 11:58 ` Boris Brezillon 2016-09-15 8:32 ` [PATCH 6/9] mtd: nand: automate NAND timings selection Sascha Hauer 2016-09-15 8:32 ` [PATCH 7/9] mtd: nand: sunxi: switch from manual to automated timing config Sascha Hauer 2016-09-15 8:32 ` [PATCH 8/9] mtd: nand: mxc: implement onfi get/set features Sascha Hauer 2016-11-21 13:56 ` Boris Brezillon 2016-11-22 10:52 ` Sascha Hauer 2016-11-22 11:03 ` Boris Brezillon 2016-09-15 8:32 ` [PATCH 9/9] mtd: nand: mxc: Add timing setup for v2 controllers Sascha Hauer 2016-09-16 12:19 ` [PATCH v4] mtd: nand: automate NAND timings selection Boris Brezillon 2016-09-19 6:43 ` Sascha Hauer -- strict thread matches above, loose matches on Subject: below -- 2016-09-09 12:05 Sascha Hauer
This is a public inbox, see mirroring instructions for how to clone and mirror all data and code used for this inbox; as well as URLs for NNTP newsgroup(s).