Commit 0d12c687 authored by Finn Thain's avatar Finn Thain Committed by David S. Miller

net/macsonic: Convert to nubus_driver

This resolves an old issue preventing any NuBus SONIC NICs from
working in a Mac with an on-board SONIC device.
Tested-by: default avatarStan Johnson <userm57@yahoo.com>
Signed-off-by: default avatarFinn Thain <fthain@telegraphics.com.au>
Signed-off-by: default avatarDavid S. Miller <davem@davemloft.net>
parent a94cf2a6
...@@ -60,8 +60,6 @@ ...@@ -60,8 +60,6 @@
#include <asm/macints.h> #include <asm/macints.h>
#include <asm/mac_via.h> #include <asm/mac_via.h>
static char mac_sonic_string[] = "macsonic";
#include "sonic.h" #include "sonic.h"
/* These should basically be bus-size and endian independent (since /* These should basically be bus-size and endian independent (since
...@@ -410,7 +408,7 @@ static int mac_onboard_sonic_probe(struct net_device *dev) ...@@ -410,7 +408,7 @@ static int mac_onboard_sonic_probe(struct net_device *dev)
return macsonic_init(dev); return macsonic_init(dev);
} }
static int mac_nubus_sonic_ethernet_addr(struct net_device *dev, static int mac_sonic_nubus_ethernet_addr(struct net_device *dev,
unsigned long prom_addr, int id) unsigned long prom_addr, int id)
{ {
int i; int i;
...@@ -449,70 +447,49 @@ static int macsonic_ident(struct nubus_rsrc *fres) ...@@ -449,70 +447,49 @@ static int macsonic_ident(struct nubus_rsrc *fres)
return -1; return -1;
} }
static int mac_nubus_sonic_probe(struct net_device *dev) static int mac_sonic_nubus_probe_board(struct nubus_board *board, int id,
struct net_device *dev)
{ {
static int slots;
struct nubus_rsrc *ndev = NULL;
struct sonic_local* lp = netdev_priv(dev); struct sonic_local* lp = netdev_priv(dev);
unsigned long base_addr, prom_addr; unsigned long base_addr, prom_addr;
u16 sonic_dcr; u16 sonic_dcr;
int id = -1;
int reg_offset, dma_bitmode; int reg_offset, dma_bitmode;
/* Find the first SONIC that hasn't been initialized already */
for_each_func_rsrc(ndev) {
if (ndev->category != NUBUS_CAT_NETWORK ||
ndev->type != NUBUS_TYPE_ETHERNET)
continue;
/* Have we seen it already? */
if (slots & (1<<ndev->board->slot))
continue;
slots |= 1<<ndev->board->slot;
/* Is it one of ours? */
if ((id = macsonic_ident(ndev)) != -1)
break;
}
if (ndev == NULL)
return -ENODEV;
switch (id) { switch (id) {
case MACSONIC_DUODOCK: case MACSONIC_DUODOCK:
base_addr = ndev->board->slot_addr + DUODOCK_SONIC_REGISTERS; base_addr = board->slot_addr + DUODOCK_SONIC_REGISTERS;
prom_addr = ndev->board->slot_addr + DUODOCK_SONIC_PROM_BASE; prom_addr = board->slot_addr + DUODOCK_SONIC_PROM_BASE;
sonic_dcr = SONIC_DCR_EXBUS | SONIC_DCR_RFT0 | SONIC_DCR_RFT1 | sonic_dcr = SONIC_DCR_EXBUS | SONIC_DCR_RFT0 | SONIC_DCR_RFT1 |
SONIC_DCR_TFT0; SONIC_DCR_TFT0;
reg_offset = 2; reg_offset = 2;
dma_bitmode = SONIC_BITMODE32; dma_bitmode = SONIC_BITMODE32;
break; break;
case MACSONIC_APPLE: case MACSONIC_APPLE:
base_addr = ndev->board->slot_addr + APPLE_SONIC_REGISTERS; base_addr = board->slot_addr + APPLE_SONIC_REGISTERS;
prom_addr = ndev->board->slot_addr + APPLE_SONIC_PROM_BASE; prom_addr = board->slot_addr + APPLE_SONIC_PROM_BASE;
sonic_dcr = SONIC_DCR_BMS | SONIC_DCR_RFT1 | SONIC_DCR_TFT0; sonic_dcr = SONIC_DCR_BMS | SONIC_DCR_RFT1 | SONIC_DCR_TFT0;
reg_offset = 0; reg_offset = 0;
dma_bitmode = SONIC_BITMODE32; dma_bitmode = SONIC_BITMODE32;
break; break;
case MACSONIC_APPLE16: case MACSONIC_APPLE16:
base_addr = ndev->board->slot_addr + APPLE_SONIC_REGISTERS; base_addr = board->slot_addr + APPLE_SONIC_REGISTERS;
prom_addr = ndev->board->slot_addr + APPLE_SONIC_PROM_BASE; prom_addr = board->slot_addr + APPLE_SONIC_PROM_BASE;
sonic_dcr = SONIC_DCR_EXBUS | SONIC_DCR_RFT1 | SONIC_DCR_TFT0 | sonic_dcr = SONIC_DCR_EXBUS | SONIC_DCR_RFT1 | SONIC_DCR_TFT0 |
SONIC_DCR_PO1 | SONIC_DCR_BMS; SONIC_DCR_PO1 | SONIC_DCR_BMS;
reg_offset = 0; reg_offset = 0;
dma_bitmode = SONIC_BITMODE16; dma_bitmode = SONIC_BITMODE16;
break; break;
case MACSONIC_DAYNALINK: case MACSONIC_DAYNALINK:
base_addr = ndev->board->slot_addr + APPLE_SONIC_REGISTERS; base_addr = board->slot_addr + APPLE_SONIC_REGISTERS;
prom_addr = ndev->board->slot_addr + DAYNALINK_PROM_BASE; prom_addr = board->slot_addr + DAYNALINK_PROM_BASE;
sonic_dcr = SONIC_DCR_RFT1 | SONIC_DCR_TFT0 | sonic_dcr = SONIC_DCR_RFT1 | SONIC_DCR_TFT0 |
SONIC_DCR_PO1 | SONIC_DCR_BMS; SONIC_DCR_PO1 | SONIC_DCR_BMS;
reg_offset = 0; reg_offset = 0;
dma_bitmode = SONIC_BITMODE16; dma_bitmode = SONIC_BITMODE16;
break; break;
case MACSONIC_DAYNA: case MACSONIC_DAYNA:
base_addr = ndev->board->slot_addr + DAYNA_SONIC_REGISTERS; base_addr = board->slot_addr + DAYNA_SONIC_REGISTERS;
prom_addr = ndev->board->slot_addr + DAYNA_SONIC_MAC_ADDR; prom_addr = board->slot_addr + DAYNA_SONIC_MAC_ADDR;
sonic_dcr = SONIC_DCR_BMS | sonic_dcr = SONIC_DCR_BMS |
SONIC_DCR_RFT1 | SONIC_DCR_TFT0 | SONIC_DCR_PO1; SONIC_DCR_RFT1 | SONIC_DCR_TFT0 | SONIC_DCR_PO1;
reg_offset = 0; reg_offset = 0;
...@@ -528,14 +505,14 @@ static int mac_nubus_sonic_probe(struct net_device *dev) ...@@ -528,14 +505,14 @@ static int mac_nubus_sonic_probe(struct net_device *dev)
dev->base_addr = base_addr; dev->base_addr = base_addr;
lp->reg_offset = reg_offset; lp->reg_offset = reg_offset;
lp->dma_bitmode = dma_bitmode; lp->dma_bitmode = dma_bitmode;
dev->irq = SLOT2IRQ(ndev->board->slot); dev->irq = SLOT2IRQ(board->slot);
if (!sonic_version_printed) { if (!sonic_version_printed) {
printk(KERN_INFO "%s", version); printk(KERN_INFO "%s", version);
sonic_version_printed = 1; sonic_version_printed = 1;
} }
printk(KERN_INFO "%s: %s in slot %X\n", printk(KERN_INFO "%s: %s in slot %X\n",
dev_name(lp->device), ndev->board->name, ndev->board->slot); dev_name(lp->device), board->name, board->slot);
printk(KERN_INFO "%s: revision 0x%04x, using %d bit DMA and register offset %d\n", printk(KERN_INFO "%s: revision 0x%04x, using %d bit DMA and register offset %d\n",
dev_name(lp->device), SONIC_READ(SONIC_SR), dma_bitmode?32:16, reg_offset); dev_name(lp->device), SONIC_READ(SONIC_SR), dma_bitmode?32:16, reg_offset);
...@@ -557,14 +534,17 @@ static int mac_nubus_sonic_probe(struct net_device *dev) ...@@ -557,14 +534,17 @@ static int mac_nubus_sonic_probe(struct net_device *dev)
SONIC_WRITE(SONIC_ISR, 0x7fff); SONIC_WRITE(SONIC_ISR, 0x7fff);
/* Now look for the MAC address. */ /* Now look for the MAC address. */
if (mac_nubus_sonic_ethernet_addr(dev, prom_addr, id) != 0) if (mac_sonic_nubus_ethernet_addr(dev, prom_addr, id) != 0)
return -ENODEV; return -ENODEV;
dev_info(&board->dev, "SONIC ethernet @%08lx, MAC %pM, IRQ %d\n",
dev->base_addr, dev->dev_addr, dev->irq);
/* Shared init code */ /* Shared init code */
return macsonic_init(dev); return macsonic_init(dev);
} }
static int mac_sonic_probe(struct platform_device *pdev) static int mac_sonic_platform_probe(struct platform_device *pdev)
{ {
struct net_device *dev; struct net_device *dev;
struct sonic_local *lp; struct sonic_local *lp;
...@@ -579,16 +559,10 @@ static int mac_sonic_probe(struct platform_device *pdev) ...@@ -579,16 +559,10 @@ static int mac_sonic_probe(struct platform_device *pdev)
SET_NETDEV_DEV(dev, &pdev->dev); SET_NETDEV_DEV(dev, &pdev->dev);
platform_set_drvdata(pdev, dev); platform_set_drvdata(pdev, dev);
/* This will catch fatal stuff like -ENOMEM as well as success */
err = mac_onboard_sonic_probe(dev); err = mac_onboard_sonic_probe(dev);
if (err == 0)
goto found;
if (err != -ENODEV)
goto out;
err = mac_nubus_sonic_probe(dev);
if (err) if (err)
goto out; goto out;
found:
err = register_netdev(dev); err = register_netdev(dev);
if (err) if (err)
goto out; goto out;
...@@ -610,7 +584,7 @@ MODULE_ALIAS("platform:macsonic"); ...@@ -610,7 +584,7 @@ MODULE_ALIAS("platform:macsonic");
#include "sonic.c" #include "sonic.c"
static int mac_sonic_device_remove(struct platform_device *pdev) static int mac_sonic_platform_remove(struct platform_device *pdev)
{ {
struct net_device *dev = platform_get_drvdata(pdev); struct net_device *dev = platform_get_drvdata(pdev);
struct sonic_local* lp = netdev_priv(dev); struct sonic_local* lp = netdev_priv(dev);
...@@ -623,12 +597,103 @@ static int mac_sonic_device_remove(struct platform_device *pdev) ...@@ -623,12 +597,103 @@ static int mac_sonic_device_remove(struct platform_device *pdev)
return 0; return 0;
} }
static struct platform_driver mac_sonic_driver = { static struct platform_driver mac_sonic_platform_driver = {
.probe = mac_sonic_probe, .probe = mac_sonic_platform_probe,
.remove = mac_sonic_device_remove, .remove = mac_sonic_platform_remove,
.driver = { .driver = {
.name = mac_sonic_string, .name = "macsonic",
},
};
static int mac_sonic_nubus_probe(struct nubus_board *board)
{
struct net_device *ndev;
struct sonic_local *lp;
struct nubus_rsrc *fres;
int id = -1;
int err;
/* The platform driver will handle a PDS or Comm Slot card (even if
* it has a pseudoslot declaration ROM).
*/
if (macintosh_config->expansion_type == MAC_EXP_PDS_COMM)
return -ENODEV;
for_each_board_func_rsrc(board, fres) {
if (fres->category != NUBUS_CAT_NETWORK ||
fres->type != NUBUS_TYPE_ETHERNET)
continue;
id = macsonic_ident(fres);
if (id != -1)
break;
}
if (!fres)
return -ENODEV;
ndev = alloc_etherdev(sizeof(struct sonic_local));
if (!ndev)
return -ENOMEM;
lp = netdev_priv(ndev);
lp->device = &board->dev;
SET_NETDEV_DEV(ndev, &board->dev);
err = mac_sonic_nubus_probe_board(board, id, ndev);
if (err)
goto out;
err = register_netdev(ndev);
if (err)
goto out;
nubus_set_drvdata(board, ndev);
return 0;
out:
free_netdev(ndev);
return err;
}
static int mac_sonic_nubus_remove(struct nubus_board *board)
{
struct net_device *ndev = nubus_get_drvdata(board);
struct sonic_local *lp = netdev_priv(ndev);
unregister_netdev(ndev);
dma_free_coherent(lp->device,
SIZEOF_SONIC_DESC * SONIC_BUS_SCALE(lp->dma_bitmode),
lp->descriptors, lp->descriptors_laddr);
free_netdev(ndev);
return 0;
}
static struct nubus_driver mac_sonic_nubus_driver = {
.probe = mac_sonic_nubus_probe,
.remove = mac_sonic_nubus_remove,
.driver = {
.name = "macsonic-nubus",
.owner = THIS_MODULE,
}, },
}; };
module_platform_driver(mac_sonic_driver); static int perr, nerr;
static int __init mac_sonic_init(void)
{
perr = platform_driver_register(&mac_sonic_platform_driver);
nerr = nubus_driver_register(&mac_sonic_nubus_driver);
return 0;
}
module_init(mac_sonic_init);
static void __exit mac_sonic_exit(void)
{
if (!perr)
platform_driver_unregister(&mac_sonic_platform_driver);
if (!nerr)
nubus_driver_unregister(&mac_sonic_nubus_driver);
}
module_exit(mac_sonic_exit);
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