edb7312.c 5.3 KB
Newer Older
L
Linus Torvalds 已提交
1 2 3
/*
 *  drivers/mtd/nand/edb7312.c
 *
4
 *  Copyright (C) 2002 Marius Gröger (mag@sysgo.de)
L
Linus Torvalds 已提交
5 6 7 8
 *
 *  Derived from drivers/mtd/nand/autcpu12.c
 *       Copyright (c) 2001 Thomas Gleixner (gleixner@autronix.de)
 *
9
 * $Id: edb7312.c,v 1.12 2005/11/07 11:14:30 gleixner Exp $
L
Linus Torvalds 已提交
10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27
 *
 * 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.
 *
 *  Overview:
 *   This is a device driver for the NAND flash device found on the
 *   CLEP7312 board which utilizes the Toshiba TC58V64AFT part. This is
 *   a 64Mibit (8MiB x 8 bits) NAND flash device.
 */

#include <linux/slab.h>
#include <linux/module.h>
#include <linux/init.h>
#include <linux/mtd/mtd.h>
#include <linux/mtd/nand.h>
#include <linux/mtd/partitions.h>
#include <asm/io.h>
28
#include <asm/arch/hardware.h>	/* for CLPS7111_VIRT_BASE */
L
Linus Torvalds 已提交
29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56
#include <asm/sizes.h>
#include <asm/hardware/clps7111.h>

/*
 * MTD structure for EDB7312 board
 */
static struct mtd_info *ep7312_mtd = NULL;

/*
 * Values specific to the EDB7312 board (used with EP7312 processor)
 */
#define EP7312_FIO_PBASE 0x10000000	/* Phys address of flash */
#define EP7312_PXDR	0x0001	/*
				 * IO offset to Port B data register
				 * where the CLE, ALE and NCE pins
				 * are wired to.
				 */
#define EP7312_PXDDR	0x0041	/*
				 * IO offset to Port B data direction
				 * register so we can control the IO
				 * lines.
				 */

/*
 * Module stuff
 */

static unsigned long ep7312_fio_pbase = EP7312_FIO_PBASE;
57 58
static void __iomem *ep7312_pxdr = (void __iomem *)EP7312_PXDR;
static void __iomem *ep7312_pxddr = (void __iomem *)EP7312_PXDDR;
L
Linus Torvalds 已提交
59 60 61 62 63 64

#ifdef CONFIG_MTD_PARTITIONS
/*
 * Define static partitions for flash device
 */
static struct mtd_partition partition_info[] = {
65 66 67
	{.name = "EP7312 Nand Flash",
	 .offset = 0,
	 .size = 8 * 1024 * 1024}
L
Linus Torvalds 已提交
68
};
69

L
Linus Torvalds 已提交
70 71 72 73
#define NUM_PARTITIONS 1

#endif

74
/*
L
Linus Torvalds 已提交
75 76
 *	hardware specific access to control-lines
 */
77
static void ep7312_hwcontrol(struct mtd_info *mtd, int cmd)
L
Linus Torvalds 已提交
78
{
79
	switch (cmd) {
80 81 82

	case NAND_CTL_SETCLE:
		clps_writeb(clps_readb(ep7312_pxdr) | 0x10, ep7312_pxdr);
L
Linus Torvalds 已提交
83
		break;
84
	case NAND_CTL_CLRCLE:
L
Linus Torvalds 已提交
85 86
		clps_writeb(clps_readb(ep7312_pxdr) & ~0x10, ep7312_pxdr);
		break;
87

L
Linus Torvalds 已提交
88 89 90 91 92 93
	case NAND_CTL_SETALE:
		clps_writeb(clps_readb(ep7312_pxdr) | 0x20, ep7312_pxdr);
		break;
	case NAND_CTL_CLRALE:
		clps_writeb(clps_readb(ep7312_pxdr) & ~0x20, ep7312_pxdr);
		break;
94

L
Linus Torvalds 已提交
95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110
	case NAND_CTL_SETNCE:
		clps_writeb((clps_readb(ep7312_pxdr) | 0x80) & ~0x40, ep7312_pxdr);
		break;
	case NAND_CTL_CLRNCE:
		clps_writeb((clps_readb(ep7312_pxdr) | 0x80) | 0x40, ep7312_pxdr);
		break;
	}
}

/*
 *	read device ready pin
 */
static int ep7312_device_ready(struct mtd_info *mtd)
{
	return 1;
}
111

L
Linus Torvalds 已提交
112 113 114 115 116 117 118
#ifdef CONFIG_MTD_PARTITIONS
const char *part_probes[] = { "cmdlinepart", NULL };
#endif

/*
 * Main initialization routine
 */
