238 lines
		
	
	
		
			5.1 KiB
		
	
	
	
		
			C
		
	
	
	
			
		
		
	
	
			238 lines
		
	
	
		
			5.1 KiB
		
	
	
	
		
			C
		
	
	
	
| // SPDX-License-Identifier: GPL-2.0+
 | |
| /*
 | |
|  * Copyright (C) 2014       Panasonic Corporation
 | |
|  * Copyright (C) 2014-2015  Masahiro Yamada <yamada.masahiro@socionext.com>
 | |
|  */
 | |
| 
 | |
| #include <common.h>
 | |
| #include <log.h>
 | |
| #include <asm/io.h>
 | |
| #include <asm/unaligned.h>
 | |
| #include <linux/delay.h>
 | |
| #include <linux/mtd/rawnand.h>
 | |
| #include "denali.h"
 | |
| 
 | |
| #define DENALI_MAP01		(1 << 26)	/* read/write pages in PIO */
 | |
| #define DENALI_MAP10		(2 << 26)	/* high-level control plane */
 | |
| 
 | |
| #define INDEX_CTRL_REG		0x0
 | |
| #define INDEX_DATA_REG		0x10
 | |
| 
 | |
| #define SPARE_ACCESS		0x41
 | |
| #define MAIN_ACCESS		0x42
 | |
| #define PIPELINE_ACCESS		0x2000
 | |
| 
 | |
| #define BANK(x) ((x) << 24)
 | |
| 
 | |
| static void __iomem *denali_flash_mem =
 | |
| 			(void __iomem *)CONFIG_SYS_NAND_DATA_BASE;
 | |
| static void __iomem *denali_flash_reg =
 | |
| 			(void __iomem *)CONFIG_SYS_NAND_REGS_BASE;
 | |
| 
 | |
| static const int flash_bank;
 | |
| static int page_size, oob_size, pages_per_block;
 | |
| 
 | |
| static void index_addr(uint32_t address, uint32_t data)
 | |
| {
 | |
| 	writel(address, denali_flash_mem + INDEX_CTRL_REG);
 | |
| 	writel(data, denali_flash_mem + INDEX_DATA_REG);
 | |
| }
 | |
| 
 | |
| static int wait_for_irq(uint32_t irq_mask)
 | |
