mirror of
https://github.com/AsahiLinux/u-boot
synced 2025-01-10 12:18:55 +00:00
38ef64e6ce
Since commit 34793598c8
("mtd: nand: mxs_nand_spl: Remove the page aligned
access") there are no longer any users of nand_get_mtd. However, it is
still important to know what the page size is so we can allocate a
large-enough buffer. If the image size is not page-aligned, we will go off
the end of the buffer and clobber some memory.
Introduce a new function nand_page_size which returns the page size. For
most drivers it is easy to determine the page size. However, a few need to
be modified since they only keep the page size around temporarily.
It's possible that this patch could cause a regression on some platforms if
the offset is non-aligned and there is invalid address space immediately
before the load address. spl_load_legacy_img does not (except when
compressing) respect bl_len, so only boards with SPL_LOAD_FIT (8 boards) or
SPL_LOAD_IMX_CONTAINER (none in tree) would be affected.
defconfig CONFIG_TEXT_BASE
======================= ================
am335x_evm 0x80800000
am43xx_evm 0x80800000
am43xx_evm_rtconly 0x80800000
am43xx_evm_usbhost_boot 0x80800000
am43xx_hs_evm 0x80800000
dra7xx_evm 0x80800000
gwventana_nand 0x17800000
imx8mn_bsh_smm_s2 0x40200000
All the sitara boards have DDR mapped at 0x80000000. gwventana is an i.MX6Q
which has DDR at 0x10000000. I don't have the IMX8MNRM handy, but on the
i.MX8M DDR starts at 0x40000000. Therefore all of these boards can handle a
little underflow.
Signed-off-by: Sean Anderson <seanga2@gmail.com>
242 lines
5.1 KiB
C
242 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 *)CFG_SYS_NAND_DATA_BASE;
|
|
static void __iomem *denali_flash_reg =
|
|
(void __iomem *)CFG_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;
|
|
}
|
|
|
|
unsigned int nand_page_size(void)
|
|
{
|
|
return page_size;
|
|
}
|
|
|
|
void nand_deselect(void) {}
|