119
static int __init ep7312_init(void)
L
Linus Torvalds 已提交
120 121 122 123 124
{
	struct nand_chip *this;
	const char *part_type = 0;
	int mtd_parts_nb = 0;
	struct mtd_partition *mtd_parts = 0;
125
	void __iomem *ep7312_fio_base;
126

L
Linus Torvalds 已提交
127
	/* Allocate memory for MTD device structure and private data */
128
	ep7312_mtd = kmalloc(sizeof(struct mtd_info) + sizeof(struct nand_chip), GFP_KERNEL);
L
Linus Torvalds 已提交
129 130 131 132
	if (!ep7312_mtd) {
		printk("Unable to allocate EDB7312 NAND MTD device structure.\n");
		return -ENOMEM;
	}
133

L
Linus Torvalds 已提交
134 135
	/* map physical adress */
	ep7312_fio_base = ioremap(ep7312_fio_pbase, SZ_1K);
136
	if (!ep7312_fio_base) {
L
Linus Torvalds 已提交
137 138 139 140
		printk("ioremap EDB7312 NAND flash failed\n");
		kfree(ep7312_mtd);
		return -EIO;
	}
141

L
Linus Torvalds 已提交
142
	/* Get pointer to private data */
143
	this = (struct nand_chip *)(&ep7312_mtd[1]);
144

L
Linus Torvalds 已提交
145
	/* Initialize structures */
146 147
	memset(ep7312_mtd, 0, sizeof(struct mtd_info));
	memset(this, 0, sizeof(struct nand_chip));
148

L
Linus Torvalds 已提交
149 150
	/* Link the private data with the MTD structure */
	ep7312_mtd->priv = this;
151
	ep7312_mtd->owner = THIS_MODULE;
152

L
Linus Torvalds 已提交
153 154 155 156 157
	/*
	 * Set GPIO Port B control register so that the pins are configured
	 * to be outputs for controlling the NAND flash.
	 */
	clps_writeb(0xf0, ep7312_pxddr);
158

L
Linus Torvalds 已提交
159 160 161 162 163 164 165
	/* insert callbacks */
	this->IO_ADDR_R = ep7312_fio_base;
	this->IO_ADDR_W = ep7312_fio_base;
	this->hwcontrol = ep7312_hwcontrol;
	this->dev_ready = ep7312_device_ready;
	/* 15 us command delay time */
	this->chip_delay = 15;
166

L
Linus Torvalds 已提交
167
	/* Scan to find existence of the device */
168
	if (nand_scan(ep7312_mtd, 1)) {
L
Linus Torvalds 已提交
169
		iounmap((void *)ep7312_fio_base);
170
		kfree(ep7312_mtd);
L
Linus Torvalds 已提交
171 172 173 174
		return -ENXIO;
	}
#ifdef CONFIG_MTD_PARTITIONS
	ep7312_mtd->name = "edb7312-nand";
175
	mtd_parts_nb = parse_mtd_partitions(ep7312_mtd, part_probes, &mtd_parts, 0);
L
Linus Torvalds 已提交
176 177 178 179 180 181 182 183 184 185
	if (mtd_parts_nb > 0)
		part_type = "command line";
	else
		mtd_parts_nb = 0;
#endif
	if (mtd_parts_nb == 0) {
		mtd_parts = partition_info;
		mtd_parts_nb = NUM_PARTITIONS;
		part_type = "static";
	}
186

L
Linus Torvalds 已提交
187 188 189
	/* Register the partitions */
	printk(KERN_NOTICE "Using %s partition definition\n", part_type);
	add_mtd_partitions(ep7312_mtd, mtd_parts, mtd_parts_nb);
190

L
Linus Torvalds 已提交
191 192 193
	/* Return happy */
	return 0;
}
194

L
Linus Torvalds 已提交
195 196 197 198 199
module_init(ep7312_init);

/*
 * Clean up routine
 */
200
static void __exit ep7312_cleanup(void)
L
Linus Torvalds 已提交
201
{
202
	struct nand_chip *this = (struct nand_chip *)&ep7312_mtd[1];
203

L
Linus Torvalds 已提交
204
	/* Release resources, unregister device */
205
	nand_release(ap7312_mtd);
206

L
Linus Torvalds 已提交
207
	/* Free internal data buffer */
208
	kfree(this->data_buf);
209

L
Linus Torvalds 已提交
210
	/* Free the MTD device structure */
211
	kfree(ep7312_mtd);
L
Linus Torvalds 已提交
212
}
213

L
Linus Torvalds 已提交
214 215 216 217 218
module_exit(ep7312_cleanup);

MODULE_LICENSE("GPL");
MODULE_AUTHOR("Marius Groeger <mag@sysgo.de>");
MODULE_DESCRIPTION("MTD map driver for Cogent EDB7312 board");