2  * Flash memory interface rev.5 driver for the Intel
 
   3  * Flash chips used on the NetWinder.
 
   5  * 20/08/2000   RMK     use __ioremap to map flash into virtual memory
 
   6  *                      make a few more places use "volatile"
 
   7  * 22/05/2001   RMK     - Lock read against write
 
   8  *                      - merge printk level changes (with mods) from Alan Cox.
 
   9  *                      - use *ppos as the file position, not file->f_pos.
 
  10  *                      - fix check for out of range pos and r/w size
 
  12  * Please note that we are tampering with the only flash chip in the
 
  13  * machine, which contains the bootup code.  We therefore have the
 
  14  * power to convert these machines into doorstops...
 
  17 #include <linux/module.h>
 
  18 #include <linux/types.h>
 
  20 #include <linux/errno.h>
 
  22 #include <linux/delay.h>
 
  23 #include <linux/proc_fs.h>
 
  24 #include <linux/sched.h>
 
  25 #include <linux/miscdevice.h>
 
  26 #include <linux/spinlock.h>
 
  27 #include <linux/rwsem.h>
 
  28 #include <linux/init.h>
 
  29 #include <linux/smp_lock.h>
 
  31 #include <asm/hardware/dec21285.h>
 
  34 #include <asm/mach-types.h>
 
  35 #include <asm/system.h>
 
  36 #include <asm/uaccess.h>
 
  38 /*****************************************************************************/
 
  39 #include <asm/nwflash.h>
 
  41 #define NWFLASH_VERSION "6.4"
 
  43 static void kick_open(void);
 
  44 static int get_flash_id(void);
 
  45 static int erase_block(int nBlock);
 
  46 static int write_block(unsigned long p, const char __user *buf, int count);
 
  48 #define KFLASH_SIZE     1024*1024       //1 Meg
 
  49 #define KFLASH_SIZE4    4*1024*1024     //4 Meg
 
  50 #define KFLASH_ID       0x89A6          //Intel flash
 
  51 #define KFLASH_ID4      0xB0D4          //Intel flash 4Meg
 
  53 static int flashdebug;          //if set - we will display progress msgs
 
  55 static int gbWriteEnable;
 
  56 static int gbWriteBase64Enable;
 
  57 static volatile unsigned char *FLASH_BASE;
 
  58 static int gbFlashSize = KFLASH_SIZE;
 
  59 static DECLARE_MUTEX(nwflash_sem);
 
  61 extern spinlock_t gpio_lock;
 
  63 static int get_flash_id(void)
 
  65         volatile unsigned int c1, c2;
 
  68          * try to get flash chip ID
 
  72         *(volatile unsigned char *) (FLASH_BASE + 0x8000) = 0x90;
 
  74         c1 = *(volatile unsigned char *) FLASH_BASE;
 
  78          * on 4 Meg flash the second byte is actually at offset 2...
 
  81                 c2 = *(volatile unsigned char *) (FLASH_BASE + 2);
 
  83                 c2 = *(volatile unsigned char *) (FLASH_BASE + 1);
 
  88          * set it back to read mode
 
  90         *(volatile unsigned char *) (FLASH_BASE + 0x8000) = 0xFF;
 
  93                 gbFlashSize = KFLASH_SIZE4;
 
  98 static int flash_ioctl(struct inode *inodep, struct file *filep, unsigned int cmd, unsigned long arg)
 
 101         case CMD_WRITE_DISABLE:
 
 102                 gbWriteBase64Enable = 0;
 
 106         case CMD_WRITE_ENABLE:
 
 110         case CMD_WRITE_BASE64K_ENABLE:
 
 111                 gbWriteBase64Enable = 1;
 
 115                 gbWriteBase64Enable = 0;
 
 122 static ssize_t flash_read(struct file *file, char __user *buf, size_t size,
 
 125         unsigned long p = *ppos;
 
 126         unsigned int count = size;
 
 130                 printk(KERN_DEBUG "flash_read: flash_read: offset=0x%lX, "
 
 131                        "buffer=%p, count=0x%X.\n", p, buf, count);
 
 136         if (p < gbFlashSize) {
 
 137                 if (count > gbFlashSize - p)
 
 138                         count = gbFlashSize - p;
 
 141                  * We now lock against reads and writes. --rmk
 
 143                 if (down_interruptible(&nwflash_sem))
 
 146                 ret = copy_to_user(buf, (void *)(FLASH_BASE + p), count);
 
 157 static ssize_t flash_write(struct file *file, const char __user *buf,
 
 158                            size_t size, loff_t * ppos)
 
 160         unsigned long p = *ppos;
 
 161         unsigned int count = size;
 
 163         int nBlock, temp, rc;
 
 167                 printk("flash_write: offset=0x%lX, buffer=0x%p, count=0x%X.\n",
 
 173         if (p < 64 * 1024 && (!gbWriteBase64Enable))
 
 177          * check for out of range pos or count
 
 179         if (p >= gbFlashSize)
 
 180                 return count ? -ENXIO : 0;
 
 182         if (count > gbFlashSize - p)
 
 183                 count = gbFlashSize - p;
 
 185         if (!access_ok(VERIFY_READ, buf, count))
 
 189          * We now lock against reads and writes. --rmk
 
 191         if (down_interruptible(&nwflash_sem))
 
 196         leds_event(led_claim);
 
 197         leds_event(led_green_on);
 
 199         nBlock = (int) p >> 16; //block # of 64K bytes
 
 202          * # of 64K blocks to erase and write
 
 204         temp = ((int) (p + count) >> 16) - nBlock + 1;
 
 207          * write ends at exactly 64k boundary?
 
 209         if (((int) (p + count) & 0xFFFF) == 0)
 
 213                 printk(KERN_DEBUG "flash_write: writing %d block(s) "
 
 214                         "starting at %d.\n", temp, nBlock);
 
 216         for (; temp; temp--, nBlock++) {
 
 218                         printk(KERN_DEBUG "flash_write: erasing block %d.\n", nBlock);
 
 221                  * first we have to erase the block(s), where we will write...
 
 227                         rc = erase_block(nBlock);
 
 229                 } while (rc && i < 10);
 
 232                         printk(KERN_ERR "flash_write: erase error %x\n", rc);
 
 236                         printk(KERN_DEBUG "flash_write: writing offset %lX, "
 
 237                                "from buf %p, bytes left %X.\n", p, buf,
 
 241                  * write_block will limit write to space left in this block
 
 243                 rc = write_block(p, buf, count - written);
 
 247                  * if somehow write verify failed? Can't happen??
 
 251                          * retry up to 10 times
 
 257                                  * else quit with error...
 
 263                         printk(KERN_ERR "flash_write: write error %X\n", rc);
 
 272                         printk(KERN_DEBUG "flash_write: written 0x%X bytes OK.\n", written);
 
 276          * restore reg on exit
 
 278         leds_event(led_release);
 
 287  * The memory devices use the full 32/64 bits of the offset, and so we cannot
 
 288  * check against negative addresses: they are ok. The return value is weird,
 
 289  * though, in that case (0).
 
 291  * also note that seeking relative to the "end of file" isn't supported:
 
 292  * it has no meaning, so it returns -EINVAL.
 
 294 static loff_t flash_llseek(struct file *file, loff_t offset, int orig)
 
 300                 printk(KERN_DEBUG "flash_llseek: offset=0x%X, orig=0x%X.\n",
 
 301                        (unsigned int) offset, orig);
 
 310                 if ((unsigned int) offset > gbFlashSize) {
 
 315                 file->f_pos = (unsigned int) offset;
 
 319                 if ((file->f_pos + offset) > gbFlashSize) {
 
 323                 if ((file->f_pos + offset) < 0) {
 
 327                 file->f_pos += offset;
 
 339  * assume that main Write routine did the parameter checking...
 
 340  * so just go ahead and erase, what requested!
 
 343 static int erase_block(int nBlock)
 
 345         volatile unsigned int c1;
 
 346         volatile unsigned char *pWritePtr;
 
 347         unsigned long timeout;
 
 351          * orange LED == erase
 
 353         leds_event(led_amber_on);
 
 356          * reset footbridge to the correct offset 0 (...0..3)
 
 358         *CSR_ROMWRITEREG = 0;
 
 363         c1 = *(volatile unsigned char *) (FLASH_BASE + 0x8000);
 
 367          * reset status if old errors
 
 369         *(volatile unsigned char *) (FLASH_BASE + 0x8000) = 0x50;
 
 373          * aim at the middle of a current block...
 
 375         pWritePtr = (unsigned char *) ((unsigned int) (FLASH_BASE + 0x8000 + (nBlock << 16)));
 
 385         *(volatile unsigned char *) pWritePtr = 0x20;
 
 390         *(volatile unsigned char *) pWritePtr = 0xD0;
 
 398          * wait while erasing in process (up to 10 sec)
 
 400         timeout = jiffies + 10 * HZ;
 
 402         while (!(c1 & 0x80) && time_before(jiffies, timeout)) {
 
 407                 c1 = *(volatile unsigned char *) (pWritePtr);
 
 408                 //              printk("Flash_erase: status=%X.\n",c1);
 
 412          * set flash for normal read access
 
 415 //      *(volatile unsigned char*)(FLASH_BASE+0x8000) = 0xFF;
 
 416         *(volatile unsigned char *) pWritePtr = 0xFF;   //back to normal operation
 
 419          * check if erase errors were reported
 
 422                 printk(KERN_ERR "flash_erase: err at %p\n", pWritePtr);
 
 427                 *(volatile unsigned char *) (FLASH_BASE + 0x8000) = 0x50;
 
 432          * just to make sure - verify if erased OK...
 
 436         pWritePtr = (unsigned char *) ((unsigned int) (FLASH_BASE + (nBlock << 16)));
 
 438         for (temp = 0; temp < 16 * 1024; temp++, pWritePtr += 4) {
 
 439                 if ((temp1 = *(volatile unsigned int *) pWritePtr) != 0xFFFFFFFF) {
 
 440                         printk(KERN_ERR "flash_erase: verify err at %p = %X\n",
 
 451  * write_block will limit number of bytes written to the space in this block
 
 453 static int write_block(unsigned long p, const char __user *buf, int count)
 
 455         volatile unsigned int c1;
 
 456         volatile unsigned int c2;
 
 457         unsigned char *pWritePtr;
 
 458         unsigned int uAddress;
 
 460         unsigned long timeout;
 
 461         unsigned long timeout1;
 
 466         leds_event(led_amber_off);
 
 467         leds_event(led_red_on);
 
 469         pWritePtr = (unsigned char *) ((unsigned int) (FLASH_BASE + p));
 
 472          * check if write will end in this block....
 
 476         if (offset + count > 0x10000)
 
 477                 count = 0x10000 - offset;
 
 480          * wait up to 30 sec for this block
 
 482         timeout = jiffies + 30 * HZ;
 
 484         for (offset = 0; offset < count; offset++, pWritePtr++) {
 
 485                 uAddress = (unsigned int) pWritePtr;
 
 486                 uAddress &= 0xFFFFFFFC;
 
 487                 if (__get_user(c2, buf + offset))
 
 494                 c1 = *(volatile unsigned char *) (FLASH_BASE + 0x8000);
 
 497                  * kick open the write gate
 
 502                  * program footbridge to the correct offset...0..3
 
 504                 *CSR_ROMWRITEREG = (unsigned int) pWritePtr & 3;
 
 509                 *(volatile unsigned char *) (uAddress) = 0x40;
 
 514                 *(volatile unsigned char *) (uAddress) = c2;
 
 519                 *(volatile unsigned char *) (FLASH_BASE + 0x10000) = 0x70;
 
 524                  * wait up to 1 sec for this byte
 
 526                 timeout1 = jiffies + 1 * HZ;
 
 531                 while (!(c1 & 0x80) && time_before(jiffies, timeout1))
 
 532                         c1 = *(volatile unsigned char *) (FLASH_BASE + 0x8000);
 
 535                  * if timeout getting status
 
 537                 if (time_after_eq(jiffies, timeout1)) {
 
 542                         *(volatile unsigned char *) (FLASH_BASE + 0x8000) = 0x50;
 
 547                  * switch on read access, as a default flash operation mode
 
 553                 *(volatile unsigned char *) (FLASH_BASE + 0x8000) = 0xFF;
 
 556                  * if hardware reports an error writing, and not timeout - 
 
 557                  * reset the chip and retry
 
 564                         *(volatile unsigned char *) (FLASH_BASE + 0x8000) = 0x50;
 
 569                         if (time_before(jiffies, timeout)) {
 
 571                                         printk(KERN_DEBUG "write_block: Retrying write at 0x%X)n",
 
 572                                                pWritePtr - FLASH_BASE);
 
 577                                 leds_event(led_amber_off);
 
 585                                 leds_event(led_red_on);
 
 589                                 printk(KERN_ERR "write_block: timeout at 0x%X\n",
 
 590                                        pWritePtr - FLASH_BASE);
 
 601          * green LED == read/verify
 
 603         leds_event(led_amber_off);
 
 604         leds_event(led_green_on);
 
 608         pWritePtr = (unsigned char *) ((unsigned int) (FLASH_BASE + p));
 
 610         for (offset = 0; offset < count; offset++) {
 
 612                 if (__get_user(c, buf))
 
 615                 if ((c1 = *pWritePtr++) != c) {
 
 616                         printk(KERN_ERR "write_block: verify error at 0x%X (%02X!=%02X)\n",
 
 617                                pWritePtr - FLASH_BASE, c1, c);
 
 626 static void kick_open(void)
 
 631          * we want to write a bit pattern XXX1 to Xilinx to enable
 
 632          * the write gate, which will be open for about the next 2ms.
 
 634         spin_lock_irqsave(&gpio_lock, flags);
 
 636         spin_unlock_irqrestore(&gpio_lock, flags);
 
 639          * let the ISA bus to catch on...
 
 644 static struct file_operations flash_fops =
 
 646         .owner          = THIS_MODULE,
 
 647         .llseek         = flash_llseek,
 
 649         .write          = flash_write,
 
 650         .ioctl          = flash_ioctl,
 
 653 static struct miscdevice flash_miscdev =
 
 660 static int __init nwflash_init(void)
 
 664         if (machine_is_netwinder()) {
 
 667                 FLASH_BASE = ioremap(DC21285_FLASH, KFLASH_SIZE4);
 
 672                 if ((id != KFLASH_ID) && (id != KFLASH_ID4)) {
 
 674                         iounmap((void *)FLASH_BASE);
 
 675                         printk("Flash: incorrect ID 0x%04X.\n", id);
 
 679                 printk("Flash ROM driver v.%s, flash device ID 0x%04X, size %d Mb.\n",
 
 680                        NWFLASH_VERSION, id, gbFlashSize / (1024 * 1024));
 
 682                 ret = misc_register(&flash_miscdev);
 
 684                         iounmap((void *)FLASH_BASE);
 
 691 static void __exit nwflash_exit(void)
 
 693         misc_deregister(&flash_miscdev);
 
 694         iounmap((void *)FLASH_BASE);
 
 697 MODULE_LICENSE("GPL");
 
 699 module_param(flashdebug, bool, 0644);
 
 701 module_init(nwflash_init);
 
 702 module_exit(nwflash_exit);