1695f7df8SLionel Debieve /* 29fe181c6SYann Gautier * Copyright (c) 2019-2020, STMicroelectronics - All Rights Reserved 3695f7df8SLionel Debieve * 4695f7df8SLionel Debieve * SPDX-License-Identifier: GPL-2.0+ OR BSD-3-Clause 5695f7df8SLionel Debieve */ 6695f7df8SLionel Debieve 7695f7df8SLionel Debieve #include <assert.h> 8695f7df8SLionel Debieve #include <errno.h> 9695f7df8SLionel Debieve #include <limits.h> 10695f7df8SLionel Debieve #include <stdint.h> 11695f7df8SLionel Debieve 12695f7df8SLionel Debieve #include <libfdt.h> 13695f7df8SLionel Debieve 14695f7df8SLionel Debieve #include <platform_def.h> 15695f7df8SLionel Debieve 16695f7df8SLionel Debieve #include <common/debug.h> 17695f7df8SLionel Debieve #include <drivers/delay_timer.h> 18695f7df8SLionel Debieve #include <drivers/raw_nand.h> 19695f7df8SLionel Debieve #include <drivers/st/stm32_fmc2_nand.h> 20695f7df8SLionel Debieve #include <drivers/st/stm32_gpio.h> 21695f7df8SLionel Debieve #include <drivers/st/stm32mp_reset.h> 22695f7df8SLionel Debieve #include <lib/mmio.h> 23695f7df8SLionel Debieve #include <lib/utils_def.h> 24695f7df8SLionel Debieve 2545c70e68SEtienne Carriere /* Timeout for device interface reset */ 2645c70e68SEtienne Carriere #define TIMEOUT_US_1_MS 1000U 2745c70e68SEtienne Carriere 28695f7df8SLionel Debieve /* FMC2 Compatibility */ 29*0c3e8acbSChristophe Kerello #define DT_FMC2_EBI_COMPAT "st,stm32mp1-fmc2-ebi" 30*0c3e8acbSChristophe Kerello #define DT_FMC2_NFC_COMPAT "st,stm32mp1-fmc2-nfc" 31695f7df8SLionel Debieve #define MAX_CS 2U 32*0c3e8acbSChristophe Kerello #define MAX_BANK 5U 33695f7df8SLionel Debieve 34695f7df8SLionel Debieve /* FMC2 Controller Registers */ 35695f7df8SLionel Debieve #define FMC2_BCR1 0x00U 36695f7df8SLionel Debieve #define FMC2_PCR 0x80U 37695f7df8SLionel Debieve #define FMC2_SR 0x84U 38695f7df8SLionel Debieve #define FMC2_PMEM 0x88U 39695f7df8SLionel Debieve #define FMC2_PATT 0x8CU 40695f7df8SLionel Debieve #define FMC2_HECCR 0x94U 41695f7df8SLionel Debieve #define FMC2_BCHISR 0x254U 4244966000SLionel Debieve #define FMC2_BCHICR 0x258U 43695f7df8SLionel Debieve #define FMC2_BCHDSR0 0x27CU 44695f7df8SLionel Debieve #define FMC2_BCHDSR1 0x280U 45695f7df8SLionel Debieve #define FMC2_BCHDSR2 0x284U 46695f7df8SLionel Debieve #define FMC2_BCHDSR3 0x288U 47695f7df8SLionel Debieve #define FMC2_BCHDSR4 0x28CU 48695f7df8SLionel Debieve 49695f7df8SLionel Debieve /* FMC2_BCR1 register */ 50695f7df8SLionel Debieve #define FMC2_BCR1_FMC2EN BIT(31) 51695f7df8SLionel Debieve /* FMC2_PCR register */ 52695f7df8SLionel Debieve #define FMC2_PCR_PWAITEN BIT(1) 53695f7df8SLionel Debieve #define FMC2_PCR_PBKEN BIT(2) 54695f7df8SLionel Debieve #define FMC2_PCR_PWID_MASK GENMASK_32(5, 4) 55695f7df8SLionel Debieve #define FMC2_PCR_PWID(x) (((x) << 4) & FMC2_PCR_PWID_MASK) 56695f7df8SLionel Debieve #define FMC2_PCR_PWID_8 0x0U 57695f7df8SLionel Debieve #define FMC2_PCR_PWID_16 0x1U 58695f7df8SLionel Debieve #define FMC2_PCR_ECCEN BIT(6) 59695f7df8SLionel Debieve #define FMC2_PCR_ECCALG BIT(8) 60695f7df8SLionel Debieve #define FMC2_PCR_TCLR_MASK GENMASK_32(12, 9) 61695f7df8SLionel Debieve #define FMC2_PCR_TCLR(x) (((x) << 9) & FMC2_PCR_TCLR_MASK) 62695f7df8SLionel Debieve #define FMC2_PCR_TCLR_DEFAULT 0xFU 63695f7df8SLionel Debieve #define FMC2_PCR_TAR_MASK GENMASK_32(16, 13) 64695f7df8SLionel Debieve #define FMC2_PCR_TAR(x) (((x) << 13) & FMC2_PCR_TAR_MASK) 65695f7df8SLionel Debieve #define FMC2_PCR_TAR_DEFAULT 0xFU 66695f7df8SLionel Debieve #define FMC2_PCR_ECCSS_MASK GENMASK_32(19, 17) 67695f7df8SLionel Debieve #define FMC2_PCR_ECCSS(x) (((x) << 17) & FMC2_PCR_ECCSS_MASK) 68695f7df8SLionel Debieve #define FMC2_PCR_ECCSS_512 0x1U 69695f7df8SLionel Debieve #define FMC2_PCR_ECCSS_2048 0x3U 70695f7df8SLionel Debieve #define FMC2_PCR_BCHECC BIT(24) 71695f7df8SLionel Debieve #define FMC2_PCR_WEN BIT(25) 72695f7df8SLionel Debieve /* FMC2_SR register */ 73695f7df8SLionel Debieve #define FMC2_SR_NWRF BIT(6) 74695f7df8SLionel Debieve /* FMC2_PMEM register*/ 75695f7df8SLionel Debieve #define FMC2_PMEM_MEMSET(x) (((x) & GENMASK_32(7, 0)) << 0) 76695f7df8SLionel Debieve #define FMC2_PMEM_MEMWAIT(x) (((x) & GENMASK_32(7, 0)) << 8) 77695f7df8SLionel Debieve #define FMC2_PMEM_MEMHOLD(x) (((x) & GENMASK_32(7, 0)) << 16) 78695f7df8SLionel Debieve #define FMC2_PMEM_MEMHIZ(x) (((x) & GENMASK_32(7, 0)) << 24) 79695f7df8SLionel Debieve #define FMC2_PMEM_DEFAULT 0x0A0A0A0AU 80695f7df8SLionel Debieve /* FMC2_PATT register */ 81695f7df8SLionel Debieve #define FMC2_PATT_ATTSET(x) (((x) & GENMASK_32(7, 0)) << 0) 82695f7df8SLionel Debieve #define FMC2_PATT_ATTWAIT(x) (((x) & GENMASK_32(7, 0)) << 8) 83695f7df8SLionel Debieve #define FMC2_PATT_ATTHOLD(x) (((x) & GENMASK_32(7, 0)) << 16) 84695f7df8SLionel Debieve #define FMC2_PATT_ATTHIZ(x) (((x) & GENMASK_32(7, 0)) << 24) 85695f7df8SLionel Debieve #define FMC2_PATT_DEFAULT 0x0A0A0A0AU 86695f7df8SLionel Debieve /* FMC2_BCHISR register */ 87695f7df8SLionel Debieve #define FMC2_BCHISR_DERF BIT(1) 8844966000SLionel Debieve /* FMC2_BCHICR register */ 8944966000SLionel Debieve #define FMC2_BCHICR_CLEAR_IRQ GENMASK_32(4, 0) 90695f7df8SLionel Debieve /* FMC2_BCHDSR0 register */ 91695f7df8SLionel Debieve #define FMC2_BCHDSR0_DUE BIT(0) 92695f7df8SLionel Debieve #define FMC2_BCHDSR0_DEF BIT(1) 93695f7df8SLionel Debieve #define FMC2_BCHDSR0_DEN_MASK GENMASK_32(7, 4) 94695f7df8SLionel Debieve #define FMC2_BCHDSR0_DEN_SHIFT 4U 95695f7df8SLionel Debieve /* FMC2_BCHDSR1 register */ 96695f7df8SLionel Debieve #define FMC2_BCHDSR1_EBP1_MASK GENMASK_32(12, 0) 97695f7df8SLionel Debieve #define FMC2_BCHDSR1_EBP2_MASK GENMASK_32(28, 16) 98695f7df8SLionel Debieve #define FMC2_BCHDSR1_EBP2_SHIFT 16U 99695f7df8SLionel Debieve /* FMC2_BCHDSR2 register */ 100695f7df8SLionel Debieve #define FMC2_BCHDSR2_EBP3_MASK GENMASK_32(12, 0) 101695f7df8SLionel Debieve #define FMC2_BCHDSR2_EBP4_MASK GENMASK_32(28, 16) 102695f7df8SLionel Debieve #define FMC2_BCHDSR2_EBP4_SHIFT 16U 103695f7df8SLionel Debieve /* FMC2_BCHDSR3 register */ 104695f7df8SLionel Debieve #define FMC2_BCHDSR3_EBP5_MASK GENMASK_32(12, 0) 105695f7df8SLionel Debieve #define FMC2_BCHDSR3_EBP6_MASK GENMASK_32(28, 16) 106695f7df8SLionel Debieve #define FMC2_BCHDSR3_EBP6_SHIFT 16U 107695f7df8SLionel Debieve /* FMC2_BCHDSR4 register */ 108695f7df8SLionel Debieve #define FMC2_BCHDSR4_EBP7_MASK GENMASK_32(12, 0) 109695f7df8SLionel Debieve #define FMC2_BCHDSR4_EBP8_MASK GENMASK_32(28, 16) 110695f7df8SLionel Debieve #define FMC2_BCHDSR4_EBP8_SHIFT 16U 111695f7df8SLionel Debieve 112695f7df8SLionel Debieve /* Timings */ 113695f7df8SLionel Debieve #define FMC2_THIZ 0x01U 114695f7df8SLionel Debieve #define FMC2_TIO 8000U 115695f7df8SLionel Debieve #define FMC2_TSYNC 3000U 116695f7df8SLionel Debieve #define FMC2_PCR_TIMING_MASK GENMASK_32(3, 0) 117695f7df8SLionel Debieve #define FMC2_PMEM_PATT_TIMING_MASK GENMASK_32(7, 0) 118695f7df8SLionel Debieve 119695f7df8SLionel Debieve #define FMC2_BBM_LEN 2U 120695f7df8SLionel Debieve #define FMC2_MAX_ECC_BYTES 14U 121695f7df8SLionel Debieve #define TIMEOUT_US_10_MS 10000U 122695f7df8SLionel Debieve #define FMC2_PSEC_PER_MSEC (1000UL * 1000UL * 1000UL) 123695f7df8SLionel Debieve 124695f7df8SLionel Debieve enum stm32_fmc2_ecc { 125695f7df8SLionel Debieve FMC2_ECC_HAM = 1U, 126695f7df8SLionel Debieve FMC2_ECC_BCH4 = 4U, 127695f7df8SLionel Debieve FMC2_ECC_BCH8 = 8U 128695f7df8SLionel Debieve }; 129695f7df8SLionel Debieve 130695f7df8SLionel Debieve struct stm32_fmc2_cs_reg { 131695f7df8SLionel Debieve uintptr_t data_base; 132695f7df8SLionel Debieve uintptr_t cmd_base; 133695f7df8SLionel Debieve uintptr_t addr_base; 134695f7df8SLionel Debieve }; 135695f7df8SLionel Debieve 136695f7df8SLionel Debieve struct stm32_fmc2_nand_timings { 137695f7df8SLionel Debieve uint8_t tclr; 138695f7df8SLionel Debieve uint8_t tar; 139695f7df8SLionel Debieve uint8_t thiz; 140695f7df8SLionel Debieve uint8_t twait; 141695f7df8SLionel Debieve uint8_t thold_mem; 142695f7df8SLionel Debieve uint8_t tset_mem; 143695f7df8SLionel Debieve uint8_t thold_att; 144695f7df8SLionel Debieve uint8_t tset_att; 145695f7df8SLionel Debieve }; 146695f7df8SLionel Debieve 147695f7df8SLionel Debieve struct stm32_fmc2_nfc { 148695f7df8SLionel Debieve uintptr_t reg_base; 149695f7df8SLionel Debieve struct stm32_fmc2_cs_reg cs[MAX_CS]; 150695f7df8SLionel Debieve unsigned long clock_id; 151695f7df8SLionel Debieve unsigned int reset_id; 152695f7df8SLionel Debieve uint8_t cs_sel; 153695f7df8SLionel Debieve }; 154695f7df8SLionel Debieve 155695f7df8SLionel Debieve static struct stm32_fmc2_nfc stm32_fmc2; 156695f7df8SLionel Debieve 157695f7df8SLionel Debieve static uintptr_t fmc2_base(void) 158695f7df8SLionel Debieve { 159695f7df8SLionel Debieve return stm32_fmc2.reg_base; 160695f7df8SLionel Debieve } 161695f7df8SLionel Debieve 162695f7df8SLionel Debieve static void stm32_fmc2_nand_setup_timing(void) 163695f7df8SLionel Debieve { 164695f7df8SLionel Debieve struct stm32_fmc2_nand_timings tims; 165695f7df8SLionel Debieve unsigned long hclk = stm32mp_clk_get_rate(stm32_fmc2.clock_id); 166695f7df8SLionel Debieve unsigned long hclkp = FMC2_PSEC_PER_MSEC / (hclk / 1000U); 167695f7df8SLionel Debieve unsigned long timing, tar, tclr, thiz, twait; 168695f7df8SLionel Debieve unsigned long tset_mem, tset_att, thold_mem, thold_att; 169695f7df8SLionel Debieve uint32_t pcr, pmem, patt; 170695f7df8SLionel Debieve 171695f7df8SLionel Debieve tar = MAX(hclkp, NAND_TAR_MIN); 172695f7df8SLionel Debieve timing = div_round_up(tar, hclkp) - 1U; 173695f7df8SLionel Debieve tims.tar = MIN(timing, (unsigned long)FMC2_PCR_TIMING_MASK); 174695f7df8SLionel Debieve 175695f7df8SLionel Debieve tclr = MAX(hclkp, NAND_TCLR_MIN); 176695f7df8SLionel Debieve timing = div_round_up(tclr, hclkp) - 1U; 177695f7df8SLionel Debieve tims.tclr = MIN(timing, (unsigned long)FMC2_PCR_TIMING_MASK); 178695f7df8SLionel Debieve 179695f7df8SLionel Debieve tims.thiz = FMC2_THIZ; 180695f7df8SLionel Debieve thiz = (tims.thiz + 1U) * hclkp; 181695f7df8SLionel Debieve 182695f7df8SLionel Debieve /* 183695f7df8SLionel Debieve * tWAIT > tRP 184695f7df8SLionel Debieve * tWAIT > tWP 185695f7df8SLionel Debieve * tWAIT > tREA + tIO 186695f7df8SLionel Debieve */ 187695f7df8SLionel Debieve twait = MAX(hclkp, NAND_TRP_MIN); 188695f7df8SLionel Debieve twait = MAX(twait, NAND_TWP_MIN); 189695f7df8SLionel Debieve twait = MAX(twait, NAND_TREA_MAX + FMC2_TIO); 190695f7df8SLionel Debieve timing = div_round_up(twait, hclkp); 191695f7df8SLionel Debieve tims.twait = CLAMP(timing, 1UL, 192695f7df8SLionel Debieve (unsigned long)FMC2_PMEM_PATT_TIMING_MASK); 193695f7df8SLionel Debieve 194695f7df8SLionel Debieve /* 195695f7df8SLionel Debieve * tSETUP_MEM > tCS - tWAIT 196695f7df8SLionel Debieve * tSETUP_MEM > tALS - tWAIT 197695f7df8SLionel Debieve * tSETUP_MEM > tDS - (tWAIT - tHIZ) 198695f7df8SLionel Debieve */ 199695f7df8SLionel Debieve tset_mem = hclkp; 200695f7df8SLionel Debieve if ((twait < NAND_TCS_MIN) && (tset_mem < (NAND_TCS_MIN - twait))) { 201695f7df8SLionel Debieve tset_mem = NAND_TCS_MIN - twait; 202695f7df8SLionel Debieve } 203695f7df8SLionel Debieve if ((twait < NAND_TALS_MIN) && (tset_mem < (NAND_TALS_MIN - twait))) { 204695f7df8SLionel Debieve tset_mem = NAND_TALS_MIN - twait; 205695f7df8SLionel Debieve } 206695f7df8SLionel Debieve if ((twait > thiz) && ((twait - thiz) < NAND_TDS_MIN) && 207695f7df8SLionel Debieve (tset_mem < (NAND_TDS_MIN - (twait - thiz)))) { 208695f7df8SLionel Debieve tset_mem = NAND_TDS_MIN - (twait - thiz); 209695f7df8SLionel Debieve } 210695f7df8SLionel Debieve timing = div_round_up(tset_mem, hclkp); 211695f7df8SLionel Debieve tims.tset_mem = CLAMP(timing, 1UL, 212695f7df8SLionel Debieve (unsigned long)FMC2_PMEM_PATT_TIMING_MASK); 213695f7df8SLionel Debieve 214695f7df8SLionel Debieve /* 215695f7df8SLionel Debieve * tHOLD_MEM > tCH 216695f7df8SLionel Debieve * tHOLD_MEM > tREH - tSETUP_MEM 217695f7df8SLionel Debieve * tHOLD_MEM > max(tRC, tWC) - (tSETUP_MEM + tWAIT) 218695f7df8SLionel Debieve */ 219695f7df8SLionel Debieve thold_mem = MAX(hclkp, NAND_TCH_MIN); 220695f7df8SLionel Debieve if ((tset_mem < NAND_TREH_MIN) && 221695f7df8SLionel Debieve (thold_mem < (NAND_TREH_MIN - tset_mem))) { 222695f7df8SLionel Debieve thold_mem = NAND_TREH_MIN - tset_mem; 223695f7df8SLionel Debieve } 224695f7df8SLionel Debieve if (((tset_mem + twait) < NAND_TRC_MIN) && 225695f7df8SLionel Debieve (thold_mem < (NAND_TRC_MIN - (tset_mem + twait)))) { 226695f7df8SLionel Debieve thold_mem = NAND_TRC_MIN - (tset_mem + twait); 227695f7df8SLionel Debieve } 228695f7df8SLionel Debieve if (((tset_mem + twait) < NAND_TWC_MIN) && 229695f7df8SLionel Debieve (thold_mem < (NAND_TWC_MIN - (tset_mem + twait)))) { 230695f7df8SLionel Debieve thold_mem = NAND_TWC_MIN - (tset_mem + twait); 231695f7df8SLionel Debieve } 232695f7df8SLionel Debieve timing = div_round_up(thold_mem, hclkp); 233695f7df8SLionel Debieve tims.thold_mem = CLAMP(timing, 1UL, 234695f7df8SLionel Debieve (unsigned long)FMC2_PMEM_PATT_TIMING_MASK); 235695f7df8SLionel Debieve 236695f7df8SLionel Debieve /* 237695f7df8SLionel Debieve * tSETUP_ATT > tCS - tWAIT 238695f7df8SLionel Debieve * tSETUP_ATT > tCLS - tWAIT 239695f7df8SLionel Debieve * tSETUP_ATT > tALS - tWAIT 240695f7df8SLionel Debieve * tSETUP_ATT > tRHW - tHOLD_MEM 241695f7df8SLionel Debieve * tSETUP_ATT > tDS - (tWAIT - tHIZ) 242695f7df8SLionel Debieve */ 243695f7df8SLionel Debieve tset_att = hclkp; 244695f7df8SLionel Debieve if ((twait < NAND_TCS_MIN) && (tset_att < (NAND_TCS_MIN - twait))) { 245695f7df8SLionel Debieve tset_att = NAND_TCS_MIN - twait; 246695f7df8SLionel Debieve } 247695f7df8SLionel Debieve if ((twait < NAND_TCLS_MIN) && (tset_att < (NAND_TCLS_MIN - twait))) { 248695f7df8SLionel Debieve tset_att = NAND_TCLS_MIN - twait; 249695f7df8SLionel Debieve } 250695f7df8SLionel Debieve if ((twait < NAND_TALS_MIN) && (tset_att < (NAND_TALS_MIN - twait))) { 251695f7df8SLionel Debieve tset_att = NAND_TALS_MIN - twait; 252695f7df8SLionel Debieve } 253695f7df8SLionel Debieve if ((thold_mem < NAND_TRHW_MIN) && 254695f7df8SLionel Debieve (tset_att < (NAND_TRHW_MIN - thold_mem))) { 255695f7df8SLionel Debieve tset_att = NAND_TRHW_MIN - thold_mem; 256695f7df8SLionel Debieve } 257695f7df8SLionel Debieve if ((twait > thiz) && ((twait - thiz) < NAND_TDS_MIN) && 258695f7df8SLionel Debieve (tset_att < (NAND_TDS_MIN - (twait - thiz)))) { 259695f7df8SLionel Debieve tset_att = NAND_TDS_MIN - (twait - thiz); 260695f7df8SLionel Debieve } 261695f7df8SLionel Debieve timing = div_round_up(tset_att, hclkp); 262695f7df8SLionel Debieve tims.tset_att = CLAMP(timing, 1UL, 263695f7df8SLionel Debieve (unsigned long)FMC2_PMEM_PATT_TIMING_MASK); 264695f7df8SLionel Debieve 265695f7df8SLionel Debieve /* 266695f7df8SLionel Debieve * tHOLD_ATT > tALH 267695f7df8SLionel Debieve * tHOLD_ATT > tCH 268695f7df8SLionel Debieve * tHOLD_ATT > tCLH 269695f7df8SLionel Debieve * tHOLD_ATT > tCOH 270695f7df8SLionel Debieve * tHOLD_ATT > tDH 271695f7df8SLionel Debieve * tHOLD_ATT > tWB + tIO + tSYNC - tSETUP_MEM 272695f7df8SLionel Debieve * tHOLD_ATT > tADL - tSETUP_MEM 273695f7df8SLionel Debieve * tHOLD_ATT > tWH - tSETUP_MEM 274695f7df8SLionel Debieve * tHOLD_ATT > tWHR - tSETUP_MEM 275695f7df8SLionel Debieve * tHOLD_ATT > tRC - (tSETUP_ATT + tWAIT) 276695f7df8SLionel Debieve * tHOLD_ATT > tWC - (tSETUP_ATT + tWAIT) 277695f7df8SLionel Debieve */ 278695f7df8SLionel Debieve thold_att = MAX(hclkp, NAND_TALH_MIN); 279695f7df8SLionel Debieve thold_att = MAX(thold_att, NAND_TCH_MIN); 280695f7df8SLionel Debieve thold_att = MAX(thold_att, NAND_TCLH_MIN); 281695f7df8SLionel Debieve thold_att = MAX(thold_att, NAND_TCOH_MIN); 282695f7df8SLionel Debieve thold_att = MAX(thold_att, NAND_TDH_MIN); 283695f7df8SLionel Debieve if (((NAND_TWB_MAX + FMC2_TIO + FMC2_TSYNC) > tset_mem) && 284695f7df8SLionel Debieve (thold_att < (NAND_TWB_MAX + FMC2_TIO + FMC2_TSYNC - tset_mem))) { 285695f7df8SLionel Debieve thold_att = NAND_TWB_MAX + FMC2_TIO + FMC2_TSYNC - tset_mem; 286695f7df8SLionel Debieve } 287695f7df8SLionel Debieve if ((tset_mem < NAND_TADL_MIN) && 288695f7df8SLionel Debieve (thold_att < (NAND_TADL_MIN - tset_mem))) { 289695f7df8SLionel Debieve thold_att = NAND_TADL_MIN - tset_mem; 290695f7df8SLionel Debieve } 291695f7df8SLionel Debieve if ((tset_mem < NAND_TWH_MIN) && 292695f7df8SLionel Debieve (thold_att < (NAND_TWH_MIN - tset_mem))) { 293695f7df8SLionel Debieve thold_att = NAND_TWH_MIN - tset_mem; 294695f7df8SLionel Debieve } 295695f7df8SLionel Debieve if ((tset_mem < NAND_TWHR_MIN) && 296695f7df8SLionel Debieve (thold_att < (NAND_TWHR_MIN - tset_mem))) { 297695f7df8SLionel Debieve thold_att = NAND_TWHR_MIN - tset_mem; 298695f7df8SLionel Debieve } 299695f7df8SLionel Debieve if (((tset_att + twait) < NAND_TRC_MIN) && 300695f7df8SLionel Debieve (thold_att < (NAND_TRC_MIN - (tset_att + twait)))) { 301695f7df8SLionel Debieve thold_att = NAND_TRC_MIN - (tset_att + twait); 302695f7df8SLionel Debieve } 303695f7df8SLionel Debieve if (((tset_att + twait) < NAND_TWC_MIN) && 304695f7df8SLionel Debieve (thold_att < (NAND_TWC_MIN - (tset_att + twait)))) { 305695f7df8SLionel Debieve thold_att = NAND_TWC_MIN - (tset_att + twait); 306695f7df8SLionel Debieve } 307695f7df8SLionel Debieve timing = div_round_up(thold_att, hclkp); 308695f7df8SLionel Debieve tims.thold_att = CLAMP(timing, 1UL, 309695f7df8SLionel Debieve (unsigned long)FMC2_PMEM_PATT_TIMING_MASK); 310695f7df8SLionel Debieve 311695f7df8SLionel Debieve VERBOSE("NAND timings: %u - %u - %u - %u - %u - %u - %u - %u\n", 312695f7df8SLionel Debieve tims.tclr, tims.tar, tims.thiz, tims.twait, 313695f7df8SLionel Debieve tims.thold_mem, tims.tset_mem, 314695f7df8SLionel Debieve tims.thold_att, tims.tset_att); 315695f7df8SLionel Debieve 316695f7df8SLionel Debieve /* Set tclr/tar timings */ 317695f7df8SLionel Debieve pcr = mmio_read_32(fmc2_base() + FMC2_PCR); 318695f7df8SLionel Debieve pcr &= ~FMC2_PCR_TCLR_MASK; 319695f7df8SLionel Debieve pcr |= FMC2_PCR_TCLR(tims.tclr); 320695f7df8SLionel Debieve pcr &= ~FMC2_PCR_TAR_MASK; 321695f7df8SLionel Debieve pcr |= FMC2_PCR_TAR(tims.tar); 322695f7df8SLionel Debieve 323695f7df8SLionel Debieve /* Set tset/twait/thold/thiz timings in common bank */ 324695f7df8SLionel Debieve pmem = FMC2_PMEM_MEMSET(tims.tset_mem); 325695f7df8SLionel Debieve pmem |= FMC2_PMEM_MEMWAIT(tims.twait); 326695f7df8SLionel Debieve pmem |= FMC2_PMEM_MEMHOLD(tims.thold_mem); 327695f7df8SLionel Debieve pmem |= FMC2_PMEM_MEMHIZ(tims.thiz); 328695f7df8SLionel Debieve 329695f7df8SLionel Debieve /* Set tset/twait/thold/thiz timings in attribute bank */ 330695f7df8SLionel Debieve patt = FMC2_PATT_ATTSET(tims.tset_att); 331695f7df8SLionel Debieve patt |= FMC2_PATT_ATTWAIT(tims.twait); 332695f7df8SLionel Debieve patt |= FMC2_PATT_ATTHOLD(tims.thold_att); 333695f7df8SLionel Debieve patt |= FMC2_PATT_ATTHIZ(tims.thiz); 334695f7df8SLionel Debieve 335695f7df8SLionel Debieve mmio_write_32(fmc2_base() + FMC2_PCR, pcr); 336695f7df8SLionel Debieve mmio_write_32(fmc2_base() + FMC2_PMEM, pmem); 337695f7df8SLionel Debieve mmio_write_32(fmc2_base() + FMC2_PATT, patt); 338695f7df8SLionel Debieve } 339695f7df8SLionel Debieve 340695f7df8SLionel Debieve static void stm32_fmc2_set_buswidth_16(bool set) 341695f7df8SLionel Debieve { 342695f7df8SLionel Debieve mmio_clrsetbits_32(fmc2_base() + FMC2_PCR, FMC2_PCR_PWID_MASK, 343695f7df8SLionel Debieve (set ? FMC2_PCR_PWID(FMC2_PCR_PWID_16) : 0U)); 344695f7df8SLionel Debieve } 345695f7df8SLionel Debieve 346695f7df8SLionel Debieve static void stm32_fmc2_set_ecc(bool enable) 347695f7df8SLionel Debieve { 348695f7df8SLionel Debieve mmio_clrsetbits_32(fmc2_base() + FMC2_PCR, FMC2_PCR_ECCEN, 349695f7df8SLionel Debieve (enable ? FMC2_PCR_ECCEN : 0U)); 350695f7df8SLionel Debieve } 351695f7df8SLionel Debieve 352695f7df8SLionel Debieve static int stm32_fmc2_ham_correct(uint8_t *buffer, uint8_t *eccbuffer, 353695f7df8SLionel Debieve uint8_t *ecc) 354695f7df8SLionel Debieve { 355695f7df8SLionel Debieve uint8_t xor_ecc_ones; 356695f7df8SLionel Debieve uint16_t xor_ecc_1b, xor_ecc_2b, xor_ecc_3b; 357695f7df8SLionel Debieve union { 358695f7df8SLionel Debieve uint32_t val; 359695f7df8SLionel Debieve uint8_t bytes[4]; 360695f7df8SLionel Debieve } xor_ecc; 361695f7df8SLionel Debieve 362695f7df8SLionel Debieve /* Page size--------ECC_Code Size 363695f7df8SLionel Debieve * 256---------------22 bits LSB (ECC_CODE & 0x003FFFFF) 364695f7df8SLionel Debieve * 512---------------24 bits (ECC_CODE & 0x00FFFFFF) 365695f7df8SLionel Debieve * 1024--------------26 bits (ECC_CODE & 0x03FFFFFF) 366695f7df8SLionel Debieve * 2048--------------28 bits (ECC_CODE & 0x0FFFFFFF) 367695f7df8SLionel Debieve * 4096--------------30 bits (ECC_CODE & 0x3FFFFFFF) 368695f7df8SLionel Debieve * 8192--------------32 bits (ECC_CODE & 0xFFFFFFFF) 369695f7df8SLionel Debieve */ 370695f7df8SLionel Debieve 371695f7df8SLionel Debieve /* For Page size 512, ECC_Code size 24 bits */ 372695f7df8SLionel Debieve xor_ecc_1b = ecc[0] ^ eccbuffer[0]; 373695f7df8SLionel Debieve xor_ecc_2b = ecc[1] ^ eccbuffer[1]; 374695f7df8SLionel Debieve xor_ecc_3b = ecc[2] ^ eccbuffer[2]; 375695f7df8SLionel Debieve 3769fe181c6SYann Gautier xor_ecc.val = 0U; 377695f7df8SLionel Debieve xor_ecc.bytes[2] = xor_ecc_3b; 378695f7df8SLionel Debieve xor_ecc.bytes[1] = xor_ecc_2b; 379695f7df8SLionel Debieve xor_ecc.bytes[0] = xor_ecc_1b; 380695f7df8SLionel Debieve 381695f7df8SLionel Debieve if (xor_ecc.val == 0U) { 382695f7df8SLionel Debieve return 0; /* No Error */ 383695f7df8SLionel Debieve } 384695f7df8SLionel Debieve 385695f7df8SLionel Debieve xor_ecc_ones = __builtin_popcount(xor_ecc.val); 386695f7df8SLionel Debieve if (xor_ecc_ones < 23U) { 387695f7df8SLionel Debieve if (xor_ecc_ones == 12U) { 388695f7df8SLionel Debieve uint16_t bit_address, byte_address; 389695f7df8SLionel Debieve 390695f7df8SLionel Debieve /* Correctable ERROR */ 391695f7df8SLionel Debieve bit_address = ((xor_ecc_1b >> 1) & BIT(0)) | 392695f7df8SLionel Debieve ((xor_ecc_1b >> 2) & BIT(1)) | 393695f7df8SLionel Debieve ((xor_ecc_1b >> 3) & BIT(2)); 394695f7df8SLionel Debieve 395695f7df8SLionel Debieve byte_address = ((xor_ecc_1b >> 7) & BIT(0)) | 396695f7df8SLionel Debieve ((xor_ecc_2b) & BIT(1)) | 397695f7df8SLionel Debieve ((xor_ecc_2b >> 1) & BIT(2)) | 398695f7df8SLionel Debieve ((xor_ecc_2b >> 2) & BIT(3)) | 399695f7df8SLionel Debieve ((xor_ecc_2b >> 3) & BIT(4)) | 400695f7df8SLionel Debieve ((xor_ecc_3b << 4) & BIT(5)) | 401695f7df8SLionel Debieve ((xor_ecc_3b << 3) & BIT(6)) | 402695f7df8SLionel Debieve ((xor_ecc_3b << 2) & BIT(7)) | 403695f7df8SLionel Debieve ((xor_ecc_3b << 1) & BIT(8)); 404695f7df8SLionel Debieve 405695f7df8SLionel Debieve /* Correct bit error in the data */ 406695f7df8SLionel Debieve buffer[byte_address] = 407695f7df8SLionel Debieve buffer[byte_address] ^ BIT(bit_address); 408695f7df8SLionel Debieve VERBOSE("Hamming: 1 ECC error corrected\n"); 409695f7df8SLionel Debieve 410695f7df8SLionel Debieve return 0; 411695f7df8SLionel Debieve } 412695f7df8SLionel Debieve 413695f7df8SLionel Debieve /* Non Correctable ERROR */ 414695f7df8SLionel Debieve ERROR("%s: Uncorrectable ECC Errors\n", __func__); 415695f7df8SLionel Debieve return -1; 416695f7df8SLionel Debieve } 417695f7df8SLionel Debieve 418695f7df8SLionel Debieve /* ECC ERROR */ 419695f7df8SLionel Debieve ERROR("%s: Hamming correction error\n", __func__); 420695f7df8SLionel Debieve return -1; 421695f7df8SLionel Debieve } 422695f7df8SLionel Debieve 423695f7df8SLionel Debieve 424695f7df8SLionel Debieve static int stm32_fmc2_ham_calculate(uint8_t *buffer, uint8_t *ecc) 425695f7df8SLionel Debieve { 426695f7df8SLionel Debieve uint32_t heccr; 427695f7df8SLionel Debieve uint64_t timeout = timeout_init_us(TIMEOUT_US_10_MS); 428695f7df8SLionel Debieve 429695f7df8SLionel Debieve while ((mmio_read_32(fmc2_base() + FMC2_SR) & FMC2_SR_NWRF) == 0U) { 430695f7df8SLionel Debieve if (timeout_elapsed(timeout)) { 431695f7df8SLionel Debieve return -ETIMEDOUT; 432695f7df8SLionel Debieve } 433695f7df8SLionel Debieve } 434695f7df8SLionel Debieve 435695f7df8SLionel Debieve heccr = mmio_read_32(fmc2_base() + FMC2_HECCR); 436695f7df8SLionel Debieve 437695f7df8SLionel Debieve ecc[0] = heccr; 438695f7df8SLionel Debieve ecc[1] = heccr >> 8; 439695f7df8SLionel Debieve ecc[2] = heccr >> 16; 440695f7df8SLionel Debieve 441695f7df8SLionel Debieve /* Disable ECC */ 442695f7df8SLionel Debieve stm32_fmc2_set_ecc(false); 443695f7df8SLionel Debieve 444695f7df8SLionel Debieve return 0; 445695f7df8SLionel Debieve } 446695f7df8SLionel Debieve 447695f7df8SLionel Debieve static int stm32_fmc2_bch_correct(uint8_t *buffer, unsigned int eccsize) 448695f7df8SLionel Debieve { 449695f7df8SLionel Debieve uint32_t bchdsr0, bchdsr1, bchdsr2, bchdsr3, bchdsr4; 450695f7df8SLionel Debieve uint16_t pos[8]; 451695f7df8SLionel Debieve int i, den; 452695f7df8SLionel Debieve uint64_t timeout = timeout_init_us(TIMEOUT_US_10_MS); 453695f7df8SLionel Debieve 454695f7df8SLionel Debieve while ((mmio_read_32(fmc2_base() + FMC2_BCHISR) & 455695f7df8SLionel Debieve FMC2_BCHISR_DERF) == 0U) { 456695f7df8SLionel Debieve if (timeout_elapsed(timeout)) { 457695f7df8SLionel Debieve return -ETIMEDOUT; 458695f7df8SLionel Debieve } 459695f7df8SLionel Debieve } 460695f7df8SLionel Debieve 461695f7df8SLionel Debieve bchdsr0 = mmio_read_32(fmc2_base() + FMC2_BCHDSR0); 462695f7df8SLionel Debieve bchdsr1 = mmio_read_32(fmc2_base() + FMC2_BCHDSR1); 463695f7df8SLionel Debieve bchdsr2 = mmio_read_32(fmc2_base() + FMC2_BCHDSR2); 464695f7df8SLionel Debieve bchdsr3 = mmio_read_32(fmc2_base() + FMC2_BCHDSR3); 465695f7df8SLionel Debieve bchdsr4 = mmio_read_32(fmc2_base() + FMC2_BCHDSR4); 466695f7df8SLionel Debieve 467695f7df8SLionel Debieve /* Disable ECC */ 468695f7df8SLionel Debieve stm32_fmc2_set_ecc(false); 469695f7df8SLionel Debieve 470695f7df8SLionel Debieve /* No error found */ 471695f7df8SLionel Debieve if ((bchdsr0 & FMC2_BCHDSR0_DEF) == 0U) { 472695f7df8SLionel Debieve return 0; 473695f7df8SLionel Debieve } 474695f7df8SLionel Debieve 475695f7df8SLionel Debieve /* Too many errors detected */ 476695f7df8SLionel Debieve if ((bchdsr0 & FMC2_BCHDSR0_DUE) != 0U) { 477695f7df8SLionel Debieve return -EBADMSG; 478695f7df8SLionel Debieve } 479695f7df8SLionel Debieve 480695f7df8SLionel Debieve pos[0] = bchdsr1 & FMC2_BCHDSR1_EBP1_MASK; 481695f7df8SLionel Debieve pos[1] = (bchdsr1 & FMC2_BCHDSR1_EBP2_MASK) >> FMC2_BCHDSR1_EBP2_SHIFT; 482695f7df8SLionel Debieve pos[2] = bchdsr2 & FMC2_BCHDSR2_EBP3_MASK; 483695f7df8SLionel Debieve pos[3] = (bchdsr2 & FMC2_BCHDSR2_EBP4_MASK) >> FMC2_BCHDSR2_EBP4_SHIFT; 484695f7df8SLionel Debieve pos[4] = bchdsr3 & FMC2_BCHDSR3_EBP5_MASK; 485695f7df8SLionel Debieve pos[5] = (bchdsr3 & FMC2_BCHDSR3_EBP6_MASK) >> FMC2_BCHDSR3_EBP6_SHIFT; 486695f7df8SLionel Debieve pos[6] = bchdsr4 & FMC2_BCHDSR4_EBP7_MASK; 487695f7df8SLionel Debieve pos[7] = (bchdsr4 & FMC2_BCHDSR4_EBP8_MASK) >> FMC2_BCHDSR4_EBP8_SHIFT; 488695f7df8SLionel Debieve 489695f7df8SLionel Debieve den = (bchdsr0 & FMC2_BCHDSR0_DEN_MASK) >> FMC2_BCHDSR0_DEN_SHIFT; 490695f7df8SLionel Debieve for (i = 0; i < den; i++) { 491695f7df8SLionel Debieve if (pos[i] < (eccsize * 8U)) { 492695f7df8SLionel Debieve uint8_t bitmask = BIT(pos[i] % 8U); 493695f7df8SLionel Debieve uint32_t offset = pos[i] / 8U; 494695f7df8SLionel Debieve 495695f7df8SLionel Debieve *(buffer + offset) ^= bitmask; 496695f7df8SLionel Debieve } 497695f7df8SLionel Debieve } 498695f7df8SLionel Debieve 499695f7df8SLionel Debieve return 0; 500695f7df8SLionel Debieve } 501695f7df8SLionel Debieve 502695f7df8SLionel Debieve static void stm32_fmc2_hwctl(struct nand_device *nand) 503695f7df8SLionel Debieve { 504695f7df8SLionel Debieve stm32_fmc2_set_ecc(false); 505695f7df8SLionel Debieve 506695f7df8SLionel Debieve if (nand->ecc.max_bit_corr != FMC2_ECC_HAM) { 507695f7df8SLionel Debieve mmio_clrbits_32(fmc2_base() + FMC2_PCR, FMC2_PCR_WEN); 50844966000SLionel Debieve mmio_write_32(fmc2_base() + FMC2_BCHICR, FMC2_BCHICR_CLEAR_IRQ); 509695f7df8SLionel Debieve } 510695f7df8SLionel Debieve 511695f7df8SLionel Debieve stm32_fmc2_set_ecc(true); 512695f7df8SLionel Debieve } 513695f7df8SLionel Debieve 514695f7df8SLionel Debieve static int stm32_fmc2_read_page(struct nand_device *nand, 515695f7df8SLionel Debieve unsigned int page, uintptr_t buffer) 516695f7df8SLionel Debieve { 517695f7df8SLionel Debieve unsigned int eccsize = nand->ecc.size; 518695f7df8SLionel Debieve unsigned int eccbytes = nand->ecc.bytes; 519695f7df8SLionel Debieve unsigned int eccsteps = nand->page_size / eccsize; 520695f7df8SLionel Debieve uint8_t ecc_corr[FMC2_MAX_ECC_BYTES]; 521695f7df8SLionel Debieve uint8_t ecc_cal[FMC2_MAX_ECC_BYTES] = {0U}; 522695f7df8SLionel Debieve uint8_t *p; 523695f7df8SLionel Debieve unsigned int i; 524695f7df8SLionel Debieve unsigned int s; 525695f7df8SLionel Debieve int ret; 526695f7df8SLionel Debieve 527695f7df8SLionel Debieve VERBOSE(">%s page %i buffer %lx\n", __func__, page, buffer); 528695f7df8SLionel Debieve 529695f7df8SLionel Debieve ret = nand_read_page_cmd(page, 0U, 0U, 0U); 530695f7df8SLionel Debieve if (ret != 0) { 531695f7df8SLionel Debieve return ret; 532695f7df8SLionel Debieve } 533695f7df8SLionel Debieve 534695f7df8SLionel Debieve for (s = 0U, i = nand->page_size + FMC2_BBM_LEN, p = (uint8_t *)buffer; 535695f7df8SLionel Debieve s < eccsteps; 536695f7df8SLionel Debieve s++, i += eccbytes, p += eccsize) { 537695f7df8SLionel Debieve stm32_fmc2_hwctl(nand); 538695f7df8SLionel Debieve 539695f7df8SLionel Debieve /* Read the NAND page sector (512 bytes) */ 540695f7df8SLionel Debieve ret = nand_change_read_column_cmd(s * eccsize, (uintptr_t)p, 541695f7df8SLionel Debieve eccsize); 542695f7df8SLionel Debieve if (ret != 0) { 543695f7df8SLionel Debieve return ret; 544695f7df8SLionel Debieve } 545695f7df8SLionel Debieve 546695f7df8SLionel Debieve if (nand->ecc.max_bit_corr == FMC2_ECC_HAM) { 547695f7df8SLionel Debieve ret = stm32_fmc2_ham_calculate(p, ecc_cal); 548695f7df8SLionel Debieve if (ret != 0) { 549695f7df8SLionel Debieve return ret; 550695f7df8SLionel Debieve } 551695f7df8SLionel Debieve } 552695f7df8SLionel Debieve 553695f7df8SLionel Debieve /* Read the corresponding ECC bytes */ 554695f7df8SLionel Debieve ret = nand_change_read_column_cmd(i, (uintptr_t)ecc_corr, 555695f7df8SLionel Debieve eccbytes); 556695f7df8SLionel Debieve if (ret != 0) { 557695f7df8SLionel Debieve return ret; 558695f7df8SLionel Debieve } 559695f7df8SLionel Debieve 560695f7df8SLionel Debieve /* Correct the data */ 561695f7df8SLionel Debieve if (nand->ecc.max_bit_corr == FMC2_ECC_HAM) { 562695f7df8SLionel Debieve ret = stm32_fmc2_ham_correct(p, ecc_corr, ecc_cal); 563695f7df8SLionel Debieve } else { 564695f7df8SLionel Debieve ret = stm32_fmc2_bch_correct(p, eccsize); 565695f7df8SLionel Debieve } 566695f7df8SLionel Debieve 567695f7df8SLionel Debieve if (ret != 0) { 568695f7df8SLionel Debieve return ret; 569695f7df8SLionel Debieve } 570695f7df8SLionel Debieve } 571695f7df8SLionel Debieve 572695f7df8SLionel Debieve return 0; 573695f7df8SLionel Debieve } 574695f7df8SLionel Debieve 575695f7df8SLionel Debieve static void stm32_fmc2_read_data(struct nand_device *nand, 576695f7df8SLionel Debieve uint8_t *buff, unsigned int length, 577695f7df8SLionel Debieve bool use_bus8) 578695f7df8SLionel Debieve { 579695f7df8SLionel Debieve uintptr_t data_base = stm32_fmc2.cs[stm32_fmc2.cs_sel].data_base; 580695f7df8SLionel Debieve 581695f7df8SLionel Debieve if (use_bus8 && (nand->buswidth == NAND_BUS_WIDTH_16)) { 582695f7df8SLionel Debieve stm32_fmc2_set_buswidth_16(false); 583695f7df8SLionel Debieve } 584695f7df8SLionel Debieve 585695f7df8SLionel Debieve if ((((uintptr_t)buff & BIT(0)) != 0U) && (length != 0U)) { 586695f7df8SLionel Debieve *buff = mmio_read_8(data_base); 587695f7df8SLionel Debieve buff += sizeof(uint8_t); 588695f7df8SLionel Debieve length -= sizeof(uint8_t); 589695f7df8SLionel Debieve } 590695f7df8SLionel Debieve 591695f7df8SLionel Debieve if ((((uintptr_t)buff & GENMASK_32(1, 0)) != 0U) && 592695f7df8SLionel Debieve (length >= sizeof(uint16_t))) { 593695f7df8SLionel Debieve *(uint16_t *)buff = mmio_read_16(data_base); 594695f7df8SLionel Debieve buff += sizeof(uint16_t); 595695f7df8SLionel Debieve length -= sizeof(uint16_t); 596695f7df8SLionel Debieve } 597695f7df8SLionel Debieve 598695f7df8SLionel Debieve /* 32bit aligned */ 599695f7df8SLionel Debieve while (length >= sizeof(uint32_t)) { 600695f7df8SLionel Debieve *(uint32_t *)buff = mmio_read_32(data_base); 601695f7df8SLionel Debieve buff += sizeof(uint32_t); 602695f7df8SLionel Debieve length -= sizeof(uint32_t); 603695f7df8SLionel Debieve } 604695f7df8SLionel Debieve 605695f7df8SLionel Debieve /* Read remaining bytes */ 606695f7df8SLionel Debieve if (length >= sizeof(uint16_t)) { 607695f7df8SLionel Debieve *(uint16_t *)buff = mmio_read_16(data_base); 608695f7df8SLionel Debieve buff += sizeof(uint16_t); 609695f7df8SLionel Debieve length -= sizeof(uint16_t); 610695f7df8SLionel Debieve } 611695f7df8SLionel Debieve 612695f7df8SLionel Debieve if (length != 0U) { 613695f7df8SLionel Debieve *buff = mmio_read_8(data_base); 614695f7df8SLionel Debieve } 615695f7df8SLionel Debieve 616695f7df8SLionel Debieve if (use_bus8 && (nand->buswidth == NAND_BUS_WIDTH_16)) { 617695f7df8SLionel Debieve /* Reconfigure bus width to 16-bit */ 618695f7df8SLionel Debieve stm32_fmc2_set_buswidth_16(true); 619695f7df8SLionel Debieve } 620695f7df8SLionel Debieve } 621695f7df8SLionel Debieve 622695f7df8SLionel Debieve static void stm32_fmc2_write_data(struct nand_device *nand, 623695f7df8SLionel Debieve uint8_t *buff, unsigned int length, 624695f7df8SLionel Debieve bool use_bus8) 625695f7df8SLionel Debieve { 626695f7df8SLionel Debieve uintptr_t data_base = stm32_fmc2.cs[stm32_fmc2.cs_sel].data_base; 627695f7df8SLionel Debieve 628695f7df8SLionel Debieve if (use_bus8 && (nand->buswidth == NAND_BUS_WIDTH_16)) { 629695f7df8SLionel Debieve /* Reconfigure bus width to 8-bit */ 630695f7df8SLionel Debieve stm32_fmc2_set_buswidth_16(false); 631695f7df8SLionel Debieve } 632695f7df8SLionel Debieve 633695f7df8SLionel Debieve if ((((uintptr_t)buff & BIT(0)) != 0U) && (length != 0U)) { 634695f7df8SLionel Debieve mmio_write_8(data_base, *buff); 635695f7df8SLionel Debieve buff += sizeof(uint8_t); 636695f7df8SLionel Debieve length -= sizeof(uint8_t); 637695f7df8SLionel Debieve } 638695f7df8SLionel Debieve 639695f7df8SLionel Debieve if ((((uintptr_t)buff & GENMASK_32(1, 0)) != 0U) && 640695f7df8SLionel Debieve (length >= sizeof(uint16_t))) { 641695f7df8SLionel Debieve mmio_write_16(data_base, *(uint16_t *)buff); 642695f7df8SLionel Debieve buff += sizeof(uint16_t); 643695f7df8SLionel Debieve length -= sizeof(uint16_t); 644695f7df8SLionel Debieve } 645695f7df8SLionel Debieve 646695f7df8SLionel Debieve /* 32bits aligned */ 647695f7df8SLionel Debieve while (length >= sizeof(uint32_t)) { 648695f7df8SLionel Debieve mmio_write_32(data_base, *(uint32_t *)buff); 649695f7df8SLionel Debieve buff += sizeof(uint32_t); 650695f7df8SLionel Debieve length -= sizeof(uint32_t); 651695f7df8SLionel Debieve } 652695f7df8SLionel Debieve 653695f7df8SLionel Debieve /* Read remaining bytes */ 654695f7df8SLionel Debieve if (length >= sizeof(uint16_t)) { 655695f7df8SLionel Debieve mmio_write_16(data_base, *(uint16_t *)buff); 656695f7df8SLionel Debieve buff += sizeof(uint16_t); 657695f7df8SLionel Debieve length -= sizeof(uint16_t); 658695f7df8SLionel Debieve } 659695f7df8SLionel Debieve 660695f7df8SLionel Debieve if (length != 0U) { 661695f7df8SLionel Debieve mmio_write_8(data_base, *buff); 662695f7df8SLionel Debieve } 663695f7df8SLionel Debieve 664695f7df8SLionel Debieve if (use_bus8 && (nand->buswidth == NAND_BUS_WIDTH_16)) { 665695f7df8SLionel Debieve /* Reconfigure bus width to 16-bit */ 666695f7df8SLionel Debieve stm32_fmc2_set_buswidth_16(true); 667695f7df8SLionel Debieve } 668695f7df8SLionel Debieve } 669695f7df8SLionel Debieve 670695f7df8SLionel Debieve static void stm32_fmc2_ctrl_init(void) 671695f7df8SLionel Debieve { 672695f7df8SLionel Debieve uint32_t pcr = mmio_read_32(fmc2_base() + FMC2_PCR); 673695f7df8SLionel Debieve uint32_t bcr1 = mmio_read_32(fmc2_base() + FMC2_BCR1); 674695f7df8SLionel Debieve 675695f7df8SLionel Debieve /* Enable wait feature and NAND flash memory bank */ 676695f7df8SLionel Debieve pcr |= FMC2_PCR_PWAITEN; 677695f7df8SLionel Debieve pcr |= FMC2_PCR_PBKEN; 678695f7df8SLionel Debieve 679695f7df8SLionel Debieve /* Set buswidth to 8 bits mode for identification */ 680695f7df8SLionel Debieve pcr &= ~FMC2_PCR_PWID_MASK; 681695f7df8SLionel Debieve 682695f7df8SLionel Debieve /* ECC logic is disabled */ 683695f7df8SLionel Debieve pcr &= ~FMC2_PCR_ECCEN; 684695f7df8SLionel Debieve 685695f7df8SLionel Debieve /* Default mode */ 686695f7df8SLionel Debieve pcr &= ~FMC2_PCR_ECCALG; 687695f7df8SLionel Debieve pcr &= ~FMC2_PCR_BCHECC; 688695f7df8SLionel Debieve pcr &= ~FMC2_PCR_WEN; 689695f7df8SLionel Debieve 690695f7df8SLionel Debieve /* Set default ECC sector size */ 691695f7df8SLionel Debieve pcr &= ~FMC2_PCR_ECCSS_MASK; 692695f7df8SLionel Debieve pcr |= FMC2_PCR_ECCSS(FMC2_PCR_ECCSS_2048); 693695f7df8SLionel Debieve 694695f7df8SLionel Debieve /* Set default TCLR/TAR timings */ 695695f7df8SLionel Debieve pcr &= ~FMC2_PCR_TCLR_MASK; 696695f7df8SLionel Debieve pcr |= FMC2_PCR_TCLR(FMC2_PCR_TCLR_DEFAULT); 697695f7df8SLionel Debieve pcr &= ~FMC2_PCR_TAR_MASK; 698695f7df8SLionel Debieve pcr |= FMC2_PCR_TAR(FMC2_PCR_TAR_DEFAULT); 699695f7df8SLionel Debieve 700695f7df8SLionel Debieve /* Enable FMC2 controller */ 701695f7df8SLionel Debieve bcr1 |= FMC2_BCR1_FMC2EN; 702695f7df8SLionel Debieve 703695f7df8SLionel Debieve mmio_write_32(fmc2_base() + FMC2_BCR1, bcr1); 704695f7df8SLionel Debieve mmio_write_32(fmc2_base() + FMC2_PCR, pcr); 705695f7df8SLionel Debieve mmio_write_32(fmc2_base() + FMC2_PMEM, FMC2_PMEM_DEFAULT); 706695f7df8SLionel Debieve mmio_write_32(fmc2_base() + FMC2_PATT, FMC2_PATT_DEFAULT); 707695f7df8SLionel Debieve } 708695f7df8SLionel Debieve 709695f7df8SLionel Debieve static int stm32_fmc2_exec(struct nand_req *req) 710695f7df8SLionel Debieve { 711695f7df8SLionel Debieve int ret = 0; 712695f7df8SLionel Debieve 713695f7df8SLionel Debieve switch (req->type & NAND_REQ_MASK) { 714695f7df8SLionel Debieve case NAND_REQ_CMD: 715695f7df8SLionel Debieve VERBOSE("Write CMD %x\n", (uint8_t)req->type); 716695f7df8SLionel Debieve mmio_write_8(stm32_fmc2.cs[stm32_fmc2.cs_sel].cmd_base, 717695f7df8SLionel Debieve (uint8_t)req->type); 718695f7df8SLionel Debieve break; 719695f7df8SLionel Debieve case NAND_REQ_ADDR: 720695f7df8SLionel Debieve VERBOSE("Write ADDR %x\n", *(req->addr)); 721695f7df8SLionel Debieve mmio_write_8(stm32_fmc2.cs[stm32_fmc2.cs_sel].addr_base, 722695f7df8SLionel Debieve *(req->addr)); 723695f7df8SLionel Debieve break; 724695f7df8SLionel Debieve case NAND_REQ_DATAIN: 725695f7df8SLionel Debieve VERBOSE("Read data\n"); 726695f7df8SLionel Debieve stm32_fmc2_read_data(req->nand, req->addr, req->length, 727695f7df8SLionel Debieve ((req->type & NAND_REQ_BUS_WIDTH_8) != 728695f7df8SLionel Debieve 0U)); 729695f7df8SLionel Debieve break; 730695f7df8SLionel Debieve case NAND_REQ_DATAOUT: 731695f7df8SLionel Debieve VERBOSE("Write data\n"); 732695f7df8SLionel Debieve stm32_fmc2_write_data(req->nand, req->addr, req->length, 733695f7df8SLionel Debieve ((req->type & NAND_REQ_BUS_WIDTH_8) != 734695f7df8SLionel Debieve 0U)); 735695f7df8SLionel Debieve break; 736695f7df8SLionel Debieve case NAND_REQ_WAIT: 737695f7df8SLionel Debieve VERBOSE("WAIT Ready\n"); 738695f7df8SLionel Debieve ret = nand_wait_ready(req->delay_ms); 739695f7df8SLionel Debieve break; 740695f7df8SLionel Debieve default: 741695f7df8SLionel Debieve ret = -EINVAL; 742695f7df8SLionel Debieve break; 743695f7df8SLionel Debieve }; 744695f7df8SLionel Debieve 745695f7df8SLionel Debieve return ret; 746695f7df8SLionel Debieve } 747695f7df8SLionel Debieve 748695f7df8SLionel Debieve static void stm32_fmc2_setup(struct nand_device *nand) 749695f7df8SLionel Debieve { 750695f7df8SLionel Debieve uint32_t pcr = mmio_read_32(fmc2_base() + FMC2_PCR); 751695f7df8SLionel Debieve 752695f7df8SLionel Debieve /* Set buswidth */ 753695f7df8SLionel Debieve pcr &= ~FMC2_PCR_PWID_MASK; 754695f7df8SLionel Debieve if (nand->buswidth == NAND_BUS_WIDTH_16) { 755695f7df8SLionel Debieve pcr |= FMC2_PCR_PWID(FMC2_PCR_PWID_16); 756695f7df8SLionel Debieve } 757695f7df8SLionel Debieve 758695f7df8SLionel Debieve if (nand->ecc.mode == NAND_ECC_HW) { 759695f7df8SLionel Debieve nand->mtd_read_page = stm32_fmc2_read_page; 760695f7df8SLionel Debieve 761695f7df8SLionel Debieve pcr &= ~FMC2_PCR_ECCALG; 762695f7df8SLionel Debieve pcr &= ~FMC2_PCR_BCHECC; 763695f7df8SLionel Debieve 764695f7df8SLionel Debieve pcr &= ~FMC2_PCR_ECCSS_MASK; 765695f7df8SLionel Debieve pcr |= FMC2_PCR_ECCSS(FMC2_PCR_ECCSS_512); 766695f7df8SLionel Debieve 767695f7df8SLionel Debieve switch (nand->ecc.max_bit_corr) { 768695f7df8SLionel Debieve case FMC2_ECC_HAM: 769695f7df8SLionel Debieve nand->ecc.bytes = 3; 770695f7df8SLionel Debieve break; 771695f7df8SLionel Debieve case FMC2_ECC_BCH8: 772695f7df8SLionel Debieve pcr |= FMC2_PCR_ECCALG; 773695f7df8SLionel Debieve pcr |= FMC2_PCR_BCHECC; 774695f7df8SLionel Debieve nand->ecc.bytes = 13; 775695f7df8SLionel Debieve break; 776695f7df8SLionel Debieve default: 777695f7df8SLionel Debieve /* Use FMC2 ECC BCH4 */ 778695f7df8SLionel Debieve pcr |= FMC2_PCR_ECCALG; 779695f7df8SLionel Debieve nand->ecc.bytes = 7; 780695f7df8SLionel Debieve break; 781695f7df8SLionel Debieve } 782695f7df8SLionel Debieve 783695f7df8SLionel Debieve if ((nand->buswidth & NAND_BUS_WIDTH_16) != 0) { 784695f7df8SLionel Debieve nand->ecc.bytes++; 785695f7df8SLionel Debieve } 786695f7df8SLionel Debieve } 787695f7df8SLionel Debieve 788695f7df8SLionel Debieve mmio_write_32(stm32_fmc2.reg_base + FMC2_PCR, pcr); 789695f7df8SLionel Debieve } 790695f7df8SLionel Debieve 791695f7df8SLionel Debieve static const struct nand_ctrl_ops ctrl_ops = { 792695f7df8SLionel Debieve .setup = stm32_fmc2_setup, 793695f7df8SLionel Debieve .exec = stm32_fmc2_exec 794695f7df8SLionel Debieve }; 795695f7df8SLionel Debieve 796695f7df8SLionel Debieve int stm32_fmc2_init(void) 797695f7df8SLionel Debieve { 798*0c3e8acbSChristophe Kerello int fmc_ebi_node; 799*0c3e8acbSChristophe Kerello int fmc_nfc_node; 800*0c3e8acbSChristophe Kerello int fmc_flash_node = 0; 801695f7df8SLionel Debieve int nchips = 0; 802695f7df8SLionel Debieve unsigned int i; 803695f7df8SLionel Debieve void *fdt = NULL; 804695f7df8SLionel Debieve const fdt32_t *cuint; 805695f7df8SLionel Debieve struct dt_node_info info; 806*0c3e8acbSChristophe Kerello uintptr_t bank_address[MAX_BANK] = { 0, 0, 0, 0, 0 }; 807*0c3e8acbSChristophe Kerello uint8_t bank_assigned = 0; 808*0c3e8acbSChristophe Kerello uint8_t bank; 80945c70e68SEtienne Carriere int ret; 810695f7df8SLionel Debieve 811695f7df8SLionel Debieve if (fdt_get_address(&fdt) == 0) { 812695f7df8SLionel Debieve return -FDT_ERR_NOTFOUND; 813695f7df8SLionel Debieve } 814695f7df8SLionel Debieve 815*0c3e8acbSChristophe Kerello fmc_ebi_node = dt_get_node(&info, -1, DT_FMC2_EBI_COMPAT); 816*0c3e8acbSChristophe Kerello if (fmc_ebi_node < 0) { 817*0c3e8acbSChristophe Kerello return fmc_ebi_node; 818695f7df8SLionel Debieve } 819695f7df8SLionel Debieve 820695f7df8SLionel Debieve if (info.status == DT_DISABLED) { 821695f7df8SLionel Debieve return -FDT_ERR_NOTFOUND; 822695f7df8SLionel Debieve } 823695f7df8SLionel Debieve 824695f7df8SLionel Debieve stm32_fmc2.reg_base = info.base; 825695f7df8SLionel Debieve 826695f7df8SLionel Debieve if ((info.clock < 0) || (info.reset < 0)) { 827695f7df8SLionel Debieve return -FDT_ERR_BADVALUE; 828695f7df8SLionel Debieve } 829695f7df8SLionel Debieve 830695f7df8SLionel Debieve stm32_fmc2.clock_id = (unsigned long)info.clock; 831695f7df8SLionel Debieve stm32_fmc2.reset_id = (unsigned int)info.reset; 832695f7df8SLionel Debieve 833*0c3e8acbSChristophe Kerello cuint = fdt_getprop(fdt, fmc_ebi_node, "ranges", NULL); 834695f7df8SLionel Debieve if (cuint == NULL) { 835695f7df8SLionel Debieve return -FDT_ERR_BADVALUE; 836695f7df8SLionel Debieve } 837695f7df8SLionel Debieve 838*0c3e8acbSChristophe Kerello for (i = 0U; i < MAX_BANK; i++) { 839*0c3e8acbSChristophe Kerello bank = fdt32_to_cpu(*cuint); 840*0c3e8acbSChristophe Kerello if ((bank >= MAX_BANK) || ((bank_assigned & BIT(bank)) != 0U)) { 841*0c3e8acbSChristophe Kerello return -FDT_ERR_BADVALUE; 842*0c3e8acbSChristophe Kerello } 843*0c3e8acbSChristophe Kerello bank_assigned |= BIT(bank); 844*0c3e8acbSChristophe Kerello bank_address[bank] = fdt32_to_cpu(*(cuint + 2)); 845*0c3e8acbSChristophe Kerello cuint += 4; 846695f7df8SLionel Debieve } 847695f7df8SLionel Debieve 848695f7df8SLionel Debieve /* Pinctrl initialization */ 849*0c3e8acbSChristophe Kerello if (dt_set_pinctrl_config(fmc_ebi_node) != 0) { 850695f7df8SLionel Debieve return -FDT_ERR_BADVALUE; 851695f7df8SLionel Debieve } 852695f7df8SLionel Debieve 853*0c3e8acbSChristophe Kerello /* Parse NFC controller node */ 854*0c3e8acbSChristophe Kerello fmc_nfc_node = fdt_node_offset_by_compatible(fdt, fmc_ebi_node, 855*0c3e8acbSChristophe Kerello DT_FMC2_NFC_COMPAT); 856*0c3e8acbSChristophe Kerello if (fmc_nfc_node < 0) { 857*0c3e8acbSChristophe Kerello return fmc_nfc_node; 858*0c3e8acbSChristophe Kerello } 859*0c3e8acbSChristophe Kerello 860*0c3e8acbSChristophe Kerello if (fdt_get_status(fmc_nfc_node) == DT_DISABLED) { 861*0c3e8acbSChristophe Kerello return -FDT_ERR_NOTFOUND; 862*0c3e8acbSChristophe Kerello } 863*0c3e8acbSChristophe Kerello 864*0c3e8acbSChristophe Kerello cuint = fdt_getprop(fdt, fmc_nfc_node, "reg", NULL); 865*0c3e8acbSChristophe Kerello if (cuint == NULL) { 866*0c3e8acbSChristophe Kerello return -FDT_ERR_BADVALUE; 867*0c3e8acbSChristophe Kerello } 868*0c3e8acbSChristophe Kerello 869*0c3e8acbSChristophe Kerello for (i = 0U; i < MAX_CS; i++) { 870*0c3e8acbSChristophe Kerello bank = fdt32_to_cpu(*cuint); 871*0c3e8acbSChristophe Kerello if (bank >= MAX_BANK) { 872*0c3e8acbSChristophe Kerello return -FDT_ERR_BADVALUE; 873*0c3e8acbSChristophe Kerello } 874*0c3e8acbSChristophe Kerello stm32_fmc2.cs[i].data_base = fdt32_to_cpu(*(cuint + 1)) + 875*0c3e8acbSChristophe Kerello bank_address[bank]; 876*0c3e8acbSChristophe Kerello 877*0c3e8acbSChristophe Kerello bank = fdt32_to_cpu(*(cuint + 3)); 878*0c3e8acbSChristophe Kerello if (bank >= MAX_BANK) { 879*0c3e8acbSChristophe Kerello return -FDT_ERR_BADVALUE; 880*0c3e8acbSChristophe Kerello } 881*0c3e8acbSChristophe Kerello stm32_fmc2.cs[i].cmd_base = fdt32_to_cpu(*(cuint + 4)) + 882*0c3e8acbSChristophe Kerello bank_address[bank]; 883*0c3e8acbSChristophe Kerello 884*0c3e8acbSChristophe Kerello bank = fdt32_to_cpu(*(cuint + 6)); 885*0c3e8acbSChristophe Kerello if (bank >= MAX_BANK) { 886*0c3e8acbSChristophe Kerello return -FDT_ERR_BADVALUE; 887*0c3e8acbSChristophe Kerello } 888*0c3e8acbSChristophe Kerello stm32_fmc2.cs[i].addr_base = fdt32_to_cpu(*(cuint + 7)) + 889*0c3e8acbSChristophe Kerello bank_address[bank]; 890*0c3e8acbSChristophe Kerello 891*0c3e8acbSChristophe Kerello cuint += 9; 892*0c3e8acbSChristophe Kerello } 893*0c3e8acbSChristophe Kerello 894695f7df8SLionel Debieve /* Parse flash nodes */ 895*0c3e8acbSChristophe Kerello fdt_for_each_subnode(fmc_flash_node, fdt, fmc_nfc_node) { 896695f7df8SLionel Debieve nchips++; 897695f7df8SLionel Debieve } 898695f7df8SLionel Debieve 899695f7df8SLionel Debieve if (nchips != 1) { 900695f7df8SLionel Debieve WARN("Only one SLC NAND device supported\n"); 901695f7df8SLionel Debieve return -FDT_ERR_BADVALUE; 902695f7df8SLionel Debieve } 903695f7df8SLionel Debieve 904*0c3e8acbSChristophe Kerello fdt_for_each_subnode(fmc_flash_node, fdt, fmc_nfc_node) { 905695f7df8SLionel Debieve /* Get chip select */ 906*0c3e8acbSChristophe Kerello cuint = fdt_getprop(fdt, fmc_flash_node, "reg", NULL); 907695f7df8SLionel Debieve if (cuint == NULL) { 908695f7df8SLionel Debieve WARN("Chip select not well defined\n"); 909695f7df8SLionel Debieve return -FDT_ERR_BADVALUE; 910695f7df8SLionel Debieve } 911695f7df8SLionel Debieve stm32_fmc2.cs_sel = fdt32_to_cpu(*cuint); 912695f7df8SLionel Debieve VERBOSE("NAND CS %i\n", stm32_fmc2.cs_sel); 913695f7df8SLionel Debieve } 914695f7df8SLionel Debieve 915695f7df8SLionel Debieve /* Enable Clock */ 916695f7df8SLionel Debieve stm32mp_clk_enable(stm32_fmc2.clock_id); 917695f7df8SLionel Debieve 918695f7df8SLionel Debieve /* Reset IP */ 91945c70e68SEtienne Carriere ret = stm32mp_reset_assert(stm32_fmc2.reset_id, TIMEOUT_US_1_MS); 92045c70e68SEtienne Carriere if (ret != 0) { 92145c70e68SEtienne Carriere panic(); 92245c70e68SEtienne Carriere } 92345c70e68SEtienne Carriere ret = stm32mp_reset_deassert(stm32_fmc2.reset_id, TIMEOUT_US_1_MS); 92445c70e68SEtienne Carriere if (ret != 0) { 92545c70e68SEtienne Carriere panic(); 92645c70e68SEtienne Carriere } 927695f7df8SLionel Debieve 928695f7df8SLionel Debieve /* Setup default IP registers */ 929695f7df8SLionel Debieve stm32_fmc2_ctrl_init(); 930695f7df8SLionel Debieve 931695f7df8SLionel Debieve /* Setup default timings */ 932695f7df8SLionel Debieve stm32_fmc2_nand_setup_timing(); 933695f7df8SLionel Debieve 934695f7df8SLionel Debieve /* Init NAND RAW framework */ 935695f7df8SLionel Debieve nand_raw_ctrl_init(&ctrl_ops); 936695f7df8SLionel Debieve 937695f7df8SLionel Debieve return 0; 938695f7df8SLionel Debieve } 939