Commit 17ad78de authored by Andrew Ruder's avatar Andrew Ruder Committed by David S. Miller

dm9000: clean up edge-triggered irq compatibility

DM9000 uses level-triggered interrupts.  Some systems (PXA270) only
support edge-triggered interrupts on GPIOs.  Some changes are necessary
to ensure that interrupts are not triggered while the GPIO interrupt is
masked or we will miss the interrupt forever.

* Make some helper functions called dm9000_mask_interrupts() and
  dm9000_unmask_interrupts() for readability.

* dm9000_init_dm9000(): ensure that this function always leaves interrupts
  masked regardless of the state when it entered the function.  This is
  primarily to support the situation in dm9000_open where the logic used
  to go:

    dm9000_open()
        dm9000_init_dm9000()
            unmask interrupts
        request_irq()

  If an interrupt occurred between unmasking the interrupt and
  requesting the irq, it would be missed forever as the edge event would
  never be seen by the GPIO hardware in the PXA270.  This allows us to
  change the logic to:

    dm9000_open()
        dm9000_init_dm9000()
            dm9000_mask_interrupts()
        request_irq()
        dm9000_unmask_interrupts()

* dm9000_timeout(), dm9000_drv_resume(): Add the missing
  dm9000_unmask_interrupts() now required by the change above.

* dm9000_shutdown(): Use mask helper function

* dm9000_interrupt(): Use mask/unmask helper functions
Signed-off-by: default avatarAndrew Ruder <andrew.ruder@elecsyscorp.com>
Signed-off-by: default avatarDavid S. Miller <davem@davemloft.net>
parent 751bb6fd
...@@ -882,6 +882,18 @@ dm9000_hash_table(struct net_device *dev) ...@@ -882,6 +882,18 @@ dm9000_hash_table(struct net_device *dev)
spin_unlock_irqrestore(&db->lock, flags); spin_unlock_irqrestore(&db->lock, flags);
} }
static void
dm9000_mask_interrupts(board_info_t *db)
{
iow(db, DM9000_IMR, IMR_PAR);
}
static void
dm9000_unmask_interrupts(board_info_t *db)
{
iow(db, DM9000_IMR, db->imr_all);
}
/* /*
* Initialize dm9000 board * Initialize dm9000 board
*/ */
...@@ -895,6 +907,7 @@ dm9000_init_dm9000(struct net_device *dev) ...@@ -895,6 +907,7 @@ dm9000_init_dm9000(struct net_device *dev)
dm9000_dbg(db, 1, "entering %s\n", __func__); dm9000_dbg(db, 1, "entering %s\n", __func__);
dm9000_reset(db); dm9000_reset(db);
dm9000_mask_interrupts(db);
/* I/O mode */ /* I/O mode */
db->io_mode = ior(db, DM9000_ISR) >> 6; /* ISR bit7:6 keeps I/O mode */ db->io_mode = ior(db, DM9000_ISR) >> 6; /* ISR bit7:6 keeps I/O mode */
...@@ -943,9 +956,6 @@ dm9000_init_dm9000(struct net_device *dev) ...@@ -943,9 +956,6 @@ dm9000_init_dm9000(struct net_device *dev)
db->imr_all = imr; db->imr_all = imr;
/* Enable TX/RX interrupt mask */
iow(db, DM9000_IMR, imr);
/* Init Driver variable */ /* Init Driver variable */
db->tx_pkt_cnt = 0; db->tx_pkt_cnt = 0;
db->queue_pkt_len = 0; db->queue_pkt_len = 0;
...@@ -965,6 +975,7 @@ static void dm9000_timeout(struct net_device *dev) ...@@ -965,6 +975,7 @@ static void dm9000_timeout(struct net_device *dev)
netif_stop_queue(dev); netif_stop_queue(dev);
dm9000_init_dm9000(dev); dm9000_init_dm9000(dev);
dm9000_unmask_interrupts(db);
/* We can accept TX packets again */ /* We can accept TX packets again */
dev->trans_start = jiffies; /* prevent tx timeout */ dev->trans_start = jiffies; /* prevent tx timeout */
netif_wake_queue(dev); netif_wake_queue(dev);
...@@ -1194,9 +1205,7 @@ static irqreturn_t dm9000_interrupt(int irq, void *dev_id) ...@@ -1194,9 +1205,7 @@ static irqreturn_t dm9000_interrupt(int irq, void *dev_id)
/* Save previous register address */ /* Save previous register address */
reg_save = readb(db->io_addr); reg_save = readb(db->io_addr);
/* Disable all interrupts */ dm9000_mask_interrupts(db);
iow(db, DM9000_IMR, IMR_PAR);
/* Got DM9000 interrupt status */ /* Got DM9000 interrupt status */
int_status = ior(db, DM9000_ISR); /* Got ISR */ int_status = ior(db, DM9000_ISR); /* Got ISR */
iow(db, DM9000_ISR, int_status); /* Clear ISR status */ iow(db, DM9000_ISR, int_status); /* Clear ISR status */
...@@ -1219,9 +1228,7 @@ static irqreturn_t dm9000_interrupt(int irq, void *dev_id) ...@@ -1219,9 +1228,7 @@ static irqreturn_t dm9000_interrupt(int irq, void *dev_id)
} }
} }
/* Re-enable interrupt mask */ dm9000_unmask_interrupts(db);
iow(db, DM9000_IMR, db->imr_all);
/* Restore previous register address */ /* Restore previous register address */
writeb(reg_save, db->io_addr); writeb(reg_save, db->io_addr);
...@@ -1309,6 +1316,10 @@ dm9000_open(struct net_device *dev) ...@@ -1309,6 +1316,10 @@ dm9000_open(struct net_device *dev)
if (request_irq(dev->irq, dm9000_interrupt, irqflags, dev->name, dev)) if (request_irq(dev->irq, dm9000_interrupt, irqflags, dev->name, dev))
return -EAGAIN; return -EAGAIN;
/* Now that we have an interrupt handler hooked up we can unmask
* our interrupts
*/
dm9000_unmask_interrupts(db);
/* Init driver variable */ /* Init driver variable */
db->dbug_cnt = 0; db->dbug_cnt = 0;
...@@ -1329,7 +1340,7 @@ dm9000_shutdown(struct net_device *dev) ...@@ -1329,7 +1340,7 @@ dm9000_shutdown(struct net_device *dev)
/* RESET device */ /* RESET device */
dm9000_phy_write(dev, 0, MII_BMCR, BMCR_RESET); /* PHY RESET */ dm9000_phy_write(dev, 0, MII_BMCR, BMCR_RESET); /* PHY RESET */
iow(db, DM9000_GPR, 0x01); /* Power-Down PHY */ iow(db, DM9000_GPR, 0x01); /* Power-Down PHY */
iow(db, DM9000_IMR, IMR_PAR); /* Disable all interrupt */ dm9000_mask_interrupts(db);
iow(db, DM9000_RCR, 0x00); /* Disable RX */ iow(db, DM9000_RCR, 0x00); /* Disable RX */
} }
...@@ -1694,6 +1705,7 @@ dm9000_drv_resume(struct device *dev) ...@@ -1694,6 +1705,7 @@ dm9000_drv_resume(struct device *dev)
* the device was powered off it is in a known state */ * the device was powered off it is in a known state */
if (!db->wake_state) { if (!db->wake_state) {
dm9000_init_dm9000(ndev); dm9000_init_dm9000(ndev);
dm9000_unmask_interrupts(db);
} }
netif_device_attach(ndev); netif_device_attach(ndev);
......
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