atmel_nand.c 14.7 KB
Newer Older
A
Andrew Victor 已提交
1 2 3 4 5 6 7 8 9
/*
 *  Copyright (C) 2003 Rick Bronson
 *
 *  Derived from drivers/mtd/nand/autcpu12.c
 *	 Copyright (c) 2001 Thomas Gleixner (gleixner@autronix.de)
 *
 *  Derived from drivers/mtd/spia.c
 *	 Copyright (C) 2000 Steven J. Hill (sjhill@cotw.com)
 *
10 11 12 13 14 15 16 17 18
 *
 *  Add Hardware ECC support for AT91SAM9260 / AT91SAM9263
 *     Richard Genoud (richard.genoud@gmail.com), Adeneo Copyright (C) 2007
 *
 *     Derived from Das U-Boot source code
 *     		(u-boot-1.1.5/board/atmel/at91sam9263ek/nand.c)
 *     (C) Copyright 2006 ATMEL Rousset, Lacressonniere Nicolas
 *
 *
A
Andrew Victor 已提交
19 20 21 22 23 24 25 26
 * 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.
 *
 */

#include <linux/slab.h>
#include <linux/module.h>
27
#include <linux/moduleparam.h>
A
Andrew Victor 已提交
28 29 30 31 32
#include <linux/platform_device.h>
#include <linux/mtd/mtd.h>
#include <linux/mtd/nand.h>
#include <linux/mtd/partitions.h>

33 34
#include <linux/gpio.h>
#include <linux/io.h>
A
Andrew Victor 已提交
35

36 37
#include <mach/board.h>
#include <mach/cpu.h>
A
Andrew Victor 已提交
38

39
#ifdef CONFIG_MTD_NAND_ATMEL_ECC_HW
40 41 42 43 44
#define hard_ecc	1
#else
#define hard_ecc	0
#endif

45
#ifdef CONFIG_MTD_NAND_ATMEL_ECC_NONE
46 47 48 49 50
#define no_ecc		1
#else
#define no_ecc		0
#endif

51 52 53
static int on_flash_bbt = 0;
module_param(on_flash_bbt, int, 0);

