patch-2.1.88 linux/drivers/acorn/net/etherh.c
Next file: linux/drivers/acorn/net/net-probe.c
Previous file: linux/drivers/acorn/net/ether3.h
Back to the patch index
Back to the overall index
- Lines: 539
- Date:
Mon Feb 16 15:49:48 1998
- Orig file:
v2.1.87/linux/drivers/acorn/net/etherh.c
- Orig date:
Wed Dec 31 16:00:00 1969
diff -u --recursive --new-file v2.1.87/linux/drivers/acorn/net/etherh.c linux/drivers/acorn/net/etherh.c
@@ -0,0 +1,538 @@
+/*
+ * linux/drivers/net/etherh.c
+ *
+ * NS8390 ANT etherh specific driver
+ * For Acorn machines
+ *
+ * By Russell King.
+ *
+ * Changelog:
+ * 08-Dec-1996 RMK 1.00 Created
+ * RMK 1.03 Added support for EtherLan500 cards
+ * 23-Nov-1997 RMK 1.04 Added media autodetection
+ *
+ * Insmod Module Parameters
+ * ------------------------
+ * io=<io_base>
+ * irq=<irqno>
+ * xcvr=<0|1> 0 = 10bT, 1=10b2 (Lan600/600A only)
+ */
+
+#include <linux/module.h>
+#include <linux/kernel.h>
+#include <linux/sched.h>
+#include <linux/types.h>
+#include <linux/fcntl.h>
+#include <linux/interrupt.h>
+#include <linux/ptrace.h>
+#include <linux/ioport.h>
+#include <linux/in.h>
+#include <linux/malloc.h>
+#include <linux/string.h>
+#include <linux/errno.h>
+#include <linux/netdevice.h>
+#include <linux/etherdevice.h>
+#include <linux/skbuff.h>
+
+#include <asm/system.h>
+#include <asm/bitops.h>
+#include <asm/ecard.h>
+#include <asm/delay.h>
+#include <asm/io.h>
+#include <asm/irq.h>
+
+#include "8390.h"
+
+#define NET_DEBUG 0
+#define DEBUG_INIT 2
+
+static unsigned int net_debug = NET_DEBUG;
+static const card_ids etherh_cids[] = {
+ { MANU_I3, PROD_I3_ETHERLAN500 },
+ { MANU_I3, PROD_I3_ETHERLAN600 },
+ { MANU_I3, PROD_I3_ETHERLAN600A },
+ { 0xffff, 0xffff }
+};
+
+static char *version = "etherh [500/600/600A] ethernet driver (c) 1997 R.M.King v1.04\n";
+
+#define ETHERH500_DATAPORT 0x200 /* MEMC */
+#define ETHERH500_NS8390 0x000 /* MEMC */
+#define ETHERH500_CTRLPORT 0x200 /* IOC */
+
+#define ETHERH600_DATAPORT 16 /* MEMC */
+#define ETHERH600_NS8390 0x200 /* MEMC */
+#define ETHERH600_CTRLPORT 0x080 /* MEMC */
+
+#define ETHERH_CP_IE 1
+#define ETHERH_CP_IF 2
+
+#define ETHERH_TX_START_PAGE 1
+#define ETHERH_STOP_PAGE 0x7f
+
+/* --------------------------------------------------------------------------- */
+
+/*
+ * Read the ethernet address string from the on board rom.
+ * This is an ascii string...
+ */
+static int
+etherh_addr (char *addr, struct expansion_card *ec)
+{
+ struct in_chunk_dir cd;
+ char *s;
+
+ if (ecard_readchunk(&cd, ec, 0xf5, 0) && (s = strchr(cd.d.string, '('))) {
+ int i;
+ for (i = 0; i < 6; i++) {
+ addr[i] = simple_strtoul(s + 1, &s, 0x10);
+ if (*s != (i == 5? ')' : ':'))
+ break;
+ }
+ if (i == 6)
+ return 0;
+ }
+ return ENODEV;
+}
+
+/*
+ * Reset the 8390 (hard reset)
+ */
+static void
+etherh_reset (struct device *dev)
+{
+ unsigned int addr = dev->base_addr;
+ int crb;
+
+ if (dev->mem_end == PROD_I3_ETHERLAN600 || dev->mem_end == PROD_I3_ETHERLAN600A) {
+ crb = inb (addr + EN0_RCNTHI) & 0xf8;
+ outb (ei_status.interface_num ? crb | 1: crb, addr+EN0_RCNTHI);
+ }
+}
+
+/*
+ * Write a block of data out to the 8390
+ */
+static void
+etherh_block_output (struct device *dev, int count, const unsigned char *buf, int start_page)
+{
+ unsigned int addr, dma_addr;
+ unsigned long dma_start;
+
+ if (ei_status.dmaing) {
+ printk ("%s: DMAing conflict in etherh_block_input: "
+ " DMAstat %d irqlock %d intr %d\n", dev->name,
+ ei_status.dmaing, ei_status.irqlock, dev->interrupt);
+ return;
+ }
+
+ ei_status.dmaing |= 1;
+
+ addr = dev->base_addr;
+ dma_addr = dev->mem_start;
+
+ count = (count + 1) & ~1;
+ outb (E8390_NODMA | E8390_PAGE0 | E8390_START, addr + E8390_CMD);
+
+ outb (0x42, addr + EN0_RCNTLO);
+ outb (0x00, addr + EN0_RCNTHI);
+ outb (0x42, addr + EN0_RSARLO);
+ outb (0x00, addr + EN0_RSARHI);
+ outb (E8390_RREAD | E8390_START, addr + E8390_CMD);
+
+ udelay (1);
+
+ outb (ENISR_RDC, addr + EN0_ISR);
+ outb (count, addr + EN0_RCNTLO);
+ outb (count >> 8, addr + EN0_RCNTHI);
+ outb (0, addr + EN0_RSARLO);
+ outb (start_page, addr + EN0_RSARHI);
+ outb (E8390_RWRITE | E8390_START, addr + E8390_CMD);
+
+ if (ei_status.word16)
+ outsw (dma_addr, buf, count >> 1);
+#ifdef BIT8
+ else
+ outsb (dma_addr, buf, count);
+#endif
+
+ dma_start = jiffies;
+
+ while ((inb (addr + EN0_ISR) & ENISR_RDC) == 0)
+ if (jiffies - dma_start > 2*HZ/100) { /* 20ms */
+ printk ("%s: timeout waiting for TX RDC\n", dev->name);
+ etherh_reset (dev);
+ NS8390_init (dev, 1);
+ break;
+ }
+
+ outb (ENISR_RDC, addr + EN0_ISR);
+ ei_status.dmaing &= ~1;
+}
+
+/*
+ * Read a block of data from the 8390
+ */
+static void
+etherh_block_input (struct device *dev, int count, struct sk_buff *skb, int ring_offset)
+{
+ unsigned int addr, dma_addr;
+ unsigned char *buf;
+
+ if (ei_status.dmaing) {
+ printk ("%s: DMAing conflict in etherh_block_input: "
+ " DMAstat %d irqlock %d intr %d\n", dev->name,
+ ei_status.dmaing, ei_status.irqlock, dev->interrupt);
+ return;
+ }
+
+ ei_status.dmaing |= 1;
+
+ addr = dev->base_addr;
+ dma_addr = dev->mem_start;
+
+ buf = skb->data;
+ outb (E8390_NODMA | E8390_PAGE0 | E8390_START, addr + E8390_CMD);
+ outb (count, addr + EN0_RCNTLO);
+ outb (count >> 8, addr + EN0_RCNTHI);
+ outb (ring_offset, addr + EN0_RSARLO);
+ outb (ring_offset >> 8, addr + EN0_RSARHI);
+ outb (E8390_RREAD | E8390_START, addr + E8390_CMD);
+
+ if (ei_status.word16) {
+ insw (dma_addr, buf, count >> 1);
+ if (count & 1)
+ buf[count - 1] = inb (dma_addr);
+ }
+#ifdef BIT8
+ else
+ insb (dma_addr, buf, count);
+#endif
+
+ outb (ENISR_RDC, addr + EN0_ISR);
+ ei_status.dmaing &= ~1;
+}
+
+/*
+ * Read a header from the 8390
+ */
+static void
+etherh_get_header (struct device *dev, struct e8390_pkt_hdr *hdr, int ring_page)
+{
+ unsigned int addr, dma_addr;
+
+ if (ei_status.dmaing) {
+ printk ("%s: DMAing conflict in etherh_get_header: "
+ " DMAstat %d irqlock %d intr %d\n", dev->name,
+ ei_status.dmaing, ei_status.irqlock, dev->interrupt);
+ return;
+ }
+
+ ei_status.dmaing |= 1;
+
+ addr = dev->base_addr;
+ dma_addr = dev->mem_start;
+
+ outb (E8390_NODMA | E8390_PAGE0 | E8390_START, addr + E8390_CMD);
+ outb (sizeof (*hdr), addr + EN0_RCNTLO);
+ outb (0, addr + EN0_RCNTHI);
+ outb (0, addr + EN0_RSARLO);
+ outb (ring_page, addr + EN0_RSARHI);
+ outb (E8390_RREAD | E8390_START, addr + E8390_CMD);
+
+ if (ei_status.word16)
+ insw (dma_addr, hdr, sizeof (*hdr) >> 1);
+#ifdef BIT8
+ else
+ insb (dma_addr, hdr, sizeof (*hdr));
+#endif
+
+ outb (ENISR_RDC, addr + EN0_ISR);
+ ei_status.dmaing &= ~1;
+}
+
+/*
+ * Open/initialize the board. This is called (in the current kernel)
+ * sometime after booting when the 'ifconfig' program is run.
+ *
+ * This routine should set everything up anew at each open, even
+ * registers that "should" only need to be set once at boot, so that
+ * there is non-reboot way to recover if something goes wrong.
+ */
+static int
+etherh_open(struct device *dev)
+{
+ unsigned int addr = dev->base_addr;
+ int crb;
+
+ MOD_INC_USE_COUNT;
+
+ if (request_irq(dev->irq, ei_interrupt, 0, "etherh", dev)) {
+ MOD_DEC_USE_COUNT;
+ return -EAGAIN;
+ }
+
+ if (dev->mem_end == PROD_I3_ETHERLAN600 || dev->mem_end == PROD_I3_ETHERLAN600A) {
+ crb = inb (addr + EN0_RCNTHI) & 0xf8;
+ outb (ei_status.interface_num ? crb | 1: crb, addr+EN0_RCNTHI);
+ }
+
+ ei_open (dev);
+ return 0;
+}
+
+/*
+ * The inverse routine to etherh_open().
+ */
+static int
+etherh_close(struct device *dev)
+{
+ ei_close (dev);
+ free_irq (dev->irq, dev);
+
+ MOD_DEC_USE_COUNT;
+ return 0;
+}
+
+/*
+ * This is the real probe routine.
+ */
+static int
+etherh_probe1(struct device *dev)
+{
+ static int version_printed;
+ unsigned int addr, i, reg0, tmp;
+ char *dev_type;
+
+ addr = dev->base_addr;
+
+ if (net_debug && version_printed++ == 0)
+ printk(version);
+
+ switch (dev->mem_end) {
+ case PROD_I3_ETHERLAN500:
+ dev_type = "500 ";
+ break;
+ case PROD_I3_ETHERLAN600:
+ dev_type = "600 ";
+ break;
+ case PROD_I3_ETHERLAN600A:
+ dev_type = "600A ";
+ break;
+ default:
+ dev_type = "";
+ }
+
+ reg0 = inb (addr);
+ if (reg0 == 0xff) {
+ if (net_debug & DEBUG_INIT)
+ printk ("%s: etherh error: NS8390 command register wrong\n", dev->name);
+ return -ENODEV;
+ }
+
+ outb (E8390_NODMA | E8390_PAGE1 | E8390_STOP, addr + E8390_CMD);
+ tmp = inb (addr + 13);
+ outb (0xff, addr + 13);
+ outb (E8390_NODMA | E8390_PAGE0, addr + E8390_CMD);
+ inb (addr + EN0_COUNTER0);
+ if (inb (addr + EN0_COUNTER0) != 0) {
+ if (net_debug & DEBUG_INIT)
+ printk ("%s: etherh error: NS8390 not found\n", dev->name);
+ outb (reg0, addr);
+ outb (tmp, addr + 13);
+ return -ENODEV;
+ }
+
+ if (ethdev_init (dev))
+ return -ENOMEM;
+
+ request_region (addr, 16, "etherh");
+
+ printk("%s: etherh %sfound at %lx, IRQ%d, ether address ",
+ dev->name, dev_type, dev->base_addr, dev->irq);
+
+ for (i = 0; i < 6; i++)
+ printk (i == 5 ? "%2.2x\n" : "%2.2x:", dev->dev_addr[i]);
+
+ ei_status.name = "etherh";
+ ei_status.word16 = 1;
+ ei_status.interface_num = dev->if_port & 1;
+ ei_status.tx_start_page = ETHERH_TX_START_PAGE;
+ ei_status.rx_start_page = ei_status.tx_start_page + TX_PAGES;
+ ei_status.stop_page = ETHERH_STOP_PAGE;
+ ei_status.reset_8390 = etherh_reset;
+ ei_status.block_input = etherh_block_input;
+ ei_status.block_output = etherh_block_output;
+ ei_status.get_8390_hdr = etherh_get_header;
+ dev->open = etherh_open;
+ dev->stop = etherh_close;
+
+ NS8390_init (dev, 0);
+ return 0;
+}
+
+static void etherh_irq_enable (ecard_t *ec, int irqnr)
+{
+ unsigned int ctrl_addr = (unsigned int)ec->irq_data;
+ outb (inb (ctrl_addr) | ETHERH_CP_IE, ctrl_addr);
+}
+
+static void etherh_irq_disable (ecard_t *ec, int irqnr)
+{
+ unsigned int ctrl_addr = (unsigned int)ec->irq_data;
+ outb (inb (ctrl_addr) & ~ETHERH_CP_IE, ctrl_addr);
+}
+
+static expansioncard_ops_t etherh_ops = {
+ etherh_irq_enable,
+ etherh_irq_disable,
+ NULL,
+ NULL
+};
+
+static void etherh_initdev (ecard_t *ec, struct device *dev)
+{
+ ecard_claim (ec);
+
+ dev->irq = ec->irq;
+ dev->mem_end = ec->cld.product;
+
+ switch (ec->cld.product) {
+ case PROD_I3_ETHERLAN500:
+ dev->base_addr = ecard_address (ec, ECARD_MEMC, 0) + ETHERH500_NS8390;
+ dev->mem_start = dev->base_addr + ETHERH500_DATAPORT;
+ ec->irq_data = (void *)ecard_address (ec, ECARD_IOC, ECARD_FAST)
+ + ETHERH500_CTRLPORT;
+ break;
+
+ case PROD_I3_ETHERLAN600:
+ case PROD_I3_ETHERLAN600A:
+ dev->base_addr = ecard_address (ec, ECARD_MEMC, 0) + ETHERH600_NS8390;
+ dev->mem_start = dev->base_addr + ETHERH600_DATAPORT;
+ ec->irq_data = (void *)(dev->base_addr + ETHERH600_CTRLPORT);
+ break;
+
+ default:
+ printk ("%s: etherh error: unknown card type\n", dev->name);
+ }
+ ec->ops = ðerh_ops;
+
+ etherh_addr (dev->dev_addr, ec);
+}
+
+#ifndef MODULE
+int
+etherh_probe(struct device *dev)
+{
+ if (!dev)
+ return ENODEV;
+
+ ecard_startfind ();
+
+ if (!dev->base_addr) {
+ struct expansion_card *ec;
+
+ if ((ec = ecard_find (0, etherh_cids)) == NULL)
+ return ENODEV;
+
+ etherh_initdev (ec, dev);
+ }
+ return etherh_probe1 (dev);
+}
+#endif
+
+#ifdef MODULE
+#define MAX_ETHERH_CARDS 2
+
+static int io[MAX_ETHERH_CARDS];
+static int irq[MAX_ETHERH_CARDS];
+static int xcvr[MAX_ETHERH_CARDS] = { 1, 1 };
+static char ethernames[MAX_ETHERH_CARDS][9];
+static struct device *my_ethers[MAX_ETHERH_CARDS];
+static struct expansion_card *ec[MAX_ETHERH_CARDS];
+
+int
+init_module(void)
+{
+ struct device *dev = NULL;
+ struct expansion_card *boguscards[MAX_ETHERH_CARDS];
+ int i, found = 0;
+
+ for (i = 0; i < MAX_ETHERH_CARDS; i++) {
+ my_ethers[i] = NULL;
+ boguscards[i] = NULL;
+ ec[i] = NULL;
+ strcpy (ethernames[i], " ");
+ }
+
+ ecard_startfind();
+
+ for (i = 0; i < MAX_ETHERH_CARDS; i++) {
+ if (!dev)
+ dev = (struct device *)kmalloc (sizeof (struct device), GFP_KERNEL);
+ if (dev)
+ memset (dev, 0, sizeof (struct device));
+
+ if (!io[i]) {
+ if ((ec[i] = ecard_find (0, etherh_cids)) == NULL)
+ continue;
+ if (!dev)
+ return -ENOMEM;
+
+ etherh_initdev (ec[i], dev);
+ } else {
+ ec[i] = NULL;
+ if (!dev)
+ return -ENOMEM;
+ dev->base_addr = io[i];
+ dev->irq = irq[i];
+ }
+
+ dev->init = etherh_probe1;
+ dev->name = ethernames[i];
+ dev->if_port = xcvr[i];
+
+ my_ethers[i] = dev;
+
+ if (register_netdev (my_ethers[i]) != 0) {
+ printk (KERN_WARNING "No etherh card found at %08lX\n", dev->base_addr);
+ if (ec[i]) {
+ boguscards[i] = ec[i];
+ ec[i] = NULL;
+ }
+ continue;
+ }
+ found ++;
+ dev = NULL;
+ }
+ if (dev)
+ kfree (dev);
+ for (i = 0; i < MAX_ETHERH_CARDS; i++)
+ if (boguscards[i]) {
+ boguscards[i]->ops = NULL;
+ ecard_release (boguscards[i]);
+ }
+ if (!found)
+ return -ENODEV;
+ return 0;
+}
+
+void
+cleanup_module(void)
+{
+ int i;
+ for (i = 0; i < MAX_ETHERH_CARDS; i++) {
+ if (my_ethers[i]) {
+ unregister_netdev(my_ethers[i]);
+ release_region (my_ethers[i]->base_addr, 16);
+ kfree (my_ethers[i]);
+ my_ethers[i] = NULL;
+ }
+ if (ec[i]) {
+ ec[i]->ops = NULL;
+ ecard_release(ec[i]);
+ ec[i] = NULL;
+ }
+ }
+}
+#endif /* MODULE */
FUNET's LINUX-ADM group, linux-adm@nic.funet.fi
TCL-scripts by Sam Shen, slshen@lbl.gov