diff options
author | Glenn Elliott <gelliott@cs.unc.edu> | 2012-03-04 19:47:13 -0500 |
---|---|---|
committer | Glenn Elliott <gelliott@cs.unc.edu> | 2012-03-04 19:47:13 -0500 |
commit | c71c03bda1e86c9d5198c5d83f712e695c4f2a1e (patch) | |
tree | ecb166cb3e2b7e2adb3b5e292245fefd23381ac8 /drivers/net/pcnet32.c | |
parent | ea53c912f8a86a8567697115b6a0d8152beee5c8 (diff) | |
parent | 6a00f206debf8a5c8899055726ad127dbeeed098 (diff) |
Merge branch 'mpi-master' into wip-k-fmlpwip-k-fmlp
Conflicts:
litmus/sched_cedf.c
Diffstat (limited to 'drivers/net/pcnet32.c')
-rw-r--r-- | drivers/net/pcnet32.c | 96 |
1 files changed, 39 insertions, 57 deletions
diff --git a/drivers/net/pcnet32.c b/drivers/net/pcnet32.c index c200c2821730..b48aba9e4227 100644 --- a/drivers/net/pcnet32.c +++ b/drivers/net/pcnet32.c | |||
@@ -295,12 +295,14 @@ struct pcnet32_private { | |||
295 | struct net_device *next; | 295 | struct net_device *next; |
296 | struct mii_if_info mii_if; | 296 | struct mii_if_info mii_if; |
297 | struct timer_list watchdog_timer; | 297 | struct timer_list watchdog_timer; |
298 | struct timer_list blink_timer; | ||
299 | u32 msg_enable; /* debug message level */ | 298 | u32 msg_enable; /* debug message level */ |
300 | 299 | ||
301 | /* each bit indicates an available PHY */ | 300 | /* each bit indicates an available PHY */ |
302 | u32 phymask; | 301 | u32 phymask; |
303 | unsigned short chip_version; /* which variant this is */ | 302 | unsigned short chip_version; /* which variant this is */ |
303 | |||
304 | /* saved registers during ethtool blink */ | ||
305 | u16 save_regs[4]; | ||
304 | }; | 306 | }; |
305 | 307 | ||
306 | static int pcnet32_probe_pci(struct pci_dev *, const struct pci_device_id *); | 308 | static int pcnet32_probe_pci(struct pci_dev *, const struct pci_device_id *); |
@@ -324,8 +326,6 @@ static void pcnet32_restart(struct net_device *dev, unsigned int csr0_bits); | |||
324 | static void pcnet32_ethtool_test(struct net_device *dev, | 326 | static void pcnet32_ethtool_test(struct net_device *dev, |
325 | struct ethtool_test *eth_test, u64 * data); | 327 | struct ethtool_test *eth_test, u64 * data); |
326 | static int pcnet32_loopback_test(struct net_device *dev, uint64_t * data1); | 328 | static int pcnet32_loopback_test(struct net_device *dev, uint64_t * data1); |
327 | static int pcnet32_phys_id(struct net_device *dev, u32 data); | ||
328 | static void pcnet32_led_blink_callback(struct net_device *dev); | ||
329 | static int pcnet32_get_regs_len(struct net_device *dev); | 329 | static int pcnet32_get_regs_len(struct net_device *dev); |
330 | static void pcnet32_get_regs(struct net_device *dev, struct ethtool_regs *regs, | 330 | static void pcnet32_get_regs(struct net_device *dev, struct ethtool_regs *regs, |
331 | void *ptr); | 331 | void *ptr); |
@@ -376,7 +376,7 @@ static void pcnet32_wio_reset(unsigned long addr) | |||
376 | static int pcnet32_wio_check(unsigned long addr) | 376 | static int pcnet32_wio_check(unsigned long addr) |
377 | { | 377 | { |
378 | outw(88, addr + PCNET32_WIO_RAP); | 378 | outw(88, addr + PCNET32_WIO_RAP); |
379 | return (inw(addr + PCNET32_WIO_RAP) == 88); | 379 | return inw(addr + PCNET32_WIO_RAP) == 88; |
380 | } | 380 | } |
381 | 381 | ||
382 | static struct pcnet32_access pcnet32_wio = { | 382 | static struct pcnet32_access pcnet32_wio = { |
@@ -431,7 +431,7 @@ static void pcnet32_dwio_reset(unsigned long addr) | |||
431 | static int pcnet32_dwio_check(unsigned long addr) | 431 | static int pcnet32_dwio_check(unsigned long addr) |
432 | { | 432 | { |
433 | outl(88, addr + PCNET32_DWIO_RAP); | 433 | outl(88, addr + PCNET32_DWIO_RAP); |
434 | return ((inl(addr + PCNET32_DWIO_RAP) & 0xffff) == 88); | 434 | return (inl(addr + PCNET32_DWIO_RAP) & 0xffff) == 88; |
435 | } | 435 | } |
436 | 436 | ||
437 | static struct pcnet32_access pcnet32_dwio = { | 437 | static struct pcnet32_access pcnet32_dwio = { |
@@ -1022,7 +1022,8 @@ clean_up: | |||
1022 | return rc; | 1022 | return rc; |
1023 | } /* end pcnet32_loopback_test */ | 1023 | } /* end pcnet32_loopback_test */ |
1024 | 1024 | ||
1025 | static void pcnet32_led_blink_callback(struct net_device *dev) | 1025 | static int pcnet32_set_phys_id(struct net_device *dev, |
1026 | enum ethtool_phys_id_state state) | ||
1026 | { | 1027 | { |
1027 | struct pcnet32_private *lp = netdev_priv(dev); | 1028 | struct pcnet32_private *lp = netdev_priv(dev); |
1028 | struct pcnet32_access *a = &lp->a; | 1029 | struct pcnet32_access *a = &lp->a; |
@@ -1030,50 +1031,31 @@ static void pcnet32_led_blink_callback(struct net_device *dev) | |||
1030 | unsigned long flags; | 1031 | unsigned long flags; |
1031 | int i; | 1032 | int i; |
1032 | 1033 | ||
1033 | spin_lock_irqsave(&lp->lock, flags); | 1034 | switch (state) { |
1034 | for (i = 4; i < 8; i++) | 1035 | case ETHTOOL_ID_ACTIVE: |
1035 | a->write_bcr(ioaddr, i, a->read_bcr(ioaddr, i) ^ 0x4000); | 1036 | /* Save the current value of the bcrs */ |
1036 | spin_unlock_irqrestore(&lp->lock, flags); | 1037 | spin_lock_irqsave(&lp->lock, flags); |
1037 | 1038 | for (i = 4; i < 8; i++) | |
1038 | mod_timer(&lp->blink_timer, PCNET32_BLINK_TIMEOUT); | 1039 | lp->save_regs[i - 4] = a->read_bcr(ioaddr, i); |
1039 | } | 1040 | spin_unlock_irqrestore(&lp->lock, flags); |
1041 | return 2; /* cycle on/off twice per second */ | ||
1040 | 1042 | ||
1041 | static int pcnet32_phys_id(struct net_device *dev, u32 data) | 1043 | case ETHTOOL_ID_ON: |
1042 | { | 1044 | case ETHTOOL_ID_OFF: |
1043 | struct pcnet32_private *lp = netdev_priv(dev); | 1045 | /* Blink the led */ |
1044 | struct pcnet32_access *a = &lp->a; | 1046 | spin_lock_irqsave(&lp->lock, flags); |
1045 | ulong ioaddr = dev->base_addr; | 1047 | for (i = 4; i < 8; i++) |
1046 | unsigned long flags; | 1048 | a->write_bcr(ioaddr, i, a->read_bcr(ioaddr, i) ^ 0x4000); |
1047 | int i, regs[4]; | 1049 | spin_unlock_irqrestore(&lp->lock, flags); |
1050 | break; | ||
1048 | 1051 | ||
1049 | if (!lp->blink_timer.function) { | 1052 | case ETHTOOL_ID_INACTIVE: |
1050 | init_timer(&lp->blink_timer); | 1053 | /* Restore the original value of the bcrs */ |
1051 | lp->blink_timer.function = (void *)pcnet32_led_blink_callback; | 1054 | spin_lock_irqsave(&lp->lock, flags); |
1052 | lp->blink_timer.data = (unsigned long)dev; | 1055 | for (i = 4; i < 8; i++) |
1056 | a->write_bcr(ioaddr, i, lp->save_regs[i - 4]); | ||
1057 | spin_unlock_irqrestore(&lp->lock, flags); | ||
1053 | } | 1058 | } |
1054 | |||
1055 | /* Save the current value of the bcrs */ | ||
1056 | spin_lock_irqsave(&lp->lock, flags); | ||
1057 | for (i = 4; i < 8; i++) | ||
1058 | regs[i - 4] = a->read_bcr(ioaddr, i); | ||
1059 | spin_unlock_irqrestore(&lp->lock, flags); | ||
1060 | |||
1061 | mod_timer(&lp->blink_timer, jiffies); | ||
1062 | set_current_state(TASK_INTERRUPTIBLE); | ||
1063 | |||
1064 | /* AV: the limit here makes no sense whatsoever */ | ||
1065 | if ((!data) || (data > (u32) (MAX_SCHEDULE_TIMEOUT / HZ))) | ||
1066 | data = (u32) (MAX_SCHEDULE_TIMEOUT / HZ); | ||
1067 | |||
1068 | msleep_interruptible(data * 1000); | ||
1069 | del_timer_sync(&lp->blink_timer); | ||
1070 | |||
1071 | /* Restore the original value of the bcrs */ | ||
1072 | spin_lock_irqsave(&lp->lock, flags); | ||
1073 | for (i = 4; i < 8; i++) | ||
1074 | a->write_bcr(ioaddr, i, regs[i - 4]); | ||
1075 | spin_unlock_irqrestore(&lp->lock, flags); | ||
1076 | |||
1077 | return 0; | 1059 | return 0; |
1078 | } | 1060 | } |
1079 | 1061 | ||
@@ -1450,7 +1432,7 @@ static const struct ethtool_ops pcnet32_ethtool_ops = { | |||
1450 | .set_ringparam = pcnet32_set_ringparam, | 1432 | .set_ringparam = pcnet32_set_ringparam, |
1451 | .get_strings = pcnet32_get_strings, | 1433 | .get_strings = pcnet32_get_strings, |
1452 | .self_test = pcnet32_ethtool_test, | 1434 | .self_test = pcnet32_ethtool_test, |
1453 | .phys_id = pcnet32_phys_id, | 1435 | .set_phys_id = pcnet32_set_phys_id, |
1454 | .get_regs_len = pcnet32_get_regs_len, | 1436 | .get_regs_len = pcnet32_get_regs_len, |
1455 | .get_regs = pcnet32_get_regs, | 1437 | .get_regs = pcnet32_get_regs, |
1456 | .get_sset_count = pcnet32_get_sset_count, | 1438 | .get_sset_count = pcnet32_get_sset_count, |
@@ -1651,7 +1633,7 @@ pcnet32_probe1(unsigned long ioaddr, int shared, struct pci_dev *pdev) | |||
1651 | /* | 1633 | /* |
1652 | * On selected chips turn on the BCR18:NOUFLO bit. This stops transmit | 1634 | * On selected chips turn on the BCR18:NOUFLO bit. This stops transmit |
1653 | * starting until the packet is loaded. Strike one for reliability, lose | 1635 | * starting until the packet is loaded. Strike one for reliability, lose |
1654 | * one for latency - although on PCI this isnt a big loss. Older chips | 1636 | * one for latency - although on PCI this isn't a big loss. Older chips |
1655 | * have FIFO's smaller than a packet, so you can't do this. | 1637 | * have FIFO's smaller than a packet, so you can't do this. |
1656 | * Turn on BCR18:BurstRdEn and BCR18:BurstWrEn. | 1638 | * Turn on BCR18:BurstRdEn and BCR18:BurstWrEn. |
1657 | */ | 1639 | */ |
@@ -2117,7 +2099,7 @@ static int pcnet32_open(struct net_device *dev) | |||
2117 | int first_phy = -1; | 2099 | int first_phy = -1; |
2118 | u16 bmcr; | 2100 | u16 bmcr; |
2119 | u32 bcr9; | 2101 | u32 bcr9; |
2120 | struct ethtool_cmd ecmd; | 2102 | struct ethtool_cmd ecmd = { .cmd = ETHTOOL_GSET }; |
2121 | 2103 | ||
2122 | /* | 2104 | /* |
2123 | * There is really no good other way to handle multiple PHYs | 2105 | * There is really no good other way to handle multiple PHYs |
@@ -2133,9 +2115,9 @@ static int pcnet32_open(struct net_device *dev) | |||
2133 | ecmd.port = PORT_MII; | 2115 | ecmd.port = PORT_MII; |
2134 | ecmd.transceiver = XCVR_INTERNAL; | 2116 | ecmd.transceiver = XCVR_INTERNAL; |
2135 | ecmd.autoneg = AUTONEG_DISABLE; | 2117 | ecmd.autoneg = AUTONEG_DISABLE; |
2136 | ecmd.speed = | 2118 | ethtool_cmd_speed_set(&ecmd, |
2137 | lp-> | 2119 | (lp->options & PCNET32_PORT_100) ? |
2138 | options & PCNET32_PORT_100 ? SPEED_100 : SPEED_10; | 2120 | SPEED_100 : SPEED_10); |
2139 | bcr9 = lp->a.read_bcr(ioaddr, 9); | 2121 | bcr9 = lp->a.read_bcr(ioaddr, 9); |
2140 | 2122 | ||
2141 | if (lp->options & PCNET32_PORT_FD) { | 2123 | if (lp->options & PCNET32_PORT_FD) { |
@@ -2781,11 +2763,11 @@ static void pcnet32_check_media(struct net_device *dev, int verbose) | |||
2781 | netif_carrier_on(dev); | 2763 | netif_carrier_on(dev); |
2782 | if (lp->mii) { | 2764 | if (lp->mii) { |
2783 | if (netif_msg_link(lp)) { | 2765 | if (netif_msg_link(lp)) { |
2784 | struct ethtool_cmd ecmd; | 2766 | struct ethtool_cmd ecmd = { |
2767 | .cmd = ETHTOOL_GSET }; | ||
2785 | mii_ethtool_gset(&lp->mii_if, &ecmd); | 2768 | mii_ethtool_gset(&lp->mii_if, &ecmd); |
2786 | netdev_info(dev, "link up, %sMbps, %s-duplex\n", | 2769 | netdev_info(dev, "link up, %uMbps, %s-duplex\n", |
2787 | (ecmd.speed == SPEED_100) | 2770 | ethtool_cmd_speed(&ecmd), |
2788 | ? "100" : "10", | ||
2789 | (ecmd.duplex == DUPLEX_FULL) | 2771 | (ecmd.duplex == DUPLEX_FULL) |
2790 | ? "full" : "half"); | 2772 | ? "full" : "half"); |
2791 | } | 2773 | } |