mirror of
https://github.com/edk2-porting/linux-next.git
synced 2024-11-20 00:26:39 +08:00
- Fix a regression in 16-bit Atmel NAND flash which was introduced in 3.1
- Fix breakage with MTD suspend caused by the API rework - Fix a problem with resetting the MX28 BCH module - A couple of other trivial fixes -----BEGIN PGP SIGNATURE----- Version: GnuPG v1.4.11 (GNU/Linux) iEYEABECAAYFAk8s6HsACgkQdwG7hYl686MIiACgxpNoUWFvq8z+2UGXxsLnNrio hhcAn31H7TY3KUuIQBo4CqG2dEjNwpCw =DRWp -----END PGP SIGNATURE----- Merge tag 'for-linus-3.3' of git://git.infradead.org/~dwmw2/mtd-3.3 - Fix a regression in 16-bit Atmel NAND flash which was introduced in 3.1 - Fix breakage with MTD suspend caused by the API rework - Fix a problem with resetting the MX28 BCH module - A couple of other trivial fixes * tag 'for-linus-3.3-20120204' of git://git.infradead.org/~dwmw2/mtd-3.3: Revert "mtd: atmel_nand: optimize read/write buffer functions" mtd: fix MTD suspend jffs2: do not initialize variable unnecessarily mtd: gpmi-nand bugfix: reset the BCH module when it is not MX23 mtd: nand: fix typo in comment
This commit is contained in:
commit
71b1b20b8a
@ -119,7 +119,7 @@ static int mtd_cls_suspend(struct device *dev, pm_message_t state)
|
|||||||
{
|
{
|
||||||
struct mtd_info *mtd = dev_get_drvdata(dev);
|
struct mtd_info *mtd = dev_get_drvdata(dev);
|
||||||
|
|
||||||
return mtd_suspend(mtd);
|
return mtd ? mtd_suspend(mtd) : 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
static int mtd_cls_resume(struct device *dev)
|
static int mtd_cls_resume(struct device *dev)
|
||||||
|
@ -161,6 +161,37 @@ static int atmel_nand_device_ready(struct mtd_info *mtd)
|
|||||||
!!host->board->rdy_pin_active_low;
|
!!host->board->rdy_pin_active_low;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Minimal-overhead PIO for data access.
|
||||||
|
*/
|
||||||
|
static void atmel_read_buf8(struct mtd_info *mtd, u8 *buf, int len)
|
||||||
|
{
|
||||||
|
struct nand_chip *nand_chip = mtd->priv;
|
||||||
|
|
||||||
|
__raw_readsb(nand_chip->IO_ADDR_R, buf, len);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void atmel_read_buf16(struct mtd_info *mtd, u8 *buf, int len)
|
||||||
|
{
|
||||||
|
struct nand_chip *nand_chip = mtd->priv;
|
||||||
|
|
||||||
|
__raw_readsw(nand_chip->IO_ADDR_R, buf, len / 2);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void atmel_write_buf8(struct mtd_info *mtd, const u8 *buf, int len)
|
||||||
|
{
|
||||||
|
struct nand_chip *nand_chip = mtd->priv;
|
||||||
|
|
||||||
|
__raw_writesb(nand_chip->IO_ADDR_W, buf, len);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void atmel_write_buf16(struct mtd_info *mtd, const u8 *buf, int len)
|
||||||
|
{
|
||||||
|
struct nand_chip *nand_chip = mtd->priv;
|
||||||
|
|
||||||
|
__raw_writesw(nand_chip->IO_ADDR_W, buf, len / 2);
|
||||||
|
}
|
||||||
|
|
||||||
static void dma_complete_func(void *completion)
|
static void dma_complete_func(void *completion)
|
||||||
{
|
{
|
||||||
complete(completion);
|
complete(completion);
|
||||||
@ -235,27 +266,33 @@ err_buf:
|
|||||||
static void atmel_read_buf(struct mtd_info *mtd, u8 *buf, int len)
|
static void atmel_read_buf(struct mtd_info *mtd, u8 *buf, int len)
|
||||||
{
|
{
|
||||||
struct nand_chip *chip = mtd->priv;
|
struct nand_chip *chip = mtd->priv;
|
||||||
|
struct atmel_nand_host *host = chip->priv;
|
||||||
|
|
||||||
if (use_dma && len > mtd->oobsize)
|
if (use_dma && len > mtd->oobsize)
|
||||||
/* only use DMA for bigger than oob size: better performances */
|
/* only use DMA for bigger than oob size: better performances */
|
||||||
if (atmel_nand_dma_op(mtd, buf, len, 1) == 0)
|
if (atmel_nand_dma_op(mtd, buf, len, 1) == 0)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
/* if no DMA operation possible, use PIO */
|
if (host->board->bus_width_16)
|
||||||
memcpy_fromio(buf, chip->IO_ADDR_R, len);
|
atmel_read_buf16(mtd, buf, len);
|
||||||
|
else
|
||||||
|
atmel_read_buf8(mtd, buf, len);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void atmel_write_buf(struct mtd_info *mtd, const u8 *buf, int len)
|
static void atmel_write_buf(struct mtd_info *mtd, const u8 *buf, int len)
|
||||||
{
|
{
|
||||||
struct nand_chip *chip = mtd->priv;
|
struct nand_chip *chip = mtd->priv;
|
||||||
|
struct atmel_nand_host *host = chip->priv;
|
||||||
|
|
||||||
if (use_dma && len > mtd->oobsize)
|
if (use_dma && len > mtd->oobsize)
|
||||||
/* only use DMA for bigger than oob size: better performances */
|
/* only use DMA for bigger than oob size: better performances */
|
||||||
if (atmel_nand_dma_op(mtd, (void *)buf, len, 0) == 0)
|
if (atmel_nand_dma_op(mtd, (void *)buf, len, 0) == 0)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
/* if no DMA operation possible, use PIO */
|
if (host->board->bus_width_16)
|
||||||
memcpy_toio(chip->IO_ADDR_W, buf, len);
|
atmel_write_buf16(mtd, buf, len);
|
||||||
|
else
|
||||||
|
atmel_write_buf8(mtd, buf, len);
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
@ -69,17 +69,19 @@ static int clear_poll_bit(void __iomem *addr, u32 mask)
|
|||||||
* [1] enable the module.
|
* [1] enable the module.
|
||||||
* [2] reset the module.
|
* [2] reset the module.
|
||||||
*
|
*
|
||||||
* In most of the cases, it's ok. But there is a hardware bug in the BCH block.
|
* In most of the cases, it's ok.
|
||||||
|
* But in MX23, there is a hardware bug in the BCH block (see erratum #2847).
|
||||||
* If you try to soft reset the BCH block, it becomes unusable until
|
* If you try to soft reset the BCH block, it becomes unusable until
|
||||||
* the next hard reset. This case occurs in the NAND boot mode. When the board
|
* the next hard reset. This case occurs in the NAND boot mode. When the board
|
||||||
* boots by NAND, the ROM of the chip will initialize the BCH blocks itself.
|
* boots by NAND, the ROM of the chip will initialize the BCH blocks itself.
|
||||||
* So If the driver tries to reset the BCH again, the BCH will not work anymore.
|
* So If the driver tries to reset the BCH again, the BCH will not work anymore.
|
||||||
* You will see a DMA timeout in this case.
|
* You will see a DMA timeout in this case. The bug has been fixed
|
||||||
|
* in the following chips, such as MX28.
|
||||||
*
|
*
|
||||||
* To avoid this bug, just add a new parameter `just_enable` for
|
* To avoid this bug, just add a new parameter `just_enable` for
|
||||||
* the mxs_reset_block(), and rewrite it here.
|
* the mxs_reset_block(), and rewrite it here.
|
||||||
*/
|
*/
|
||||||
int gpmi_reset_block(void __iomem *reset_addr, bool just_enable)
|
static int gpmi_reset_block(void __iomem *reset_addr, bool just_enable)
|
||||||
{
|
{
|
||||||
int ret;
|
int ret;
|
||||||
int timeout = 0x400;
|
int timeout = 0x400;
|
||||||
@ -206,7 +208,15 @@ int bch_set_geometry(struct gpmi_nand_data *this)
|
|||||||
if (ret)
|
if (ret)
|
||||||
goto err_out;
|
goto err_out;
|
||||||
|
|
||||||
ret = gpmi_reset_block(r->bch_regs, true);
|
/*
|
||||||
|
* Due to erratum #2847 of the MX23, the BCH cannot be soft reset on this
|
||||||
|
* chip, otherwise it will lock up. So we skip resetting BCH on the MX23.
|
||||||
|
* On the other hand, the MX28 needs the reset, because one case has been
|
||||||
|
* seen where the BCH produced ECC errors constantly after 10000
|
||||||
|
* consecutive reboots. The latter case has not been seen on the MX23 yet,
|
||||||
|
* still we don't know if it could happen there as well.
|
||||||
|
*/
|
||||||
|
ret = gpmi_reset_block(r->bch_regs, GPMI_IS_MX23(this));
|
||||||
if (ret)
|
if (ret)
|
||||||
goto err_out;
|
goto err_out;
|
||||||
|
|
||||||
|
@ -2588,7 +2588,7 @@ int nand_erase_nand(struct mtd_info *mtd, struct erase_info *instr,
|
|||||||
instr->state = MTD_ERASING;
|
instr->state = MTD_ERASING;
|
||||||
|
|
||||||
while (len) {
|
while (len) {
|
||||||
/* Heck if we have a bad block, we do not erase bad blocks! */
|
/* Check if we have a bad block, we do not erase bad blocks! */
|
||||||
if (nand_block_checkbad(mtd, ((loff_t) page) <<
|
if (nand_block_checkbad(mtd, ((loff_t) page) <<
|
||||||
chip->page_shift, 0, allowbbt)) {
|
chip->page_shift, 0, allowbbt)) {
|
||||||
pr_warn("%s: attempt to erase a bad block at page 0x%08x\n",
|
pr_warn("%s: attempt to erase a bad block at page 0x%08x\n",
|
||||||
|
@ -335,7 +335,7 @@ static int jffs2_block_check_erase(struct jffs2_sb_info *c, struct jffs2_erasebl
|
|||||||
void *ebuf;
|
void *ebuf;
|
||||||
uint32_t ofs;
|
uint32_t ofs;
|
||||||
size_t retlen;
|
size_t retlen;
|
||||||
int ret = -EIO;
|
int ret;
|
||||||
unsigned long *wordebuf;
|
unsigned long *wordebuf;
|
||||||
|
|
||||||
ret = mtd_point(c->mtd, jeb->offset, c->sector_size, &retlen,
|
ret = mtd_point(c->mtd, jeb->offset, c->sector_size, &retlen,
|
||||||
|
@ -427,9 +427,7 @@ static inline int mtd_is_locked(struct mtd_info *mtd, loff_t ofs, uint64_t len)
|
|||||||
|
|
||||||
static inline int mtd_suspend(struct mtd_info *mtd)
|
static inline int mtd_suspend(struct mtd_info *mtd)
|
||||||
{
|
{
|
||||||
if (!mtd->suspend)
|
return mtd->suspend ? mtd->suspend(mtd) : 0;
|
||||||
return -EOPNOTSUPP;
|
|
||||||
return mtd->suspend(mtd);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline void mtd_resume(struct mtd_info *mtd)
|
static inline void mtd_resume(struct mtd_info *mtd)
|
||||||
|
Loading…
Reference in New Issue
Block a user