Linux 2.6.17.7
[linux/fpc-iii.git] / drivers / mtd / chips / sharp.c
blob3cc0b23c5865cd3cd3fb1ffe5ad4496e7b5aeb59
1 /*
2 * MTD chip driver for pre-CFI Sharp flash chips
4 * Copyright 2000,2001 David A. Schleef <ds@schleef.org>
5 * 2000,2001 Lineo, Inc.
7 * $Id: sharp.c,v 1.17 2005/11/29 14:28:28 gleixner Exp $
9 * Devices supported:
10 * LH28F016SCT Symmetrical block flash memory, 2Mx8
11 * LH28F008SCT Symmetrical block flash memory, 1Mx8
13 * Documentation:
14 * http://www.sharpmeg.com/datasheets/memic/flashcmp/
15 * http://www.sharpmeg.com/datasheets/memic/flashcmp/01symf/16m/016sctl9.pdf
16 * 016sctl9.pdf
18 * Limitations:
19 * This driver only supports 4x1 arrangement of chips.
20 * Not tested on anything but PowerPC.
23 #include <linux/kernel.h>
24 #include <linux/module.h>
25 #include <linux/types.h>
26 #include <linux/sched.h>
27 #include <linux/errno.h>
28 #include <linux/interrupt.h>
29 #include <linux/mtd/map.h>
30 #include <linux/mtd/mtd.h>
31 #include <linux/mtd/cfi.h>
32 #include <linux/delay.h>
33 #include <linux/init.h>
34 #include <linux/slab.h>
36 #define CMD_RESET 0xffffffff
37 #define CMD_READ_ID 0x90909090
38 #define CMD_READ_STATUS 0x70707070
39 #define CMD_CLEAR_STATUS 0x50505050
40 #define CMD_BLOCK_ERASE_1 0x20202020
41 #define CMD_BLOCK_ERASE_2 0xd0d0d0d0
42 #define CMD_BYTE_WRITE 0x40404040
43 #define CMD_SUSPEND 0xb0b0b0b0
44 #define CMD_RESUME 0xd0d0d0d0
45 #define CMD_SET_BLOCK_LOCK_1 0x60606060
46 #define CMD_SET_BLOCK_LOCK_2 0x01010101
47 #define CMD_SET_MASTER_LOCK_1 0x60606060
48 #define CMD_SET_MASTER_LOCK_2 0xf1f1f1f1
49 #define CMD_CLEAR_BLOCK_LOCKS_1 0x60606060
50 #define CMD_CLEAR_BLOCK_LOCKS_2 0xd0d0d0d0
52 #define SR_READY 0x80808080 // 1 = ready
53 #define SR_ERASE_SUSPEND 0x40404040 // 1 = block erase suspended
54 #define SR_ERROR_ERASE 0x20202020 // 1 = error in block erase or clear lock bits
55 #define SR_ERROR_WRITE 0x10101010 // 1 = error in byte write or set lock bit
56 #define SR_VPP 0x08080808 // 1 = Vpp is low
57 #define SR_WRITE_SUSPEND 0x04040404 // 1 = byte write suspended
58 #define SR_PROTECT 0x02020202 // 1 = lock bit set
59 #define SR_RESERVED 0x01010101
61 #define SR_ERRORS (SR_ERROR_ERASE|SR_ERROR_WRITE|SR_VPP|SR_PROTECT)
63 /* Configuration options */
65 #undef AUTOUNLOCK /* automatically unlocks blocks before erasing */
67 static struct mtd_info *sharp_probe(struct map_info *);
69 static int sharp_probe_map(struct map_info *map,struct mtd_info *mtd);
71 static int sharp_read(struct mtd_info *mtd, loff_t from, size_t len,
72 size_t *retlen, u_char *buf);
73 static int sharp_write(struct mtd_info *mtd, loff_t from, size_t len,
74 size_t *retlen, const u_char *buf);
75 static int sharp_erase(struct mtd_info *mtd, struct erase_info *instr);
76 static void sharp_sync(struct mtd_info *mtd);
77 static int sharp_suspend(struct mtd_info *mtd);
78 static void sharp_resume(struct mtd_info *mtd);
79 static void sharp_destroy(struct mtd_info *mtd);
81 static int sharp_write_oneword(struct map_info *map, struct flchip *chip,
82 unsigned long adr, __u32 datum);
83 static int sharp_erase_oneblock(struct map_info *map, struct flchip *chip,
84 unsigned long adr);
85 #ifdef AUTOUNLOCK
86 static void sharp_unlock_oneblock(struct map_info *map, struct flchip *chip,
87 unsigned long adr);
88 #endif
91 struct sharp_info{
92 struct flchip *chip;
93 int bogus;
94 int chipshift;
95 int numchips;
96 struct flchip chips[1];
99 static void sharp_destroy(struct mtd_info *mtd);
101 static struct mtd_chip_driver sharp_chipdrv = {
102 .probe = sharp_probe,
103 .destroy = sharp_destroy,
104 .name = "sharp",
105 .module = THIS_MODULE
109 static struct mtd_info *sharp_probe(struct map_info *map)
111 struct mtd_info *mtd = NULL;
112 struct sharp_info *sharp = NULL;
113 int width;
115 mtd = kmalloc(sizeof(*mtd), GFP_KERNEL);
116 if(!mtd)
117 return NULL;
119 sharp = kmalloc(sizeof(*sharp), GFP_KERNEL);
120 if(!sharp) {
121 kfree(mtd);
122 return NULL;
125 memset(mtd, 0, sizeof(*mtd));
127 width = sharp_probe_map(map,mtd);
128 if(!width){
129 kfree(mtd);
130 kfree(sharp);
131 return NULL;
134 mtd->priv = map;
135 mtd->type = MTD_NORFLASH;
136 mtd->erase = sharp_erase;
137 mtd->read = sharp_read;
138 mtd->write = sharp_write;
139 mtd->sync = sharp_sync;
140 mtd->suspend = sharp_suspend;
141 mtd->resume = sharp_resume;
142 mtd->flags = MTD_CAP_NORFLASH;
143 mtd->name = map->name;
145 memset(sharp, 0, sizeof(*sharp));
146 sharp->chipshift = 23;
147 sharp->numchips = 1;
148 sharp->chips[0].start = 0;
149 sharp->chips[0].state = FL_READY;
150 sharp->chips[0].mutex = &sharp->chips[0]._spinlock;
151 sharp->chips[0].word_write_time = 0;
152 init_waitqueue_head(&sharp->chips[0].wq);
153 spin_lock_init(&sharp->chips[0]._spinlock);
155 map->fldrv = &sharp_chipdrv;
156 map->fldrv_priv = sharp;
158 __module_get(THIS_MODULE);
159 return mtd;
162 static inline void sharp_send_cmd(struct map_info *map, unsigned long cmd, unsigned long adr)
164 map_word map_cmd;
165 map_cmd.x[0] = cmd;
166 map_write(map, map_cmd, adr);
169 static int sharp_probe_map(struct map_info *map,struct mtd_info *mtd)
171 map_word tmp, read0, read4;
172 unsigned long base = 0;
173 int width = 4;
175 tmp = map_read(map, base+0);
177 sharp_send_cmd(map, CMD_READ_ID, base+0);
179 read0 = map_read(map, base+0);
180 read4 = map_read(map, base+4);
181 if(read0.x[0] == 0x89898989){
182 printk("Looks like sharp flash\n");
183 switch(read4.x[0]){
184 case 0xaaaaaaaa:
185 case 0xa0a0a0a0:
186 /* aa - LH28F016SCT-L95 2Mx8, 32 64k blocks*/
187 /* a0 - LH28F016SCT-Z4 2Mx8, 32 64k blocks*/
188 mtd->erasesize = 0x10000 * width;
189 mtd->size = 0x200000 * width;
190 return width;
191 case 0xa6a6a6a6:
192 /* a6 - LH28F008SCT-L12 1Mx8, 16 64k blocks*/
193 /* a6 - LH28F008SCR-L85 1Mx8, 16 64k blocks*/
194 mtd->erasesize = 0x10000 * width;
195 mtd->size = 0x100000 * width;
196 return width;
197 #if 0
198 case 0x00000000: /* unknown */
199 /* XX - LH28F004SCT 512kx8, 8 64k blocks*/
200 mtd->erasesize = 0x10000 * width;
201 mtd->size = 0x80000 * width;
202 return width;
203 #endif
204 default:
205 printk("Sort-of looks like sharp flash, 0x%08lx 0x%08lx\n",
206 read0.x[0], read4.x[0]);
208 }else if((map_read(map, base+0).x[0] == CMD_READ_ID)){
209 /* RAM, probably */
210 printk("Looks like RAM\n");
211 map_write(map, tmp, base+0);
212 }else{
213 printk("Doesn't look like sharp flash, 0x%08lx 0x%08lx\n",
214 read0.x[0], read4.x[0]);
217 return 0;
220 /* This function returns with the chip->mutex lock held. */
221 static int sharp_wait(struct map_info *map, struct flchip *chip)
223 int i;
224 map_word status;
225 unsigned long timeo = jiffies + HZ;
226 DECLARE_WAITQUEUE(wait, current);
227 int adr = 0;
229 retry:
230 spin_lock_bh(chip->mutex);
232 switch(chip->state){
233 case FL_READY:
234 sharp_send_cmd(map, CMD_READ_STATUS, adr);
235 chip->state = FL_STATUS;
236 case FL_STATUS:
237 for(i=0;i<100;i++){
238 status = map_read(map, adr);
239 if((status.x[0] & SR_READY)==SR_READY)
240 break;
241 udelay(1);
243 break;
244 default:
245 printk("Waiting for chip\n");
247 set_current_state(TASK_INTERRUPTIBLE);
248 add_wait_queue(&chip->wq, &wait);
250 spin_unlock_bh(chip->mutex);
252 schedule();
253 remove_wait_queue(&chip->wq, &wait);
255 if(signal_pending(current))
256 return -EINTR;
258 timeo = jiffies + HZ;
260 goto retry;
263 sharp_send_cmd(map, CMD_RESET, adr);
265 chip->state = FL_READY;
267 return 0;
270 static void sharp_release(struct flchip *chip)
272 wake_up(&chip->wq);
273 spin_unlock_bh(chip->mutex);
276 static int sharp_read(struct mtd_info *mtd, loff_t from, size_t len,
277 size_t *retlen, u_char *buf)
279 struct map_info *map = mtd->priv;
280 struct sharp_info *sharp = map->fldrv_priv;
281 int chipnum;
282 int ret = 0;
283 int ofs = 0;
285 chipnum = (from >> sharp->chipshift);
286 ofs = from & ((1 << sharp->chipshift)-1);
288 *retlen = 0;
290 while(len){
291 unsigned long thislen;
293 if(chipnum>=sharp->numchips)
294 break;
296 thislen = len;
297 if(ofs+thislen >= (1<<sharp->chipshift))
298 thislen = (1<<sharp->chipshift) - ofs;
300 ret = sharp_wait(map,&sharp->chips[chipnum]);
301 if(ret<0)
302 break;
304 map_copy_from(map,buf,ofs,thislen);
306 sharp_release(&sharp->chips[chipnum]);
308 *retlen += thislen;
309 len -= thislen;
310 buf += thislen;
312 ofs = 0;
313 chipnum++;
315 return ret;
318 static int sharp_write(struct mtd_info *mtd, loff_t to, size_t len,
319 size_t *retlen, const u_char *buf)
321 struct map_info *map = mtd->priv;
322 struct sharp_info *sharp = map->fldrv_priv;
323 int ret = 0;
324 int i,j;
325 int chipnum;
326 unsigned long ofs;
327 union { u32 l; unsigned char uc[4]; } tbuf;
329 *retlen = 0;
331 while(len){
332 tbuf.l = 0xffffffff;
333 chipnum = to >> sharp->chipshift;
334 ofs = to & ((1<<sharp->chipshift)-1);
336 j=0;
337 for(i=ofs&3;i<4 && len;i++){
338 tbuf.uc[i] = *buf;
339 buf++;
340 to++;
341 len--;
342 j++;
344 sharp_write_oneword(map, &sharp->chips[chipnum], ofs&~3, tbuf.l);
345 if(ret<0)
346 return ret;
347 (*retlen)+=j;
350 return 0;
353 static int sharp_write_oneword(struct map_info *map, struct flchip *chip,
354 unsigned long adr, __u32 datum)
356 int ret;
357 int timeo;
358 int try;
359 int i;
360 map_word data, status;
362 status.x[0] = 0;
363 ret = sharp_wait(map,chip);
365 for(try=0;try<10;try++){
366 sharp_send_cmd(map, CMD_BYTE_WRITE, adr);
367 /* cpu_to_le32 -> hack to fix the writel be->le conversion */
368 data.x[0] = cpu_to_le32(datum);
369 map_write(map, data, adr);
371 chip->state = FL_WRITING;
373 timeo = jiffies + (HZ/2);
375 sharp_send_cmd(map, CMD_READ_STATUS, adr);
376 for(i=0;i<100;i++){
377 status = map_read(map, adr);
378 if((status.x[0] & SR_READY) == SR_READY)
379 break;
381 if(i==100){
382 printk("sharp: timed out writing\n");
385 if(!(status.x[0] & SR_ERRORS))
386 break;
388 printk("sharp: error writing byte at addr=%08lx status=%08lx\n", adr, status.x[0]);
390 sharp_send_cmd(map, CMD_CLEAR_STATUS, adr);
392 sharp_send_cmd(map, CMD_RESET, adr);
393 chip->state = FL_READY;
395 wake_up(&chip->wq);
396 spin_unlock_bh(chip->mutex);
398 return 0;
401 static int sharp_erase(struct mtd_info *mtd, struct erase_info *instr)
403 struct map_info *map = mtd->priv;
404 struct sharp_info *sharp = map->fldrv_priv;
405 unsigned long adr,len;
406 int chipnum, ret=0;
408 //printk("sharp_erase()\n");
409 if(instr->addr & (mtd->erasesize - 1))
410 return -EINVAL;
411 if(instr->len & (mtd->erasesize - 1))
412 return -EINVAL;
413 if(instr->len + instr->addr > mtd->size)
414 return -EINVAL;
416 chipnum = instr->addr >> sharp->chipshift;
417 adr = instr->addr & ((1<<sharp->chipshift)-1);
418 len = instr->len;
420 while(len){
421 ret = sharp_erase_oneblock(map, &sharp->chips[chipnum], adr);
422 if(ret)return ret;
424 adr += mtd->erasesize;
425 len -= mtd->erasesize;
426 if(adr >> sharp->chipshift){
427 adr = 0;
428 chipnum++;
429 if(chipnum>=sharp->numchips)
430 break;
434 instr->state = MTD_ERASE_DONE;
435 mtd_erase_callback(instr);
437 return 0;
440 static int sharp_do_wait_for_ready(struct map_info *map, struct flchip *chip,
441 unsigned long adr)
443 int ret;
444 unsigned long timeo;
445 map_word status;
446 DECLARE_WAITQUEUE(wait, current);
448 sharp_send_cmd(map, CMD_READ_STATUS, adr);
449 status = map_read(map, adr);
451 timeo = jiffies + HZ;
453 while(time_before(jiffies, timeo)){
454 sharp_send_cmd(map, CMD_READ_STATUS, adr);
455 status = map_read(map, adr);
456 if((status.x[0] & SR_READY)==SR_READY){
457 ret = 0;
458 goto out;
460 set_current_state(TASK_INTERRUPTIBLE);
461 add_wait_queue(&chip->wq, &wait);
463 //spin_unlock_bh(chip->mutex);
465 schedule_timeout(1);
466 schedule();
467 remove_wait_queue(&chip->wq, &wait);
469 //spin_lock_bh(chip->mutex);
471 if (signal_pending(current)){
472 ret = -EINTR;
473 goto out;
477 ret = -ETIME;
478 out:
479 return ret;
482 static int sharp_erase_oneblock(struct map_info *map, struct flchip *chip,
483 unsigned long adr)
485 int ret;
486 //int timeo;
487 map_word status;
488 //int i;
490 //printk("sharp_erase_oneblock()\n");
492 #ifdef AUTOUNLOCK
493 /* This seems like a good place to do an unlock */
494 sharp_unlock_oneblock(map,chip,adr);
495 #endif
497 sharp_send_cmd(map, CMD_BLOCK_ERASE_1, adr);
498 sharp_send_cmd(map, CMD_BLOCK_ERASE_2, adr);
500 chip->state = FL_ERASING;
502 ret = sharp_do_wait_for_ready(map,chip,adr);
503 if(ret<0)return ret;
505 sharp_send_cmd(map, CMD_READ_STATUS, adr);
506 status = map_read(map, adr);
508 if(!(status.x[0] & SR_ERRORS)){
509 sharp_send_cmd(map, CMD_RESET, adr);
510 chip->state = FL_READY;
511 //spin_unlock_bh(chip->mutex);
512 return 0;
515 printk("sharp: error erasing block at addr=%08lx status=%08lx\n", adr, status.x[0]);
516 sharp_send_cmd(map, CMD_CLEAR_STATUS, adr);
518 //spin_unlock_bh(chip->mutex);
520 return -EIO;
523 #ifdef AUTOUNLOCK
524 static void sharp_unlock_oneblock(struct map_info *map, struct flchip *chip,
525 unsigned long adr)
527 int i;
528 map_word status;
530 sharp_send_cmd(map, CMD_CLEAR_BLOCK_LOCKS_1, adr);
531 sharp_send_cmd(map, CMD_CLEAR_BLOCK_LOCKS_2, adr);
533 udelay(100);
535 status = map_read(map, adr);
536 printk("status=%08lx\n", status.x[0]);
538 for(i=0;i<1000;i++){
539 //sharp_send_cmd(map, CMD_READ_STATUS, adr);
540 status = map_read(map, adr);
541 if((status.x[0] & SR_READY) == SR_READY)
542 break;
543 udelay(100);
545 if(i==1000){
546 printk("sharp: timed out unlocking block\n");
549 if(!(status.x[0] & SR_ERRORS)){
550 sharp_send_cmd(map, CMD_RESET, adr);
551 chip->state = FL_READY;
552 return;
555 printk("sharp: error unlocking block at addr=%08lx status=%08lx\n", adr, status.x[0]);
556 sharp_send_cmd(map, CMD_CLEAR_STATUS, adr);
558 #endif
560 static void sharp_sync(struct mtd_info *mtd)
562 //printk("sharp_sync()\n");
565 static int sharp_suspend(struct mtd_info *mtd)
567 printk("sharp_suspend()\n");
568 return -EINVAL;
571 static void sharp_resume(struct mtd_info *mtd)
573 printk("sharp_resume()\n");
577 static void sharp_destroy(struct mtd_info *mtd)
579 printk("sharp_destroy()\n");
583 static int __init sharp_probe_init(void)
585 printk("MTD Sharp chip driver <ds@lineo.com>\n");
587 register_mtd_chip_driver(&sharp_chipdrv);
589 return 0;
592 static void __exit sharp_probe_exit(void)
594 unregister_mtd_chip_driver(&sharp_chipdrv);
597 module_init(sharp_probe_init);
598 module_exit(sharp_probe_exit);
601 MODULE_LICENSE("GPL");
602 MODULE_AUTHOR("David Schleef <ds@schleef.org>");
603 MODULE_DESCRIPTION("Old MTD chip driver for pre-CFI Sharp flash chips");