aboutsummaryrefslogtreecommitdiffstats
path: root/drivers/net/skge.c
diff options
context:
space:
mode:
Diffstat (limited to 'drivers/net/skge.c')
-rw-r--r--drivers/net/skge.c56
1 files changed, 33 insertions, 23 deletions
diff --git a/drivers/net/skge.c b/drivers/net/skge.c
index f11e0e1a3d0e..290d6aa92383 100644
--- a/drivers/net/skge.c
+++ b/drivers/net/skge.c
@@ -97,10 +97,12 @@ static void genesis_mac_init(struct skge_hw *hw, int port);
97static void genesis_reset(struct skge_hw *hw, int port); 97static void genesis_reset(struct skge_hw *hw, int port);
98static void genesis_link_up(struct skge_port *skge); 98static void genesis_link_up(struct skge_port *skge);
99 99
100/* Avoid conditionals by using array */
100static const int txqaddr[] = { Q_XA1, Q_XA2 }; 101static const int txqaddr[] = { Q_XA1, Q_XA2 };
101static const int rxqaddr[] = { Q_R1, Q_R2 }; 102static const int rxqaddr[] = { Q_R1, Q_R2 };
102static const u32 rxirqmask[] = { IS_R1_F, IS_R2_F }; 103static const u32 rxirqmask[] = { IS_R1_F, IS_R2_F };
103static const u32 txirqmask[] = { IS_XA1_F, IS_XA2_F }; 104static const u32 txirqmask[] = { IS_XA1_F, IS_XA2_F };
105static const u32 portirqmask[] = { IS_PORT_1, IS_PORT_2 };
104 106
105/* Don't need to look at whole 16K. 107/* Don't need to look at whole 16K.
106 * last interesting register is descriptor poll timer. 108 * last interesting register is descriptor poll timer.
@@ -1382,7 +1384,9 @@ static void genesis_mac_intr(struct skge_hw *hw, int port)
1382 struct skge_port *skge = netdev_priv(hw->dev[port]); 1384 struct skge_port *skge = netdev_priv(hw->dev[port]);
1383 u16 status = xm_read16(hw, port, XM_ISRC); 1385 u16 status = xm_read16(hw, port, XM_ISRC);
1384 1386
1385 pr_debug("genesis_intr status %x\n", status); 1387 if (netif_msg_intr(skge))
1388 printk(KERN_DEBUG PFX "%s: mac interrupt status 0x%x\n",
1389 skge->netdev->name, status);
1386 1390
1387 if (status & XM_IS_TXF_UR) { 1391 if (status & XM_IS_TXF_UR) {
1388 xm_write32(hw, port, XM_MODE, XM_MD_FTF); 1392 xm_write32(hw, port, XM_MODE, XM_MD_FTF);
@@ -1446,6 +1450,7 @@ static void genesis_link_up(struct skge_port *skge)
1446 */ 1450 */
1447 if (skge->flow_control == FLOW_MODE_NONE || 1451 if (skge->flow_control == FLOW_MODE_NONE ||
1448 skge->flow_control == FLOW_MODE_LOC_SEND) 1452 skge->flow_control == FLOW_MODE_LOC_SEND)
1453 /* Disable Pause Frame Reception */
1449 cmd |= XM_MMU_IGN_PF; 1454 cmd |= XM_MMU_IGN_PF;
1450 else 1455 else
1451 /* Enable Pause Frame Reception */ 1456 /* Enable Pause Frame Reception */
@@ -1519,7 +1524,9 @@ static inline void bcom_phy_intr(struct skge_port *skge)
1519 u16 isrc; 1524 u16 isrc;
1520 1525
1521 isrc = xm_phy_read(hw, port, PHY_BCOM_INT_STAT); 1526 isrc = xm_phy_read(hw, port, PHY_BCOM_INT_STAT);
1522 pr_debug("bcom_phy_interrupt status=0x%x\n", isrc); 1527 if (netif_msg_intr(skge))
1528 printk(KERN_DEBUG PFX "%s: phy interrupt status 0x%x\n",
1529 skge->netdev->name, isrc);
1523 1530
1524 if (isrc & PHY_B_IS_PSE) 1531 if (isrc & PHY_B_IS_PSE)
1525 printk(KERN_ERR PFX "%s: uncorrectable pair swap error\n", 1532 printk(KERN_ERR PFX "%s: uncorrectable pair swap error\n",
@@ -1823,10 +1830,14 @@ static void yukon_get_stats(struct skge_port *skge, u64 *data)
1823 1830
1824static void yukon_mac_intr(struct skge_hw *hw, int port) 1831static void yukon_mac_intr(struct skge_hw *hw, int port)
1825{ 1832{
1826 struct skge_port *skge = netdev_priv(hw->dev[port]); 1833 struct net_device *dev = hw->dev[port];
1834 struct skge_port *skge = netdev_priv(dev);
1827 u8 status = skge_read8(hw, SK_REG(port, GMAC_IRQ_SRC)); 1835 u8 status = skge_read8(hw, SK_REG(port, GMAC_IRQ_SRC));
1828 1836
1829 pr_debug("yukon_intr status %x\n", status); 1837 if (netif_msg_intr(skge))
1838 printk(KERN_DEBUG PFX "%s: mac interrupt status 0x%x\n",
1839 dev->name, status);
1840
1830 if (status & GM_IS_RX_FF_OR) { 1841 if (status & GM_IS_RX_FF_OR) {
1831 ++skge->net_stats.rx_fifo_errors; 1842 ++skge->net_stats.rx_fifo_errors;
1832 gma_write8(hw, port, RX_GMF_CTRL_T, GMF_CLI_RX_FO); 1843 gma_write8(hw, port, RX_GMF_CTRL_T, GMF_CLI_RX_FO);
@@ -1908,7 +1919,10 @@ static void yukon_phy_intr(struct skge_port *skge)
1908 1919
1909 istatus = gm_phy_read(hw, port, PHY_MARV_INT_STAT); 1920 istatus = gm_phy_read(hw, port, PHY_MARV_INT_STAT);
1910 phystat = gm_phy_read(hw, port, PHY_MARV_PHY_STAT); 1921 phystat = gm_phy_read(hw, port, PHY_MARV_PHY_STAT);
1911 pr_debug("yukon phy intr istat=%x phy_stat=%x\n", istatus, phystat); 1922
1923 if (netif_msg_intr(skge))
1924 printk(KERN_DEBUG PFX "%s: phy interrupt status 0x%x 0x%x\n",
1925 skge->netdev->name, istatus, phystat);
1912 1926
1913 if (istatus & PHY_M_IS_AN_COMPL) { 1927 if (istatus & PHY_M_IS_AN_COMPL) {
1914 if (gm_phy_read(hw, port, PHY_MARV_AUNE_LP) 1928 if (gm_phy_read(hw, port, PHY_MARV_AUNE_LP)
@@ -2055,6 +2069,10 @@ static int skge_up(struct net_device *dev)
2055 2069
2056 skge->tx_avail = skge->tx_ring.count - 1; 2070 skge->tx_avail = skge->tx_ring.count - 1;
2057 2071
2072 /* Enable IRQ from port */
2073 hw->intr_mask |= portirqmask[port];
2074 skge_write32(hw, B0_IMSK, hw->intr_mask);
2075
2058 /* Initialze MAC */ 2076 /* Initialze MAC */
2059 if (hw->chip_id == CHIP_ID_GENESIS) 2077 if (hw->chip_id == CHIP_ID_GENESIS)
2060 genesis_mac_init(hw, port); 2078 genesis_mac_init(hw, port);
@@ -2449,7 +2467,8 @@ static int skge_poll(struct net_device *dev, int *budget)
2449 unsigned int to_do = min(dev->quota, *budget); 2467 unsigned int to_do = min(dev->quota, *budget);
2450 unsigned int work_done = 0; 2468 unsigned int work_done = 0;
2451 int done; 2469 int done;
2452 static const u32 irqmask[] = { IS_PORT_1, IS_PORT_2 }; 2470
2471 pr_debug("skge_poll\n");
2453 2472
2454 for (e = ring->to_clean; e != ring->to_use && work_done < to_do; 2473 for (e = ring->to_clean; e != ring->to_use && work_done < to_do;
2455 e = e->next) { 2474 e = e->next) {
@@ -2512,10 +2531,9 @@ static int skge_poll(struct net_device *dev, int *budget)
2512 2531
2513 if (done) { 2532 if (done) {
2514 local_irq_disable(); 2533 local_irq_disable();
2515 hw->intr_mask |= irqmask[skge->port];
2516 /* Order is important since data can get interrupted */
2517 skge_write32(hw, B0_IMSK, hw->intr_mask);
2518 __netif_rx_complete(dev); 2534 __netif_rx_complete(dev);
2535 hw->intr_mask |= portirqmask[skge->port];
2536 skge_write32(hw, B0_IMSK, hw->intr_mask);
2519 local_irq_enable(); 2537 local_irq_enable();
2520 } 2538 }
2521 2539
@@ -2697,19 +2715,14 @@ static irqreturn_t skge_intr(int irq, void *dev_id, struct pt_regs *regs)
2697 return IRQ_NONE; 2715 return IRQ_NONE;
2698 2716
2699 status &= hw->intr_mask; 2717 status &= hw->intr_mask;
2700 2718 if (status & IS_R1_F) {
2701 if ((status & IS_R1_F) && netif_rx_schedule_prep(hw->dev[0])) {
2702 status &= ~IS_R1_F;
2703 hw->intr_mask &= ~IS_R1_F; 2719 hw->intr_mask &= ~IS_R1_F;
2704 skge_write32(hw, B0_IMSK, hw->intr_mask); 2720 netif_rx_schedule(hw->dev[0]);
2705 __netif_rx_schedule(hw->dev[0]);
2706 } 2721 }
2707 2722
2708 if ((status & IS_R2_F) && netif_rx_schedule_prep(hw->dev[1])) { 2723 if (status & IS_R2_F) {
2709 status &= ~IS_R2_F;
2710 hw->intr_mask &= ~IS_R2_F; 2724 hw->intr_mask &= ~IS_R2_F;
2711 skge_write32(hw, B0_IMSK, hw->intr_mask); 2725 netif_rx_schedule(hw->dev[1]);
2712 __netif_rx_schedule(hw->dev[1]);
2713 } 2726 }
2714 2727
2715 if (status & IS_XA1_F) 2728 if (status & IS_XA1_F)
@@ -2732,8 +2745,7 @@ static irqreturn_t skge_intr(int irq, void *dev_id, struct pt_regs *regs)
2732 tasklet_schedule(&hw->ext_tasklet); 2745 tasklet_schedule(&hw->ext_tasklet);
2733 } 2746 }
2734 2747
2735 if (status) 2748 skge_write32(hw, B0_IMSK, hw->intr_mask);
2736 skge_write32(hw, B0_IMSK, hw->intr_mask);
2737 2749
2738 return IRQ_HANDLED; 2750 return IRQ_HANDLED;
2739} 2751}
@@ -2918,9 +2930,7 @@ static int skge_reset(struct skge_hw *hw)
2918 skge_write32(hw, B2_IRQM_INI, skge_usecs2clk(hw, 100)); 2930 skge_write32(hw, B2_IRQM_INI, skge_usecs2clk(hw, 100));
2919 skge_write32(hw, B2_IRQM_CTRL, TIM_START); 2931 skge_write32(hw, B2_IRQM_CTRL, TIM_START);
2920 2932
2921 hw->intr_mask = IS_HW_ERR | IS_EXT_REG | IS_PORT_1; 2933 hw->intr_mask = IS_HW_ERR | IS_EXT_REG;
2922 if (hw->ports > 1)
2923 hw->intr_mask |= IS_PORT_2;
2924 skge_write32(hw, B0_IMSK, hw->intr_mask); 2934 skge_write32(hw, B0_IMSK, hw->intr_mask);
2925 2935
2926 if (hw->chip_id != CHIP_ID_GENESIS) 2936 if (hw->chip_id != CHIP_ID_GENESIS)