aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--drivers/mtd/onenand/onenand_base.c38
-rw-r--r--include/linux/mtd/onenand.h3
2 files changed, 41 insertions, 0 deletions
diff --git a/drivers/mtd/onenand/onenand_base.c b/drivers/mtd/onenand/onenand_base.c
index d6c13f7ae5a1..1439c9fa1d23 100644
--- a/drivers/mtd/onenand/onenand_base.c
+++ b/drivers/mtd/onenand/onenand_base.c
@@ -373,6 +373,17 @@ static int onenand_read_bufferram(struct mtd_info *mtd, int area,
373 373
374 bufferram += onenand_bufferram_offset(mtd, area); 374 bufferram += onenand_bufferram_offset(mtd, area);
375 375
376 if (ONENAND_CHECK_BYTE_ACCESS(count)) {
377 unsigned short word;
378
379 /* Align with word(16-bit) size */
380 count--;
381
382 /* Read word and save byte */
383 word = this->read_word(bufferram + offset + count);
384 buffer[count] = (word & 0xff);
385 }
386
376 memcpy(buffer, bufferram + offset, count); 387 memcpy(buffer, bufferram + offset, count);
377 388
378 return 0; 389 return 0;
@@ -400,6 +411,17 @@ static int onenand_sync_read_bufferram(struct mtd_info *mtd, int area,
400 411
401 this->mmcontrol(mtd, ONENAND_SYS_CFG1_SYNC_READ); 412 this->mmcontrol(mtd, ONENAND_SYS_CFG1_SYNC_READ);
402 413
414 if (ONENAND_CHECK_BYTE_ACCESS(count)) {
415 unsigned short word;
416
417 /* Align with word(16-bit) size */
418 count--;
419
420 /* Read word and save byte */
421 word = this->read_word(bufferram + offset + count);
422 buffer[count] = (word & 0xff);
423 }
424
403 memcpy(buffer, bufferram + offset, count); 425 memcpy(buffer, bufferram + offset, count);
404 426
405 this->mmcontrol(mtd, 0); 427 this->mmcontrol(mtd, 0);
@@ -427,6 +449,22 @@ static int onenand_write_bufferram(struct mtd_info *mtd, int area,
427 449
428 bufferram += onenand_bufferram_offset(mtd, area); 450 bufferram += onenand_bufferram_offset(mtd, area);
429 451
452 if (ONENAND_CHECK_BYTE_ACCESS(count)) {
453 unsigned short word;
454 int byte_offset;
455
456 /* Align with word(16-bit) size */
457 count--;
458
459 /* Calculate byte access offset */
460 byte_offset = offset + count;
461
462 /* Read word and save byte */
463 word = this->read_word(bufferram + byte_offset);
464 word = (word & ~0xff) | buffer[count];
465 this->write_word(word, bufferram + byte_offset);
466 }
467
430 memcpy(bufferram + offset, buffer, count); 468 memcpy(bufferram + offset, buffer, count);
431 469
432 return 0; 470 return 0;
diff --git a/include/linux/mtd/onenand.h b/include/linux/mtd/onenand.h
index 7419b5fab133..22322c8a7729 100644
--- a/include/linux/mtd/onenand.h
+++ b/include/linux/mtd/onenand.h
@@ -130,6 +130,9 @@ struct onenand_chip {
130#define ONENAND_SET_SYS_CFG1(v, this) \ 130#define ONENAND_SET_SYS_CFG1(v, this) \
131 (this->write_word(v, this->base + ONENAND_REG_SYS_CFG1)) 131 (this->write_word(v, this->base + ONENAND_REG_SYS_CFG1))
132 132
133/* Check byte access in OneNAND */
134#define ONENAND_CHECK_BYTE_ACCESS(addr) (addr & 0x1)
135
133/* 136/*
134 * Options bits 137 * Options bits
135 */ 138 */