| {
 | |
| 	unsigned long timeout = 1000000;
 | |
| 	uint32_t intr_status;
 | |
| 
 | |
| 	do {
 | |
| 		intr_status = readl(denali_flash_reg + INTR_STATUS(flash_bank));
 | |
| 
 | |
| 		if (intr_status & INTR__ECC_UNCOR_ERR) {
 | |
| 			debug("Uncorrected ECC detected\n");
 | |
| 			return -EBADMSG;
 | |
| 		}
 | |
| 
 | |
| 		if (intr_status & irq_mask)
 | |
| 			break;
 | |
| 
 | |
| 		udelay(1);
 | |
| 		timeout--;
 | |
| 	} while (timeout);
 | |
| 
 | |
| 	if (!timeout) {
 | |
| 		debug("Timeout with interrupt status %08x\n", intr_status);
 | |
| 		return -EIO;
 | |
| 	}
 | |
| 
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| static void read_data_from_flash_mem(uint8_t *buf, int len)
 | |
| {
 | |
| 	int i;
 | |
| 	uint32_t *buf32;
 | |
| 
 | |
| 	/* transfer the data from the flash */
 | |
| 	buf32 = (uint32_t *)buf;
 | |
| 
 | |
| 	/*
 | |
| 	 * Let's take care of unaligned access although it rarely happens.
 | |
| 	 * Avoid put_unaligned() for the normal use cases since it leads to
 | |
| 	 * a bit performance regression.
 | |
| 	 */
 | |
| 	if ((unsigned long)buf32 % 4) {
 | |
| 		for (i = 0; i < len / 4; i++)
 | |
| 			put_unaligned(readl(denali_flash_mem + INDEX_DATA_REG),
 | |
| 				      buf32++);
 | |
| 	} else {
 | |
| 		for (i = 0; i < len / 4; i++)
 | |
| 			*buf32++ = readl(denali_flash_mem + INDEX_DATA_REG);
 | |
| 	}
 | |
| 
 | |
| 	if (len % 4) {
 | |
| 		u32 tmp;
 | |
| 
 | |
| 		tmp = cpu_to_le32(readl(denali_flash_mem + INDEX_DATA_REG));
 | |
| 		buf = (uint8_t *)buf32;
 | |
| 		for (i = 0; i < len % 4; i++) {
 | |
| 			*buf++ = tmp;
 | |
| 			tmp >>= 8;
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| 
 | |
| int denali_send_pipeline_cmd(int page, int ecc_en, int access_type)
 | |
| {
 | |
| 	uint32_t addr, cmd;
 | |
| 	static uint32_t page_count = 1;
 | |
| 
 | |
| 	writel(ecc_en, denali_flash_reg + ECC_ENABLE);
 | |
| 
 | |
| 	/* clear all bits of intr_status. */
 | |
| 	writel(0xffff, denali_flash_reg + INTR_STATUS(flash_bank));
 | |
| 
 | |
| 	addr = BANK(flash_bank) | page;
 | |
| 
 | |
| 	/* setup the acccess type */
 | |
| 	cmd = DENALI_MAP10 | addr;
 | |
| 	index_addr(cmd, access_type);
 | |
| 
 | |
| 	/* setup the pipeline command */
 | |
| 	index_addr(cmd, PIPELINE_ACCESS | page_count);
 | |
| 
 | |
| 	cmd = DENALI_MAP01 | addr;
 | |
| 	writel(cmd, denali_flash_mem + INDEX_CTRL_REG);
 | |
| 
 | |
| 	return wait_for_irq(INTR__LOAD_COMP);
 | |
| }
 | |
| 
 | |
| static int nand_read_oob(void *buf, int page)
 | |
| {
 | |
| 	int ret;
 | |
| 
 | |
| 	ret = denali_send_pipeline_cmd(page, 0, SPARE_ACCESS);
 | |
| 	if (ret < 0)
 | |
| 		return ret;
 | |
| 
 | |
| 	read_data_from_flash_mem(buf, oob_size);
 | |
| 
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| static int nand_read_page(void *buf, int page)
 | |
| {
 | |
| 	int ret;
 | |
| 
 | |
| 	ret = denali_send_pipeline_cmd(page, 1, MAIN_ACCESS);
 | |
| 	if (ret < 0)
 | |
| 		return ret;
 | |
| 
 | |
| 	read_data_from_flash_mem(buf, page_size);
 | |
| 
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| static int nand_block_isbad(void *buf, int block)
 | |
| {
 | |
| 	int ret;
 | |
| 
 | |
| 	ret = nand_read_oob(buf, block * pages_per_block);
 | |
| 	if (ret < 0)
 | |
| 		return ret;
 | |
| 
 | |
| 	return *((uint8_t *)buf + CONFIG_SYS_NAND_BAD_BLOCK_POS) != 0xff;
 | |
| }
 | |
| 
 | |
| /* nand_init() - initialize data to make nand usable by SPL */
 | |
| void nand_init(void)
 | |
| {
 | |
| 	/* access to main area */
 | |
| 	writel(0, denali_flash_reg + TRANSFER_SPARE_REG);
 | |
| 
 | |
| 	/*
 | |
| 	 * These registers are expected to be already set by the hardware
 | |
| 	 * or earlier boot code.  So we read these values out.
 | |
| 	 */
 | |
| 	page_size = readl(denali_flash_reg + DEVICE_MAIN_AREA_SIZE);
 | |
| 	oob_size = readl(denali_flash_reg + DEVICE_SPARE_AREA_SIZE);
 | |
| 	pages_per_block = readl(denali_flash_reg + PAGES_PER_BLOCK);
 | |
| 
 | |
| 	/* Do as denali_hw_init() does. */
 | |
| 	writel(CONFIG_NAND_DENALI_SPARE_AREA_SKIP_BYTES,
 | |
| 	       denali_flash_reg + SPARE_AREA_SKIP_BYTES);
 | |
| 	writel(0x0F, denali_flash_reg + RB_PIN_ENABLED);
 | |
| 	writel(CHIP_EN_DONT_CARE__FLAG, denali_flash_reg + CHIP_ENABLE_DONT_CARE);
 | |
| 	writel(0xffff, denali_flash_reg + SPARE_AREA_MARKER);
 | |
| }
 | |
| 
 | |
| int nand_spl_load_image(uint32_t offs, unsigned int size, void *dst)
 | |
| {
 | |
| 	int block, page, column, readlen;
 | |
| 	int ret;
 | |
| 	int force_bad_block_check = 1;
 | |
| 
 | |
| 	page = offs / page_size;
 | |
| 	column = offs % page_size;
 | |
| 
 | |
| 	block = page / pages_per_block;
 | |
| 	page = page % pages_per_block;
 | |
| 
 | |
| 	while (size) {
 | |
| 		if (force_bad_block_check || page == 0) {
 | |
| 			ret = nand_block_isbad(dst, block);
 | |
| 			if (ret < 0)
 | |
| 				return ret;
 | |
| 
 | |
| 			if (ret) {
 | |
| 				block++;
 | |
| 				continue;
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		force_bad_block_check = 0;
 | |
| 
 | |
| 		ret = nand_read_page(dst, block * pages_per_block + page);
 | |
| 		if (ret < 0)
 | |
| 			return ret;
 | |
| 
 | |
| 		readlen = min(page_size - column, (int)size);
 | |
| 
 | |
| 		if (unlikely(column)) {
 | |
| 			/* Partial page read */
 | |
| 			memmove(dst, dst + column, readlen);
 | |
| 			column = 0;
 | |
| 		}
 | |
| 
 | |
| 		size -= readlen;
 | |
| 		dst += readlen;
 | |
| 		page++;
 | |
| 		if (page == pages_per_block) {
 | |
| 			block++;
 | |
| 			page = 0;
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| void nand_deselect(void) {}
 |