diff options
| author | Linus Torvalds <torvalds@linux-foundation.org> | 2009-05-08 20:00:41 -0400 |
|---|---|---|
| committer | Linus Torvalds <torvalds@linux-foundation.org> | 2009-05-08 20:00:41 -0400 |
| commit | 621c2559c16fc703e24fc0e3ffad28d5477b49c7 (patch) | |
| tree | cc46b4501f53423e81572d1c47174a5fd0afa1bf | |
| parent | d9d05fda919cb6414ae4889b696b2fada7a23217 (diff) | |
| parent | cd1a6de7d4a492bf3405a6c070075a4cb8c90262 (diff) | |
Merge git://git.infradead.org/mtd-2.6
* git://git.infradead.org/mtd-2.6:
mtd: fix timeout in M25P80 driver
mtd: Bug in m25p80.c during whole-chip erase
mtd: expose subpage size via sysfs
mtd: mtd in mtd_release is unused without CONFIG_MTD_CHAR
| -rw-r--r-- | drivers/mtd/devices/m25p80.c | 26 | ||||
| -rw-r--r-- | drivers/mtd/mtdcore.c | 18 |
2 files changed, 29 insertions, 15 deletions
diff --git a/drivers/mtd/devices/m25p80.c b/drivers/mtd/devices/m25p80.c index 8185b1f3e5e6..cc6369ea67dd 100644 --- a/drivers/mtd/devices/m25p80.c +++ b/drivers/mtd/devices/m25p80.c | |||
| @@ -54,7 +54,7 @@ | |||
| 54 | #define SR_SRWD 0x80 /* SR write protect */ | 54 | #define SR_SRWD 0x80 /* SR write protect */ |
| 55 | 55 | ||
| 56 | /* Define max times to check status register before we give up. */ | 56 | /* Define max times to check status register before we give up. */ |
| 57 | #define MAX_READY_WAIT_COUNT 100000 | 57 | #define MAX_READY_WAIT_JIFFIES (10 * HZ) /* eg. M25P128 specs 6s max sector erase */ |
| 58 | #define CMD_SIZE 4 | 58 | #define CMD_SIZE 4 |
| 59 | 59 | ||
| 60 | #ifdef CONFIG_M25PXX_USE_FAST_READ | 60 | #ifdef CONFIG_M25PXX_USE_FAST_READ |
| @@ -139,20 +139,20 @@ static inline int write_enable(struct m25p *flash) | |||
| 139 | */ | 139 | */ |
| 140 | static int wait_till_ready(struct m25p *flash) | 140 | static int wait_till_ready(struct m25p *flash) |
| 141 | { | 141 | { |
| 142 | int count; | 142 | unsigned long deadline; |
| 143 | int sr; | 143 | int sr; |
| 144 | 144 | ||
| 145 | /* one chip guarantees max 5 msec wait here after page writes, | 145 | deadline = jiffies + MAX_READY_WAIT_JIFFIES; |
| 146 | * but potentially three seconds (!) after page erase. | 146 | |
| 147 | */ | 147 | do { |
| 148 | for (count = 0; count < MAX_READY_WAIT_COUNT; count++) { | ||
| 149 | if ((sr = read_sr(flash)) < 0) | 148 | if ((sr = read_sr(flash)) < 0) |
| 150 | break; | 149 | break; |
| 151 | else if (!(sr & SR_WIP)) | 150 | else if (!(sr & SR_WIP)) |
| 152 | return 0; | 151 | return 0; |
| 153 | 152 | ||
| 154 | /* REVISIT sometimes sleeping would be best */ | 153 | cond_resched(); |
| 155 | } | 154 | |
| 155 | } while (!time_after_eq(jiffies, deadline)); | ||
| 156 | 156 | ||
| 157 | return 1; | 157 | return 1; |
| 158 | } | 158 | } |
| @@ -246,10 +246,12 @@ static int m25p80_erase(struct mtd_info *mtd, struct erase_info *instr) | |||
| 246 | mutex_lock(&flash->lock); | 246 | mutex_lock(&flash->lock); |
| 247 | 247 | ||
| 248 | /* whole-chip erase? */ | 248 | /* whole-chip erase? */ |
| 249 | if (len == flash->mtd.size && erase_chip(flash)) { | 249 | if (len == flash->mtd.size) { |
| 250 | instr->state = MTD_ERASE_FAILED; | 250 | if (erase_chip(flash)) { |
| 251 | mutex_unlock(&flash->lock); | 251 | instr->state = MTD_ERASE_FAILED; |
| 252 | return -EIO; | 252 | mutex_unlock(&flash->lock); |
| 253 | return -EIO; | ||
| 254 | } | ||
| 253 | 255 | ||
| 254 | /* REVISIT in some cases we could speed up erasing large regions | 256 | /* REVISIT in some cases we could speed up erasing large regions |
| 255 | * by using OPCODE_SE instead of OPCODE_BE_4K. We may have set up | 257 | * by using OPCODE_SE instead of OPCODE_BE_4K. We may have set up |
diff --git a/drivers/mtd/mtdcore.c b/drivers/mtd/mtdcore.c index fdd6ae859397..bccb4b1ffc46 100644 --- a/drivers/mtd/mtdcore.c +++ b/drivers/mtd/mtdcore.c | |||
| @@ -48,11 +48,11 @@ static LIST_HEAD(mtd_notifiers); | |||
| 48 | */ | 48 | */ |
| 49 | static void mtd_release(struct device *dev) | 49 | static void mtd_release(struct device *dev) |
| 50 | { | 50 | { |
| 51 | struct mtd_info *mtd = dev_to_mtd(dev); | 51 | dev_t index = MTD_DEVT(dev_to_mtd(dev)->index); |
| 52 | 52 | ||
| 53 | /* remove /dev/mtdXro node if needed */ | 53 | /* remove /dev/mtdXro node if needed */ |
| 54 | if (MTD_DEVT(mtd->index)) | 54 | if (index) |
| 55 | device_destroy(mtd_class, MTD_DEVT(mtd->index) + 1); | 55 | device_destroy(mtd_class, index + 1); |
| 56 | } | 56 | } |
| 57 | 57 | ||
| 58 | static ssize_t mtd_type_show(struct device *dev, | 58 | static ssize_t mtd_type_show(struct device *dev, |
| @@ -132,6 +132,17 @@ static ssize_t mtd_writesize_show(struct device *dev, | |||
| 132 | } | 132 | } |
| 133 | static DEVICE_ATTR(writesize, S_IRUGO, mtd_writesize_show, NULL); | 133 | static DEVICE_ATTR(writesize, S_IRUGO, mtd_writesize_show, NULL); |
| 134 | 134 | ||
| 135 | static ssize_t mtd_subpagesize_show(struct device *dev, | ||
| 136 | struct device_attribute *attr, char *buf) | ||
| 137 | { | ||
| 138 | struct mtd_info *mtd = dev_to_mtd(dev); | ||
| 139 | unsigned int subpagesize = mtd->writesize >> mtd->subpage_sft; | ||
| 140 | |||
| 141 | return snprintf(buf, PAGE_SIZE, "%u\n", subpagesize); | ||
| 142 | |||
| 143 | } | ||
| 144 | static DEVICE_ATTR(subpagesize, S_IRUGO, mtd_subpagesize_show, NULL); | ||
| 145 | |||
| 135 | static ssize_t mtd_oobsize_show(struct device *dev, | 146 | static ssize_t mtd_oobsize_show(struct device *dev, |
| 136 | struct device_attribute *attr, char *buf) | 147 | struct device_attribute *attr, char *buf) |
| 137 | { | 148 | { |
| @@ -169,6 +180,7 @@ static struct attribute *mtd_attrs[] = { | |||
| 169 | &dev_attr_size.attr, | 180 | &dev_attr_size.attr, |
| 170 | &dev_attr_erasesize.attr, | 181 | &dev_attr_erasesize.attr, |
| 171 | &dev_attr_writesize.attr, | 182 | &dev_attr_writesize.attr, |
| 183 | &dev_attr_subpagesize.attr, | ||
| 172 | &dev_attr_oobsize.attr, | 184 | &dev_attr_oobsize.attr, |
| 173 | &dev_attr_numeraseregions.attr, | 185 | &dev_attr_numeraseregions.attr, |
| 174 | &dev_attr_name.attr, | 186 | &dev_attr_name.attr, |
