Commit 9489be8c authored by Maxim Levitsky's avatar Maxim Levitsky Committed by David Woodhouse

mtd/r852: Fixes in case of DMA timeout

* Don't call complete on dma completion
* do a INIT_COMPLETE before using it each time
* Report DMA read error via ecc 'correct'

I finally managed to make my system do suspend to ram propertly, and I see that
if card was inserted during suspend (while system was off), I get dma timeouts
on resume. Simple card reinsert solves the issue.
This patch solves a crash that would happen otherwise
Signed-off-by: default avatarMaxim Levitsky <maximlevitsky@gmail.com>
Signed-off-by: default avatarDavid Woodhouse <David.Woodhouse@intel.com>
parent cc1fed00
...@@ -150,7 +150,6 @@ static void r852_dma_done(struct r852_device *dev, int error) ...@@ -150,7 +150,6 @@ static void r852_dma_done(struct r852_device *dev, int error)
if (dev->phys_dma_addr && dev->phys_dma_addr != dev->phys_bounce_buffer) if (dev->phys_dma_addr && dev->phys_dma_addr != dev->phys_bounce_buffer)
pci_unmap_single(dev->pci_dev, dev->phys_dma_addr, R852_DMA_LEN, pci_unmap_single(dev->pci_dev, dev->phys_dma_addr, R852_DMA_LEN,
dev->dma_dir ? PCI_DMA_FROMDEVICE : PCI_DMA_TODEVICE); dev->dma_dir ? PCI_DMA_FROMDEVICE : PCI_DMA_TODEVICE);
complete(&dev->dma_done);
} }
/* /*
...@@ -182,6 +181,7 @@ static void r852_do_dma(struct r852_device *dev, uint8_t *buf, int do_read) ...@@ -182,6 +181,7 @@ static void r852_do_dma(struct r852_device *dev, uint8_t *buf, int do_read)
/* Set dma direction */ /* Set dma direction */
dev->dma_dir = do_read; dev->dma_dir = do_read;
dev->dma_stage = 1; dev->dma_stage = 1;
INIT_COMPLETION(dev->dma_done);
dbg_verbose("doing dma %s ", do_read ? "read" : "write"); dbg_verbose("doing dma %s ", do_read ? "read" : "write");
...@@ -494,6 +494,11 @@ int r852_ecc_correct(struct mtd_info *mtd, uint8_t *dat, ...@@ -494,6 +494,11 @@ int r852_ecc_correct(struct mtd_info *mtd, uint8_t *dat,
if (dev->card_unstable) if (dev->card_unstable)
return 0; return 0;
if (dev->dma_error) {
dev->dma_error = 0;
return -1;
}
r852_write_reg(dev, R852_CTL, dev->ctlreg | R852_CTL_ECC_ACCESS); r852_write_reg(dev, R852_CTL, dev->ctlreg | R852_CTL_ECC_ACCESS);
ecc_reg = r852_read_reg_dword(dev, R852_DATALINE); ecc_reg = r852_read_reg_dword(dev, R852_DATALINE);
r852_write_reg(dev, R852_CTL, dev->ctlreg); r852_write_reg(dev, R852_CTL, dev->ctlreg);
...@@ -796,6 +801,7 @@ static irqreturn_t r852_irq(int irq, void *data) ...@@ -796,6 +801,7 @@ static irqreturn_t r852_irq(int irq, void *data)
if (dma_status & R852_DMA_IRQ_ERROR) { if (dma_status & R852_DMA_IRQ_ERROR) {
dbg("recieved dma error IRQ"); dbg("recieved dma error IRQ");
r852_dma_done(dev, -EIO); r852_dma_done(dev, -EIO);
complete(&dev->dma_done);
goto out; goto out;
} }
...@@ -825,8 +831,10 @@ static irqreturn_t r852_irq(int irq, void *data) ...@@ -825,8 +831,10 @@ static irqreturn_t r852_irq(int irq, void *data)
r852_dma_enable(dev); r852_dma_enable(dev);
/* Operation done */ /* Operation done */
if (dev->dma_stage == 3) if (dev->dma_stage == 3) {
r852_dma_done(dev, 0); r852_dma_done(dev, 0);
complete(&dev->dma_done);
}
goto out; goto out;
} }
...@@ -1082,7 +1090,7 @@ int r852_resume(struct device *device) ...@@ -1082,7 +1090,7 @@ int r852_resume(struct device *device)
dev->card_detected ? "added" : "removed"); dev->card_detected ? "added" : "removed");
queue_delayed_work(dev->card_workqueue, queue_delayed_work(dev->card_workqueue,
&dev->card_detect_work, 1000); &dev->card_detect_work, msecs_to_jiffies(1000));
return 0; return 0;
} }
......
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