|
@@ -45,6 +45,7 @@
|
|
|
#define I82802AB 0x00ad
|
|
|
#define I82802AC 0x00ac
|
|
|
#define PF38F4476 0x881c
|
|
|
+#define M28F00AP30 0x8963
|
|
|
/* STMicroelectronics chips */
|
|
|
#define M50LPW080 0x002F
|
|
|
#define M50FLW080A 0x0080
|
|
@@ -375,6 +376,17 @@ static void cfi_fixup_major_minor(struct cfi_private *cfi,
|
|
|
extp->MinorVersion = '1';
|
|
|
}
|
|
|
|
|
|
+static int cfi_is_micron_28F00AP30(struct cfi_private *cfi, struct flchip *chip)
|
|
|
+{
|
|
|
+ /*
|
|
|
+ * Micron(was Numonyx) 1Gbit bottom boot are buggy w.r.t
|
|
|
+ * Erase Supend for their small Erase Blocks(0x8000)
|
|
|
+ */
|
|
|
+ if (cfi->mfr == CFI_MFR_INTEL && cfi->id == M28F00AP30)
|
|
|
+ return 1;
|
|
|
+ return 0;
|
|
|
+}
|
|
|
+
|
|
|
static inline struct cfi_pri_intelext *
|
|
|
read_pri_intelext(struct map_info *map, __u16 adr)
|
|
|
{
|
|
@@ -831,21 +843,30 @@ static int chip_ready (struct map_info *map, struct flchip *chip, unsigned long
|
|
|
(mode == FL_WRITING && (cfip->SuspendCmdSupport & 1))))
|
|
|
goto sleep;
|
|
|
|
|
|
+ /* Do not allow suspend iff read/write to EB address */
|
|
|
+ if ((adr & chip->in_progress_block_mask) ==
|
|
|
+ chip->in_progress_block_addr)
|
|
|
+ goto sleep;
|
|
|
+
|
|
|
+ /* do not suspend small EBs, buggy Micron Chips */
|
|
|
+ if (cfi_is_micron_28F00AP30(cfi, chip) &&
|
|
|
+ (chip->in_progress_block_mask == ~(0x8000-1)))
|
|
|
+ goto sleep;
|
|
|
|
|
|
/* Erase suspend */
|
|
|
- map_write(map, CMD(0xB0), adr);
|
|
|
+ map_write(map, CMD(0xB0), chip->in_progress_block_addr);
|
|
|
|
|
|
/* If the flash has finished erasing, then 'erase suspend'
|
|
|
* appears to make some (28F320) flash devices switch to
|
|
|
* 'read' mode. Make sure that we switch to 'read status'
|
|
|
* mode so we get the right data. --rmk
|
|
|
*/
|
|
|
- map_write(map, CMD(0x70), adr);
|
|
|
+ map_write(map, CMD(0x70), chip->in_progress_block_addr);
|
|
|
chip->oldstate = FL_ERASING;
|
|
|
chip->state = FL_ERASE_SUSPENDING;
|
|
|
chip->erase_suspended = 1;
|
|
|
for (;;) {
|
|
|
- status = map_read(map, adr);
|
|
|
+ status = map_read(map, chip->in_progress_block_addr);
|
|
|
if (map_word_andequal(map, status, status_OK, status_OK))
|
|
|
break;
|
|
|
|
|
@@ -1041,8 +1062,8 @@ static void put_chip(struct map_info *map, struct flchip *chip, unsigned long ad
|
|
|
sending the 0x70 (Read Status) command to an erasing
|
|
|
chip and expecting it to be ignored, that's what we
|
|
|
do. */
|
|
|
- map_write(map, CMD(0xd0), adr);
|
|
|
- map_write(map, CMD(0x70), adr);
|
|
|
+ map_write(map, CMD(0xd0), chip->in_progress_block_addr);
|
|
|
+ map_write(map, CMD(0x70), chip->in_progress_block_addr);
|
|
|
chip->oldstate = FL_READY;
|
|
|
chip->state = FL_ERASING;
|
|
|
break;
|
|
@@ -1933,6 +1954,8 @@ static int __xipram do_erase_oneblock(struct map_info *map, struct flchip *chip,
|
|
|
map_write(map, CMD(0xD0), adr);
|
|
|
chip->state = FL_ERASING;
|
|
|
chip->erase_suspended = 0;
|
|
|
+ chip->in_progress_block_addr = adr;
|
|
|
+ chip->in_progress_block_mask = ~(len - 1);
|
|
|
|
|
|
ret = INVAL_CACHE_AND_WAIT(map, chip, adr,
|
|
|
adr, len,
|