54 55
/* Register access macros */
#define ecc_readl(add, reg)				\
56
	__raw_readl(add + ATMEL_ECC_##reg)
57
#define ecc_writel(add, reg, value)			\
58
	__raw_writel((value), add + ATMEL_ECC_##reg)
59

60
#include "atmel_nand_ecc.h"	/* Hardware ECC registers */
61 62 63 64 65 66

/* oob layout for large page size
 * bad block info is on bytes 0 and 1
 * the bytes have to be consecutives to avoid
 * several NAND_CMD_RNDOUT during read
 */
67
static struct nand_ecclayout atmel_oobinfo_large = {
68 69 70 71 72 73 74 75 76 77 78 79
	.eccbytes = 4,
	.eccpos = {60, 61, 62, 63},
	.oobfree = {
		{2, 58}
	},
};

/* oob layout for small page size
 * bad block info is on bytes 4 and 5
 * the bytes have to be consecutives to avoid
 * several NAND_CMD_RNDOUT during read
 */
80
static struct nand_ecclayout atmel_oobinfo_small = {
81 82 83 84 85 86 87
	.eccbytes = 4,
	.eccpos = {0, 1, 2, 3},
	.oobfree = {
		{6, 10}
	},
};

88
struct atmel_nand_host {
A
Andrew Victor 已提交
89 90 91
	struct nand_chip	nand_chip;
	struct mtd_info		mtd;
	void __iomem		*io_base;
92
	struct atmel_nand_data	*board;
93 94
	struct device		*dev;
	void __iomem		*ecc;
A
Andrew Victor 已提交
95 96
};

97 98 99
/*
 * Enable NAND.
 */
100
static void atmel_nand_enable(struct atmel_nand_host *host)
101 102
{
	if (host->board->enable_pin)
103
		gpio_set_value(host->board->enable_pin, 0);
104 105 106 107 108
}

/*
 * Disable NAND.
 */
109
static void atmel_nand_disable(struct atmel_nand_host *host)
110 111
{
	if (host->board->enable_pin)
112
		gpio_set_value(host->board->enable_pin, 1);
113 114
}

A
Andrew Victor 已提交
115 116 117
/*
 * Hardware specific access to control-lines
 */
118
static void atmel_nand_cmd_ctrl(struct mtd_info *mtd, int cmd, unsigned int ctrl)
A
Andrew Victor 已提交
119 120
{
	struct nand_chip *nand_chip = mtd->priv;
121
	struct atmel_nand_host *host = nand_chip->priv;
A
Andrew Victor 已提交
122

123
	if (ctrl & NAND_CTRL_CHANGE) {
124
		if (ctrl & NAND_NCE)
125
			atmel_nand_enable(host);
126
		else
127
			atmel_nand_disable(host);
128
	}
A
Andrew Victor 已提交
129 130 131 132 133 134 135 136 137 138 139 140
	if (cmd == NAND_CMD_NONE)
		return;

	if (ctrl & NAND_CLE)
		writeb(cmd, host->io_base + (1 << host->board->cle));
	else
		writeb(cmd, host->io_base + (1 << host->board->ale));
}

/*
 * Read the Device Ready pin.
 */
141
static int atmel_nand_device_ready(struct mtd_info *mtd)
A
Andrew Victor 已提交
142 143
{
	struct nand_chip *nand_chip = mtd->priv;
144
	struct atmel_nand_host *host = nand_chip->priv;
A
Andrew Victor 已提交
145

146 147
	return gpio_get_value(host->board->rdy_pin) ^
                !!host->board->rdy_pin_active_low;
A
Andrew Victor 已提交
148 149
}

150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180
/*
 * Minimal-overhead PIO for data access.
 */
static void atmel_read_buf(struct mtd_info *mtd, u8 *buf, int len)
{
	struct nand_chip	*nand_chip = mtd->priv;

	__raw_readsb(nand_chip->IO_ADDR_R, buf, len);
}

static void atmel_read_buf16(struct mtd_info *mtd, u8 *buf, int len)
{
	struct nand_chip	*nand_chip = mtd->priv;

	__raw_readsw(nand_chip->IO_ADDR_R, buf, len / 2);
}

static void atmel_write_buf(struct mtd_info *mtd, const u8 *buf, int len)
{
	struct nand_chip	*nand_chip = mtd->priv;

	__raw_writesb(nand_chip->IO_ADDR_W, buf, len);
}

static void atmel_write_buf16(struct mtd_info *mtd, const u8 *buf, int len)
{
	struct nand_chip	*nand_chip = mtd->priv;

	__raw_writesw(nand_chip->IO_ADDR_W, buf, len / 2);
}

181 182 183 184 185 186 187 188 189
/*
 * Calculate HW ECC
 *
 * function called after a write
 *
 * mtd:        MTD block structure
 * dat:        raw data (unused)
 * ecc_code:   buffer for ECC
 */
190
static int atmel_nand_calculate(struct mtd_info *mtd,
191 192 193
		const u_char *dat, unsigned char *ecc_code)
{
	struct nand_chip *nand_chip = mtd->priv;
194
	struct atmel_nand_host *host = nand_chip->priv;
195 196 197
	unsigned int ecc_value;

	/* get the first 2 ECC bytes */
198
	ecc_value = ecc_readl(host->ecc, PR);
199

200 201
	ecc_code[0] = ecc_value & 0xFF;
	ecc_code[1] = (ecc_value >> 8) & 0xFF;
202 203

	/* get the last 2 ECC bytes */
204
	ecc_value = ecc_readl(host->ecc, NPR) & ATMEL_ECC_NPARITY;
205

206 207
	ecc_code[2] = ecc_value & 0xFF;
	ecc_code[3] = (ecc_value >> 8) & 0xFF;
208 209 210 211 212 213 214 215 216 217 218

	return 0;
}

/*
 * HW ECC read page function
 *
 * mtd:        mtd info structure
 * chip:       nand chip info structure
 * buf:        buffer to store read data
 */
219
static int atmel_nand_read_page(struct mtd_info *mtd,
220
		struct nand_chip *chip, uint8_t *buf, int page)
221 222 223 224 225 226 227 228 229
{
	int eccsize = chip->ecc.size;
	int eccbytes = chip->ecc.bytes;
	uint32_t *eccpos = chip->ecc.layout->eccpos;
	uint8_t *p = buf;
	uint8_t *oob = chip->oob_poi;
	uint8_t *ecc_pos;
	int stat;

230 231 232 233 234 235 236 237 238 239 240 241 242
	/*
	 * Errata: ALE is incorrectly wired up to the ECC controller
	 * on the AP7000, so it will include the address cycles in the
	 * ECC calculation.
	 *
	 * Workaround: Reset the parity registers before reading the
	 * actual data.
	 */
	if (cpu_is_at32ap7000()) {
		struct atmel_nand_host *host = chip->priv;
		ecc_writel(host->ecc, CR, ATMEL_ECC_RST);
	}

243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 272 273 274 275 276 277 278 279 280 281 282 283 284 285 286 287 288 289 290
	/* read the page */
	chip->read_buf(mtd, p, eccsize);

	/* move to ECC position if needed */
	if (eccpos[0] != 0) {
		/* This only works on large pages
		 * because the ECC controller waits for
		 * NAND_CMD_RNDOUTSTART after the
		 * NAND_CMD_RNDOUT.
		 * anyway, for small pages, the eccpos[0] == 0
		 */
		chip->cmdfunc(mtd, NAND_CMD_RNDOUT,
				mtd->writesize + eccpos[0], -1);
	}

	/* the ECC controller needs to read the ECC just after the data */
	ecc_pos = oob + eccpos[0];
	chip->read_buf(mtd, ecc_pos, eccbytes);

	/* check if there's an error */
	stat = chip->ecc.correct(mtd, p, oob, NULL);

	if (stat < 0)
		mtd->ecc_stats.failed++;
	else
		mtd->ecc_stats.corrected += stat;

	/* get back to oob start (end of page) */
	chip->cmdfunc(mtd, NAND_CMD_RNDOUT, mtd->writesize, -1);

	/* read the oob */
	chip->read_buf(mtd, oob, mtd->oobsize);

	return 0;
}

/*
 * HW ECC Correction
 *
 * function called after a read
 *
 * mtd:        MTD block structure
 * dat:        raw data read from the chip
 * read_ecc:   ECC from the chip (unused)
 * isnull:     unused
 *
 * Detect and correct a 1 bit error for a page
 */
291
static int atmel_nand_correct(struct mtd_info *mtd, u_char *dat,
292 293 294
		u_char *read_ecc, u_char *isnull)
{
	struct nand_chip *nand_chip = mtd->priv;
295
	struct atmel_nand_host *host = nand_chip->priv;
296 297 298 299 300 301 302
	unsigned int ecc_status;
	unsigned int ecc_word, ecc_bit;

	/* get the status from the Status Register */
	ecc_status = ecc_readl(host->ecc, SR);

	/* if there's no error */
303
	if (likely(!(ecc_status & ATMEL_ECC_RECERR)))
304 305 306
		return 0;

	/* get error bit offset (4 bits) */
307
	ecc_bit = ecc_readl(host->ecc, PR) & ATMEL_ECC_BITADDR;
308
	/* get word address (12 bits) */
309
	ecc_word = ecc_readl(host->ecc, PR) & ATMEL_ECC_WORDADDR;
310 311 312
	ecc_word >>= 4;

	/* if there are multiple errors */
313
	if (ecc_status & ATMEL_ECC_MULERR) {
314 315
		/* check if it is a freshly erased block
		 * (filled with 0xff) */
316 317
		if ((ecc_bit == ATMEL_ECC_BITADDR)
				&& (ecc_word == (ATMEL_ECC_WORDADDR >> 4))) {
318 319 320 321 322 323
			/* the block has just been erased, return OK */
			return 0;
		}
		/* it doesn't seems to be a freshly
		 * erased block.
		 * We can't correct so many errors */
324
		dev_dbg(host->dev, "atmel_nand : multiple errors detected."
325 326 327 328 329
				" Unable to correct.\n");
		return -EIO;
	}

	/* if there's a single bit error : we can correct it */
330
	if (ecc_status & ATMEL_ECC_ECCERR) {
331 332 333
		/* there's nothing much to do here.
		 * the bit error is on the ECC itself.
		 */
334
		dev_dbg(host->dev, "atmel_nand : one bit error on ECC code."
335 336 337 338
				" Nothing to correct\n");
		return 0;
	}

339
	dev_dbg(host->dev, "atmel_nand : one bit error on data."
340 341 342 343 344 345 346 347 348 349 350
			" (word offset in the page :"
			" 0x%x bit offset : 0x%x)\n",
			ecc_word, ecc_bit);
	/* correct the error */
	if (nand_chip->options & NAND_BUSWIDTH_16) {
		/* 16 bits words */
		((unsigned short *) dat)[ecc_word] ^= (1 << ecc_bit);
	} else {
		/* 8 bits words */
		dat[ecc_word] ^= (1 << ecc_bit);
	}
351
	dev_dbg(host->dev, "atmel_nand : error corrected\n");
352 353 354 355
	return 1;
}

/*
356
 * Enable HW ECC : unused on most chips
357
 */
358 359 360 361 362 363 364 365
static void atmel_nand_hwctl(struct mtd_info *mtd, int mode)
{
	if (cpu_is_at32ap7000()) {
		struct nand_chip *nand_chip = mtd->priv;
		struct atmel_nand_host *host = nand_chip->priv;
		ecc_writel(host->ecc, CR, ATMEL_ECC_RST);
	}
}
366

367
#ifdef CONFIG_MTD_PARTITIONS
368
static const char *part_probes[] = { "cmdlinepart", NULL };
369 370
#endif

A
Andrew Victor 已提交
371 372 373
/*
 * Probe for the NAND device.
 */
374
static int __init atmel_nand_probe(struct platform_device *pdev)
A
Andrew Victor 已提交
375
{
376
	struct atmel_nand_host *host;
A
Andrew Victor 已提交
377 378
	struct mtd_info *mtd;
	struct nand_chip *nand_chip;
379 380
	struct resource *regs;
	struct resource *mem;
A
Andrew Victor 已提交
381 382 383 384 385 386 387
	int res;

#ifdef CONFIG_MTD_PARTITIONS
	struct mtd_partition *partitions = NULL;
	int num_partitions = 0;
#endif

388 389 390 391 392 393
	mem = platform_get_resource(pdev, IORESOURCE_MEM, 0);
	if (!mem) {
		printk(KERN_ERR "atmel_nand: can't get I/O resource mem\n");
		return -ENXIO;
	}

A
Andrew Victor 已提交
394
	/* Allocate memory for the device structure (and zero it) */
395
	host = kzalloc(sizeof(struct atmel_nand_host), GFP_KERNEL);
A
Andrew Victor 已提交
396
	if (!host) {
397
		printk(KERN_ERR "atmel_nand: failed to allocate device structure.\n");
A
Andrew Victor 已提交
398 399 400
		return -ENOMEM;
	}

401
	host->io_base = ioremap(mem->start, mem->end - mem->start + 1);
A
Andrew Victor 已提交
402
	if (host->io_base == NULL) {
403
		printk(KERN_ERR "atmel_nand: ioremap failed\n");
404 405
		res = -EIO;
		goto err_nand_ioremap;
A
Andrew Victor 已提交
406 407 408 409 410
	}

	mtd = &host->mtd;
	nand_chip = &host->nand_chip;
	host->board = pdev->dev.platform_data;
411
	host->dev = &pdev->dev;
A
Andrew Victor 已提交
412 413 414 415 416 417 418 419

	nand_chip->priv = host;		/* link the private data structures */
	mtd->priv = nand_chip;
	mtd->owner = THIS_MODULE;

	/* Set address of NAND IO lines */
	nand_chip->IO_ADDR_R = host->io_base;
	nand_chip->IO_ADDR_W = host->io_base;
420
	nand_chip->cmd_ctrl = atmel_nand_cmd_ctrl;
I
Ivan Kuten 已提交
421 422

	if (host->board->rdy_pin)
423
		nand_chip->dev_ready = atmel_nand_device_ready;
I
Ivan Kuten 已提交
424

425 426
	regs = platform_get_resource(pdev, IORESOURCE_MEM, 1);
	if (!regs && hard_ecc) {
427
		printk(KERN_ERR "atmel_nand: can't get I/O resource "
428 429 430
				"regs\nFalling back on software ECC\n");
	}

A
Andrew Victor 已提交
431
	nand_chip->ecc.mode = NAND_ECC_SOFT;	/* enable ECC */
432 433 434 435 436
	if (no_ecc)
		nand_chip->ecc.mode = NAND_ECC_NONE;
	if (hard_ecc && regs) {
		host->ecc = ioremap(regs->start, regs->end - regs->start + 1);
		if (host->ecc == NULL) {
437
			printk(KERN_ERR "atmel_nand: ioremap failed\n");
438 439 440
			res = -EIO;
			goto err_ecc_ioremap;
		}
441
		nand_chip->ecc.mode = NAND_ECC_HW;
442 443 444 445
		nand_chip->ecc.calculate = atmel_nand_calculate;
		nand_chip->ecc.correct = atmel_nand_correct;
		nand_chip->ecc.hwctl = atmel_nand_hwctl;
		nand_chip->ecc.read_page = atmel_nand_read_page;
446 447 448
		nand_chip->ecc.bytes = 4;
	}

A
Andrew Victor 已提交
449 450
	nand_chip->chip_delay = 20;		/* 20us command delay time */

451
	if (host->board->bus_width_16) {	/* 16-bit bus width */
452
		nand_chip->options |= NAND_BUSWIDTH_16;
453 454 455 456 457 458
		nand_chip->read_buf = atmel_read_buf16;
		nand_chip->write_buf = atmel_write_buf16;
	} else {
		nand_chip->read_buf = atmel_read_buf;
		nand_chip->write_buf = atmel_write_buf;
	}
459

A
Andrew Victor 已提交
460
	platform_set_drvdata(pdev, host);
461
	atmel_nand_enable(host);
A
Andrew Victor 已提交
462 463

	if (host->board->det_pin) {
464
		if (gpio_get_value(host->board->det_pin)) {
465
			printk(KERN_INFO "No SmartMedia card inserted.\n");
A
Andrew Victor 已提交
466
			res = ENXIO;
467
			goto err_no_card;
A
Andrew Victor 已提交
468 469 470
		}
	}

471 472 473 474 475
	if (on_flash_bbt) {
		printk(KERN_INFO "atmel_nand: Use On Flash BBT\n");
		nand_chip->options |= NAND_USE_FLASH_BBT;
	}

476 477 478
	/* first scan to find the device and get the page size */
	if (nand_scan_ident(mtd, 1)) {
		res = -ENXIO;
479
		goto err_scan_ident;
480 481
	}

482
	if (nand_chip->ecc.mode == NAND_ECC_HW) {
483 484 485 486 487 488
		/* ECC is calculated for the whole page (1 step) */
		nand_chip->ecc.size = mtd->writesize;

		/* set ECC page size and oob layout */
		switch (mtd->writesize) {
		case 512:
489 490
			nand_chip->ecc.layout = &atmel_oobinfo_small;
			ecc_writel(host->ecc, MR, ATMEL_ECC_PAGESIZE_528);
491 492
			break;
		case 1024:
493 494
			nand_chip->ecc.layout = &atmel_oobinfo_large;
			ecc_writel(host->ecc, MR, ATMEL_ECC_PAGESIZE_1056);
495 496
			break;
		case 2048:
497 498
			nand_chip->ecc.layout = &atmel_oobinfo_large;
			ecc_writel(host->ecc, MR, ATMEL_ECC_PAGESIZE_2112);
499 500
			break;
		case 4096:
501 502
			nand_chip->ecc.layout = &atmel_oobinfo_large;
			ecc_writel(host->ecc, MR, ATMEL_ECC_PAGESIZE_4224);
503 504 505 506 507 508 509 510 511 512 513 514 515 516 517 518 519 520
			break;
		default:
			/* page size not handled by HW ECC */
			/* switching back to soft ECC */
			nand_chip->ecc.mode = NAND_ECC_SOFT;
			nand_chip->ecc.calculate = NULL;
			nand_chip->ecc.correct = NULL;
			nand_chip->ecc.hwctl = NULL;
			nand_chip->ecc.read_page = NULL;
			nand_chip->ecc.postpad = 0;
			nand_chip->ecc.prepad = 0;
			nand_chip->ecc.bytes = 0;
			break;
		}
	}

	/* second phase scan */
	if (nand_scan_tail(mtd)) {
A
Andrew Victor 已提交
521
		res = -ENXIO;
522
		goto err_scan_tail;
A
Andrew Victor 已提交
523 524 525
	}

#ifdef CONFIG_MTD_PARTITIONS
526
#ifdef CONFIG_MTD_CMDLINE_PARTS
527
	mtd->name = "atmel_nand";
528 529
	num_partitions = parse_mtd_partitions(mtd, part_probes,
					      &partitions, 0);
530
#endif
531 532 533
	if (num_partitions <= 0 && host->board->partition_info)
		partitions = host->board->partition_info(mtd->size,
							 &num_partitions);
A
Andrew Victor 已提交
534 535

	if ((!partitions) || (num_partitions == 0)) {
536
		printk(KERN_ERR "atmel_nand: No partitions defined, or unsupported device.\n");
A
Andrew Victor 已提交
537
		res = ENXIO;
538
		goto err_no_partitions;
A
Andrew Victor 已提交
539 540 541 542 543 544 545 546 547 548
	}

	res = add_mtd_partitions(mtd, partitions, num_partitions);
#else
	res = add_mtd_device(mtd);
#endif

	if (!res)
		return res;

549
#ifdef CONFIG_MTD_PARTITIONS
550
err_no_partitions:
551
#endif
A
Andrew Victor 已提交
552
	nand_release(mtd);
553 554 555
err_scan_tail:
err_scan_ident:
err_no_card:
556
	atmel_nand_disable(host);
A
Andrew Victor 已提交
557
	platform_set_drvdata(pdev, NULL);
558 559 560
	if (host->ecc)
		iounmap(host->ecc);
err_ecc_ioremap:
A
Andrew Victor 已提交
561
	iounmap(host->io_base);
562
err_nand_ioremap:
A
Andrew Victor 已提交
563 564 565 566 567 568 569
	kfree(host);
	return res;
}

/*
 * Remove a NAND device.
 */
570
static int __exit atmel_nand_remove(struct platform_device *pdev)
A
Andrew Victor 已提交
571
{
572
	struct atmel_nand_host *host = platform_get_drvdata(pdev);
A
Andrew Victor 已提交
573 574 575 576
	struct mtd_info *mtd = &host->mtd;

	nand_release(mtd);

577
	atmel_nand_disable(host);
A
Andrew Victor 已提交
578

579 580
	if (host->ecc)
		iounmap(host->ecc);
A
Andrew Victor 已提交
581 582 583 584 585 586
	iounmap(host->io_base);
	kfree(host);

	return 0;
}

587
static struct platform_driver atmel_nand_driver = {
588
	.remove		= __exit_p(atmel_nand_remove),
A
Andrew Victor 已提交
589
	.driver		= {
590
		.name	= "atmel_nand",
A
Andrew Victor 已提交
591 592 593 594
		.owner	= THIS_MODULE,
	},
};

595
static int __init atmel_nand_init(void)
A
Andrew Victor 已提交
596
{
597
	return platform_driver_probe(&atmel_nand_driver, atmel_nand_probe);
A
Andrew Victor 已提交
598 599 600
}


601
static void __exit atmel_nand_exit(void)
A
Andrew Victor 已提交
602
{
603
	platform_driver_unregister(&atmel_nand_driver);
A
Andrew Victor 已提交
604 605 606
}


607 608
module_init(atmel_nand_init);
module_exit(atmel_nand_exit);
A
Andrew Victor 已提交
609 610 611

MODULE_LICENSE("GPL");
MODULE_AUTHOR("Rick Bronson");
612
MODULE_DESCRIPTION("NAND/SmartMedia driver for AT91 / AVR32");
613
MODULE_ALIAS("platform:atmel_nand");