atmel_nand.c 14.8 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 198
	uint32_t *eccpos = nand_chip->ecc.layout->eccpos;
	unsigned int ecc_value;

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

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

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

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

	return 0;
}

/*
 * HW ECC read page function
 *
 * mtd:        mtd info structure
 * chip:       nand chip info structure
 * buf:        buffer to store read data
 */
220
static int atmel_nand_read_page(struct mtd_info *mtd,
221
		struct nand_chip *chip, uint8_t *buf, int page)
222 223 224 225 226 227 228 229 230
{
	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;

231 232 233 234 235 236 237 238 239 240 241 242 243
	/*
	 * 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);
	}

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 291
	/* 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
 */
292
static int atmel_nand_correct(struct mtd_info *mtd, u_char *dat,
293 294 295
		u_char *read_ecc, u_char *isnull)
{
	struct nand_chip *nand_chip = mtd->priv;
296
	struct atmel_nand_host *host = nand_chip->priv;
297 298 299 300 301 302 303
	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 */
304
	if (likely(!(ecc_status & ATMEL_ECC_RECERR)))
305 306 307
		return 0;

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

	/* if there are multiple errors */
314
	if (ecc_status & ATMEL_ECC_MULERR) {
315 316
		/* check if it is a freshly erased block
		 * (filled with 0xff) */
317 318
		if ((ecc_bit == ATMEL_ECC_BITADDR)
				&& (ecc_word == (ATMEL_ECC_WORDADDR >> 4))) {
319 320 321 322 323 324
			/* 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 */
325
		dev_dbg(host->dev, "atmel_nand : multiple errors detected."
326 327 328 329 330
				" Unable to correct.\n");
		return -EIO;
	}

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

340
	dev_dbg(host->dev, "atmel_nand : one bit error on data."
341 342 343 344 345 346 347 348 349 350 351
			" (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);
	}
352
	dev_dbg(host->dev, "atmel_nand : error corrected\n");
353 354 355 356
	return 1;
}

/*
357
 * Enable HW ECC : unused on most chips
358
 */
359 360 361 362 363 364 365 366
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);
	}
}
367

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

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

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

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

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

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

	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;
421
	nand_chip->cmd_ctrl = atmel_nand_cmd_ctrl;
I
Ivan Kuten 已提交
422 423

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

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

A
Andrew Victor 已提交
432
	nand_chip->ecc.mode = NAND_ECC_SOFT;	/* enable ECC */
433 434 435 436 437
	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) {
438
			printk(KERN_ERR "atmel_nand: ioremap failed\n");
439 440 441
			res = -EIO;
			goto err_ecc_ioremap;
		}
442
		nand_chip->ecc.mode = NAND_ECC_HW;
443 444 445 446
		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;
447 448 449
		nand_chip->ecc.bytes = 4;
	}

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

452
	if (host->board->bus_width_16) {	/* 16-bit bus width */
453
		nand_chip->options |= NAND_BUSWIDTH_16;
454 455 456 457 458 459
		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;
	}
460

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

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

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

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

483
	if (nand_chip->ecc.mode == NAND_ECC_HW) {
484 485 486 487 488 489
		/* 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:
490 491
			nand_chip->ecc.layout = &atmel_oobinfo_small;
			ecc_writel(host->ecc, MR, ATMEL_ECC_PAGESIZE_528);
492 493
			break;
		case 1024:
494 495
			nand_chip->ecc.layout = &atmel_oobinfo_large;
			ecc_writel(host->ecc, MR, ATMEL_ECC_PAGESIZE_1056);
496 497
			break;
		case 2048:
498 499
			nand_chip->ecc.layout = &atmel_oobinfo_large;
			ecc_writel(host->ecc, MR, ATMEL_ECC_PAGESIZE_2112);
500 501
			break;
		case 4096:
502 503
			nand_chip->ecc.layout = &atmel_oobinfo_large;
			ecc_writel(host->ecc, MR, ATMEL_ECC_PAGESIZE_4224);
504 505 506 507 508 509 510 511 512 513 514 515 516 517 518 519 520 521
			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 已提交
522
		res = -ENXIO;
523
		goto err_scan_tail;
A
Andrew Victor 已提交
524 525 526
	}

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

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

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

	if (!res)
		return res;

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

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

	nand_release(mtd);

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

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

	return 0;
}

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

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


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


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

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