Skip to content
Snippets Groups Projects
nand_base.c 70.7 KiB
Newer Older
  • Learn to ignore specific revisions
  • /*
     *  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.
     *   Basic support for AG-AND chips is provided.
    
    Wolfgang Denk's avatar
    Wolfgang Denk committed
     *
    
     *	Additional technical information is available on
     *	http://www.linux-mtd.infradead.org/tech/nand.html
    
    Wolfgang Denk's avatar
    Wolfgang Denk committed
     *
    
     *  Copyright (C) 2000 Steven J. Hill (sjhill@realitydiluted.com)
    
     *		  2002-2006 Thomas Gleixner (tglx@linutronix.de)
    
     *  Credits:
    
    Wolfgang Denk's avatar
    Wolfgang Denk committed
     *	David Woodhouse for adding multichip support
     *
    
     *	Aleph One Ltd. and Toby Churchill Ltd. for supporting the
     *	rework for 2K page size chips
     *
    
     *	Enable cached programming for 2k page size chips
     *	Check, if mtd->ecctype should be set to MTD_ECC_HW
     *	if we have HW ecc support.
     *	The AG-AND chips have nice features for speed improvement,
     *	which are not supported yet. Read / program 4 pages in one go.
     *
     * 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.
     *
     */
    
    /* XXX U-BOOT XXX */
    #if 0
    
    #include <linux/module.h>
    
    #include <linux/delay.h>
    #include <linux/errno.h>
    
    #include <linux/err.h>
    
    #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>
    #include <linux/mtd/compatmac.h>
    #include <linux/interrupt.h>
    #include <linux/bitops.h>
    
    #include <linux/leds.h>
    
    #include <asm/io.h>
    
    #ifdef CONFIG_MTD_PARTITIONS
    #include <linux/mtd/partitions.h>
    #endif
    
    
    Wolfgang Denk's avatar
    Wolfgang Denk committed
    #endif
    
    #define ENOTSUPP	524	/* Operation is not supported */
    
    
    #if defined(CONFIG_CMD_NAND) && !defined(CFG_NAND_LEGACY)
    
    
    #include <malloc.h>
    #include <watchdog.h>
    
    #include <linux/err.h>
    
    #include <linux/mtd/compat.h>
    #include <linux/mtd/mtd.h>
    #include <linux/mtd/nand.h>
    #include <linux/mtd/nand_ecc.h>
    
    #include <asm/io.h>
    #include <asm/errno.h>
    
    #ifdef CONFIG_JFFS2_NAND
    #include <jffs2/jffs2.h>
    #endif
    
    /* Define default oob placement schemes for large and small page devices */
    
    static struct nand_ecclayout nand_oob_8 = {
    
    	.eccbytes = 3,
    	.eccpos = {0, 1, 2},
    
    	.oobfree = {
    		{.offset = 3,
    		 .length = 2},
    		{.offset = 6,
    		 .length = 2}}
    
    static struct nand_ecclayout nand_oob_16 = {
    
    	.eccbytes = 6,
    	.eccpos = {0, 1, 2, 3, 6, 7},
    
    	.oobfree = {
    		{.offset = 8,
    		 . length = 8}}
    
    static struct nand_ecclayout nand_oob_64 = {
    
    		   40, 41, 42, 43, 44, 45, 46, 47,
    		   48, 49, 50, 51, 52, 53, 54, 55,
    		   56, 57, 58, 59, 60, 61, 62, 63},
    	.oobfree = {
    		{.offset = 2,
    		 .length = 38}}
    
    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,
    		 .length = 78}}
    
    
    static int nand_get_device(struct nand_chip *chip, struct mtd_info *mtd,
    			   int new_state);
    
    static int nand_do_write_oob(struct mtd_info *mtd, loff_t to,
    			     struct mtd_oob_ops *ops);
    
    static int nand_wait(struct mtd_info *mtd, struct nand_chip *this);
    
     * For devices which display every fart in the system on a seperate LED. Is
     * compiled away when LED support is disabled.
    
    DEFINE_LED_TRIGGER(nand_led_trigger);
    
    /**
     * nand_release_device - [GENERIC] release chip
     * @mtd:	MTD device structure
    
    Wolfgang Denk's avatar
    Wolfgang Denk committed
     *
     * Deselect, release chip lock and wake up anyone waiting on the device
    
    static void nand_release_device(struct mtd_info *mtd)
    
    	struct nand_chip *chip = mtd->priv;
    
    
    	/* De-select the NAND device */
    
    	chip->select_chip(mtd, -1);
    
    	/* Release the controller and the chip */
    	spin_lock(&chip->controller->lock);
    	chip->controller->active = NULL;
    	chip->state = FL_READY;
    	wake_up(&chip->controller->wq);
    	spin_unlock(&chip->controller->lock);
    
    static void nand_release_device (struct mtd_info *mtd)
    {
    	struct nand_chip *this = mtd->priv;
    	this->select_chip(mtd, -1);	/* De-select the NAND device */
    }
    
    #endif
    
    /**
     * nand_read_byte - [DEFAULT] read one byte from the chip
     * @mtd:	MTD device structure
     *
     * Default read function for 8bit buswith
     */
    
    static uint8_t nand_read_byte(struct mtd_info *mtd)
    
    	struct nand_chip *chip = mtd->priv;
    	return readb(chip->IO_ADDR_R);
    
    }
    
    /**
     * nand_read_byte16 - [DEFAULT] read one byte endianess aware from the chip
     * @mtd:	MTD device structure
     *
    
    Wolfgang Denk's avatar
    Wolfgang Denk committed
     * Default read function for 16bit buswith with
    
    static uint8_t nand_read_byte16(struct mtd_info *mtd)
    
    	struct nand_chip *chip = mtd->priv;
    	return (uint8_t) cpu_to_le16(readw(chip->IO_ADDR_R));
    
    }
    
    /**
     * nand_read_word - [DEFAULT] read one word from the chip
     * @mtd:	MTD device structure
     *
    
    Wolfgang Denk's avatar
    Wolfgang Denk committed
     * Default read function for 16bit buswith without
    
     * endianess conversion
     */
    static u16 nand_read_word(struct mtd_info *mtd)
    {
    
    	struct nand_chip *chip = mtd->priv;
    	return readw(chip->IO_ADDR_R);
    
    }
    
    /**
     * nand_select_chip - [DEFAULT] control CE line
     * @mtd:	MTD device structure
    
     * @chipnr:	chipnumber to select, -1 for deselect
    
     *
     * Default select function for 1 chip devices.
     */
    
    static void nand_select_chip(struct mtd_info *mtd, int chipnr)
    
    	struct nand_chip *chip = mtd->priv;
    
    	switch (chipnr) {
    
    		chip->cmd_ctrl(mtd, NAND_CMD_NONE, 0 | NAND_CTRL_CHANGE);
    
    		break;
    	case 0:
    		break;
    
    	default:
    		BUG();
    	}
    }
    
    /**
     * nand_write_buf - [DEFAULT] write buffer to chip
     * @mtd:	MTD device structure
     * @buf:	data buffer
     * @len:	number of bytes to write
     *
     * Default write function for 8bit buswith
     */
    
    static void nand_write_buf(struct mtd_info *mtd, const uint8_t *buf, int len)
    
    	struct nand_chip *chip = mtd->priv;
    
    	for (i = 0; i < len; i++)
    		writeb(buf[i], chip->IO_ADDR_W);
    
    Wolfgang Denk's avatar
    Wolfgang Denk committed
     * nand_read_buf - [DEFAULT] read chip data into buffer
    
     * @mtd:	MTD device structure
     * @buf:	buffer to store date
     * @len:	number of bytes to read
     *
     * Default read function for 8bit buswith
     */
    
    static void nand_read_buf(struct mtd_info *mtd, uint8_t *buf, int len)
    
    	struct nand_chip *chip = mtd->priv;
    
    	for (i = 0; i < len; i++)
    		buf[i] = readb(chip->IO_ADDR_R);
    
    Wolfgang Denk's avatar
    Wolfgang Denk committed
     * nand_verify_buf - [DEFAULT] Verify chip data against buffer
    
     * @mtd:	MTD device structure
     * @buf:	buffer containing the data to compare
     * @len:	number of bytes to compare
     *
     * Default verify function for 8bit buswith
     */
    
    static int nand_verify_buf(struct mtd_info *mtd, const uint8_t *buf, int len)
    
    	struct nand_chip *chip = mtd->priv;
    
    	for (i = 0; i < len; i++)
    		if (buf[i] != readb(chip->IO_ADDR_R))
    
    			return -EFAULT;
    	return 0;
    }
    
    /**
     * nand_write_buf16 - [DEFAULT] write buffer to chip
     * @mtd:	MTD device structure
     * @buf:	data buffer
     * @len:	number of bytes to write
     *
     * Default write function for 16bit buswith
     */
    
    static void nand_write_buf16(struct mtd_info *mtd, const uint8_t *buf, int len)
    
    	struct nand_chip *chip = mtd->priv;
    
    	u16 *p = (u16 *) buf;
    	len >>= 1;
    
    	for (i = 0; i < len; i++)
    		writew(p[i], chip->IO_ADDR_W);
    
    Wolfgang Denk's avatar
    Wolfgang Denk committed
     * nand_read_buf16 - [DEFAULT] read chip data into buffer
    
     * @mtd:	MTD device structure
     * @buf:	buffer to store date
     * @len:	number of bytes to read
     *
     * Default read function for 16bit buswith
     */
    
    static void nand_read_buf16(struct mtd_info *mtd, uint8_t *buf, int len)
    
    	struct nand_chip *chip = mtd->priv;
    
    	for (i = 0; i < len; i++)
    		p[i] = readw(chip->IO_ADDR_R);
    
    Wolfgang Denk's avatar
    Wolfgang Denk committed
     * nand_verify_buf16 - [DEFAULT] Verify chip data against buffer
    
     * @mtd:	MTD device structure
     * @buf:	buffer containing the data to compare
     * @len:	number of bytes to compare
     *
     * Default verify function for 16bit buswith
     */
    
    static int nand_verify_buf16(struct mtd_info *mtd, const uint8_t *buf, int len)
    
    	struct nand_chip *chip = mtd->priv;
    
    	for (i = 0; i < len; i++)
    		if (p[i] != readw(chip->IO_ADDR_R))
    
    			return -EFAULT;
    
    	return 0;
    }
    
    /**
     * nand_block_bad - [DEFAULT] Read bad block marker from the chip
     * @mtd:	MTD device structure
     * @ofs:	offset from device start
     * @getchip:	0, if the chip is already selected
     *
    
    Wolfgang Denk's avatar
    Wolfgang Denk committed
     * Check, if the block is bad.
    
     */
    static int nand_block_bad(struct mtd_info *mtd, loff_t ofs, int getchip)
    {
    	int page, chipnr, res = 0;
    
    	struct nand_chip *chip = mtd->priv;
    
    	page = (int)(ofs >> chip->page_shift) & chip->pagemask;
    
    		chipnr = (int)(ofs >> chip->chip_shift);
    
    		nand_get_device(chip, mtd, FL_READING);
    
    		chip->select_chip(mtd, chipnr);
    
    	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;
    
    		if ((bad & 0xFF) != 0xff)
    			res = 1;
    	} else {
    
    		chip->cmdfunc(mtd, NAND_CMD_READOOB, chip->badblockpos, page);
    		if (chip->read_byte(mtd) != 0xff)
    
    	if (getchip)
    
    	return res;
    }
    
    /**
     * nand_default_block_markbad - [DEFAULT] mark a block bad
     * @mtd:	MTD device structure
     * @ofs:	offset from device start
     *
     * This is the default implementation, which can be overridden by
     * a hardware specific driver.
    */
    static int nand_default_block_markbad(struct mtd_info *mtd, loff_t ofs)
    {
    
    	struct nand_chip *chip = mtd->priv;
    	uint8_t buf[2] = { 0, 0 };
    	int block, ret;
    
    	block = (int)(ofs >> chip->bbt_erase_shift);
    	if (chip->bbt)
    		chip->bbt[block >> 2] |= 0x01 << ((block & 0x03) << 1);
    
    
    	/* Do we have a flash based bad block table ? */
    
    	if (chip->options & NAND_USE_FLASH_BBT)
    		ret = nand_update_bbt(mtd, ofs);
    	else {
    		/* We write two bytes, so we dont have to mess with 16 bit
    		 * access
    		 */
    		ofs += mtd->oobsize;
    		chip->ops.len = chip->ops.ooblen = 2;
    		chip->ops.datbuf = NULL;
    		chip->ops.oobbuf = buf;
    		chip->ops.ooboffs = chip->badblockpos & ~0x01;
    
    		ret = nand_do_write_oob(mtd, ofs, &chip->ops);
    	}
    	if (!ret)
    		mtd->ecc_stats.badblocks++;
    	return ret;
    
    Wolfgang Denk's avatar
    Wolfgang Denk committed
    /**
    
     * nand_check_wp - [GENERIC] check if the chip is write protected
     * @mtd:	MTD device structure
    
    Wolfgang Denk's avatar
    Wolfgang Denk committed
     * Check, if the device is write protected
    
    Wolfgang Denk's avatar
    Wolfgang Denk committed
     * The function expects, that the device is already selected
    
    static int nand_check_wp(struct mtd_info *mtd)
    
    	struct nand_chip *chip = mtd->priv;
    
    	chip->cmdfunc(mtd, NAND_CMD_STATUS, -1, -1);
    	return (chip->read_byte(mtd) & NAND_STATUS_WP) ? 0 : 1;
    
    /**
     * nand_block_checkbad - [GENERIC] Check if a block is marked bad
     * @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
     *
     * Check, if the block is bad. Either by reading the bad block table or
     * calling of the scan function.
     */
    
    static int nand_block_checkbad(struct mtd_info *mtd, loff_t ofs, int getchip,
    			       int allowbbt)
    
    	struct nand_chip *chip = mtd->priv;
    
    	if (!(chip->options & NAND_BBT_SCANNED)) {
    		chip->scan_bbt(mtd);
    		chip->options |= NAND_BBT_SCANNED;
    	}
    
    
    	if (!chip->bbt)
    		return chip->block_bad(mtd, ofs, getchip);
    
    	/* Return info from the table */
    
    	return nand_isbad_bbt(mtd, ofs, allowbbt);
    }
    
    /*
     * Wait for the ready pin, after a command
     * The timeout is catched later.
     */
    /* XXX U-BOOT XXX */
    #if 0
    void nand_wait_ready(struct mtd_info *mtd)
    {
    	struct nand_chip *chip = mtd->priv;
    	unsigned long timeo = jiffies + 2;
    
    	led_trigger_event(nand_led_trigger, LED_FULL);
    	/* wait until command is processed or timeout occures */
    	do {
    		if (chip->dev_ready(mtd))
    			break;
    		touch_softlockup_watchdog();
    	} while (time_before(jiffies, timeo));
    	led_trigger_event(nand_led_trigger, LED_OFF);
    
    EXPORT_SYMBOL_GPL(nand_wait_ready);
    #else
    void nand_wait_ready(struct mtd_info *mtd)
    {
    	struct nand_chip *chip = mtd->priv;
    
    	u32 timeo = (CFG_HZ * 20) / 1000;
    
    	reset_timer();
    
    	/* wait until command is processed or timeout occures */
    	while (get_timer(0) < timeo) {
    		if (chip->dev_ready)
    			if (chip->dev_ready(mtd))
    				break;
    	}
    
    
    /**
     * nand_command - [DEFAULT] Send command to NAND device
     * @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
     *
     * Send command to NAND device. This function is used for small page
     * devices (256/512 Bytes per page)
     */
    
    static void nand_command(struct mtd_info *mtd, unsigned int command,
    			 int column, int page_addr)
    
    	register struct nand_chip *chip = mtd->priv;
    	int ctrl = NAND_CTRL_CLE | NAND_CTRL_CHANGE;
    
    
    	/*
    	 * Write out the command to the device.
    	 */
    	if (command == NAND_CMD_SEQIN) {
    		int readcmd;
    
    
    		if (column >= mtd->writesize) {
    
    			column -= mtd->writesize;
    
    			readcmd = NAND_CMD_READOOB;
    		} else if (column < 256) {
    			/* First 256 bytes --> READ0 */
    			readcmd = NAND_CMD_READ0;
    		} else {
    			column -= 256;
    			readcmd = NAND_CMD_READ1;
    		}
    
    		chip->cmd_ctrl(mtd, readcmd, ctrl);
    		ctrl &= ~NAND_CTRL_CHANGE;
    
    	chip->cmd_ctrl(mtd, command, ctrl);
    
    	/*
    	 * Address cycle, when necessary
    	 */
    	ctrl = NAND_CTRL_ALE | NAND_CTRL_CHANGE;
    	/* Serially input address */
    	if (column != -1) {
    		/* Adjust columns for 16 bit buswidth */
    		if (chip->options & NAND_BUSWIDTH_16)
    			column >>= 1;
    		chip->cmd_ctrl(mtd, column, ctrl);
    		ctrl &= ~NAND_CTRL_CHANGE;
    
    	if (page_addr != -1) {
    		chip->cmd_ctrl(mtd, page_addr, ctrl);
    		ctrl &= ~NAND_CTRL_CHANGE;
    		chip->cmd_ctrl(mtd, page_addr >> 8, ctrl);
    		/* One more address cycle for devices > 32MiB */
    		if (chip->chipsize > (32 << 20))
    			chip->cmd_ctrl(mtd, page_addr >> 16, ctrl);
    	}
    	chip->cmd_ctrl(mtd, NAND_CMD_NONE, NAND_NCE | NAND_CTRL_CHANGE);
    
    Wolfgang Denk's avatar
    Wolfgang Denk committed
    
    	/*
    	 * program and erase have their own busy handlers
    
    	 * status and sequential in needs no delay
    
    	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:
    
    		if (chip->dev_ready)
    
    		udelay(chip->chip_delay);
    		chip->cmd_ctrl(mtd, NAND_CMD_STATUS,
    			       NAND_CTRL_CLE | NAND_CTRL_CHANGE);
    		chip->cmd_ctrl(mtd,
    			       NAND_CMD_NONE, NAND_NCE | NAND_CTRL_CHANGE);
    		while (!(chip->read_byte(mtd) & NAND_STATUS_READY)) ;
    
    		/* This applies to read commands */
    
    Wolfgang Denk's avatar
    Wolfgang Denk committed
    		/*
    
    		 * If we don't have access to the busy pin, we apply the given
    		 * command delay
    
    		 */
    		if (!chip->dev_ready) {
    			udelay(chip->chip_delay);
    
    	}
    	/* Apply this short delay always to ensure that we do wait tWB in
    	 * any case on any machine. */
    
    	ndelay(100);
    
    	nand_wait_ready(mtd);
    
    }
    
    /**
     * nand_command_lp - [DEFAULT] Send command to NAND large page device
     * @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
     *
    
     * Send command to NAND device. This is the version for the new large page
     * devices We dont have the separate regions as we have in the small page
     * devices.  We must emulate NAND_CMD_READOOB to keep the code compatible.
    
    static void nand_command_lp(struct mtd_info *mtd, unsigned int command,
    			    int column, int page_addr)
    
    	register struct nand_chip *chip = mtd->priv;
    
    
    	/* Emulate NAND_CMD_READOOB */
    	if (command == NAND_CMD_READOOB) {
    
    		column += mtd->writesize;
    
    	/* Command latch cycle */
    	chip->cmd_ctrl(mtd, command & 0xff,
    		       NAND_NCE | NAND_CLE | NAND_CTRL_CHANGE);
    
    
    	if (column != -1 || page_addr != -1) {
    
    		int ctrl = NAND_CTRL_CHANGE | NAND_NCE | NAND_ALE;
    
    
    		/* Serially input address */
    		if (column != -1) {
    			/* Adjust columns for 16 bit buswidth */
    
    			if (chip->options & NAND_BUSWIDTH_16)
    
    			chip->cmd_ctrl(mtd, column, ctrl);
    			ctrl &= ~NAND_CTRL_CHANGE;
    			chip->cmd_ctrl(mtd, column >> 8, ctrl);
    
    			chip->cmd_ctrl(mtd, page_addr, ctrl);
    			chip->cmd_ctrl(mtd, page_addr >> 8,
    				       NAND_NCE | NAND_ALE);
    
    			/* One more address cycle for devices > 128MiB */
    
    			if (chip->chipsize > (128 << 20))
    				chip->cmd_ctrl(mtd, page_addr >> 16,
    					       NAND_NCE | NAND_ALE);
    
    	chip->cmd_ctrl(mtd, NAND_CMD_NONE, NAND_NCE | NAND_CTRL_CHANGE);
    
    Wolfgang Denk's avatar
    Wolfgang Denk committed
    
    	/*
    	 * program and erase have their own busy handlers
    
    	 * status, sequential in, and deplete1 need no delay
    	 */
    
    	case NAND_CMD_CACHEDPROG:
    	case NAND_CMD_PAGEPROG:
    	case NAND_CMD_ERASE1:
    	case NAND_CMD_ERASE2:
    	case NAND_CMD_SEQIN:
    
    	case NAND_CMD_RNDIN:
    
    	case NAND_CMD_DEPLETE1:
    
    		/*
    		 * read error status commands require only a short delay
    		 */
    	case NAND_CMD_STATUS_ERROR:
    	case NAND_CMD_STATUS_ERROR0:
    	case NAND_CMD_STATUS_ERROR1:
    	case NAND_CMD_STATUS_ERROR2:
    	case NAND_CMD_STATUS_ERROR3:
    		udelay(chip->chip_delay);
    		return;
    
    		if (chip->dev_ready)
    
    		udelay(chip->chip_delay);
    		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);
    		while (!(chip->read_byte(mtd) & NAND_STATUS_READY)) ;
    		return;
    
    	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);
    
    		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);
    
    		/* This applies to read commands */
    
    Wolfgang Denk's avatar
    Wolfgang Denk committed
    		/*
    
    		 * If we don't have access to the busy pin, we apply the given
    		 * command delay
    
    		 */
    		if (!chip->dev_ready) {
    			udelay(chip->chip_delay);
    
    	/* Apply this short delay always to ensure that we do wait tWB in
    	 * any case on any machine. */
    
    	ndelay(100);
    
    	nand_wait_ready(mtd);
    
    }
    
    /**
     * nand_get_device - [GENERIC] Get chip for selected access
    
     * @chip:	the nand chip descriptor
    
     * @mtd:	MTD device structure
    
    Wolfgang Denk's avatar
    Wolfgang Denk committed
     * @new_state:	the state which is requested
    
     *
     * Get the device and lock it for exclusive access
     */
    /* XXX U-BOOT XXX */
    #if 0
    
    static int
    nand_get_device(struct nand_chip *chip, struct mtd_info *mtd, int new_state)
    
    	spinlock_t *lock = &chip->controller->lock;
    	wait_queue_head_t *wq = &chip->controller->wq;
    	DECLARE_WAITQUEUE(wait, current);
     retry:
    	spin_lock(lock);
    
    
    	/* Hardware controller shared among independend devices */
    
    	/* Hardware controller shared among independend devices */
    	if (!chip->controller->active)
    		chip->controller->active = chip;
    
    	if (chip->controller->active == chip && chip->state == FL_READY) {
    		chip->state = new_state;
    		spin_unlock(lock);
    		return 0;
    
    	if (new_state == FL_PM_SUSPENDED) {
    		spin_unlock(lock);
    		return (chip->state == FL_PM_SUSPENDED) ? 0 : -EAGAIN;
    	}
    	set_current_state(TASK_UNINTERRUPTIBLE);
    	add_wait_queue(wq, &wait);
    	spin_unlock(lock);
    	schedule();
    	remove_wait_queue(wq, &wait);
    
    static int nand_get_device (struct nand_chip *this, struct mtd_info *mtd, int new_state)
    {
    	return 0;
    }
    
    #endif
    
    /**
     * nand_wait - [DEFAULT]  wait until the command is done
     * @mtd:	MTD device structure
    
     * @chip:	NAND chip structure
    
     *
     * Wait for command done. This applies to erase and program only
    
    Wolfgang Denk's avatar
    Wolfgang Denk committed
     * Erase can take up to 400ms and program up to 20ms according to
    
     * general NAND and SmartMedia specs
    
    static int nand_wait(struct mtd_info *mtd, struct nand_chip *chip)
    
    
    	unsigned long timeo = jiffies;
    	int status, state = chip->state;
    
    		timeo += (HZ * 400) / 1000;
    
    		timeo += (HZ * 20) / 1000;
    
    	led_trigger_event(nand_led_trigger, LED_FULL);
    
    
    	/* Apply this short delay always to ensure that we do wait tWB in
    	 * any case on any machine. */
    
    	ndelay(100);
    
    	if ((state == FL_ERASING) && (chip->options & NAND_IS_AND))
    		chip->cmdfunc(mtd, NAND_CMD_STATUS_MULTI, -1, -1);
    
    Wolfgang Denk's avatar
    Wolfgang Denk committed
    	else
    
    		chip->cmdfunc(mtd, NAND_CMD_STATUS, -1, -1);
    
    Wolfgang Denk's avatar
    Wolfgang Denk committed
    	while (time_before(jiffies, timeo)) {
    
    		if (chip->dev_ready) {
    			if (chip->dev_ready(mtd))
    
    Wolfgang Denk's avatar
    Wolfgang Denk committed
    				break;
    
    			if (chip->read_byte(mtd) & NAND_STATUS_READY)
    
    		cond_resched();
    
    	led_trigger_event(nand_led_trigger, LED_OFF);
    
    	status = (int)chip->read_byte(mtd);
    	return status;
    
    static int nand_wait(struct mtd_info *mtd, struct nand_chip *this)
    
    	int state = this->state;
    
    Wolfgang Denk's avatar
    Wolfgang Denk committed
    		timeo = (CFG_HZ * 400) / 1000;
    
    		timeo = (CFG_HZ * 20) / 1000;
    
    
    	if ((state == FL_ERASING) && (this->options & NAND_IS_AND))
    		this->cmdfunc(mtd, NAND_CMD_STATUS_MULTI, -1, -1);
    	else
    		this->cmdfunc(mtd, NAND_CMD_STATUS, -1, -1);
    
    
    		if (get_timer(0) > timeo) {
    			printf("Timeout!");
    
    			return 0x01;
    		}
    
    
    		if (this->dev_ready) {
    			if (this->dev_ready(mtd))
    				break;
    		} else {
    			if (this->read_byte(mtd) & NAND_STATUS_READY)
    				break;
    		}
    	}
    
    	reset_timer();
    	while (get_timer(0) < 10);
    
    #endif /*  PPCHAMELON_NAND_TIMER_HACK */
    
    	return this->read_byte(mtd);
    
     * nand_read_page_raw - [Intern] read raw page data without ecc
     * @mtd:	mtd info structure
     * @chip:	nand chip info structure
     * @buf:	buffer to store read data
    
    static int nand_read_page_raw(struct mtd_info *mtd, struct nand_chip *chip,
    			      uint8_t *buf)
    
    	chip->read_buf(mtd, buf, mtd->writesize);
    	chip->read_buf(mtd, chip->oob_poi, mtd->oobsize);
    	return 0;
    }
    
    /**
     * nand_read_page_swecc - [REPLACABLE] software ecc based page read function
     * @mtd:	mtd info structure
     * @chip:	nand chip info structure
     * @buf:	buffer to store read data
     */
    static int nand_read_page_swecc(struct mtd_info *mtd, struct nand_chip *chip,
    				uint8_t *buf)
    {
    	int i, eccsize = chip->ecc.size;
    	int eccbytes = chip->ecc.bytes;
    	int eccsteps = chip->ecc.steps;
    	uint8_t *p = buf;
    	uint8_t *ecc_calc = chip->buffers->ecccalc;
    	uint8_t *ecc_code = chip->buffers->ecccode;
    	uint32_t *eccpos = chip->ecc.layout->eccpos;
    
    	chip->ecc.read_page_raw(mtd, chip, buf);
    
    	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++)
    		ecc_code[i] = chip->oob_poi[eccpos[i]];
    
    	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]);
    		if (stat == -1)
    			mtd->ecc_stats.failed++;
    		else
    			mtd->ecc_stats.corrected += stat;
    
    Wolfgang Denk's avatar
    Wolfgang Denk committed
    	return 0;
    
     * nand_read_page_hwecc - [REPLACABLE] hardware ecc based page read function
     * @mtd:	mtd info structure
     * @chip:	nand chip info structure
     * @buf:	buffer to store read data
    
     * Not for syndrome calculating ecc controllers which need a special oob layout
    
    static int nand_read_page_hwecc(struct mtd_info *mtd, struct nand_chip *chip,
    				uint8_t *buf)
    {
    	int i, eccsize = chip->ecc.size;
    	int eccbytes = chip->ecc.bytes;
    	int eccsteps = chip->ecc.steps;
    	uint8_t *p = buf;
    	uint8_t *ecc_calc = chip->buffers->ecccalc;
    	uint8_t *ecc_code = chip->buffers->ecccode;
    	uint32_t *eccpos = chip->ecc.layout->eccpos;
    
    	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]);
    	}
    	chip->read_buf(mtd, chip->oob_poi, mtd->oobsize);
    
    	for (i = 0; i < chip->ecc.total; i++)
    		ecc_code[i] = chip->oob_poi[eccpos[i]];
    
    	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]);
    		if (stat == -1)
    			mtd->ecc_stats.failed++;
    		else
    			mtd->ecc_stats.corrected += stat;
    
     * nand_read_page_syndrome - [REPLACABLE] hardware ecc syndrom based page read
     * @mtd:	mtd info structure
     * @chip:	nand chip info structure
     * @buf:	buffer to store read data
    
     * The hw generator calculates the error syndrome automatically. Therefor
     * we need a special oob layout and handling.
     */
    static int nand_read_page_syndrome(struct mtd_info *mtd, struct nand_chip *chip,
    				   uint8_t *buf)
    
    	int i, eccsize = chip->ecc.size;
    	int eccbytes = chip->ecc.bytes;
    	int eccsteps = chip->ecc.steps;
    	uint8_t *p = buf;
    	uint8_t *oob = chip->oob_poi;
    
    	for (i = 0; eccsteps; eccsteps--, i += eccbytes, p += eccsize) {
    		int stat;
    
    		chip->ecc.hwctl(mtd, NAND_ECC_READ);
    		chip->read_buf(mtd, p, eccsize);
    
    		if (chip->ecc.prepad) {
    			chip->read_buf(mtd, oob, chip->ecc.prepad);
    			oob += chip->ecc.prepad;
    		}
    
    		chip->ecc.hwctl(mtd, NAND_ECC_READSYN);
    		chip->read_buf(mtd, oob, eccbytes);
    		stat = chip->ecc.correct(mtd, p, oob, NULL);
    
    		if (stat == -1)