Commit 8df3c8e0 authored by Nicolas Pitre's avatar Nicolas Pitre Committed by Linus Torvalds

[PATCH] L18 flash corruption fix

Another fix to the put_chip() concurrency logic.

Problem was occurring when:

1) one thread was erasing a block in partition x;
2) another thread suspended the erase in order to write to
   partition y;
3) a third thread came along to read a different block from
   partition x and, when it called put_chip(), chip->oldstate was
   FL_ERASING and the erase (mistakenly) resumed;
4) the write in partition y obviously failed at that point.

Incidentally, the fix for this problem also fixed the case where
suspending writes for MTD XIP usage was not working properly.
Signed-off-by: default avatarNicolas Pitre <nico@cam.org>
Signed-off-by: default avatarDavid Woodhouse <dwmw2@infradead.org>
Signed-off-by: default avatarLinus Torvalds <torvalds@osdl.org>
parent 68f12028
...@@ -36,10 +36,7 @@ ...@@ -36,10 +36,7 @@
#include <linux/mtd/cfi.h> #include <linux/mtd/cfi.h>
/* #define CMDSET0001_DISABLE_ERASE_SUSPEND_ON_WRITE */ /* #define CMDSET0001_DISABLE_ERASE_SUSPEND_ON_WRITE */
/* #define CMDSET0001_DISABLE_WRITE_SUSPEND */
#ifdef CONFIG_MTD_XIP
#define CMDSET0001_DISABLE_WRITE_SUSPEND
#endif
// debugging, turns off buffer write mode if set to 1 // debugging, turns off buffer write mode if set to 1
#define FORCE_WORD_WRITE 0 #define FORCE_WORD_WRITE 0
...@@ -152,7 +149,6 @@ static void fixup_intel_strataflash(struct mtd_info *mtd, void* param) ...@@ -152,7 +149,6 @@ static void fixup_intel_strataflash(struct mtd_info *mtd, void* param)
#endif #endif
#ifdef CMDSET0001_DISABLE_WRITE_SUSPEND #ifdef CMDSET0001_DISABLE_WRITE_SUSPEND
/* The XIP config appears to have problems using write suspend at the moment */
static void fixup_no_write_suspend(struct mtd_info *mtd, void* param) static void fixup_no_write_suspend(struct mtd_info *mtd, void* param)
{ {
struct map_info *map = mtd->priv; struct map_info *map = mtd->priv;
...@@ -733,7 +729,7 @@ static void put_chip(struct map_info *map, struct flchip *chip, unsigned long ad ...@@ -733,7 +729,7 @@ static void put_chip(struct map_info *map, struct flchip *chip, unsigned long ad
if (chip->priv) { if (chip->priv) {
struct flchip_shared *shared = chip->priv; struct flchip_shared *shared = chip->priv;
spin_lock(&shared->lock); spin_lock(&shared->lock);
if (shared->writing == chip) { if (shared->writing == chip && chip->oldstate == FL_READY) {
/* We own the ability to write, but we're done */ /* We own the ability to write, but we're done */
shared->writing = shared->erasing; shared->writing = shared->erasing;
if (shared->writing && shared->writing != chip) { if (shared->writing && shared->writing != chip) {
...@@ -745,18 +741,25 @@ static void put_chip(struct map_info *map, struct flchip *chip, unsigned long ad ...@@ -745,18 +741,25 @@ static void put_chip(struct map_info *map, struct flchip *chip, unsigned long ad
put_chip(map, loaner, loaner->start); put_chip(map, loaner, loaner->start);
spin_lock(chip->mutex); spin_lock(chip->mutex);
spin_unlock(loaner->mutex); spin_unlock(loaner->mutex);
} else { wake_up(&chip->wq);
if (chip->oldstate != FL_ERASING) { return;
}
shared->erasing = NULL; shared->erasing = NULL;
if (chip->oldstate != FL_WRITING)
shared->writing = NULL; shared->writing = NULL;
} } else if (shared->erasing == chip && shared->writing != chip) {
/*
* We own the ability to erase without the ability
* to write, which means the erase was suspended
* and some other partition is currently writing.
* Don't let the switch below mess things up since
* we don't have ownership to resume anything.
*/
spin_unlock(&shared->lock); spin_unlock(&shared->lock);
wake_up(&chip->wq);
return;
} }
} else {
spin_unlock(&shared->lock); spin_unlock(&shared->lock);
} }
}
switch(chip->oldstate) { switch(chip->oldstate) {
case FL_ERASING: case FL_ERASING:
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment