mirror of
https://github.com/AsahiLinux/u-boot
synced 2025-01-09 19:58:55 +00:00
8502b5bf20
Add a SPL test for the NAND load method. We use some different functions to do the writing from the main test since things like nand_write_skip_bad aren't available in SPL. We disable BBT scanning, since scan_bbt is only populated when not in SPL. We use nand_spl_loaders.c as it seems to be common to at least a few boards already. However, we do not use nand_spl_simple.c because it would require us to implement cmd_ctrl. The various nand load functions are adapted from omap_gpmc. However, they have been modified for simplicity/correctness. Signed-off-by: Sean Anderson <seanga2@gmail.com>
707 lines
17 KiB
C
707 lines
17 KiB
C
// SPDX-License-Identifier: GPL-2.0+
|
|
/*
|
|
* Copyright (C) Sean Anderson <seanga2@gmail.com>
|
|
*/
|
|
|
|
#define LOG_CATEGORY UCLASS_MTD
|
|
#include <errno.h>
|
|
#include <hexdump.h>
|
|
#include <log.h>
|
|
#include <nand.h>
|
|
#include <os.h>
|
|
#include <rand.h>
|
|
#include <spl.h>
|
|
#include <system-constants.h>
|
|
#include <dm/device_compat.h>
|
|
#include <dm/read.h>
|
|
#include <dm/uclass.h>
|
|
#include <asm/bitops.h>
|
|
#include <linux/bitmap.h>
|
|
#include <linux/mtd/rawnand.h>
|
|
#include <linux/sizes.h>
|
|
|
|
enum sand_nand_state {
|
|
STATE_READY,
|
|
STATE_IDLE,
|
|
STATE_READ,
|
|
STATE_READ_ID,
|
|
STATE_READ_ONFI,
|
|
STATE_PARAM_ONFI,
|
|
STATE_STATUS,
|
|
STATE_PROG,
|
|
STATE_ERASE,
|
|
};
|
|
|
|
static const char *const state_name[] = {
|
|
[STATE_READY] = "READY",
|
|
[STATE_IDLE] = "IDLE",
|
|
[STATE_READ] = "READ",
|
|
[STATE_READ_ID] = "READ_ID",
|
|
[STATE_READ_ONFI] = "READ_ONFI",
|
|
[STATE_PARAM_ONFI] = "PARAM_ONFI",
|
|
[STATE_STATUS] = "STATUS",
|
|
[STATE_PROG] = "PROG",
|
|
[STATE_ERASE] = "ERASE",
|
|
};
|
|
|
|
/**
|
|
* struct sand_nand_chip - Per-device private data
|
|
* @nand: The nand chip
|
|
* @node: The next device in this controller
|
|
* @programmed: Bitmap of whether sectors are programmed
|
|
* @id: ID to report for NAND_CMD_READID
|
|
* @id_len: Length of @id
|
|
* @onfi: Three copies of ONFI parameter page
|
|
* @status: Status to report for NAND_CMD_STATUS
|
|
* @chunksize: Size of one "chunk" (page + oob) in bytes
|
|
* @pageize: Size of one page in bytes
|
|
* @pages: Total number of pages
|
|
* @pages_per_erase: Number of pages per eraseblock
|
|
* @err_count: Number of errors to inject per @err_step_bits of data
|
|
* @err_step_bits: Number of data bits per error "step"
|
|
* @err_steps: Number of err steps in a page
|
|
* @cs: Chip select for this device
|
|
* @state: Current state of the device
|
|
* @column: Column of the most-recent command
|
|
* @page_addr: Page address of the most-recent command
|
|
* @fd: File descriptor for the backing data
|
|
* @fd_page_addr: Page address that @fd is seek'd to
|
|
* @selected: Whether this device is selected
|
|
* @tmp: "Cache" buffer used to store transferred data before committing it
|
|
* @tmp_dirty: Whether @tmp is dirty (modified) or clean (all ones)
|
|
*
|
|
* Data is stored with the OOB area in-line. For example, with 512-byte pages
|
|
* and and 16-byte OOB areas, the first page would start at offset 0, the second
|
|
* at offset 528, the third at offset 1056, and so on
|
|
*/
|
|
struct sand_nand_chip {
|
|
struct nand_chip nand;
|
|
struct list_head node;
|
|
long *programmed;
|
|
const u8 *id;
|
|
u32 chunksize, pagesize, pages, pages_per_erase;
|
|
u32 err_count, err_step_bits, err_steps, ecc_bits;
|
|
unsigned int cs;
|
|
enum sand_nand_state state;
|
|
int column, page_addr, fd, fd_page_addr;
|
|
bool selected, tmp_dirty;
|
|
u8 status;
|
|
u8 id_len;
|
|
u8 tmp[NAND_MAX_PAGESIZE + NAND_MAX_OOBSIZE];
|
|
u8 onfi[sizeof(struct nand_onfi_params) * 3];
|
|
};
|
|
|
|
#define SAND_DEBUG(chip, fmt, ...) \
|
|
dev_dbg((chip)->nand.mtd.dev, "%u (%s): " fmt, (chip)->cs, \
|
|
state_name[(chip)->state], ##__VA_ARGS__)
|
|
|
|
static inline void to_state(struct sand_nand_chip *chip,
|
|
enum sand_nand_state new_state)
|
|
{
|
|
if (new_state != chip->state)
|
|
SAND_DEBUG(chip, "to state %s\n", state_name[new_state]);
|
|
chip->state = new_state;
|
|
}
|
|
|
|
static inline struct sand_nand_chip *to_sand_nand(struct nand_chip *nand)
|
|
{
|
|
return container_of(nand, struct sand_nand_chip, nand);
|
|
}
|
|
|
|
struct sand_nand_priv {
|
|
struct list_head chips;
|
|
};
|
|
|
|
static int sand_nand_dev_ready(struct mtd_info *mtd)
|
|
{
|
|
return 1;
|
|
}
|
|
|
|
static int sand_nand_wait(struct mtd_info *mtd, struct nand_chip *chip)
|
|
{
|
|
u8 status;
|
|
|
|
return nand_status_op(chip, &status) ?: status;
|
|
}
|
|
|
|
static int sand_nand_seek(struct sand_nand_chip *chip)
|
|
{
|
|
if (chip->fd_page_addr == chip->page_addr)
|
|
return 0;
|
|
|
|
if (os_lseek(chip->fd, (off_t)chip->page_addr * chip->chunksize,
|
|
OS_SEEK_SET) < 0) {
|
|
SAND_DEBUG(chip, "could not seek: %d\n", errno);
|
|
return -EIO;
|
|
}
|
|
|
|
chip->fd_page_addr = chip->page_addr;
|
|
return 0;
|
|
}
|
|
|
|
static void sand_nand_inject_error(struct sand_nand_chip *chip,
|
|
unsigned int step, unsigned int pos)
|
|
{
|
|
int byte, index;
|
|
|
|
if (pos < chip->err_step_bits) {
|
|
__change_bit(step * chip->err_step_bits + pos, chip->tmp);
|
|
return;
|
|
}
|
|
|
|
/*
|
|
* Only ECC bytes are covered in the OOB area, so
|
|
* pretend that those are the only bytes which can have
|
|
* errors.
|
|
*/
|
|
byte = (pos - chip->err_step_bits + step * chip->ecc_bits) / 8;
|
|
index = chip->nand.ecc.layout->eccpos[byte];
|
|
/* Avoid endianness issues by working with bytes */
|
|
chip->tmp[chip->pagesize + index] ^= BIT(pos & 0x7);
|
|
}
|
|
|
|
static int sand_nand_read(struct sand_nand_chip *chip)
|
|
{
|
|
unsigned int i, stop = 0;
|
|
|
|
if (chip->column == chip->pagesize)
|
|
stop = chip->err_step_bits;
|
|
|
|
if (test_bit(chip->page_addr, chip->programmed)) {
|
|
if (sand_nand_seek(chip))
|
|
return -EIO;
|
|
|
|
if (os_read(chip->fd, chip->tmp, chip->chunksize) !=
|
|
chip->chunksize) {
|
|
SAND_DEBUG(chip, "could not read: %d\n", errno);
|
|
return -EIO;
|
|
}
|
|
chip->fd_page_addr++;
|
|
} else if (chip->tmp_dirty) {
|
|
memset(chip->tmp + chip->column, 0xff,
|
|
chip->chunksize - chip->column);
|
|
}
|
|
|
|
/*
|
|
* Inject some errors; this is Method A from "An Efficient Algorithm for
|
|
* Sequential Random Sampling" (Vitter 87). This is still slow when
|
|
* generating a lot (dozens) of ECC errors.
|
|
*
|
|
* To avoid generating too many errors in any one ECC step, we separate
|
|
* our error generation by ECC step.
|
|
*/
|
|
chip->tmp_dirty = true;
|
|
for (i = 0; i < chip->err_steps; i++) {
|
|
u32 bit_errors = chip->err_count;
|
|
unsigned int j = chip->err_step_bits + chip->ecc_bits;
|
|
|
|
while (bit_errors) {
|
|
unsigned int u = rand();
|
|
float quot = 1ULL << 32;
|
|
|
|
do {
|
|
quot *= j - bit_errors;
|
|
quot /= j;
|
|
j--;
|
|
|
|
if (j < stop)
|
|
goto next;
|
|
} while (u < quot);
|
|
|
|
sand_nand_inject_error(chip, i, j);
|
|
bit_errors--;
|
|
}
|
|
next:
|
|
;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static void sand_nand_command(struct mtd_info *mtd, unsigned int command,
|
|
int column, int page_addr)
|
|
{
|
|
struct nand_chip *nand = mtd_to_nand(mtd);
|
|
struct sand_nand_chip *chip = to_sand_nand(nand);
|
|
enum sand_nand_state new_state = chip->state;
|
|
|
|
SAND_DEBUG(chip, "command=%02x column=%d page_addr=%d\n", command,
|
|
column, page_addr);
|
|
|
|
if (!chip->selected)
|
|
return;
|
|
|
|
switch (chip->state) {
|
|
case STATE_READY:
|
|
if (command == NAND_CMD_RESET)
|
|
goto reset;
|
|
break;
|
|
case STATE_PROG:
|
|
new_state = STATE_IDLE;
|
|
if (command != NAND_CMD_PAGEPROG ||
|
|
test_and_set_bit(chip->page_addr, chip->programmed)) {
|
|
chip->status |= NAND_STATUS_FAIL;
|
|
break;
|
|
}
|
|
|
|
if (sand_nand_seek(chip)) {
|
|
chip->status |= NAND_STATUS_FAIL;
|
|
break;
|
|
}
|
|
|
|
if (os_write(chip->fd, chip->tmp, chip->chunksize) !=
|
|
chip->chunksize) {
|
|
SAND_DEBUG(chip, "could not write: %d\n", errno);
|
|
chip->status |= NAND_STATUS_FAIL;
|
|
break;
|
|
}
|
|
|
|
chip->fd_page_addr++;
|
|
break;
|
|
case STATE_ERASE:
|
|
new_state = STATE_IDLE;
|
|
if (command != NAND_CMD_ERASE2) {
|
|
chip->status |= NAND_STATUS_FAIL;
|
|
break;
|
|
}
|
|
|
|
if (chip->page_addr < 0 ||
|
|
chip->page_addr >= chip->pages ||
|
|
chip->page_addr % chip->pages_per_erase)
|
|
chip->status |= NAND_STATUS_FAIL;
|
|
else
|
|
bitmap_clear(chip->programmed, chip->page_addr,
|
|
chip->pages_per_erase);
|
|
break;
|
|
default:
|
|
chip->column = column;
|
|
chip->page_addr = page_addr;
|
|
switch (command) {
|
|
case NAND_CMD_READOOB:
|
|
if (column >= 0)
|
|
chip->column += chip->pagesize;
|
|
fallthrough;
|
|
case NAND_CMD_READ0:
|
|
new_state = STATE_IDLE;
|
|
if (page_addr < 0 || page_addr >= chip->pages)
|
|
break;
|
|
|
|
if (chip->column < 0 || chip->column >= chip->chunksize)
|
|
break;
|
|
|
|
if (sand_nand_read(chip))
|
|
break;
|
|
|
|
chip->page_addr = page_addr;
|
|
new_state = STATE_READ;
|
|
break;
|
|
case NAND_CMD_ERASE1:
|
|
new_state = STATE_ERASE;
|
|
chip->status = ~NAND_STATUS_FAIL;
|
|
break;
|
|
case NAND_CMD_STATUS:
|
|
new_state = STATE_STATUS;
|
|
chip->column = 0;
|
|
break;
|
|
case NAND_CMD_SEQIN:
|
|
new_state = STATE_PROG;
|
|
chip->status = ~NAND_STATUS_FAIL;
|
|
if (page_addr < 0 || page_addr >= chip->pages ||
|
|
chip->column < 0 ||
|
|
chip->column >= chip->chunksize) {
|
|
chip->status |= NAND_STATUS_FAIL;
|
|
} else if (chip->tmp_dirty) {
|
|
memset(chip->tmp, 0xff, chip->chunksize);
|
|
chip->tmp_dirty = false;
|
|
}
|
|
break;
|
|
case NAND_CMD_READID:
|
|
if (chip->onfi[0] && column == 0x20)
|
|
new_state = STATE_READ_ONFI;
|
|
else
|
|
new_state = STATE_READ_ID;
|
|
chip->column = 0;
|
|
break;
|
|
case NAND_CMD_PARAM:
|
|
if (chip->onfi[0] && !column)
|
|
new_state = STATE_PARAM_ONFI;
|
|
else
|
|
new_state = STATE_IDLE;
|
|
break;
|
|
case NAND_CMD_RESET:
|
|
reset:
|
|
new_state = STATE_IDLE;
|
|
chip->column = -1;
|
|
chip->page_addr = -1;
|
|
chip->status = ~NAND_STATUS_FAIL;
|
|
break;
|
|
default:
|
|
new_state = STATE_IDLE;
|
|
SAND_DEBUG(chip, "Unsupported command %02x\n", command);
|
|
}
|
|
}
|
|
|
|
to_state(chip, new_state);
|
|
}
|
|
|
|
static void sand_nand_select_chip(struct mtd_info *mtd, int n)
|
|
{
|
|
struct nand_chip *nand = mtd_to_nand(mtd);
|
|
struct sand_nand_chip *chip = to_sand_nand(nand);
|
|
|
|
chip->selected = !n;
|
|
}
|
|
|
|
static void sand_nand_read_buf(struct mtd_info *mtd, u8 *buf, int len)
|
|
{
|
|
struct nand_chip *nand = mtd_to_nand(mtd);
|
|
struct sand_nand_chip *chip = to_sand_nand(nand);
|
|
unsigned int to_copy;
|
|
int src_len = 0;
|
|
const u8 *src = NULL;
|
|
|
|
if (!chip->selected)
|
|
goto copy;
|
|
|
|
switch (chip->state) {
|
|
case STATE_READ:
|
|
src = chip->tmp;
|
|
src_len = chip->chunksize;
|
|
break;
|
|
case STATE_READ_ID:
|
|
src = chip->id;
|
|
src_len = chip->id_len;
|
|
break;
|
|
case STATE_READ_ONFI:
|
|
src = "ONFI";
|
|
src_len = 4;
|
|
break;
|
|
case STATE_PARAM_ONFI:
|
|
src = chip->onfi;
|
|
src_len = sizeof(chip->onfi);
|
|
break;
|
|
case STATE_STATUS:
|
|
src = &chip->status;
|
|
src_len = 1;
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
|
|
copy:
|
|
if (chip->column >= 0)
|
|
to_copy = max(min(len, src_len - chip->column), 0);
|
|
else
|
|
to_copy = 0;
|
|
memcpy(buf, src + chip->column, to_copy);
|
|
memset(buf + to_copy, 0xff, len - to_copy);
|
|
chip->column += to_copy;
|
|
|
|
if (len == 1) {
|
|
SAND_DEBUG(chip, "read [ %02x ]\n", buf[0]);
|
|
} else if (src_len) {
|
|
SAND_DEBUG(chip, "read %d bytes\n", len);
|
|
#ifdef VERBOSE_DEBUG
|
|
print_hex_dump_bytes("", DUMP_PREFIX_OFFSET, buf, len);
|
|
#endif
|
|
}
|
|
|
|
if (src_len && chip->column == src_len)
|
|
to_state(chip, STATE_IDLE);
|
|
}
|
|
|
|
static u8 sand_nand_read_byte(struct mtd_info *mtd)
|
|
{
|
|
u8 ret;
|
|
|
|
sand_nand_read_buf(mtd, &ret, 1);
|
|
return ret;
|
|
}
|
|
|
|
static u16 sand_nand_read_word(struct mtd_info *mtd)
|
|
{
|
|
struct nand_chip *nand = mtd_to_nand(mtd);
|
|
struct sand_nand_chip *chip = to_sand_nand(nand);
|
|
|
|
SAND_DEBUG(chip, "16-bit access unsupported\n");
|
|
return sand_nand_read_byte(mtd) | 0xff00;
|
|
}
|
|
|
|
static void sand_nand_write_buf(struct mtd_info *mtd, const u8 *buf, int len)
|
|
{
|
|
struct nand_chip *nand = mtd_to_nand(mtd);
|
|
struct sand_nand_chip *chip = to_sand_nand(nand);
|
|
|
|
SAND_DEBUG(chip, "write %d bytes\n", len);
|
|
#ifdef VERBOSE_DEBUG
|
|
print_hex_dump_bytes("", DUMP_PREFIX_OFFSET, buf, len);
|
|
#endif
|
|
|
|
if (chip->state != STATE_PROG || chip->status & NAND_STATUS_FAIL)
|
|
return;
|
|
|
|
chip->tmp_dirty = true;
|
|
len = min((unsigned int)len, chip->chunksize - chip->column);
|
|
memcpy(chip->tmp + chip->column, buf, len);
|
|
chip->column += len;
|
|
}
|
|
|
|
static struct nand_chip *nand_chip;
|
|
|
|
int sand_nand_remove(struct udevice *dev)
|
|
{
|
|
struct sand_nand_priv *priv = dev_get_priv(dev);
|
|
struct sand_nand_chip *chip;
|
|
|
|
list_for_each_entry(chip, &priv->chips, node) {
|
|
struct nand_chip *nand = &chip->nand;
|
|
|
|
if (nand_chip == nand)
|
|
nand_chip = NULL;
|
|
|
|
nand_unregister(nand_to_mtd(nand));
|
|
free(chip->programmed);
|
|
os_close(chip->fd);
|
|
free(chip);
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int sand_nand_probe(struct udevice *dev)
|
|
{
|
|
struct sand_nand_priv *priv = dev_get_priv(dev);
|
|
struct sand_nand_chip *chip;
|
|
int ret, devnum = 0;
|
|
ofnode np;
|
|
|
|
INIT_LIST_HEAD(&priv->chips);
|
|
|
|
dev_for_each_subnode(np, dev) {
|
|
struct nand_chip *nand;
|
|
struct mtd_info *mtd;
|
|
u32 erasesize, oobsize, pagesize, pages;
|
|
u32 err_count, err_step_size;
|
|
off_t expected_size;
|
|
char filename[30];
|
|
fdt_addr_t cs;
|
|
const u8 *id, *onfi;
|
|
int id_len, onfi_len;
|
|
|
|
cs = ofnode_get_addr_size_index_notrans(np, 0, NULL);
|
|
if (cs == FDT_ADDR_T_NONE) {
|
|
dev_dbg(dev, "Invalid cs for chip %s\n",
|
|
ofnode_get_name(np));
|
|
ret = -ENOENT;
|
|
goto err;
|
|
}
|
|
|
|
id = ofnode_read_prop(np, "sandbox,id", &id_len);
|
|
if (!id) {
|
|
dev_dbg(dev, "No sandbox,id property for chip %s\n",
|
|
ofnode_get_name(np));
|
|
ret = -EINVAL;
|
|
goto err;
|
|
}
|
|
|
|
onfi = ofnode_read_prop(np, "sandbox,onfi", &onfi_len);
|
|
if (onfi && onfi_len != sizeof(struct nand_onfi_params)) {
|
|
dev_dbg(dev, "Invalid length %d for onfi params\n",
|
|
onfi_len);
|
|
ret = -EINVAL;
|
|
goto err;
|
|
}
|
|
|
|
ret = ofnode_read_u32(np, "sandbox,erasesize", &erasesize);
|
|
if (ret) {
|
|
dev_dbg(dev, "No sandbox,erasesize property for chip %s",
|
|
ofnode_get_name(np));
|
|
goto err;
|
|
}
|
|
|
|
ret = ofnode_read_u32(np, "sandbox,oobsize", &oobsize);
|
|
if (ret) {
|
|
dev_dbg(dev, "No sandbox,oobsize property for chip %s",
|
|
ofnode_get_name(np));
|
|
goto err;
|
|
}
|
|
|
|
ret = ofnode_read_u32(np, "sandbox,pagesize", &pagesize);
|
|
if (ret) {
|
|
dev_dbg(dev, "No sandbox,pagesize property for chip %s",
|
|
ofnode_get_name(np));
|
|
goto err;
|
|
}
|
|
|
|
ret = ofnode_read_u32(np, "sandbox,pages", &pages);
|
|
if (ret) {
|
|
dev_dbg(dev, "No sandbox,pages property for chip %s",
|
|
ofnode_get_name(np));
|
|
goto err;
|
|
}
|
|
|
|
ret = ofnode_read_u32(np, "sandbox,err-count", &err_count);
|
|
if (ret) {
|
|
dev_dbg(dev,
|
|
"No sandbox,err-count property for chip %s",
|
|
ofnode_get_name(np));
|
|
goto err;
|
|
}
|
|
|
|
ret = ofnode_read_u32(np, "sandbox,err-step-size",
|
|
&err_step_size);
|
|
if (ret) {
|
|
dev_dbg(dev,
|
|
"No sandbox,err-step-size property for chip %s",
|
|
ofnode_get_name(np));
|
|
goto err;
|
|
}
|
|
|
|
chip = calloc(sizeof(*chip), 1);
|
|
if (!chip) {
|
|
ret = -ENOMEM;
|
|
goto err;
|
|
}
|
|
|
|
chip->cs = cs;
|
|
chip->id = id;
|
|
chip->id_len = id_len;
|
|
chip->chunksize = pagesize + oobsize;
|
|
chip->pagesize = pagesize;
|
|
chip->pages = pages;
|
|
chip->pages_per_erase = erasesize / pagesize;
|
|
memset(chip->tmp, 0xff, chip->chunksize);
|
|
|
|
chip->err_count = err_count;
|
|
chip->err_step_bits = err_step_size * 8;
|
|
chip->err_steps = pagesize / err_step_size;
|
|
|
|
expected_size = (off_t)pages * chip->chunksize;
|
|
snprintf(filename, sizeof(filename),
|
|
"/tmp/u-boot.nand%d.XXXXXX", devnum);
|
|
chip->fd = os_mktemp(filename, expected_size);
|
|
if (chip->fd < 0) {
|
|
dev_dbg(dev, "Could not create temp file %s\n",
|
|
filename);
|
|
ret = chip->fd;
|
|
goto err_chip;
|
|
}
|
|
|
|
chip->programmed = calloc(sizeof(long),
|
|
BITS_TO_LONGS(pages));
|
|
if (!chip->programmed) {
|
|
ret = -ENOMEM;
|
|
goto err_fd;
|
|
}
|
|
|
|
if (onfi) {
|
|
memcpy(chip->onfi, onfi, onfi_len);
|
|
memcpy(chip->onfi + onfi_len, onfi, onfi_len);
|
|
memcpy(chip->onfi + 2 * onfi_len, onfi, onfi_len);
|
|
}
|
|
|
|
nand = &chip->nand;
|
|
nand->options = spl_in_proper() ? 0 : NAND_SKIP_BBTSCAN;
|
|
nand->flash_node = np;
|
|
nand->dev_ready = sand_nand_dev_ready;
|
|
nand->cmdfunc = sand_nand_command;
|
|
nand->waitfunc = sand_nand_wait;
|
|
nand->select_chip = sand_nand_select_chip;
|
|
nand->read_byte = sand_nand_read_byte;
|
|
nand->read_word = sand_nand_read_word;
|
|
nand->read_buf = sand_nand_read_buf;
|
|
nand->write_buf = sand_nand_write_buf;
|
|
nand->ecc.options = NAND_ECC_GENERIC_ERASED_CHECK;
|
|
|
|
mtd = nand_to_mtd(nand);
|
|
mtd->dev = dev;
|
|
|
|
ret = nand_scan(mtd, CONFIG_SYS_NAND_MAX_CHIPS);
|
|
if (ret) {
|
|
dev_dbg(dev, "Could not scan chip %s: %d\n",
|
|
ofnode_get_name(np), ret);
|
|
goto err_prog;
|
|
}
|
|
chip->ecc_bits = nand->ecc.layout->eccbytes * 8 /
|
|
chip->err_steps;
|
|
|
|
ret = nand_register(devnum, mtd);
|
|
if (ret) {
|
|
dev_dbg(dev, "Could not register nand %d: %d\n", devnum,
|
|
ret);
|
|
goto err_prog;
|
|
}
|
|
|
|
if (!nand_chip)
|
|
nand_chip = nand;
|
|
|
|
list_add_tail(&chip->node, &priv->chips);
|
|
devnum++;
|
|
continue;
|
|
|
|
err_prog:
|
|
free(chip->programmed);
|
|
err_fd:
|
|
os_close(chip->fd);
|
|
err_chip:
|
|
free(chip);
|
|
goto err;
|
|
}
|
|
|
|
return 0;
|
|
|
|
err:
|
|
sand_nand_remove(dev);
|
|
return ret;
|
|
}
|
|
|
|
static const struct udevice_id sand_nand_ids[] = {
|
|
{ .compatible = "sandbox,nand" },
|
|
{ }
|
|
};
|
|
|
|
U_BOOT_DRIVER(sand_nand) = {
|
|
.name = "sand-nand",
|
|
.id = UCLASS_MTD,
|
|
.of_match = sand_nand_ids,
|
|
.probe = sand_nand_probe,
|
|
.remove = sand_nand_remove,
|
|
.priv_auto = sizeof(struct sand_nand_priv),
|
|
};
|
|
|
|
void board_nand_init(void)
|
|
{
|
|
struct udevice *dev;
|
|
int err;
|
|
|
|
err = uclass_get_device_by_driver(UCLASS_MTD, DM_DRIVER_REF(sand_nand),
|
|
&dev);
|
|
if (err && err != -ENODEV)
|
|
log_info("Failed to get sandbox NAND: %d\n", err);
|
|
}
|
|
|
|
#if IS_ENABLED(CONFIG_SPL_BUILD) && IS_ENABLED(CONFIG_SPL_NAND_INIT)
|
|
void nand_deselect(void)
|
|
{
|
|
nand_chip->select_chip(nand_to_mtd(nand_chip), -1);
|
|
}
|
|
|
|
static int nand_is_bad_block(int block)
|
|
{
|
|
struct mtd_info *mtd = nand_to_mtd(nand_chip);
|
|
|
|
return mtd_block_isbad(mtd, block << mtd->erasesize_shift);
|
|
}
|
|
|
|
static int nand_read_page(int block, int page, uchar *dst)
|
|
{
|
|
struct mtd_info *mtd = nand_to_mtd(nand_chip);
|
|
loff_t ofs = ((loff_t)block << mtd->erasesize_shift) +
|
|
((loff_t)page << mtd->writesize_shift);
|
|
size_t len = mtd->writesize;
|
|
|
|
return nand_read(mtd, ofs, &len, dst);
|
|
}
|
|
|
|
#include "nand_spl_loaders.c"
|
|
#endif /* CONFIG_SPL_NAND_INIT */
|