2005-04-17 05:20:36 +07:00
|
|
|
/*
|
|
|
|
* drivers/mtd/nand.c
|
|
|
|
*
|
|
|
|
* Overview:
|
|
|
|
* This is the generic MTD driver for NAND flash devices. It should be
|
|
|
|
* capable of working with almost all NAND chips currently available.
|
2005-11-07 18:15:49 +07:00
|
|
|
*
|
2005-04-17 05:20:36 +07:00
|
|
|
* Additional technical information is available on
|
2007-07-28 18:07:16 +07:00
|
|
|
* http://www.linux-mtd.infradead.org/doc/nand.html
|
2005-11-07 18:15:49 +07:00
|
|
|
*
|
2005-04-17 05:20:36 +07:00
|
|
|
* Copyright (C) 2000 Steven J. Hill (sjhill@realitydiluted.com)
|
2006-05-24 17:07:37 +07:00
|
|
|
* 2002-2006 Thomas Gleixner (tglx@linutronix.de)
|
2005-04-17 05:20:36 +07:00
|
|
|
*
|
2006-05-24 17:07:37 +07:00
|
|
|
* Credits:
|
2005-11-07 18:15:49 +07:00
|
|
|
* David Woodhouse for adding multichip support
|
|
|
|
*
|
2005-04-17 05:20:36 +07:00
|
|
|
* Aleph One Ltd. and Toby Churchill Ltd. for supporting the
|
|
|
|
* rework for 2K page size chips
|
|
|
|
*
|
2006-05-24 17:07:37 +07:00
|
|
|
* TODO:
|
2005-04-17 05:20:36 +07:00
|
|
|
* Enable cached programming for 2k page size chips
|
|
|
|
* Check, if mtd->ecctype should be set to MTD_ECC_HW
|
2011-06-24 04:12:08 +07:00
|
|
|
* if we have HW ECC support.
|
2007-07-23 20:06:50 +07:00
|
|
|
* BBT table is not serialized, has to be fixed
|
2005-04-17 05:20:36 +07:00
|
|
|
*
|
|
|
|
* This program is free software; you can redistribute it and/or modify
|
|
|
|
* it under the terms of the GNU General Public License version 2 as
|
|
|
|
* published by the Free Software Foundation.
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
2006-05-14 07:20:46 +07:00
|
|
|
#include <linux/module.h>
|
2005-04-17 05:20:36 +07:00
|
|
|
#include <linux/delay.h>
|
|
|
|
#include <linux/errno.h>
|
2006-05-23 16:54:38 +07:00
|
|
|
#include <linux/err.h>
|
2005-04-17 05:20:36 +07:00
|
|
|
#include <linux/sched.h>
|
|
|
|
#include <linux/slab.h>
|
|
|
|
#include <linux/types.h>
|
|
|
|
#include <linux/mtd/mtd.h>
|
|
|
|
#include <linux/mtd/nand.h>
|
|
|
|
#include <linux/mtd/nand_ecc.h>
|
2011-03-11 17:05:33 +07:00
|
|
|
#include <linux/mtd/nand_bch.h>
|
2005-04-17 05:20:36 +07:00
|
|
|
#include <linux/interrupt.h>
|
|
|
|
#include <linux/bitops.h>
|
2006-03-31 17:31:14 +07:00
|
|
|
#include <linux/leds.h>
|
2010-09-07 18:23:45 +07:00
|
|
|
#include <linux/io.h>
|
2005-04-17 05:20:36 +07:00
|
|
|
#include <linux/mtd/partitions.h>
|
|
|
|
|
|
|
|
/* Define default oob placement schemes for large and small page devices */
|
2006-05-28 03:16:10 +07:00
|
|
|
static struct nand_ecclayout nand_oob_8 = {
|
2005-04-17 05:20:36 +07:00
|
|
|
.eccbytes = 3,
|
|
|
|
.eccpos = {0, 1, 2},
|
2006-05-28 03:16:10 +07:00
|
|
|
.oobfree = {
|
|
|
|
{.offset = 3,
|
|
|
|
.length = 2},
|
|
|
|
{.offset = 6,
|
2010-09-07 18:23:43 +07:00
|
|
|
.length = 2} }
|
2005-04-17 05:20:36 +07:00
|
|
|
};
|
|
|
|
|
2006-05-28 03:16:10 +07:00
|
|
|
static struct nand_ecclayout nand_oob_16 = {
|
2005-04-17 05:20:36 +07:00
|
|
|
.eccbytes = 6,
|
|
|
|
.eccpos = {0, 1, 2, 3, 6, 7},
|
2006-05-28 03:16:10 +07:00
|
|
|
.oobfree = {
|
|
|
|
{.offset = 8,
|
2010-09-07 18:23:43 +07:00
|
|
|
. length = 8} }
|
2005-04-17 05:20:36 +07:00
|
|
|
};
|
|
|
|
|
2006-05-28 03:16:10 +07:00
|
|
|
static struct nand_ecclayout nand_oob_64 = {
|
2005-04-17 05:20:36 +07:00
|
|
|
.eccbytes = 24,
|
|
|
|
.eccpos = {
|
2006-05-14 00:07:53 +07:00
|
|
|
40, 41, 42, 43, 44, 45, 46, 47,
|
|
|
|
48, 49, 50, 51, 52, 53, 54, 55,
|
|
|
|
56, 57, 58, 59, 60, 61, 62, 63},
|
2006-05-28 03:16:10 +07:00
|
|
|
.oobfree = {
|
|
|
|
{.offset = 2,
|
2010-09-07 18:23:43 +07:00
|
|
|
.length = 38} }
|
2005-04-17 05:20:36 +07:00
|
|
|
};
|
|
|
|
|
2007-12-12 23:27:03 +07:00
|
|
|
static struct nand_ecclayout nand_oob_128 = {
|
|
|
|
.eccbytes = 48,
|
|
|
|
.eccpos = {
|
|
|
|
80, 81, 82, 83, 84, 85, 86, 87,
|
|
|
|
88, 89, 90, 91, 92, 93, 94, 95,
|
|
|
|
96, 97, 98, 99, 100, 101, 102, 103,
|
|
|
|
104, 105, 106, 107, 108, 109, 110, 111,
|
|
|
|
112, 113, 114, 115, 116, 117, 118, 119,
|
|
|
|
120, 121, 122, 123, 124, 125, 126, 127},
|
|
|
|
.oobfree = {
|
|
|
|
{.offset = 2,
|
2010-09-07 18:23:43 +07:00
|
|
|
.length = 78} }
|
2007-12-12 23:27:03 +07:00
|
|
|
};
|
|
|
|
|
2012-11-19 13:43:30 +07:00
|
|
|
static int nand_get_device(struct mtd_info *mtd, int new_state);
|
2005-04-17 05:20:36 +07:00
|
|
|
|
2006-05-29 08:26:58 +07:00
|
|
|
static int nand_do_write_oob(struct mtd_info *mtd, loff_t to,
|
|
|
|
struct mtd_oob_ops *ops);
|
|
|
|
|
2006-05-24 04:48:57 +07:00
|
|
|
/*
|
2008-02-03 22:22:34 +07:00
|
|
|
* For devices which display every fart in the system on a separate LED. Is
|
2006-05-24 04:48:57 +07:00
|
|
|
* compiled away when LED support is disabled.
|
|
|
|
*/
|
|
|
|
DEFINE_LED_TRIGGER(nand_led_trigger);
|
|
|
|
|
2010-02-03 15:42:24 +07:00
|
|
|
static int check_offs_len(struct mtd_info *mtd,
|
|
|
|
loff_t ofs, uint64_t len)
|
|
|
|
{
|
|
|
|
struct nand_chip *chip = mtd->priv;
|
|
|
|
int ret = 0;
|
|
|
|
|
|
|
|
/* Start address must align on block boundary */
|
2013-08-09 16:49:05 +07:00
|
|
|
if (ofs & ((1ULL << chip->phys_erase_shift) - 1)) {
|
2011-07-20 00:06:09 +07:00
|
|
|
pr_debug("%s: unaligned address\n", __func__);
|
2010-02-03 15:42:24 +07:00
|
|
|
ret = -EINVAL;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Length must align on block boundary */
|
2013-08-09 16:49:05 +07:00
|
|
|
if (len & ((1ULL << chip->phys_erase_shift) - 1)) {
|
2011-07-20 00:06:09 +07:00
|
|
|
pr_debug("%s: length not block aligned\n", __func__);
|
2010-02-03 15:42:24 +07:00
|
|
|
ret = -EINVAL;
|
|
|
|
}
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2005-04-17 05:20:36 +07:00
|
|
|
/**
|
|
|
|
* nand_release_device - [GENERIC] release chip
|
2011-05-26 04:59:01 +07:00
|
|
|
* @mtd: MTD device structure
|
2005-11-07 18:15:49 +07:00
|
|
|
*
|
2012-11-19 13:43:29 +07:00
|
|
|
* Release chip lock and wake up anyone waiting on the device.
|
2005-04-17 05:20:36 +07:00
|
|
|
*/
|
2006-05-14 00:07:53 +07:00
|
|
|
static void nand_release_device(struct mtd_info *mtd)
|
2005-04-17 05:20:36 +07:00
|
|
|
{
|
2006-05-24 17:07:37 +07:00
|
|
|
struct nand_chip *chip = mtd->priv;
|
2005-04-17 05:20:36 +07:00
|
|
|
|
2006-05-23 16:37:03 +07:00
|
|
|
/* Release the controller and the chip */
|
2006-05-24 17:07:37 +07:00
|
|
|
spin_lock(&chip->controller->lock);
|
|
|
|
chip->controller->active = NULL;
|
|
|
|
chip->state = FL_READY;
|
|
|
|
wake_up(&chip->controller->wq);
|
|
|
|
spin_unlock(&chip->controller->lock);
|
2005-04-17 05:20:36 +07:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* nand_read_byte - [DEFAULT] read one byte from the chip
|
2011-05-26 04:59:01 +07:00
|
|
|
* @mtd: MTD device structure
|
2005-04-17 05:20:36 +07:00
|
|
|
*
|
2011-06-24 04:12:08 +07:00
|
|
|
* Default read function for 8bit buswidth
|
2005-04-17 05:20:36 +07:00
|
|
|
*/
|
2006-05-23 16:52:35 +07:00
|
|
|
static uint8_t nand_read_byte(struct mtd_info *mtd)
|
2005-04-17 05:20:36 +07:00
|
|
|
{
|
2006-05-24 17:07:37 +07:00
|
|
|
struct nand_chip *chip = mtd->priv;
|
|
|
|
return readb(chip->IO_ADDR_R);
|
2005-04-17 05:20:36 +07:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2012-11-09 21:20:58 +07:00
|
|
|
* nand_read_byte16 - [DEFAULT] read one byte endianness aware from the chip
|
2011-06-24 04:12:08 +07:00
|
|
|
* nand_read_byte16 - [DEFAULT] read one byte endianness aware from the chip
|
2011-05-26 04:59:01 +07:00
|
|
|
* @mtd: MTD device structure
|
2005-04-17 05:20:36 +07:00
|
|
|
*
|
2011-06-24 04:12:08 +07:00
|
|
|
* Default read function for 16bit buswidth with endianness conversion.
|
|
|
|
*
|
2005-04-17 05:20:36 +07:00
|
|
|
*/
|
2006-05-23 16:52:35 +07:00
|
|
|
static uint8_t nand_read_byte16(struct mtd_info *mtd)
|
2005-04-17 05:20:36 +07:00
|
|
|
{
|
2006-05-24 17:07:37 +07:00
|
|
|
struct nand_chip *chip = mtd->priv;
|
|
|
|
return (uint8_t) cpu_to_le16(readw(chip->IO_ADDR_R));
|
2005-04-17 05:20:36 +07:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* nand_read_word - [DEFAULT] read one word from the chip
|
2011-05-26 04:59:01 +07:00
|
|
|
* @mtd: MTD device structure
|
2005-04-17 05:20:36 +07:00
|
|
|
*
|
2011-06-24 04:12:08 +07:00
|
|
|
* Default read function for 16bit buswidth without endianness conversion.
|
2005-04-17 05:20:36 +07:00
|
|
|
*/
|
|
|
|
static u16 nand_read_word(struct mtd_info *mtd)
|
|
|
|
{
|
2006-05-24 17:07:37 +07:00
|
|
|
struct nand_chip *chip = mtd->priv;
|
|
|
|
return readw(chip->IO_ADDR_R);
|
2005-04-17 05:20:36 +07:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* nand_select_chip - [DEFAULT] control CE line
|
2011-05-26 04:59:01 +07:00
|
|
|
* @mtd: MTD device structure
|
|
|
|
* @chipnr: chipnumber to select, -1 for deselect
|
2005-04-17 05:20:36 +07:00
|
|
|
*
|
|
|
|
* Default select function for 1 chip devices.
|
|
|
|
*/
|
2006-05-24 17:07:37 +07:00
|
|
|
static void nand_select_chip(struct mtd_info *mtd, int chipnr)
|
2005-04-17 05:20:36 +07:00
|
|
|
{
|
2006-05-24 17:07:37 +07:00
|
|
|
struct nand_chip *chip = mtd->priv;
|
|
|
|
|
|
|
|
switch (chipnr) {
|
2005-04-17 05:20:36 +07:00
|
|
|
case -1:
|
2006-05-24 17:07:37 +07:00
|
|
|
chip->cmd_ctrl(mtd, NAND_CMD_NONE, 0 | NAND_CTRL_CHANGE);
|
2005-04-17 05:20:36 +07:00
|
|
|
break;
|
|
|
|
case 0:
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
BUG();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* nand_write_buf - [DEFAULT] write buffer to chip
|
2011-05-26 04:59:01 +07:00
|
|
|
* @mtd: MTD device structure
|
|
|
|
* @buf: data buffer
|
|
|
|
* @len: number of bytes to write
|
2005-04-17 05:20:36 +07:00
|
|
|
*
|
2011-06-24 04:12:08 +07:00
|
|
|
* Default write function for 8bit buswidth.
|
2005-04-17 05:20:36 +07:00
|
|
|
*/
|
2006-05-23 16:52:35 +07:00
|
|
|
static void nand_write_buf(struct mtd_info *mtd, const uint8_t *buf, int len)
|
2005-04-17 05:20:36 +07:00
|
|
|
{
|
2006-05-24 17:07:37 +07:00
|
|
|
struct nand_chip *chip = mtd->priv;
|
2005-04-17 05:20:36 +07:00
|
|
|
|
2013-04-13 12:32:13 +07:00
|
|
|
iowrite8_rep(chip->IO_ADDR_W, buf, len);
|
2005-04-17 05:20:36 +07:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2005-11-07 18:15:49 +07:00
|
|
|
* nand_read_buf - [DEFAULT] read chip data into buffer
|
2011-05-26 04:59:01 +07:00
|
|
|
* @mtd: MTD device structure
|
|
|
|
* @buf: buffer to store date
|
|
|
|
* @len: number of bytes to read
|
2005-04-17 05:20:36 +07:00
|
|
|
*
|
2011-06-24 04:12:08 +07:00
|
|
|
* Default read function for 8bit buswidth.
|
2005-04-17 05:20:36 +07:00
|
|
|
*/
|
2006-05-23 16:52:35 +07:00
|
|
|
static void nand_read_buf(struct mtd_info *mtd, uint8_t *buf, int len)
|
2005-04-17 05:20:36 +07:00
|
|
|
{
|
2006-05-24 17:07:37 +07:00
|
|
|
struct nand_chip *chip = mtd->priv;
|
2005-04-17 05:20:36 +07:00
|
|
|
|
2013-04-13 12:32:13 +07:00
|
|
|
ioread8_rep(chip->IO_ADDR_R, buf, len);
|
2005-04-17 05:20:36 +07:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* nand_write_buf16 - [DEFAULT] write buffer to chip
|
2011-05-26 04:59:01 +07:00
|
|
|
* @mtd: MTD device structure
|
|
|
|
* @buf: data buffer
|
|
|
|
* @len: number of bytes to write
|
2005-04-17 05:20:36 +07:00
|
|
|
*
|
2011-06-24 04:12:08 +07:00
|
|
|
* Default write function for 16bit buswidth.
|
2005-04-17 05:20:36 +07:00
|
|
|
*/
|
2006-05-23 16:52:35 +07:00
|
|
|
static void nand_write_buf16(struct mtd_info *mtd, const uint8_t *buf, int len)
|
2005-04-17 05:20:36 +07:00
|
|
|
{
|
2006-05-24 17:07:37 +07:00
|
|
|
struct nand_chip *chip = mtd->priv;
|
2005-04-17 05:20:36 +07:00
|
|
|
u16 *p = (u16 *) buf;
|
2005-11-07 18:15:49 +07:00
|
|
|
|
2013-04-13 12:32:13 +07:00
|
|
|
iowrite16_rep(chip->IO_ADDR_W, p, len >> 1);
|
2005-04-17 05:20:36 +07:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2005-11-07 18:15:49 +07:00
|
|
|
* nand_read_buf16 - [DEFAULT] read chip data into buffer
|
2011-05-26 04:59:01 +07:00
|
|
|
* @mtd: MTD device structure
|
|
|
|
* @buf: buffer to store date
|
|
|
|
* @len: number of bytes to read
|
2005-04-17 05:20:36 +07:00
|
|
|
*
|
2011-06-24 04:12:08 +07:00
|
|
|
* Default read function for 16bit buswidth.
|
2005-04-17 05:20:36 +07:00
|
|
|
*/
|
2006-05-23 16:52:35 +07:00
|
|
|
static void nand_read_buf16(struct mtd_info *mtd, uint8_t *buf, int len)
|
2005-04-17 05:20:36 +07:00
|
|
|
{
|
2006-05-24 17:07:37 +07:00
|
|
|
struct nand_chip *chip = mtd->priv;
|
2005-04-17 05:20:36 +07:00
|
|
|
u16 *p = (u16 *) buf;
|
|
|
|
|
2013-04-13 12:32:13 +07:00
|
|
|
ioread16_rep(chip->IO_ADDR_R, p, len >> 1);
|
2005-04-17 05:20:36 +07:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* nand_block_bad - [DEFAULT] Read bad block marker from the chip
|
2011-05-26 04:59:01 +07:00
|
|
|
* @mtd: MTD device structure
|
|
|
|
* @ofs: offset from device start
|
|
|
|
* @getchip: 0, if the chip is already selected
|
2005-04-17 05:20:36 +07:00
|
|
|
*
|
2005-11-07 18:15:49 +07:00
|
|
|
* Check, if the block is bad.
|
2005-04-17 05:20:36 +07:00
|
|
|
*/
|
|
|
|
static int nand_block_bad(struct mtd_info *mtd, loff_t ofs, int getchip)
|
|
|
|
{
|
2012-01-14 09:11:48 +07:00
|
|
|
int page, chipnr, res = 0, i = 0;
|
2006-05-24 17:07:37 +07:00
|
|
|
struct nand_chip *chip = mtd->priv;
|
2005-04-17 05:20:36 +07:00
|
|
|
u16 bad;
|
|
|
|
|
2011-06-01 06:31:21 +07:00
|
|
|
if (chip->bbt_options & NAND_BBT_SCANLASTPAGE)
|
2010-05-05 10:58:10 +07:00
|
|
|
ofs += mtd->erasesize - mtd->writesize;
|
|
|
|
|
2007-05-03 13:39:37 +07:00
|
|
|
page = (int)(ofs >> chip->page_shift) & chip->pagemask;
|
|
|
|
|
2005-04-17 05:20:36 +07:00
|
|
|
if (getchip) {
|
2006-05-24 17:07:37 +07:00
|
|
|
chipnr = (int)(ofs >> chip->chip_shift);
|
2005-04-17 05:20:36 +07:00
|
|
|
|
2012-11-19 13:43:30 +07:00
|
|
|
nand_get_device(mtd, FL_READING);
|
2005-04-17 05:20:36 +07:00
|
|
|
|
|
|
|
/* Select the NAND device */
|
2006-05-24 17:07:37 +07:00
|
|
|
chip->select_chip(mtd, chipnr);
|
2007-05-03 13:39:37 +07:00
|
|
|
}
|
2005-04-17 05:20:36 +07:00
|
|
|
|
2012-01-14 09:11:48 +07:00
|
|
|
do {
|
|
|
|
if (chip->options & NAND_BUSWIDTH_16) {
|
|
|
|
chip->cmdfunc(mtd, NAND_CMD_READOOB,
|
|
|
|
chip->badblockpos & 0xFE, page);
|
|
|
|
bad = cpu_to_le16(chip->read_word(mtd));
|
|
|
|
if (chip->badblockpos & 0x1)
|
|
|
|
bad >>= 8;
|
|
|
|
else
|
|
|
|
bad &= 0xFF;
|
|
|
|
} else {
|
|
|
|
chip->cmdfunc(mtd, NAND_CMD_READOOB, chip->badblockpos,
|
|
|
|
page);
|
|
|
|
bad = chip->read_byte(mtd);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (likely(chip->badblockbits == 8))
|
|
|
|
res = bad != 0xFF;
|
2010-02-23 01:39:38 +07:00
|
|
|
else
|
2012-01-14 09:11:48 +07:00
|
|
|
res = hweight8(bad) < chip->badblockbits;
|
|
|
|
ofs += mtd->writesize;
|
|
|
|
page = (int)(ofs >> chip->page_shift) & chip->pagemask;
|
|
|
|
i++;
|
|
|
|
} while (!res && i < 2 && (chip->bbt_options & NAND_BBT_SCAN2NDPAGE));
|
2010-02-23 01:39:38 +07:00
|
|
|
|
2012-11-19 13:43:29 +07:00
|
|
|
if (getchip) {
|
|
|
|
chip->select_chip(mtd, -1);
|
2005-04-17 05:20:36 +07:00
|
|
|
nand_release_device(mtd);
|
2012-11-19 13:43:29 +07:00
|
|
|
}
|
2005-11-07 18:15:49 +07:00
|
|
|
|
2005-04-17 05:20:36 +07:00
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2013-07-31 07:52:58 +07:00
|
|
|
* nand_default_block_markbad - [DEFAULT] mark a block bad via bad block marker
|
2011-05-26 04:59:01 +07:00
|
|
|
* @mtd: MTD device structure
|
|
|
|
* @ofs: offset from device start
|
2005-04-17 05:20:36 +07:00
|
|
|
*
|
2011-05-26 04:59:01 +07:00
|
|
|
* This is the default implementation, which can be overridden by a hardware
|
2013-07-31 07:52:58 +07:00
|
|
|
* specific driver. It provides the details for writing a bad block marker to a
|
|
|
|
* block.
|
|
|
|
*/
|
|
|
|
static int nand_default_block_markbad(struct mtd_info *mtd, loff_t ofs)
|
|
|
|
{
|
|
|
|
struct nand_chip *chip = mtd->priv;
|
|
|
|
struct mtd_oob_ops ops;
|
|
|
|
uint8_t buf[2] = { 0, 0 };
|
|
|
|
int ret = 0, res, i = 0;
|
|
|
|
|
|
|
|
ops.datbuf = NULL;
|
|
|
|
ops.oobbuf = buf;
|
|
|
|
ops.ooboffs = chip->badblockpos;
|
|
|
|
if (chip->options & NAND_BUSWIDTH_16) {
|
|
|
|
ops.ooboffs &= ~0x01;
|
|
|
|
ops.len = ops.ooblen = 2;
|
|
|
|
} else {
|
|
|
|
ops.len = ops.ooblen = 1;
|
|
|
|
}
|
|
|
|
ops.mode = MTD_OPS_PLACE_OOB;
|
|
|
|
|
|
|
|
/* Write to first/last page(s) if necessary */
|
|
|
|
if (chip->bbt_options & NAND_BBT_SCANLASTPAGE)
|
|
|
|
ofs += mtd->erasesize - mtd->writesize;
|
|
|
|
do {
|
|
|
|
res = nand_do_write_oob(mtd, ofs, &ops);
|
|
|
|
if (!ret)
|
|
|
|
ret = res;
|
|
|
|
|
|
|
|
i++;
|
|
|
|
ofs += mtd->writesize;
|
|
|
|
} while ((chip->bbt_options & NAND_BBT_SCAN2NDPAGE) && i < 2);
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* nand_block_markbad_lowlevel - mark a block bad
|
|
|
|
* @mtd: MTD device structure
|
|
|
|
* @ofs: offset from device start
|
|
|
|
*
|
|
|
|
* This function performs the generic NAND bad block marking steps (i.e., bad
|
|
|
|
* block table(s) and/or marker(s)). We only allow the hardware driver to
|
|
|
|
* specify how to write bad block markers to OOB (chip->block_markbad).
|
|
|
|
*
|
2013-07-31 07:52:59 +07:00
|
|
|
* We try operations in the following order:
|
mtd: nand: write BBM to OOB even with flash-based BBT
Currently, the flash-based BBT implementation writes bad block data only
to its flash-based table and not to the OOB marker area. Then, as new bad
blocks are marked over time, the OOB markers become incomplete and the
flash-based table becomes the only source of current bad block
information. This becomes an obvious problem when, for example:
* bootloader cannot read the flash-based BBT format
* BBT is corrupted and the flash must be rescanned for bad
blocks; we want to remember bad blocks that were marked from Linux
So to keep the bad block markers in sync with the flash-based BBT, this
patch changes the default so that we write bad block markers to the proper
OOB area on each block in addition to flash-based BBT. Comments are
updated, expanded, and/or relocated as necessary.
The new flash-based BBT procedure for marking bad blocks:
(1) erase the affected block, to allow OOB marker to be written cleanly
(2) update in-memory BBT
(3) write bad block marker to OOB area of affected block
(4) update flash-based BBT
Note that we retain the first error encountered in (3) or (4), finish the
procedures, and dump the error in the end.
This should handle power cuts gracefully enough. (1) and (2) are mostly
harmless (note that (1) will not erase an already-recognized bad block).
The OOB and BBT may be "out of sync" if we experience power loss bewteen
(3) and (4), but we can reasonably expect that on next boot, subsequent
I/O operations will discover that the block should be marked bad again,
thus re-syncing the OOB and BBT.
Note that this is a change from the previous default flash-based BBT
behavior. If your system cannot support writing bad block markers to OOB,
use the new NAND_BBT_NO_OOB_BBM option (in combination with
NAND_BBT_USE_FLASH and NAND_BBT_NO_OOB).
Signed-off-by: Brian Norris <computersforpeace@gmail.com>
Signed-off-by: Artem Bityutskiy <artem.bityutskiy@linux.intel.com>
Signed-off-by: David Woodhouse <David.Woodhouse@intel.com>
2012-02-07 04:44:00 +07:00
|
|
|
* (1) erase the affected block, to allow OOB marker to be written cleanly
|
2013-07-31 07:52:59 +07:00
|
|
|
* (2) write bad block marker to OOB area of affected block (unless flag
|
|
|
|
* NAND_BBT_NO_OOB_BBM is present)
|
|
|
|
* (3) update the BBT
|
|
|
|
* Note that we retain the first error encountered in (2) or (3), finish the
|
mtd: nand: write BBM to OOB even with flash-based BBT
Currently, the flash-based BBT implementation writes bad block data only
to its flash-based table and not to the OOB marker area. Then, as new bad
blocks are marked over time, the OOB markers become incomplete and the
flash-based table becomes the only source of current bad block
information. This becomes an obvious problem when, for example:
* bootloader cannot read the flash-based BBT format
* BBT is corrupted and the flash must be rescanned for bad
blocks; we want to remember bad blocks that were marked from Linux
So to keep the bad block markers in sync with the flash-based BBT, this
patch changes the default so that we write bad block markers to the proper
OOB area on each block in addition to flash-based BBT. Comments are
updated, expanded, and/or relocated as necessary.
The new flash-based BBT procedure for marking bad blocks:
(1) erase the affected block, to allow OOB marker to be written cleanly
(2) update in-memory BBT
(3) write bad block marker to OOB area of affected block
(4) update flash-based BBT
Note that we retain the first error encountered in (3) or (4), finish the
procedures, and dump the error in the end.
This should handle power cuts gracefully enough. (1) and (2) are mostly
harmless (note that (1) will not erase an already-recognized bad block).
The OOB and BBT may be "out of sync" if we experience power loss bewteen
(3) and (4), but we can reasonably expect that on next boot, subsequent
I/O operations will discover that the block should be marked bad again,
thus re-syncing the OOB and BBT.
Note that this is a change from the previous default flash-based BBT
behavior. If your system cannot support writing bad block markers to OOB,
use the new NAND_BBT_NO_OOB_BBM option (in combination with
NAND_BBT_USE_FLASH and NAND_BBT_NO_OOB).
Signed-off-by: Brian Norris <computersforpeace@gmail.com>
Signed-off-by: Artem Bityutskiy <artem.bityutskiy@linux.intel.com>
Signed-off-by: David Woodhouse <David.Woodhouse@intel.com>
2012-02-07 04:44:00 +07:00
|
|
|
* procedures, and dump the error in the end.
|
2005-04-17 05:20:36 +07:00
|
|
|
*/
|
2013-07-31 07:52:58 +07:00
|
|
|
static int nand_block_markbad_lowlevel(struct mtd_info *mtd, loff_t ofs)
|
2005-04-17 05:20:36 +07:00
|
|
|
{
|
2006-05-24 17:07:37 +07:00
|
|
|
struct nand_chip *chip = mtd->priv;
|
2013-07-31 07:52:59 +07:00
|
|
|
int res, ret = 0;
|
2005-11-07 18:15:49 +07:00
|
|
|
|
2013-07-31 07:52:59 +07:00
|
|
|
if (!(chip->bbt_options & NAND_BBT_NO_OOB_BBM)) {
|
2012-01-14 09:11:47 +07:00
|
|
|
struct erase_info einfo;
|
|
|
|
|
|
|
|
/* Attempt erase before marking OOB */
|
|
|
|
memset(&einfo, 0, sizeof(einfo));
|
|
|
|
einfo.mtd = mtd;
|
|
|
|
einfo.addr = ofs;
|
2013-08-09 16:49:05 +07:00
|
|
|
einfo.len = 1ULL << chip->phys_erase_shift;
|
2012-01-14 09:11:47 +07:00
|
|
|
nand_erase_nand(mtd, &einfo, 0);
|
2005-04-17 05:20:36 +07:00
|
|
|
|
2013-07-31 07:52:59 +07:00
|
|
|
/* Write bad block marker to OOB */
|
2012-11-19 13:43:30 +07:00
|
|
|
nand_get_device(mtd, FL_WRITING);
|
2013-07-31 07:52:58 +07:00
|
|
|
ret = chip->block_markbad(mtd, ofs);
|
2007-07-23 20:06:50 +07:00
|
|
|
nand_release_device(mtd);
|
2006-05-30 05:37:34 +07:00
|
|
|
}
|
mtd: nand: write BBM to OOB even with flash-based BBT
Currently, the flash-based BBT implementation writes bad block data only
to its flash-based table and not to the OOB marker area. Then, as new bad
blocks are marked over time, the OOB markers become incomplete and the
flash-based table becomes the only source of current bad block
information. This becomes an obvious problem when, for example:
* bootloader cannot read the flash-based BBT format
* BBT is corrupted and the flash must be rescanned for bad
blocks; we want to remember bad blocks that were marked from Linux
So to keep the bad block markers in sync with the flash-based BBT, this
patch changes the default so that we write bad block markers to the proper
OOB area on each block in addition to flash-based BBT. Comments are
updated, expanded, and/or relocated as necessary.
The new flash-based BBT procedure for marking bad blocks:
(1) erase the affected block, to allow OOB marker to be written cleanly
(2) update in-memory BBT
(3) write bad block marker to OOB area of affected block
(4) update flash-based BBT
Note that we retain the first error encountered in (3) or (4), finish the
procedures, and dump the error in the end.
This should handle power cuts gracefully enough. (1) and (2) are mostly
harmless (note that (1) will not erase an already-recognized bad block).
The OOB and BBT may be "out of sync" if we experience power loss bewteen
(3) and (4), but we can reasonably expect that on next boot, subsequent
I/O operations will discover that the block should be marked bad again,
thus re-syncing the OOB and BBT.
Note that this is a change from the previous default flash-based BBT
behavior. If your system cannot support writing bad block markers to OOB,
use the new NAND_BBT_NO_OOB_BBM option (in combination with
NAND_BBT_USE_FLASH and NAND_BBT_NO_OOB).
Signed-off-by: Brian Norris <computersforpeace@gmail.com>
Signed-off-by: Artem Bityutskiy <artem.bityutskiy@linux.intel.com>
Signed-off-by: David Woodhouse <David.Woodhouse@intel.com>
2012-02-07 04:44:00 +07:00
|
|
|
|
2013-07-31 07:52:59 +07:00
|
|
|
/* Mark block bad in BBT */
|
|
|
|
if (chip->bbt) {
|
|
|
|
res = nand_markbad_bbt(mtd, ofs);
|
mtd: nand: write BBM to OOB even with flash-based BBT
Currently, the flash-based BBT implementation writes bad block data only
to its flash-based table and not to the OOB marker area. Then, as new bad
blocks are marked over time, the OOB markers become incomplete and the
flash-based table becomes the only source of current bad block
information. This becomes an obvious problem when, for example:
* bootloader cannot read the flash-based BBT format
* BBT is corrupted and the flash must be rescanned for bad
blocks; we want to remember bad blocks that were marked from Linux
So to keep the bad block markers in sync with the flash-based BBT, this
patch changes the default so that we write bad block markers to the proper
OOB area on each block in addition to flash-based BBT. Comments are
updated, expanded, and/or relocated as necessary.
The new flash-based BBT procedure for marking bad blocks:
(1) erase the affected block, to allow OOB marker to be written cleanly
(2) update in-memory BBT
(3) write bad block marker to OOB area of affected block
(4) update flash-based BBT
Note that we retain the first error encountered in (3) or (4), finish the
procedures, and dump the error in the end.
This should handle power cuts gracefully enough. (1) and (2) are mostly
harmless (note that (1) will not erase an already-recognized bad block).
The OOB and BBT may be "out of sync" if we experience power loss bewteen
(3) and (4), but we can reasonably expect that on next boot, subsequent
I/O operations will discover that the block should be marked bad again,
thus re-syncing the OOB and BBT.
Note that this is a change from the previous default flash-based BBT
behavior. If your system cannot support writing bad block markers to OOB,
use the new NAND_BBT_NO_OOB_BBM option (in combination with
NAND_BBT_USE_FLASH and NAND_BBT_NO_OOB).
Signed-off-by: Brian Norris <computersforpeace@gmail.com>
Signed-off-by: Artem Bityutskiy <artem.bityutskiy@linux.intel.com>
Signed-off-by: David Woodhouse <David.Woodhouse@intel.com>
2012-02-07 04:44:00 +07:00
|
|
|
if (!ret)
|
|
|
|
ret = res;
|
|
|
|
}
|
|
|
|
|
2006-05-30 05:37:34 +07:00
|
|
|
if (!ret)
|
|
|
|
mtd->ecc_stats.badblocks++;
|
2007-07-23 20:06:50 +07:00
|
|
|
|
2006-05-30 05:37:34 +07:00
|
|
|
return ret;
|
2005-04-17 05:20:36 +07:00
|
|
|
}
|
|
|
|
|
2005-11-07 18:15:49 +07:00
|
|
|
/**
|
2005-04-17 05:20:36 +07:00
|
|
|
* nand_check_wp - [GENERIC] check if the chip is write protected
|
2011-05-26 04:59:01 +07:00
|
|
|
* @mtd: MTD device structure
|
2005-04-17 05:20:36 +07:00
|
|
|
*
|
2011-05-26 04:59:01 +07:00
|
|
|
* Check, if the device is write protected. The function expects, that the
|
|
|
|
* device is already selected.
|
2005-04-17 05:20:36 +07:00
|
|
|
*/
|
2006-05-14 00:07:53 +07:00
|
|
|
static int nand_check_wp(struct mtd_info *mtd)
|
2005-04-17 05:20:36 +07:00
|
|
|
{
|
2006-05-24 17:07:37 +07:00
|
|
|
struct nand_chip *chip = mtd->priv;
|
2010-02-23 01:39:40 +07:00
|
|
|
|
2011-05-26 04:59:01 +07:00
|
|
|
/* Broken xD cards report WP despite being writable */
|
2010-02-23 01:39:40 +07:00
|
|
|
if (chip->options & NAND_BROKEN_XD)
|
|
|
|
return 0;
|
|
|
|
|
2005-04-17 05:20:36 +07:00
|
|
|
/* Check the WP bit */
|
2006-05-24 17:07:37 +07:00
|
|
|
chip->cmdfunc(mtd, NAND_CMD_STATUS, -1, -1);
|
|
|
|
return (chip->read_byte(mtd) & NAND_STATUS_WP) ? 0 : 1;
|
2005-04-17 05:20:36 +07:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* nand_block_checkbad - [GENERIC] Check if a block is marked bad
|
2011-05-26 04:59:01 +07:00
|
|
|
* @mtd: MTD device structure
|
|
|
|
* @ofs: offset from device start
|
|
|
|
* @getchip: 0, if the chip is already selected
|
|
|
|
* @allowbbt: 1, if its allowed to access the bbt area
|
2005-04-17 05:20:36 +07:00
|
|
|
*
|
|
|
|
* Check, if the block is bad. Either by reading the bad block table or
|
|
|
|
* calling of the scan function.
|
|
|
|
*/
|
2006-05-23 16:50:56 +07:00
|
|
|
static int nand_block_checkbad(struct mtd_info *mtd, loff_t ofs, int getchip,
|
|
|
|
int allowbbt)
|
2005-04-17 05:20:36 +07:00
|
|
|
{
|
2006-05-24 17:07:37 +07:00
|
|
|
struct nand_chip *chip = mtd->priv;
|
2005-11-07 18:15:49 +07:00
|
|
|
|
2006-05-24 17:07:37 +07:00
|
|
|
if (!chip->bbt)
|
|
|
|
return chip->block_bad(mtd, ofs, getchip);
|
2005-11-07 18:15:49 +07:00
|
|
|
|
2005-04-17 05:20:36 +07:00
|
|
|
/* Return info from the table */
|
2006-05-14 00:07:53 +07:00
|
|
|
return nand_isbad_bbt(mtd, ofs, allowbbt);
|
2005-04-17 05:20:36 +07:00
|
|
|
}
|
|
|
|
|
2009-10-05 20:55:52 +07:00
|
|
|
/**
|
|
|
|
* panic_nand_wait_ready - [GENERIC] Wait for the ready pin after commands.
|
2011-05-26 04:59:01 +07:00
|
|
|
* @mtd: MTD device structure
|
|
|
|
* @timeo: Timeout
|
2009-10-05 20:55:52 +07:00
|
|
|
*
|
|
|
|
* Helper function for nand_wait_ready used when needing to wait in interrupt
|
|
|
|
* context.
|
|
|
|
*/
|
|
|
|
static void panic_nand_wait_ready(struct mtd_info *mtd, unsigned long timeo)
|
|
|
|
{
|
|
|
|
struct nand_chip *chip = mtd->priv;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
/* Wait for the device to get ready */
|
|
|
|
for (i = 0; i < timeo; i++) {
|
|
|
|
if (chip->dev_ready(mtd))
|
|
|
|
break;
|
|
|
|
touch_softlockup_watchdog();
|
|
|
|
mdelay(1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-06-24 04:12:08 +07:00
|
|
|
/* Wait for the ready pin, after a command. The timeout is caught later. */
|
2006-09-25 23:05:24 +07:00
|
|
|
void nand_wait_ready(struct mtd_info *mtd)
|
2005-02-23 04:56:49 +07:00
|
|
|
{
|
2006-05-24 17:07:37 +07:00
|
|
|
struct nand_chip *chip = mtd->priv;
|
2012-11-23 00:31:28 +07:00
|
|
|
unsigned long timeo = jiffies + msecs_to_jiffies(20);
|
2005-02-23 04:56:49 +07:00
|
|
|
|
2009-10-05 20:55:52 +07:00
|
|
|
/* 400ms timeout */
|
|
|
|
if (in_interrupt() || oops_in_progress)
|
|
|
|
return panic_nand_wait_ready(mtd, 400);
|
|
|
|
|
2006-03-31 17:31:14 +07:00
|
|
|
led_trigger_event(nand_led_trigger, LED_FULL);
|
2011-06-24 04:12:08 +07:00
|
|
|
/* Wait until command is processed or timeout occurs */
|
2005-02-23 04:56:49 +07:00
|
|
|
do {
|
2006-05-24 17:07:37 +07:00
|
|
|
if (chip->dev_ready(mtd))
|
2006-03-31 17:31:14 +07:00
|
|
|
break;
|
2005-09-07 05:16:27 +07:00
|
|
|
touch_softlockup_watchdog();
|
2005-11-07 18:15:49 +07:00
|
|
|
} while (time_before(jiffies, timeo));
|
2006-03-31 17:31:14 +07:00
|
|
|
led_trigger_event(nand_led_trigger, LED_OFF);
|
2005-02-23 04:56:49 +07:00
|
|
|
}
|
2006-09-25 23:05:24 +07:00
|
|
|
EXPORT_SYMBOL_GPL(nand_wait_ready);
|
2005-02-23 04:56:49 +07:00
|
|
|
|
2005-04-17 05:20:36 +07:00
|
|
|
/**
|
|
|
|
* nand_command - [DEFAULT] Send command to NAND device
|
2011-05-26 04:59:01 +07:00
|
|
|
* @mtd: MTD device structure
|
|
|
|
* @command: the command to be sent
|
|
|
|
* @column: the column address for this command, -1 if none
|
|
|
|
* @page_addr: the page address for this command, -1 if none
|
2005-04-17 05:20:36 +07:00
|
|
|
*
|
2011-05-26 04:59:01 +07:00
|
|
|
* Send command to NAND device. This function is used for small page devices
|
2013-03-05 20:00:51 +07:00
|
|
|
* (512 Bytes per page).
|
2005-04-17 05:20:36 +07:00
|
|
|
*/
|
2006-05-24 04:25:53 +07:00
|
|
|
static void nand_command(struct mtd_info *mtd, unsigned int command,
|
|
|
|
int column, int page_addr)
|
2005-04-17 05:20:36 +07:00
|
|
|
{
|
2006-05-24 17:07:37 +07:00
|
|
|
register struct nand_chip *chip = mtd->priv;
|
2006-05-24 04:25:53 +07:00
|
|
|
int ctrl = NAND_CTRL_CLE | NAND_CTRL_CHANGE;
|
2005-04-17 05:20:36 +07:00
|
|
|
|
2011-05-26 04:59:01 +07:00
|
|
|
/* Write out the command to the device */
|
2005-04-17 05:20:36 +07:00
|
|
|
if (command == NAND_CMD_SEQIN) {
|
|
|
|
int readcmd;
|
|
|
|
|
2006-05-23 04:18:05 +07:00
|
|
|
if (column >= mtd->writesize) {
|
2005-04-17 05:20:36 +07:00
|
|
|
/* OOB area */
|
2006-05-23 04:18:05 +07:00
|
|
|
column -= mtd->writesize;
|
2005-04-17 05:20:36 +07:00
|
|
|
readcmd = NAND_CMD_READOOB;
|
|
|
|
} else if (column < 256) {
|
|
|
|
/* First 256 bytes --> READ0 */
|
|
|
|
readcmd = NAND_CMD_READ0;
|
|
|
|
} else {
|
|
|
|
column -= 256;
|
|
|
|
readcmd = NAND_CMD_READ1;
|
|
|
|
}
|
2006-05-24 17:07:37 +07:00
|
|
|
chip->cmd_ctrl(mtd, readcmd, ctrl);
|
2006-05-24 04:25:53 +07:00
|
|
|
ctrl &= ~NAND_CTRL_CHANGE;
|
2005-04-17 05:20:36 +07:00
|
|
|
}
|
2006-05-24 17:07:37 +07:00
|
|
|
chip->cmd_ctrl(mtd, command, ctrl);
|
2005-04-17 05:20:36 +07:00
|
|
|
|
2011-05-26 04:59:01 +07:00
|
|
|
/* Address cycle, when necessary */
|
2006-05-24 04:25:53 +07:00
|
|
|
ctrl = NAND_CTRL_ALE | NAND_CTRL_CHANGE;
|
|
|
|
/* Serially input address */
|
|
|
|
if (column != -1) {
|
|
|
|
/* Adjust columns for 16 bit buswidth */
|
2006-05-24 17:07:37 +07:00
|
|
|
if (chip->options & NAND_BUSWIDTH_16)
|
2006-05-24 04:25:53 +07:00
|
|
|
column >>= 1;
|
2006-05-24 17:07:37 +07:00
|
|
|
chip->cmd_ctrl(mtd, column, ctrl);
|
2006-05-24 04:25:53 +07:00
|
|
|
ctrl &= ~NAND_CTRL_CHANGE;
|
|
|
|
}
|
|
|
|
if (page_addr != -1) {
|
2006-05-24 17:07:37 +07:00
|
|
|
chip->cmd_ctrl(mtd, page_addr, ctrl);
|
2006-05-24 04:25:53 +07:00
|
|
|
ctrl &= ~NAND_CTRL_CHANGE;
|
2006-05-24 17:07:37 +07:00
|
|
|
chip->cmd_ctrl(mtd, page_addr >> 8, ctrl);
|
2006-05-24 04:25:53 +07:00
|
|
|
/* One more address cycle for devices > 32MiB */
|
2006-05-24 17:07:37 +07:00
|
|
|
if (chip->chipsize > (32 << 20))
|
|
|
|
chip->cmd_ctrl(mtd, page_addr >> 16, ctrl);
|
2005-04-17 05:20:36 +07:00
|
|
|
}
|
2006-05-24 17:07:37 +07:00
|
|
|
chip->cmd_ctrl(mtd, NAND_CMD_NONE, NAND_NCE | NAND_CTRL_CHANGE);
|
2005-11-07 18:15:49 +07:00
|
|
|
|
|
|
|
/*
|
2011-05-26 04:59:01 +07:00
|
|
|
* Program and erase have their own busy handlers status and sequential
|
|
|
|
* in needs no delay
|
2006-05-14 00:07:53 +07:00
|
|
|
*/
|
2005-04-17 05:20:36 +07:00
|
|
|
switch (command) {
|
2005-11-07 18:15:49 +07:00
|
|
|
|
2005-04-17 05:20:36 +07:00
|
|
|
case NAND_CMD_PAGEPROG:
|
|
|
|
case NAND_CMD_ERASE1:
|
|
|
|
case NAND_CMD_ERASE2:
|
|
|
|
case NAND_CMD_SEQIN:
|
|
|
|
case NAND_CMD_STATUS:
|
|
|
|
return;
|
|
|
|
|
|
|
|
case NAND_CMD_RESET:
|
2006-05-24 17:07:37 +07:00
|
|
|
if (chip->dev_ready)
|
2005-04-17 05:20:36 +07:00
|
|
|
break;
|
2006-05-24 17:07:37 +07:00
|
|
|
udelay(chip->chip_delay);
|
|
|
|
chip->cmd_ctrl(mtd, NAND_CMD_STATUS,
|
2006-05-24 04:25:53 +07:00
|
|
|
NAND_CTRL_CLE | NAND_CTRL_CHANGE);
|
2006-05-25 03:57:09 +07:00
|
|
|
chip->cmd_ctrl(mtd,
|
|
|
|
NAND_CMD_NONE, NAND_NCE | NAND_CTRL_CHANGE);
|
2010-09-07 18:23:43 +07:00
|
|
|
while (!(chip->read_byte(mtd) & NAND_STATUS_READY))
|
|
|
|
;
|
2005-04-17 05:20:36 +07:00
|
|
|
return;
|
|
|
|
|
2006-05-14 00:07:53 +07:00
|
|
|
/* This applies to read commands */
|
2005-04-17 05:20:36 +07:00
|
|
|
default:
|
2005-11-07 18:15:49 +07:00
|
|
|
/*
|
2005-04-17 05:20:36 +07:00
|
|
|
* If we don't have access to the busy pin, we apply the given
|
|
|
|
* command delay
|
2006-05-14 00:07:53 +07:00
|
|
|
*/
|
2006-05-24 17:07:37 +07:00
|
|
|
if (!chip->dev_ready) {
|
|
|
|
udelay(chip->chip_delay);
|
2005-04-17 05:20:36 +07:00
|
|
|
return;
|
2005-11-07 18:15:49 +07:00
|
|
|
}
|
2005-04-17 05:20:36 +07:00
|
|
|
}
|
2011-05-26 04:59:01 +07:00
|
|
|
/*
|
|
|
|
* Apply this short delay always to ensure that we do wait tWB in
|
|
|
|
* any case on any machine.
|
|
|
|
*/
|
2006-05-14 00:07:53 +07:00
|
|
|
ndelay(100);
|
2005-02-23 04:56:49 +07:00
|
|
|
|
|
|
|
nand_wait_ready(mtd);
|
2005-04-17 05:20:36 +07:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* nand_command_lp - [DEFAULT] Send command to NAND large page device
|
2011-05-26 04:59:01 +07:00
|
|
|
* @mtd: MTD device structure
|
|
|
|
* @command: the command to be sent
|
|
|
|
* @column: the column address for this command, -1 if none
|
|
|
|
* @page_addr: the page address for this command, -1 if none
|
2005-04-17 05:20:36 +07:00
|
|
|
*
|
2006-05-24 04:25:53 +07:00
|
|
|
* Send command to NAND device. This is the version for the new large page
|
2011-06-24 04:12:08 +07:00
|
|
|
* devices. We don't have the separate regions as we have in the small page
|
|
|
|
* devices. We must emulate NAND_CMD_READOOB to keep the code compatible.
|
2005-04-17 05:20:36 +07:00
|
|
|
*/
|
2006-05-24 04:25:53 +07:00
|
|
|
static void nand_command_lp(struct mtd_info *mtd, unsigned int command,
|
|
|
|
int column, int page_addr)
|
2005-04-17 05:20:36 +07:00
|
|
|
{
|
2006-05-24 17:07:37 +07:00
|
|
|
register struct nand_chip *chip = mtd->priv;
|
2005-04-17 05:20:36 +07:00
|
|
|
|
|
|
|
/* Emulate NAND_CMD_READOOB */
|
|
|
|
if (command == NAND_CMD_READOOB) {
|
2006-05-23 04:18:05 +07:00
|
|
|
column += mtd->writesize;
|
2005-04-17 05:20:36 +07:00
|
|
|
command = NAND_CMD_READ0;
|
|
|
|
}
|
2005-11-07 18:15:49 +07:00
|
|
|
|
2006-05-24 04:25:53 +07:00
|
|
|
/* Command latch cycle */
|
2013-02-28 15:02:19 +07:00
|
|
|
chip->cmd_ctrl(mtd, command, NAND_NCE | NAND_CLE | NAND_CTRL_CHANGE);
|
2005-04-17 05:20:36 +07:00
|
|
|
|
|
|
|
if (column != -1 || page_addr != -1) {
|
2006-05-24 04:25:53 +07:00
|
|
|
int ctrl = NAND_CTRL_CHANGE | NAND_NCE | NAND_ALE;
|
2005-04-17 05:20:36 +07:00
|
|
|
|
|
|
|
/* Serially input address */
|
|
|
|
if (column != -1) {
|
|
|
|
/* Adjust columns for 16 bit buswidth */
|
2006-05-24 17:07:37 +07:00
|
|
|
if (chip->options & NAND_BUSWIDTH_16)
|
2005-04-17 05:20:36 +07:00
|
|
|
column >>= 1;
|
2006-05-24 17:07:37 +07:00
|
|
|
chip->cmd_ctrl(mtd, column, ctrl);
|
2006-05-24 04:25:53 +07:00
|
|
|
ctrl &= ~NAND_CTRL_CHANGE;
|
2006-05-24 17:07:37 +07:00
|
|
|
chip->cmd_ctrl(mtd, column >> 8, ctrl);
|
2005-11-07 18:15:49 +07:00
|
|
|
}
|
2005-04-17 05:20:36 +07:00
|
|
|
if (page_addr != -1) {
|
2006-05-24 17:07:37 +07:00
|
|
|
chip->cmd_ctrl(mtd, page_addr, ctrl);
|
|
|
|
chip->cmd_ctrl(mtd, page_addr >> 8,
|
2006-05-24 04:25:53 +07:00
|
|
|
NAND_NCE | NAND_ALE);
|
2005-04-17 05:20:36 +07:00
|
|
|
/* One more address cycle for devices > 128MiB */
|
2006-05-24 17:07:37 +07:00
|
|
|
if (chip->chipsize > (128 << 20))
|
|
|
|
chip->cmd_ctrl(mtd, page_addr >> 16,
|
2006-05-24 04:25:53 +07:00
|
|
|
NAND_NCE | NAND_ALE);
|
2005-04-17 05:20:36 +07:00
|
|
|
}
|
|
|
|
}
|
2006-05-24 17:07:37 +07:00
|
|
|
chip->cmd_ctrl(mtd, NAND_CMD_NONE, NAND_NCE | NAND_CTRL_CHANGE);
|
2005-11-07 18:15:49 +07:00
|
|
|
|
|
|
|
/*
|
2011-05-26 04:59:01 +07:00
|
|
|
* Program and erase have their own busy handlers status, sequential
|
|
|
|
* in, and deplete1 need no delay.
|
2005-01-18 01:35:25 +07:00
|
|
|
*/
|
2005-04-17 05:20:36 +07:00
|
|
|
switch (command) {
|
2005-11-07 18:15:49 +07:00
|
|
|
|
2005-04-17 05:20:36 +07:00
|
|
|
case NAND_CMD_CACHEDPROG:
|
|
|
|
case NAND_CMD_PAGEPROG:
|
|
|
|
case NAND_CMD_ERASE1:
|
|
|
|
case NAND_CMD_ERASE2:
|
|
|
|
case NAND_CMD_SEQIN:
|
2006-06-21 01:05:05 +07:00
|
|
|
case NAND_CMD_RNDIN:
|
2005-04-17 05:20:36 +07:00
|
|
|
case NAND_CMD_STATUS:
|
2005-01-18 01:35:25 +07:00
|
|
|
return;
|
2005-04-17 05:20:36 +07:00
|
|
|
|
|
|
|
case NAND_CMD_RESET:
|
2006-05-24 17:07:37 +07:00
|
|
|
if (chip->dev_ready)
|
2005-04-17 05:20:36 +07:00
|
|
|
break;
|
2006-05-24 17:07:37 +07:00
|
|
|
udelay(chip->chip_delay);
|
2006-05-25 03:57:09 +07:00
|
|
|
chip->cmd_ctrl(mtd, NAND_CMD_STATUS,
|
|
|
|
NAND_NCE | NAND_CLE | NAND_CTRL_CHANGE);
|
|
|
|
chip->cmd_ctrl(mtd, NAND_CMD_NONE,
|
|
|
|
NAND_NCE | NAND_CTRL_CHANGE);
|
2010-09-07 18:23:43 +07:00
|
|
|
while (!(chip->read_byte(mtd) & NAND_STATUS_READY))
|
|
|
|
;
|
2005-04-17 05:20:36 +07:00
|
|
|
return;
|
|
|
|
|
2006-06-21 01:05:05 +07:00
|
|
|
case NAND_CMD_RNDOUT:
|
|
|
|
/* No ready / busy check necessary */
|
|
|
|
chip->cmd_ctrl(mtd, NAND_CMD_RNDOUTSTART,
|
|
|
|
NAND_NCE | NAND_CLE | NAND_CTRL_CHANGE);
|
|
|
|
chip->cmd_ctrl(mtd, NAND_CMD_NONE,
|
|
|
|
NAND_NCE | NAND_CTRL_CHANGE);
|
|
|
|
return;
|
|
|
|
|
2005-04-17 05:20:36 +07:00
|
|
|
case NAND_CMD_READ0:
|
2006-05-25 03:57:09 +07:00
|
|
|
chip->cmd_ctrl(mtd, NAND_CMD_READSTART,
|
|
|
|
NAND_NCE | NAND_CLE | NAND_CTRL_CHANGE);
|
|
|
|
chip->cmd_ctrl(mtd, NAND_CMD_NONE,
|
|
|
|
NAND_NCE | NAND_CTRL_CHANGE);
|
2005-11-07 18:15:49 +07:00
|
|
|
|
2006-05-14 00:07:53 +07:00
|
|
|
/* This applies to read commands */
|
2005-04-17 05:20:36 +07:00
|
|
|
default:
|
2005-11-07 18:15:49 +07:00
|
|
|
/*
|
2005-04-17 05:20:36 +07:00
|
|
|
* If we don't have access to the busy pin, we apply the given
|
2011-05-26 04:59:01 +07:00
|
|
|
* command delay.
|
2006-05-14 00:07:53 +07:00
|
|
|
*/
|
2006-05-24 17:07:37 +07:00
|
|
|
if (!chip->dev_ready) {
|
|
|
|
udelay(chip->chip_delay);
|
2005-04-17 05:20:36 +07:00
|
|
|
return;
|
2005-11-07 18:15:49 +07:00
|
|
|
}
|
2005-04-17 05:20:36 +07:00
|
|
|
}
|
2005-02-23 04:56:49 +07:00
|
|
|
|
2011-05-26 04:59:01 +07:00
|
|
|
/*
|
|
|
|
* Apply this short delay always to ensure that we do wait tWB in
|
|
|
|
* any case on any machine.
|
|
|
|
*/
|
2006-05-14 00:07:53 +07:00
|
|
|
ndelay(100);
|
2005-02-23 04:56:49 +07:00
|
|
|
|
|
|
|
nand_wait_ready(mtd);
|
2005-04-17 05:20:36 +07:00
|
|
|
}
|
|
|
|
|
2009-10-05 20:55:52 +07:00
|
|
|
/**
|
|
|
|
* panic_nand_get_device - [GENERIC] Get chip for selected access
|
2011-05-26 04:59:01 +07:00
|
|
|
* @chip: the nand chip descriptor
|
|
|
|
* @mtd: MTD device structure
|
|
|
|
* @new_state: the state which is requested
|
2009-10-05 20:55:52 +07:00
|
|
|
*
|
|
|
|
* Used when in panic, no locks are taken.
|
|
|
|
*/
|
|
|
|
static void panic_nand_get_device(struct nand_chip *chip,
|
|
|
|
struct mtd_info *mtd, int new_state)
|
|
|
|
{
|
2011-06-24 04:12:08 +07:00
|
|
|
/* Hardware controller shared among independent devices */
|
2009-10-05 20:55:52 +07:00
|
|
|
chip->controller->active = chip;
|
|
|
|
chip->state = new_state;
|
|
|
|
}
|
|
|
|
|
2005-04-17 05:20:36 +07:00
|
|
|
/**
|
|
|
|
* nand_get_device - [GENERIC] Get chip for selected access
|
2011-05-26 04:59:01 +07:00
|
|
|
* @mtd: MTD device structure
|
|
|
|
* @new_state: the state which is requested
|
2005-04-17 05:20:36 +07:00
|
|
|
*
|
|
|
|
* Get the device and lock it for exclusive access
|
|
|
|
*/
|
2006-05-23 16:50:56 +07:00
|
|
|
static int
|
2012-11-19 13:43:30 +07:00
|
|
|
nand_get_device(struct mtd_info *mtd, int new_state)
|
2005-04-17 05:20:36 +07:00
|
|
|
{
|
2012-11-19 13:43:30 +07:00
|
|
|
struct nand_chip *chip = mtd->priv;
|
2006-05-24 17:07:37 +07:00
|
|
|
spinlock_t *lock = &chip->controller->lock;
|
|
|
|
wait_queue_head_t *wq = &chip->controller->wq;
|
2006-05-14 00:07:53 +07:00
|
|
|
DECLARE_WAITQUEUE(wait, current);
|
2010-09-07 18:23:45 +07:00
|
|
|
retry:
|
2005-06-01 02:39:20 +07:00
|
|
|
spin_lock(lock);
|
|
|
|
|
2009-07-09 22:11:22 +07:00
|
|
|
/* Hardware controller shared among independent devices */
|
2006-05-24 17:07:37 +07:00
|
|
|
if (!chip->controller->active)
|
|
|
|
chip->controller->active = chip;
|
2006-05-23 16:37:03 +07:00
|
|
|
|
2006-05-24 17:07:37 +07:00
|
|
|
if (chip->controller->active == chip && chip->state == FL_READY) {
|
|
|
|
chip->state = new_state;
|
2005-06-01 02:39:20 +07:00
|
|
|
spin_unlock(lock);
|
2005-09-15 20:58:53 +07:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
if (new_state == FL_PM_SUSPENDED) {
|
2009-11-18 05:45:49 +07:00
|
|
|
if (chip->controller->active->state == FL_PM_SUSPENDED) {
|
|
|
|
chip->state = FL_PM_SUSPENDED;
|
|
|
|
spin_unlock(lock);
|
|
|
|
return 0;
|
|
|
|
}
|
2005-06-01 02:39:20 +07:00
|
|
|
}
|
|
|
|
set_current_state(TASK_UNINTERRUPTIBLE);
|
|
|
|
add_wait_queue(wq, &wait);
|
|
|
|
spin_unlock(lock);
|
|
|
|
schedule();
|
|
|
|
remove_wait_queue(wq, &wait);
|
2005-04-17 05:20:36 +07:00
|
|
|
goto retry;
|
|
|
|
}
|
|
|
|
|
2009-10-05 20:55:52 +07:00
|
|
|
/**
|
2011-05-26 04:59:01 +07:00
|
|
|
* panic_nand_wait - [GENERIC] wait until the command is done
|
|
|
|
* @mtd: MTD device structure
|
|
|
|
* @chip: NAND chip structure
|
|
|
|
* @timeo: timeout
|
2009-10-05 20:55:52 +07:00
|
|
|
*
|
|
|
|
* Wait for command done. This is a helper function for nand_wait used when
|
|
|
|
* we are in interrupt context. May happen when in panic and trying to write
|
tree-wide: fix comment/printk typos
"gadget", "through", "command", "maintain", "maintain", "controller", "address",
"between", "initiali[zs]e", "instead", "function", "select", "already",
"equal", "access", "management", "hierarchy", "registration", "interest",
"relative", "memory", "offset", "already",
Signed-off-by: Uwe Kleine-König <u.kleine-koenig@pengutronix.de>
Signed-off-by: Jiri Kosina <jkosina@suse.cz>
2010-11-02 02:38:34 +07:00
|
|
|
* an oops through mtdoops.
|
2009-10-05 20:55:52 +07:00
|
|
|
*/
|
|
|
|
static void panic_nand_wait(struct mtd_info *mtd, struct nand_chip *chip,
|
|
|
|
unsigned long timeo)
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
for (i = 0; i < timeo; i++) {
|
|
|
|
if (chip->dev_ready) {
|
|
|
|
if (chip->dev_ready(mtd))
|
|
|
|
break;
|
|
|
|
} else {
|
|
|
|
if (chip->read_byte(mtd) & NAND_STATUS_READY)
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
mdelay(1);
|
2010-09-07 18:23:43 +07:00
|
|
|
}
|
2009-10-05 20:55:52 +07:00
|
|
|
}
|
|
|
|
|
2005-04-17 05:20:36 +07:00
|
|
|
/**
|
2011-05-26 04:59:01 +07:00
|
|
|
* nand_wait - [DEFAULT] wait until the command is done
|
|
|
|
* @mtd: MTD device structure
|
|
|
|
* @chip: NAND chip structure
|
2005-04-17 05:20:36 +07:00
|
|
|
*
|
2011-05-26 04:59:01 +07:00
|
|
|
* Wait for command done. This applies to erase and program only. Erase can
|
|
|
|
* take up to 400ms and program up to 20ms according to general NAND and
|
|
|
|
* SmartMedia specs.
|
2006-06-29 11:48:27 +07:00
|
|
|
*/
|
2006-06-21 01:05:05 +07:00
|
|
|
static int nand_wait(struct mtd_info *mtd, struct nand_chip *chip)
|
2005-04-17 05:20:36 +07:00
|
|
|
{
|
|
|
|
|
2006-06-21 01:05:05 +07:00
|
|
|
int status, state = chip->state;
|
2013-01-30 09:03:56 +07:00
|
|
|
unsigned long timeo = (state == FL_ERASING ? 400 : 20);
|
2005-04-17 05:20:36 +07:00
|
|
|
|
2006-03-31 17:31:14 +07:00
|
|
|
led_trigger_event(nand_led_trigger, LED_FULL);
|
|
|
|
|
2011-05-26 04:59:01 +07:00
|
|
|
/*
|
|
|
|
* Apply this short delay always to ensure that we do wait tWB in any
|
|
|
|
* case on any machine.
|
|
|
|
*/
|
2006-05-14 00:07:53 +07:00
|
|
|
ndelay(100);
|
2005-04-17 05:20:36 +07:00
|
|
|
|
2013-03-04 19:21:34 +07:00
|
|
|
chip->cmdfunc(mtd, NAND_CMD_STATUS, -1, -1);
|
2005-04-17 05:20:36 +07:00
|
|
|
|
2009-10-05 20:55:52 +07:00
|
|
|
if (in_interrupt() || oops_in_progress)
|
|
|
|
panic_nand_wait(mtd, chip, timeo);
|
|
|
|
else {
|
2013-01-30 09:03:56 +07:00
|
|
|
timeo = jiffies + msecs_to_jiffies(timeo);
|
2009-10-05 20:55:52 +07:00
|
|
|
while (time_before(jiffies, timeo)) {
|
|
|
|
if (chip->dev_ready) {
|
|
|
|
if (chip->dev_ready(mtd))
|
|
|
|
break;
|
|
|
|
} else {
|
|
|
|
if (chip->read_byte(mtd) & NAND_STATUS_READY)
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
cond_resched();
|
2005-04-17 05:20:36 +07:00
|
|
|
}
|
|
|
|
}
|
2006-03-31 17:31:14 +07:00
|
|
|
led_trigger_event(nand_led_trigger, LED_OFF);
|
|
|
|
|
2006-05-24 17:07:37 +07:00
|
|
|
status = (int)chip->read_byte(mtd);
|
2012-11-05 21:00:44 +07:00
|
|
|
/* This can happen if in case of timeout or buggy dev_ready */
|
|
|
|
WARN_ON(!(status & NAND_STATUS_READY));
|
2005-04-17 05:20:36 +07:00
|
|
|
return status;
|
|
|
|
}
|
|
|
|
|
2010-02-08 17:20:49 +07:00
|
|
|
/**
|
2010-08-11 08:02:50 +07:00
|
|
|
* __nand_unlock - [REPLACEABLE] unlocks specified locked blocks
|
|
|
|
* @mtd: mtd info
|
|
|
|
* @ofs: offset to start unlock from
|
|
|
|
* @len: length to unlock
|
2011-05-26 04:59:01 +07:00
|
|
|
* @invert: when = 0, unlock the range of blocks within the lower and
|
|
|
|
* upper boundary address
|
|
|
|
* when = 1, unlock the range of blocks outside the boundaries
|
|
|
|
* of the lower and upper boundary address
|
2010-02-08 17:20:49 +07:00
|
|
|
*
|
2011-05-26 04:59:01 +07:00
|
|
|
* Returs unlock status.
|
2010-02-08 17:20:49 +07:00
|
|
|
*/
|
|
|
|
static int __nand_unlock(struct mtd_info *mtd, loff_t ofs,
|
|
|
|
uint64_t len, int invert)
|
|
|
|
{
|
|
|
|
int ret = 0;
|
|
|
|
int status, page;
|
|
|
|
struct nand_chip *chip = mtd->priv;
|
|
|
|
|
|
|
|
/* Submit address of first page to unlock */
|
|
|
|
page = ofs >> chip->page_shift;
|
|
|
|
chip->cmdfunc(mtd, NAND_CMD_UNLOCK1, -1, page & chip->pagemask);
|
|
|
|
|
|
|
|
/* Submit address of last page to unlock */
|
|
|
|
page = (ofs + len) >> chip->page_shift;
|
|
|
|
chip->cmdfunc(mtd, NAND_CMD_UNLOCK2, -1,
|
|
|
|
(page | invert) & chip->pagemask);
|
|
|
|
|
|
|
|
/* Call wait ready function */
|
|
|
|
status = chip->waitfunc(mtd, chip);
|
|
|
|
/* See if device thinks it succeeded */
|
2012-10-15 10:47:24 +07:00
|
|
|
if (status & NAND_STATUS_FAIL) {
|
2011-07-20 00:06:09 +07:00
|
|
|
pr_debug("%s: error status = 0x%08x\n",
|
2010-02-08 17:20:49 +07:00
|
|
|
__func__, status);
|
|
|
|
ret = -EIO;
|
|
|
|
}
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2010-08-11 08:02:50 +07:00
|
|
|
* nand_unlock - [REPLACEABLE] unlocks specified locked blocks
|
|
|
|
* @mtd: mtd info
|
|
|
|
* @ofs: offset to start unlock from
|
|
|
|
* @len: length to unlock
|
2010-02-08 17:20:49 +07:00
|
|
|
*
|
2011-05-26 04:59:01 +07:00
|
|
|
* Returns unlock status.
|
2010-02-08 17:20:49 +07:00
|
|
|
*/
|
|
|
|
int nand_unlock(struct mtd_info *mtd, loff_t ofs, uint64_t len)
|
|
|
|
{
|
|
|
|
int ret = 0;
|
|
|
|
int chipnr;
|
|
|
|
struct nand_chip *chip = mtd->priv;
|
|
|
|
|
2011-07-20 00:06:09 +07:00
|
|
|
pr_debug("%s: start = 0x%012llx, len = %llu\n",
|
2010-02-08 17:20:49 +07:00
|
|
|
__func__, (unsigned long long)ofs, len);
|
|
|
|
|
|
|
|
if (check_offs_len(mtd, ofs, len))
|
|
|
|
ret = -EINVAL;
|
|
|
|
|
|
|
|
/* Align to last block address if size addresses end of the device */
|
|
|
|
if (ofs + len == mtd->size)
|
|
|
|
len -= mtd->erasesize;
|
|
|
|
|
2012-11-19 13:43:30 +07:00
|
|
|
nand_get_device(mtd, FL_UNLOCKING);
|
2010-02-08 17:20:49 +07:00
|
|
|
|
|
|
|
/* Shift to get chip number */
|
|
|
|
chipnr = ofs >> chip->chip_shift;
|
|
|
|
|
|
|
|
chip->select_chip(mtd, chipnr);
|
|
|
|
|
|
|
|
/* Check, if it is write protected */
|
|
|
|
if (nand_check_wp(mtd)) {
|
2011-07-20 00:06:09 +07:00
|
|
|
pr_debug("%s: device is write protected!\n",
|
2010-02-08 17:20:49 +07:00
|
|
|
__func__);
|
|
|
|
ret = -EIO;
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
|
|
|
ret = __nand_unlock(mtd, ofs, len, 0);
|
|
|
|
|
|
|
|
out:
|
2012-11-19 13:43:29 +07:00
|
|
|
chip->select_chip(mtd, -1);
|
2010-02-08 17:20:49 +07:00
|
|
|
nand_release_device(mtd);
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
2010-09-07 18:23:45 +07:00
|
|
|
EXPORT_SYMBOL(nand_unlock);
|
2010-02-08 17:20:49 +07:00
|
|
|
|
|
|
|
/**
|
2010-08-11 08:02:50 +07:00
|
|
|
* nand_lock - [REPLACEABLE] locks all blocks present in the device
|
|
|
|
* @mtd: mtd info
|
|
|
|
* @ofs: offset to start unlock from
|
|
|
|
* @len: length to unlock
|
2010-02-08 17:20:49 +07:00
|
|
|
*
|
2011-05-26 04:59:01 +07:00
|
|
|
* This feature is not supported in many NAND parts. 'Micron' NAND parts do
|
|
|
|
* have this feature, but it allows only to lock all blocks, not for specified
|
|
|
|
* range for block. Implementing 'lock' feature by making use of 'unlock', for
|
|
|
|
* now.
|
2010-02-08 17:20:49 +07:00
|
|
|
*
|
2011-05-26 04:59:01 +07:00
|
|
|
* Returns lock status.
|
2010-02-08 17:20:49 +07:00
|
|
|
*/
|
|
|
|
int nand_lock(struct mtd_info *mtd, loff_t ofs, uint64_t len)
|
|
|
|
{
|
|
|
|
int ret = 0;
|
|
|
|
int chipnr, status, page;
|
|
|
|
struct nand_chip *chip = mtd->priv;
|
|
|
|
|
2011-07-20 00:06:09 +07:00
|
|
|
pr_debug("%s: start = 0x%012llx, len = %llu\n",
|
2010-02-08 17:20:49 +07:00
|
|
|
__func__, (unsigned long long)ofs, len);
|
|
|
|
|
|
|
|
if (check_offs_len(mtd, ofs, len))
|
|
|
|
ret = -EINVAL;
|
|
|
|
|
2012-11-19 13:43:30 +07:00
|
|
|
nand_get_device(mtd, FL_LOCKING);
|
2010-02-08 17:20:49 +07:00
|
|
|
|
|
|
|
/* Shift to get chip number */
|
|
|
|
chipnr = ofs >> chip->chip_shift;
|
|
|
|
|
|
|
|
chip->select_chip(mtd, chipnr);
|
|
|
|
|
|
|
|
/* Check, if it is write protected */
|
|
|
|
if (nand_check_wp(mtd)) {
|
2011-07-20 00:06:09 +07:00
|
|
|
pr_debug("%s: device is write protected!\n",
|
2010-02-08 17:20:49 +07:00
|
|
|
__func__);
|
|
|
|
status = MTD_ERASE_FAILED;
|
|
|
|
ret = -EIO;
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Submit address of first page to lock */
|
|
|
|
page = ofs >> chip->page_shift;
|
|
|
|
chip->cmdfunc(mtd, NAND_CMD_LOCK, -1, page & chip->pagemask);
|
|
|
|
|
|
|
|
/* Call wait ready function */
|
|
|
|
status = chip->waitfunc(mtd, chip);
|
|
|
|
/* See if device thinks it succeeded */
|
2012-10-15 10:47:24 +07:00
|
|
|
if (status & NAND_STATUS_FAIL) {
|
2011-07-20 00:06:09 +07:00
|
|
|
pr_debug("%s: error status = 0x%08x\n",
|
2010-02-08 17:20:49 +07:00
|
|
|
__func__, status);
|
|
|
|
ret = -EIO;
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
|
|
|
ret = __nand_unlock(mtd, ofs, len, 0x1);
|
|
|
|
|
|
|
|
out:
|
2012-11-19 13:43:29 +07:00
|
|
|
chip->select_chip(mtd, -1);
|
2010-02-08 17:20:49 +07:00
|
|
|
nand_release_device(mtd);
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
2010-09-07 18:23:45 +07:00
|
|
|
EXPORT_SYMBOL(nand_lock);
|
2010-02-08 17:20:49 +07:00
|
|
|
|
2006-05-29 08:26:58 +07:00
|
|
|
/**
|
2011-06-24 04:12:08 +07:00
|
|
|
* nand_read_page_raw - [INTERN] read raw page data without ecc
|
2011-05-26 04:59:01 +07:00
|
|
|
* @mtd: mtd info structure
|
|
|
|
* @chip: nand chip info structure
|
|
|
|
* @buf: buffer to store read data
|
2012-05-03 00:14:55 +07:00
|
|
|
* @oob_required: caller requires OOB data read to chip->oob_poi
|
2011-05-26 04:59:01 +07:00
|
|
|
* @page: page number to read
|
[MTD] [NAND] fix "raw" reads with ECC syndrome layouts
The syndrome based page read/write routines store ECC, and possibly other
"OOB" data, right after each chunk of ECC'd data. With ECC chunk size of
512 bytes and a large page (2KiB) NAND, the layout is:
data-0 OOB-0 data-1 OOB-1 data-2 OOB-2 data-3 OOB-3 OOB-leftover
Where OOBx is (prepad, ECC, postpad). However, the current "raw" routines
use a traditional layout -- data OOB, disregarding the prepad and postpad
values -- so when they're used with that type of ECC hardware, those calls
mix up the data and OOB. Which means, in particular, that bad block
tables won't be found on startup, with data corruption and related chaos
ensuing.
The current syndrome-based drivers in mainline all seem to use one chunk
per page; presumably they haven't noticed such bugs.
Fix this, by adding read/write page_raw_syndrome() routines as siblings of
the existing non-raw routines; "raw" just means to bypass the ECC
computations, not change data and OOB layout.
Signed-off-by: David Brownell <dbrownell@users.sourceforge.net>
Signed-off-by: Andrew Morton <akpm@linux-foundation.org>
Signed-off-by: David Woodhouse <David.Woodhouse@intel.com>
2009-03-05 03:01:36 +07:00
|
|
|
*
|
2011-06-24 04:12:08 +07:00
|
|
|
* Not for syndrome calculating ECC controllers, which use a special oob layout.
|
2006-05-29 08:26:58 +07:00
|
|
|
*/
|
|
|
|
static int nand_read_page_raw(struct mtd_info *mtd, struct nand_chip *chip,
|
2012-05-03 00:14:55 +07:00
|
|
|
uint8_t *buf, int oob_required, int page)
|
2006-05-29 08:26:58 +07:00
|
|
|
{
|
|
|
|
chip->read_buf(mtd, buf, mtd->writesize);
|
2012-05-03 00:15:03 +07:00
|
|
|
if (oob_required)
|
|
|
|
chip->read_buf(mtd, chip->oob_poi, mtd->oobsize);
|
2006-05-29 08:26:58 +07:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
[MTD] [NAND] fix "raw" reads with ECC syndrome layouts
The syndrome based page read/write routines store ECC, and possibly other
"OOB" data, right after each chunk of ECC'd data. With ECC chunk size of
512 bytes and a large page (2KiB) NAND, the layout is:
data-0 OOB-0 data-1 OOB-1 data-2 OOB-2 data-3 OOB-3 OOB-leftover
Where OOBx is (prepad, ECC, postpad). However, the current "raw" routines
use a traditional layout -- data OOB, disregarding the prepad and postpad
values -- so when they're used with that type of ECC hardware, those calls
mix up the data and OOB. Which means, in particular, that bad block
tables won't be found on startup, with data corruption and related chaos
ensuing.
The current syndrome-based drivers in mainline all seem to use one chunk
per page; presumably they haven't noticed such bugs.
Fix this, by adding read/write page_raw_syndrome() routines as siblings of
the existing non-raw routines; "raw" just means to bypass the ECC
computations, not change data and OOB layout.
Signed-off-by: David Brownell <dbrownell@users.sourceforge.net>
Signed-off-by: Andrew Morton <akpm@linux-foundation.org>
Signed-off-by: David Woodhouse <David.Woodhouse@intel.com>
2009-03-05 03:01:36 +07:00
|
|
|
/**
|
2011-06-24 04:12:08 +07:00
|
|
|
* nand_read_page_raw_syndrome - [INTERN] read raw page data without ecc
|
2011-05-26 04:59:01 +07:00
|
|
|
* @mtd: mtd info structure
|
|
|
|
* @chip: nand chip info structure
|
|
|
|
* @buf: buffer to store read data
|
2012-05-03 00:14:55 +07:00
|
|
|
* @oob_required: caller requires OOB data read to chip->oob_poi
|
2011-05-26 04:59:01 +07:00
|
|
|
* @page: page number to read
|
[MTD] [NAND] fix "raw" reads with ECC syndrome layouts
The syndrome based page read/write routines store ECC, and possibly other
"OOB" data, right after each chunk of ECC'd data. With ECC chunk size of
512 bytes and a large page (2KiB) NAND, the layout is:
data-0 OOB-0 data-1 OOB-1 data-2 OOB-2 data-3 OOB-3 OOB-leftover
Where OOBx is (prepad, ECC, postpad). However, the current "raw" routines
use a traditional layout -- data OOB, disregarding the prepad and postpad
values -- so when they're used with that type of ECC hardware, those calls
mix up the data and OOB. Which means, in particular, that bad block
tables won't be found on startup, with data corruption and related chaos
ensuing.
The current syndrome-based drivers in mainline all seem to use one chunk
per page; presumably they haven't noticed such bugs.
Fix this, by adding read/write page_raw_syndrome() routines as siblings of
the existing non-raw routines; "raw" just means to bypass the ECC
computations, not change data and OOB layout.
Signed-off-by: David Brownell <dbrownell@users.sourceforge.net>
Signed-off-by: Andrew Morton <akpm@linux-foundation.org>
Signed-off-by: David Woodhouse <David.Woodhouse@intel.com>
2009-03-05 03:01:36 +07:00
|
|
|
*
|
|
|
|
* We need a special oob layout and handling even when OOB isn't used.
|
|
|
|
*/
|
2010-09-07 18:23:45 +07:00
|
|
|
static int nand_read_page_raw_syndrome(struct mtd_info *mtd,
|
2012-05-03 00:14:55 +07:00
|
|
|
struct nand_chip *chip, uint8_t *buf,
|
|
|
|
int oob_required, int page)
|
[MTD] [NAND] fix "raw" reads with ECC syndrome layouts
The syndrome based page read/write routines store ECC, and possibly other
"OOB" data, right after each chunk of ECC'd data. With ECC chunk size of
512 bytes and a large page (2KiB) NAND, the layout is:
data-0 OOB-0 data-1 OOB-1 data-2 OOB-2 data-3 OOB-3 OOB-leftover
Where OOBx is (prepad, ECC, postpad). However, the current "raw" routines
use a traditional layout -- data OOB, disregarding the prepad and postpad
values -- so when they're used with that type of ECC hardware, those calls
mix up the data and OOB. Which means, in particular, that bad block
tables won't be found on startup, with data corruption and related chaos
ensuing.
The current syndrome-based drivers in mainline all seem to use one chunk
per page; presumably they haven't noticed such bugs.
Fix this, by adding read/write page_raw_syndrome() routines as siblings of
the existing non-raw routines; "raw" just means to bypass the ECC
computations, not change data and OOB layout.
Signed-off-by: David Brownell <dbrownell@users.sourceforge.net>
Signed-off-by: Andrew Morton <akpm@linux-foundation.org>
Signed-off-by: David Woodhouse <David.Woodhouse@intel.com>
2009-03-05 03:01:36 +07:00
|
|
|
{
|
|
|
|
int eccsize = chip->ecc.size;
|
|
|
|
int eccbytes = chip->ecc.bytes;
|
|
|
|
uint8_t *oob = chip->oob_poi;
|
|
|
|
int steps, size;
|
|
|
|
|
|
|
|
for (steps = chip->ecc.steps; steps > 0; steps--) {
|
|
|
|
chip->read_buf(mtd, buf, eccsize);
|
|
|
|
buf += eccsize;
|
|
|
|
|
|
|
|
if (chip->ecc.prepad) {
|
|
|
|
chip->read_buf(mtd, oob, chip->ecc.prepad);
|
|
|
|
oob += chip->ecc.prepad;
|
|
|
|
}
|
|
|
|
|
|
|
|
chip->read_buf(mtd, oob, eccbytes);
|
|
|
|
oob += eccbytes;
|
|
|
|
|
|
|
|
if (chip->ecc.postpad) {
|
|
|
|
chip->read_buf(mtd, oob, chip->ecc.postpad);
|
|
|
|
oob += chip->ecc.postpad;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
size = mtd->oobsize - (oob - chip->oob_poi);
|
|
|
|
if (size)
|
|
|
|
chip->read_buf(mtd, oob, size);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2005-04-17 05:20:36 +07:00
|
|
|
/**
|
2011-06-24 04:12:08 +07:00
|
|
|
* nand_read_page_swecc - [REPLACEABLE] software ECC based page read function
|
2011-05-26 04:59:01 +07:00
|
|
|
* @mtd: mtd info structure
|
|
|
|
* @chip: nand chip info structure
|
|
|
|
* @buf: buffer to store read data
|
2012-05-03 00:14:55 +07:00
|
|
|
* @oob_required: caller requires OOB data read to chip->oob_poi
|
2011-05-26 04:59:01 +07:00
|
|
|
* @page: page number to read
|
2005-01-24 10:07:46 +07:00
|
|
|
*/
|
2006-05-25 15:07:16 +07:00
|
|
|
static int nand_read_page_swecc(struct mtd_info *mtd, struct nand_chip *chip,
|
2012-05-03 00:14:55 +07:00
|
|
|
uint8_t *buf, int oob_required, int page)
|
2005-04-17 05:20:36 +07:00
|
|
|
{
|
2006-05-25 15:07:16 +07:00
|
|
|
int i, eccsize = chip->ecc.size;
|
|
|
|
int eccbytes = chip->ecc.bytes;
|
|
|
|
int eccsteps = chip->ecc.steps;
|
|
|
|
uint8_t *p = buf;
|
2006-09-25 23:08:04 +07:00
|
|
|
uint8_t *ecc_calc = chip->buffers->ecccalc;
|
|
|
|
uint8_t *ecc_code = chip->buffers->ecccode;
|
2007-05-29 01:17:54 +07:00
|
|
|
uint32_t *eccpos = chip->ecc.layout->eccpos;
|
2012-04-26 02:06:09 +07:00
|
|
|
unsigned int max_bitflips = 0;
|
2006-05-25 15:07:16 +07:00
|
|
|
|
2012-05-03 00:14:55 +07:00
|
|
|
chip->ecc.read_page_raw(mtd, chip, buf, 1, page);
|
2006-05-25 15:07:16 +07:00
|
|
|
|
|
|
|
for (i = 0; eccsteps; eccsteps--, i += eccbytes, p += eccsize)
|
|
|
|
chip->ecc.calculate(mtd, p, &ecc_calc[i]);
|
|
|
|
|
|
|
|
for (i = 0; i < chip->ecc.total; i++)
|
2006-05-26 23:52:08 +07:00
|
|
|
ecc_code[i] = chip->oob_poi[eccpos[i]];
|
2006-05-25 15:07:16 +07:00
|
|
|
|
|
|
|
eccsteps = chip->ecc.steps;
|
|
|
|
p = buf;
|
|
|
|
|
|
|
|
for (i = 0 ; eccsteps; eccsteps--, i += eccbytes, p += eccsize) {
|
|
|
|
int stat;
|
|
|
|
|
|
|
|
stat = chip->ecc.correct(mtd, p, &ecc_code[i], &ecc_calc[i]);
|
2012-04-26 02:06:09 +07:00
|
|
|
if (stat < 0) {
|
2006-05-25 15:07:16 +07:00
|
|
|
mtd->ecc_stats.failed++;
|
2012-04-26 02:06:09 +07:00
|
|
|
} else {
|
2006-05-25 15:07:16 +07:00
|
|
|
mtd->ecc_stats.corrected += stat;
|
2012-04-26 02:06:09 +07:00
|
|
|
max_bitflips = max_t(unsigned int, max_bitflips, stat);
|
|
|
|
}
|
2006-05-25 15:07:16 +07:00
|
|
|
}
|
2012-04-26 02:06:09 +07:00
|
|
|
return max_bitflips;
|
2005-04-05 01:56:32 +07:00
|
|
|
}
|
2005-04-17 05:20:36 +07:00
|
|
|
|
2008-05-15 23:23:18 +07:00
|
|
|
/**
|
2013-03-15 19:25:53 +07:00
|
|
|
* nand_read_subpage - [REPLACEABLE] ECC based sub-page read function
|
2011-05-26 04:59:01 +07:00
|
|
|
* @mtd: mtd info structure
|
|
|
|
* @chip: nand chip info structure
|
|
|
|
* @data_offs: offset of requested data within the page
|
|
|
|
* @readlen: data length
|
|
|
|
* @bufpoi: buffer to store read data
|
2008-05-15 23:23:18 +07:00
|
|
|
*/
|
2010-09-07 18:23:45 +07:00
|
|
|
static int nand_read_subpage(struct mtd_info *mtd, struct nand_chip *chip,
|
|
|
|
uint32_t data_offs, uint32_t readlen, uint8_t *bufpoi)
|
2008-05-15 23:23:18 +07:00
|
|
|
{
|
|
|
|
int start_step, end_step, num_steps;
|
|
|
|
uint32_t *eccpos = chip->ecc.layout->eccpos;
|
|
|
|
uint8_t *p;
|
|
|
|
int data_col_addr, i, gaps = 0;
|
|
|
|
int datafrag_len, eccfrag_len, aligned_len, aligned_pos;
|
|
|
|
int busw = (chip->options & NAND_BUSWIDTH_16) ? 2 : 1;
|
2010-09-07 18:23:45 +07:00
|
|
|
int index = 0;
|
2012-04-26 02:06:09 +07:00
|
|
|
unsigned int max_bitflips = 0;
|
2008-05-15 23:23:18 +07:00
|
|
|
|
2011-06-24 04:12:08 +07:00
|
|
|
/* Column address within the page aligned to ECC size (256bytes) */
|
2008-05-15 23:23:18 +07:00
|
|
|
start_step = data_offs / chip->ecc.size;
|
|
|
|
end_step = (data_offs + readlen - 1) / chip->ecc.size;
|
|
|
|
num_steps = end_step - start_step + 1;
|
|
|
|
|
2011-05-26 04:59:01 +07:00
|
|
|
/* Data size aligned to ECC ecc.size */
|
2008-05-15 23:23:18 +07:00
|
|
|
datafrag_len = num_steps * chip->ecc.size;
|
|
|
|
eccfrag_len = num_steps * chip->ecc.bytes;
|
|
|
|
|
|
|
|
data_col_addr = start_step * chip->ecc.size;
|
|
|
|
/* If we read not a page aligned data */
|
|
|
|
if (data_col_addr != 0)
|
|
|
|
chip->cmdfunc(mtd, NAND_CMD_RNDOUT, data_col_addr, -1);
|
|
|
|
|
|
|
|
p = bufpoi + data_col_addr;
|
|
|
|
chip->read_buf(mtd, p, datafrag_len);
|
|
|
|
|
2011-05-26 04:59:01 +07:00
|
|
|
/* Calculate ECC */
|
2008-05-15 23:23:18 +07:00
|
|
|
for (i = 0; i < eccfrag_len ; i += chip->ecc.bytes, p += chip->ecc.size)
|
|
|
|
chip->ecc.calculate(mtd, p, &chip->buffers->ecccalc[i]);
|
|
|
|
|
2011-05-26 04:59:01 +07:00
|
|
|
/*
|
|
|
|
* The performance is faster if we position offsets according to
|
2011-06-24 04:12:08 +07:00
|
|
|
* ecc.pos. Let's make sure that there are no gaps in ECC positions.
|
2011-05-26 04:59:01 +07:00
|
|
|
*/
|
2008-05-15 23:23:18 +07:00
|
|
|
for (i = 0; i < eccfrag_len - 1; i++) {
|
|
|
|
if (eccpos[i + start_step * chip->ecc.bytes] + 1 !=
|
|
|
|
eccpos[i + start_step * chip->ecc.bytes + 1]) {
|
|
|
|
gaps = 1;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (gaps) {
|
|
|
|
chip->cmdfunc(mtd, NAND_CMD_RNDOUT, mtd->writesize, -1);
|
|
|
|
chip->read_buf(mtd, chip->oob_poi, mtd->oobsize);
|
|
|
|
} else {
|
2011-05-26 04:59:01 +07:00
|
|
|
/*
|
2011-06-24 04:12:08 +07:00
|
|
|
* Send the command to read the particular ECC bytes take care
|
2011-05-26 04:59:01 +07:00
|
|
|
* about buswidth alignment in read_buf.
|
|
|
|
*/
|
2010-09-07 18:23:45 +07:00
|
|
|
index = start_step * chip->ecc.bytes;
|
|
|
|
|
|
|
|
aligned_pos = eccpos[index] & ~(busw - 1);
|
2008-05-15 23:23:18 +07:00
|
|
|
aligned_len = eccfrag_len;
|
2010-09-07 18:23:45 +07:00
|
|
|
if (eccpos[index] & (busw - 1))
|
2008-05-15 23:23:18 +07:00
|
|
|
aligned_len++;
|
2010-09-07 18:23:45 +07:00
|
|
|
if (eccpos[index + (num_steps * chip->ecc.bytes)] & (busw - 1))
|
2008-05-15 23:23:18 +07:00
|
|
|
aligned_len++;
|
|
|
|
|
2010-09-07 18:23:45 +07:00
|
|
|
chip->cmdfunc(mtd, NAND_CMD_RNDOUT,
|
|
|
|
mtd->writesize + aligned_pos, -1);
|
2008-05-15 23:23:18 +07:00
|
|
|
chip->read_buf(mtd, &chip->oob_poi[aligned_pos], aligned_len);
|
|
|
|
}
|
|
|
|
|
|
|
|
for (i = 0; i < eccfrag_len; i++)
|
2010-09-07 18:23:45 +07:00
|
|
|
chip->buffers->ecccode[i] = chip->oob_poi[eccpos[i + index]];
|
2008-05-15 23:23:18 +07:00
|
|
|
|
|
|
|
p = bufpoi + data_col_addr;
|
|
|
|
for (i = 0; i < eccfrag_len ; i += chip->ecc.bytes, p += chip->ecc.size) {
|
|
|
|
int stat;
|
|
|
|
|
2010-09-07 18:23:45 +07:00
|
|
|
stat = chip->ecc.correct(mtd, p,
|
|
|
|
&chip->buffers->ecccode[i], &chip->buffers->ecccalc[i]);
|
2012-04-26 02:06:09 +07:00
|
|
|
if (stat < 0) {
|
2008-05-15 23:23:18 +07:00
|
|
|
mtd->ecc_stats.failed++;
|
2012-04-26 02:06:09 +07:00
|
|
|
} else {
|
2008-05-15 23:23:18 +07:00
|
|
|
mtd->ecc_stats.corrected += stat;
|
2012-04-26 02:06:09 +07:00
|
|
|
max_bitflips = max_t(unsigned int, max_bitflips, stat);
|
|
|
|
}
|
2008-05-15 23:23:18 +07:00
|
|
|
}
|
2012-04-26 02:06:09 +07:00
|
|
|
return max_bitflips;
|
2008-05-15 23:23:18 +07:00
|
|
|
}
|
|
|
|
|
2005-01-24 10:07:46 +07:00
|
|
|
/**
|
2011-06-24 04:12:08 +07:00
|
|
|
* nand_read_page_hwecc - [REPLACEABLE] hardware ECC based page read function
|
2011-05-26 04:59:01 +07:00
|
|
|
* @mtd: mtd info structure
|
|
|
|
* @chip: nand chip info structure
|
|
|
|
* @buf: buffer to store read data
|
2012-05-03 00:14:55 +07:00
|
|
|
* @oob_required: caller requires OOB data read to chip->oob_poi
|
2011-05-26 04:59:01 +07:00
|
|
|
* @page: page number to read
|
2005-01-24 10:07:46 +07:00
|
|
|
*
|
2011-06-24 04:12:08 +07:00
|
|
|
* Not for syndrome calculating ECC controllers which need a special oob layout.
|
2005-01-24 10:07:46 +07:00
|
|
|
*/
|
2006-05-25 15:07:16 +07:00
|
|
|
static int nand_read_page_hwecc(struct mtd_info *mtd, struct nand_chip *chip,
|
2012-05-03 00:14:55 +07:00
|
|
|
uint8_t *buf, int oob_required, int page)
|
2005-04-17 05:20:36 +07:00
|
|
|
{
|
2006-05-25 15:07:16 +07:00
|
|
|
int i, eccsize = chip->ecc.size;
|
|
|
|
int eccbytes = chip->ecc.bytes;
|
|
|
|
int eccsteps = chip->ecc.steps;
|
|
|
|
uint8_t *p = buf;
|
2006-09-25 23:08:04 +07:00
|
|
|
uint8_t *ecc_calc = chip->buffers->ecccalc;
|
|
|
|
uint8_t *ecc_code = chip->buffers->ecccode;
|
2007-05-29 01:17:54 +07:00
|
|
|
uint32_t *eccpos = chip->ecc.layout->eccpos;
|
2012-04-26 02:06:09 +07:00
|
|
|
unsigned int max_bitflips = 0;
|
2006-05-25 15:07:16 +07:00
|
|
|
|
|
|
|
for (i = 0; eccsteps; eccsteps--, i += eccbytes, p += eccsize) {
|
|
|
|
chip->ecc.hwctl(mtd, NAND_ECC_READ);
|
|
|
|
chip->read_buf(mtd, p, eccsize);
|
|
|
|
chip->ecc.calculate(mtd, p, &ecc_calc[i]);
|
2005-04-17 05:20:36 +07:00
|
|
|
}
|
2006-05-26 23:52:08 +07:00
|
|
|
chip->read_buf(mtd, chip->oob_poi, mtd->oobsize);
|
2005-04-17 05:20:36 +07:00
|
|
|
|
2006-05-25 15:07:16 +07:00
|
|
|
for (i = 0; i < chip->ecc.total; i++)
|
2006-05-26 23:52:08 +07:00
|
|
|
ecc_code[i] = chip->oob_poi[eccpos[i]];
|
2005-04-17 05:20:36 +07:00
|
|
|
|
2006-05-25 15:07:16 +07:00
|
|
|
eccsteps = chip->ecc.steps;
|
|
|
|
p = buf;
|
2005-11-07 18:15:49 +07:00
|
|
|
|
2006-05-25 15:07:16 +07:00
|
|
|
for (i = 0 ; eccsteps; eccsteps--, i += eccbytes, p += eccsize) {
|
|
|
|
int stat;
|
2005-04-17 05:20:36 +07:00
|
|
|
|
2006-05-25 15:07:16 +07:00
|
|
|
stat = chip->ecc.correct(mtd, p, &ecc_code[i], &ecc_calc[i]);
|
2012-04-26 02:06:09 +07:00
|
|
|
if (stat < 0) {
|
2006-05-25 15:07:16 +07:00
|
|
|
mtd->ecc_stats.failed++;
|
2012-04-26 02:06:09 +07:00
|
|
|
} else {
|
2006-05-25 15:07:16 +07:00
|
|
|
mtd->ecc_stats.corrected += stat;
|
2012-04-26 02:06:09 +07:00
|
|
|
max_bitflips = max_t(unsigned int, max_bitflips, stat);
|
|
|
|
}
|
2006-05-25 15:07:16 +07:00
|
|
|
}
|
2012-04-26 02:06:09 +07:00
|
|
|
return max_bitflips;
|
2006-05-25 15:07:16 +07:00
|
|
|
}
|
2005-04-17 05:20:36 +07:00
|
|
|
|
2009-09-19 02:51:47 +07:00
|
|
|
/**
|
2011-06-24 04:12:08 +07:00
|
|
|
* nand_read_page_hwecc_oob_first - [REPLACEABLE] hw ecc, read oob first
|
2011-05-26 04:59:01 +07:00
|
|
|
* @mtd: mtd info structure
|
|
|
|
* @chip: nand chip info structure
|
|
|
|
* @buf: buffer to store read data
|
2012-05-03 00:14:55 +07:00
|
|
|
* @oob_required: caller requires OOB data read to chip->oob_poi
|
2011-05-26 04:59:01 +07:00
|
|
|
* @page: page number to read
|
2009-09-19 02:51:47 +07:00
|
|
|
*
|
2011-05-26 04:59:01 +07:00
|
|
|
* Hardware ECC for large page chips, require OOB to be read first. For this
|
|
|
|
* ECC mode, the write_page method is re-used from ECC_HW. These methods
|
|
|
|
* read/write ECC from the OOB area, unlike the ECC_HW_SYNDROME support with
|
|
|
|
* multiple ECC steps, follows the "infix ECC" scheme and reads/writes ECC from
|
|
|
|
* the data area, by overwriting the NAND manufacturer bad block markings.
|
2009-09-19 02:51:47 +07:00
|
|
|
*/
|
|
|
|
static int nand_read_page_hwecc_oob_first(struct mtd_info *mtd,
|
2012-05-03 00:14:55 +07:00
|
|
|
struct nand_chip *chip, uint8_t *buf, int oob_required, int page)
|
2009-09-19 02:51:47 +07:00
|
|
|
{
|
|
|
|
int i, eccsize = chip->ecc.size;
|
|
|
|
int eccbytes = chip->ecc.bytes;
|
|
|
|
int eccsteps = chip->ecc.steps;
|
|
|
|
uint8_t *p = buf;
|
|
|
|
uint8_t *ecc_code = chip->buffers->ecccode;
|
|
|
|
uint32_t *eccpos = chip->ecc.layout->eccpos;
|
|
|
|
uint8_t *ecc_calc = chip->buffers->ecccalc;
|
2012-04-26 02:06:09 +07:00
|
|
|
unsigned int max_bitflips = 0;
|
2009-09-19 02:51:47 +07:00
|
|
|
|
|
|
|
/* Read the OOB area first */
|
|
|
|
chip->cmdfunc(mtd, NAND_CMD_READOOB, 0, page);
|
|
|
|
chip->read_buf(mtd, chip->oob_poi, mtd->oobsize);
|
|
|
|
chip->cmdfunc(mtd, NAND_CMD_READ0, 0, page);
|
|
|
|
|
|
|
|
for (i = 0; i < chip->ecc.total; i++)
|
|
|
|
ecc_code[i] = chip->oob_poi[eccpos[i]];
|
|
|
|
|
|
|
|
for (i = 0; eccsteps; eccsteps--, i += eccbytes, p += eccsize) {
|
|
|
|
int stat;
|
|
|
|
|
|
|
|
chip->ecc.hwctl(mtd, NAND_ECC_READ);
|
|
|
|
chip->read_buf(mtd, p, eccsize);
|
|
|
|
chip->ecc.calculate(mtd, p, &ecc_calc[i]);
|
|
|
|
|
|
|
|
stat = chip->ecc.correct(mtd, p, &ecc_code[i], NULL);
|
2012-04-26 02:06:09 +07:00
|
|
|
if (stat < 0) {
|
2009-09-19 02:51:47 +07:00
|
|
|
mtd->ecc_stats.failed++;
|
2012-04-26 02:06:09 +07:00
|
|
|
} else {
|
2009-09-19 02:51:47 +07:00
|
|
|
mtd->ecc_stats.corrected += stat;
|
2012-04-26 02:06:09 +07:00
|
|
|
max_bitflips = max_t(unsigned int, max_bitflips, stat);
|
|
|
|
}
|
2009-09-19 02:51:47 +07:00
|
|
|
}
|
2012-04-26 02:06:09 +07:00
|
|
|
return max_bitflips;
|
2009-09-19 02:51:47 +07:00
|
|
|
}
|
|
|
|
|
2006-05-25 15:07:16 +07:00
|
|
|
/**
|
2011-06-24 04:12:08 +07:00
|
|
|
* nand_read_page_syndrome - [REPLACEABLE] hardware ECC syndrome based page read
|
2011-05-26 04:59:01 +07:00
|
|
|
* @mtd: mtd info structure
|
|
|
|
* @chip: nand chip info structure
|
|
|
|
* @buf: buffer to store read data
|
2012-05-03 00:14:55 +07:00
|
|
|
* @oob_required: caller requires OOB data read to chip->oob_poi
|
2011-05-26 04:59:01 +07:00
|
|
|
* @page: page number to read
|
2006-05-25 15:07:16 +07:00
|
|
|
*
|
2011-05-26 04:59:01 +07:00
|
|
|
* The hw generator calculates the error syndrome automatically. Therefore we
|
|
|
|
* need a special oob layout and handling.
|
2006-05-25 15:07:16 +07:00
|
|
|
*/
|
|
|
|
static int nand_read_page_syndrome(struct mtd_info *mtd, struct nand_chip *chip,
|
2012-05-03 00:14:55 +07:00
|
|
|
uint8_t *buf, int oob_required, int page)
|
2006-05-25 15:07:16 +07:00
|
|
|
{
|
|
|
|
int i, eccsize = chip->ecc.size;
|
|
|
|
int eccbytes = chip->ecc.bytes;
|
|
|
|
int eccsteps = chip->ecc.steps;
|
|
|
|
uint8_t *p = buf;
|
2006-05-26 23:52:08 +07:00
|
|
|
uint8_t *oob = chip->oob_poi;
|
2012-04-26 02:06:09 +07:00
|
|
|
unsigned int max_bitflips = 0;
|
2005-04-17 05:20:36 +07:00
|
|
|
|
2006-05-25 15:07:16 +07:00
|
|
|
for (i = 0; eccsteps; eccsteps--, i += eccbytes, p += eccsize) {
|
|
|
|
int stat;
|
2005-11-07 18:15:49 +07:00
|
|
|
|
2006-05-25 15:07:16 +07:00
|
|
|
chip->ecc.hwctl(mtd, NAND_ECC_READ);
|
|
|
|
chip->read_buf(mtd, p, eccsize);
|
2005-04-17 05:20:36 +07:00
|
|
|
|
2006-05-25 15:07:16 +07:00
|
|
|
if (chip->ecc.prepad) {
|
|
|
|
chip->read_buf(mtd, oob, chip->ecc.prepad);
|
|
|
|
oob += chip->ecc.prepad;
|
|
|
|
}
|
2005-04-17 05:20:36 +07:00
|
|
|
|
2006-05-25 15:07:16 +07:00
|
|
|
chip->ecc.hwctl(mtd, NAND_ECC_READSYN);
|
|
|
|
chip->read_buf(mtd, oob, eccbytes);
|
|
|
|
stat = chip->ecc.correct(mtd, p, oob, NULL);
|
2005-11-07 18:15:49 +07:00
|
|
|
|
2012-04-26 02:06:09 +07:00
|
|
|
if (stat < 0) {
|
2006-05-25 15:07:16 +07:00
|
|
|
mtd->ecc_stats.failed++;
|
2012-04-26 02:06:09 +07:00
|
|
|
} else {
|
2006-05-25 15:07:16 +07:00
|
|
|
mtd->ecc_stats.corrected += stat;
|
2012-04-26 02:06:09 +07:00
|
|
|
max_bitflips = max_t(unsigned int, max_bitflips, stat);
|
|
|
|
}
|
2005-11-07 18:15:49 +07:00
|
|
|
|
2006-05-25 15:07:16 +07:00
|
|
|
oob += eccbytes;
|
2005-04-17 05:20:36 +07:00
|
|
|
|
2006-05-25 15:07:16 +07:00
|
|
|
if (chip->ecc.postpad) {
|
|
|
|
chip->read_buf(mtd, oob, chip->ecc.postpad);
|
|
|
|
oob += chip->ecc.postpad;
|
2005-11-07 18:15:49 +07:00
|
|
|
}
|
2006-05-25 15:07:16 +07:00
|
|
|
}
|
2005-04-17 05:20:36 +07:00
|
|
|
|
2006-05-25 15:07:16 +07:00
|
|
|
/* Calculate remaining oob bytes */
|
2006-06-07 12:34:37 +07:00
|
|
|
i = mtd->oobsize - (oob - chip->oob_poi);
|
2006-05-25 15:07:16 +07:00
|
|
|
if (i)
|
|
|
|
chip->read_buf(mtd, oob, i);
|
2005-11-07 18:15:49 +07:00
|
|
|
|
2012-04-26 02:06:09 +07:00
|
|
|
return max_bitflips;
|
2006-05-25 15:07:16 +07:00
|
|
|
}
|
2005-04-17 05:20:36 +07:00
|
|
|
|
2006-05-25 15:07:16 +07:00
|
|
|
/**
|
2011-06-24 04:12:08 +07:00
|
|
|
* nand_transfer_oob - [INTERN] Transfer oob to client buffer
|
2011-05-26 04:59:01 +07:00
|
|
|
* @chip: nand chip structure
|
|
|
|
* @oob: oob destination address
|
|
|
|
* @ops: oob ops structure
|
|
|
|
* @len: size of oob to transfer
|
2006-05-29 08:26:58 +07:00
|
|
|
*/
|
|
|
|
static uint8_t *nand_transfer_oob(struct nand_chip *chip, uint8_t *oob,
|
2006-11-03 22:20:38 +07:00
|
|
|
struct mtd_oob_ops *ops, size_t len)
|
2006-05-29 08:26:58 +07:00
|
|
|
{
|
2010-09-07 18:23:43 +07:00
|
|
|
switch (ops->mode) {
|
2006-05-29 08:26:58 +07:00
|
|
|
|
2011-08-31 08:45:40 +07:00
|
|
|
case MTD_OPS_PLACE_OOB:
|
|
|
|
case MTD_OPS_RAW:
|
2006-05-29 08:26:58 +07:00
|
|
|
memcpy(oob, chip->oob_poi + ops->ooboffs, len);
|
|
|
|
return oob + len;
|
|
|
|
|
2011-08-31 08:45:40 +07:00
|
|
|
case MTD_OPS_AUTO_OOB: {
|
2006-05-29 08:26:58 +07:00
|
|
|
struct nand_oobfree *free = chip->ecc.layout->oobfree;
|
2006-06-21 01:05:05 +07:00
|
|
|
uint32_t boffs = 0, roffs = ops->ooboffs;
|
|
|
|
size_t bytes = 0;
|
2006-05-29 08:26:58 +07:00
|
|
|
|
2010-09-07 18:23:43 +07:00
|
|
|
for (; free->length && len; free++, len -= bytes) {
|
2011-05-26 04:59:01 +07:00
|
|
|
/* Read request not from offset 0? */
|
2006-06-21 01:05:05 +07:00
|
|
|
if (unlikely(roffs)) {
|
|
|
|
if (roffs >= free->length) {
|
|
|
|
roffs -= free->length;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
boffs = free->offset + roffs;
|
|
|
|
bytes = min_t(size_t, len,
|
|
|
|
(free->length - roffs));
|
|
|
|
roffs = 0;
|
|
|
|
} else {
|
|
|
|
bytes = min_t(size_t, len, free->length);
|
|
|
|
boffs = free->offset;
|
|
|
|
}
|
|
|
|
memcpy(oob, chip->oob_poi + boffs, bytes);
|
2006-05-29 08:26:58 +07:00
|
|
|
oob += bytes;
|
|
|
|
}
|
|
|
|
return oob;
|
|
|
|
}
|
|
|
|
default:
|
|
|
|
BUG();
|
|
|
|
}
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2011-06-24 04:12:08 +07:00
|
|
|
* nand_do_read_ops - [INTERN] Read data with ECC
|
2011-05-26 04:59:01 +07:00
|
|
|
* @mtd: MTD device structure
|
|
|
|
* @from: offset to read from
|
|
|
|
* @ops: oob ops structure
|
2006-05-25 15:07:16 +07:00
|
|
|
*
|
|
|
|
* Internal function. Called with chip held.
|
|
|
|
*/
|
2006-05-29 08:26:58 +07:00
|
|
|
static int nand_do_read_ops(struct mtd_info *mtd, loff_t from,
|
|
|
|
struct mtd_oob_ops *ops)
|
2006-05-25 15:07:16 +07:00
|
|
|
{
|
2012-05-03 00:14:56 +07:00
|
|
|
int chipnr, page, realpage, col, bytes, aligned, oob_required;
|
2006-05-25 15:07:16 +07:00
|
|
|
struct nand_chip *chip = mtd->priv;
|
|
|
|
struct mtd_ecc_stats stats;
|
|
|
|
int ret = 0;
|
2006-05-29 08:26:58 +07:00
|
|
|
uint32_t readlen = ops->len;
|
2006-11-03 22:20:38 +07:00
|
|
|
uint32_t oobreadlen = ops->ooblen;
|
2011-08-31 08:45:40 +07:00
|
|
|
uint32_t max_oobsize = ops->mode == MTD_OPS_AUTO_OOB ?
|
2010-02-23 01:39:35 +07:00
|
|
|
mtd->oobavail : mtd->oobsize;
|
|
|
|
|
2006-05-29 08:26:58 +07:00
|
|
|
uint8_t *bufpoi, *oob, *buf;
|
2012-04-26 02:06:11 +07:00
|
|
|
unsigned int max_bitflips = 0;
|
2005-04-17 05:20:36 +07:00
|
|
|
|
2006-05-25 15:07:16 +07:00
|
|
|
stats = mtd->ecc_stats;
|
2005-04-17 05:20:36 +07:00
|
|
|
|
2006-05-25 15:07:16 +07:00
|
|
|
chipnr = (int)(from >> chip->chip_shift);
|
|
|
|
chip->select_chip(mtd, chipnr);
|
2005-11-07 18:15:49 +07:00
|
|
|
|
2006-05-25 15:07:16 +07:00
|
|
|
realpage = (int)(from >> chip->page_shift);
|
|
|
|
page = realpage & chip->pagemask;
|
2005-04-17 05:20:36 +07:00
|
|
|
|
2006-05-25 15:07:16 +07:00
|
|
|
col = (int)(from & (mtd->writesize - 1));
|
2005-11-07 18:15:49 +07:00
|
|
|
|
2006-05-29 08:26:58 +07:00
|
|
|
buf = ops->datbuf;
|
|
|
|
oob = ops->oobbuf;
|
2012-05-03 00:14:56 +07:00
|
|
|
oob_required = oob ? 1 : 0;
|
2006-05-29 08:26:58 +07:00
|
|
|
|
2010-09-07 18:23:43 +07:00
|
|
|
while (1) {
|
2006-05-25 15:07:16 +07:00
|
|
|
bytes = min(mtd->writesize - col, readlen);
|
|
|
|
aligned = (bytes == mtd->writesize);
|
2005-11-07 18:15:49 +07:00
|
|
|
|
2011-05-26 04:59:01 +07:00
|
|
|
/* Is the current page in the buffer? */
|
2006-05-29 08:26:58 +07:00
|
|
|
if (realpage != chip->pagebuf || oob) {
|
2006-09-25 23:08:04 +07:00
|
|
|
bufpoi = aligned ? buf : chip->buffers->databuf;
|
2005-11-07 18:15:49 +07:00
|
|
|
|
2012-05-02 07:12:54 +07:00
|
|
|
chip->cmdfunc(mtd, NAND_CMD_READ0, 0x00, page);
|
2005-04-17 05:20:36 +07:00
|
|
|
|
2012-04-26 02:06:11 +07:00
|
|
|
/*
|
|
|
|
* Now read the page into the buffer. Absent an error,
|
|
|
|
* the read methods return max bitflips per ecc step.
|
|
|
|
*/
|
2011-08-31 08:45:40 +07:00
|
|
|
if (unlikely(ops->mode == MTD_OPS_RAW))
|
2012-05-03 00:14:55 +07:00
|
|
|
ret = chip->ecc.read_page_raw(mtd, chip, bufpoi,
|
2012-05-03 00:14:56 +07:00
|
|
|
oob_required,
|
|
|
|
page);
|
2012-08-14 04:35:30 +07:00
|
|
|
else if (!aligned && NAND_HAS_SUBPAGE_READ(chip) &&
|
|
|
|
!oob)
|
2010-09-07 18:23:45 +07:00
|
|
|
ret = chip->ecc.read_subpage(mtd, chip,
|
|
|
|
col, bytes, bufpoi);
|
2006-09-25 23:12:39 +07:00
|
|
|
else
|
2009-09-19 02:51:46 +07:00
|
|
|
ret = chip->ecc.read_page(mtd, chip, bufpoi,
|
2012-05-03 00:14:56 +07:00
|
|
|
oob_required, page);
|
mtd: nand: invalidate cache on unaligned reads
In rare cases, we are given an unaligned parameter `from' in
`nand_do_read_ops()'. In such cases, we use the page cache
(chip->buffers->databuf) as an intermediate buffer before dumping to the
client buffer. However, there are also cases where this buffer is not
cleanly reusable. In those cases, we need to make sure that we
explicitly invalidate the cache.
This patch prevents accidental reusage of the page cache, and for me,
this solves some problems I come across when reading a corrupted BBT
from flash (NAND_BBT_USE_FLASH and NAND_BBT_NO_OOB).
Note: the rare "unaligned" case is a result of the extra BBT pattern +
version located in the data area instead of OOB.
Also, this patch disables caching on raw reads, since we are reading
without error correction. This is, obviously, prone to errors and should
not be cached.
Signed-off-by: Brian Norris <computersforpeace@gmail.com>
Signed-off-by: Artem Bityutskiy <artem.bityutskiy@intel.com>
2011-09-08 03:13:40 +07:00
|
|
|
if (ret < 0) {
|
|
|
|
if (!aligned)
|
|
|
|
/* Invalidate page cache */
|
|
|
|
chip->pagebuf = -1;
|
2005-04-17 05:20:36 +07:00
|
|
|
break;
|
mtd: nand: invalidate cache on unaligned reads
In rare cases, we are given an unaligned parameter `from' in
`nand_do_read_ops()'. In such cases, we use the page cache
(chip->buffers->databuf) as an intermediate buffer before dumping to the
client buffer. However, there are also cases where this buffer is not
cleanly reusable. In those cases, we need to make sure that we
explicitly invalidate the cache.
This patch prevents accidental reusage of the page cache, and for me,
this solves some problems I come across when reading a corrupted BBT
from flash (NAND_BBT_USE_FLASH and NAND_BBT_NO_OOB).
Note: the rare "unaligned" case is a result of the extra BBT pattern +
version located in the data area instead of OOB.
Also, this patch disables caching on raw reads, since we are reading
without error correction. This is, obviously, prone to errors and should
not be cached.
Signed-off-by: Brian Norris <computersforpeace@gmail.com>
Signed-off-by: Artem Bityutskiy <artem.bityutskiy@intel.com>
2011-09-08 03:13:40 +07:00
|
|
|
}
|
2006-05-25 15:07:16 +07:00
|
|
|
|
2012-04-26 02:06:11 +07:00
|
|
|
max_bitflips = max_t(unsigned int, max_bitflips, ret);
|
|
|
|
|
2006-05-25 15:07:16 +07:00
|
|
|
/* Transfer not aligned data */
|
|
|
|
if (!aligned) {
|
2012-08-14 04:35:30 +07:00
|
|
|
if (!NAND_HAS_SUBPAGE_READ(chip) && !oob &&
|
mtd: nand: invalidate cache on unaligned reads
In rare cases, we are given an unaligned parameter `from' in
`nand_do_read_ops()'. In such cases, we use the page cache
(chip->buffers->databuf) as an intermediate buffer before dumping to the
client buffer. However, there are also cases where this buffer is not
cleanly reusable. In those cases, we need to make sure that we
explicitly invalidate the cache.
This patch prevents accidental reusage of the page cache, and for me,
this solves some problems I come across when reading a corrupted BBT
from flash (NAND_BBT_USE_FLASH and NAND_BBT_NO_OOB).
Note: the rare "unaligned" case is a result of the extra BBT pattern +
version located in the data area instead of OOB.
Also, this patch disables caching on raw reads, since we are reading
without error correction. This is, obviously, prone to errors and should
not be cached.
Signed-off-by: Brian Norris <computersforpeace@gmail.com>
Signed-off-by: Artem Bityutskiy <artem.bityutskiy@intel.com>
2011-09-08 03:13:40 +07:00
|
|
|
!(mtd->ecc_stats.failed - stats.failed) &&
|
2012-04-26 02:06:11 +07:00
|
|
|
(ops->mode != MTD_OPS_RAW)) {
|
2008-05-15 23:23:18 +07:00
|
|
|
chip->pagebuf = realpage;
|
2012-04-26 02:06:11 +07:00
|
|
|
chip->pagebuf_bitflips = ret;
|
|
|
|
} else {
|
mtd: nand: invalidate cache on unaligned reads
In rare cases, we are given an unaligned parameter `from' in
`nand_do_read_ops()'. In such cases, we use the page cache
(chip->buffers->databuf) as an intermediate buffer before dumping to the
client buffer. However, there are also cases where this buffer is not
cleanly reusable. In those cases, we need to make sure that we
explicitly invalidate the cache.
This patch prevents accidental reusage of the page cache, and for me,
this solves some problems I come across when reading a corrupted BBT
from flash (NAND_BBT_USE_FLASH and NAND_BBT_NO_OOB).
Note: the rare "unaligned" case is a result of the extra BBT pattern +
version located in the data area instead of OOB.
Also, this patch disables caching on raw reads, since we are reading
without error correction. This is, obviously, prone to errors and should
not be cached.
Signed-off-by: Brian Norris <computersforpeace@gmail.com>
Signed-off-by: Artem Bityutskiy <artem.bityutskiy@intel.com>
2011-09-08 03:13:40 +07:00
|
|
|
/* Invalidate page cache */
|
|
|
|
chip->pagebuf = -1;
|
2012-04-26 02:06:11 +07:00
|
|
|
}
|
2006-09-25 23:08:04 +07:00
|
|
|
memcpy(buf, chip->buffers->databuf + col, bytes);
|
2006-05-25 15:07:16 +07:00
|
|
|
}
|
|
|
|
|
2006-05-29 08:26:58 +07:00
|
|
|
buf += bytes;
|
|
|
|
|
|
|
|
if (unlikely(oob)) {
|
2010-02-23 01:39:37 +07:00
|
|
|
int toread = min(oobreadlen, max_oobsize);
|
|
|
|
|
|
|
|
if (toread) {
|
|
|
|
oob = nand_transfer_oob(chip,
|
|
|
|
oob, ops, toread);
|
|
|
|
oobreadlen -= toread;
|
|
|
|
}
|
2006-05-29 08:26:58 +07:00
|
|
|
}
|
2013-03-13 23:51:31 +07:00
|
|
|
|
|
|
|
if (chip->options & NAND_NEED_READRDY) {
|
|
|
|
/* Apply delay or wait for ready/busy pin */
|
|
|
|
if (!chip->dev_ready)
|
|
|
|
udelay(chip->chip_delay);
|
|
|
|
else
|
|
|
|
nand_wait_ready(mtd);
|
|
|
|
}
|
2006-05-29 08:26:58 +07:00
|
|
|
} else {
|
2006-09-25 23:08:04 +07:00
|
|
|
memcpy(buf, chip->buffers->databuf + col, bytes);
|
2006-05-29 08:26:58 +07:00
|
|
|
buf += bytes;
|
2012-04-26 02:06:11 +07:00
|
|
|
max_bitflips = max_t(unsigned int, max_bitflips,
|
|
|
|
chip->pagebuf_bitflips);
|
2006-05-29 08:26:58 +07:00
|
|
|
}
|
2005-04-17 05:20:36 +07:00
|
|
|
|
2006-05-25 15:07:16 +07:00
|
|
|
readlen -= bytes;
|
2005-11-07 18:15:49 +07:00
|
|
|
|
2006-05-25 15:07:16 +07:00
|
|
|
if (!readlen)
|
2005-11-07 18:15:49 +07:00
|
|
|
break;
|
2005-04-17 05:20:36 +07:00
|
|
|
|
2011-05-26 04:59:01 +07:00
|
|
|
/* For subsequent reads align to page boundary */
|
2005-04-17 05:20:36 +07:00
|
|
|
col = 0;
|
|
|
|
/* Increment page address */
|
|
|
|
realpage++;
|
|
|
|
|
2006-05-24 17:07:37 +07:00
|
|
|
page = realpage & chip->pagemask;
|
2005-04-17 05:20:36 +07:00
|
|
|
/* Check, if we cross a chip boundary */
|
|
|
|
if (!page) {
|
|
|
|
chipnr++;
|
2006-05-24 17:07:37 +07:00
|
|
|
chip->select_chip(mtd, -1);
|
|
|
|
chip->select_chip(mtd, chipnr);
|
2005-04-17 05:20:36 +07:00
|
|
|
}
|
|
|
|
}
|
2012-11-19 13:43:29 +07:00
|
|
|
chip->select_chip(mtd, -1);
|
2005-04-17 05:20:36 +07:00
|
|
|
|
2006-05-29 08:26:58 +07:00
|
|
|
ops->retlen = ops->len - (size_t) readlen;
|
2006-11-03 22:20:38 +07:00
|
|
|
if (oob)
|
|
|
|
ops->oobretlen = ops->ooblen - oobreadlen;
|
2005-04-17 05:20:36 +07:00
|
|
|
|
2012-04-26 02:06:09 +07:00
|
|
|
if (ret < 0)
|
2006-05-25 15:07:16 +07:00
|
|
|
return ret;
|
|
|
|
|
2006-05-29 19:56:39 +07:00
|
|
|
if (mtd->ecc_stats.failed - stats.failed)
|
|
|
|
return -EBADMSG;
|
|
|
|
|
2012-04-26 02:06:11 +07:00
|
|
|
return max_bitflips;
|
2006-05-25 15:07:16 +07:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2011-03-31 08:57:33 +07:00
|
|
|
* nand_read - [MTD Interface] MTD compatibility function for nand_do_read_ecc
|
2011-05-26 04:59:01 +07:00
|
|
|
* @mtd: MTD device structure
|
|
|
|
* @from: offset to read from
|
|
|
|
* @len: number of bytes to read
|
|
|
|
* @retlen: pointer to variable to store the number of read bytes
|
|
|
|
* @buf: the databuffer to put data
|
2006-05-25 15:07:16 +07:00
|
|
|
*
|
2011-05-26 04:59:01 +07:00
|
|
|
* Get hold of the chip and call nand_do_read.
|
2006-05-25 15:07:16 +07:00
|
|
|
*/
|
|
|
|
static int nand_read(struct mtd_info *mtd, loff_t from, size_t len,
|
|
|
|
size_t *retlen, uint8_t *buf)
|
|
|
|
{
|
2011-08-31 08:45:45 +07:00
|
|
|
struct mtd_oob_ops ops;
|
2006-05-25 15:07:16 +07:00
|
|
|
int ret;
|
|
|
|
|
2012-11-19 13:43:30 +07:00
|
|
|
nand_get_device(mtd, FL_READING);
|
2011-08-31 08:45:45 +07:00
|
|
|
ops.len = len;
|
|
|
|
ops.datbuf = buf;
|
|
|
|
ops.oobbuf = NULL;
|
2012-07-03 15:44:14 +07:00
|
|
|
ops.mode = MTD_OPS_PLACE_OOB;
|
2011-08-31 08:45:45 +07:00
|
|
|
ret = nand_do_read_ops(mtd, from, &ops);
|
|
|
|
*retlen = ops.retlen;
|
2006-05-25 15:07:16 +07:00
|
|
|
nand_release_device(mtd);
|
|
|
|
return ret;
|
2005-04-17 05:20:36 +07:00
|
|
|
}
|
|
|
|
|
2006-06-21 01:05:05 +07:00
|
|
|
/**
|
2011-06-24 04:12:08 +07:00
|
|
|
* nand_read_oob_std - [REPLACEABLE] the most common OOB data read function
|
2011-05-26 04:59:01 +07:00
|
|
|
* @mtd: mtd info structure
|
|
|
|
* @chip: nand chip info structure
|
|
|
|
* @page: page number to read
|
2006-06-21 01:05:05 +07:00
|
|
|
*/
|
|
|
|
static int nand_read_oob_std(struct mtd_info *mtd, struct nand_chip *chip,
|
2012-05-09 17:06:35 +07:00
|
|
|
int page)
|
2006-06-21 01:05:05 +07:00
|
|
|
{
|
2012-05-09 17:06:35 +07:00
|
|
|
chip->cmdfunc(mtd, NAND_CMD_READOOB, 0, page);
|
2006-06-21 01:05:05 +07:00
|
|
|
chip->read_buf(mtd, chip->oob_poi, mtd->oobsize);
|
2012-05-09 17:06:35 +07:00
|
|
|
return 0;
|
2006-06-21 01:05:05 +07:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2011-06-24 04:12:08 +07:00
|
|
|
* nand_read_oob_syndrome - [REPLACEABLE] OOB data read function for HW ECC
|
2006-06-21 01:05:05 +07:00
|
|
|
* with syndromes
|
2011-05-26 04:59:01 +07:00
|
|
|
* @mtd: mtd info structure
|
|
|
|
* @chip: nand chip info structure
|
|
|
|
* @page: page number to read
|
2006-06-21 01:05:05 +07:00
|
|
|
*/
|
|
|
|
static int nand_read_oob_syndrome(struct mtd_info *mtd, struct nand_chip *chip,
|
2012-05-09 17:06:35 +07:00
|
|
|
int page)
|
2006-06-21 01:05:05 +07:00
|
|
|
{
|
|
|
|
uint8_t *buf = chip->oob_poi;
|
|
|
|
int length = mtd->oobsize;
|
|
|
|
int chunk = chip->ecc.bytes + chip->ecc.prepad + chip->ecc.postpad;
|
|
|
|
int eccsize = chip->ecc.size;
|
|
|
|
uint8_t *bufpoi = buf;
|
|
|
|
int i, toread, sndrnd = 0, pos;
|
|
|
|
|
|
|
|
chip->cmdfunc(mtd, NAND_CMD_READ0, chip->ecc.size, page);
|
|
|
|
for (i = 0; i < chip->ecc.steps; i++) {
|
|
|
|
if (sndrnd) {
|
|
|
|
pos = eccsize + i * (eccsize + chunk);
|
|
|
|
if (mtd->writesize > 512)
|
|
|
|
chip->cmdfunc(mtd, NAND_CMD_RNDOUT, pos, -1);
|
|
|
|
else
|
|
|
|
chip->cmdfunc(mtd, NAND_CMD_READ0, pos, page);
|
|
|
|
} else
|
|
|
|
sndrnd = 1;
|
|
|
|
toread = min_t(int, length, chunk);
|
|
|
|
chip->read_buf(mtd, bufpoi, toread);
|
|
|
|
bufpoi += toread;
|
|
|
|
length -= toread;
|
|
|
|
}
|
|
|
|
if (length > 0)
|
|
|
|
chip->read_buf(mtd, bufpoi, length);
|
|
|
|
|
2012-05-09 17:06:35 +07:00
|
|
|
return 0;
|
2006-06-21 01:05:05 +07:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2011-06-24 04:12:08 +07:00
|
|
|
* nand_write_oob_std - [REPLACEABLE] the most common OOB data write function
|
2011-05-26 04:59:01 +07:00
|
|
|
* @mtd: mtd info structure
|
|
|
|
* @chip: nand chip info structure
|
|
|
|
* @page: page number to write
|
2006-06-21 01:05:05 +07:00
|
|
|
*/
|
|
|
|
static int nand_write_oob_std(struct mtd_info *mtd, struct nand_chip *chip,
|
|
|
|
int page)
|
|
|
|
{
|
|
|
|
int status = 0;
|
|
|
|
const uint8_t *buf = chip->oob_poi;
|
|
|
|
int length = mtd->oobsize;
|
|
|
|
|
|
|
|
chip->cmdfunc(mtd, NAND_CMD_SEQIN, mtd->writesize, page);
|
|
|
|
chip->write_buf(mtd, buf, length);
|
|
|
|
/* Send command to program the OOB data */
|
|
|
|
chip->cmdfunc(mtd, NAND_CMD_PAGEPROG, -1, -1);
|
|
|
|
|
|
|
|
status = chip->waitfunc(mtd, chip);
|
|
|
|
|
2006-06-21 16:51:20 +07:00
|
|
|
return status & NAND_STATUS_FAIL ? -EIO : 0;
|
2006-06-21 01:05:05 +07:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2011-06-24 04:12:08 +07:00
|
|
|
* nand_write_oob_syndrome - [REPLACEABLE] OOB data write function for HW ECC
|
2011-05-26 04:59:01 +07:00
|
|
|
* with syndrome - only for large page flash
|
|
|
|
* @mtd: mtd info structure
|
|
|
|
* @chip: nand chip info structure
|
|
|
|
* @page: page number to write
|
2006-06-21 01:05:05 +07:00
|
|
|
*/
|
|
|
|
static int nand_write_oob_syndrome(struct mtd_info *mtd,
|
|
|
|
struct nand_chip *chip, int page)
|
|
|
|
{
|
|
|
|
int chunk = chip->ecc.bytes + chip->ecc.prepad + chip->ecc.postpad;
|
|
|
|
int eccsize = chip->ecc.size, length = mtd->oobsize;
|
|
|
|
int i, len, pos, status = 0, sndcmd = 0, steps = chip->ecc.steps;
|
|
|
|
const uint8_t *bufpoi = chip->oob_poi;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* data-ecc-data-ecc ... ecc-oob
|
|
|
|
* or
|
|
|
|
* data-pad-ecc-pad-data-pad .... ecc-pad-oob
|
|
|
|
*/
|
|
|
|
if (!chip->ecc.prepad && !chip->ecc.postpad) {
|
|
|
|
pos = steps * (eccsize + chunk);
|
|
|
|
steps = 0;
|
|
|
|
} else
|
2006-07-11 14:11:25 +07:00
|
|
|
pos = eccsize;
|
2006-06-21 01:05:05 +07:00
|
|
|
|
|
|
|
chip->cmdfunc(mtd, NAND_CMD_SEQIN, pos, page);
|
|
|
|
for (i = 0; i < steps; i++) {
|
|
|
|
if (sndcmd) {
|
|
|
|
if (mtd->writesize <= 512) {
|
|
|
|
uint32_t fill = 0xFFFFFFFF;
|
|
|
|
|
|
|
|
len = eccsize;
|
|
|
|
while (len > 0) {
|
|
|
|
int num = min_t(int, len, 4);
|
|
|
|
chip->write_buf(mtd, (uint8_t *)&fill,
|
|
|
|
num);
|
|
|
|
len -= num;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
pos = eccsize + i * (eccsize + chunk);
|
|
|
|
chip->cmdfunc(mtd, NAND_CMD_RNDIN, pos, -1);
|
|
|
|
}
|
|
|
|
} else
|
|
|
|
sndcmd = 1;
|
|
|
|
len = min_t(int, length, chunk);
|
|
|
|
chip->write_buf(mtd, bufpoi, len);
|
|
|
|
bufpoi += len;
|
|
|
|
length -= len;
|
|
|
|
}
|
|
|
|
if (length > 0)
|
|
|
|
chip->write_buf(mtd, bufpoi, length);
|
|
|
|
|
|
|
|
chip->cmdfunc(mtd, NAND_CMD_PAGEPROG, -1, -1);
|
|
|
|
status = chip->waitfunc(mtd, chip);
|
|
|
|
|
|
|
|
return status & NAND_STATUS_FAIL ? -EIO : 0;
|
|
|
|
}
|
|
|
|
|
2005-04-17 05:20:36 +07:00
|
|
|
/**
|
2011-06-24 04:12:08 +07:00
|
|
|
* nand_do_read_oob - [INTERN] NAND read out-of-band
|
2011-05-26 04:59:01 +07:00
|
|
|
* @mtd: MTD device structure
|
|
|
|
* @from: offset to read from
|
|
|
|
* @ops: oob operations description structure
|
2005-04-17 05:20:36 +07:00
|
|
|
*
|
2011-05-26 04:59:01 +07:00
|
|
|
* NAND read out-of-band data from the spare area.
|
2005-04-17 05:20:36 +07:00
|
|
|
*/
|
2006-05-29 08:26:58 +07:00
|
|
|
static int nand_do_read_oob(struct mtd_info *mtd, loff_t from,
|
|
|
|
struct mtd_oob_ops *ops)
|
2005-04-17 05:20:36 +07:00
|
|
|
{
|
2012-05-02 07:12:54 +07:00
|
|
|
int page, realpage, chipnr;
|
2006-05-24 17:07:37 +07:00
|
|
|
struct nand_chip *chip = mtd->priv;
|
2011-06-24 06:45:24 +07:00
|
|
|
struct mtd_ecc_stats stats;
|
2006-11-03 22:20:38 +07:00
|
|
|
int readlen = ops->ooblen;
|
|
|
|
int len;
|
2006-06-21 01:05:05 +07:00
|
|
|
uint8_t *buf = ops->oobbuf;
|
2012-05-09 17:13:34 +07:00
|
|
|
int ret = 0;
|
2005-11-07 18:15:49 +07:00
|
|
|
|
2011-07-20 00:06:09 +07:00
|
|
|
pr_debug("%s: from = 0x%08Lx, len = %i\n",
|
2009-07-07 17:19:49 +07:00
|
|
|
__func__, (unsigned long long)from, readlen);
|
2005-04-17 05:20:36 +07:00
|
|
|
|
2011-06-24 06:45:24 +07:00
|
|
|
stats = mtd->ecc_stats;
|
|
|
|
|
2011-08-31 08:45:40 +07:00
|
|
|
if (ops->mode == MTD_OPS_AUTO_OOB)
|
2006-11-03 22:20:38 +07:00
|
|
|
len = chip->ecc.layout->oobavail;
|
2007-01-31 22:58:29 +07:00
|
|
|
else
|
|
|
|
len = mtd->oobsize;
|
|
|
|
|
|
|
|
if (unlikely(ops->ooboffs >= len)) {
|
2011-07-20 00:06:09 +07:00
|
|
|
pr_debug("%s: attempt to start read outside oob\n",
|
|
|
|
__func__);
|
2007-01-31 22:58:29 +07:00
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Do not allow reads past end of device */
|
|
|
|
if (unlikely(from >= mtd->size ||
|
|
|
|
ops->ooboffs + readlen > ((mtd->size >> chip->page_shift) -
|
|
|
|
(from >> chip->page_shift)) * len)) {
|
2011-07-20 00:06:09 +07:00
|
|
|
pr_debug("%s: attempt to read beyond end of device\n",
|
|
|
|
__func__);
|
2007-01-31 22:58:29 +07:00
|
|
|
return -EINVAL;
|
|
|
|
}
|
2006-11-03 22:20:38 +07:00
|
|
|
|
2006-05-25 14:51:54 +07:00
|
|
|
chipnr = (int)(from >> chip->chip_shift);
|
2006-05-24 17:07:37 +07:00
|
|
|
chip->select_chip(mtd, chipnr);
|
2005-04-17 05:20:36 +07:00
|
|
|
|
2006-05-25 14:51:54 +07:00
|
|
|
/* Shift to get page */
|
|
|
|
realpage = (int)(from >> chip->page_shift);
|
|
|
|
page = realpage & chip->pagemask;
|
2005-04-17 05:20:36 +07:00
|
|
|
|
2010-09-07 18:23:43 +07:00
|
|
|
while (1) {
|
2011-08-31 08:45:40 +07:00
|
|
|
if (ops->mode == MTD_OPS_RAW)
|
2012-05-09 17:13:34 +07:00
|
|
|
ret = chip->ecc.read_oob_raw(mtd, chip, page);
|
2011-08-31 08:45:38 +07:00
|
|
|
else
|
2012-05-09 17:13:34 +07:00
|
|
|
ret = chip->ecc.read_oob(mtd, chip, page);
|
|
|
|
|
|
|
|
if (ret < 0)
|
|
|
|
break;
|
2006-11-03 22:20:38 +07:00
|
|
|
|
|
|
|
len = min(len, readlen);
|
|
|
|
buf = nand_transfer_oob(chip, buf, ops, len);
|
2006-05-29 08:26:58 +07:00
|
|
|
|
2013-03-13 23:51:31 +07:00
|
|
|
if (chip->options & NAND_NEED_READRDY) {
|
|
|
|
/* Apply delay or wait for ready/busy pin */
|
|
|
|
if (!chip->dev_ready)
|
|
|
|
udelay(chip->chip_delay);
|
|
|
|
else
|
|
|
|
nand_wait_ready(mtd);
|
|
|
|
}
|
|
|
|
|
2006-11-03 22:20:38 +07:00
|
|
|
readlen -= len;
|
2006-06-21 16:51:20 +07:00
|
|
|
if (!readlen)
|
|
|
|
break;
|
|
|
|
|
2006-05-25 14:51:54 +07:00
|
|
|
/* Increment page address */
|
|
|
|
realpage++;
|
|
|
|
|
|
|
|
page = realpage & chip->pagemask;
|
|
|
|
/* Check, if we cross a chip boundary */
|
|
|
|
if (!page) {
|
|
|
|
chipnr++;
|
|
|
|
chip->select_chip(mtd, -1);
|
|
|
|
chip->select_chip(mtd, chipnr);
|
2005-04-17 05:20:36 +07:00
|
|
|
}
|
|
|
|
}
|
2012-11-19 13:43:29 +07:00
|
|
|
chip->select_chip(mtd, -1);
|
2005-04-17 05:20:36 +07:00
|
|
|
|
2012-05-09 17:13:34 +07:00
|
|
|
ops->oobretlen = ops->ooblen - readlen;
|
|
|
|
|
|
|
|
if (ret < 0)
|
|
|
|
return ret;
|
2011-06-24 06:45:24 +07:00
|
|
|
|
|
|
|
if (mtd->ecc_stats.failed - stats.failed)
|
|
|
|
return -EBADMSG;
|
|
|
|
|
|
|
|
return mtd->ecc_stats.corrected - stats.corrected ? -EUCLEAN : 0;
|
2005-04-17 05:20:36 +07:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2006-05-29 08:26:58 +07:00
|
|
|
* nand_read_oob - [MTD Interface] NAND read data and/or out-of-band
|
2011-05-26 04:59:01 +07:00
|
|
|
* @mtd: MTD device structure
|
|
|
|
* @from: offset to read from
|
|
|
|
* @ops: oob operation description structure
|
2005-04-17 05:20:36 +07:00
|
|
|
*
|
2011-05-26 04:59:01 +07:00
|
|
|
* NAND read data and/or out-of-band data.
|
2005-04-17 05:20:36 +07:00
|
|
|
*/
|
2006-05-29 08:26:58 +07:00
|
|
|
static int nand_read_oob(struct mtd_info *mtd, loff_t from,
|
|
|
|
struct mtd_oob_ops *ops)
|
2005-04-17 05:20:36 +07:00
|
|
|
{
|
2006-05-29 08:26:58 +07:00
|
|
|
int ret = -ENOTSUPP;
|
|
|
|
|
|
|
|
ops->retlen = 0;
|
2005-04-17 05:20:36 +07:00
|
|
|
|
|
|
|
/* Do not allow reads past end of device */
|
2006-11-03 22:20:38 +07:00
|
|
|
if (ops->datbuf && (from + ops->len) > mtd->size) {
|
2011-07-20 00:06:09 +07:00
|
|
|
pr_debug("%s: attempt to read beyond end of device\n",
|
|
|
|
__func__);
|
2005-04-17 05:20:36 +07:00
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
2012-11-19 13:43:30 +07:00
|
|
|
nand_get_device(mtd, FL_READING);
|
2005-04-17 05:20:36 +07:00
|
|
|
|
2010-09-07 18:23:43 +07:00
|
|
|
switch (ops->mode) {
|
2011-08-31 08:45:40 +07:00
|
|
|
case MTD_OPS_PLACE_OOB:
|
|
|
|
case MTD_OPS_AUTO_OOB:
|
|
|
|
case MTD_OPS_RAW:
|
2006-05-29 08:26:58 +07:00
|
|
|
break;
|
2005-04-17 05:20:36 +07:00
|
|
|
|
2006-05-29 08:26:58 +07:00
|
|
|
default:
|
|
|
|
goto out;
|
|
|
|
}
|
2005-04-17 05:20:36 +07:00
|
|
|
|
2006-05-29 08:26:58 +07:00
|
|
|
if (!ops->datbuf)
|
|
|
|
ret = nand_do_read_oob(mtd, from, ops);
|
|
|
|
else
|
|
|
|
ret = nand_do_read_ops(mtd, from, ops);
|
2005-11-07 18:15:49 +07:00
|
|
|
|
2010-09-07 18:23:45 +07:00
|
|
|
out:
|
2006-05-29 08:26:58 +07:00
|
|
|
nand_release_device(mtd);
|
|
|
|
return ret;
|
|
|
|
}
|
2005-11-07 18:15:49 +07:00
|
|
|
|
2005-04-17 05:20:36 +07:00
|
|
|
|
2006-05-29 08:26:58 +07:00
|
|
|
/**
|
2011-06-24 04:12:08 +07:00
|
|
|
* nand_write_page_raw - [INTERN] raw page write function
|
2011-05-26 04:59:01 +07:00
|
|
|
* @mtd: mtd info structure
|
|
|
|
* @chip: nand chip info structure
|
|
|
|
* @buf: data buffer
|
2012-05-03 00:14:55 +07:00
|
|
|
* @oob_required: must write chip->oob_poi to OOB
|
[MTD] [NAND] fix "raw" reads with ECC syndrome layouts
The syndrome based page read/write routines store ECC, and possibly other
"OOB" data, right after each chunk of ECC'd data. With ECC chunk size of
512 bytes and a large page (2KiB) NAND, the layout is:
data-0 OOB-0 data-1 OOB-1 data-2 OOB-2 data-3 OOB-3 OOB-leftover
Where OOBx is (prepad, ECC, postpad). However, the current "raw" routines
use a traditional layout -- data OOB, disregarding the prepad and postpad
values -- so when they're used with that type of ECC hardware, those calls
mix up the data and OOB. Which means, in particular, that bad block
tables won't be found on startup, with data corruption and related chaos
ensuing.
The current syndrome-based drivers in mainline all seem to use one chunk
per page; presumably they haven't noticed such bugs.
Fix this, by adding read/write page_raw_syndrome() routines as siblings of
the existing non-raw routines; "raw" just means to bypass the ECC
computations, not change data and OOB layout.
Signed-off-by: David Brownell <dbrownell@users.sourceforge.net>
Signed-off-by: Andrew Morton <akpm@linux-foundation.org>
Signed-off-by: David Woodhouse <David.Woodhouse@intel.com>
2009-03-05 03:01:36 +07:00
|
|
|
*
|
2011-06-24 04:12:08 +07:00
|
|
|
* Not for syndrome calculating ECC controllers, which use a special oob layout.
|
2006-05-29 08:26:58 +07:00
|
|
|
*/
|
2012-06-25 17:07:45 +07:00
|
|
|
static int nand_write_page_raw(struct mtd_info *mtd, struct nand_chip *chip,
|
2012-05-03 00:14:55 +07:00
|
|
|
const uint8_t *buf, int oob_required)
|
2006-05-29 08:26:58 +07:00
|
|
|
{
|
|
|
|
chip->write_buf(mtd, buf, mtd->writesize);
|
2012-05-03 00:15:03 +07:00
|
|
|
if (oob_required)
|
|
|
|
chip->write_buf(mtd, chip->oob_poi, mtd->oobsize);
|
2012-06-25 17:07:45 +07:00
|
|
|
|
|
|
|
return 0;
|
2005-04-17 05:20:36 +07:00
|
|
|
}
|
|
|
|
|
[MTD] [NAND] fix "raw" reads with ECC syndrome layouts
The syndrome based page read/write routines store ECC, and possibly other
"OOB" data, right after each chunk of ECC'd data. With ECC chunk size of
512 bytes and a large page (2KiB) NAND, the layout is:
data-0 OOB-0 data-1 OOB-1 data-2 OOB-2 data-3 OOB-3 OOB-leftover
Where OOBx is (prepad, ECC, postpad). However, the current "raw" routines
use a traditional layout -- data OOB, disregarding the prepad and postpad
values -- so when they're used with that type of ECC hardware, those calls
mix up the data and OOB. Which means, in particular, that bad block
tables won't be found on startup, with data corruption and related chaos
ensuing.
The current syndrome-based drivers in mainline all seem to use one chunk
per page; presumably they haven't noticed such bugs.
Fix this, by adding read/write page_raw_syndrome() routines as siblings of
the existing non-raw routines; "raw" just means to bypass the ECC
computations, not change data and OOB layout.
Signed-off-by: David Brownell <dbrownell@users.sourceforge.net>
Signed-off-by: Andrew Morton <akpm@linux-foundation.org>
Signed-off-by: David Woodhouse <David.Woodhouse@intel.com>
2009-03-05 03:01:36 +07:00
|
|
|
/**
|
2011-06-24 04:12:08 +07:00
|
|
|
* nand_write_page_raw_syndrome - [INTERN] raw page write function
|
2011-05-26 04:59:01 +07:00
|
|
|
* @mtd: mtd info structure
|
|
|
|
* @chip: nand chip info structure
|
|
|
|
* @buf: data buffer
|
2012-05-03 00:14:55 +07:00
|
|
|
* @oob_required: must write chip->oob_poi to OOB
|
[MTD] [NAND] fix "raw" reads with ECC syndrome layouts
The syndrome based page read/write routines store ECC, and possibly other
"OOB" data, right after each chunk of ECC'd data. With ECC chunk size of
512 bytes and a large page (2KiB) NAND, the layout is:
data-0 OOB-0 data-1 OOB-1 data-2 OOB-2 data-3 OOB-3 OOB-leftover
Where OOBx is (prepad, ECC, postpad). However, the current "raw" routines
use a traditional layout -- data OOB, disregarding the prepad and postpad
values -- so when they're used with that type of ECC hardware, those calls
mix up the data and OOB. Which means, in particular, that bad block
tables won't be found on startup, with data corruption and related chaos
ensuing.
The current syndrome-based drivers in mainline all seem to use one chunk
per page; presumably they haven't noticed such bugs.
Fix this, by adding read/write page_raw_syndrome() routines as siblings of
the existing non-raw routines; "raw" just means to bypass the ECC
computations, not change data and OOB layout.
Signed-off-by: David Brownell <dbrownell@users.sourceforge.net>
Signed-off-by: Andrew Morton <akpm@linux-foundation.org>
Signed-off-by: David Woodhouse <David.Woodhouse@intel.com>
2009-03-05 03:01:36 +07:00
|
|
|
*
|
|
|
|
* We need a special oob layout and handling even when ECC isn't checked.
|
|
|
|
*/
|
2012-06-25 17:07:45 +07:00
|
|
|
static int nand_write_page_raw_syndrome(struct mtd_info *mtd,
|
2010-09-07 18:23:45 +07:00
|
|
|
struct nand_chip *chip,
|
2012-05-03 00:14:55 +07:00
|
|
|
const uint8_t *buf, int oob_required)
|
[MTD] [NAND] fix "raw" reads with ECC syndrome layouts
The syndrome based page read/write routines store ECC, and possibly other
"OOB" data, right after each chunk of ECC'd data. With ECC chunk size of
512 bytes and a large page (2KiB) NAND, the layout is:
data-0 OOB-0 data-1 OOB-1 data-2 OOB-2 data-3 OOB-3 OOB-leftover
Where OOBx is (prepad, ECC, postpad). However, the current "raw" routines
use a traditional layout -- data OOB, disregarding the prepad and postpad
values -- so when they're used with that type of ECC hardware, those calls
mix up the data and OOB. Which means, in particular, that bad block
tables won't be found on startup, with data corruption and related chaos
ensuing.
The current syndrome-based drivers in mainline all seem to use one chunk
per page; presumably they haven't noticed such bugs.
Fix this, by adding read/write page_raw_syndrome() routines as siblings of
the existing non-raw routines; "raw" just means to bypass the ECC
computations, not change data and OOB layout.
Signed-off-by: David Brownell <dbrownell@users.sourceforge.net>
Signed-off-by: Andrew Morton <akpm@linux-foundation.org>
Signed-off-by: David Woodhouse <David.Woodhouse@intel.com>
2009-03-05 03:01:36 +07:00
|
|
|
{
|
|
|
|
int eccsize = chip->ecc.size;
|
|
|
|
int eccbytes = chip->ecc.bytes;
|
|
|
|
uint8_t *oob = chip->oob_poi;
|
|
|
|
int steps, size;
|
|
|
|
|
|
|
|
for (steps = chip->ecc.steps; steps > 0; steps--) {
|
|
|
|
chip->write_buf(mtd, buf, eccsize);
|
|
|
|
buf += eccsize;
|
|
|
|
|
|
|
|
if (chip->ecc.prepad) {
|
|
|
|
chip->write_buf(mtd, oob, chip->ecc.prepad);
|
|
|
|
oob += chip->ecc.prepad;
|
|
|
|
}
|
|
|
|
|
|
|
|
chip->read_buf(mtd, oob, eccbytes);
|
|
|
|
oob += eccbytes;
|
|
|
|
|
|
|
|
if (chip->ecc.postpad) {
|
|
|
|
chip->write_buf(mtd, oob, chip->ecc.postpad);
|
|
|
|
oob += chip->ecc.postpad;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
size = mtd->oobsize - (oob - chip->oob_poi);
|
|
|
|
if (size)
|
|
|
|
chip->write_buf(mtd, oob, size);
|
2012-06-25 17:07:45 +07:00
|
|
|
|
|
|
|
return 0;
|
[MTD] [NAND] fix "raw" reads with ECC syndrome layouts
The syndrome based page read/write routines store ECC, and possibly other
"OOB" data, right after each chunk of ECC'd data. With ECC chunk size of
512 bytes and a large page (2KiB) NAND, the layout is:
data-0 OOB-0 data-1 OOB-1 data-2 OOB-2 data-3 OOB-3 OOB-leftover
Where OOBx is (prepad, ECC, postpad). However, the current "raw" routines
use a traditional layout -- data OOB, disregarding the prepad and postpad
values -- so when they're used with that type of ECC hardware, those calls
mix up the data and OOB. Which means, in particular, that bad block
tables won't be found on startup, with data corruption and related chaos
ensuing.
The current syndrome-based drivers in mainline all seem to use one chunk
per page; presumably they haven't noticed such bugs.
Fix this, by adding read/write page_raw_syndrome() routines as siblings of
the existing non-raw routines; "raw" just means to bypass the ECC
computations, not change data and OOB layout.
Signed-off-by: David Brownell <dbrownell@users.sourceforge.net>
Signed-off-by: Andrew Morton <akpm@linux-foundation.org>
Signed-off-by: David Woodhouse <David.Woodhouse@intel.com>
2009-03-05 03:01:36 +07:00
|
|
|
}
|
2006-05-23 22:21:03 +07:00
|
|
|
/**
|
2011-06-24 04:12:08 +07:00
|
|
|
* nand_write_page_swecc - [REPLACEABLE] software ECC based page write function
|
2011-05-26 04:59:01 +07:00
|
|
|
* @mtd: mtd info structure
|
|
|
|
* @chip: nand chip info structure
|
|
|
|
* @buf: data buffer
|
2012-05-03 00:14:55 +07:00
|
|
|
* @oob_required: must write chip->oob_poi to OOB
|
2006-05-23 22:21:03 +07:00
|
|
|
*/
|
2012-06-25 17:07:45 +07:00
|
|
|
static int nand_write_page_swecc(struct mtd_info *mtd, struct nand_chip *chip,
|
2012-05-03 00:14:55 +07:00
|
|
|
const uint8_t *buf, int oob_required)
|
2006-05-23 22:21:03 +07:00
|
|
|
{
|
2006-05-26 23:52:08 +07:00
|
|
|
int i, eccsize = chip->ecc.size;
|
|
|
|
int eccbytes = chip->ecc.bytes;
|
|
|
|
int eccsteps = chip->ecc.steps;
|
2006-09-25 23:08:04 +07:00
|
|
|
uint8_t *ecc_calc = chip->buffers->ecccalc;
|
2006-05-26 23:52:08 +07:00
|
|
|
const uint8_t *p = buf;
|
2007-05-29 01:17:54 +07:00
|
|
|
uint32_t *eccpos = chip->ecc.layout->eccpos;
|
2006-05-23 22:21:03 +07:00
|
|
|
|
2011-06-24 04:12:08 +07:00
|
|
|
/* Software ECC calculation */
|
2006-05-29 08:26:58 +07:00
|
|
|
for (i = 0; eccsteps; eccsteps--, i += eccbytes, p += eccsize)
|
|
|
|
chip->ecc.calculate(mtd, p, &ecc_calc[i]);
|
2006-05-23 22:21:03 +07:00
|
|
|
|
2006-05-29 08:26:58 +07:00
|
|
|
for (i = 0; i < chip->ecc.total; i++)
|
|
|
|
chip->oob_poi[eccpos[i]] = ecc_calc[i];
|
2006-05-23 22:21:03 +07:00
|
|
|
|
2012-06-25 17:07:45 +07:00
|
|
|
return chip->ecc.write_page_raw(mtd, chip, buf, 1);
|
2006-05-26 23:52:08 +07:00
|
|
|
}
|
2006-05-23 22:21:03 +07:00
|
|
|
|
2006-05-26 23:52:08 +07:00
|
|
|
/**
|
2011-06-24 04:12:08 +07:00
|
|
|
* nand_write_page_hwecc - [REPLACEABLE] hardware ECC based page write function
|
2011-05-26 04:59:01 +07:00
|
|
|
* @mtd: mtd info structure
|
|
|
|
* @chip: nand chip info structure
|
|
|
|
* @buf: data buffer
|
2012-05-03 00:14:55 +07:00
|
|
|
* @oob_required: must write chip->oob_poi to OOB
|
2006-05-26 23:52:08 +07:00
|
|
|
*/
|
2012-06-25 17:07:45 +07:00
|
|
|
static int nand_write_page_hwecc(struct mtd_info *mtd, struct nand_chip *chip,
|
2012-05-03 00:14:55 +07:00
|
|
|
const uint8_t *buf, int oob_required)
|
2006-05-26 23:52:08 +07:00
|
|
|
{
|
|
|
|
int i, eccsize = chip->ecc.size;
|
|
|
|
int eccbytes = chip->ecc.bytes;
|
|
|
|
int eccsteps = chip->ecc.steps;
|
2006-09-25 23:08:04 +07:00
|
|
|
uint8_t *ecc_calc = chip->buffers->ecccalc;
|
2006-05-26 23:52:08 +07:00
|
|
|
const uint8_t *p = buf;
|
2007-05-29 01:17:54 +07:00
|
|
|
uint32_t *eccpos = chip->ecc.layout->eccpos;
|
2006-05-23 22:21:03 +07:00
|
|
|
|
2006-05-26 23:52:08 +07:00
|
|
|
for (i = 0; eccsteps; eccsteps--, i += eccbytes, p += eccsize) {
|
|
|
|
chip->ecc.hwctl(mtd, NAND_ECC_WRITE);
|
2006-05-27 05:05:44 +07:00
|
|
|
chip->write_buf(mtd, p, eccsize);
|
2006-05-26 23:52:08 +07:00
|
|
|
chip->ecc.calculate(mtd, p, &ecc_calc[i]);
|
2006-05-23 22:21:03 +07:00
|
|
|
}
|
|
|
|
|
2006-05-26 23:52:08 +07:00
|
|
|
for (i = 0; i < chip->ecc.total; i++)
|
|
|
|
chip->oob_poi[eccpos[i]] = ecc_calc[i];
|
|
|
|
|
|
|
|
chip->write_buf(mtd, chip->oob_poi, mtd->oobsize);
|
2012-06-25 17:07:45 +07:00
|
|
|
|
|
|
|
return 0;
|
2006-05-23 22:21:03 +07:00
|
|
|
}
|
|
|
|
|
2013-03-15 19:25:53 +07:00
|
|
|
|
|
|
|
/**
|
|
|
|
* nand_write_subpage_hwecc - [REPLACABLE] hardware ECC based subpage write
|
|
|
|
* @mtd: mtd info structure
|
|
|
|
* @chip: nand chip info structure
|
2013-08-09 07:16:36 +07:00
|
|
|
* @offset: column address of subpage within the page
|
2013-03-15 19:25:53 +07:00
|
|
|
* @data_len: data length
|
2013-08-09 07:16:36 +07:00
|
|
|
* @buf: data buffer
|
2013-03-15 19:25:53 +07:00
|
|
|
* @oob_required: must write chip->oob_poi to OOB
|
|
|
|
*/
|
|
|
|
static int nand_write_subpage_hwecc(struct mtd_info *mtd,
|
|
|
|
struct nand_chip *chip, uint32_t offset,
|
2013-08-09 07:16:36 +07:00
|
|
|
uint32_t data_len, const uint8_t *buf,
|
2013-03-15 19:25:53 +07:00
|
|
|
int oob_required)
|
|
|
|
{
|
|
|
|
uint8_t *oob_buf = chip->oob_poi;
|
|
|
|
uint8_t *ecc_calc = chip->buffers->ecccalc;
|
|
|
|
int ecc_size = chip->ecc.size;
|
|
|
|
int ecc_bytes = chip->ecc.bytes;
|
|
|
|
int ecc_steps = chip->ecc.steps;
|
|
|
|
uint32_t *eccpos = chip->ecc.layout->eccpos;
|
|
|
|
uint32_t start_step = offset / ecc_size;
|
|
|
|
uint32_t end_step = (offset + data_len - 1) / ecc_size;
|
|
|
|
int oob_bytes = mtd->oobsize / ecc_steps;
|
|
|
|
int step, i;
|
|
|
|
|
|
|
|
for (step = 0; step < ecc_steps; step++) {
|
|
|
|
/* configure controller for WRITE access */
|
|
|
|
chip->ecc.hwctl(mtd, NAND_ECC_WRITE);
|
|
|
|
|
|
|
|
/* write data (untouched subpages already masked by 0xFF) */
|
2013-08-09 07:16:36 +07:00
|
|
|
chip->write_buf(mtd, buf, ecc_size);
|
2013-03-15 19:25:53 +07:00
|
|
|
|
|
|
|
/* mask ECC of un-touched subpages by padding 0xFF */
|
|
|
|
if ((step < start_step) || (step > end_step))
|
|
|
|
memset(ecc_calc, 0xff, ecc_bytes);
|
|
|
|
else
|
2013-08-09 07:16:36 +07:00
|
|
|
chip->ecc.calculate(mtd, buf, ecc_calc);
|
2013-03-15 19:25:53 +07:00
|
|
|
|
|
|
|
/* mask OOB of un-touched subpages by padding 0xFF */
|
|
|
|
/* if oob_required, preserve OOB metadata of written subpage */
|
|
|
|
if (!oob_required || (step < start_step) || (step > end_step))
|
|
|
|
memset(oob_buf, 0xff, oob_bytes);
|
|
|
|
|
2013-08-09 07:16:36 +07:00
|
|
|
buf += ecc_size;
|
2013-03-15 19:25:53 +07:00
|
|
|
ecc_calc += ecc_bytes;
|
|
|
|
oob_buf += oob_bytes;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* copy calculated ECC for whole page to chip->buffer->oob */
|
|
|
|
/* this include masked-value(0xFF) for unwritten subpages */
|
|
|
|
ecc_calc = chip->buffers->ecccalc;
|
|
|
|
for (i = 0; i < chip->ecc.total; i++)
|
|
|
|
chip->oob_poi[eccpos[i]] = ecc_calc[i];
|
|
|
|
|
|
|
|
/* write OOB buffer to NAND device */
|
|
|
|
chip->write_buf(mtd, chip->oob_poi, mtd->oobsize);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2005-11-07 18:15:49 +07:00
|
|
|
/**
|
2011-06-24 04:12:08 +07:00
|
|
|
* nand_write_page_syndrome - [REPLACEABLE] hardware ECC syndrome based page write
|
2011-05-26 04:59:01 +07:00
|
|
|
* @mtd: mtd info structure
|
|
|
|
* @chip: nand chip info structure
|
|
|
|
* @buf: data buffer
|
2012-05-03 00:14:55 +07:00
|
|
|
* @oob_required: must write chip->oob_poi to OOB
|
2005-04-17 05:20:36 +07:00
|
|
|
*
|
2011-05-26 04:59:01 +07:00
|
|
|
* The hw generator calculates the error syndrome automatically. Therefore we
|
|
|
|
* need a special oob layout and handling.
|
2006-05-26 23:52:08 +07:00
|
|
|
*/
|
2012-06-25 17:07:45 +07:00
|
|
|
static int nand_write_page_syndrome(struct mtd_info *mtd,
|
2012-05-03 00:14:55 +07:00
|
|
|
struct nand_chip *chip,
|
|
|
|
const uint8_t *buf, int oob_required)
|
2005-04-17 05:20:36 +07:00
|
|
|
{
|
2006-05-26 23:52:08 +07:00
|
|
|
int i, eccsize = chip->ecc.size;
|
|
|
|
int eccbytes = chip->ecc.bytes;
|
|
|
|
int eccsteps = chip->ecc.steps;
|
|
|
|
const uint8_t *p = buf;
|
|
|
|
uint8_t *oob = chip->oob_poi;
|
2005-04-17 05:20:36 +07:00
|
|
|
|
2006-05-26 23:52:08 +07:00
|
|
|
for (i = 0; eccsteps; eccsteps--, i += eccbytes, p += eccsize) {
|
2005-04-17 05:20:36 +07:00
|
|
|
|
2006-05-26 23:52:08 +07:00
|
|
|
chip->ecc.hwctl(mtd, NAND_ECC_WRITE);
|
|
|
|
chip->write_buf(mtd, p, eccsize);
|
2005-11-07 18:15:49 +07:00
|
|
|
|
2006-05-26 23:52:08 +07:00
|
|
|
if (chip->ecc.prepad) {
|
|
|
|
chip->write_buf(mtd, oob, chip->ecc.prepad);
|
|
|
|
oob += chip->ecc.prepad;
|
|
|
|
}
|
|
|
|
|
|
|
|
chip->ecc.calculate(mtd, p, oob);
|
|
|
|
chip->write_buf(mtd, oob, eccbytes);
|
|
|
|
oob += eccbytes;
|
|
|
|
|
|
|
|
if (chip->ecc.postpad) {
|
|
|
|
chip->write_buf(mtd, oob, chip->ecc.postpad);
|
|
|
|
oob += chip->ecc.postpad;
|
2005-04-17 05:20:36 +07:00
|
|
|
}
|
|
|
|
}
|
2006-05-26 23:52:08 +07:00
|
|
|
|
|
|
|
/* Calculate remaining oob bytes */
|
2006-06-07 12:34:37 +07:00
|
|
|
i = mtd->oobsize - (oob - chip->oob_poi);
|
2006-05-26 23:52:08 +07:00
|
|
|
if (i)
|
|
|
|
chip->write_buf(mtd, oob, i);
|
2012-06-25 17:07:45 +07:00
|
|
|
|
|
|
|
return 0;
|
2006-05-26 23:52:08 +07:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2006-09-25 23:12:39 +07:00
|
|
|
* nand_write_page - [REPLACEABLE] write one page
|
2011-05-26 04:59:01 +07:00
|
|
|
* @mtd: MTD device structure
|
|
|
|
* @chip: NAND chip descriptor
|
2013-03-15 19:25:53 +07:00
|
|
|
* @offset: address offset within the page
|
|
|
|
* @data_len: length of actual data to be written
|
2011-05-26 04:59:01 +07:00
|
|
|
* @buf: the data to write
|
2012-05-03 00:14:55 +07:00
|
|
|
* @oob_required: must write chip->oob_poi to OOB
|
2011-05-26 04:59:01 +07:00
|
|
|
* @page: page number to write
|
|
|
|
* @cached: cached programming
|
|
|
|
* @raw: use _raw version of write_page
|
2006-05-26 23:52:08 +07:00
|
|
|
*/
|
|
|
|
static int nand_write_page(struct mtd_info *mtd, struct nand_chip *chip,
|
2013-03-15 19:25:53 +07:00
|
|
|
uint32_t offset, int data_len, const uint8_t *buf,
|
|
|
|
int oob_required, int page, int cached, int raw)
|
2006-05-26 23:52:08 +07:00
|
|
|
{
|
2013-03-15 19:25:53 +07:00
|
|
|
int status, subpage;
|
|
|
|
|
|
|
|
if (!(chip->options & NAND_NO_SUBPAGE_WRITE) &&
|
|
|
|
chip->ecc.write_subpage)
|
|
|
|
subpage = offset || (data_len < mtd->writesize);
|
|
|
|
else
|
|
|
|
subpage = 0;
|
2006-05-26 23:52:08 +07:00
|
|
|
|
|
|
|
chip->cmdfunc(mtd, NAND_CMD_SEQIN, 0x00, page);
|
|
|
|
|
2006-09-25 23:12:39 +07:00
|
|
|
if (unlikely(raw))
|
2013-03-15 19:25:53 +07:00
|
|
|
status = chip->ecc.write_page_raw(mtd, chip, buf,
|
|
|
|
oob_required);
|
|
|
|
else if (subpage)
|
|
|
|
status = chip->ecc.write_subpage(mtd, chip, offset, data_len,
|
|
|
|
buf, oob_required);
|
2006-09-25 23:12:39 +07:00
|
|
|
else
|
2012-06-25 17:07:45 +07:00
|
|
|
status = chip->ecc.write_page(mtd, chip, buf, oob_required);
|
|
|
|
|
|
|
|
if (status < 0)
|
|
|
|
return status;
|
2006-05-26 23:52:08 +07:00
|
|
|
|
|
|
|
/*
|
2011-06-24 04:12:08 +07:00
|
|
|
* Cached progamming disabled for now. Not sure if it's worth the
|
2011-05-26 04:59:01 +07:00
|
|
|
* trouble. The speed gain is not very impressive. (2.3->2.6Mib/s).
|
2006-05-26 23:52:08 +07:00
|
|
|
*/
|
|
|
|
cached = 0;
|
|
|
|
|
2013-03-04 19:56:18 +07:00
|
|
|
if (!cached || !NAND_HAS_CACHEPROG(chip)) {
|
2006-05-26 23:52:08 +07:00
|
|
|
|
|
|
|
chip->cmdfunc(mtd, NAND_CMD_PAGEPROG, -1, -1);
|
2006-06-21 01:05:05 +07:00
|
|
|
status = chip->waitfunc(mtd, chip);
|
2006-05-26 23:52:08 +07:00
|
|
|
/*
|
|
|
|
* See if operation failed and additional status checks are
|
2011-05-26 04:59:01 +07:00
|
|
|
* available.
|
2006-05-26 23:52:08 +07:00
|
|
|
*/
|
|
|
|
if ((status & NAND_STATUS_FAIL) && (chip->errstat))
|
|
|
|
status = chip->errstat(mtd, chip, FL_WRITING, status,
|
|
|
|
page);
|
|
|
|
|
|
|
|
if (status & NAND_STATUS_FAIL)
|
|
|
|
return -EIO;
|
|
|
|
} else {
|
|
|
|
chip->cmdfunc(mtd, NAND_CMD_CACHEDPROG, -1, -1);
|
2006-06-21 01:05:05 +07:00
|
|
|
status = chip->waitfunc(mtd, chip);
|
2006-05-26 23:52:08 +07:00
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
2005-04-17 05:20:36 +07:00
|
|
|
}
|
|
|
|
|
2006-05-29 08:26:58 +07:00
|
|
|
/**
|
2011-06-24 04:12:08 +07:00
|
|
|
* nand_fill_oob - [INTERN] Transfer client buffer to oob
|
2011-06-14 21:52:38 +07:00
|
|
|
* @mtd: MTD device structure
|
2011-05-26 04:59:01 +07:00
|
|
|
* @oob: oob data buffer
|
|
|
|
* @len: oob data write length
|
|
|
|
* @ops: oob ops structure
|
2006-05-29 08:26:58 +07:00
|
|
|
*/
|
2011-06-14 21:52:38 +07:00
|
|
|
static uint8_t *nand_fill_oob(struct mtd_info *mtd, uint8_t *oob, size_t len,
|
|
|
|
struct mtd_oob_ops *ops)
|
2006-05-29 08:26:58 +07:00
|
|
|
{
|
2011-06-14 21:52:38 +07:00
|
|
|
struct nand_chip *chip = mtd->priv;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Initialise to all 0xFF, to avoid the possibility of left over OOB
|
|
|
|
* data from a previous OOB read.
|
|
|
|
*/
|
|
|
|
memset(chip->oob_poi, 0xff, mtd->oobsize);
|
|
|
|
|
2010-09-07 18:23:43 +07:00
|
|
|
switch (ops->mode) {
|
2006-05-29 08:26:58 +07:00
|
|
|
|
2011-08-31 08:45:40 +07:00
|
|
|
case MTD_OPS_PLACE_OOB:
|
|
|
|
case MTD_OPS_RAW:
|
2006-05-29 08:26:58 +07:00
|
|
|
memcpy(chip->oob_poi + ops->ooboffs, oob, len);
|
|
|
|
return oob + len;
|
|
|
|
|
2011-08-31 08:45:40 +07:00
|
|
|
case MTD_OPS_AUTO_OOB: {
|
2006-05-29 08:26:58 +07:00
|
|
|
struct nand_oobfree *free = chip->ecc.layout->oobfree;
|
2006-06-21 01:05:05 +07:00
|
|
|
uint32_t boffs = 0, woffs = ops->ooboffs;
|
|
|
|
size_t bytes = 0;
|
2006-05-29 08:26:58 +07:00
|
|
|
|
2010-09-07 18:23:43 +07:00
|
|
|
for (; free->length && len; free++, len -= bytes) {
|
2011-05-26 04:59:01 +07:00
|
|
|
/* Write request not from offset 0? */
|
2006-06-21 01:05:05 +07:00
|
|
|
if (unlikely(woffs)) {
|
|
|
|
if (woffs >= free->length) {
|
|
|
|
woffs -= free->length;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
boffs = free->offset + woffs;
|
|
|
|
bytes = min_t(size_t, len,
|
|
|
|
(free->length - woffs));
|
|
|
|
woffs = 0;
|
|
|
|
} else {
|
|
|
|
bytes = min_t(size_t, len, free->length);
|
|
|
|
boffs = free->offset;
|
|
|
|
}
|
2006-07-11 14:11:25 +07:00
|
|
|
memcpy(chip->oob_poi + boffs, oob, bytes);
|
2006-05-29 08:26:58 +07:00
|
|
|
oob += bytes;
|
|
|
|
}
|
|
|
|
return oob;
|
|
|
|
}
|
|
|
|
default:
|
|
|
|
BUG();
|
|
|
|
}
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2010-09-07 18:23:43 +07:00
|
|
|
#define NOTALIGNED(x) ((x & (chip->subpagesize - 1)) != 0)
|
2005-04-17 05:20:36 +07:00
|
|
|
|
|
|
|
/**
|
2011-06-24 04:12:08 +07:00
|
|
|
* nand_do_write_ops - [INTERN] NAND write with ECC
|
2011-05-26 04:59:01 +07:00
|
|
|
* @mtd: MTD device structure
|
|
|
|
* @to: offset to write to
|
|
|
|
* @ops: oob operations description structure
|
2005-04-17 05:20:36 +07:00
|
|
|
*
|
2011-05-26 04:59:01 +07:00
|
|
|
* NAND write with ECC.
|
2005-04-17 05:20:36 +07:00
|
|
|
*/
|
2006-05-29 08:26:58 +07:00
|
|
|
static int nand_do_write_ops(struct mtd_info *mtd, loff_t to,
|
|
|
|
struct mtd_oob_ops *ops)
|
2005-04-17 05:20:36 +07:00
|
|
|
{
|
2006-09-28 20:38:36 +07:00
|
|
|
int chipnr, realpage, page, blockmask, column;
|
2006-05-24 17:07:37 +07:00
|
|
|
struct nand_chip *chip = mtd->priv;
|
2006-05-29 08:26:58 +07:00
|
|
|
uint32_t writelen = ops->len;
|
2010-02-23 01:39:36 +07:00
|
|
|
|
|
|
|
uint32_t oobwritelen = ops->ooblen;
|
2011-08-31 08:45:40 +07:00
|
|
|
uint32_t oobmaxlen = ops->mode == MTD_OPS_AUTO_OOB ?
|
2010-02-23 01:39:36 +07:00
|
|
|
mtd->oobavail : mtd->oobsize;
|
|
|
|
|
2006-05-29 08:26:58 +07:00
|
|
|
uint8_t *oob = ops->oobbuf;
|
|
|
|
uint8_t *buf = ops->datbuf;
|
2013-03-15 19:25:53 +07:00
|
|
|
int ret;
|
2012-05-03 00:14:56 +07:00
|
|
|
int oob_required = oob ? 1 : 0;
|
2005-04-17 05:20:36 +07:00
|
|
|
|
2006-05-29 08:26:58 +07:00
|
|
|
ops->retlen = 0;
|
2006-09-28 20:38:36 +07:00
|
|
|
if (!writelen)
|
|
|
|
return 0;
|
2005-04-17 05:20:36 +07:00
|
|
|
|
2011-05-26 04:59:01 +07:00
|
|
|
/* Reject writes, which are not page aligned */
|
2006-05-29 08:26:58 +07:00
|
|
|
if (NOTALIGNED(to) || NOTALIGNED(ops->len)) {
|
2011-07-20 00:06:08 +07:00
|
|
|
pr_notice("%s: attempt to write non page aligned data\n",
|
|
|
|
__func__);
|
2005-04-17 05:20:36 +07:00
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
2006-09-28 20:38:36 +07:00
|
|
|
column = to & (mtd->writesize - 1);
|
2005-04-17 05:20:36 +07:00
|
|
|
|
2006-06-28 05:11:45 +07:00
|
|
|
chipnr = (int)(to >> chip->chip_shift);
|
|
|
|
chip->select_chip(mtd, chipnr);
|
|
|
|
|
2005-04-17 05:20:36 +07:00
|
|
|
/* Check, if it is write protected */
|
2012-11-19 13:43:29 +07:00
|
|
|
if (nand_check_wp(mtd)) {
|
|
|
|
ret = -EIO;
|
|
|
|
goto err_out;
|
|
|
|
}
|
2005-04-17 05:20:36 +07:00
|
|
|
|
2006-05-26 23:52:08 +07:00
|
|
|
realpage = (int)(to >> chip->page_shift);
|
|
|
|
page = realpage & chip->pagemask;
|
|
|
|
blockmask = (1 << (chip->phys_erase_shift - chip->page_shift)) - 1;
|
|
|
|
|
|
|
|
/* Invalidate the page cache, when we write to the cached page */
|
|
|
|
if (to <= (chip->pagebuf << chip->page_shift) &&
|
2006-05-29 08:26:58 +07:00
|
|
|
(chip->pagebuf << chip->page_shift) < (to + ops->len))
|
2006-05-24 17:07:37 +07:00
|
|
|
chip->pagebuf = -1;
|
2005-11-07 18:15:49 +07:00
|
|
|
|
2010-02-23 01:39:36 +07:00
|
|
|
/* Don't allow multipage oob writes with offset */
|
2012-11-19 13:43:29 +07:00
|
|
|
if (oob && ops->ooboffs && (ops->ooboffs + ops->ooblen > oobmaxlen)) {
|
|
|
|
ret = -EINVAL;
|
|
|
|
goto err_out;
|
|
|
|
}
|
2010-02-23 01:39:36 +07:00
|
|
|
|
2010-09-07 18:23:43 +07:00
|
|
|
while (1) {
|
2006-09-28 20:38:36 +07:00
|
|
|
int bytes = mtd->writesize;
|
2006-05-26 23:52:08 +07:00
|
|
|
int cached = writelen > bytes && page != blockmask;
|
2006-09-28 20:38:36 +07:00
|
|
|
uint8_t *wbuf = buf;
|
|
|
|
|
2011-05-26 04:59:01 +07:00
|
|
|
/* Partial page write? */
|
2006-09-28 20:38:36 +07:00
|
|
|
if (unlikely(column || writelen < (mtd->writesize - 1))) {
|
|
|
|
cached = 0;
|
|
|
|
bytes = min_t(int, bytes - column, (int) writelen);
|
|
|
|
chip->pagebuf = -1;
|
|
|
|
memset(chip->buffers->databuf, 0xff, mtd->writesize);
|
|
|
|
memcpy(&chip->buffers->databuf[column], buf, bytes);
|
|
|
|
wbuf = chip->buffers->databuf;
|
|
|
|
}
|
2005-04-17 05:20:36 +07:00
|
|
|
|
2010-02-23 01:39:36 +07:00
|
|
|
if (unlikely(oob)) {
|
|
|
|
size_t len = min(oobwritelen, oobmaxlen);
|
2011-06-14 21:52:38 +07:00
|
|
|
oob = nand_fill_oob(mtd, oob, len, ops);
|
2010-02-23 01:39:36 +07:00
|
|
|
oobwritelen -= len;
|
2011-06-14 21:52:38 +07:00
|
|
|
} else {
|
|
|
|
/* We still need to erase leftover OOB data */
|
|
|
|
memset(chip->oob_poi, 0xff, mtd->oobsize);
|
2010-02-23 01:39:36 +07:00
|
|
|
}
|
2013-03-15 19:25:53 +07:00
|
|
|
ret = chip->write_page(mtd, chip, column, bytes, wbuf,
|
|
|
|
oob_required, page, cached,
|
|
|
|
(ops->mode == MTD_OPS_RAW));
|
2006-05-26 23:52:08 +07:00
|
|
|
if (ret)
|
|
|
|
break;
|
|
|
|
|
|
|
|
writelen -= bytes;
|
|
|
|
if (!writelen)
|
|
|
|
break;
|
|
|
|
|
2006-09-28 20:38:36 +07:00
|
|
|
column = 0;
|
2006-05-26 23:52:08 +07:00
|
|
|
buf += bytes;
|
|
|
|
realpage++;
|
|
|
|
|
|
|
|
page = realpage & chip->pagemask;
|
|
|
|
/* Check, if we cross a chip boundary */
|
|
|
|
if (!page) {
|
|
|
|
chipnr++;
|
|
|
|
chip->select_chip(mtd, -1);
|
|
|
|
chip->select_chip(mtd, chipnr);
|
2005-04-17 05:20:36 +07:00
|
|
|
}
|
|
|
|
}
|
2006-05-29 08:26:58 +07:00
|
|
|
|
|
|
|
ops->retlen = ops->len - writelen;
|
2006-11-03 22:20:38 +07:00
|
|
|
if (unlikely(oob))
|
|
|
|
ops->oobretlen = ops->ooblen;
|
2012-11-19 13:43:29 +07:00
|
|
|
|
|
|
|
err_out:
|
|
|
|
chip->select_chip(mtd, -1);
|
2005-04-17 05:20:36 +07:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2009-10-05 20:55:52 +07:00
|
|
|
/**
|
|
|
|
* panic_nand_write - [MTD Interface] NAND write with ECC
|
2011-05-26 04:59:01 +07:00
|
|
|
* @mtd: MTD device structure
|
|
|
|
* @to: offset to write to
|
|
|
|
* @len: number of bytes to write
|
|
|
|
* @retlen: pointer to variable to store the number of written bytes
|
|
|
|
* @buf: the data to write
|
2009-10-05 20:55:52 +07:00
|
|
|
*
|
|
|
|
* NAND write with ECC. Used when performing writes in interrupt context, this
|
|
|
|
* may for example be called by mtdoops when writing an oops while in panic.
|
|
|
|
*/
|
|
|
|
static int panic_nand_write(struct mtd_info *mtd, loff_t to, size_t len,
|
|
|
|
size_t *retlen, const uint8_t *buf)
|
|
|
|
{
|
|
|
|
struct nand_chip *chip = mtd->priv;
|
2011-08-31 08:45:45 +07:00
|
|
|
struct mtd_oob_ops ops;
|
2009-10-05 20:55:52 +07:00
|
|
|
int ret;
|
|
|
|
|
2011-05-26 04:59:01 +07:00
|
|
|
/* Wait for the device to get ready */
|
2009-10-05 20:55:52 +07:00
|
|
|
panic_nand_wait(mtd, chip, 400);
|
|
|
|
|
2011-05-26 04:59:01 +07:00
|
|
|
/* Grab the device */
|
2009-10-05 20:55:52 +07:00
|
|
|
panic_nand_get_device(chip, mtd, FL_WRITING);
|
|
|
|
|
2011-08-31 08:45:45 +07:00
|
|
|
ops.len = len;
|
|
|
|
ops.datbuf = (uint8_t *)buf;
|
|
|
|
ops.oobbuf = NULL;
|
2012-07-03 15:44:14 +07:00
|
|
|
ops.mode = MTD_OPS_PLACE_OOB;
|
2009-10-05 20:55:52 +07:00
|
|
|
|
2011-08-31 08:45:45 +07:00
|
|
|
ret = nand_do_write_ops(mtd, to, &ops);
|
2009-10-05 20:55:52 +07:00
|
|
|
|
2011-08-31 08:45:45 +07:00
|
|
|
*retlen = ops.retlen;
|
2009-10-05 20:55:52 +07:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2006-05-26 23:52:08 +07:00
|
|
|
/**
|
2006-05-29 08:26:58 +07:00
|
|
|
* nand_write - [MTD Interface] NAND write with ECC
|
2011-05-26 04:59:01 +07:00
|
|
|
* @mtd: MTD device structure
|
|
|
|
* @to: offset to write to
|
|
|
|
* @len: number of bytes to write
|
|
|
|
* @retlen: pointer to variable to store the number of written bytes
|
|
|
|
* @buf: the data to write
|
2006-05-26 23:52:08 +07:00
|
|
|
*
|
2011-05-26 04:59:01 +07:00
|
|
|
* NAND write with ECC.
|
2006-05-26 23:52:08 +07:00
|
|
|
*/
|
2006-05-29 08:26:58 +07:00
|
|
|
static int nand_write(struct mtd_info *mtd, loff_t to, size_t len,
|
|
|
|
size_t *retlen, const uint8_t *buf)
|
2006-05-26 23:52:08 +07:00
|
|
|
{
|
2011-08-31 08:45:45 +07:00
|
|
|
struct mtd_oob_ops ops;
|
2006-05-26 23:52:08 +07:00
|
|
|
int ret;
|
|
|
|
|
2012-11-19 13:43:30 +07:00
|
|
|
nand_get_device(mtd, FL_WRITING);
|
2011-08-31 08:45:45 +07:00
|
|
|
ops.len = len;
|
|
|
|
ops.datbuf = (uint8_t *)buf;
|
|
|
|
ops.oobbuf = NULL;
|
2012-07-03 15:44:14 +07:00
|
|
|
ops.mode = MTD_OPS_PLACE_OOB;
|
2011-08-31 08:45:45 +07:00
|
|
|
ret = nand_do_write_ops(mtd, to, &ops);
|
|
|
|
*retlen = ops.retlen;
|
2006-05-26 23:52:08 +07:00
|
|
|
nand_release_device(mtd);
|
2006-05-29 08:26:58 +07:00
|
|
|
return ret;
|
2006-05-26 23:52:08 +07:00
|
|
|
}
|
2006-05-25 14:51:54 +07:00
|
|
|
|
2005-04-17 05:20:36 +07:00
|
|
|
/**
|
2006-05-29 08:26:58 +07:00
|
|
|
* nand_do_write_oob - [MTD Interface] NAND write out-of-band
|
2011-05-26 04:59:01 +07:00
|
|
|
* @mtd: MTD device structure
|
|
|
|
* @to: offset to write to
|
|
|
|
* @ops: oob operation description structure
|
2005-04-17 05:20:36 +07:00
|
|
|
*
|
2011-05-26 04:59:01 +07:00
|
|
|
* NAND write out-of-band.
|
2005-04-17 05:20:36 +07:00
|
|
|
*/
|
2006-05-29 08:26:58 +07:00
|
|
|
static int nand_do_write_oob(struct mtd_info *mtd, loff_t to,
|
|
|
|
struct mtd_oob_ops *ops)
|
2005-04-17 05:20:36 +07:00
|
|
|
{
|
2007-01-31 22:58:29 +07:00
|
|
|
int chipnr, page, status, len;
|
2006-05-24 17:07:37 +07:00
|
|
|
struct nand_chip *chip = mtd->priv;
|
2005-04-17 05:20:36 +07:00
|
|
|
|
2011-07-20 00:06:09 +07:00
|
|
|
pr_debug("%s: to = 0x%08x, len = %i\n",
|
2009-07-07 17:19:49 +07:00
|
|
|
__func__, (unsigned int)to, (int)ops->ooblen);
|
2005-04-17 05:20:36 +07:00
|
|
|
|
2011-08-31 08:45:40 +07:00
|
|
|
if (ops->mode == MTD_OPS_AUTO_OOB)
|
2007-01-31 22:58:29 +07:00
|
|
|
len = chip->ecc.layout->oobavail;
|
|
|
|
else
|
|
|
|
len = mtd->oobsize;
|
|
|
|
|
2005-04-17 05:20:36 +07:00
|
|
|
/* Do not allow write past end of page */
|
2007-01-31 22:58:29 +07:00
|
|
|
if ((ops->ooboffs + ops->ooblen) > len) {
|
2011-07-20 00:06:09 +07:00
|
|
|
pr_debug("%s: attempt to write past end of page\n",
|
|
|
|
__func__);
|
2005-04-17 05:20:36 +07:00
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
2007-01-31 22:58:29 +07:00
|
|
|
if (unlikely(ops->ooboffs >= len)) {
|
2011-07-20 00:06:09 +07:00
|
|
|
pr_debug("%s: attempt to start write outside oob\n",
|
|
|
|
__func__);
|
2007-01-31 22:58:29 +07:00
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
2011-02-25 12:06:18 +07:00
|
|
|
/* Do not allow write past end of device */
|
2007-01-31 22:58:29 +07:00
|
|
|
if (unlikely(to >= mtd->size ||
|
|
|
|
ops->ooboffs + ops->ooblen >
|
|
|
|
((mtd->size >> chip->page_shift) -
|
|
|
|
(to >> chip->page_shift)) * len)) {
|
2011-07-20 00:06:09 +07:00
|
|
|
pr_debug("%s: attempt to write beyond end of device\n",
|
|
|
|
__func__);
|
2007-01-31 22:58:29 +07:00
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
2006-05-25 14:51:54 +07:00
|
|
|
chipnr = (int)(to >> chip->chip_shift);
|
2006-05-24 17:07:37 +07:00
|
|
|
chip->select_chip(mtd, chipnr);
|
2005-04-17 05:20:36 +07:00
|
|
|
|
2006-05-25 14:51:54 +07:00
|
|
|
/* Shift to get page */
|
|
|
|
page = (int)(to >> chip->page_shift);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Reset the chip. Some chips (like the Toshiba TC5832DC found in one
|
|
|
|
* of my DiskOnChip 2000 test units) will clear the whole data page too
|
|
|
|
* if we don't do this. I have no clue why, but I seem to have 'fixed'
|
|
|
|
* it in the doc2000 driver in August 1999. dwmw2.
|
|
|
|
*/
|
2006-05-24 17:07:37 +07:00
|
|
|
chip->cmdfunc(mtd, NAND_CMD_RESET, -1, -1);
|
2005-04-17 05:20:36 +07:00
|
|
|
|
|
|
|
/* Check, if it is write protected */
|
2012-11-19 13:43:29 +07:00
|
|
|
if (nand_check_wp(mtd)) {
|
|
|
|
chip->select_chip(mtd, -1);
|
2006-05-29 08:26:58 +07:00
|
|
|
return -EROFS;
|
2012-11-19 13:43:29 +07:00
|
|
|
}
|
2005-11-07 18:15:49 +07:00
|
|
|
|
2005-04-17 05:20:36 +07:00
|
|
|
/* Invalidate the page cache, if we write to the cached page */
|
2006-05-24 17:07:37 +07:00
|
|
|
if (page == chip->pagebuf)
|
|
|
|
chip->pagebuf = -1;
|
2005-04-17 05:20:36 +07:00
|
|
|
|
2011-06-14 21:52:38 +07:00
|
|
|
nand_fill_oob(mtd, ops->oobbuf, ops->ooblen, ops);
|
2011-08-31 08:45:37 +07:00
|
|
|
|
2011-08-31 08:45:40 +07:00
|
|
|
if (ops->mode == MTD_OPS_RAW)
|
2011-08-31 08:45:37 +07:00
|
|
|
status = chip->ecc.write_oob_raw(mtd, chip, page & chip->pagemask);
|
|
|
|
else
|
|
|
|
status = chip->ecc.write_oob(mtd, chip, page & chip->pagemask);
|
2005-04-17 05:20:36 +07:00
|
|
|
|
2012-11-19 13:43:29 +07:00
|
|
|
chip->select_chip(mtd, -1);
|
|
|
|
|
2006-06-21 01:05:05 +07:00
|
|
|
if (status)
|
|
|
|
return status;
|
2005-04-17 05:20:36 +07:00
|
|
|
|
2006-11-03 22:20:38 +07:00
|
|
|
ops->oobretlen = ops->ooblen;
|
2005-04-17 05:20:36 +07:00
|
|
|
|
2006-06-21 01:05:05 +07:00
|
|
|
return 0;
|
2006-05-29 08:26:58 +07:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* nand_write_oob - [MTD Interface] NAND write data and/or out-of-band
|
2011-05-26 04:59:01 +07:00
|
|
|
* @mtd: MTD device structure
|
|
|
|
* @to: offset to write to
|
|
|
|
* @ops: oob operation description structure
|
2006-05-29 08:26:58 +07:00
|
|
|
*/
|
|
|
|
static int nand_write_oob(struct mtd_info *mtd, loff_t to,
|
|
|
|
struct mtd_oob_ops *ops)
|
|
|
|
{
|
|
|
|
int ret = -ENOTSUPP;
|
|
|
|
|
|
|
|
ops->retlen = 0;
|
|
|
|
|
|
|
|
/* Do not allow writes past end of device */
|
2006-11-03 22:20:38 +07:00
|
|
|
if (ops->datbuf && (to + ops->len) > mtd->size) {
|
2011-07-20 00:06:09 +07:00
|
|
|
pr_debug("%s: attempt to write beyond end of device\n",
|
|
|
|
__func__);
|
2006-05-29 08:26:58 +07:00
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
2012-11-19 13:43:30 +07:00
|
|
|
nand_get_device(mtd, FL_WRITING);
|
2006-05-29 08:26:58 +07:00
|
|
|
|
2010-09-07 18:23:43 +07:00
|
|
|
switch (ops->mode) {
|
2011-08-31 08:45:40 +07:00
|
|
|
case MTD_OPS_PLACE_OOB:
|
|
|
|
case MTD_OPS_AUTO_OOB:
|
|
|
|
case MTD_OPS_RAW:
|
2006-05-29 08:26:58 +07:00
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!ops->datbuf)
|
|
|
|
ret = nand_do_write_oob(mtd, to, ops);
|
|
|
|
else
|
|
|
|
ret = nand_do_write_ops(mtd, to, ops);
|
|
|
|
|
2010-09-07 18:23:45 +07:00
|
|
|
out:
|
2005-04-17 05:20:36 +07:00
|
|
|
nand_release_device(mtd);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2011-06-24 04:12:08 +07:00
|
|
|
* single_erase_cmd - [GENERIC] NAND standard block erase command function
|
2011-05-26 04:59:01 +07:00
|
|
|
* @mtd: MTD device structure
|
|
|
|
* @page: the page address of the block which will be erased
|
2005-04-17 05:20:36 +07:00
|
|
|
*
|
2011-05-26 04:59:01 +07:00
|
|
|
* Standard erase command for NAND chips.
|
2005-04-17 05:20:36 +07:00
|
|
|
*/
|
2006-05-14 00:07:53 +07:00
|
|
|
static void single_erase_cmd(struct mtd_info *mtd, int page)
|
2005-04-17 05:20:36 +07:00
|
|
|
{
|
2006-05-24 17:07:37 +07:00
|
|
|
struct nand_chip *chip = mtd->priv;
|
2005-04-17 05:20:36 +07:00
|
|
|
/* Send commands to erase a block */
|
2006-05-24 17:07:37 +07:00
|
|
|
chip->cmdfunc(mtd, NAND_CMD_ERASE1, -1, page);
|
|
|
|
chip->cmdfunc(mtd, NAND_CMD_ERASE2, -1, -1);
|
2005-04-17 05:20:36 +07:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* nand_erase - [MTD Interface] erase block(s)
|
2011-05-26 04:59:01 +07:00
|
|
|
* @mtd: MTD device structure
|
|
|
|
* @instr: erase instruction
|
2005-04-17 05:20:36 +07:00
|
|
|
*
|
2011-05-26 04:59:01 +07:00
|
|
|
* Erase one ore more blocks.
|
2005-04-17 05:20:36 +07:00
|
|
|
*/
|
2006-05-14 00:07:53 +07:00
|
|
|
static int nand_erase(struct mtd_info *mtd, struct erase_info *instr)
|
2005-04-17 05:20:36 +07:00
|
|
|
{
|
2006-05-14 00:07:53 +07:00
|
|
|
return nand_erase_nand(mtd, instr, 0);
|
2005-04-17 05:20:36 +07:00
|
|
|
}
|
2005-11-07 18:15:49 +07:00
|
|
|
|
2005-04-17 05:20:36 +07:00
|
|
|
/**
|
2011-06-24 04:12:08 +07:00
|
|
|
* nand_erase_nand - [INTERN] erase block(s)
|
2011-05-26 04:59:01 +07:00
|
|
|
* @mtd: MTD device structure
|
|
|
|
* @instr: erase instruction
|
|
|
|
* @allowbbt: allow erasing the bbt area
|
2005-04-17 05:20:36 +07:00
|
|
|
*
|
2011-05-26 04:59:01 +07:00
|
|
|
* Erase one ore more blocks.
|
2005-04-17 05:20:36 +07:00
|
|
|
*/
|
2006-05-24 17:07:37 +07:00
|
|
|
int nand_erase_nand(struct mtd_info *mtd, struct erase_info *instr,
|
|
|
|
int allowbbt)
|
2005-04-17 05:20:36 +07:00
|
|
|
{
|
2008-12-10 20:37:21 +07:00
|
|
|
int page, status, pages_per_block, ret, chipnr;
|
2006-05-24 17:07:37 +07:00
|
|
|
struct nand_chip *chip = mtd->priv;
|
2008-12-10 20:37:21 +07:00
|
|
|
loff_t len;
|
2005-04-17 05:20:36 +07:00
|
|
|
|
2011-07-20 00:06:09 +07:00
|
|
|
pr_debug("%s: start = 0x%012llx, len = %llu\n",
|
|
|
|
__func__, (unsigned long long)instr->addr,
|
|
|
|
(unsigned long long)instr->len);
|
2005-04-17 05:20:36 +07:00
|
|
|
|
2010-02-03 15:42:24 +07:00
|
|
|
if (check_offs_len(mtd, instr->addr, instr->len))
|
2005-04-17 05:20:36 +07:00
|
|
|
return -EINVAL;
|
|
|
|
|
|
|
|
/* Grab the lock and see if the device is available */
|
2012-11-19 13:43:30 +07:00
|
|
|
nand_get_device(mtd, FL_ERASING);
|
2005-04-17 05:20:36 +07:00
|
|
|
|
|
|
|
/* Shift to get first page */
|
2006-05-24 17:07:37 +07:00
|
|
|
page = (int)(instr->addr >> chip->page_shift);
|
|
|
|
chipnr = (int)(instr->addr >> chip->chip_shift);
|
2005-04-17 05:20:36 +07:00
|
|
|
|
|
|
|
/* Calculate pages in each block */
|
2006-05-24 17:07:37 +07:00
|
|
|
pages_per_block = 1 << (chip->phys_erase_shift - chip->page_shift);
|
2005-04-17 05:20:36 +07:00
|
|
|
|
|
|
|
/* Select the NAND device */
|
2006-05-24 17:07:37 +07:00
|
|
|
chip->select_chip(mtd, chipnr);
|
2005-04-17 05:20:36 +07:00
|
|
|
|
|
|
|
/* Check, if it is write protected */
|
|
|
|
if (nand_check_wp(mtd)) {
|
2011-07-20 00:06:09 +07:00
|
|
|
pr_debug("%s: device is write protected!\n",
|
|
|
|
__func__);
|
2005-04-17 05:20:36 +07:00
|
|
|
instr->state = MTD_ERASE_FAILED;
|
|
|
|
goto erase_exit;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Loop through the pages */
|
|
|
|
len = instr->len;
|
|
|
|
|
|
|
|
instr->state = MTD_ERASING;
|
|
|
|
|
|
|
|
while (len) {
|
2011-12-22 05:01:20 +07:00
|
|
|
/* Check if we have a bad block, we do not erase bad blocks! */
|
2006-05-24 17:07:37 +07:00
|
|
|
if (nand_block_checkbad(mtd, ((loff_t) page) <<
|
|
|
|
chip->page_shift, 0, allowbbt)) {
|
2011-07-20 00:06:08 +07:00
|
|
|
pr_warn("%s: attempt to erase a bad block at page 0x%08x\n",
|
|
|
|
__func__, page);
|
2005-04-17 05:20:36 +07:00
|
|
|
instr->state = MTD_ERASE_FAILED;
|
|
|
|
goto erase_exit;
|
|
|
|
}
|
2005-11-07 18:15:49 +07:00
|
|
|
|
2006-05-24 17:07:37 +07:00
|
|
|
/*
|
|
|
|
* Invalidate the page cache, if we erase the block which
|
2011-05-26 04:59:01 +07:00
|
|
|
* contains the current cached page.
|
2006-05-24 17:07:37 +07:00
|
|
|
*/
|
|
|
|
if (page <= chip->pagebuf && chip->pagebuf <
|
|
|
|
(page + pages_per_block))
|
|
|
|
chip->pagebuf = -1;
|
2005-04-17 05:20:36 +07:00
|
|
|
|
2006-05-24 17:07:37 +07:00
|
|
|
chip->erase_cmd(mtd, page & chip->pagemask);
|
2005-11-07 18:15:49 +07:00
|
|
|
|
2006-06-21 01:05:05 +07:00
|
|
|
status = chip->waitfunc(mtd, chip);
|
2005-04-17 05:20:36 +07:00
|
|
|
|
2006-05-24 17:07:37 +07:00
|
|
|
/*
|
|
|
|
* See if operation failed and additional status checks are
|
|
|
|
* available
|
|
|
|
*/
|
|
|
|
if ((status & NAND_STATUS_FAIL) && (chip->errstat))
|
|
|
|
status = chip->errstat(mtd, chip, FL_ERASING,
|
|
|
|
status, page);
|
2005-01-24 10:07:46 +07:00
|
|
|
|
2005-04-17 05:20:36 +07:00
|
|
|
/* See if block erase succeeded */
|
2005-01-24 01:30:53 +07:00
|
|
|
if (status & NAND_STATUS_FAIL) {
|
2011-07-20 00:06:09 +07:00
|
|
|
pr_debug("%s: failed erase, page 0x%08x\n",
|
|
|
|
__func__, page);
|
2005-04-17 05:20:36 +07:00
|
|
|
instr->state = MTD_ERASE_FAILED;
|
2008-12-10 20:37:21 +07:00
|
|
|
instr->fail_addr =
|
|
|
|
((loff_t)page << chip->page_shift);
|
2005-04-17 05:20:36 +07:00
|
|
|
goto erase_exit;
|
|
|
|
}
|
2005-01-18 01:35:25 +07:00
|
|
|
|
2005-04-17 05:20:36 +07:00
|
|
|
/* Increment page address and decrement length */
|
2013-08-09 16:49:05 +07:00
|
|
|
len -= (1ULL << chip->phys_erase_shift);
|
2005-04-17 05:20:36 +07:00
|
|
|
page += pages_per_block;
|
|
|
|
|
|
|
|
/* Check, if we cross a chip boundary */
|
2006-05-24 17:07:37 +07:00
|
|
|
if (len && !(page & chip->pagemask)) {
|
2005-04-17 05:20:36 +07:00
|
|
|
chipnr++;
|
2006-05-24 17:07:37 +07:00
|
|
|
chip->select_chip(mtd, -1);
|
|
|
|
chip->select_chip(mtd, chipnr);
|
2005-04-17 05:20:36 +07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
instr->state = MTD_ERASE_DONE;
|
|
|
|
|
2010-09-07 18:23:45 +07:00
|
|
|
erase_exit:
|
2005-04-17 05:20:36 +07:00
|
|
|
|
|
|
|
ret = instr->state == MTD_ERASE_DONE ? 0 : -EIO;
|
|
|
|
|
|
|
|
/* Deselect and wake up anyone waiting on the device */
|
2012-11-19 13:43:29 +07:00
|
|
|
chip->select_chip(mtd, -1);
|
2005-04-17 05:20:36 +07:00
|
|
|
nand_release_device(mtd);
|
|
|
|
|
2007-10-07 02:01:59 +07:00
|
|
|
/* Do call back function */
|
|
|
|
if (!ret)
|
|
|
|
mtd_erase_callback(instr);
|
|
|
|
|
2005-04-17 05:20:36 +07:00
|
|
|
/* Return more or less happy */
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* nand_sync - [MTD Interface] sync
|
2011-05-26 04:59:01 +07:00
|
|
|
* @mtd: MTD device structure
|
2005-04-17 05:20:36 +07:00
|
|
|
*
|
2011-05-26 04:59:01 +07:00
|
|
|
* Sync is actually a wait for chip ready function.
|
2005-04-17 05:20:36 +07:00
|
|
|
*/
|
2006-05-14 00:07:53 +07:00
|
|
|
static void nand_sync(struct mtd_info *mtd)
|
2005-04-17 05:20:36 +07:00
|
|
|
{
|
2011-07-20 00:06:09 +07:00
|
|
|
pr_debug("%s: called\n", __func__);
|
2005-04-17 05:20:36 +07:00
|
|
|
|
|
|
|
/* Grab the lock and see if the device is available */
|
2012-11-19 13:43:30 +07:00
|
|
|
nand_get_device(mtd, FL_SYNCING);
|
2005-04-17 05:20:36 +07:00
|
|
|
/* Release it and go back */
|
2006-05-14 00:07:53 +07:00
|
|
|
nand_release_device(mtd);
|
2005-04-17 05:20:36 +07:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2006-05-24 17:07:37 +07:00
|
|
|
* nand_block_isbad - [MTD Interface] Check if block at offset is bad
|
2011-05-26 04:59:01 +07:00
|
|
|
* @mtd: MTD device structure
|
|
|
|
* @offs: offset relative to mtd start
|
2005-04-17 05:20:36 +07:00
|
|
|
*/
|
2006-05-24 17:07:37 +07:00
|
|
|
static int nand_block_isbad(struct mtd_info *mtd, loff_t offs)
|
2005-04-17 05:20:36 +07:00
|
|
|
{
|
2006-05-24 17:07:37 +07:00
|
|
|
return nand_block_checkbad(mtd, offs, 1, 0);
|
2005-04-17 05:20:36 +07:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2006-05-24 17:07:37 +07:00
|
|
|
* nand_block_markbad - [MTD Interface] Mark block at the given offset as bad
|
2011-05-26 04:59:01 +07:00
|
|
|
* @mtd: MTD device structure
|
|
|
|
* @ofs: offset relative to mtd start
|
2005-04-17 05:20:36 +07:00
|
|
|
*/
|
2006-05-14 00:07:53 +07:00
|
|
|
static int nand_block_markbad(struct mtd_info *mtd, loff_t ofs)
|
2005-04-17 05:20:36 +07:00
|
|
|
{
|
|
|
|
int ret;
|
|
|
|
|
2010-09-07 18:23:43 +07:00
|
|
|
ret = nand_block_isbad(mtd, ofs);
|
|
|
|
if (ret) {
|
2011-05-26 04:59:01 +07:00
|
|
|
/* If it was bad already, return success and do nothing */
|
2005-04-17 05:20:36 +07:00
|
|
|
if (ret > 0)
|
|
|
|
return 0;
|
2006-05-14 00:07:53 +07:00
|
|
|
return ret;
|
|
|
|
}
|
2005-04-17 05:20:36 +07:00
|
|
|
|
2013-07-31 07:52:58 +07:00
|
|
|
return nand_block_markbad_lowlevel(mtd, ofs);
|
2005-04-17 05:20:36 +07:00
|
|
|
}
|
|
|
|
|
2012-09-13 13:57:52 +07:00
|
|
|
/**
|
|
|
|
* nand_onfi_set_features- [REPLACEABLE] set features for ONFI nand
|
|
|
|
* @mtd: MTD device structure
|
|
|
|
* @chip: nand chip info structure
|
|
|
|
* @addr: feature address.
|
|
|
|
* @subfeature_param: the subfeature parameters, a four bytes array.
|
|
|
|
*/
|
|
|
|
static int nand_onfi_set_features(struct mtd_info *mtd, struct nand_chip *chip,
|
|
|
|
int addr, uint8_t *subfeature_param)
|
|
|
|
{
|
|
|
|
int status;
|
|
|
|
|
2013-05-29 19:30:13 +07:00
|
|
|
if (!chip->onfi_version ||
|
|
|
|
!(le16_to_cpu(chip->onfi_params.opt_cmd)
|
|
|
|
& ONFI_OPT_CMD_SET_GET_FEATURES))
|
2012-09-13 13:57:52 +07:00
|
|
|
return -EINVAL;
|
|
|
|
|
|
|
|
chip->cmdfunc(mtd, NAND_CMD_SET_FEATURES, addr, -1);
|
|
|
|
chip->write_buf(mtd, subfeature_param, ONFI_SUBFEATURE_PARAM_LEN);
|
|
|
|
status = chip->waitfunc(mtd, chip);
|
|
|
|
if (status & NAND_STATUS_FAIL)
|
|
|
|
return -EIO;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* nand_onfi_get_features- [REPLACEABLE] get features for ONFI nand
|
|
|
|
* @mtd: MTD device structure
|
|
|
|
* @chip: nand chip info structure
|
|
|
|
* @addr: feature address.
|
|
|
|
* @subfeature_param: the subfeature parameters, a four bytes array.
|
|
|
|
*/
|
|
|
|
static int nand_onfi_get_features(struct mtd_info *mtd, struct nand_chip *chip,
|
|
|
|
int addr, uint8_t *subfeature_param)
|
|
|
|
{
|
2013-05-29 19:30:13 +07:00
|
|
|
if (!chip->onfi_version ||
|
|
|
|
!(le16_to_cpu(chip->onfi_params.opt_cmd)
|
|
|
|
& ONFI_OPT_CMD_SET_GET_FEATURES))
|
2012-09-13 13:57:52 +07:00
|
|
|
return -EINVAL;
|
|
|
|
|
|
|
|
/* clear the sub feature parameters */
|
|
|
|
memset(subfeature_param, 0, ONFI_SUBFEATURE_PARAM_LEN);
|
|
|
|
|
|
|
|
chip->cmdfunc(mtd, NAND_CMD_GET_FEATURES, addr, -1);
|
|
|
|
chip->read_buf(mtd, subfeature_param, ONFI_SUBFEATURE_PARAM_LEN);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2005-09-15 20:58:53 +07:00
|
|
|
/**
|
|
|
|
* nand_suspend - [MTD Interface] Suspend the NAND flash
|
2011-05-26 04:59:01 +07:00
|
|
|
* @mtd: MTD device structure
|
2005-09-15 20:58:53 +07:00
|
|
|
*/
|
|
|
|
static int nand_suspend(struct mtd_info *mtd)
|
|
|
|
{
|
2012-11-19 13:43:30 +07:00
|
|
|
return nand_get_device(mtd, FL_PM_SUSPENDED);
|
2005-09-15 20:58:53 +07:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* nand_resume - [MTD Interface] Resume the NAND flash
|
2011-05-26 04:59:01 +07:00
|
|
|
* @mtd: MTD device structure
|
2005-09-15 20:58:53 +07:00
|
|
|
*/
|
|
|
|
static void nand_resume(struct mtd_info *mtd)
|
|
|
|
{
|
2006-05-24 17:07:37 +07:00
|
|
|
struct nand_chip *chip = mtd->priv;
|
2005-09-15 20:58:53 +07:00
|
|
|
|
2006-05-24 17:07:37 +07:00
|
|
|
if (chip->state == FL_PM_SUSPENDED)
|
2005-09-15 20:58:53 +07:00
|
|
|
nand_release_device(mtd);
|
|
|
|
else
|
2011-07-20 00:06:08 +07:00
|
|
|
pr_err("%s called for a chip which is not in suspended state\n",
|
|
|
|
__func__);
|
2005-09-15 20:58:53 +07:00
|
|
|
}
|
|
|
|
|
2011-05-26 04:59:01 +07:00
|
|
|
/* Set default functions */
|
2006-05-24 17:07:37 +07:00
|
|
|
static void nand_set_defaults(struct nand_chip *chip, int busw)
|
2006-05-23 16:54:38 +07:00
|
|
|
{
|
2005-04-17 05:20:36 +07:00
|
|
|
/* check for proper chip_delay setup, set 20us if not */
|
2006-05-24 17:07:37 +07:00
|
|
|
if (!chip->chip_delay)
|
|
|
|
chip->chip_delay = 20;
|
2005-04-17 05:20:36 +07:00
|
|
|
|
|
|
|
/* check, if a user supplied command function given */
|
2006-05-24 17:07:37 +07:00
|
|
|
if (chip->cmdfunc == NULL)
|
|
|
|
chip->cmdfunc = nand_command;
|
2005-04-17 05:20:36 +07:00
|
|
|
|
|
|
|
/* check, if a user supplied wait function given */
|
2006-05-24 17:07:37 +07:00
|
|
|
if (chip->waitfunc == NULL)
|
|
|
|
chip->waitfunc = nand_wait;
|
|
|
|
|
|
|
|
if (!chip->select_chip)
|
|
|
|
chip->select_chip = nand_select_chip;
|
2013-07-18 15:17:02 +07:00
|
|
|
|
2013-08-16 09:10:07 +07:00
|
|
|
/* set for ONFI nand */
|
|
|
|
if (!chip->onfi_set_features)
|
|
|
|
chip->onfi_set_features = nand_onfi_set_features;
|
|
|
|
if (!chip->onfi_get_features)
|
|
|
|
chip->onfi_get_features = nand_onfi_get_features;
|
|
|
|
|
2013-07-18 15:17:02 +07:00
|
|
|
/* If called twice, pointers that depend on busw may need to be reset */
|
|
|
|
if (!chip->read_byte || chip->read_byte == nand_read_byte)
|
2006-05-24 17:07:37 +07:00
|
|
|
chip->read_byte = busw ? nand_read_byte16 : nand_read_byte;
|
|
|
|
if (!chip->read_word)
|
|
|
|
chip->read_word = nand_read_word;
|
|
|
|
if (!chip->block_bad)
|
|
|
|
chip->block_bad = nand_block_bad;
|
|
|
|
if (!chip->block_markbad)
|
|
|
|
chip->block_markbad = nand_default_block_markbad;
|
2013-07-18 15:17:02 +07:00
|
|
|
if (!chip->write_buf || chip->write_buf == nand_write_buf)
|
2006-05-24 17:07:37 +07:00
|
|
|
chip->write_buf = busw ? nand_write_buf16 : nand_write_buf;
|
2013-07-18 15:17:02 +07:00
|
|
|
if (!chip->read_buf || chip->read_buf == nand_read_buf)
|
2006-05-24 17:07:37 +07:00
|
|
|
chip->read_buf = busw ? nand_read_buf16 : nand_read_buf;
|
|
|
|
if (!chip->scan_bbt)
|
|
|
|
chip->scan_bbt = nand_default_bbt;
|
2006-05-26 23:52:08 +07:00
|
|
|
|
|
|
|
if (!chip->controller) {
|
|
|
|
chip->controller = &chip->hwcontrol;
|
|
|
|
spin_lock_init(&chip->controller->lock);
|
|
|
|
init_waitqueue_head(&chip->controller->wq);
|
|
|
|
}
|
|
|
|
|
2006-05-23 16:54:38 +07:00
|
|
|
}
|
|
|
|
|
2011-05-26 04:59:01 +07:00
|
|
|
/* Sanitize ONFI strings so we can safely print them */
|
2010-08-30 23:32:24 +07:00
|
|
|
static void sanitize_string(uint8_t *s, size_t len)
|
|
|
|
{
|
|
|
|
ssize_t i;
|
|
|
|
|
2011-05-26 04:59:01 +07:00
|
|
|
/* Null terminate */
|
2010-08-30 23:32:24 +07:00
|
|
|
s[len - 1] = 0;
|
|
|
|
|
2011-05-26 04:59:01 +07:00
|
|
|
/* Remove non printable chars */
|
2010-08-30 23:32:24 +07:00
|
|
|
for (i = 0; i < len - 1; i++) {
|
|
|
|
if (s[i] < ' ' || s[i] > 127)
|
|
|
|
s[i] = '?';
|
|
|
|
}
|
|
|
|
|
2011-05-26 04:59:01 +07:00
|
|
|
/* Remove trailing spaces */
|
2010-08-30 23:32:24 +07:00
|
|
|
strim(s);
|
|
|
|
}
|
|
|
|
|
|
|
|
static u16 onfi_crc16(u16 crc, u8 const *p, size_t len)
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
while (len--) {
|
|
|
|
crc ^= *p++ << 8;
|
|
|
|
for (i = 0; i < 8; i++)
|
|
|
|
crc = (crc << 1) ^ ((crc & 0x8000) ? 0x8005 : 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
return crc;
|
|
|
|
}
|
|
|
|
|
2013-05-22 09:28:27 +07:00
|
|
|
/* Parse the Extended Parameter Page. */
|
|
|
|
static int nand_flash_detect_ext_param_page(struct mtd_info *mtd,
|
|
|
|
struct nand_chip *chip, struct nand_onfi_params *p)
|
|
|
|
{
|
|
|
|
struct onfi_ext_param_page *ep;
|
|
|
|
struct onfi_ext_section *s;
|
|
|
|
struct onfi_ext_ecc_info *ecc;
|
|
|
|
uint8_t *cursor;
|
|
|
|
int ret = -EINVAL;
|
|
|
|
int len;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
len = le16_to_cpu(p->ext_param_page_length) * 16;
|
|
|
|
ep = kmalloc(len, GFP_KERNEL);
|
2013-09-17 07:59:20 +07:00
|
|
|
if (!ep)
|
|
|
|
return -ENOMEM;
|
2013-05-22 09:28:27 +07:00
|
|
|
|
|
|
|
/* Send our own NAND_CMD_PARAM. */
|
|
|
|
chip->cmdfunc(mtd, NAND_CMD_PARAM, 0, -1);
|
|
|
|
|
|
|
|
/* Use the Change Read Column command to skip the ONFI param pages. */
|
|
|
|
chip->cmdfunc(mtd, NAND_CMD_RNDOUT,
|
|
|
|
sizeof(*p) * p->num_of_param_pages , -1);
|
|
|
|
|
|
|
|
/* Read out the Extended Parameter Page. */
|
|
|
|
chip->read_buf(mtd, (uint8_t *)ep, len);
|
|
|
|
if ((onfi_crc16(ONFI_CRC_BASE, ((uint8_t *)ep) + 2, len - 2)
|
|
|
|
!= le16_to_cpu(ep->crc))) {
|
|
|
|
pr_debug("fail in the CRC.\n");
|
|
|
|
goto ext_out;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Check the signature.
|
|
|
|
* Do not strictly follow the ONFI spec, maybe changed in future.
|
|
|
|
*/
|
|
|
|
if (strncmp(ep->sig, "EPPS", 4)) {
|
|
|
|
pr_debug("The signature is invalid.\n");
|
|
|
|
goto ext_out;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* find the ECC section. */
|
|
|
|
cursor = (uint8_t *)(ep + 1);
|
|
|
|
for (i = 0; i < ONFI_EXT_SECTION_MAX; i++) {
|
|
|
|
s = ep->sections + i;
|
|
|
|
if (s->type == ONFI_SECTION_TYPE_2)
|
|
|
|
break;
|
|
|
|
cursor += s->length * 16;
|
|
|
|
}
|
|
|
|
if (i == ONFI_EXT_SECTION_MAX) {
|
|
|
|
pr_debug("We can not find the ECC section.\n");
|
|
|
|
goto ext_out;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* get the info we want. */
|
|
|
|
ecc = (struct onfi_ext_ecc_info *)cursor;
|
|
|
|
|
2013-09-17 08:20:21 +07:00
|
|
|
if (!ecc->codeword_size) {
|
|
|
|
pr_debug("Invalid codeword size\n");
|
|
|
|
goto ext_out;
|
2013-05-22 09:28:27 +07:00
|
|
|
}
|
|
|
|
|
2013-09-17 08:20:21 +07:00
|
|
|
chip->ecc_strength_ds = ecc->ecc_bits;
|
|
|
|
chip->ecc_step_ds = 1 << ecc->codeword_size;
|
2013-09-17 07:59:20 +07:00
|
|
|
ret = 0;
|
2013-05-22 09:28:27 +07:00
|
|
|
|
|
|
|
ext_out:
|
|
|
|
kfree(ep);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2010-09-02 03:28:59 +07:00
|
|
|
/*
|
2011-05-26 04:59:01 +07:00
|
|
|
* Check if the NAND chip is ONFI compliant, returns 1 if it is, 0 otherwise.
|
2010-09-02 03:28:59 +07:00
|
|
|
*/
|
|
|
|
static int nand_flash_detect_onfi(struct mtd_info *mtd, struct nand_chip *chip,
|
2011-06-26 23:26:55 +07:00
|
|
|
int *busw)
|
2010-09-02 03:28:59 +07:00
|
|
|
{
|
|
|
|
struct nand_onfi_params *p = &chip->onfi_params;
|
|
|
|
int i;
|
|
|
|
int val;
|
|
|
|
|
2011-06-24 04:12:08 +07:00
|
|
|
/* Try ONFI for unknown chip or LP */
|
2010-09-02 03:28:59 +07:00
|
|
|
chip->cmdfunc(mtd, NAND_CMD_READID, 0x20, -1);
|
|
|
|
if (chip->read_byte(mtd) != 'O' || chip->read_byte(mtd) != 'N' ||
|
|
|
|
chip->read_byte(mtd) != 'F' || chip->read_byte(mtd) != 'I')
|
|
|
|
return 0;
|
|
|
|
|
2013-08-14 00:51:55 +07:00
|
|
|
/*
|
|
|
|
* ONFI must be probed in 8-bit mode or with NAND_BUSWIDTH_AUTO, not
|
|
|
|
* with NAND_BUSWIDTH_16
|
|
|
|
*/
|
|
|
|
if (chip->options & NAND_BUSWIDTH_16) {
|
|
|
|
pr_err("ONFI cannot be probed in 16-bit mode; aborting\n");
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2010-09-02 03:28:59 +07:00
|
|
|
chip->cmdfunc(mtd, NAND_CMD_PARAM, 0, -1);
|
|
|
|
for (i = 0; i < 3; i++) {
|
|
|
|
chip->read_buf(mtd, (uint8_t *)p, sizeof(*p));
|
|
|
|
if (onfi_crc16(ONFI_CRC_BASE, (uint8_t *)p, 254) ==
|
|
|
|
le16_to_cpu(p->crc)) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-08-14 00:51:55 +07:00
|
|
|
if (i == 3) {
|
|
|
|
pr_err("Could not find valid ONFI parameter page; aborting\n");
|
2010-09-02 03:28:59 +07:00
|
|
|
return 0;
|
2013-08-14 00:51:55 +07:00
|
|
|
}
|
2010-09-02 03:28:59 +07:00
|
|
|
|
2011-05-26 04:59:01 +07:00
|
|
|
/* Check version */
|
2010-09-02 03:28:59 +07:00
|
|
|
val = le16_to_cpu(p->revision);
|
2010-12-12 15:23:33 +07:00
|
|
|
if (val & (1 << 5))
|
|
|
|
chip->onfi_version = 23;
|
|
|
|
else if (val & (1 << 4))
|
2010-09-02 03:28:59 +07:00
|
|
|
chip->onfi_version = 22;
|
|
|
|
else if (val & (1 << 3))
|
|
|
|
chip->onfi_version = 21;
|
|
|
|
else if (val & (1 << 2))
|
|
|
|
chip->onfi_version = 20;
|
2010-12-12 15:23:33 +07:00
|
|
|
else if (val & (1 << 1))
|
2010-09-02 03:28:59 +07:00
|
|
|
chip->onfi_version = 10;
|
2010-12-12 15:23:33 +07:00
|
|
|
|
|
|
|
if (!chip->onfi_version) {
|
2011-07-20 00:06:08 +07:00
|
|
|
pr_info("%s: unsupported ONFI version: %d\n", __func__, val);
|
2010-12-12 15:23:33 +07:00
|
|
|
return 0;
|
|
|
|
}
|
2010-09-02 03:28:59 +07:00
|
|
|
|
|
|
|
sanitize_string(p->manufacturer, sizeof(p->manufacturer));
|
|
|
|
sanitize_string(p->model, sizeof(p->model));
|
|
|
|
if (!mtd->name)
|
|
|
|
mtd->name = p->model;
|
|
|
|
mtd->writesize = le32_to_cpu(p->byte_per_page);
|
|
|
|
mtd->erasesize = le32_to_cpu(p->pages_per_block) * mtd->writesize;
|
|
|
|
mtd->oobsize = le16_to_cpu(p->spare_bytes_per_page);
|
2012-03-19 21:35:25 +07:00
|
|
|
chip->chipsize = le32_to_cpu(p->blocks_per_lun);
|
|
|
|
chip->chipsize *= (uint64_t)mtd->erasesize * p->lun_count;
|
2013-09-25 13:58:13 +07:00
|
|
|
chip->bits_per_cell = p->bits_per_cell;
|
2013-05-17 10:17:30 +07:00
|
|
|
|
|
|
|
if (onfi_feature(chip) & ONFI_FEATURE_16_BIT_BUS)
|
2011-06-26 23:26:55 +07:00
|
|
|
*busw = NAND_BUSWIDTH_16;
|
2013-05-17 10:17:30 +07:00
|
|
|
else
|
|
|
|
*busw = 0;
|
2010-09-02 03:28:59 +07:00
|
|
|
|
2013-05-17 10:17:26 +07:00
|
|
|
if (p->ecc_bits != 0xff) {
|
|
|
|
chip->ecc_strength_ds = p->ecc_bits;
|
|
|
|
chip->ecc_step_ds = 512;
|
2013-05-22 09:28:27 +07:00
|
|
|
} else if (chip->onfi_version >= 21 &&
|
|
|
|
(onfi_feature(chip) & ONFI_FEATURE_EXT_PARAM_PAGE)) {
|
|
|
|
|
|
|
|
/*
|
|
|
|
* The nand_flash_detect_ext_param_page() uses the
|
|
|
|
* Change Read Column command which maybe not supported
|
|
|
|
* by the chip->cmdfunc. So try to update the chip->cmdfunc
|
|
|
|
* now. We do not replace user supplied command function.
|
|
|
|
*/
|
|
|
|
if (mtd->writesize > 512 && chip->cmdfunc == nand_command)
|
|
|
|
chip->cmdfunc = nand_command_lp;
|
|
|
|
|
|
|
|
/* The Extended Parameter Page is supported since ONFI 2.1. */
|
|
|
|
if (nand_flash_detect_ext_param_page(mtd, chip, p))
|
2013-08-14 00:51:55 +07:00
|
|
|
pr_warn("Failed to detect ONFI extended param page\n");
|
|
|
|
} else {
|
|
|
|
pr_warn("Could not retrieve ONFI ECC requirements\n");
|
2013-05-17 10:17:26 +07:00
|
|
|
}
|
|
|
|
|
2010-09-02 03:28:59 +07:00
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2012-09-25 10:40:52 +07:00
|
|
|
/*
|
|
|
|
* nand_id_has_period - Check if an ID string has a given wraparound period
|
|
|
|
* @id_data: the ID string
|
|
|
|
* @arrlen: the length of the @id_data array
|
|
|
|
* @period: the period of repitition
|
|
|
|
*
|
|
|
|
* Check if an ID string is repeated within a given sequence of bytes at
|
|
|
|
* specific repetition interval period (e.g., {0x20,0x01,0x7F,0x20} has a
|
2012-11-15 12:54:20 +07:00
|
|
|
* period of 3). This is a helper function for nand_id_len(). Returns non-zero
|
2012-09-25 10:40:52 +07:00
|
|
|
* if the repetition has a period of @period; otherwise, returns zero.
|
|
|
|
*/
|
|
|
|
static int nand_id_has_period(u8 *id_data, int arrlen, int period)
|
|
|
|
{
|
|
|
|
int i, j;
|
|
|
|
for (i = 0; i < period; i++)
|
|
|
|
for (j = i + period; j < arrlen; j += period)
|
|
|
|
if (id_data[i] != id_data[j])
|
|
|
|
return 0;
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* nand_id_len - Get the length of an ID string returned by CMD_READID
|
|
|
|
* @id_data: the ID string
|
|
|
|
* @arrlen: the length of the @id_data array
|
|
|
|
|
|
|
|
* Returns the length of the ID string, according to known wraparound/trailing
|
|
|
|
* zero patterns. If no pattern exists, returns the length of the array.
|
|
|
|
*/
|
|
|
|
static int nand_id_len(u8 *id_data, int arrlen)
|
|
|
|
{
|
|
|
|
int last_nonzero, period;
|
|
|
|
|
|
|
|
/* Find last non-zero byte */
|
|
|
|
for (last_nonzero = arrlen - 1; last_nonzero >= 0; last_nonzero--)
|
|
|
|
if (id_data[last_nonzero])
|
|
|
|
break;
|
|
|
|
|
|
|
|
/* All zeros */
|
|
|
|
if (last_nonzero < 0)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
/* Calculate wraparound period */
|
|
|
|
for (period = 1; period < arrlen; period++)
|
|
|
|
if (nand_id_has_period(id_data, arrlen, period))
|
|
|
|
break;
|
|
|
|
|
|
|
|
/* There's a repeated pattern */
|
|
|
|
if (period < arrlen)
|
|
|
|
return period;
|
|
|
|
|
|
|
|
/* There are trailing zeros */
|
|
|
|
if (last_nonzero < arrlen - 1)
|
|
|
|
return last_nonzero + 1;
|
|
|
|
|
|
|
|
/* No pattern detected */
|
|
|
|
return arrlen;
|
|
|
|
}
|
|
|
|
|
2013-09-25 13:58:11 +07:00
|
|
|
/* Extract the bits of per cell from the 3rd byte of the extended ID */
|
|
|
|
static int nand_get_bits_per_cell(u8 cellinfo)
|
|
|
|
{
|
|
|
|
int bits;
|
|
|
|
|
|
|
|
bits = cellinfo & NAND_CI_CELLTYPE_MSK;
|
|
|
|
bits >>= NAND_CI_CELLTYPE_SHIFT;
|
|
|
|
return bits + 1;
|
|
|
|
}
|
|
|
|
|
2012-09-25 10:40:50 +07:00
|
|
|
/*
|
|
|
|
* Many new NAND share similar device ID codes, which represent the size of the
|
|
|
|
* chip. The rest of the parameters must be decoded according to generic or
|
|
|
|
* manufacturer-specific "extended ID" decoding patterns.
|
|
|
|
*/
|
|
|
|
static void nand_decode_ext_id(struct mtd_info *mtd, struct nand_chip *chip,
|
|
|
|
u8 id_data[8], int *busw)
|
|
|
|
{
|
2012-09-25 10:40:52 +07:00
|
|
|
int extid, id_len;
|
2012-09-25 10:40:50 +07:00
|
|
|
/* The 3rd id byte holds MLC / multichip data */
|
2013-09-25 13:58:11 +07:00
|
|
|
chip->bits_per_cell = nand_get_bits_per_cell(id_data[2]);
|
2012-09-25 10:40:50 +07:00
|
|
|
/* The 4th id byte is the important one */
|
|
|
|
extid = id_data[3];
|
|
|
|
|
2012-09-25 10:40:52 +07:00
|
|
|
id_len = nand_id_len(id_data, 8);
|
|
|
|
|
2012-09-25 10:40:50 +07:00
|
|
|
/*
|
|
|
|
* Field definitions are in the following datasheets:
|
|
|
|
* Old style (4,5 byte ID): Samsung K9GAG08U0M (p.32)
|
2012-10-10 13:26:06 +07:00
|
|
|
* New Samsung (6 byte ID): Samsung K9GAG08U0F (p.44)
|
2012-09-25 10:40:54 +07:00
|
|
|
* Hynix MLC (6 byte ID): Hynix H27UBG8T2B (p.22)
|
2012-09-25 10:40:50 +07:00
|
|
|
*
|
2012-10-10 13:26:06 +07:00
|
|
|
* Check for ID length, non-zero 6th byte, cell type, and Hynix/Samsung
|
|
|
|
* ID to decide what to do.
|
2012-09-25 10:40:50 +07:00
|
|
|
*/
|
2012-10-10 13:26:06 +07:00
|
|
|
if (id_len == 6 && id_data[0] == NAND_MFR_SAMSUNG &&
|
2013-09-25 13:58:10 +07:00
|
|
|
!nand_is_slc(chip) && id_data[5] != 0x00) {
|
2012-09-25 10:40:50 +07:00
|
|
|
/* Calc pagesize */
|
|
|
|
mtd->writesize = 2048 << (extid & 0x03);
|
|
|
|
extid >>= 2;
|
|
|
|
/* Calc oobsize */
|
mtd: nand: detect Samsung K9GBG08U0A, K9GAG08U0F ID
Datasheets for the following Samsung NAND parts (both MLC and SLC) describe
extensions to the Samsung 6-byte extended ID decoding table:
K9GBG08U0A (MLC, 6-byte ID)
K9GAG08U0F (MLC, 6-byte ID)
K9FAG08U0M (SLC, 6-byte ID)
The table found in K9GAG08U0F, p.44, contains a superset of the information
found in other previous datasheets.
This patch adds support for all of these chips, with 512B and 640B OOB sizes.
It also changes the detection pattern such that this table applies to all
Samsung 6-byte ID NAND, not just MLC. This is safe, according to the NAND
parameter data I have collected:
Note that nand_base.c does not yet support the bad block marker scheme defined
for these chips (i.e., scan 1st and last page for BB markers).
Signed-off-by: Brian Norris <computersforpeace@gmail.com>
Signed-off-by: Artem Bityutskiy <artem.bityutskiy@linux.intel.com>
Signed-off-by: David Woodhouse <David.Woodhouse@intel.com>
2012-09-25 10:40:55 +07:00
|
|
|
switch (((extid >> 2) & 0x04) | (extid & 0x03)) {
|
2012-09-25 10:40:50 +07:00
|
|
|
case 1:
|
|
|
|
mtd->oobsize = 128;
|
|
|
|
break;
|
|
|
|
case 2:
|
|
|
|
mtd->oobsize = 218;
|
|
|
|
break;
|
|
|
|
case 3:
|
|
|
|
mtd->oobsize = 400;
|
|
|
|
break;
|
mtd: nand: detect Samsung K9GBG08U0A, K9GAG08U0F ID
Datasheets for the following Samsung NAND parts (both MLC and SLC) describe
extensions to the Samsung 6-byte extended ID decoding table:
K9GBG08U0A (MLC, 6-byte ID)
K9GAG08U0F (MLC, 6-byte ID)
K9FAG08U0M (SLC, 6-byte ID)
The table found in K9GAG08U0F, p.44, contains a superset of the information
found in other previous datasheets.
This patch adds support for all of these chips, with 512B and 640B OOB sizes.
It also changes the detection pattern such that this table applies to all
Samsung 6-byte ID NAND, not just MLC. This is safe, according to the NAND
parameter data I have collected:
Note that nand_base.c does not yet support the bad block marker scheme defined
for these chips (i.e., scan 1st and last page for BB markers).
Signed-off-by: Brian Norris <computersforpeace@gmail.com>
Signed-off-by: Artem Bityutskiy <artem.bityutskiy@linux.intel.com>
Signed-off-by: David Woodhouse <David.Woodhouse@intel.com>
2012-09-25 10:40:55 +07:00
|
|
|
case 4:
|
2012-09-25 10:40:50 +07:00
|
|
|
mtd->oobsize = 436;
|
|
|
|
break;
|
mtd: nand: detect Samsung K9GBG08U0A, K9GAG08U0F ID
Datasheets for the following Samsung NAND parts (both MLC and SLC) describe
extensions to the Samsung 6-byte extended ID decoding table:
K9GBG08U0A (MLC, 6-byte ID)
K9GAG08U0F (MLC, 6-byte ID)
K9FAG08U0M (SLC, 6-byte ID)
The table found in K9GAG08U0F, p.44, contains a superset of the information
found in other previous datasheets.
This patch adds support for all of these chips, with 512B and 640B OOB sizes.
It also changes the detection pattern such that this table applies to all
Samsung 6-byte ID NAND, not just MLC. This is safe, according to the NAND
parameter data I have collected:
Note that nand_base.c does not yet support the bad block marker scheme defined
for these chips (i.e., scan 1st and last page for BB markers).
Signed-off-by: Brian Norris <computersforpeace@gmail.com>
Signed-off-by: Artem Bityutskiy <artem.bityutskiy@linux.intel.com>
Signed-off-by: David Woodhouse <David.Woodhouse@intel.com>
2012-09-25 10:40:55 +07:00
|
|
|
case 5:
|
|
|
|
mtd->oobsize = 512;
|
|
|
|
break;
|
|
|
|
case 6:
|
|
|
|
default: /* Other cases are "reserved" (unknown) */
|
|
|
|
mtd->oobsize = 640;
|
|
|
|
break;
|
2012-09-25 10:40:50 +07:00
|
|
|
}
|
|
|
|
extid >>= 2;
|
|
|
|
/* Calc blocksize */
|
|
|
|
mtd->erasesize = (128 * 1024) <<
|
|
|
|
(((extid >> 1) & 0x04) | (extid & 0x03));
|
|
|
|
*busw = 0;
|
2012-09-25 10:40:54 +07:00
|
|
|
} else if (id_len == 6 && id_data[0] == NAND_MFR_HYNIX &&
|
2013-09-25 13:58:10 +07:00
|
|
|
!nand_is_slc(chip)) {
|
2012-09-25 10:40:54 +07:00
|
|
|
unsigned int tmp;
|
|
|
|
|
|
|
|
/* Calc pagesize */
|
|
|
|
mtd->writesize = 2048 << (extid & 0x03);
|
|
|
|
extid >>= 2;
|
|
|
|
/* Calc oobsize */
|
|
|
|
switch (((extid >> 2) & 0x04) | (extid & 0x03)) {
|
|
|
|
case 0:
|
|
|
|
mtd->oobsize = 128;
|
|
|
|
break;
|
|
|
|
case 1:
|
|
|
|
mtd->oobsize = 224;
|
|
|
|
break;
|
|
|
|
case 2:
|
|
|
|
mtd->oobsize = 448;
|
|
|
|
break;
|
|
|
|
case 3:
|
|
|
|
mtd->oobsize = 64;
|
|
|
|
break;
|
|
|
|
case 4:
|
|
|
|
mtd->oobsize = 32;
|
|
|
|
break;
|
|
|
|
case 5:
|
|
|
|
mtd->oobsize = 16;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
mtd->oobsize = 640;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
extid >>= 2;
|
|
|
|
/* Calc blocksize */
|
|
|
|
tmp = ((extid >> 1) & 0x04) | (extid & 0x03);
|
|
|
|
if (tmp < 0x03)
|
|
|
|
mtd->erasesize = (128 * 1024) << tmp;
|
|
|
|
else if (tmp == 0x03)
|
|
|
|
mtd->erasesize = 768 * 1024;
|
|
|
|
else
|
|
|
|
mtd->erasesize = (64 * 1024) << tmp;
|
|
|
|
*busw = 0;
|
2012-09-25 10:40:50 +07:00
|
|
|
} else {
|
|
|
|
/* Calc pagesize */
|
|
|
|
mtd->writesize = 1024 << (extid & 0x03);
|
|
|
|
extid >>= 2;
|
|
|
|
/* Calc oobsize */
|
|
|
|
mtd->oobsize = (8 << (extid & 0x01)) *
|
|
|
|
(mtd->writesize >> 9);
|
|
|
|
extid >>= 2;
|
|
|
|
/* Calc blocksize. Blocksize is multiples of 64KiB */
|
|
|
|
mtd->erasesize = (64 * 1024) << (extid & 0x03);
|
|
|
|
extid >>= 2;
|
|
|
|
/* Get buswidth information */
|
|
|
|
*busw = (extid & 0x01) ? NAND_BUSWIDTH_16 : 0;
|
mtd: nand: detect OOB size for Toshiba 24nm raw SLC
Toshiba NAND datasheets have not been very forthcoming on OOB size
information; they do not provide any bitfields in the ID string for
spare area. In their 24nm technology flash, however, Toshiba migrated
their NAND to have 32 bytes spare per 512 bytes of page area (up from
the traditional 16 bytes), as they now require 8-bit ECC or higher.
I have discussed this issue directly with Toshiba representatives, and
they acknowledge this problem. They recommend detecting these flash
based on their technology node as follows:
For 24nm Toshiba SLC raw NAND (not BENAND -- Built-in Ecc NAND), there
are 32 bytes of spare area for every 512 bytes of in-band data area.
We can implement this rule with the following snippet of a device ID
decode table, which applies to all their 43nm, 32nm, and 24nm SLC NAND
(this table is not fully in the NAND datasheets, but it was provided
directly by Toshiba representatives):
- ID byte 5, bit[7]:
1 -> BENAND
0 -> raw SLC
- ID byte 6, bits[2:0]:
100b -> 43nm
101b -> 32nm
110b -> 24nm
111b -> Reserved
I'm also working with Toshiba on including this bitfield description for
their 5th and 6th ID bytes in their public data sheets.
I will provide the 8-byte ID strings from the two 24nm Toshiba samples I
have; their first 6 bytes match the documentation I received from
Toshiba:
24nm SLC 1Gbit TC58NVG0S3HTA00
0x98 0xf1 0x80 0x15 0x72 0x16 0x08 0x00
24nm SLC 2Gbit TC58NVG1S3HTA00
0x98 0xda 0x90 0x15 0x76 0x16 0x08 0x00
I have also tested for regressions with:
43nm SLC 4Gbit TC58NVG2S3ETA00
0x98 0xdc 0x90 0x15 0x76 0x14 0x03 0x10
32nm SLC 8Gbit TC58NVG3SOFA00
0x98 0xd3 0x90 0x26 0x76 0x15 0x02 0x08
Signed-off-by: Brian Norris <computersforpeace@gmail.com>
Signed-off-by: Artem Bityutskiy <artem.bityutskiy@linux.intel.com>
Signed-off-by: David Woodhouse <David.Woodhouse@intel.com>
2013-06-26 03:17:59 +07:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Toshiba 24nm raw SLC (i.e., not BENAND) have 32B OOB per
|
|
|
|
* 512B page. For Toshiba SLC, we decode the 5th/6th byte as
|
|
|
|
* follows:
|
|
|
|
* - ID byte 6, bits[2:0]: 100b -> 43nm, 101b -> 32nm,
|
|
|
|
* 110b -> 24nm
|
|
|
|
* - ID byte 5, bit[7]: 1 -> BENAND, 0 -> raw SLC
|
|
|
|
*/
|
|
|
|
if (id_len >= 6 && id_data[0] == NAND_MFR_TOSHIBA &&
|
2013-09-25 13:58:10 +07:00
|
|
|
nand_is_slc(chip) &&
|
mtd: nand: detect OOB size for Toshiba 24nm raw SLC
Toshiba NAND datasheets have not been very forthcoming on OOB size
information; they do not provide any bitfields in the ID string for
spare area. In their 24nm technology flash, however, Toshiba migrated
their NAND to have 32 bytes spare per 512 bytes of page area (up from
the traditional 16 bytes), as they now require 8-bit ECC or higher.
I have discussed this issue directly with Toshiba representatives, and
they acknowledge this problem. They recommend detecting these flash
based on their technology node as follows:
For 24nm Toshiba SLC raw NAND (not BENAND -- Built-in Ecc NAND), there
are 32 bytes of spare area for every 512 bytes of in-band data area.
We can implement this rule with the following snippet of a device ID
decode table, which applies to all their 43nm, 32nm, and 24nm SLC NAND
(this table is not fully in the NAND datasheets, but it was provided
directly by Toshiba representatives):
- ID byte 5, bit[7]:
1 -> BENAND
0 -> raw SLC
- ID byte 6, bits[2:0]:
100b -> 43nm
101b -> 32nm
110b -> 24nm
111b -> Reserved
I'm also working with Toshiba on including this bitfield description for
their 5th and 6th ID bytes in their public data sheets.
I will provide the 8-byte ID strings from the two 24nm Toshiba samples I
have; their first 6 bytes match the documentation I received from
Toshiba:
24nm SLC 1Gbit TC58NVG0S3HTA00
0x98 0xf1 0x80 0x15 0x72 0x16 0x08 0x00
24nm SLC 2Gbit TC58NVG1S3HTA00
0x98 0xda 0x90 0x15 0x76 0x16 0x08 0x00
I have also tested for regressions with:
43nm SLC 4Gbit TC58NVG2S3ETA00
0x98 0xdc 0x90 0x15 0x76 0x14 0x03 0x10
32nm SLC 8Gbit TC58NVG3SOFA00
0x98 0xd3 0x90 0x26 0x76 0x15 0x02 0x08
Signed-off-by: Brian Norris <computersforpeace@gmail.com>
Signed-off-by: Artem Bityutskiy <artem.bityutskiy@linux.intel.com>
Signed-off-by: David Woodhouse <David.Woodhouse@intel.com>
2013-06-26 03:17:59 +07:00
|
|
|
(id_data[5] & 0x7) == 0x6 /* 24nm */ &&
|
|
|
|
!(id_data[4] & 0x80) /* !BENAND */) {
|
|
|
|
mtd->oobsize = 32 * mtd->writesize >> 9;
|
|
|
|
}
|
|
|
|
|
2012-09-25 10:40:50 +07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-09-25 10:40:51 +07:00
|
|
|
/*
|
|
|
|
* Old devices have chip data hardcoded in the device ID table. nand_decode_id
|
|
|
|
* decodes a matching ID table entry and assigns the MTD size parameters for
|
|
|
|
* the chip.
|
|
|
|
*/
|
|
|
|
static void nand_decode_id(struct mtd_info *mtd, struct nand_chip *chip,
|
|
|
|
struct nand_flash_dev *type, u8 id_data[8],
|
|
|
|
int *busw)
|
|
|
|
{
|
|
|
|
int maf_id = id_data[0];
|
|
|
|
|
|
|
|
mtd->erasesize = type->erasesize;
|
|
|
|
mtd->writesize = type->pagesize;
|
|
|
|
mtd->oobsize = mtd->writesize / 32;
|
|
|
|
*busw = type->options & NAND_BUSWIDTH_16;
|
|
|
|
|
2013-09-25 13:58:12 +07:00
|
|
|
/* All legacy ID NAND are small-page, SLC */
|
|
|
|
chip->bits_per_cell = 1;
|
|
|
|
|
2012-09-25 10:40:51 +07:00
|
|
|
/*
|
|
|
|
* Check for Spansion/AMD ID + repeating 5th, 6th byte since
|
|
|
|
* some Spansion chips have erasesize that conflicts with size
|
|
|
|
* listed in nand_ids table.
|
|
|
|
* Data sheet (5 byte ID): Spansion S30ML-P ORNAND (p.39)
|
|
|
|
*/
|
|
|
|
if (maf_id == NAND_MFR_AMD && id_data[4] != 0x00 && id_data[5] == 0x00
|
|
|
|
&& id_data[6] == 0x00 && id_data[7] == 0x00
|
|
|
|
&& mtd->writesize == 512) {
|
|
|
|
mtd->erasesize = 128 * 1024;
|
|
|
|
mtd->erasesize <<= ((id_data[3] & 0x03) << 1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-09-25 10:40:49 +07:00
|
|
|
/*
|
|
|
|
* Set the bad block marker/indicator (BBM/BBI) patterns according to some
|
|
|
|
* heuristic patterns using various detected parameters (e.g., manufacturer,
|
|
|
|
* page size, cell-type information).
|
|
|
|
*/
|
|
|
|
static void nand_decode_bbm_options(struct mtd_info *mtd,
|
|
|
|
struct nand_chip *chip, u8 id_data[8])
|
|
|
|
{
|
|
|
|
int maf_id = id_data[0];
|
|
|
|
|
|
|
|
/* Set the bad block position */
|
|
|
|
if (mtd->writesize > 512 || (chip->options & NAND_BUSWIDTH_16))
|
|
|
|
chip->badblockpos = NAND_LARGE_BADBLOCK_POS;
|
|
|
|
else
|
|
|
|
chip->badblockpos = NAND_SMALL_BADBLOCK_POS;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Bad block marker is stored in the last page of each block on Samsung
|
|
|
|
* and Hynix MLC devices; stored in first two pages of each block on
|
|
|
|
* Micron devices with 2KiB pages and on SLC Samsung, Hynix, Toshiba,
|
|
|
|
* AMD/Spansion, and Macronix. All others scan only the first page.
|
|
|
|
*/
|
2013-09-25 13:58:10 +07:00
|
|
|
if (!nand_is_slc(chip) &&
|
2012-09-25 10:40:49 +07:00
|
|
|
(maf_id == NAND_MFR_SAMSUNG ||
|
|
|
|
maf_id == NAND_MFR_HYNIX))
|
|
|
|
chip->bbt_options |= NAND_BBT_SCANLASTPAGE;
|
2013-09-25 13:58:10 +07:00
|
|
|
else if ((nand_is_slc(chip) &&
|
2012-09-25 10:40:49 +07:00
|
|
|
(maf_id == NAND_MFR_SAMSUNG ||
|
|
|
|
maf_id == NAND_MFR_HYNIX ||
|
|
|
|
maf_id == NAND_MFR_TOSHIBA ||
|
|
|
|
maf_id == NAND_MFR_AMD ||
|
|
|
|
maf_id == NAND_MFR_MACRONIX)) ||
|
|
|
|
(mtd->writesize == 2048 &&
|
|
|
|
maf_id == NAND_MFR_MICRON))
|
|
|
|
chip->bbt_options |= NAND_BBT_SCAN2NDPAGE;
|
|
|
|
}
|
|
|
|
|
2013-03-15 10:01:00 +07:00
|
|
|
static inline bool is_full_id_nand(struct nand_flash_dev *type)
|
|
|
|
{
|
|
|
|
return type->id_len;
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool find_full_id_nand(struct mtd_info *mtd, struct nand_chip *chip,
|
|
|
|
struct nand_flash_dev *type, u8 *id_data, int *busw)
|
|
|
|
{
|
|
|
|
if (!strncmp(type->id, id_data, type->id_len)) {
|
|
|
|
mtd->writesize = type->pagesize;
|
|
|
|
mtd->erasesize = type->erasesize;
|
|
|
|
mtd->oobsize = type->oobsize;
|
|
|
|
|
2013-09-25 13:58:11 +07:00
|
|
|
chip->bits_per_cell = nand_get_bits_per_cell(id_data[2]);
|
2013-03-15 10:01:00 +07:00
|
|
|
chip->chipsize = (uint64_t)type->chipsize << 20;
|
|
|
|
chip->options |= type->options;
|
2013-05-17 10:17:32 +07:00
|
|
|
chip->ecc_strength_ds = NAND_ECC_STRENGTH(type);
|
|
|
|
chip->ecc_step_ds = NAND_ECC_STEP(type);
|
2013-03-15 10:01:00 +07:00
|
|
|
|
|
|
|
*busw = type->options & NAND_BUSWIDTH_16;
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2006-05-23 16:54:38 +07:00
|
|
|
/*
|
2011-05-26 04:59:01 +07:00
|
|
|
* Get the flash and manufacturer id and lookup if the type is supported.
|
2006-05-23 16:54:38 +07:00
|
|
|
*/
|
|
|
|
static struct nand_flash_dev *nand_get_flash_type(struct mtd_info *mtd,
|
2006-05-24 17:07:37 +07:00
|
|
|
struct nand_chip *chip,
|
2010-09-07 18:23:45 +07:00
|
|
|
int busw,
|
|
|
|
int *maf_id, int *dev_id,
|
2010-02-27 01:32:56 +07:00
|
|
|
struct nand_flash_dev *type)
|
2006-05-23 16:54:38 +07:00
|
|
|
{
|
2010-08-30 23:32:24 +07:00
|
|
|
int i, maf_idx;
|
mtd: nand: extend NAND flash detection to new MLC chips
Some of the newer MLC devices have a 6-byte ID sequence in which
several field definitions differ from older chips in a manner that is
not backward compatible. For instance:
Samsung K9GAG08U0M (5-byte sequence): ec d5 14 b6 74
4th byte, bits 1:0 encode the page size: 0=1KiB, 1=2KiB, 2=4KiB, 3=8KiB
4th byte, bits 5:4 encode the block size: 0=64KiB, 1=128KiB, ...
4th byte, bit 6 encodes the OOB size: 0=8B/512B, 1=16B/512B
Samsung K9GAG08U0D (6-byte sequence): ec d5 94 29 34 41
4th byte, bits 1:0 encode the page size: 0=2KiB, 1=4KiB, 3=8KiB, 4=rsvd
4th byte, bits 7;5:4 encode the block size: 0=128KiB, 1=256KiB, ...
4th byte, bits 6;3:2 encode the OOB size: 1=128B/page, 2=218B/page
This patch uses the new 6-byte scheme if the following conditions are
all true:
1) The ID code wraps around after exactly 6 bytes
2) Manufacturer is Samsung
3) 6th byte is zero
The patch also extends the maximum OOB size from 128B to 256B.
Signed-off-by: Kevin Cernekee <cernekee@gmail.com>
Signed-off-by: David Woodhouse <David.Woodhouse@intel.com>
2010-05-05 10:58:03 +07:00
|
|
|
u8 id_data[8];
|
2005-04-17 05:20:36 +07:00
|
|
|
|
|
|
|
/* Select the device */
|
2006-05-24 17:07:37 +07:00
|
|
|
chip->select_chip(mtd, 0);
|
2005-04-17 05:20:36 +07:00
|
|
|
|
2008-09-15 19:37:29 +07:00
|
|
|
/*
|
|
|
|
* Reset the chip, required by some chips (e.g. Micron MT29FxGxxxxx)
|
2011-05-26 04:59:01 +07:00
|
|
|
* after power-up.
|
2008-09-15 19:37:29 +07:00
|
|
|
*/
|
|
|
|
chip->cmdfunc(mtd, NAND_CMD_RESET, -1, -1);
|
|
|
|
|
2005-04-17 05:20:36 +07:00
|
|
|
/* Send the command for reading device ID */
|
2006-05-24 17:07:37 +07:00
|
|
|
chip->cmdfunc(mtd, NAND_CMD_READID, 0x00, -1);
|
2005-04-17 05:20:36 +07:00
|
|
|
|
|
|
|
/* Read manufacturer and device IDs */
|
2006-05-24 17:07:37 +07:00
|
|
|
*maf_id = chip->read_byte(mtd);
|
2010-08-30 23:32:24 +07:00
|
|
|
*dev_id = chip->read_byte(mtd);
|
2005-04-17 05:20:36 +07:00
|
|
|
|
2011-05-26 04:59:01 +07:00
|
|
|
/*
|
|
|
|
* Try again to make sure, as some systems the bus-hold or other
|
2008-04-14 20:58:58 +07:00
|
|
|
* interface concerns can cause random data which looks like a
|
|
|
|
* possibly credible NAND flash to appear. If the two results do
|
|
|
|
* not match, ignore the device completely.
|
|
|
|
*/
|
|
|
|
|
|
|
|
chip->cmdfunc(mtd, NAND_CMD_READID, 0x00, -1);
|
|
|
|
|
2012-09-25 10:40:48 +07:00
|
|
|
/* Read entire ID string */
|
|
|
|
for (i = 0; i < 8; i++)
|
mtd: nand: extend NAND flash detection to new MLC chips
Some of the newer MLC devices have a 6-byte ID sequence in which
several field definitions differ from older chips in a manner that is
not backward compatible. For instance:
Samsung K9GAG08U0M (5-byte sequence): ec d5 14 b6 74
4th byte, bits 1:0 encode the page size: 0=1KiB, 1=2KiB, 2=4KiB, 3=8KiB
4th byte, bits 5:4 encode the block size: 0=64KiB, 1=128KiB, ...
4th byte, bit 6 encodes the OOB size: 0=8B/512B, 1=16B/512B
Samsung K9GAG08U0D (6-byte sequence): ec d5 94 29 34 41
4th byte, bits 1:0 encode the page size: 0=2KiB, 1=4KiB, 3=8KiB, 4=rsvd
4th byte, bits 7;5:4 encode the block size: 0=128KiB, 1=256KiB, ...
4th byte, bits 6;3:2 encode the OOB size: 1=128B/page, 2=218B/page
This patch uses the new 6-byte scheme if the following conditions are
all true:
1) The ID code wraps around after exactly 6 bytes
2) Manufacturer is Samsung
3) 6th byte is zero
The patch also extends the maximum OOB size from 128B to 256B.
Signed-off-by: Kevin Cernekee <cernekee@gmail.com>
Signed-off-by: David Woodhouse <David.Woodhouse@intel.com>
2010-05-05 10:58:03 +07:00
|
|
|
id_data[i] = chip->read_byte(mtd);
|
2008-04-14 20:58:58 +07:00
|
|
|
|
2010-08-30 23:32:24 +07:00
|
|
|
if (id_data[0] != *maf_id || id_data[1] != *dev_id) {
|
2011-07-20 00:06:07 +07:00
|
|
|
pr_info("%s: second ID read did not match "
|
2011-07-20 00:06:08 +07:00
|
|
|
"%02x,%02x against %02x,%02x\n", __func__,
|
|
|
|
*maf_id, *dev_id, id_data[0], id_data[1]);
|
2008-04-14 20:58:58 +07:00
|
|
|
return ERR_PTR(-ENODEV);
|
|
|
|
}
|
|
|
|
|
2006-05-23 16:54:38 +07:00
|
|
|
if (!type)
|
2010-02-27 01:32:56 +07:00
|
|
|
type = nand_flash_ids;
|
|
|
|
|
2013-03-15 10:01:00 +07:00
|
|
|
for (; type->name != NULL; type++) {
|
|
|
|
if (is_full_id_nand(type)) {
|
|
|
|
if (find_full_id_nand(mtd, chip, type, id_data, &busw))
|
|
|
|
goto ident_done;
|
|
|
|
} else if (*dev_id == type->dev_id) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2010-02-27 01:32:56 +07:00
|
|
|
|
2010-08-30 23:32:24 +07:00
|
|
|
chip->onfi_version = 0;
|
|
|
|
if (!type->name || !type->pagesize) {
|
2010-09-02 03:28:59 +07:00
|
|
|
/* Check is chip is ONFI compliant */
|
2012-09-25 10:40:47 +07:00
|
|
|
if (nand_flash_detect_onfi(mtd, chip, &busw))
|
2010-09-02 03:28:59 +07:00
|
|
|
goto ident_done;
|
2010-08-30 23:32:24 +07:00
|
|
|
}
|
|
|
|
|
2010-02-27 01:32:56 +07:00
|
|
|
if (!type->name)
|
2006-05-23 16:54:38 +07:00
|
|
|
return ERR_PTR(-ENODEV);
|
|
|
|
|
2006-05-27 06:02:13 +07:00
|
|
|
if (!mtd->name)
|
|
|
|
mtd->name = type->name;
|
|
|
|
|
2008-12-10 20:37:21 +07:00
|
|
|
chip->chipsize = (uint64_t)type->chipsize << 20;
|
2006-05-23 16:54:38 +07:00
|
|
|
|
2010-09-27 09:43:53 +07:00
|
|
|
if (!type->pagesize && chip->init_size) {
|
2011-05-26 04:59:01 +07:00
|
|
|
/* Set the pagesize, oobsize, erasesize by the driver */
|
2010-09-27 09:43:53 +07:00
|
|
|
busw = chip->init_size(mtd, chip, id_data);
|
|
|
|
} else if (!type->pagesize) {
|
2012-09-25 10:40:50 +07:00
|
|
|
/* Decode parameters from extended ID */
|
|
|
|
nand_decode_ext_id(mtd, chip, id_data, &busw);
|
2006-05-23 16:54:38 +07:00
|
|
|
} else {
|
2012-09-25 10:40:51 +07:00
|
|
|
nand_decode_id(mtd, chip, type, id_data, &busw);
|
2006-05-23 16:54:38 +07:00
|
|
|
}
|
2012-07-13 23:28:24 +07:00
|
|
|
/* Get chip options */
|
|
|
|
chip->options |= type->options;
|
2010-08-30 23:32:24 +07:00
|
|
|
|
2011-05-26 04:59:01 +07:00
|
|
|
/*
|
|
|
|
* Check if chip is not a Samsung device. Do not clear the
|
|
|
|
* options for chips which do not have an extended id.
|
2010-08-30 23:32:24 +07:00
|
|
|
*/
|
|
|
|
if (*maf_id != NAND_MFR_SAMSUNG && !type->pagesize)
|
|
|
|
chip->options &= ~NAND_SAMSUNG_LP_OPTIONS;
|
|
|
|
ident_done:
|
|
|
|
|
2006-05-23 16:54:38 +07:00
|
|
|
/* Try to identify manufacturer */
|
2006-07-15 19:26:18 +07:00
|
|
|
for (maf_idx = 0; nand_manuf_ids[maf_idx].id != 0x0; maf_idx++) {
|
2006-05-23 16:54:38 +07:00
|
|
|
if (nand_manuf_ids[maf_idx].id == *maf_id)
|
|
|
|
break;
|
|
|
|
}
|
2005-02-16 16:39:39 +07:00
|
|
|
|
2012-11-06 17:51:44 +07:00
|
|
|
if (chip->options & NAND_BUSWIDTH_AUTO) {
|
|
|
|
WARN_ON(chip->options & NAND_BUSWIDTH_16);
|
|
|
|
chip->options |= busw;
|
|
|
|
nand_set_defaults(chip, busw);
|
|
|
|
} else if (busw != (chip->options & NAND_BUSWIDTH_16)) {
|
|
|
|
/*
|
|
|
|
* Check, if buswidth is correct. Hardware drivers should set
|
|
|
|
* chip correct!
|
|
|
|
*/
|
2011-07-20 00:06:07 +07:00
|
|
|
pr_info("NAND device: Manufacturer ID:"
|
2011-07-20 00:06:08 +07:00
|
|
|
" 0x%02x, Chip ID: 0x%02x (%s %s)\n", *maf_id,
|
|
|
|
*dev_id, nand_manuf_ids[maf_idx].name, mtd->name);
|
2011-07-20 00:06:07 +07:00
|
|
|
pr_warn("NAND bus width %d instead %d bit\n",
|
2011-07-20 00:06:08 +07:00
|
|
|
(chip->options & NAND_BUSWIDTH_16) ? 16 : 8,
|
|
|
|
busw ? 16 : 8);
|
2006-05-23 16:54:38 +07:00
|
|
|
return ERR_PTR(-EINVAL);
|
|
|
|
}
|
2005-11-07 18:15:49 +07:00
|
|
|
|
2012-09-25 10:40:49 +07:00
|
|
|
nand_decode_bbm_options(mtd, chip, id_data);
|
|
|
|
|
2006-05-23 16:54:38 +07:00
|
|
|
/* Calculate the address shift from the page size */
|
2006-05-24 17:07:37 +07:00
|
|
|
chip->page_shift = ffs(mtd->writesize) - 1;
|
2011-05-26 04:59:01 +07:00
|
|
|
/* Convert chipsize to number of pages per chip -1 */
|
2006-05-24 17:07:37 +07:00
|
|
|
chip->pagemask = (chip->chipsize >> chip->page_shift) - 1;
|
2005-11-07 18:15:49 +07:00
|
|
|
|
2006-05-24 17:07:37 +07:00
|
|
|
chip->bbt_erase_shift = chip->phys_erase_shift =
|
2006-05-23 16:54:38 +07:00
|
|
|
ffs(mtd->erasesize) - 1;
|
2008-12-10 20:37:21 +07:00
|
|
|
if (chip->chipsize & 0xffffffff)
|
|
|
|
chip->chip_shift = ffs((unsigned)chip->chipsize) - 1;
|
2010-09-07 18:23:45 +07:00
|
|
|
else {
|
|
|
|
chip->chip_shift = ffs((unsigned)(chip->chipsize >> 32));
|
|
|
|
chip->chip_shift += 32 - 1;
|
|
|
|
}
|
2005-04-17 05:20:36 +07:00
|
|
|
|
2011-04-29 00:26:59 +07:00
|
|
|
chip->badblockbits = 8;
|
2013-03-04 19:21:34 +07:00
|
|
|
chip->erase_cmd = single_erase_cmd;
|
2006-05-23 16:54:38 +07:00
|
|
|
|
2011-05-26 04:59:01 +07:00
|
|
|
/* Do not replace user supplied command function! */
|
2006-05-24 17:07:37 +07:00
|
|
|
if (mtd->writesize > 512 && chip->cmdfunc == nand_command)
|
|
|
|
chip->cmdfunc = nand_command_lp;
|
2006-05-23 16:54:38 +07:00
|
|
|
|
2013-09-25 13:58:14 +07:00
|
|
|
pr_info("NAND device: Manufacturer ID: 0x%02x, Chip ID: 0x%02x (%s %s)\n",
|
2012-04-09 10:41:37 +07:00
|
|
|
*maf_id, *dev_id, nand_manuf_ids[maf_idx].name,
|
2013-09-25 13:58:14 +07:00
|
|
|
chip->onfi_version ? chip->onfi_params.model : type->name);
|
|
|
|
|
|
|
|
pr_info("NAND device: %dMiB, %s, page size: %d, OOB size: %d\n",
|
|
|
|
(int)(chip->chipsize >> 20), nand_is_slc(chip) ? "SLC" : "MLC",
|
|
|
|
mtd->writesize, mtd->oobsize);
|
2006-05-23 16:54:38 +07:00
|
|
|
|
|
|
|
return type;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2006-09-25 23:06:53 +07:00
|
|
|
* nand_scan_ident - [NAND Interface] Scan for the NAND device
|
2011-05-26 04:59:01 +07:00
|
|
|
* @mtd: MTD device structure
|
|
|
|
* @maxchips: number of chips to scan for
|
|
|
|
* @table: alternative NAND ID table
|
2006-05-23 16:54:38 +07:00
|
|
|
*
|
2011-05-26 04:59:01 +07:00
|
|
|
* This is the first phase of the normal nand_scan() function. It reads the
|
|
|
|
* flash ID and sets up MTD fields accordingly.
|
2006-05-23 16:54:38 +07:00
|
|
|
*
|
2006-09-25 23:06:53 +07:00
|
|
|
* The mtd->owner field must be set to the module of the caller.
|
2006-05-23 16:54:38 +07:00
|
|
|
*/
|
2010-02-27 01:32:56 +07:00
|
|
|
int nand_scan_ident(struct mtd_info *mtd, int maxchips,
|
|
|
|
struct nand_flash_dev *table)
|
2006-05-23 16:54:38 +07:00
|
|
|
{
|
2010-08-30 23:32:24 +07:00
|
|
|
int i, busw, nand_maf_id, nand_dev_id;
|
2006-05-24 17:07:37 +07:00
|
|
|
struct nand_chip *chip = mtd->priv;
|
2006-05-23 16:54:38 +07:00
|
|
|
struct nand_flash_dev *type;
|
|
|
|
|
|
|
|
/* Get buswidth to select the correct functions */
|
2006-05-24 17:07:37 +07:00
|
|
|
busw = chip->options & NAND_BUSWIDTH_16;
|
2006-05-23 16:54:38 +07:00
|
|
|
/* Set the default functions */
|
2006-05-24 17:07:37 +07:00
|
|
|
nand_set_defaults(chip, busw);
|
2006-05-23 16:54:38 +07:00
|
|
|
|
|
|
|
/* Read the flash type */
|
2010-09-07 18:23:45 +07:00
|
|
|
type = nand_get_flash_type(mtd, chip, busw,
|
|
|
|
&nand_maf_id, &nand_dev_id, table);
|
2006-05-23 16:54:38 +07:00
|
|
|
|
|
|
|
if (IS_ERR(type)) {
|
2009-11-03 01:12:33 +07:00
|
|
|
if (!(chip->options & NAND_SCAN_SILENT_NODEV))
|
2011-07-20 00:06:08 +07:00
|
|
|
pr_warn("No NAND device found\n");
|
2006-05-24 17:07:37 +07:00
|
|
|
chip->select_chip(mtd, -1);
|
2006-05-23 16:54:38 +07:00
|
|
|
return PTR_ERR(type);
|
2005-04-17 05:20:36 +07:00
|
|
|
}
|
|
|
|
|
2012-11-09 15:23:45 +07:00
|
|
|
chip->select_chip(mtd, -1);
|
|
|
|
|
2006-05-23 16:54:38 +07:00
|
|
|
/* Check for a chip array */
|
2006-05-14 00:07:53 +07:00
|
|
|
for (i = 1; i < maxchips; i++) {
|
2006-05-24 17:07:37 +07:00
|
|
|
chip->select_chip(mtd, i);
|
2008-09-15 19:37:29 +07:00
|
|
|
/* See comment in nand_get_flash_type for reset */
|
|
|
|
chip->cmdfunc(mtd, NAND_CMD_RESET, -1, -1);
|
2005-04-17 05:20:36 +07:00
|
|
|
/* Send the command for reading device ID */
|
2006-05-24 17:07:37 +07:00
|
|
|
chip->cmdfunc(mtd, NAND_CMD_READID, 0x00, -1);
|
2005-04-17 05:20:36 +07:00
|
|
|
/* Read manufacturer and device IDs */
|
2006-05-24 17:07:37 +07:00
|
|
|
if (nand_maf_id != chip->read_byte(mtd) ||
|
2012-11-09 15:23:45 +07:00
|
|
|
nand_dev_id != chip->read_byte(mtd)) {
|
|
|
|
chip->select_chip(mtd, -1);
|
2005-04-17 05:20:36 +07:00
|
|
|
break;
|
2012-11-09 15:23:45 +07:00
|
|
|
}
|
|
|
|
chip->select_chip(mtd, -1);
|
2005-04-17 05:20:36 +07:00
|
|
|
}
|
|
|
|
if (i > 1)
|
2011-07-20 00:06:07 +07:00
|
|
|
pr_info("%d NAND chips detected\n", i);
|
2005-11-07 18:15:49 +07:00
|
|
|
|
2005-04-17 05:20:36 +07:00
|
|
|
/* Store the number of chips and calc total size for mtd */
|
2006-05-24 17:07:37 +07:00
|
|
|
chip->numchips = i;
|
|
|
|
mtd->size = i * chip->chipsize;
|
2006-05-23 16:54:38 +07:00
|
|
|
|
2006-09-25 23:06:53 +07:00
|
|
|
return 0;
|
|
|
|
}
|
2010-09-07 18:23:45 +07:00
|
|
|
EXPORT_SYMBOL(nand_scan_ident);
|
2006-09-25 23:06:53 +07:00
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* nand_scan_tail - [NAND Interface] Scan for the NAND device
|
2011-05-26 04:59:01 +07:00
|
|
|
* @mtd: MTD device structure
|
2006-09-25 23:06:53 +07:00
|
|
|
*
|
2011-05-26 04:59:01 +07:00
|
|
|
* This is the second phase of the normal nand_scan() function. It fills out
|
|
|
|
* all the uninitialized function pointers with the defaults and scans for a
|
|
|
|
* bad block table if appropriate.
|
2006-09-25 23:06:53 +07:00
|
|
|
*/
|
|
|
|
int nand_scan_tail(struct mtd_info *mtd)
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
struct nand_chip *chip = mtd->priv;
|
|
|
|
|
mtd: nand: write BBM to OOB even with flash-based BBT
Currently, the flash-based BBT implementation writes bad block data only
to its flash-based table and not to the OOB marker area. Then, as new bad
blocks are marked over time, the OOB markers become incomplete and the
flash-based table becomes the only source of current bad block
information. This becomes an obvious problem when, for example:
* bootloader cannot read the flash-based BBT format
* BBT is corrupted and the flash must be rescanned for bad
blocks; we want to remember bad blocks that were marked from Linux
So to keep the bad block markers in sync with the flash-based BBT, this
patch changes the default so that we write bad block markers to the proper
OOB area on each block in addition to flash-based BBT. Comments are
updated, expanded, and/or relocated as necessary.
The new flash-based BBT procedure for marking bad blocks:
(1) erase the affected block, to allow OOB marker to be written cleanly
(2) update in-memory BBT
(3) write bad block marker to OOB area of affected block
(4) update flash-based BBT
Note that we retain the first error encountered in (3) or (4), finish the
procedures, and dump the error in the end.
This should handle power cuts gracefully enough. (1) and (2) are mostly
harmless (note that (1) will not erase an already-recognized bad block).
The OOB and BBT may be "out of sync" if we experience power loss bewteen
(3) and (4), but we can reasonably expect that on next boot, subsequent
I/O operations will discover that the block should be marked bad again,
thus re-syncing the OOB and BBT.
Note that this is a change from the previous default flash-based BBT
behavior. If your system cannot support writing bad block markers to OOB,
use the new NAND_BBT_NO_OOB_BBM option (in combination with
NAND_BBT_USE_FLASH and NAND_BBT_NO_OOB).
Signed-off-by: Brian Norris <computersforpeace@gmail.com>
Signed-off-by: Artem Bityutskiy <artem.bityutskiy@linux.intel.com>
Signed-off-by: David Woodhouse <David.Woodhouse@intel.com>
2012-02-07 04:44:00 +07:00
|
|
|
/* New bad blocks should be marked in OOB, flash-based BBT, or both */
|
|
|
|
BUG_ON((chip->bbt_options & NAND_BBT_NO_OOB_BBM) &&
|
|
|
|
!(chip->bbt_options & NAND_BBT_USE_FLASH));
|
|
|
|
|
2006-09-25 23:08:04 +07:00
|
|
|
if (!(chip->options & NAND_OWN_BUFFERS))
|
|
|
|
chip->buffers = kmalloc(sizeof(*chip->buffers), GFP_KERNEL);
|
|
|
|
if (!chip->buffers)
|
|
|
|
return -ENOMEM;
|
|
|
|
|
2006-10-21 23:09:53 +07:00
|
|
|
/* Set the internal oob buffer location, just after the page data */
|
2006-10-22 07:47:45 +07:00
|
|
|
chip->oob_poi = chip->buffers->databuf + mtd->writesize;
|
2005-04-17 05:20:36 +07:00
|
|
|
|
2006-05-23 16:54:38 +07:00
|
|
|
/*
|
2011-05-26 04:59:01 +07:00
|
|
|
* If no default placement scheme is given, select an appropriate one.
|
2006-05-23 16:54:38 +07:00
|
|
|
*/
|
2011-03-11 17:05:33 +07:00
|
|
|
if (!chip->ecc.layout && (chip->ecc.mode != NAND_ECC_SOFT_BCH)) {
|
2005-11-07 18:15:49 +07:00
|
|
|
switch (mtd->oobsize) {
|
2005-04-17 05:20:36 +07:00
|
|
|
case 8:
|
2006-05-28 03:16:10 +07:00
|
|
|
chip->ecc.layout = &nand_oob_8;
|
2005-04-17 05:20:36 +07:00
|
|
|
break;
|
|
|
|
case 16:
|
2006-05-28 03:16:10 +07:00
|
|
|
chip->ecc.layout = &nand_oob_16;
|
2005-04-17 05:20:36 +07:00
|
|
|
break;
|
|
|
|
case 64:
|
2006-05-28 03:16:10 +07:00
|
|
|
chip->ecc.layout = &nand_oob_64;
|
2005-04-17 05:20:36 +07:00
|
|
|
break;
|
2007-12-12 23:27:03 +07:00
|
|
|
case 128:
|
|
|
|
chip->ecc.layout = &nand_oob_128;
|
|
|
|
break;
|
2005-04-17 05:20:36 +07:00
|
|
|
default:
|
2011-07-20 00:06:08 +07:00
|
|
|
pr_warn("No oob scheme defined for oobsize %d\n",
|
|
|
|
mtd->oobsize);
|
2005-04-17 05:20:36 +07:00
|
|
|
BUG();
|
|
|
|
}
|
|
|
|
}
|
2005-11-07 18:15:49 +07:00
|
|
|
|
2006-09-25 23:12:39 +07:00
|
|
|
if (!chip->write_page)
|
|
|
|
chip->write_page = nand_write_page;
|
|
|
|
|
2005-11-07 18:15:49 +07:00
|
|
|
/*
|
2011-05-26 04:59:01 +07:00
|
|
|
* Check ECC mode, default to software if 3byte/512byte hardware ECC is
|
2006-05-23 16:54:38 +07:00
|
|
|
* selected and we have 256 byte pagesize fallback to software ECC
|
2006-05-14 00:07:53 +07:00
|
|
|
*/
|
2006-09-25 23:12:39 +07:00
|
|
|
|
2006-05-24 17:07:37 +07:00
|
|
|
switch (chip->ecc.mode) {
|
2009-09-19 02:51:47 +07:00
|
|
|
case NAND_ECC_HW_OOB_FIRST:
|
|
|
|
/* Similar to NAND_ECC_HW, but a separate read_page handle */
|
|
|
|
if (!chip->ecc.calculate || !chip->ecc.correct ||
|
|
|
|
!chip->ecc.hwctl) {
|
2011-07-20 00:06:07 +07:00
|
|
|
pr_warn("No ECC functions supplied; "
|
2011-07-20 00:06:08 +07:00
|
|
|
"hardware ECC not possible\n");
|
2009-09-19 02:51:47 +07:00
|
|
|
BUG();
|
|
|
|
}
|
|
|
|
if (!chip->ecc.read_page)
|
|
|
|
chip->ecc.read_page = nand_read_page_hwecc_oob_first;
|
|
|
|
|
2006-05-23 17:00:46 +07:00
|
|
|
case NAND_ECC_HW:
|
2011-05-26 04:59:01 +07:00
|
|
|
/* Use standard hwecc read page function? */
|
2006-05-25 15:07:16 +07:00
|
|
|
if (!chip->ecc.read_page)
|
|
|
|
chip->ecc.read_page = nand_read_page_hwecc;
|
2006-05-26 23:52:08 +07:00
|
|
|
if (!chip->ecc.write_page)
|
|
|
|
chip->ecc.write_page = nand_write_page_hwecc;
|
[MTD] [NAND] fix "raw" reads with ECC syndrome layouts
The syndrome based page read/write routines store ECC, and possibly other
"OOB" data, right after each chunk of ECC'd data. With ECC chunk size of
512 bytes and a large page (2KiB) NAND, the layout is:
data-0 OOB-0 data-1 OOB-1 data-2 OOB-2 data-3 OOB-3 OOB-leftover
Where OOBx is (prepad, ECC, postpad). However, the current "raw" routines
use a traditional layout -- data OOB, disregarding the prepad and postpad
values -- so when they're used with that type of ECC hardware, those calls
mix up the data and OOB. Which means, in particular, that bad block
tables won't be found on startup, with data corruption and related chaos
ensuing.
The current syndrome-based drivers in mainline all seem to use one chunk
per page; presumably they haven't noticed such bugs.
Fix this, by adding read/write page_raw_syndrome() routines as siblings of
the existing non-raw routines; "raw" just means to bypass the ECC
computations, not change data and OOB layout.
Signed-off-by: David Brownell <dbrownell@users.sourceforge.net>
Signed-off-by: Andrew Morton <akpm@linux-foundation.org>
Signed-off-by: David Woodhouse <David.Woodhouse@intel.com>
2009-03-05 03:01:36 +07:00
|
|
|
if (!chip->ecc.read_page_raw)
|
|
|
|
chip->ecc.read_page_raw = nand_read_page_raw;
|
|
|
|
if (!chip->ecc.write_page_raw)
|
|
|
|
chip->ecc.write_page_raw = nand_write_page_raw;
|
2006-06-21 01:05:05 +07:00
|
|
|
if (!chip->ecc.read_oob)
|
|
|
|
chip->ecc.read_oob = nand_read_oob_std;
|
|
|
|
if (!chip->ecc.write_oob)
|
|
|
|
chip->ecc.write_oob = nand_write_oob_std;
|
2013-03-15 19:25:53 +07:00
|
|
|
if (!chip->ecc.read_subpage)
|
|
|
|
chip->ecc.read_subpage = nand_read_subpage;
|
|
|
|
if (!chip->ecc.write_subpage)
|
|
|
|
chip->ecc.write_subpage = nand_write_subpage_hwecc;
|
2006-05-25 15:07:16 +07:00
|
|
|
|
2006-05-23 17:00:46 +07:00
|
|
|
case NAND_ECC_HW_SYNDROME:
|
2007-12-14 00:15:28 +07:00
|
|
|
if ((!chip->ecc.calculate || !chip->ecc.correct ||
|
|
|
|
!chip->ecc.hwctl) &&
|
|
|
|
(!chip->ecc.read_page ||
|
2008-01-16 23:36:03 +07:00
|
|
|
chip->ecc.read_page == nand_read_page_hwecc ||
|
2007-12-14 00:15:28 +07:00
|
|
|
!chip->ecc.write_page ||
|
2008-01-16 23:36:03 +07:00
|
|
|
chip->ecc.write_page == nand_write_page_hwecc)) {
|
2011-07-20 00:06:07 +07:00
|
|
|
pr_warn("No ECC functions supplied; "
|
2011-07-20 00:06:08 +07:00
|
|
|
"hardware ECC not possible\n");
|
2006-05-23 17:00:46 +07:00
|
|
|
BUG();
|
|
|
|
}
|
2011-05-26 04:59:01 +07:00
|
|
|
/* Use standard syndrome read/write page function? */
|
2006-05-25 15:07:16 +07:00
|
|
|
if (!chip->ecc.read_page)
|
|
|
|
chip->ecc.read_page = nand_read_page_syndrome;
|
2006-05-26 23:52:08 +07:00
|
|
|
if (!chip->ecc.write_page)
|
|
|
|
chip->ecc.write_page = nand_write_page_syndrome;
|
[MTD] [NAND] fix "raw" reads with ECC syndrome layouts
The syndrome based page read/write routines store ECC, and possibly other
"OOB" data, right after each chunk of ECC'd data. With ECC chunk size of
512 bytes and a large page (2KiB) NAND, the layout is:
data-0 OOB-0 data-1 OOB-1 data-2 OOB-2 data-3 OOB-3 OOB-leftover
Where OOBx is (prepad, ECC, postpad). However, the current "raw" routines
use a traditional layout -- data OOB, disregarding the prepad and postpad
values -- so when they're used with that type of ECC hardware, those calls
mix up the data and OOB. Which means, in particular, that bad block
tables won't be found on startup, with data corruption and related chaos
ensuing.
The current syndrome-based drivers in mainline all seem to use one chunk
per page; presumably they haven't noticed such bugs.
Fix this, by adding read/write page_raw_syndrome() routines as siblings of
the existing non-raw routines; "raw" just means to bypass the ECC
computations, not change data and OOB layout.
Signed-off-by: David Brownell <dbrownell@users.sourceforge.net>
Signed-off-by: Andrew Morton <akpm@linux-foundation.org>
Signed-off-by: David Woodhouse <David.Woodhouse@intel.com>
2009-03-05 03:01:36 +07:00
|
|
|
if (!chip->ecc.read_page_raw)
|
|
|
|
chip->ecc.read_page_raw = nand_read_page_raw_syndrome;
|
|
|
|
if (!chip->ecc.write_page_raw)
|
|
|
|
chip->ecc.write_page_raw = nand_write_page_raw_syndrome;
|
2006-06-21 01:05:05 +07:00
|
|
|
if (!chip->ecc.read_oob)
|
|
|
|
chip->ecc.read_oob = nand_read_oob_syndrome;
|
|
|
|
if (!chip->ecc.write_oob)
|
|
|
|
chip->ecc.write_oob = nand_write_oob_syndrome;
|
2006-05-25 15:07:16 +07:00
|
|
|
|
2012-04-26 02:06:10 +07:00
|
|
|
if (mtd->writesize >= chip->ecc.size) {
|
|
|
|
if (!chip->ecc.strength) {
|
|
|
|
pr_warn("Driver must set ecc.strength when using hardware ECC\n");
|
|
|
|
BUG();
|
|
|
|
}
|
2006-05-23 17:00:46 +07:00
|
|
|
break;
|
2012-04-26 02:06:10 +07:00
|
|
|
}
|
2011-07-20 00:06:07 +07:00
|
|
|
pr_warn("%d byte HW ECC not possible on "
|
2011-07-20 00:06:08 +07:00
|
|
|
"%d byte page size, fallback to SW ECC\n",
|
|
|
|
chip->ecc.size, mtd->writesize);
|
2006-05-24 17:07:37 +07:00
|
|
|
chip->ecc.mode = NAND_ECC_SOFT;
|
2005-11-07 18:15:49 +07:00
|
|
|
|
2006-05-23 17:00:46 +07:00
|
|
|
case NAND_ECC_SOFT:
|
2006-05-24 17:07:37 +07:00
|
|
|
chip->ecc.calculate = nand_calculate_ecc;
|
|
|
|
chip->ecc.correct = nand_correct_data;
|
2006-05-25 15:07:16 +07:00
|
|
|
chip->ecc.read_page = nand_read_page_swecc;
|
2008-05-15 23:23:18 +07:00
|
|
|
chip->ecc.read_subpage = nand_read_subpage;
|
2006-05-26 23:52:08 +07:00
|
|
|
chip->ecc.write_page = nand_write_page_swecc;
|
[MTD] [NAND] fix "raw" reads with ECC syndrome layouts
The syndrome based page read/write routines store ECC, and possibly other
"OOB" data, right after each chunk of ECC'd data. With ECC chunk size of
512 bytes and a large page (2KiB) NAND, the layout is:
data-0 OOB-0 data-1 OOB-1 data-2 OOB-2 data-3 OOB-3 OOB-leftover
Where OOBx is (prepad, ECC, postpad). However, the current "raw" routines
use a traditional layout -- data OOB, disregarding the prepad and postpad
values -- so when they're used with that type of ECC hardware, those calls
mix up the data and OOB. Which means, in particular, that bad block
tables won't be found on startup, with data corruption and related chaos
ensuing.
The current syndrome-based drivers in mainline all seem to use one chunk
per page; presumably they haven't noticed such bugs.
Fix this, by adding read/write page_raw_syndrome() routines as siblings of
the existing non-raw routines; "raw" just means to bypass the ECC
computations, not change data and OOB layout.
Signed-off-by: David Brownell <dbrownell@users.sourceforge.net>
Signed-off-by: Andrew Morton <akpm@linux-foundation.org>
Signed-off-by: David Woodhouse <David.Woodhouse@intel.com>
2009-03-05 03:01:36 +07:00
|
|
|
chip->ecc.read_page_raw = nand_read_page_raw;
|
|
|
|
chip->ecc.write_page_raw = nand_write_page_raw;
|
2006-06-21 01:05:05 +07:00
|
|
|
chip->ecc.read_oob = nand_read_oob_std;
|
|
|
|
chip->ecc.write_oob = nand_write_oob_std;
|
2008-12-12 07:10:57 +07:00
|
|
|
if (!chip->ecc.size)
|
|
|
|
chip->ecc.size = 256;
|
2006-05-24 17:07:37 +07:00
|
|
|
chip->ecc.bytes = 3;
|
2012-03-12 04:21:11 +07:00
|
|
|
chip->ecc.strength = 1;
|
2005-04-17 05:20:36 +07:00
|
|
|
break;
|
2005-11-07 18:15:49 +07:00
|
|
|
|
2011-03-11 17:05:33 +07:00
|
|
|
case NAND_ECC_SOFT_BCH:
|
|
|
|
if (!mtd_nand_has_bch()) {
|
2011-07-20 00:06:07 +07:00
|
|
|
pr_warn("CONFIG_MTD_ECC_BCH not enabled\n");
|
2011-03-11 17:05:33 +07:00
|
|
|
BUG();
|
|
|
|
}
|
|
|
|
chip->ecc.calculate = nand_bch_calculate_ecc;
|
|
|
|
chip->ecc.correct = nand_bch_correct_data;
|
|
|
|
chip->ecc.read_page = nand_read_page_swecc;
|
|
|
|
chip->ecc.read_subpage = nand_read_subpage;
|
|
|
|
chip->ecc.write_page = nand_write_page_swecc;
|
|
|
|
chip->ecc.read_page_raw = nand_read_page_raw;
|
|
|
|
chip->ecc.write_page_raw = nand_write_page_raw;
|
|
|
|
chip->ecc.read_oob = nand_read_oob_std;
|
|
|
|
chip->ecc.write_oob = nand_write_oob_std;
|
|
|
|
/*
|
|
|
|
* Board driver should supply ecc.size and ecc.bytes values to
|
|
|
|
* select how many bits are correctable; see nand_bch_init()
|
2011-05-26 04:59:01 +07:00
|
|
|
* for details. Otherwise, default to 4 bits for large page
|
|
|
|
* devices.
|
2011-03-11 17:05:33 +07:00
|
|
|
*/
|
|
|
|
if (!chip->ecc.size && (mtd->oobsize >= 64)) {
|
|
|
|
chip->ecc.size = 512;
|
|
|
|
chip->ecc.bytes = 7;
|
|
|
|
}
|
|
|
|
chip->ecc.priv = nand_bch_init(mtd,
|
|
|
|
chip->ecc.size,
|
|
|
|
chip->ecc.bytes,
|
|
|
|
&chip->ecc.layout);
|
|
|
|
if (!chip->ecc.priv) {
|
2011-07-20 00:06:07 +07:00
|
|
|
pr_warn("BCH ECC initialization failed!\n");
|
2011-03-11 17:05:33 +07:00
|
|
|
BUG();
|
|
|
|
}
|
2012-03-12 04:21:11 +07:00
|
|
|
chip->ecc.strength =
|
2012-04-26 02:06:10 +07:00
|
|
|
chip->ecc.bytes * 8 / fls(8 * chip->ecc.size);
|
2011-03-11 17:05:33 +07:00
|
|
|
break;
|
|
|
|
|
2005-11-07 18:15:49 +07:00
|
|
|
case NAND_ECC_NONE:
|
2011-07-20 00:06:07 +07:00
|
|
|
pr_warn("NAND_ECC_NONE selected by board driver. "
|
2011-07-20 00:06:08 +07:00
|
|
|
"This is not recommended!\n");
|
2006-05-29 08:26:58 +07:00
|
|
|
chip->ecc.read_page = nand_read_page_raw;
|
|
|
|
chip->ecc.write_page = nand_write_page_raw;
|
2006-06-21 01:05:05 +07:00
|
|
|
chip->ecc.read_oob = nand_read_oob_std;
|
[MTD] [NAND] fix "raw" reads with ECC syndrome layouts
The syndrome based page read/write routines store ECC, and possibly other
"OOB" data, right after each chunk of ECC'd data. With ECC chunk size of
512 bytes and a large page (2KiB) NAND, the layout is:
data-0 OOB-0 data-1 OOB-1 data-2 OOB-2 data-3 OOB-3 OOB-leftover
Where OOBx is (prepad, ECC, postpad). However, the current "raw" routines
use a traditional layout -- data OOB, disregarding the prepad and postpad
values -- so when they're used with that type of ECC hardware, those calls
mix up the data and OOB. Which means, in particular, that bad block
tables won't be found on startup, with data corruption and related chaos
ensuing.
The current syndrome-based drivers in mainline all seem to use one chunk
per page; presumably they haven't noticed such bugs.
Fix this, by adding read/write page_raw_syndrome() routines as siblings of
the existing non-raw routines; "raw" just means to bypass the ECC
computations, not change data and OOB layout.
Signed-off-by: David Brownell <dbrownell@users.sourceforge.net>
Signed-off-by: Andrew Morton <akpm@linux-foundation.org>
Signed-off-by: David Woodhouse <David.Woodhouse@intel.com>
2009-03-05 03:01:36 +07:00
|
|
|
chip->ecc.read_page_raw = nand_read_page_raw;
|
|
|
|
chip->ecc.write_page_raw = nand_write_page_raw;
|
2006-06-21 01:05:05 +07:00
|
|
|
chip->ecc.write_oob = nand_write_oob_std;
|
2006-05-24 17:07:37 +07:00
|
|
|
chip->ecc.size = mtd->writesize;
|
|
|
|
chip->ecc.bytes = 0;
|
2012-03-12 04:21:11 +07:00
|
|
|
chip->ecc.strength = 0;
|
2005-04-17 05:20:36 +07:00
|
|
|
break;
|
2006-09-25 23:12:39 +07:00
|
|
|
|
2005-04-17 05:20:36 +07:00
|
|
|
default:
|
2011-07-20 00:06:08 +07:00
|
|
|
pr_warn("Invalid NAND_ECC_MODE %d\n", chip->ecc.mode);
|
2005-11-07 18:15:49 +07:00
|
|
|
BUG();
|
2005-04-17 05:20:36 +07:00
|
|
|
}
|
2005-11-07 18:15:49 +07:00
|
|
|
|
2011-08-31 08:45:37 +07:00
|
|
|
/* For many systems, the standard OOB write also works for raw */
|
2011-08-31 08:45:38 +07:00
|
|
|
if (!chip->ecc.read_oob_raw)
|
|
|
|
chip->ecc.read_oob_raw = chip->ecc.read_oob;
|
2011-08-31 08:45:37 +07:00
|
|
|
if (!chip->ecc.write_oob_raw)
|
|
|
|
chip->ecc.write_oob_raw = chip->ecc.write_oob;
|
|
|
|
|
2006-05-28 03:16:10 +07:00
|
|
|
/*
|
|
|
|
* The number of bytes available for a client to place data into
|
2011-05-26 04:59:01 +07:00
|
|
|
* the out of band area.
|
2006-05-28 03:16:10 +07:00
|
|
|
*/
|
|
|
|
chip->ecc.layout->oobavail = 0;
|
2009-04-22 09:51:20 +07:00
|
|
|
for (i = 0; chip->ecc.layout->oobfree[i].length
|
|
|
|
&& i < ARRAY_SIZE(chip->ecc.layout->oobfree); i++)
|
2006-05-28 03:16:10 +07:00
|
|
|
chip->ecc.layout->oobavail +=
|
|
|
|
chip->ecc.layout->oobfree[i].length;
|
2007-03-06 20:56:34 +07:00
|
|
|
mtd->oobavail = chip->ecc.layout->oobavail;
|
2006-05-28 03:16:10 +07:00
|
|
|
|
2006-05-23 16:54:38 +07:00
|
|
|
/*
|
|
|
|
* Set the number of read / write steps for one page depending on ECC
|
2011-05-26 04:59:01 +07:00
|
|
|
* mode.
|
2006-05-23 16:54:38 +07:00
|
|
|
*/
|
2006-05-24 17:07:37 +07:00
|
|
|
chip->ecc.steps = mtd->writesize / chip->ecc.size;
|
2010-09-07 18:23:43 +07:00
|
|
|
if (chip->ecc.steps * chip->ecc.size != mtd->writesize) {
|
2011-07-20 00:06:07 +07:00
|
|
|
pr_warn("Invalid ECC parameters\n");
|
2006-05-23 17:00:46 +07:00
|
|
|
BUG();
|
2005-04-17 05:20:36 +07:00
|
|
|
}
|
2006-05-25 15:07:16 +07:00
|
|
|
chip->ecc.total = chip->ecc.steps * chip->ecc.bytes;
|
2005-11-07 18:15:49 +07:00
|
|
|
|
2011-05-26 04:59:01 +07:00
|
|
|
/* Allow subpage writes up to ecc.steps. Not possible for MLC flash */
|
2013-09-25 13:58:10 +07:00
|
|
|
if (!(chip->options & NAND_NO_SUBPAGE_WRITE) && nand_is_slc(chip)) {
|
2010-09-07 18:23:43 +07:00
|
|
|
switch (chip->ecc.steps) {
|
2006-09-28 20:38:36 +07:00
|
|
|
case 2:
|
|
|
|
mtd->subpage_sft = 1;
|
|
|
|
break;
|
|
|
|
case 4:
|
|
|
|
case 8:
|
2007-12-12 23:27:03 +07:00
|
|
|
case 16:
|
2006-09-28 20:38:36 +07:00
|
|
|
mtd->subpage_sft = 2;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
chip->subpagesize = mtd->writesize >> mtd->subpage_sft;
|
|
|
|
|
2006-05-25 14:45:29 +07:00
|
|
|
/* Initialize state */
|
2006-05-24 17:07:37 +07:00
|
|
|
chip->state = FL_READY;
|
2005-04-17 05:20:36 +07:00
|
|
|
|
|
|
|
/* Invalidate the pagebuffer reference */
|
2006-05-24 17:07:37 +07:00
|
|
|
chip->pagebuf = -1;
|
2005-04-17 05:20:36 +07:00
|
|
|
|
2012-08-14 04:35:30 +07:00
|
|
|
/* Large page NAND with SOFT_ECC should support subpage reads */
|
|
|
|
if ((chip->ecc.mode == NAND_ECC_SOFT) && (chip->page_shift > 9))
|
|
|
|
chip->options |= NAND_SUBPAGE_READ;
|
|
|
|
|
2005-04-17 05:20:36 +07:00
|
|
|
/* Fill in remaining MTD driver data */
|
|
|
|
mtd->type = MTD_NANDFLASH;
|
2010-02-23 01:39:40 +07:00
|
|
|
mtd->flags = (chip->options & NAND_ROM) ? MTD_CAP_ROM :
|
|
|
|
MTD_CAP_NANDFLASH;
|
2012-01-30 19:58:32 +07:00
|
|
|
mtd->_erase = nand_erase;
|
|
|
|
mtd->_point = NULL;
|
|
|
|
mtd->_unpoint = NULL;
|
|
|
|
mtd->_read = nand_read;
|
|
|
|
mtd->_write = nand_write;
|
|
|
|
mtd->_panic_write = panic_nand_write;
|
|
|
|
mtd->_read_oob = nand_read_oob;
|
|
|
|
mtd->_write_oob = nand_write_oob;
|
|
|
|
mtd->_sync = nand_sync;
|
|
|
|
mtd->_lock = NULL;
|
|
|
|
mtd->_unlock = NULL;
|
|
|
|
mtd->_suspend = nand_suspend;
|
|
|
|
mtd->_resume = nand_resume;
|
|
|
|
mtd->_block_isbad = nand_block_isbad;
|
|
|
|
mtd->_block_markbad = nand_block_markbad;
|
2010-12-17 05:42:16 +07:00
|
|
|
mtd->writebufsize = mtd->writesize;
|
2005-04-17 05:20:36 +07:00
|
|
|
|
2012-03-12 04:21:11 +07:00
|
|
|
/* propagate ecc info to mtd_info */
|
2006-05-28 03:16:10 +07:00
|
|
|
mtd->ecclayout = chip->ecc.layout;
|
2012-04-26 02:06:05 +07:00
|
|
|
mtd->ecc_strength = chip->ecc.strength;
|
2013-08-16 09:10:06 +07:00
|
|
|
mtd->ecc_step_size = chip->ecc.size;
|
2012-06-08 22:29:06 +07:00
|
|
|
/*
|
|
|
|
* Initialize bitflip_threshold to its default prior scan_bbt() call.
|
|
|
|
* scan_bbt() might invoke mtd_read(), thus bitflip_threshold must be
|
|
|
|
* properly set.
|
|
|
|
*/
|
|
|
|
if (!mtd->bitflip_threshold)
|
|
|
|
mtd->bitflip_threshold = mtd->ecc_strength;
|
2005-04-17 05:20:36 +07:00
|
|
|
|
2005-02-09 19:20:00 +07:00
|
|
|
/* Check, if we should skip the bad block table scan */
|
2006-05-24 17:07:37 +07:00
|
|
|
if (chip->options & NAND_SKIP_BBTSCAN)
|
2005-02-09 19:20:00 +07:00
|
|
|
return 0;
|
2005-04-17 05:20:36 +07:00
|
|
|
|
|
|
|
/* Build bad block table */
|
2006-05-24 17:07:37 +07:00
|
|
|
return chip->scan_bbt(mtd);
|
2005-04-17 05:20:36 +07:00
|
|
|
}
|
2010-09-07 18:23:45 +07:00
|
|
|
EXPORT_SYMBOL(nand_scan_tail);
|
2005-04-17 05:20:36 +07:00
|
|
|
|
2011-05-26 04:59:01 +07:00
|
|
|
/*
|
|
|
|
* is_module_text_address() isn't exported, and it's mostly a pointless
|
2010-09-07 18:23:45 +07:00
|
|
|
* test if this is a module _anyway_ -- they'd have to try _really_ hard
|
2011-05-26 04:59:01 +07:00
|
|
|
* to call us from in-kernel code if the core NAND support is modular.
|
|
|
|
*/
|
2006-09-25 23:06:53 +07:00
|
|
|
#ifdef MODULE
|
|
|
|
#define caller_is_module() (1)
|
|
|
|
#else
|
|
|
|
#define caller_is_module() \
|
2009-04-01 02:05:31 +07:00
|
|
|
is_module_text_address((unsigned long)__builtin_return_address(0))
|
2006-09-25 23:06:53 +07:00
|
|
|
#endif
|
|
|
|
|
|
|
|
/**
|
|
|
|
* nand_scan - [NAND Interface] Scan for the NAND device
|
2011-05-26 04:59:01 +07:00
|
|
|
* @mtd: MTD device structure
|
|
|
|
* @maxchips: number of chips to scan for
|
2006-09-25 23:06:53 +07:00
|
|
|
*
|
2011-05-26 04:59:01 +07:00
|
|
|
* This fills out all the uninitialized function pointers with the defaults.
|
|
|
|
* The flash ID is read and the mtd/chip structures are filled with the
|
|
|
|
* appropriate values. The mtd->owner field must be set to the module of the
|
|
|
|
* caller.
|
2006-09-25 23:06:53 +07:00
|
|
|
*/
|
|
|
|
int nand_scan(struct mtd_info *mtd, int maxchips)
|
|
|
|
{
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
/* Many callers got this wrong, so check for it for a while... */
|
|
|
|
if (!mtd->owner && caller_is_module()) {
|
2011-07-20 00:06:08 +07:00
|
|
|
pr_crit("%s called with NULL mtd->owner!\n", __func__);
|
2006-09-25 23:06:53 +07:00
|
|
|
BUG();
|
|
|
|
}
|
|
|
|
|
2010-02-27 01:32:56 +07:00
|
|
|
ret = nand_scan_ident(mtd, maxchips, NULL);
|
2006-09-25 23:06:53 +07:00
|
|
|
if (!ret)
|
|
|
|
ret = nand_scan_tail(mtd);
|
|
|
|
return ret;
|
|
|
|
}
|
2010-09-07 18:23:45 +07:00
|
|
|
EXPORT_SYMBOL(nand_scan);
|
2006-09-25 23:06:53 +07:00
|
|
|
|
2005-04-17 05:20:36 +07:00
|
|
|
/**
|
2005-11-07 18:15:49 +07:00
|
|
|
* nand_release - [NAND Interface] Free resources held by the NAND device
|
2011-05-26 04:59:01 +07:00
|
|
|
* @mtd: MTD device structure
|
|
|
|
*/
|
2006-05-14 00:07:53 +07:00
|
|
|
void nand_release(struct mtd_info *mtd)
|
2005-04-17 05:20:36 +07:00
|
|
|
{
|
2006-05-24 17:07:37 +07:00
|
|
|
struct nand_chip *chip = mtd->priv;
|
2005-04-17 05:20:36 +07:00
|
|
|
|
2011-03-11 17:05:33 +07:00
|
|
|
if (chip->ecc.mode == NAND_ECC_SOFT_BCH)
|
|
|
|
nand_bch_free((struct nand_bch_control *)chip->ecc.priv);
|
|
|
|
|
2011-05-23 16:22:46 +07:00
|
|
|
mtd_device_unregister(mtd);
|
2005-04-17 05:20:36 +07:00
|
|
|
|
2005-11-07 16:01:27 +07:00
|
|
|
/* Free bad block table memory */
|
2006-05-24 17:07:37 +07:00
|
|
|
kfree(chip->bbt);
|
2006-09-25 23:08:04 +07:00
|
|
|
if (!(chip->options & NAND_OWN_BUFFERS))
|
|
|
|
kfree(chip->buffers);
|
mtd: nand: more BB Detection refactoring and dynamic scan options
This is a revision to PATCH 2/2 that I sent. Link:
http://lists.infradead.org/pipermail/linux-mtd/2010-July/030911.html
Added new flag for scanning of both bytes 1 and 6 of the OOB for
a BB marker (instead of simply one or the other).
The "check_pattern" and "check_short_pattern" functions were updated
to include support for scanning the two different locations in the OOB.
In order to handle increases in variety of necessary scanning patterns,
I implemented dynamic memory allocation of nand_bbt_descr structs
in new function 'nand_create_default_bbt_descr()'. This replaces
some increasingly-unwieldy, statically-declared descriptors. It can
replace several more (e.g. "flashbased" structs). However, I do not
test the flashbased options personally.
How this was tested:
I referenced 30+ data sheets (covering 100+ parts), and I tested a
selection of 10 different chips to varying degrees. Particularly, I
tested the creation of bad-block descriptors and basic BB scanning on
three parts:
ST NAND04GW3B2D, 2K page
ST NAND128W3A, 512B page
Samsung K9F1G08U0A, 2K page
To test these, I wrote some fake bad block markers to the flash (in OOB
bytes 1, 6, and elsewhere) to see if the scanning routine would detect
them properly. However, this method was somewhat limited because the
driver I am using has some bugs in its OOB write functionality.
Signed-off-by: Brian Norris <norris@broadcom.com>
Signed-off-by: Artem Bityutskiy <Artem.Bityutskiy@nokia.com>
Signed-off-by: David Woodhouse <David.Woodhouse@intel.com>
2010-07-16 02:15:44 +07:00
|
|
|
|
|
|
|
/* Free bad block descriptor memory */
|
|
|
|
if (chip->badblock_pattern && chip->badblock_pattern->options
|
|
|
|
& NAND_BBT_DYNAMICSTRUCT)
|
|
|
|
kfree(chip->badblock_pattern);
|
2005-04-17 05:20:36 +07:00
|
|
|
}
|
2006-05-14 00:07:53 +07:00
|
|
|
EXPORT_SYMBOL_GPL(nand_release);
|
2006-03-31 17:31:14 +07:00
|
|
|
|
|
|
|
static int __init nand_base_init(void)
|
|
|
|
{
|
|
|
|
led_trigger_register_simple("nand-disk", &nand_led_trigger);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void __exit nand_base_exit(void)
|
|
|
|
{
|
|
|
|
led_trigger_unregister_simple(nand_led_trigger);
|
|
|
|
}
|
|
|
|
|
|
|
|
module_init(nand_base_init);
|
|
|
|
module_exit(nand_base_exit);
|
|
|
|
|
2006-05-14 00:07:53 +07:00
|
|
|
MODULE_LICENSE("GPL");
|
2010-09-07 18:23:45 +07:00
|
|
|
MODULE_AUTHOR("Steven J. Hill <sjhill@realitydiluted.com>");
|
|
|
|
MODULE_AUTHOR("Thomas Gleixner <tglx@linutronix.de>");
|
2006-05-14 00:07:53 +07:00
|
|
|
MODULE_DESCRIPTION("Generic NAND flash driver code");
|