Commit 50b94f1d authored by Stephen Hemminger's avatar Stephen Hemminger

[PATCH] (19/42) 3c515-T10

NE43-3c515
	* convert to dynamic allocation
	* fixed up device list handling
parent e394dc33
...@@ -307,7 +307,8 @@ struct boom_tx_desc { ...@@ -307,7 +307,8 @@ struct boom_tx_desc {
struct corkscrew_private { struct corkscrew_private {
const char *product_name; const char *product_name;
struct net_device *next_module; struct list_head list;
struct net_device *our_dev;
/* The Rx and Tx rings are here to keep them quad-word-aligned. */ /* The Rx and Tx rings are here to keep them quad-word-aligned. */
struct boom_rx_desc rx_ring[RX_RING_SIZE]; struct boom_rx_desc rx_ring[RX_RING_SIZE];
struct boom_tx_desc tx_ring[TX_RING_SIZE]; struct boom_tx_desc tx_ring[TX_RING_SIZE];
...@@ -329,6 +330,7 @@ struct corkscrew_private { ...@@ -329,6 +330,7 @@ struct corkscrew_private {
full_bus_master_tx:1, full_bus_master_rx:1, /* Boomerang */ full_bus_master_tx:1, full_bus_master_rx:1, /* Boomerang */
tx_full:1; tx_full:1;
spinlock_t lock; spinlock_t lock;
struct device *dev;
}; };
/* The action to take with a media selection timer tick. /* The action to take with a media selection timer tick.
...@@ -367,17 +369,12 @@ static struct isapnp_device_id corkscrew_isapnp_adapters[] = { ...@@ -367,17 +369,12 @@ static struct isapnp_device_id corkscrew_isapnp_adapters[] = {
MODULE_DEVICE_TABLE(isapnp, corkscrew_isapnp_adapters); MODULE_DEVICE_TABLE(isapnp, corkscrew_isapnp_adapters);
static int corkscrew_isapnp_phys_addr[3];
static int nopnp; static int nopnp;
#endif /* __ISAPNP__ */ #endif /* __ISAPNP__ */
static int corkscrew_scan(struct net_device *dev); static struct net_device *corkscrew_scan(int unit);
static struct net_device *corkscrew_found_device(struct net_device *dev, static void corkscrew_setup(struct net_device *dev, int ioaddr,
int ioaddr, int irq, struct pnp_dev *idev, int card_number);
int product_index,
int options);
static int corkscrew_probe1(struct net_device *dev);
static int corkscrew_open(struct net_device *dev); static int corkscrew_open(struct net_device *dev);
static void corkscrew_timer(unsigned long arg); static void corkscrew_timer(unsigned long arg);
static int corkscrew_start_xmit(struct sk_buff *skb, static int corkscrew_start_xmit(struct sk_buff *skb,
...@@ -413,47 +410,99 @@ static int options[MAX_UNITS] = { -1, -1, -1, -1, -1, -1, -1, -1, }; ...@@ -413,47 +410,99 @@ static int options[MAX_UNITS] = { -1, -1, -1, -1, -1, -1, -1, -1, };
#ifdef MODULE #ifdef MODULE
static int debug = -1; static int debug = -1;
/* A list of all installed Vortex devices, for removing the driver module. */ /* A list of all installed Vortex devices, for removing the driver module. */
static struct net_device *root_corkscrew_dev; /* we will need locking (and refcounting) if we ever use it for more */
static LIST_HEAD(root_corkscrew_dev);
int init_module(void) int init_module(void)
{ {
int cards_found; int found = 0;
if (debug >= 0) if (debug >= 0)
corkscrew_debug = debug; corkscrew_debug = debug;
if (corkscrew_debug) if (corkscrew_debug)
printk(version); printk(version);
while (corkscrew_scan(-1))
root_corkscrew_dev = NULL; found++;
cards_found = corkscrew_scan(NULL); return found ? 0 : -ENODEV;
return cards_found ? 0 : -ENODEV;
} }
#else #else
int tc515_probe(struct net_device *dev) struct net_device *tc515_probe(int unit)
{ {
int cards_found = 0; struct net_device *dev = corkscrew_scan(unit);
static int printed;
SET_MODULE_OWNER(dev); if (!dev)
return ERR_PTR(-ENODEV);
cards_found = corkscrew_scan(dev);
if (corkscrew_debug > 0 && cards_found) if (corkscrew_debug > 0 && !printed) {
printed = 1;
printk(version); printk(version);
}
return cards_found ? 0 : -ENODEV; return dev;
} }
#endif /* not MODULE */ #endif /* not MODULE */
static int corkscrew_scan(struct net_device *dev) static int check_device(unsigned ioaddr)
{
int timer;
if (!request_region(ioaddr, CORKSCREW_TOTAL_SIZE, "3c515"))
return 0;
/* Check the resource configuration for a matching ioaddr. */
if ((inw(ioaddr + 0x2002) & 0x1f0) != (ioaddr & 0x1f0)) {
release_region(ioaddr, CORKSCREW_TOTAL_SIZE);
return 0;
}
/* Verify by reading the device ID from the EEPROM. */
outw(EEPROM_Read + 7, ioaddr + Wn0EepromCmd);
/* Pause for at least 162 us. for the read to take place. */
for (timer = 4; timer >= 0; timer--) {
udelay(162);
if ((inw(ioaddr + Wn0EepromCmd) & 0x0200) == 0)
break;
}
if (inw(ioaddr + Wn0EepromData) != 0x6d50) {
release_region(ioaddr, CORKSCREW_TOTAL_SIZE);
return 0;
}
return 1;
}
static void cleanup_card(struct net_device *dev)
{
struct corkscrew_private *vp = (struct corkscrew_private *) dev->priv;
list_del_init(&vp->list);
if (dev->dma)
free_dma(dev->dma);
outw(TotalReset, dev->base_addr + EL3_CMD);
release_region(dev->base_addr, CORKSCREW_TOTAL_SIZE);
if (vp->dev)
pnp_device_detach(to_pnp_dev(vp->dev));
}
static struct net_device *corkscrew_scan(int unit)
{ {
int cards_found = 0; struct net_device *dev;
static int cards_found = 0;
static int ioaddr; static int ioaddr;
int err;
#ifdef __ISAPNP__ #ifdef __ISAPNP__
short i; short i;
static int pnp_cards; static int pnp_cards;
#endif #endif
dev = alloc_etherdev(sizeof(struct corkscrew_private));
if (!dev)
return ERR_PTR(-ENOMEM);
if (unit >= 0) {
sprintf(dev->name, "eth%d", unit);
netdev_boot_setup_check(dev);
}
SET_MODULE_OWNER(dev);
#ifdef __ISAPNP__ #ifdef __ISAPNP__
if(nopnp == 1) if(nopnp == 1)
goto no_pnp; goto no_pnp;
...@@ -470,7 +519,7 @@ static int corkscrew_scan(struct net_device *dev) ...@@ -470,7 +519,7 @@ static int corkscrew_scan(struct net_device *dev)
if (pnp_activate_dev(idev) < 0) { if (pnp_activate_dev(idev) < 0) {
printk("pnp activate failed (out of resources?)\n"); printk("pnp activate failed (out of resources?)\n");
pnp_device_detach(idev); pnp_device_detach(idev);
return -ENOMEM; continue;
} }
if (!pnp_port_valid(idev, 0) || !pnp_irq_valid(idev, 0)) { if (!pnp_port_valid(idev, 0) || !pnp_irq_valid(idev, 0)) {
pnp_device_detach(idev); pnp_device_detach(idev);
...@@ -478,40 +527,22 @@ static int corkscrew_scan(struct net_device *dev) ...@@ -478,40 +527,22 @@ static int corkscrew_scan(struct net_device *dev)
} }
ioaddr = pnp_port_start(idev, 0); ioaddr = pnp_port_start(idev, 0);
irq = pnp_irq(idev, 0); irq = pnp_irq(idev, 0);
if(corkscrew_debug) if (!check_device(ioaddr)) {
printk ("ISAPNP reports %s at i/o 0x%x, irq %d\n",
(char*) corkscrew_isapnp_adapters[i].driver_data, ioaddr, irq);
if ((inw(ioaddr + 0x2002) & 0x1f0) != (ioaddr & 0x1f0)) {
pnp_device_detach(idev); pnp_device_detach(idev);
continue; continue;
} }
/* Verify by reading the device ID from the EEPROM. */ if(corkscrew_debug)
{ printk ("ISAPNP reports %s at i/o 0x%x, irq %d\n",
int timer; (char*) corkscrew_isapnp_adapters[i].driver_data, ioaddr, irq);
outw(EEPROM_Read + 7, ioaddr + Wn0EepromCmd);
/* Pause for at least 162 us. for the read to take place. */
for (timer = 4; timer >= 0; timer--) {
udelay(162);
if ((inw(ioaddr + Wn0EepromCmd) & 0x0200)
== 0)
break;
}
if (inw(ioaddr + Wn0EepromData) != 0x6d50) {
pnp_device_detach(idev);
continue;
}
}
printk(KERN_INFO "3c515 Resource configuration register %#4.4x, DCR %4.4x.\n", printk(KERN_INFO "3c515 Resource configuration register %#4.4x, DCR %4.4x.\n",
inl(ioaddr + 0x2002), inw(ioaddr + 0x2000)); inl(ioaddr + 0x2002), inw(ioaddr + 0x2000));
/* irq = inw(ioaddr + 0x2002) & 15; */ /* Use the irq from isapnp */ /* irq = inw(ioaddr + 0x2002) & 15; */ /* Use the irq from isapnp */
corkscrew_isapnp_phys_addr[pnp_cards] = ioaddr; corkscrew_setup(dev, ioaddr, idev, cards_found++);
corkscrew_found_device(dev, ioaddr, irq, CORKSCREW_ID, dev
&& dev->mem_start ? dev->
mem_start : options[cards_found]);
dev = 0;
pnp_cards++; pnp_cards++;
cards_found++; err = register_netdev(dev);
if (!err)
return dev;
cleanup_card(dev);
} }
} }
no_pnp: no_pnp:
...@@ -519,123 +550,62 @@ static int corkscrew_scan(struct net_device *dev) ...@@ -519,123 +550,62 @@ static int corkscrew_scan(struct net_device *dev)
/* Check all locations on the ISA bus -- evil! */ /* Check all locations on the ISA bus -- evil! */
for (ioaddr = 0x100; ioaddr < 0x400; ioaddr += 0x20) { for (ioaddr = 0x100; ioaddr < 0x400; ioaddr += 0x20) {
int irq; if (!check_device(ioaddr))
#ifdef __ISAPNP__
/* Make sure this was not already picked up by isapnp */
if(ioaddr == corkscrew_isapnp_phys_addr[0]) continue;
if(ioaddr == corkscrew_isapnp_phys_addr[1]) continue;
if(ioaddr == corkscrew_isapnp_phys_addr[2]) continue;
#endif /* __ISAPNP__ */
if (check_region(ioaddr, CORKSCREW_TOTAL_SIZE))
continue;
/* Check the resource configuration for a matching ioaddr. */
if ((inw(ioaddr + 0x2002) & 0x1f0) != (ioaddr & 0x1f0))
continue; continue;
/* Verify by reading the device ID from the EEPROM. */
{
int timer;
outw(EEPROM_Read + 7, ioaddr + Wn0EepromCmd);
/* Pause for at least 162 us. for the read to take place. */
for (timer = 4; timer >= 0; timer--) {
udelay(162);
if ((inw(ioaddr + Wn0EepromCmd) & 0x0200)
== 0)
break;
}
if (inw(ioaddr + Wn0EepromData) != 0x6d50)
continue;
}
printk(KERN_INFO "3c515 Resource configuration register %#4.4x, DCR %4.4x.\n", printk(KERN_INFO "3c515 Resource configuration register %#4.4x, DCR %4.4x.\n",
inl(ioaddr + 0x2002), inw(ioaddr + 0x2000)); inl(ioaddr + 0x2002), inw(ioaddr + 0x2000));
irq = inw(ioaddr + 0x2002) & 15; corkscrew_setup(dev, ioaddr, NULL, cards_found++);
corkscrew_found_device(dev, ioaddr, irq, CORKSCREW_ID, err = register_netdev(dev);
dev && dev->mem_start ? dev->mem_start : if (!err)
(cards_found >= MAX_UNITS ? -1 : return dev;
options[cards_found])); cleanup_card(dev);
dev = 0;
cards_found++;
} }
if (corkscrew_debug) free_netdev(dev);
printk(KERN_INFO "%d 3c515 cards found.\n", cards_found); return NULL;
return cards_found;
} }
static struct net_device *corkscrew_found_device(struct net_device *dev, static void corkscrew_setup(struct net_device *dev, int ioaddr,
int ioaddr, int irq, struct pnp_dev *idev, int card_number)
int product_index,
int options)
{ {
struct corkscrew_private *vp; struct corkscrew_private *vp = (struct corkscrew_private *) dev->priv;
unsigned int eeprom[0x40], checksum = 0; /* EEPROM contents */
#ifdef MODULE int i;
/* Allocate and fill new device structure. */ int irq;
int dev_size = sizeof(struct corkscrew_private);
dev = alloc_etherdev(dev_size);
if (!dev)
goto err_out;
memset(dev, 0, dev_size);
vp = (struct corkscrew_private *) dev->priv; if (idev) {
dev->base_addr = ioaddr; irq = pnp_irq(idev, 0);
dev->irq = irq; vp->dev = &idev->dev;
dev->dma = (product_index == CORKSCREW_ID ? inw(ioaddr + 0x2000) & 7 : 0);
dev->init = corkscrew_probe1;
vp->product_name = "3c515";
vp->options = options;
if (options >= 0) {
vp->media_override = ((options & 7) == 2) ? 0 : options & 7;
vp->full_duplex = (options & 8) ? 1 : 0;
vp->bus_master = (options & 16) ? 1 : 0;
} else { } else {
vp->media_override = 7; irq = inw(ioaddr + 0x2002) & 15;
vp->full_duplex = 0;
vp->bus_master = 0;
} }
vp->next_module = root_corkscrew_dev;
root_corkscrew_dev = dev;
SET_MODULE_OWNER(dev);
if (register_netdev(dev) < 0)
goto err_free_dev;
#else /* not a MODULE */
/* Caution: quad-word alignment required for rings! */
dev->priv = kmalloc(sizeof(struct corkscrew_private), GFP_KERNEL);
if (!dev->priv)
goto err_out;
memset(dev->priv, 0, sizeof(struct corkscrew_private));
dev = init_etherdev(dev, sizeof(struct corkscrew_private));
dev->base_addr = ioaddr; dev->base_addr = ioaddr;
dev->irq = irq; dev->irq = irq;
dev->dma = (product_index == CORKSCREW_ID ? inw(ioaddr + 0x2000) & 7 : 0); dev->dma = inw(ioaddr + 0x2000) & 7;
vp = (struct corkscrew_private *) dev->priv;
vp->product_name = "3c515"; vp->product_name = "3c515";
vp->options = options; vp->options = dev->mem_start;
if (options >= 0) { vp->our_dev = dev;
vp->media_override = ((options & 7) == 2) ? 0 : options & 7;
vp->full_duplex = (options & 8) ? 1 : 0; if (!vp->options) {
vp->bus_master = (options & 16) ? 1 : 0; if (card_number >= MAX_UNITS)
vp->options = -1;
else
vp->options = options[card_number];
}
if (vp->options >= 0) {
vp->media_override = vp->options & 7;
if (vp->media_override == 2)
vp->media_override = 0;
vp->full_duplex = (vp->options & 8) ? 1 : 0;
vp->bus_master = (vp->options & 16) ? 1 : 0;
} else { } else {
vp->media_override = 7; vp->media_override = 7;
vp->full_duplex = 0; vp->full_duplex = 0;
vp->bus_master = 0; vp->bus_master = 0;
} }
list_add(&vp->list, &root_corkscrew_dev);
corkscrew_probe1(dev);
#endif /* MODULE */
return dev;
err_free_dev:
free_netdev(dev);
err_out:
return NULL;
}
static int corkscrew_probe1(struct net_device *dev)
{
int ioaddr = dev->base_addr;
struct corkscrew_private *vp = (struct corkscrew_private *) dev->priv;
unsigned int eeprom[0x40], checksum = 0; /* EEPROM contents */
int i;
printk(KERN_INFO "%s: 3Com %s at %#3x,", dev->name, vp->product_name, ioaddr); printk(KERN_INFO "%s: 3Com %s at %#3x,", dev->name, vp->product_name, ioaddr);
...@@ -707,9 +677,6 @@ static int corkscrew_probe1(struct net_device *dev) ...@@ -707,9 +677,6 @@ static int corkscrew_probe1(struct net_device *dev)
/* vp->full_bus_master_rx = 0; */ /* vp->full_bus_master_rx = 0; */
vp->full_bus_master_rx = (vp->capabilities & 0x20) ? 1 : 0; vp->full_bus_master_rx = (vp->capabilities & 0x20) ? 1 : 0;
/* We do a request_region() to register /proc/ioports info. */
request_region(ioaddr, CORKSCREW_TOTAL_SIZE, vp->product_name);
/* The 3c51x-specific entries in the device structure. */ /* The 3c51x-specific entries in the device structure. */
dev->open = &corkscrew_open; dev->open = &corkscrew_open;
dev->hard_start_xmit = &corkscrew_start_xmit; dev->hard_start_xmit = &corkscrew_start_xmit;
...@@ -719,8 +686,6 @@ static int corkscrew_probe1(struct net_device *dev) ...@@ -719,8 +686,6 @@ static int corkscrew_probe1(struct net_device *dev)
dev->get_stats = &corkscrew_get_stats; dev->get_stats = &corkscrew_get_stats;
dev->set_multicast_list = &set_rx_mode; dev->set_multicast_list = &set_rx_mode;
dev->ethtool_ops = &netdev_ethtool_ops; dev->ethtool_ops = &netdev_ethtool_ops;
return 0;
} }
...@@ -1608,20 +1573,16 @@ static struct ethtool_ops netdev_ethtool_ops = { ...@@ -1608,20 +1573,16 @@ static struct ethtool_ops netdev_ethtool_ops = {
#ifdef MODULE #ifdef MODULE
void cleanup_module(void) void cleanup_module(void)
{ {
struct net_device *next_dev; while (!list_empty(&root_corkscrew_dev)) {
struct net_device *dev;
while (root_corkscrew_dev) { struct corkscrew_private *vp;
next_dev =
((struct corkscrew_private *) root_corkscrew_dev-> vp = list_entry(root_corkscrew_dev.next,
priv)->next_module; struct corkscrew_private, list);
if (root_corkscrew_dev->dma) dev = vp->our_dev;
free_dma(root_corkscrew_dev->dma); unregister_netdev(dev);
unregister_netdev(root_corkscrew_dev); cleanup_card(dev);
outw(TotalReset, root_corkscrew_dev->base_addr + EL3_CMD); free_netdev(dev);
release_region(root_corkscrew_dev->base_addr,
CORKSCREW_TOTAL_SIZE);
free_netdev(root_corkscrew_dev);
root_corkscrew_dev = next_dev;
} }
} }
#endif /* MODULE */ #endif /* MODULE */
......
...@@ -84,7 +84,7 @@ extern struct net_device *cs89x0_probe(int unit); ...@@ -84,7 +84,7 @@ extern struct net_device *cs89x0_probe(int unit);
extern int hplance_probe(struct net_device *dev); extern int hplance_probe(struct net_device *dev);
extern int bagetlance_probe(struct net_device *); extern int bagetlance_probe(struct net_device *);
extern int mvme147lance_probe(struct net_device *dev); extern int mvme147lance_probe(struct net_device *dev);
extern int tc515_probe(struct net_device *dev); extern struct net_device *tc515_probe(int unit);
extern struct net_device *lance_probe(int unit); extern struct net_device *lance_probe(int unit);
extern int mace_probe(struct net_device *dev); extern int mace_probe(struct net_device *dev);
extern int macsonic_probe(struct net_device *dev); extern int macsonic_probe(struct net_device *dev);
...@@ -201,13 +201,13 @@ static struct devprobe isa_probes[] __initdata = { ...@@ -201,13 +201,13 @@ static struct devprobe isa_probes[] __initdata = {
#ifdef CONFIG_HP100 /* ISA, EISA & PCI */ #ifdef CONFIG_HP100 /* ISA, EISA & PCI */
{hp100_probe, 0}, {hp100_probe, 0},
#endif #endif
#ifdef CONFIG_3C515
{tc515_probe, 0},
#endif
{NULL, 0}, {NULL, 0},
}; };
static struct devprobe2 isa_probes2[] __initdata = { static struct devprobe2 isa_probes2[] __initdata = {
#ifdef CONFIG_3C515
{tc515_probe, 0},
#endif
#ifdef CONFIG_ULTRA #ifdef CONFIG_ULTRA
{ultra_probe, 0}, {ultra_probe, 0},
#endif #endif
......
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