2  *  linux/drivers/mtd/nand/cmx270-nand.c
 
   4  *  Copyright (C) 2006 Compulab, Ltd.
 
   5  *  Mike Rapoport <mike@compulab.co.il>
 
   7  *  Derived from drivers/mtd/nand/h1910.c
 
   8  *       Copyright (C) 2002 Marius Gröger (mag@sysgo.de)
 
   9  *       Copyright (c) 2001 Thomas Gleixner (gleixner@autronix.de)
 
  12  * This program is free software; you can redistribute it and/or modify
 
  13  * it under the terms of the GNU General Public License version 2 as
 
  14  * published by the Free Software Foundation.
 
  17  *   This is a device driver for the NAND flash device found on the
 
  21 #include <linux/mtd/nand.h>
 
  22 #include <linux/mtd/partitions.h>
 
  23 #include <linux/gpio.h>
 
  27 #include <asm/mach-types.h>
 
  29 #include <mach/pxa2xx-regs.h>
 
  31 #define GPIO_NAND_CS    (11)
 
  32 #define GPIO_NAND_RB    (89)
 
  34 /* MTD structure for CM-X270 board */
 
  35 static struct mtd_info *cmx270_nand_mtd;
 
  37 /* remaped IO address of the device */
 
  38 static void __iomem *cmx270_nand_io;
 
  41  * Define static partitions for flash device
 
  43 static struct mtd_partition partition_info[] = {
 
  47                 .size   = MTDPART_SIZ_FULL
 
  50 #define NUM_PARTITIONS (ARRAY_SIZE(partition_info))
 
  52 const char *part_probes[] = { "cmdlinepart", NULL };
 
  54 static u_char cmx270_read_byte(struct mtd_info *mtd)
 
  56         struct nand_chip *this = mtd->priv;
 
  58         return (readl(this->IO_ADDR_R) >> 16);
 
  61 static void cmx270_write_buf(struct mtd_info *mtd, const u_char *buf, int len)
 
  64         struct nand_chip *this = mtd->priv;
 
  67                 writel((*buf++ << 16), this->IO_ADDR_W);
 
  70 static void cmx270_read_buf(struct mtd_info *mtd, u_char *buf, int len)
 
  73         struct nand_chip *this = mtd->priv;
 
  76                 *buf++ = readl(this->IO_ADDR_R) >> 16;
 
  79 static int cmx270_verify_buf(struct mtd_info *mtd, const u_char *buf, int len)
 
  82         struct nand_chip *this = mtd->priv;
 
  85                 if (buf[i] != (u_char)(readl(this->IO_ADDR_R) >> 16))
 
  91 static inline void nand_cs_on(void)
 
  93         gpio_set_value(GPIO_NAND_CS, 0);
 
  96 static void nand_cs_off(void)
 
 100         gpio_set_value(GPIO_NAND_CS, 1);
 
 104  *      hardware specific access to control-lines
 
 106 static void cmx270_hwcontrol(struct mtd_info *mtd, int dat,
 
 109         struct nand_chip* this = mtd->priv;
 
 110         unsigned int nandaddr = (unsigned int)this->IO_ADDR_W;
 
 114         if (ctrl & NAND_CTRL_CHANGE) {
 
 115                 if ( ctrl & NAND_ALE )
 
 116                         nandaddr |=  (1 << 3);
 
 118                         nandaddr &= ~(1 << 3);
 
 119                 if ( ctrl & NAND_CLE )
 
 120                         nandaddr |=  (1 << 2);
 
 122                         nandaddr &= ~(1 << 2);
 
 123                 if ( ctrl & NAND_NCE )
 
 130         this->IO_ADDR_W = (void __iomem*)nandaddr;
 
 131         if (dat != NAND_CMD_NONE)
 
 132                 writel((dat << 16), this->IO_ADDR_W);
 
 138  *      read device ready pin
 
 140 static int cmx270_device_ready(struct mtd_info *mtd)
 
 144         return (gpio_get_value(GPIO_NAND_RB));
 
 148  * Main initialization routine
 
 150 static int cmx270_init(void)
 
 152         struct nand_chip *this;
 
 153         const char *part_type;
 
 154         struct mtd_partition *mtd_parts;
 
 155         int mtd_parts_nb = 0;
 
 158         if (!(machine_is_armcore() && cpu_is_pxa27x()))
 
 161         ret = gpio_request(GPIO_NAND_CS, "NAND CS");
 
 163                 pr_warning("CM-X270: failed to request NAND CS gpio\n");
 
 167         gpio_direction_output(GPIO_NAND_CS, 1);
 
 169         ret = gpio_request(GPIO_NAND_RB, "NAND R/B");
 
 171                 pr_warning("CM-X270: failed to request NAND R/B gpio\n");
 
 172                 goto err_gpio_request;
 
 175         gpio_direction_input(GPIO_NAND_RB);
 
 177         /* Allocate memory for MTD device structure and private data */
 
 178         cmx270_nand_mtd = kzalloc(sizeof(struct mtd_info) +
 
 179                                   sizeof(struct nand_chip),
 
 181         if (!cmx270_nand_mtd) {
 
 182                 pr_debug("Unable to allocate CM-X270 NAND MTD device structure.\n");
 
 187         cmx270_nand_io = ioremap(PXA_CS1_PHYS, 12);
 
 188         if (!cmx270_nand_io) {
 
 189                 pr_debug("Unable to ioremap NAND device\n");
 
 194         /* Get pointer to private data */
 
 195         this = (struct nand_chip *)(&cmx270_nand_mtd[1]);
 
 197         /* Link the private data with the MTD structure */
 
 198         cmx270_nand_mtd->owner = THIS_MODULE;
 
 199         cmx270_nand_mtd->priv = this;
 
 201         /* insert callbacks */
 
 202         this->IO_ADDR_R = cmx270_nand_io;
 
 203         this->IO_ADDR_W = cmx270_nand_io;
 
 204         this->cmd_ctrl = cmx270_hwcontrol;
 
 205         this->dev_ready = cmx270_device_ready;
 
 207         /* 15 us command delay time */
 
 208         this->chip_delay = 20;
 
 209         this->ecc.mode = NAND_ECC_SOFT;
 
 211         /* read/write functions */
 
 212         this->read_byte = cmx270_read_byte;
 
 213         this->read_buf = cmx270_read_buf;
 
 214         this->write_buf = cmx270_write_buf;
 
 215         this->verify_buf = cmx270_verify_buf;
 
 217         /* Scan to find existence of the device */
 
 218         if (nand_scan (cmx270_nand_mtd, 1)) {
 
 219                 pr_notice("No NAND device\n");
 
 224 #ifdef CONFIG_MTD_CMDLINE_PARTS
 
 225         mtd_parts_nb = parse_mtd_partitions(cmx270_nand_mtd, part_probes,
 
 227         if (mtd_parts_nb > 0)
 
 228                 part_type = "command line";
 
 233                 mtd_parts = partition_info;
 
 234                 mtd_parts_nb = NUM_PARTITIONS;
 
 235                 part_type = "static";
 
 238         /* Register the partitions */
 
 239         pr_notice("Using %s partition definition\n", part_type);
 
 240         ret = add_mtd_partitions(cmx270_nand_mtd, mtd_parts, mtd_parts_nb);
 
 248         iounmap(cmx270_nand_io);
 
 250         kfree(cmx270_nand_mtd);
 
 252         gpio_free(GPIO_NAND_RB);
 
 254         gpio_free(GPIO_NAND_CS);
 
 259 module_init(cmx270_init);
 
 264 static void cmx270_cleanup(void)
 
 266         /* Release resources, unregister device */
 
 267         nand_release(cmx270_nand_mtd);
 
 269         gpio_free(GPIO_NAND_RB);
 
 270         gpio_free(GPIO_NAND_CS);
 
 272         iounmap(cmx270_nand_io);
 
 274         /* Free the MTD device structure */
 
 275         kfree (cmx270_nand_mtd);
 
 277 module_exit(cmx270_cleanup);
 
 279 MODULE_LICENSE("GPL");
 
 280 MODULE_AUTHOR("Mike Rapoport <mike@compulab.co.il>");
 
 281 MODULE_DESCRIPTION("NAND flash driver for Compulab CM-X270 Module");