diff options
Diffstat (limited to 'drivers/net/pcmcia/3c574_cs.c')
-rw-r--r-- | drivers/net/pcmcia/3c574_cs.c | 129 |
1 files changed, 44 insertions, 85 deletions
diff --git a/drivers/net/pcmcia/3c574_cs.c b/drivers/net/pcmcia/3c574_cs.c index c683f77c6f42..34c5e1cbf65d 100644 --- a/drivers/net/pcmcia/3c574_cs.c +++ b/drivers/net/pcmcia/3c574_cs.c | |||
@@ -62,13 +62,15 @@ invalid ramWidth is Very Bad. | |||
62 | V. References | 62 | V. References |
63 | 63 | ||
64 | http://www.scyld.com/expert/NWay.html | 64 | http://www.scyld.com/expert/NWay.html |
65 | http://www.national.com/pf/DP/DP83840.html | 65 | http://www.national.com/opf/DP/DP83840A.html |
66 | 66 | ||
67 | Thanks to Terry Murphy of 3Com for providing development information for | 67 | Thanks to Terry Murphy of 3Com for providing development information for |
68 | earlier 3Com products. | 68 | earlier 3Com products. |
69 | 69 | ||
70 | */ | 70 | */ |
71 | 71 | ||
72 | #define pr_fmt(fmt) KBUILD_MODNAME ": " fmt | ||
73 | |||
72 | #include <linux/module.h> | 74 | #include <linux/module.h> |
73 | #include <linux/kernel.h> | 75 | #include <linux/kernel.h> |
74 | #include <linux/init.h> | 76 | #include <linux/init.h> |
@@ -83,11 +85,9 @@ earlier 3Com products. | |||
83 | #include <linux/skbuff.h> | 85 | #include <linux/skbuff.h> |
84 | #include <linux/if_arp.h> | 86 | #include <linux/if_arp.h> |
85 | #include <linux/ioport.h> | 87 | #include <linux/ioport.h> |
86 | #include <linux/ethtool.h> | ||
87 | #include <linux/bitops.h> | 88 | #include <linux/bitops.h> |
88 | #include <linux/mii.h> | 89 | #include <linux/mii.h> |
89 | 90 | ||
90 | #include <pcmcia/cs.h> | ||
91 | #include <pcmcia/cistpl.h> | 91 | #include <pcmcia/cistpl.h> |
92 | #include <pcmcia/cisreg.h> | 92 | #include <pcmcia/cisreg.h> |
93 | #include <pcmcia/ciscode.h> | 93 | #include <pcmcia/ciscode.h> |
@@ -238,7 +238,6 @@ static int el3_rx(struct net_device *dev, int worklimit); | |||
238 | static int el3_close(struct net_device *dev); | 238 | static int el3_close(struct net_device *dev); |
239 | static void el3_tx_timeout(struct net_device *dev); | 239 | static void el3_tx_timeout(struct net_device *dev); |
240 | static int el3_ioctl(struct net_device *dev, struct ifreq *rq, int cmd); | 240 | static int el3_ioctl(struct net_device *dev, struct ifreq *rq, int cmd); |
241 | static const struct ethtool_ops netdev_ethtool_ops; | ||
242 | static void set_rx_mode(struct net_device *dev); | 241 | static void set_rx_mode(struct net_device *dev); |
243 | static void set_multicast_list(struct net_device *dev); | 242 | static void set_multicast_list(struct net_device *dev); |
244 | 243 | ||
@@ -280,25 +279,14 @@ static int tc574_probe(struct pcmcia_device *link) | |||
280 | spin_lock_init(&lp->window_lock); | 279 | spin_lock_init(&lp->window_lock); |
281 | link->resource[0]->end = 32; | 280 | link->resource[0]->end = 32; |
282 | link->resource[0]->flags |= IO_DATA_PATH_WIDTH_16; | 281 | link->resource[0]->flags |= IO_DATA_PATH_WIDTH_16; |
283 | link->conf.Attributes = CONF_ENABLE_IRQ; | 282 | link->config_flags |= CONF_ENABLE_IRQ; |
284 | link->conf.IntType = INT_MEMORY_AND_IO; | 283 | link->config_index = 1; |
285 | link->conf.ConfigIndex = 1; | ||
286 | 284 | ||
287 | dev->netdev_ops = &el3_netdev_ops; | 285 | dev->netdev_ops = &el3_netdev_ops; |
288 | SET_ETHTOOL_OPS(dev, &netdev_ethtool_ops); | ||
289 | dev->watchdog_timeo = TX_TIMEOUT; | 286 | dev->watchdog_timeo = TX_TIMEOUT; |
290 | 287 | ||
291 | return tc574_config(link); | 288 | return tc574_config(link); |
292 | } /* tc574_attach */ | 289 | } |
293 | |||
294 | /* | ||
295 | |||
296 | This deletes a driver "instance". The device is de-registered | ||
297 | with Card Services. If it has been released, all local data | ||
298 | structures are freed. Otherwise, the structures will be freed | ||
299 | when the device is released. | ||
300 | |||
301 | */ | ||
302 | 290 | ||
303 | static void tc574_detach(struct pcmcia_device *link) | 291 | static void tc574_detach(struct pcmcia_device *link) |
304 | { | 292 | { |
@@ -313,12 +301,6 @@ static void tc574_detach(struct pcmcia_device *link) | |||
313 | free_netdev(dev); | 301 | free_netdev(dev); |
314 | } /* tc574_detach */ | 302 | } /* tc574_detach */ |
315 | 303 | ||
316 | /* | ||
317 | tc574_config() is scheduled to run after a CARD_INSERTION event | ||
318 | is received, to configure the PCMCIA socket, and to make the | ||
319 | ethernet device available to the system. | ||
320 | */ | ||
321 | |||
322 | static const char *ram_split[] = {"5:3", "3:1", "1:1", "3:5"}; | 304 | static const char *ram_split[] = {"5:3", "3:1", "1:1", "3:5"}; |
323 | 305 | ||
324 | static int tc574_config(struct pcmcia_device *link) | 306 | static int tc574_config(struct pcmcia_device *link) |
@@ -352,7 +334,7 @@ static int tc574_config(struct pcmcia_device *link) | |||
352 | if (ret) | 334 | if (ret) |
353 | goto failed; | 335 | goto failed; |
354 | 336 | ||
355 | ret = pcmcia_request_configuration(link, &link->conf); | 337 | ret = pcmcia_enable_device(link); |
356 | if (ret) | 338 | if (ret) |
357 | goto failed; | 339 | goto failed; |
358 | 340 | ||
@@ -376,8 +358,8 @@ static int tc574_config(struct pcmcia_device *link) | |||
376 | for (i = 0; i < 3; i++) | 358 | for (i = 0; i < 3; i++) |
377 | phys_addr[i] = htons(read_eeprom(ioaddr, i + 10)); | 359 | phys_addr[i] = htons(read_eeprom(ioaddr, i + 10)); |
378 | if (phys_addr[0] == htons(0x6060)) { | 360 | if (phys_addr[0] == htons(0x6060)) { |
379 | printk(KERN_NOTICE "3c574_cs: IO port conflict at 0x%03lx" | 361 | pr_notice("IO port conflict at 0x%03lx-0x%03lx\n", |
380 | "-0x%03lx\n", dev->base_addr, dev->base_addr+15); | 362 | dev->base_addr, dev->base_addr+15); |
381 | goto failed; | 363 | goto failed; |
382 | } | 364 | } |
383 | } | 365 | } |
@@ -391,7 +373,7 @@ static int tc574_config(struct pcmcia_device *link) | |||
391 | outw(2<<11, ioaddr + RunnerRdCtrl); | 373 | outw(2<<11, ioaddr + RunnerRdCtrl); |
392 | mcr = inb(ioaddr + 2); | 374 | mcr = inb(ioaddr + 2); |
393 | outw(0<<11, ioaddr + RunnerRdCtrl); | 375 | outw(0<<11, ioaddr + RunnerRdCtrl); |
394 | printk(KERN_INFO " ASIC rev %d,", mcr>>3); | 376 | pr_info(" ASIC rev %d,", mcr>>3); |
395 | EL3WINDOW(3); | 377 | EL3WINDOW(3); |
396 | config = inl(ioaddr + Wn3_Config); | 378 | config = inl(ioaddr + Wn3_Config); |
397 | lp->default_media = (config & Xcvr) >> Xcvr_shift; | 379 | lp->default_media = (config & Xcvr) >> Xcvr_shift; |
@@ -428,7 +410,7 @@ static int tc574_config(struct pcmcia_device *link) | |||
428 | } | 410 | } |
429 | } | 411 | } |
430 | if (phy > 32) { | 412 | if (phy > 32) { |
431 | printk(KERN_NOTICE " No MII transceivers found!\n"); | 413 | pr_notice(" No MII transceivers found!\n"); |
432 | goto failed; | 414 | goto failed; |
433 | } | 415 | } |
434 | i = mdio_read(ioaddr, lp->phys, 16) | 0x40; | 416 | i = mdio_read(ioaddr, lp->phys, 16) | 0x40; |
@@ -444,18 +426,16 @@ static int tc574_config(struct pcmcia_device *link) | |||
444 | SET_NETDEV_DEV(dev, &link->dev); | 426 | SET_NETDEV_DEV(dev, &link->dev); |
445 | 427 | ||
446 | if (register_netdev(dev) != 0) { | 428 | if (register_netdev(dev) != 0) { |
447 | printk(KERN_NOTICE "3c574_cs: register_netdev() failed\n"); | 429 | pr_notice("register_netdev() failed\n"); |
448 | goto failed; | 430 | goto failed; |
449 | } | 431 | } |
450 | 432 | ||
451 | printk(KERN_INFO "%s: %s at io %#3lx, irq %d, " | 433 | netdev_info(dev, "%s at io %#3lx, irq %d, hw_addr %pM\n", |
452 | "hw_addr %pM.\n", | 434 | cardname, dev->base_addr, dev->irq, dev->dev_addr); |
453 | dev->name, cardname, dev->base_addr, dev->irq, | 435 | netdev_info(dev, " %dK FIFO split %s Rx:Tx, %sMII interface.\n", |
454 | dev->dev_addr); | 436 | 8 << config & Ram_size, |
455 | printk(" %dK FIFO split %s Rx:Tx, %sMII interface.\n", | 437 | ram_split[(config & Ram_split) >> Ram_split_shift], |
456 | 8 << config & Ram_size, | 438 | config & Autoselect ? "autoselect " : ""); |
457 | ram_split[(config & Ram_split) >> Ram_split_shift], | ||
458 | config & Autoselect ? "autoselect " : ""); | ||
459 | 439 | ||
460 | return 0; | 440 | return 0; |
461 | 441 | ||
@@ -465,12 +445,6 @@ failed: | |||
465 | 445 | ||
466 | } /* tc574_config */ | 446 | } /* tc574_config */ |
467 | 447 | ||
468 | /* | ||
469 | After a card is removed, tc574_release() will unregister the net | ||
470 | device, and release the PCMCIA configuration. If the device is | ||
471 | still open, this will be postponed until it is closed. | ||
472 | */ | ||
473 | |||
474 | static void tc574_release(struct pcmcia_device *link) | 448 | static void tc574_release(struct pcmcia_device *link) |
475 | { | 449 | { |
476 | pcmcia_disable_device(link); | 450 | pcmcia_disable_device(link); |
@@ -502,14 +476,14 @@ static void dump_status(struct net_device *dev) | |||
502 | { | 476 | { |
503 | unsigned int ioaddr = dev->base_addr; | 477 | unsigned int ioaddr = dev->base_addr; |
504 | EL3WINDOW(1); | 478 | EL3WINDOW(1); |
505 | printk(KERN_INFO " irq status %04x, rx status %04x, tx status " | 479 | netdev_info(dev, " irq status %04x, rx status %04x, tx status %02x, tx free %04x\n", |
506 | "%02x, tx free %04x\n", inw(ioaddr+EL3_STATUS), | 480 | inw(ioaddr+EL3_STATUS), |
507 | inw(ioaddr+RxStatus), inb(ioaddr+TxStatus), | 481 | inw(ioaddr+RxStatus), inb(ioaddr+TxStatus), |
508 | inw(ioaddr+TxFree)); | 482 | inw(ioaddr+TxFree)); |
509 | EL3WINDOW(4); | 483 | EL3WINDOW(4); |
510 | printk(KERN_INFO " diagnostics: fifo %04x net %04x ethernet %04x" | 484 | netdev_info(dev, " diagnostics: fifo %04x net %04x ethernet %04x media %04x\n", |
511 | " media %04x\n", inw(ioaddr+0x04), inw(ioaddr+0x06), | 485 | inw(ioaddr+0x04), inw(ioaddr+0x06), |
512 | inw(ioaddr+0x08), inw(ioaddr+0x0a)); | 486 | inw(ioaddr+0x08), inw(ioaddr+0x0a)); |
513 | EL3WINDOW(1); | 487 | EL3WINDOW(1); |
514 | } | 488 | } |
515 | 489 | ||
@@ -523,7 +497,7 @@ static void tc574_wait_for_completion(struct net_device *dev, int cmd) | |||
523 | while (--i > 0) | 497 | while (--i > 0) |
524 | if (!(inw(dev->base_addr + EL3_STATUS) & 0x1000)) break; | 498 | if (!(inw(dev->base_addr + EL3_STATUS) & 0x1000)) break; |
525 | if (i == 0) | 499 | if (i == 0) |
526 | printk(KERN_NOTICE "%s: command 0x%04x did not complete!\n", dev->name, cmd); | 500 | netdev_notice(dev, "command 0x%04x did not complete!\n", cmd); |
527 | } | 501 | } |
528 | 502 | ||
529 | /* Read a word from the EEPROM using the regular EEPROM access register. | 503 | /* Read a word from the EEPROM using the regular EEPROM access register. |
@@ -710,7 +684,7 @@ static int el3_open(struct net_device *dev) | |||
710 | netif_start_queue(dev); | 684 | netif_start_queue(dev); |
711 | 685 | ||
712 | tc574_reset(dev); | 686 | tc574_reset(dev); |
713 | lp->media.function = &media_check; | 687 | lp->media.function = media_check; |
714 | lp->media.data = (unsigned long) dev; | 688 | lp->media.data = (unsigned long) dev; |
715 | lp->media.expires = jiffies + HZ; | 689 | lp->media.expires = jiffies + HZ; |
716 | add_timer(&lp->media); | 690 | add_timer(&lp->media); |
@@ -725,7 +699,7 @@ static void el3_tx_timeout(struct net_device *dev) | |||
725 | { | 699 | { |
726 | unsigned int ioaddr = dev->base_addr; | 700 | unsigned int ioaddr = dev->base_addr; |
727 | 701 | ||
728 | printk(KERN_NOTICE "%s: Transmit timed out!\n", dev->name); | 702 | netdev_notice(dev, "Transmit timed out!\n"); |
729 | dump_status(dev); | 703 | dump_status(dev); |
730 | dev->stats.tx_errors++; | 704 | dev->stats.tx_errors++; |
731 | dev->trans_start = jiffies; /* prevent tx timeout */ | 705 | dev->trans_start = jiffies; /* prevent tx timeout */ |
@@ -848,8 +822,8 @@ static irqreturn_t el3_interrupt(int irq, void *dev_id) | |||
848 | EL3WINDOW(4); | 822 | EL3WINDOW(4); |
849 | fifo_diag = inw(ioaddr + Wn4_FIFODiag); | 823 | fifo_diag = inw(ioaddr + Wn4_FIFODiag); |
850 | EL3WINDOW(1); | 824 | EL3WINDOW(1); |
851 | printk(KERN_NOTICE "%s: adapter failure, FIFO diagnostic" | 825 | netdev_notice(dev, "adapter failure, FIFO diagnostic register %04x\n", |
852 | " register %04x.\n", dev->name, fifo_diag); | 826 | fifo_diag); |
853 | if (fifo_diag & 0x0400) { | 827 | if (fifo_diag & 0x0400) { |
854 | /* Tx overrun */ | 828 | /* Tx overrun */ |
855 | tc574_wait_for_completion(dev, TxReset); | 829 | tc574_wait_for_completion(dev, TxReset); |
@@ -903,7 +877,7 @@ static void media_check(unsigned long arg) | |||
903 | this, we can limp along even if the interrupt is blocked */ | 877 | this, we can limp along even if the interrupt is blocked */ |
904 | if ((inw(ioaddr + EL3_STATUS) & IntLatch) && (inb(ioaddr + Timer) == 0xff)) { | 878 | if ((inw(ioaddr + EL3_STATUS) & IntLatch) && (inb(ioaddr + Timer) == 0xff)) { |
905 | if (!lp->fast_poll) | 879 | if (!lp->fast_poll) |
906 | printk(KERN_INFO "%s: interrupt(s) dropped!\n", dev->name); | 880 | netdev_info(dev, "interrupt(s) dropped!\n"); |
907 | 881 | ||
908 | local_irq_save(flags); | 882 | local_irq_save(flags); |
909 | el3_interrupt(dev->irq, dev); | 883 | el3_interrupt(dev->irq, dev); |
@@ -926,23 +900,21 @@ static void media_check(unsigned long arg) | |||
926 | 900 | ||
927 | if (media != lp->media_status) { | 901 | if (media != lp->media_status) { |
928 | if ((media ^ lp->media_status) & 0x0004) | 902 | if ((media ^ lp->media_status) & 0x0004) |
929 | printk(KERN_INFO "%s: %s link beat\n", dev->name, | 903 | netdev_info(dev, "%s link beat\n", |
930 | (lp->media_status & 0x0004) ? "lost" : "found"); | 904 | (lp->media_status & 0x0004) ? "lost" : "found"); |
931 | if ((media ^ lp->media_status) & 0x0020) { | 905 | if ((media ^ lp->media_status) & 0x0020) { |
932 | lp->partner = 0; | 906 | lp->partner = 0; |
933 | if (lp->media_status & 0x0020) { | 907 | if (lp->media_status & 0x0020) { |
934 | printk(KERN_INFO "%s: autonegotiation restarted\n", | 908 | netdev_info(dev, "autonegotiation restarted\n"); |
935 | dev->name); | ||
936 | } else if (partner) { | 909 | } else if (partner) { |
937 | partner &= lp->advertising; | 910 | partner &= lp->advertising; |
938 | lp->partner = partner; | 911 | lp->partner = partner; |
939 | printk(KERN_INFO "%s: autonegotiation complete: " | 912 | netdev_info(dev, "autonegotiation complete: " |
940 | "%sbaseT-%cD selected\n", dev->name, | 913 | "%dbaseT-%cD selected\n", |
941 | ((partner & 0x0180) ? "100" : "10"), | 914 | (partner & 0x0180) ? 100 : 10, |
942 | ((partner & 0x0140) ? 'F' : 'H')); | 915 | (partner & 0x0140) ? 'F' : 'H'); |
943 | } else { | 916 | } else { |
944 | printk(KERN_INFO "%s: link partner did not autonegotiate\n", | 917 | netdev_info(dev, "link partner did not autonegotiate\n"); |
945 | dev->name); | ||
946 | } | 918 | } |
947 | 919 | ||
948 | EL3WINDOW(3); | 920 | EL3WINDOW(3); |
@@ -952,10 +924,9 @@ static void media_check(unsigned long arg) | |||
952 | 924 | ||
953 | } | 925 | } |
954 | if (media & 0x0010) | 926 | if (media & 0x0010) |
955 | printk(KERN_INFO "%s: remote fault detected\n", | 927 | netdev_info(dev, "remote fault detected\n"); |
956 | dev->name); | ||
957 | if (media & 0x0002) | 928 | if (media & 0x0002) |
958 | printk(KERN_INFO "%s: jabber detected\n", dev->name); | 929 | netdev_info(dev, "jabber detected\n"); |
959 | lp->media_status = media; | 930 | lp->media_status = media; |
960 | } | 931 | } |
961 | spin_unlock_irqrestore(&lp->window_lock, flags); | 932 | spin_unlock_irqrestore(&lp->window_lock, flags); |
@@ -979,7 +950,7 @@ static struct net_device_stats *el3_get_stats(struct net_device *dev) | |||
979 | } | 950 | } |
980 | 951 | ||
981 | /* Update statistics. | 952 | /* Update statistics. |
982 | Suprisingly this need not be run single-threaded, but it effectively is. | 953 | Surprisingly this need not be run single-threaded, but it effectively is. |
983 | The counters clear when read, so the adds must merely be atomic. | 954 | The counters clear when read, so the adds must merely be atomic. |
984 | */ | 955 | */ |
985 | static void update_stats(struct net_device *dev) | 956 | static void update_stats(struct net_device *dev) |
@@ -1065,16 +1036,6 @@ static int el3_rx(struct net_device *dev, int worklimit) | |||
1065 | return worklimit; | 1036 | return worklimit; |
1066 | } | 1037 | } |
1067 | 1038 | ||
1068 | static void netdev_get_drvinfo(struct net_device *dev, | ||
1069 | struct ethtool_drvinfo *info) | ||
1070 | { | ||
1071 | strcpy(info->driver, "3c574_cs"); | ||
1072 | } | ||
1073 | |||
1074 | static const struct ethtool_ops netdev_ethtool_ops = { | ||
1075 | .get_drvinfo = netdev_get_drvinfo, | ||
1076 | }; | ||
1077 | |||
1078 | /* Provide ioctl() calls to examine the MII xcvr state. */ | 1039 | /* Provide ioctl() calls to examine the MII xcvr state. */ |
1079 | static int el3_ioctl(struct net_device *dev, struct ifreq *rq, int cmd) | 1040 | static int el3_ioctl(struct net_device *dev, struct ifreq *rq, int cmd) |
1080 | { | 1041 | { |
@@ -1189,7 +1150,7 @@ static int el3_close(struct net_device *dev) | |||
1189 | return 0; | 1150 | return 0; |
1190 | } | 1151 | } |
1191 | 1152 | ||
1192 | static struct pcmcia_device_id tc574_ids[] = { | 1153 | static const struct pcmcia_device_id tc574_ids[] = { |
1193 | PCMCIA_DEVICE_MANF_CARD(0x0101, 0x0574), | 1154 | PCMCIA_DEVICE_MANF_CARD(0x0101, 0x0574), |
1194 | PCMCIA_MFC_DEVICE_CIS_MANF_CARD(0, 0x0101, 0x0556, "cis/3CCFEM556.cis"), | 1155 | PCMCIA_MFC_DEVICE_CIS_MANF_CARD(0, 0x0101, 0x0556, "cis/3CCFEM556.cis"), |
1195 | PCMCIA_DEVICE_NULL, | 1156 | PCMCIA_DEVICE_NULL, |
@@ -1198,9 +1159,7 @@ MODULE_DEVICE_TABLE(pcmcia, tc574_ids); | |||
1198 | 1159 | ||
1199 | static struct pcmcia_driver tc574_driver = { | 1160 | static struct pcmcia_driver tc574_driver = { |
1200 | .owner = THIS_MODULE, | 1161 | .owner = THIS_MODULE, |
1201 | .drv = { | 1162 | .name = "3c574_cs", |
1202 | .name = "3c574_cs", | ||
1203 | }, | ||
1204 | .probe = tc574_probe, | 1163 | .probe = tc574_probe, |
1205 | .remove = tc574_detach, | 1164 | .remove = tc574_detach, |
1206 | .id_table = tc574_ids, | 1165 | .id_table = tc574_ids, |