1*c3e57739SLionel Debieve /* 2*c3e57739SLionel Debieve * Copyright (c) 2019, STMicroelectronics - All Rights Reserved 3*c3e57739SLionel Debieve * 4*c3e57739SLionel Debieve * SPDX-License-Identifier: BSD-3-Clause 5*c3e57739SLionel Debieve */ 6*c3e57739SLionel Debieve 7*c3e57739SLionel Debieve #include <assert.h> 8*c3e57739SLionel Debieve #include <errno.h> 9*c3e57739SLionel Debieve #include <stddef.h> 10*c3e57739SLionel Debieve 11*c3e57739SLionel Debieve #include <platform_def.h> 12*c3e57739SLionel Debieve 13*c3e57739SLionel Debieve #include <common/debug.h> 14*c3e57739SLionel Debieve #include <drivers/delay_timer.h> 15*c3e57739SLionel Debieve #include <drivers/spi_nand.h> 16*c3e57739SLionel Debieve #include <lib/utils.h> 17*c3e57739SLionel Debieve 18*c3e57739SLionel Debieve #define SPI_NAND_MAX_ID_LEN 4U 19*c3e57739SLionel Debieve #define DELAY_US_400MS 400000U 20*c3e57739SLionel Debieve #define MACRONIX_ID 0xC2U 21*c3e57739SLionel Debieve 22*c3e57739SLionel Debieve static struct spinand_device spinand_dev; 23*c3e57739SLionel Debieve 24*c3e57739SLionel Debieve #pragma weak plat_get_spi_nand_data 25*c3e57739SLionel Debieve int plat_get_spi_nand_data(struct spinand_device *device) 26*c3e57739SLionel Debieve { 27*c3e57739SLionel Debieve return 0; 28*c3e57739SLionel Debieve } 29*c3e57739SLionel Debieve 30*c3e57739SLionel Debieve static int spi_nand_reg(bool read_reg, uint8_t reg, uint8_t *val, 31*c3e57739SLionel Debieve enum spi_mem_data_dir dir) 32*c3e57739SLionel Debieve { 33*c3e57739SLionel Debieve struct spi_mem_op op; 34*c3e57739SLionel Debieve 35*c3e57739SLionel Debieve zeromem(&op, sizeof(struct spi_mem_op)); 36*c3e57739SLionel Debieve if (read_reg) { 37*c3e57739SLionel Debieve op.cmd.opcode = SPI_NAND_OP_GET_FEATURE; 38*c3e57739SLionel Debieve } else { 39*c3e57739SLionel Debieve op.cmd.opcode = SPI_NAND_OP_SET_FEATURE; 40*c3e57739SLionel Debieve } 41*c3e57739SLionel Debieve 42*c3e57739SLionel Debieve op.cmd.buswidth = SPI_MEM_BUSWIDTH_1_LINE; 43*c3e57739SLionel Debieve op.addr.val = reg; 44*c3e57739SLionel Debieve op.addr.nbytes = 1U; 45*c3e57739SLionel Debieve op.addr.buswidth = SPI_MEM_BUSWIDTH_1_LINE; 46*c3e57739SLionel Debieve op.data.buswidth = SPI_MEM_BUSWIDTH_1_LINE; 47*c3e57739SLionel Debieve op.data.dir = dir; 48*c3e57739SLionel Debieve op.data.nbytes = 1U; 49*c3e57739SLionel Debieve op.data.buf = val; 50*c3e57739SLionel Debieve 51*c3e57739SLionel Debieve return spi_mem_exec_op(&op); 52*c3e57739SLionel Debieve } 53*c3e57739SLionel Debieve 54*c3e57739SLionel Debieve static int spi_nand_read_reg(uint8_t reg, uint8_t *val) 55*c3e57739SLionel Debieve { 56*c3e57739SLionel Debieve return spi_nand_reg(true, reg, val, SPI_MEM_DATA_IN); 57*c3e57739SLionel Debieve } 58*c3e57739SLionel Debieve 59*c3e57739SLionel Debieve static int spi_nand_write_reg(uint8_t reg, uint8_t val) 60*c3e57739SLionel Debieve { 61*c3e57739SLionel Debieve return spi_nand_reg(false, reg, &val, SPI_MEM_DATA_OUT); 62*c3e57739SLionel Debieve } 63*c3e57739SLionel Debieve 64*c3e57739SLionel Debieve static int spi_nand_update_cfg(uint8_t mask, uint8_t val) 65*c3e57739SLionel Debieve { 66*c3e57739SLionel Debieve int ret; 67*c3e57739SLionel Debieve uint8_t cfg = spinand_dev.cfg_cache; 68*c3e57739SLionel Debieve 69*c3e57739SLionel Debieve cfg &= ~mask; 70*c3e57739SLionel Debieve cfg |= val; 71*c3e57739SLionel Debieve 72*c3e57739SLionel Debieve if (cfg == spinand_dev.cfg_cache) { 73*c3e57739SLionel Debieve return 0; 74*c3e57739SLionel Debieve } 75*c3e57739SLionel Debieve 76*c3e57739SLionel Debieve ret = spi_nand_write_reg(SPI_NAND_REG_CFG, cfg); 77*c3e57739SLionel Debieve if (ret == 0) { 78*c3e57739SLionel Debieve spinand_dev.cfg_cache = cfg; 79*c3e57739SLionel Debieve } 80*c3e57739SLionel Debieve 81*c3e57739SLionel Debieve return ret; 82*c3e57739SLionel Debieve } 83*c3e57739SLionel Debieve 84*c3e57739SLionel Debieve static int spi_nand_ecc_enable(bool enable) 85*c3e57739SLionel Debieve { 86*c3e57739SLionel Debieve return spi_nand_update_cfg(SPI_NAND_CFG_ECC_EN, 87*c3e57739SLionel Debieve enable ? SPI_NAND_CFG_ECC_EN : 0U); 88*c3e57739SLionel Debieve } 89*c3e57739SLionel Debieve 90*c3e57739SLionel Debieve static int spi_nand_quad_enable(uint8_t manufacturer_id) 91*c3e57739SLionel Debieve { 92*c3e57739SLionel Debieve bool enable = false; 93*c3e57739SLionel Debieve 94*c3e57739SLionel Debieve if (manufacturer_id != MACRONIX_ID) { 95*c3e57739SLionel Debieve return 0; 96*c3e57739SLionel Debieve } 97*c3e57739SLionel Debieve 98*c3e57739SLionel Debieve if (spinand_dev.spi_read_cache_op.data.buswidth == 99*c3e57739SLionel Debieve SPI_MEM_BUSWIDTH_4_LINE) { 100*c3e57739SLionel Debieve enable = true; 101*c3e57739SLionel Debieve } 102*c3e57739SLionel Debieve 103*c3e57739SLionel Debieve return spi_nand_update_cfg(SPI_NAND_CFG_QE, 104*c3e57739SLionel Debieve enable ? SPI_NAND_CFG_QE : 0U); 105*c3e57739SLionel Debieve } 106*c3e57739SLionel Debieve 107*c3e57739SLionel Debieve static int spi_nand_wait_ready(uint8_t *status) 108*c3e57739SLionel Debieve { 109*c3e57739SLionel Debieve int ret; 110*c3e57739SLionel Debieve uint64_t timeout = timeout_init_us(DELAY_US_400MS); 111*c3e57739SLionel Debieve 112*c3e57739SLionel Debieve while (!timeout_elapsed(timeout)) { 113*c3e57739SLionel Debieve ret = spi_nand_read_reg(SPI_NAND_REG_STATUS, status); 114*c3e57739SLionel Debieve if (ret != 0) { 115*c3e57739SLionel Debieve return ret; 116*c3e57739SLionel Debieve } 117*c3e57739SLionel Debieve 118*c3e57739SLionel Debieve VERBOSE("%s Status %x\n", __func__, *status); 119*c3e57739SLionel Debieve if ((*status & SPI_NAND_STATUS_BUSY) == 0U) { 120*c3e57739SLionel Debieve return 0; 121*c3e57739SLionel Debieve } 122*c3e57739SLionel Debieve } 123*c3e57739SLionel Debieve 124*c3e57739SLionel Debieve return -ETIMEDOUT; 125*c3e57739SLionel Debieve } 126*c3e57739SLionel Debieve 127*c3e57739SLionel Debieve static int spi_nand_reset(void) 128*c3e57739SLionel Debieve { 129*c3e57739SLionel Debieve struct spi_mem_op op; 130*c3e57739SLionel Debieve uint8_t status; 131*c3e57739SLionel Debieve int ret; 132*c3e57739SLionel Debieve 133*c3e57739SLionel Debieve zeromem(&op, sizeof(struct spi_mem_op)); 134*c3e57739SLionel Debieve op.cmd.opcode = SPI_NAND_OP_RESET; 135*c3e57739SLionel Debieve op.cmd.buswidth = SPI_MEM_BUSWIDTH_1_LINE; 136*c3e57739SLionel Debieve 137*c3e57739SLionel Debieve ret = spi_mem_exec_op(&op); 138*c3e57739SLionel Debieve if (ret != 0) { 139*c3e57739SLionel Debieve return ret; 140*c3e57739SLionel Debieve } 141*c3e57739SLionel Debieve 142*c3e57739SLionel Debieve return spi_nand_wait_ready(&status); 143*c3e57739SLionel Debieve } 144*c3e57739SLionel Debieve 145*c3e57739SLionel Debieve static int spi_nand_read_id(uint8_t *id) 146*c3e57739SLionel Debieve { 147*c3e57739SLionel Debieve struct spi_mem_op op; 148*c3e57739SLionel Debieve 149*c3e57739SLionel Debieve zeromem(&op, sizeof(struct spi_mem_op)); 150*c3e57739SLionel Debieve op.cmd.opcode = SPI_NAND_OP_READ_ID; 151*c3e57739SLionel Debieve op.cmd.buswidth = SPI_MEM_BUSWIDTH_1_LINE; 152*c3e57739SLionel Debieve op.data.dir = SPI_MEM_DATA_IN; 153*c3e57739SLionel Debieve op.data.nbytes = SPI_NAND_MAX_ID_LEN; 154*c3e57739SLionel Debieve op.data.buf = id; 155*c3e57739SLionel Debieve op.data.buswidth = SPI_MEM_BUSWIDTH_1_LINE; 156*c3e57739SLionel Debieve 157*c3e57739SLionel Debieve return spi_mem_exec_op(&op); 158*c3e57739SLionel Debieve } 159*c3e57739SLionel Debieve 160*c3e57739SLionel Debieve static int spi_nand_load_page(unsigned int page) 161*c3e57739SLionel Debieve { 162*c3e57739SLionel Debieve struct spi_mem_op op; 163*c3e57739SLionel Debieve uint32_t block_nb = page / spinand_dev.nand_dev->block_size; 164*c3e57739SLionel Debieve uint32_t page_nb = page - (block_nb * spinand_dev.nand_dev->page_size); 165*c3e57739SLionel Debieve uint32_t nbpages_per_block = spinand_dev.nand_dev->block_size / 166*c3e57739SLionel Debieve spinand_dev.nand_dev->page_size; 167*c3e57739SLionel Debieve uint32_t block_sh = __builtin_ctz(nbpages_per_block) + 1U; 168*c3e57739SLionel Debieve 169*c3e57739SLionel Debieve zeromem(&op, sizeof(struct spi_mem_op)); 170*c3e57739SLionel Debieve op.cmd.opcode = SPI_NAND_OP_LOAD_PAGE; 171*c3e57739SLionel Debieve op.cmd.buswidth = SPI_MEM_BUSWIDTH_1_LINE; 172*c3e57739SLionel Debieve op.addr.val = (block_nb << block_sh) | page_nb; 173*c3e57739SLionel Debieve op.addr.nbytes = 3U; 174*c3e57739SLionel Debieve op.addr.buswidth = SPI_MEM_BUSWIDTH_1_LINE; 175*c3e57739SLionel Debieve 176*c3e57739SLionel Debieve return spi_mem_exec_op(&op); 177*c3e57739SLionel Debieve } 178*c3e57739SLionel Debieve 179*c3e57739SLionel Debieve static int spi_nand_read_from_cache(unsigned int page, unsigned int offset, 180*c3e57739SLionel Debieve uint8_t *buffer, unsigned int len) 181*c3e57739SLionel Debieve { 182*c3e57739SLionel Debieve uint32_t nbpages_per_block = spinand_dev.nand_dev->block_size / 183*c3e57739SLionel Debieve spinand_dev.nand_dev->page_size; 184*c3e57739SLionel Debieve uint32_t block_nb = page / nbpages_per_block; 185*c3e57739SLionel Debieve uint32_t page_sh = __builtin_ctz(spinand_dev.nand_dev->page_size) + 1U; 186*c3e57739SLionel Debieve 187*c3e57739SLionel Debieve spinand_dev.spi_read_cache_op.addr.val = offset; 188*c3e57739SLionel Debieve 189*c3e57739SLionel Debieve if ((spinand_dev.nand_dev->nb_planes > 1U) && ((block_nb % 2U) == 1U)) { 190*c3e57739SLionel Debieve spinand_dev.spi_read_cache_op.addr.val |= 1U << page_sh; 191*c3e57739SLionel Debieve } 192*c3e57739SLionel Debieve 193*c3e57739SLionel Debieve spinand_dev.spi_read_cache_op.data.buf = buffer; 194*c3e57739SLionel Debieve spinand_dev.spi_read_cache_op.data.nbytes = len; 195*c3e57739SLionel Debieve 196*c3e57739SLionel Debieve return spi_mem_exec_op(&spinand_dev.spi_read_cache_op); 197*c3e57739SLionel Debieve } 198*c3e57739SLionel Debieve 199*c3e57739SLionel Debieve static int spi_nand_read_page(unsigned int page, unsigned int offset, 200*c3e57739SLionel Debieve uint8_t *buffer, unsigned int len, 201*c3e57739SLionel Debieve bool ecc_enabled) 202*c3e57739SLionel Debieve { 203*c3e57739SLionel Debieve uint8_t status; 204*c3e57739SLionel Debieve int ret; 205*c3e57739SLionel Debieve 206*c3e57739SLionel Debieve ret = spi_nand_ecc_enable(ecc_enabled); 207*c3e57739SLionel Debieve if (ret != 0) { 208*c3e57739SLionel Debieve return ret; 209*c3e57739SLionel Debieve } 210*c3e57739SLionel Debieve 211*c3e57739SLionel Debieve ret = spi_nand_load_page(page); 212*c3e57739SLionel Debieve if (ret != 0) { 213*c3e57739SLionel Debieve return ret; 214*c3e57739SLionel Debieve } 215*c3e57739SLionel Debieve 216*c3e57739SLionel Debieve ret = spi_nand_wait_ready(&status); 217*c3e57739SLionel Debieve if (ret != 0) { 218*c3e57739SLionel Debieve return ret; 219*c3e57739SLionel Debieve } 220*c3e57739SLionel Debieve 221*c3e57739SLionel Debieve ret = spi_nand_read_from_cache(page, offset, buffer, len); 222*c3e57739SLionel Debieve if (ret != 0) { 223*c3e57739SLionel Debieve return ret; 224*c3e57739SLionel Debieve } 225*c3e57739SLionel Debieve 226*c3e57739SLionel Debieve if (ecc_enabled && ((status & SPI_NAND_STATUS_ECC_UNCOR) != 0U)) { 227*c3e57739SLionel Debieve return -EBADMSG; 228*c3e57739SLionel Debieve } 229*c3e57739SLionel Debieve 230*c3e57739SLionel Debieve return 0; 231*c3e57739SLionel Debieve } 232*c3e57739SLionel Debieve 233*c3e57739SLionel Debieve static int spi_nand_mtd_block_is_bad(unsigned int block) 234*c3e57739SLionel Debieve { 235*c3e57739SLionel Debieve unsigned int nbpages_per_block = spinand_dev.nand_dev->block_size / 236*c3e57739SLionel Debieve spinand_dev.nand_dev->page_size; 237*c3e57739SLionel Debieve uint8_t bbm_marker[2]; 238*c3e57739SLionel Debieve int ret; 239*c3e57739SLionel Debieve 240*c3e57739SLionel Debieve ret = spi_nand_read_page(block * nbpages_per_block, 241*c3e57739SLionel Debieve spinand_dev.nand_dev->page_size, 242*c3e57739SLionel Debieve bbm_marker, sizeof(bbm_marker), false); 243*c3e57739SLionel Debieve if (ret != 0) { 244*c3e57739SLionel Debieve return ret; 245*c3e57739SLionel Debieve } 246*c3e57739SLionel Debieve 247*c3e57739SLionel Debieve if ((bbm_marker[0] != GENMASK_32(7, 0)) || 248*c3e57739SLionel Debieve (bbm_marker[1] != GENMASK_32(7, 0))) { 249*c3e57739SLionel Debieve WARN("Block %i is bad\n", block); 250*c3e57739SLionel Debieve return 1; 251*c3e57739SLionel Debieve } 252*c3e57739SLionel Debieve 253*c3e57739SLionel Debieve return 0; 254*c3e57739SLionel Debieve } 255*c3e57739SLionel Debieve 256*c3e57739SLionel Debieve static int spi_nand_mtd_read_page(struct nand_device *nand, unsigned int page, 257*c3e57739SLionel Debieve uintptr_t buffer) 258*c3e57739SLionel Debieve { 259*c3e57739SLionel Debieve return spi_nand_read_page(page, 0, (uint8_t *)buffer, 260*c3e57739SLionel Debieve spinand_dev.nand_dev->page_size, true); 261*c3e57739SLionel Debieve } 262*c3e57739SLionel Debieve 263*c3e57739SLionel Debieve int spi_nand_init(unsigned long long *size, unsigned int *erase_size) 264*c3e57739SLionel Debieve { 265*c3e57739SLionel Debieve uint8_t id[SPI_NAND_MAX_ID_LEN]; 266*c3e57739SLionel Debieve int ret; 267*c3e57739SLionel Debieve 268*c3e57739SLionel Debieve spinand_dev.nand_dev = get_nand_device(); 269*c3e57739SLionel Debieve if (spinand_dev.nand_dev == NULL) { 270*c3e57739SLionel Debieve return -EINVAL; 271*c3e57739SLionel Debieve } 272*c3e57739SLionel Debieve 273*c3e57739SLionel Debieve spinand_dev.nand_dev->mtd_block_is_bad = spi_nand_mtd_block_is_bad; 274*c3e57739SLionel Debieve spinand_dev.nand_dev->mtd_read_page = spi_nand_mtd_read_page; 275*c3e57739SLionel Debieve spinand_dev.nand_dev->nb_planes = 1; 276*c3e57739SLionel Debieve 277*c3e57739SLionel Debieve spinand_dev.spi_read_cache_op.cmd.opcode = SPI_NAND_OP_READ_FROM_CACHE; 278*c3e57739SLionel Debieve spinand_dev.spi_read_cache_op.cmd.buswidth = SPI_MEM_BUSWIDTH_1_LINE; 279*c3e57739SLionel Debieve spinand_dev.spi_read_cache_op.addr.nbytes = 2U; 280*c3e57739SLionel Debieve spinand_dev.spi_read_cache_op.addr.buswidth = SPI_MEM_BUSWIDTH_1_LINE; 281*c3e57739SLionel Debieve spinand_dev.spi_read_cache_op.dummy.nbytes = 1U; 282*c3e57739SLionel Debieve spinand_dev.spi_read_cache_op.dummy.buswidth = SPI_MEM_BUSWIDTH_1_LINE; 283*c3e57739SLionel Debieve spinand_dev.spi_read_cache_op.data.buswidth = SPI_MEM_BUSWIDTH_1_LINE; 284*c3e57739SLionel Debieve 285*c3e57739SLionel Debieve if (plat_get_spi_nand_data(&spinand_dev) != 0) { 286*c3e57739SLionel Debieve return -EINVAL; 287*c3e57739SLionel Debieve } 288*c3e57739SLionel Debieve 289*c3e57739SLionel Debieve ret = spi_nand_reset(); 290*c3e57739SLionel Debieve if (ret != 0) { 291*c3e57739SLionel Debieve return ret; 292*c3e57739SLionel Debieve } 293*c3e57739SLionel Debieve 294*c3e57739SLionel Debieve ret = spi_nand_read_id(id); 295*c3e57739SLionel Debieve if (ret != 0) { 296*c3e57739SLionel Debieve return ret; 297*c3e57739SLionel Debieve } 298*c3e57739SLionel Debieve 299*c3e57739SLionel Debieve ret = spi_nand_read_reg(SPI_NAND_REG_CFG, &spinand_dev.cfg_cache); 300*c3e57739SLionel Debieve if (ret != 0) { 301*c3e57739SLionel Debieve return ret; 302*c3e57739SLionel Debieve } 303*c3e57739SLionel Debieve 304*c3e57739SLionel Debieve ret = spi_nand_quad_enable(id[0]); 305*c3e57739SLionel Debieve if (ret != 0) { 306*c3e57739SLionel Debieve return ret; 307*c3e57739SLionel Debieve } 308*c3e57739SLionel Debieve 309*c3e57739SLionel Debieve VERBOSE("SPI_NAND Detected ID 0x%x 0x%x\n", id[0], id[1]); 310*c3e57739SLionel Debieve 311*c3e57739SLionel Debieve VERBOSE("Page size %i, Block size %i, size %lli\n", 312*c3e57739SLionel Debieve spinand_dev.nand_dev->page_size, 313*c3e57739SLionel Debieve spinand_dev.nand_dev->block_size, 314*c3e57739SLionel Debieve spinand_dev.nand_dev->size); 315*c3e57739SLionel Debieve 316*c3e57739SLionel Debieve *size = spinand_dev.nand_dev->size; 317*c3e57739SLionel Debieve *erase_size = spinand_dev.nand_dev->block_size; 318*c3e57739SLionel Debieve 319*c3e57739SLionel Debieve return 0; 320*c3e57739SLionel Debieve } 321