diff options
author | Kishore Kadiyala <kishore.kadiyala@ti.com> | 2011-05-11 11:47:27 -0400 |
---|---|---|
committer | David Woodhouse <David.Woodhouse@intel.com> | 2011-05-24 21:03:37 -0400 |
commit | c5d8c0cae4af7d78823d32fcd1c458ee1a1b5489 (patch) | |
tree | 57037c57d2e422028f0a13c32149c83b59c6a147 /drivers/mtd/nand | |
parent | 61cc8276fa776ced06b0e5b67b57e12c4997d388 (diff) |
mtd: omap: fix subpage ecc issue with prefetch
When reading/writing a subpage (When HW ECC is not available/enabled)
for number of bytes not aligned to 4, the mis-aligned bytes are handled
first (by cpu copy method) before enabling the Prefetch engine to/from
'p'(start of buffer 'buf'). Then it reads/writes rest of the bytes with
the help of Prefetch engine, if available, or again using cpu copy method.
Currently, reading/writing of rest of bytes, is not done correctly since
its trying to read/write again to/from begining of buffer 'buf',
overwriting the mis-aligned bytes.
Read & write using prefetch engine got broken in commit '2c01946c'.
We never hit a scenario of not getting 'gpmc_prefetch_enable' call
success. So, problem did not get caught up.
Signed-off-by: Kishore Kadiyala <kishore.kadiyala@ti.com>
Signed-off-by: Vimal Singh <vimal.newwork@gmail.com>
Reported-by: Bryan DE FARIA <bdefaria@adeneo-embedded.com>
Cc: stable@kernel.org [2.6.35+]
Signed-off-by: Artem Bityutskiy <Artem.Bityutskiy@nokia.com>
Signed-off-by: David Woodhouse <David.Woodhouse@intel.com>
Diffstat (limited to 'drivers/mtd/nand')
-rw-r--r-- | drivers/mtd/nand/omap2.c | 12 |
1 files changed, 5 insertions, 7 deletions
diff --git a/drivers/mtd/nand/omap2.c b/drivers/mtd/nand/omap2.c index 288423f7b7b1..39f46a0cf34e 100644 --- a/drivers/mtd/nand/omap2.c +++ b/drivers/mtd/nand/omap2.c | |||
@@ -263,11 +263,10 @@ static void omap_read_buf_pref(struct mtd_info *mtd, u_char *buf, int len) | |||
263 | if (ret) { | 263 | if (ret) { |
264 | /* PFPW engine is busy, use cpu copy method */ | 264 | /* PFPW engine is busy, use cpu copy method */ |
265 | if (info->nand.options & NAND_BUSWIDTH_16) | 265 | if (info->nand.options & NAND_BUSWIDTH_16) |
266 | omap_read_buf16(mtd, buf, len); | 266 | omap_read_buf16(mtd, (u_char *)p, len); |
267 | else | 267 | else |
268 | omap_read_buf8(mtd, buf, len); | 268 | omap_read_buf8(mtd, (u_char *)p, len); |
269 | } else { | 269 | } else { |
270 | p = (u32 *) buf; | ||
271 | do { | 270 | do { |
272 | r_count = gpmc_read_status(GPMC_PREFETCH_FIFO_CNT); | 271 | r_count = gpmc_read_status(GPMC_PREFETCH_FIFO_CNT); |
273 | r_count = r_count >> 2; | 272 | r_count = r_count >> 2; |
@@ -293,7 +292,7 @@ static void omap_write_buf_pref(struct mtd_info *mtd, | |||
293 | struct omap_nand_info, mtd); | 292 | struct omap_nand_info, mtd); |
294 | uint32_t w_count = 0; | 293 | uint32_t w_count = 0; |
295 | int i = 0, ret = 0; | 294 | int i = 0, ret = 0; |
296 | u16 *p; | 295 | u16 *p = (u16 *)buf; |
297 | unsigned long tim, limit; | 296 | unsigned long tim, limit; |
298 | 297 | ||
299 | /* take care of subpage writes */ | 298 | /* take care of subpage writes */ |
@@ -309,11 +308,10 @@ static void omap_write_buf_pref(struct mtd_info *mtd, | |||
309 | if (ret) { | 308 | if (ret) { |
310 | /* PFPW engine is busy, use cpu copy method */ | 309 | /* PFPW engine is busy, use cpu copy method */ |
311 | if (info->nand.options & NAND_BUSWIDTH_16) | 310 | if (info->nand.options & NAND_BUSWIDTH_16) |
312 | omap_write_buf16(mtd, buf, len); | 311 | omap_write_buf16(mtd, (u_char *)p, len); |
313 | else | 312 | else |
314 | omap_write_buf8(mtd, buf, len); | 313 | omap_write_buf8(mtd, (u_char *)p, len); |
315 | } else { | 314 | } else { |
316 | p = (u16 *) buf; | ||
317 | while (len) { | 315 | while (len) { |
318 | w_count = gpmc_read_status(GPMC_PREFETCH_FIFO_CNT); | 316 | w_count = gpmc_read_status(GPMC_PREFETCH_FIFO_CNT); |
319 | w_count = w_count >> 1; | 317 | w_count = w_count >> 1; |