2076 lines
56 KiB
Diff
2076 lines
56 KiB
Diff
diff -urN --exclude-from=/usr/src/exclude-file linux-2.6.13-i386/drivers/char/16c552.h linux-2.6.13/drivers/char/16c552.h
|
||
--- linux-2.6.13-i386/drivers/char/16c552.h 1970-01-01 01:00:00.000000000 +0100
|
||
+++ linux-2.6.13/drivers/char/16c552.h 2001-10-22 11:34:32.000000000 +0200
|
||
@@ -0,0 +1,165 @@
|
||
+/*
|
||
+ * Definitions for the 16c552 DACE
|
||
+ * (dual-asynchronous-communications-element) used on the GVP
|
||
+ * IO-Extender.
|
||
+ *
|
||
+ * Basically this is two 16c550 uarts's and a parallel port, which is
|
||
+ * why the serial definitions should be valid for the 16c550 uart
|
||
+ * aswell.
|
||
+ *
|
||
+ * Data was taken from National Semiconductors duart 16c552
|
||
+ * data-sheets and the Texas Instruments DACE 16c552 data-sheets (the
|
||
+ * NS version of the chip is _non_ standard and their data-sheets did
|
||
+ * cost me several wasted hours of work).
|
||
+ *
|
||
+ * This file is (C) 1995 Jes Sorensen (jds@kom.auc.dk)
|
||
+ *
|
||
+ * Moved from drivers/char/ to include/linux/, because it's useful
|
||
+ * on more than just the one card. I'm using it on the hp300 DCA
|
||
+ * serial driver, for example.
|
||
+ * -- Peter Maydell <pmaydell@chiark.greenend.org.uk> 05/1998
|
||
+ */
|
||
+
|
||
+#ifndef _16C552_H_
|
||
+#define _16C552_H_
|
||
+
|
||
+/* Serial stuff */
|
||
+
|
||
+struct uart_16c550 {
|
||
+ volatile u_char skip0;
|
||
+ volatile u_char RBR;
|
||
+ volatile u_char skip1;
|
||
+ volatile u_char IER;
|
||
+ volatile u_char skip2;
|
||
+ volatile u_char IIR;
|
||
+ volatile u_char skip3;
|
||
+ volatile u_char LCR;
|
||
+ volatile u_char skip4;
|
||
+ volatile u_char MCR;
|
||
+ volatile u_char skip5;
|
||
+ volatile u_char LSR;
|
||
+ volatile u_char skip6;
|
||
+ volatile u_char MSR;
|
||
+ volatile u_char skip7;
|
||
+ volatile u_char SCR;
|
||
+};
|
||
+
|
||
+#define THR RBR
|
||
+#define FCR IIR
|
||
+#define DLL RBR
|
||
+#define DLM IER
|
||
+#define AFR IIR
|
||
+
|
||
+/*
|
||
+ * Bit-defines for the various registers.
|
||
+ */
|
||
+
|
||
+
|
||
+/* IER */
|
||
+
|
||
+#define ERDAI (1<<0)
|
||
+#define ETHREI (1<<1)
|
||
+#define ELSI (1<<2)
|
||
+#define EMSI (1<<3)
|
||
+
|
||
+/* IIR - Interrupt Ident. Register */
|
||
+
|
||
+#define IRQ_PEND (1<<0) /* NOTE: IRQ_PEND=0 implies irq pending */
|
||
+#define IRQ_ID1 (1<<1)
|
||
+#define IRQ_ID2 (1<<2)
|
||
+#define IRQ_ID3 (1<<3)
|
||
+#define FIFO_ENA0 (1<<6) /* Both these are set when FCR(1<<0)=1 */
|
||
+#define FIFO_ENA1 (1<<7)
|
||
+
|
||
+#define IRQ_RLS (IRQ_ID1 | IRQ_ID2)
|
||
+#define IRQ_RDA (IRQ_ID2)
|
||
+#define IRQ_CTI (IRQ_ID2 | IRQ_ID3)
|
||
+#define IRQ_THRE (IRQ_ID1)
|
||
+#define IRQ_MS 0
|
||
+
|
||
+/* FCR - FIFO Control Register */
|
||
+
|
||
+#define FIFO_ENA (1<<0)
|
||
+#define RCVR_FIFO_RES (1<<1)
|
||
+#define XMIT_FIFO_RES (1<<2)
|
||
+#define DMA_MODE_SEL (1<<3)
|
||
+#define RCVR_TRIG_LSB (1<<6)
|
||
+#define RCVR_TRIG_MSB (1<<7)
|
||
+
|
||
+#define FIFO_TRIG_1 0x00
|
||
+#define FIFO_TRIG_4 RCVR_TRIG_LSB
|
||
+#define FIFO_TRIG_8 RCVR_TRIG_MSB
|
||
+#define FIFO_TRIG_14 RCVR_TRIG_LSB|RCVR_TRIG_MSB
|
||
+
|
||
+/* LCR - Line Control Register */
|
||
+
|
||
+#define WLS0 (1<<0)
|
||
+#define WLS1 (1<<1)
|
||
+#define STB (1<<2)
|
||
+#define PEN (1<<3)
|
||
+#define EPS (1<<4)
|
||
+#define STICK_PARITY (1<<5)
|
||
+#define SET_BREAK (1<<6)
|
||
+#define DLAB (1<<7)
|
||
+
|
||
+#define data_5bit 0x00
|
||
+#define data_6bit 0x01
|
||
+#define data_7bit 0x02
|
||
+#define data_8bit 0x03
|
||
+
|
||
+
|
||
+/* MCR - Modem Control Register */
|
||
+
|
||
+#define DTR (1<<0)
|
||
+#define RTS (1<<1)
|
||
+#define OUT1 (1<<2)
|
||
+#define OUT2 (1<<3)
|
||
+#define LOOP (1<<4)
|
||
+
|
||
+/* LSR - Line Status Register */
|
||
+
|
||
+#define DR (1<<0)
|
||
+#define OE (1<<1)
|
||
+#define PE (1<<2)
|
||
+#define FE (1<<3)
|
||
+#define BI (1<<4)
|
||
+#define THRE (1<<5)
|
||
+#define TEMT (1<<6)
|
||
+#define RCVR_FIFO_ERR (1<<7)
|
||
+
|
||
+/* MSR - Modem Status Register */
|
||
+
|
||
+#define DCTS (1<<0)
|
||
+#define DDSR (1<<1)
|
||
+#define TERI (1<<2)
|
||
+#define DDCD (1<<3)
|
||
+#define CTS (1<<4)
|
||
+#define DSR (1<<5)
|
||
+#define RING_I (1<<6)
|
||
+#define DCD (1<<7)
|
||
+
|
||
+/* AFR - Alternate Function Register */
|
||
+
|
||
+#define CONCUR_WRITE (1<<0)
|
||
+#define BAUDOUT (1<<1)
|
||
+#define RXRDY (1<<2)
|
||
+
|
||
+/* Parallel stuff */
|
||
+
|
||
+/*
|
||
+ * Unfortunately National Semiconductors did not supply the
|
||
+ * specifications for the parallel port in the chip :-(
|
||
+ * TI succed though, so here they are :-)
|
||
+ *
|
||
+ * Defines for the bits can be found by including <linux/lp.h>
|
||
+ */
|
||
+struct IOEXT_par {
|
||
+ volatile u_char skip0;
|
||
+ volatile u_char DATA;
|
||
+ volatile u_char skip1;
|
||
+ volatile u_char STATUS;
|
||
+ volatile u_char skip2;
|
||
+ volatile u_char CTRL;
|
||
+};
|
||
+
|
||
+#endif
|
||
diff -urN --exclude-from=/usr/src/exclude-file linux-2.6.13-i386/drivers/char/ioext.h linux-2.6.13/drivers/char/ioext.h
|
||
--- linux-2.6.13-i386/drivers/char/ioext.h 1970-01-01 01:00:00.000000000 +0100
|
||
+++ linux-2.6.13/drivers/char/ioext.h 2001-10-22 11:34:32.000000000 +0200
|
||
@@ -0,0 +1,108 @@
|
||
+/*
|
||
+ * Shared data structure for GVP IO-Extender support.
|
||
+ *
|
||
+ * Merge of ioext.h and ser_ioext.h
|
||
+ */
|
||
+#ifndef _IOEXT_H_
|
||
+#define _IOEXT_H_
|
||
+
|
||
+#include <linux/config.h>
|
||
+#include <linux/netdevice.h>
|
||
+
|
||
+#include "16c552.h"
|
||
+
|
||
+#define MAX_IOEXT 5 /*
|
||
+ * The maximum number of io-extenders is 5, as you
|
||
+ * can't have more than 5 ZII boards in any Amiga.
|
||
+ */
|
||
+
|
||
+#define UART_CLK 7372800
|
||
+
|
||
+#define IOEXT_BAUD_BASE (UART_CLK / 16)
|
||
+
|
||
+#define IOEXT_MAX_LINES 2
|
||
+
|
||
+#define IOEXT_PAR_PLIP 0x0001
|
||
+#define IOEXT_PAR_LP 0x0002
|
||
+
|
||
+
|
||
+/*
|
||
+ * Macros for the serial driver.
|
||
+ */
|
||
+#define curruart(info) ((struct uart_16c550 *)(info->port))
|
||
+
|
||
+#define ser_DTRon(info) curruart(info)->MCR |= DTR
|
||
+#define ser_RTSon(info) curruart(info)->MCR |= RTS
|
||
+#define ser_DTRoff(info) curruart(info)->MCR &= ~DTR
|
||
+#define ser_RTSoff(info) curruart(info)->MCR &= ~RTS
|
||
+
|
||
+
|
||
+/*
|
||
+ * CNTR defines (copied from the GVP SCSI-driver file gvp11.h
|
||
+ */
|
||
+#define GVP_BUSY (1<<0)
|
||
+#define GVP_IRQ_PEND (1<<1)
|
||
+#define GVP_IRQ_ENA (1<<3)
|
||
+#define GVP_DIR_WRITE (1<<4)
|
||
+
|
||
+
|
||
+/*
|
||
+ * CTRL defines
|
||
+ */
|
||
+#define PORT0_MIDI (1<<0) /* CLR = DRIVERS SET = MIDI */
|
||
+#define PORT1_MIDI (1<<1) /* CLR = DRIVERS SET = MIDI */
|
||
+#define PORT0_DRIVER (1<<2) /* CLR = RS232, SET = MIDI */
|
||
+#define PORT1_DRIVER (1<<3) /* CLR = RS232, SET = MIDI */
|
||
+#define IRQ_SEL (1<<4) /* CLR = INT2, SET = INT6 */
|
||
+#define ROM_BANK_SEL (1<<5) /* CLR = LOW 32K, SET = HIGH 32K */
|
||
+#define PORT0_CTRL (1<<6) /* CLR = RTSx or RXRDYx, SET = RTSx ONLY */
|
||
+#define PORT1_CTRL (1<<7) /* CLR = RTSx or RXRDYx, SET = RTSx ONLY */
|
||
+
|
||
+
|
||
+/*
|
||
+ * This is the struct describing the registers on the IO-Extender.
|
||
+ * NOTE: The board uses a dual uart (16c552), which should be equal to
|
||
+ * two 16c550 uarts.
|
||
+ */
|
||
+typedef struct {
|
||
+ char gap0[0x41];
|
||
+ volatile unsigned char CNTR; /* GVP DMAC CNTR (status register) */
|
||
+ char gap1[0x11e];
|
||
+ struct uart_16c550 uart0; /* The first uart */
|
||
+ char gap2[0xf0];
|
||
+ struct uart_16c550 uart1; /* The second uart */
|
||
+ char gap3[0xf0];
|
||
+ struct IOEXT_par par; /* The parallel port */
|
||
+ char gap4[0xfb];
|
||
+ volatile unsigned char CTRL; /* The control-register on the board */
|
||
+} IOEXT_struct;
|
||
+
|
||
+
|
||
+typedef struct {
|
||
+ int num_uarts;
|
||
+ int line[IOEXT_MAX_LINES];
|
||
+ volatile struct uart_16c550 *uart[IOEXT_MAX_LINES];
|
||
+ IOEXT_struct *board;
|
||
+ int spurious_count;
|
||
+ unsigned char par_use; /* IOEXT_PAR_xxx */
|
||
+#if defined(CONFIG_GVPIOEXT_PLIP) || defined(CONFIG_GVPIOEXT_PLIP_MODULE)
|
||
+ struct nt_device *dev;
|
||
+#endif
|
||
+#if defined(CONFIG_GVPIOEXT_LP) || defined(CONFIG_GVPIOEXT_LP_MODULE)
|
||
+ struct lp_struct *lp_table;
|
||
+ int lp_dev;
|
||
+ int lp_interrupt;
|
||
+#endif
|
||
+} IOExtInfoType;
|
||
+
|
||
+/* Number of detected boards. */
|
||
+extern int ioext_num;
|
||
+extern IOExtInfoType ioext_info[MAX_IOEXT];
|
||
+
|
||
+void ioext_plip_interrupt(struct net_device *dev, int *spurious_count);
|
||
+void ioext_lp_interrupt(int dev, int *spurious_count);
|
||
+
|
||
+extern struct net_device ioext_dev_plip[3];
|
||
+extern struct lp_struct ioext_lp_table[1];
|
||
+
|
||
+#endif
|
||
diff -urN --exclude-from=/usr/src/exclude-file linux-2.6.13-i386/drivers/char/mc68681.h linux-2.6.13/drivers/char/mc68681.h
|
||
--- linux-2.6.13-i386/drivers/char/mc68681.h 1970-01-01 01:00:00.000000000 +0100
|
||
+++ linux-2.6.13/drivers/char/mc68681.h 2001-10-22 11:34:32.000000000 +0200
|
||
@@ -0,0 +1,131 @@
|
||
+#ifndef _MC68681_H_
|
||
+#define _MC68681_H_
|
||
+
|
||
+/*
|
||
+ * This describes an MC68681 DUART. It has almost only overlayed registers, which
|
||
+ * the structure very ugly.
|
||
+ * Note that the ri-register isn't really a register of the duart but a kludge of bsc
|
||
+ * to make the ring indicator available.
|
||
+ *
|
||
+ * The data came from the MFC-31-Developer Kit (from Ralph Seidel,
|
||
+ * zodiac@darkness.gun.de) and the data sheet of Phillip's clone device (SCN68681)
|
||
+ * (from Richard Hirst, srh@gpt.co.uk)
|
||
+ *
|
||
+ * 11.11.95 copyright Joerg Dorchain (dorchain@mpi-sb.mpg.de)
|
||
+ *
|
||
+ */
|
||
+
|
||
+struct duarthalf {
|
||
+union {
|
||
+volatile u_char mr1; /* rw */
|
||
+volatile u_char mr2; /* rw */
|
||
+} mr;
|
||
+volatile u_char ri; /* special, read */
|
||
+union {
|
||
+volatile u_char sr; /* read */
|
||
+volatile u_char csr; /* write */
|
||
+} sr_csr;
|
||
+u_char pad1;
|
||
+volatile u_char cr; /* write */
|
||
+u_char pad2;
|
||
+union {
|
||
+volatile u_char rhr; /* read */
|
||
+volatile u_char thr; /* write */
|
||
+} hr;
|
||
+u_char pad3;
|
||
+};
|
||
+
|
||
+struct duart {
|
||
+struct duarthalf pa;
|
||
+union {
|
||
+volatile u_char ipcr; /* read */
|
||
+volatile u_char acr; /* write */
|
||
+} ipcr_acr;
|
||
+u_char pad1;
|
||
+union {
|
||
+volatile u_char isr; /* read */
|
||
+volatile u_char imr; /* write */
|
||
+} ir;
|
||
+u_char pad2;
|
||
+volatile u_char ctu;
|
||
+u_char pad3;
|
||
+volatile u_char ctl;
|
||
+u_char pad4;
|
||
+struct duarthalf pb;
|
||
+volatile u_char ivr;
|
||
+u_char pad5;
|
||
+union {
|
||
+volatile u_char ipr; /* read */
|
||
+volatile u_char opcr; /* write */
|
||
+} ipr_opcr;
|
||
+u_char pad6;
|
||
+union {
|
||
+volatile u_char start; /* read */
|
||
+volatile u_char sopc; /* write */
|
||
+} start_sopc;
|
||
+u_char pad7;
|
||
+union {
|
||
+volatile u_char stop; /* read */
|
||
+volatile u_char ropc; /* write */
|
||
+} stop_ropc;
|
||
+u_char pad8;
|
||
+};
|
||
+
|
||
+#define MR1_BITS 3
|
||
+#define MR1_5BITS 0
|
||
+#define MR1_6BITS 1
|
||
+#define MR1_7BITS 2
|
||
+#define MR1_8BITS 3
|
||
+
|
||
+#define MR1_PARITY_ODD 4
|
||
+
|
||
+#define MR1_PARITY 24
|
||
+#define MR1_PARITY_WITH 0
|
||
+#define MR1_PARITY_FORCE 8
|
||
+#define MR1_PARITY_NO 16
|
||
+#define MR1_PARITY_MULTIDROP 24
|
||
+
|
||
+#define MR1_ERROR_BLOCK 32
|
||
+#define MR1_FFULL_IRQ 64
|
||
+#define MR1_RxRTS_ON 128
|
||
+
|
||
+#define MR2_STOPS 15
|
||
+#define MR2_1STOP 7
|
||
+#define MR2_2STOP 15
|
||
+
|
||
+#define MR2_CTS_ON 16
|
||
+#define MR2_TxRTS_ON 32
|
||
+
|
||
+#define MR2_MODE 192
|
||
+#define MR2_NORMAL 0
|
||
+#define MR2_ECHO 64
|
||
+#define MR2_LOCALLOOP 128
|
||
+#define MR2_REMOTELOOP 192
|
||
+
|
||
+#define CR_RXCOMMAND 3
|
||
+#define CR_NONE 0
|
||
+#define CR_RX_ON 1
|
||
+#define CR_RX_OFF 2
|
||
+#define CR_TXCOMMAND 12
|
||
+#define CR_TX_ON 4
|
||
+#define CR_TX_OFF 8
|
||
+#define CR_MISC 112
|
||
+#define CR_RESET_MR 16
|
||
+#define CR_RESET_RX 32
|
||
+#define CR_RESET_TX 48
|
||
+#define CR_RESET_ERR 64
|
||
+#define CR_RESET_BREAK 80
|
||
+#define CR_START_BREAK 96
|
||
+#define CR_STOP_BREAK 112
|
||
+
|
||
+#define SR_RXRDY 1
|
||
+#define SR_FFULL 2
|
||
+#define SR_TXRDY 4
|
||
+#define SR_TXEMPT 8
|
||
+#define SR_OVERRUN 16
|
||
+#define SR_PARITY 32
|
||
+#define SR_FRAMING 64
|
||
+#define SR_BREAK 128
|
||
+
|
||
+
|
||
+#endif
|
||
diff -urN --exclude-from=/usr/src/exclude-file linux-2.6.13-i386/drivers/char/plip_ioext.c linux-2.6.13/drivers/char/plip_ioext.c
|
||
--- linux-2.6.13-i386/drivers/char/plip_ioext.c 1970-01-01 01:00:00.000000000 +0100
|
||
+++ linux-2.6.13/drivers/char/plip_ioext.c 2004-10-25 16:38:25.000000000 +0200
|
||
@@ -0,0 +1,1058 @@
|
||
+/*
|
||
+ * plip_ioext: A parallel port "network" driver for GVP IO-Extender.
|
||
+ *
|
||
+ * Authors: See drivers/net/plip.c
|
||
+ * IO-Extender version by Steve Bennett, <msteveb@ozemail.com.au>
|
||
+ *
|
||
+ * This driver is for use with a 5-bit cable (LapLink (R) cable).
|
||
+ */
|
||
+
|
||
+static const char *version = "NET3 PLIP version 2.2/m68k";
|
||
+
|
||
+#define __NO_VERSION__
|
||
+
|
||
+#include <linux/module.h>
|
||
+#include <linux/types.h>
|
||
+#include <linux/sched.h>
|
||
+#include <linux/errno.h>
|
||
+#include <linux/interrupt.h>
|
||
+#include <linux/slab.h>
|
||
+#include <linux/termios.h>
|
||
+#include <linux/tty.h>
|
||
+#include <linux/serial.h>
|
||
+
|
||
+#include <asm/setup.h>
|
||
+#include <asm/irq.h>
|
||
+#include <asm/amigahw.h>
|
||
+#include <asm/amigaints.h>
|
||
+#include <linux/zorro.h>
|
||
+
|
||
+#include <linux/config.h>
|
||
+#include <linux/kernel.h>
|
||
+#include <linux/fcntl.h>
|
||
+#include <linux/string.h>
|
||
+#include <linux/ptrace.h>
|
||
+#include <linux/if_ether.h>
|
||
+
|
||
+#include <asm/system.h>
|
||
+
|
||
+#include <linux/in.h>
|
||
+#include <linux/delay.h>
|
||
+/*#include <linux/lp_m68k.h>*/
|
||
+
|
||
+#include <linux/netdevice.h>
|
||
+#include <linux/etherdevice.h>
|
||
+#include <linux/inetdevice.h>
|
||
+#include <linux/skbuff.h>
|
||
+#include <linux/if_plip.h>
|
||
+
|
||
+#include <linux/tqueue.h>
|
||
+#include <linux/ioport.h>
|
||
+#include <linux/bitops.h>
|
||
+#include <asm/byteorder.h>
|
||
+
|
||
+#include "ioext.h"
|
||
+
|
||
+#define DEBUG 0
|
||
+
|
||
+/* Map 'struct device *' to our control structure */
|
||
+#define PLIP_DEV(DEV) (&ioext_info[(DEV)->irq])
|
||
+
|
||
+/************************************************************************
|
||
+**
|
||
+** PLIP definitions
|
||
+**
|
||
+*************************************************************************
|
||
+*/
|
||
+
|
||
+/* Use 0 for production, 1 for verification, >2 for debug */
|
||
+#ifndef NET_DEBUG
|
||
+#define NET_DEBUG 2
|
||
+#endif
|
||
+static unsigned int net_debug = NET_DEBUG;
|
||
+
|
||
+/* In micro second */
|
||
+#define PLIP_DELAY_UNIT 1
|
||
+
|
||
+/* Connection time out = PLIP_TRIGGER_WAIT * PLIP_DELAY_UNIT usec */
|
||
+#define PLIP_TRIGGER_WAIT 500
|
||
+
|
||
+/* Nibble time out = PLIP_NIBBLE_WAIT * PLIP_DELAY_UNIT usec */
|
||
+#define PLIP_NIBBLE_WAIT 3000
|
||
+
|
||
+#define PAR_DATA(dev) ((dev)->base_addr+0)
|
||
+#define PAR_STATUS(dev) ((dev)->base_addr+2)
|
||
+#define PAR_CONTROL(dev) ((dev)->base_addr+4)
|
||
+
|
||
+static void enable_par_irq(struct device *dev, int on);
|
||
+static int plip_init(struct device *dev);
|
||
+
|
||
+/* Bottom halfs */
|
||
+static void plip_kick_bh(struct device *dev);
|
||
+static void plip_bh(struct device *dev);
|
||
+
|
||
+/* Functions for DEV methods */
|
||
+static int plip_rebuild_header(struct sk_buff *skb);
|
||
+static int plip_tx_packet(struct sk_buff *skb, struct device *dev);
|
||
+static int plip_open(struct device *dev);
|
||
+static int plip_close(struct device *dev);
|
||
+static struct enet_statistics *plip_get_stats(struct device *dev);
|
||
+static int plip_config(struct device *dev, struct ifmap *map);
|
||
+static int plip_ioctl(struct device *dev, struct ifreq *ifr, int cmd);
|
||
+
|
||
+enum plip_connection_state {
|
||
+ PLIP_CN_NONE=0,
|
||
+ PLIP_CN_RECEIVE,
|
||
+ PLIP_CN_SEND,
|
||
+ PLIP_CN_CLOSING,
|
||
+ PLIP_CN_ERROR
|
||
+};
|
||
+
|
||
+enum plip_packet_state {
|
||
+ PLIP_PK_DONE=0,
|
||
+ PLIP_PK_TRIGGER,
|
||
+ PLIP_PK_LENGTH_LSB,
|
||
+ PLIP_PK_LENGTH_MSB,
|
||
+ PLIP_PK_DATA,
|
||
+ PLIP_PK_CHECKSUM
|
||
+};
|
||
+
|
||
+enum plip_nibble_state {
|
||
+ PLIP_NB_BEGIN,
|
||
+ PLIP_NB_1,
|
||
+ PLIP_NB_2,
|
||
+};
|
||
+
|
||
+struct plip_local {
|
||
+ enum plip_packet_state state;
|
||
+ enum plip_nibble_state nibble;
|
||
+ union {
|
||
+ struct {
|
||
+#if defined(__LITTLE_ENDIAN)
|
||
+ unsigned char lsb;
|
||
+ unsigned char msb;
|
||
+#elif defined(__BIG_ENDIAN)
|
||
+ unsigned char msb;
|
||
+ unsigned char lsb;
|
||
+#else
|
||
+#error "Please fix the endianness defines in <asm/byteorder.h>"
|
||
+#endif
|
||
+ } b;
|
||
+ unsigned short h;
|
||
+ } length;
|
||
+ unsigned short byte;
|
||
+ unsigned char checksum;
|
||
+ unsigned char data;
|
||
+ struct sk_buff *skb;
|
||
+};
|
||
+
|
||
+struct net_local {
|
||
+ struct enet_statistics enet_stats;
|
||
+ struct tq_struct immediate;
|
||
+ struct tq_struct deferred;
|
||
+ struct plip_local snd_data;
|
||
+ struct plip_local rcv_data;
|
||
+ unsigned long trigger;
|
||
+ unsigned long nibble;
|
||
+ enum plip_connection_state connection;
|
||
+ unsigned short timeout_count;
|
||
+ char is_deferred;
|
||
+ int (*orig_rebuild_header)(struct sk_buff *skb);
|
||
+};
|
||
+
|
||
+struct device ioext_dev_plip[] = {
|
||
+ {
|
||
+ "plip0",
|
||
+ 0, 0, 0, 0, /* memory */
|
||
+ 0, 0, /* base, irq */
|
||
+ 0, 0, 0, NULL, plip_init
|
||
+ },
|
||
+ {
|
||
+ "plip1",
|
||
+ 0, 0, 0, 0, /* memory */
|
||
+ 0, 0, /* base, irq */
|
||
+ 0, 0, 0, NULL, plip_init
|
||
+ },
|
||
+ {
|
||
+ "plip2",
|
||
+ 0, 0, 0, 0, /* memory */
|
||
+ 0, 0, /* base, irq */
|
||
+ 0, 0, 0, NULL, plip_init
|
||
+ }
|
||
+};
|
||
+
|
||
+/*
|
||
+ * Check for and handle an interrupt for this PLIP device.
|
||
+ *
|
||
+ */
|
||
+void ioext_plip_interrupt(struct device *dev, int *spurious_count)
|
||
+{
|
||
+ struct net_local *nl;
|
||
+ struct plip_local *rcv;
|
||
+ unsigned char c0;
|
||
+ unsigned long flags;
|
||
+
|
||
+ nl = (struct net_local *)dev->priv;
|
||
+ rcv = &nl->rcv_data;
|
||
+
|
||
+ c0 = z_readb(PAR_STATUS(dev));
|
||
+
|
||
+ if (dev->interrupt) {
|
||
+ return;
|
||
+ }
|
||
+
|
||
+ if ((c0 & 0xf8) != 0xc0) {
|
||
+ /* Not for us */
|
||
+ ++*spurious_count;
|
||
+ return;
|
||
+ }
|
||
+
|
||
+ *spurious_count = 0;
|
||
+ dev->interrupt = 1;
|
||
+
|
||
+ local_irq_save(flags);
|
||
+
|
||
+ switch (nl->connection) {
|
||
+ case PLIP_CN_CLOSING:
|
||
+ dev->tbusy = 0;
|
||
+ case PLIP_CN_NONE:
|
||
+ case PLIP_CN_SEND:
|
||
+ dev->last_rx = jiffies;
|
||
+ rcv->state = PLIP_PK_TRIGGER;
|
||
+ nl->connection = PLIP_CN_RECEIVE;
|
||
+ nl->timeout_count = 0;
|
||
+ queue_task(&nl->immediate, &tq_immediate);
|
||
+ mark_bh(IMMEDIATE_BH);
|
||
+ local_irq_restore(flags);
|
||
+#if 0
|
||
+ printk("%s: receive irq in SEND/NONE/CLOSING (%d) ok\n",
|
||
+ dev->name, nl->connection);
|
||
+#endif
|
||
+ break;
|
||
+
|
||
+ case PLIP_CN_RECEIVE:
|
||
+ local_irq_restore(flags);
|
||
+ printk("%s: receive interrupt when receiving packet\n",
|
||
+ dev->name);
|
||
+ break;
|
||
+
|
||
+ case PLIP_CN_ERROR:
|
||
+ local_irq_restore(flags);
|
||
+ printk("%s: receive interrupt in error state\n", dev->name);
|
||
+ break;
|
||
+ }
|
||
+}
|
||
+
|
||
+
|
||
+/* Bottom half handler for the delayed request.
|
||
+ This routine is kicked by do_timer().
|
||
+ Request `plip_bh' to be invoked. */
|
||
+static void
|
||
+plip_kick_bh(struct device *dev)
|
||
+{
|
||
+ struct net_local *nl = (struct net_local *)dev->priv;
|
||
+
|
||
+ if (nl->is_deferred) {
|
||
+ queue_task(&nl->immediate, &tq_immediate);
|
||
+ mark_bh(IMMEDIATE_BH);
|
||
+ }
|
||
+}
|
||
+
|
||
+/* Forward declarations of internal routines */
|
||
+static int plip_none(struct device *, struct net_local *,
|
||
+ struct plip_local *, struct plip_local *);
|
||
+static int plip_receive_packet(struct device *, struct net_local *,
|
||
+ struct plip_local *, struct plip_local *);
|
||
+static int plip_send_packet(struct device *, struct net_local *,
|
||
+ struct plip_local *, struct plip_local *);
|
||
+static int plip_connection_close(struct device *, struct net_local *,
|
||
+ struct plip_local *, struct plip_local *);
|
||
+static int plip_error(struct device *, struct net_local *,
|
||
+ struct plip_local *, struct plip_local *);
|
||
+static int plip_bh_timeout_error(struct device *dev, struct net_local *nl,
|
||
+ struct plip_local *snd,
|
||
+ struct plip_local *rcv,
|
||
+ int error);
|
||
+
|
||
+#define OK 0
|
||
+#define TIMEOUT 1
|
||
+#define ERROR 2
|
||
+
|
||
+typedef int (*plip_func)(struct device *dev, struct net_local *nl,
|
||
+ struct plip_local *snd, struct plip_local *rcv);
|
||
+
|
||
+static plip_func connection_state_table[] =
|
||
+{
|
||
+ plip_none,
|
||
+ plip_receive_packet,
|
||
+ plip_send_packet,
|
||
+ plip_connection_close,
|
||
+ plip_error
|
||
+};
|
||
+
|
||
+/*
|
||
+** enable_par_irq()
|
||
+**
|
||
+** Enable or disable parallel irq for 'dev' according to 'on'.
|
||
+**
|
||
+** It is NOT possible to disable only the parallel irq.
|
||
+** So we disable the board interrupt instead. This means that
|
||
+** during reception of a PLIP packet, no serial interrupts can
|
||
+** happen. Sorry.
|
||
+*/
|
||
+static void enable_par_irq(struct device *dev, int on)
|
||
+{
|
||
+ if (on) {
|
||
+ PLIP_DEV(dev)->board->CNTR |= GVP_IRQ_ENA;
|
||
+ }
|
||
+ else {
|
||
+ PLIP_DEV(dev)->board->CNTR &= ~GVP_IRQ_ENA;
|
||
+ }
|
||
+}
|
||
+
|
||
+/* Bottom half handler of PLIP. */
|
||
+static void
|
||
+plip_bh(struct device *dev)
|
||
+{
|
||
+ struct net_local *nl = (struct net_local *)dev->priv;
|
||
+ struct plip_local *snd = &nl->snd_data;
|
||
+ struct plip_local *rcv = &nl->rcv_data;
|
||
+ plip_func f;
|
||
+ int r;
|
||
+
|
||
+ nl->is_deferred = 0;
|
||
+ f = connection_state_table[nl->connection];
|
||
+ if ((r = (*f)(dev, nl, snd, rcv)) != OK
|
||
+ && (r = plip_bh_timeout_error(dev, nl, snd, rcv, r)) != OK) {
|
||
+ nl->is_deferred = 1;
|
||
+ queue_task(&nl->deferred, &tq_timer);
|
||
+ }
|
||
+}
|
||
+
|
||
+static int
|
||
+plip_bh_timeout_error(struct device *dev, struct net_local *nl,
|
||
+ struct plip_local *snd, struct plip_local *rcv,
|
||
+ int error)
|
||
+{
|
||
+ unsigned char c0;
|
||
+ unsigned long flags;
|
||
+
|
||
+ local_irq_save(flags);
|
||
+ if (nl->connection == PLIP_CN_SEND) {
|
||
+
|
||
+ if (error != ERROR) { /* Timeout */
|
||
+ nl->timeout_count++;
|
||
+ if ((snd->state == PLIP_PK_TRIGGER
|
||
+ && nl->timeout_count <= 10)
|
||
+ || nl->timeout_count <= 3) {
|
||
+ local_irq_restore(flags);
|
||
+ /* Try again later */
|
||
+ return TIMEOUT;
|
||
+ }
|
||
+ c0 = z_readb(PAR_STATUS(dev));
|
||
+ printk(KERN_INFO "%s: transmit timeout(%d,%02x)\n",
|
||
+ dev->name, snd->state, c0);
|
||
+ }
|
||
+ nl->enet_stats.tx_errors++;
|
||
+ nl->enet_stats.tx_aborted_errors++;
|
||
+ } else if (nl->connection == PLIP_CN_RECEIVE) {
|
||
+ if (rcv->state == PLIP_PK_TRIGGER) {
|
||
+ /* Transmission was interrupted. */
|
||
+ local_irq_restore(flags);
|
||
+ return OK;
|
||
+ }
|
||
+ if (error != ERROR) { /* Timeout */
|
||
+ if (++nl->timeout_count <= 3) {
|
||
+ local_irq_restore(flags);
|
||
+ /* Try again later */
|
||
+ return TIMEOUT;
|
||
+ }
|
||
+ c0 = z_readb(PAR_STATUS(dev));
|
||
+ printk(KERN_INFO "%s: receive timeout(%d,%02x)\n",
|
||
+ dev->name, rcv->state, c0);
|
||
+ }
|
||
+ nl->enet_stats.rx_dropped++;
|
||
+ }
|
||
+ rcv->state = PLIP_PK_DONE;
|
||
+ if (rcv->skb) {
|
||
+ kfree_skb(rcv->skb);
|
||
+ rcv->skb = NULL;
|
||
+ }
|
||
+ snd->state = PLIP_PK_DONE;
|
||
+ if (snd->skb) {
|
||
+ dev_kfree_skb(snd->skb);
|
||
+ snd->skb = NULL;
|
||
+ }
|
||
+ enable_par_irq(dev, 0);
|
||
+ dev->tbusy = 1;
|
||
+ nl->connection = PLIP_CN_ERROR;
|
||
+ z_writeb(0x00, PAR_DATA(dev));
|
||
+ local_irq_restore(flags);
|
||
+
|
||
+ return TIMEOUT;
|
||
+}
|
||
+
|
||
+static int
|
||
+plip_none(struct device *dev, struct net_local *nl,
|
||
+ struct plip_local *snd, struct plip_local *rcv)
|
||
+{
|
||
+ return OK;
|
||
+}
|
||
+
|
||
+/* PLIP_RECEIVE --- receive a byte(two nibbles)
|
||
+ Returns OK on success, TIMEOUT on timeout */
|
||
+inline static int
|
||
+plip_receive(struct device *dev, unsigned short nibble_timeout,
|
||
+ enum plip_nibble_state *ns_p, unsigned char *data_p)
|
||
+{
|
||
+ unsigned char c0, c1;
|
||
+ unsigned int cx;
|
||
+
|
||
+ switch (*ns_p) {
|
||
+ case PLIP_NB_BEGIN:
|
||
+ cx = nibble_timeout;
|
||
+ while (1) {
|
||
+ c0 = z_readb(PAR_STATUS(dev));
|
||
+ udelay(PLIP_DELAY_UNIT);
|
||
+ if ((c0 & 0x80) == 0) {
|
||
+ c1 = z_readb(PAR_STATUS(dev));
|
||
+ if (c0 == c1)
|
||
+ break;
|
||
+ }
|
||
+ if (--cx == 0)
|
||
+ return TIMEOUT;
|
||
+ }
|
||
+#if 0
|
||
+ printk("received first nybble: %02X -> %02X\n",
|
||
+ c0, (c0 >> 3) & 0x0F);
|
||
+#endif
|
||
+ *data_p = (c0 >> 3) & 0x0f;
|
||
+ z_writeb(0x10, PAR_DATA(dev)); /* send ACK */
|
||
+ *ns_p = PLIP_NB_1;
|
||
+
|
||
+ case PLIP_NB_1:
|
||
+ cx = nibble_timeout;
|
||
+ while (1) {
|
||
+ c0 = z_readb(PAR_STATUS(dev));
|
||
+ udelay(PLIP_DELAY_UNIT);
|
||
+ if (c0 & 0x80) {
|
||
+ c1 = z_readb(PAR_STATUS(dev));
|
||
+ if (c0 == c1)
|
||
+ break;
|
||
+ }
|
||
+ if (--cx == 0)
|
||
+ return TIMEOUT;
|
||
+ }
|
||
+#if 0
|
||
+ printk("received second nybble: %02X -> %02X\n",
|
||
+ c0, (c0 << 1) & 0xF0);
|
||
+#endif
|
||
+ *data_p |= (c0 << 1) & 0xf0;
|
||
+ z_writeb(0x00, PAR_DATA(dev)); /* send ACK */
|
||
+ *ns_p = PLIP_NB_BEGIN;
|
||
+ case PLIP_NB_2:
|
||
+ break;
|
||
+ }
|
||
+ return OK;
|
||
+}
|
||
+
|
||
+/* PLIP_RECEIVE_PACKET --- receive a packet */
|
||
+static int
|
||
+plip_receive_packet(struct device *dev, struct net_local *nl,
|
||
+ struct plip_local *snd, struct plip_local *rcv)
|
||
+{
|
||
+ unsigned short nibble_timeout = nl->nibble;
|
||
+ unsigned char *lbuf;
|
||
+ unsigned long flags;
|
||
+
|
||
+ switch (rcv->state) {
|
||
+ case PLIP_PK_TRIGGER:
|
||
+ enable_par_irq(dev, 0);
|
||
+ dev->interrupt = 0;
|
||
+ z_writeb(0x01, PAR_DATA(dev)); /* send ACK */
|
||
+ if (net_debug > 2)
|
||
+ printk(KERN_DEBUG "%s: receive start\n", dev->name);
|
||
+ rcv->state = PLIP_PK_LENGTH_LSB;
|
||
+ rcv->nibble = PLIP_NB_BEGIN;
|
||
+
|
||
+ case PLIP_PK_LENGTH_LSB:
|
||
+ if (snd->state != PLIP_PK_DONE) {
|
||
+ if (plip_receive(dev, nl->trigger,
|
||
+ &rcv->nibble, &rcv->length.b.lsb)) {
|
||
+ /* collision, here dev->tbusy == 1 */
|
||
+ rcv->state = PLIP_PK_DONE;
|
||
+ nl->is_deferred = 1;
|
||
+ nl->connection = PLIP_CN_SEND;
|
||
+ queue_task(&nl->deferred, &tq_timer);
|
||
+ enable_par_irq(dev, 1);
|
||
+ return OK;
|
||
+ }
|
||
+ } else {
|
||
+ if (plip_receive(dev, nibble_timeout,
|
||
+ &rcv->nibble, &rcv->length.b.lsb))
|
||
+ return TIMEOUT;
|
||
+ }
|
||
+ rcv->state = PLIP_PK_LENGTH_MSB;
|
||
+
|
||
+ case PLIP_PK_LENGTH_MSB:
|
||
+ if (plip_receive(dev, nibble_timeout,
|
||
+ &rcv->nibble, &rcv->length.b.msb))
|
||
+ return TIMEOUT;
|
||
+ if (rcv->length.h > dev->mtu + dev->hard_header_len
|
||
+ || rcv->length.h < 8) {
|
||
+ printk(KERN_INFO "%s: bogus packet size %d.\n",
|
||
+ dev->name, rcv->length.h);
|
||
+ return ERROR;
|
||
+ }
|
||
+ /* Malloc up new buffer. */
|
||
+ rcv->skb = dev_alloc_skb(rcv->length.h);
|
||
+ if (rcv->skb == NULL) {
|
||
+ printk(KERN_INFO "%s: Memory squeeze.\n", dev->name);
|
||
+ return ERROR;
|
||
+ }
|
||
+ skb_put(rcv->skb,rcv->length.h);
|
||
+ rcv->skb->dev = dev;
|
||
+ rcv->state = PLIP_PK_DATA;
|
||
+ rcv->byte = 0;
|
||
+ rcv->checksum = 0;
|
||
+
|
||
+ case PLIP_PK_DATA:
|
||
+ lbuf = rcv->skb->data;
|
||
+ do
|
||
+ if (plip_receive(dev, nibble_timeout,
|
||
+ &rcv->nibble, &lbuf[rcv->byte]))
|
||
+ return TIMEOUT;
|
||
+ while (++rcv->byte < rcv->length.h);
|
||
+ do
|
||
+ rcv->checksum += lbuf[--rcv->byte];
|
||
+ while (rcv->byte);
|
||
+ rcv->state = PLIP_PK_CHECKSUM;
|
||
+
|
||
+ case PLIP_PK_CHECKSUM:
|
||
+ if (plip_receive(dev, nibble_timeout,
|
||
+ &rcv->nibble, &rcv->data))
|
||
+ return TIMEOUT;
|
||
+ if (rcv->data != rcv->checksum) {
|
||
+ nl->enet_stats.rx_crc_errors++;
|
||
+ if (net_debug)
|
||
+ printk(KERN_INFO "%s: checksum error\n",
|
||
+ dev->name);
|
||
+ return ERROR;
|
||
+ }
|
||
+ rcv->state = PLIP_PK_DONE;
|
||
+
|
||
+ case PLIP_PK_DONE:
|
||
+ /* Inform the upper layer for the arrival of a packet. */
|
||
+ rcv->skb->protocol=eth_type_trans(rcv->skb, dev);
|
||
+ netif_rx(rcv->skb);
|
||
+ nl->enet_stats.rx_packets++;
|
||
+ rcv->skb = NULL;
|
||
+ if (net_debug > 2)
|
||
+ printk(KERN_DEBUG "%s: receive end\n", dev->name);
|
||
+
|
||
+ /* Close the connection. */
|
||
+ z_writeb (0x00, PAR_DATA(dev));
|
||
+
|
||
+ local_irq_save(flags);
|
||
+ if (snd->state != PLIP_PK_DONE) {
|
||
+ nl->connection = PLIP_CN_SEND;
|
||
+ local_irq_restore(flags);
|
||
+ queue_task(&nl->immediate, &tq_immediate);
|
||
+ mark_bh(IMMEDIATE_BH);
|
||
+ enable_par_irq(dev, 1);
|
||
+ return OK;
|
||
+ } else {
|
||
+ nl->connection = PLIP_CN_NONE;
|
||
+ local_irq_restore(flags);
|
||
+ enable_par_irq(dev, 1);
|
||
+ return OK;
|
||
+ }
|
||
+ }
|
||
+ return OK;
|
||
+}
|
||
+
|
||
+/* PLIP_SEND --- send a byte (two nibbles)
|
||
+ Returns OK on success, TIMEOUT when timeout */
|
||
+inline static int
|
||
+plip_send(struct device *dev, unsigned short nibble_timeout,
|
||
+ enum plip_nibble_state *ns_p, unsigned char data)
|
||
+{
|
||
+ unsigned char c0;
|
||
+ unsigned int cx;
|
||
+
|
||
+ switch (*ns_p) {
|
||
+ case PLIP_NB_BEGIN:
|
||
+ z_writeb((data & 0x0f), PAR_DATA(dev));
|
||
+ *ns_p = PLIP_NB_1;
|
||
+
|
||
+ case PLIP_NB_1:
|
||
+ z_writeb(0x10 | (data & 0x0f), PAR_DATA(dev));
|
||
+ cx = nibble_timeout;
|
||
+ while (1) {
|
||
+ c0 = z_readb(PAR_STATUS(dev));
|
||
+ if ((c0 & 0x80) == 0)
|
||
+ break;
|
||
+ if (--cx == 0)
|
||
+ return TIMEOUT;
|
||
+ udelay(PLIP_DELAY_UNIT);
|
||
+ }
|
||
+ z_writeb(0x10 | (data >> 4), PAR_DATA(dev));
|
||
+ *ns_p = PLIP_NB_2;
|
||
+
|
||
+ case PLIP_NB_2:
|
||
+ z_writeb((data >> 4), PAR_DATA(dev));
|
||
+ cx = nibble_timeout;
|
||
+ while (1) {
|
||
+ c0 = z_readb(PAR_STATUS(dev));
|
||
+ if (c0 & 0x80)
|
||
+ break;
|
||
+ if (--cx == 0)
|
||
+ return TIMEOUT;
|
||
+ udelay(PLIP_DELAY_UNIT);
|
||
+ }
|
||
+ *ns_p = PLIP_NB_BEGIN;
|
||
+ return OK;
|
||
+ }
|
||
+ return OK;
|
||
+}
|
||
+
|
||
+/* PLIP_SEND_PACKET --- send a packet */
|
||
+static int
|
||
+plip_send_packet(struct device *dev, struct net_local *nl,
|
||
+ struct plip_local *snd, struct plip_local *rcv)
|
||
+{
|
||
+ unsigned short nibble_timeout = nl->nibble;
|
||
+ unsigned char *lbuf;
|
||
+ unsigned char c0;
|
||
+ unsigned int cx;
|
||
+ unsigned long flags;
|
||
+
|
||
+ if (snd->skb == NULL || (lbuf = snd->skb->data) == NULL) {
|
||
+ printk(KERN_INFO "%s: send skb lost\n", dev->name);
|
||
+ snd->state = PLIP_PK_DONE;
|
||
+ snd->skb = NULL;
|
||
+ return ERROR;
|
||
+ }
|
||
+
|
||
+ if (snd->length.h == 0) {
|
||
+ return OK;
|
||
+ }
|
||
+
|
||
+ switch (snd->state) {
|
||
+ case PLIP_PK_TRIGGER:
|
||
+ if ((z_readb(PAR_STATUS(dev)) & 0xf8) != 0x80)
|
||
+ return TIMEOUT;
|
||
+
|
||
+ /* Trigger remote rx interrupt. */
|
||
+ z_writeb(0x08, PAR_DATA(dev));
|
||
+ cx = nl->trigger;
|
||
+ while (1) {
|
||
+ udelay(PLIP_DELAY_UNIT);
|
||
+ local_irq_save(flags);
|
||
+ if (nl->connection == PLIP_CN_RECEIVE) {
|
||
+ local_irq_restore(flags);
|
||
+ /* interrupted */
|
||
+ nl->enet_stats.collisions++;
|
||
+ if (net_debug > 1)
|
||
+ printk(KERN_INFO "%s: collision.\n",
|
||
+ dev->name);
|
||
+ return OK;
|
||
+ }
|
||
+ c0 = z_readb(PAR_STATUS(dev));
|
||
+ if (c0 & 0x08) {
|
||
+ enable_par_irq(dev, 0);
|
||
+ if (net_debug > 2)
|
||
+ printk(KERN_DEBUG "%s: send start\n",
|
||
+ dev->name);
|
||
+ snd->state = PLIP_PK_LENGTH_LSB;
|
||
+ snd->nibble = PLIP_NB_BEGIN;
|
||
+ nl->timeout_count = 0;
|
||
+ local_irq_restore(flags);
|
||
+ break;
|
||
+ }
|
||
+ local_irq_restore(flags);
|
||
+ if (--cx == 0) {
|
||
+ z_writeb(0x00, PAR_DATA(dev));
|
||
+ return TIMEOUT;
|
||
+ }
|
||
+ }
|
||
+
|
||
+ case PLIP_PK_LENGTH_LSB:
|
||
+ if (plip_send(dev, nibble_timeout,
|
||
+ &snd->nibble, snd->length.b.lsb))
|
||
+ return TIMEOUT;
|
||
+ snd->state = PLIP_PK_LENGTH_MSB;
|
||
+
|
||
+ case PLIP_PK_LENGTH_MSB:
|
||
+ if (plip_send(dev, nibble_timeout,
|
||
+ &snd->nibble, snd->length.b.msb))
|
||
+ return TIMEOUT;
|
||
+ snd->state = PLIP_PK_DATA;
|
||
+ snd->byte = 0;
|
||
+ snd->checksum = 0;
|
||
+
|
||
+ case PLIP_PK_DATA:
|
||
+ do
|
||
+ if (plip_send(dev, nibble_timeout,
|
||
+ &snd->nibble, lbuf[snd->byte]))
|
||
+ return TIMEOUT;
|
||
+ while (++snd->byte < snd->length.h);
|
||
+ do
|
||
+ snd->checksum += lbuf[--snd->byte];
|
||
+ while (snd->byte);
|
||
+ snd->state = PLIP_PK_CHECKSUM;
|
||
+
|
||
+ case PLIP_PK_CHECKSUM:
|
||
+ if (plip_send(dev, nibble_timeout,
|
||
+ &snd->nibble, snd->checksum))
|
||
+ return TIMEOUT;
|
||
+
|
||
+ dev_kfree_skb(snd->skb);
|
||
+ nl->enet_stats.tx_packets++;
|
||
+ snd->state = PLIP_PK_DONE;
|
||
+
|
||
+ case PLIP_PK_DONE:
|
||
+ /* Close the connection */
|
||
+ z_writeb (0x00, PAR_DATA(dev));
|
||
+ snd->skb = NULL;
|
||
+ if (net_debug > 2)
|
||
+ printk(KERN_DEBUG "%s: send end\n", dev->name);
|
||
+ nl->connection = PLIP_CN_CLOSING;
|
||
+ nl->is_deferred = 1;
|
||
+ queue_task(&nl->deferred, &tq_timer);
|
||
+ enable_par_irq(dev, 1);
|
||
+ return OK;
|
||
+ }
|
||
+ return OK;
|
||
+}
|
||
+
|
||
+static int
|
||
+plip_connection_close(struct device *dev, struct net_local *nl,
|
||
+ struct plip_local *snd, struct plip_local *rcv)
|
||
+{
|
||
+ unsigned long flags;
|
||
+
|
||
+ local_irq_save(flags);
|
||
+ if (nl->connection == PLIP_CN_CLOSING) {
|
||
+ nl->connection = PLIP_CN_NONE;
|
||
+ dev->tbusy = 0;
|
||
+ mark_bh(NET_BH);
|
||
+ }
|
||
+ local_irq_restore(flags);
|
||
+ return OK;
|
||
+}
|
||
+
|
||
+/* PLIP_ERROR --- wait till other end settled */
|
||
+static int
|
||
+plip_error(struct device *dev, struct net_local *nl,
|
||
+ struct plip_local *snd, struct plip_local *rcv)
|
||
+{
|
||
+ unsigned char status;
|
||
+
|
||
+ status = z_readb(PAR_STATUS(dev));
|
||
+ if ((status & 0xf8) == 0x80) {
|
||
+ if (net_debug > 2)
|
||
+ printk(KERN_DEBUG "%s: reset interface.\n", dev->name);
|
||
+ nl->connection = PLIP_CN_NONE;
|
||
+ dev->tbusy = 0;
|
||
+ dev->interrupt = 0;
|
||
+ enable_par_irq(dev, 1);
|
||
+ mark_bh(NET_BH);
|
||
+ } else {
|
||
+ nl->is_deferred = 1;
|
||
+ queue_task(&nl->deferred, &tq_timer);
|
||
+ }
|
||
+
|
||
+ return OK;
|
||
+}
|
||
+
|
||
+/* We don't need to send arp, for plip is point-to-point. */
|
||
+static int
|
||
+plip_rebuild_header(struct sk_buff *skb)
|
||
+{
|
||
+ struct device *dev = skb->dev;
|
||
+ struct net_local *nl = (struct net_local *)dev->priv;
|
||
+ struct ethhdr *eth = (struct ethhdr *)skb->data;
|
||
+ int i;
|
||
+
|
||
+ if ((dev->flags & IFF_NOARP)==0)
|
||
+ return nl->orig_rebuild_header(skb);
|
||
+
|
||
+ if (eth->h_proto != __constant_htons(ETH_P_IP)
|
||
+#if defined(CONFIG_IPV6) || defined(CONFIG_IPV6_MODULE)
|
||
+ && eth->h_proto != __constant_htons(ETH_P_IPV6)
|
||
+#endif
|
||
+ ) {
|
||
+ printk(KERN_ERR "plip_rebuild_header: Don't know how to resolve type %d addresses?\n", (int)eth->h_proto);
|
||
+ memcpy(eth->h_source, dev->dev_addr, dev->addr_len);
|
||
+ return 0;
|
||
+ }
|
||
+
|
||
+ for (i=0; i < ETH_ALEN - sizeof(u32); i++)
|
||
+ eth->h_dest[i] = 0xfc;
|
||
+#if 0
|
||
+ *(u32 *)(eth->h_dest+i) = dst;
|
||
+#else
|
||
+ /* Do not want to include net/route.h here.
|
||
+ * In any case, it is TOP of silliness to emulate
|
||
+ * hardware addresses on PtP link. --ANK
|
||
+ */
|
||
+ *(u32 *)(eth->h_dest+i) = 0;
|
||
+#endif
|
||
+ return 0;
|
||
+}
|
||
+
|
||
+static int
|
||
+plip_tx_packet(struct sk_buff *skb, struct device *dev)
|
||
+{
|
||
+ struct net_local *nl = (struct net_local *)dev->priv;
|
||
+ struct plip_local *snd = &nl->snd_data;
|
||
+ unsigned long flags;
|
||
+
|
||
+ if (dev->tbusy)
|
||
+ return 1;
|
||
+
|
||
+ if (test_and_set_bit(0, (void*)&dev->tbusy) != 0) {
|
||
+ printk(KERN_ERR "%s: Transmitter access conflict.\n",
|
||
+ dev->name);
|
||
+ return 1;
|
||
+ }
|
||
+
|
||
+ if (skb->len > dev->mtu + dev->hard_header_len) {
|
||
+ printk(KERN_ERR "%s: packet too big, %d.\n",
|
||
+ dev->name, (int)skb->len);
|
||
+ dev->tbusy = 0;
|
||
+ return 0;
|
||
+ }
|
||
+
|
||
+ if (net_debug > 2)
|
||
+ printk(KERN_DEBUG "%s: send request\n", dev->name);
|
||
+
|
||
+ local_irq_save(flags);
|
||
+ dev->trans_start = jiffies;
|
||
+ snd->skb = skb;
|
||
+ snd->length.h = skb->len;
|
||
+ snd->state = PLIP_PK_TRIGGER;
|
||
+ if (nl->connection == PLIP_CN_NONE) {
|
||
+ nl->connection = PLIP_CN_SEND;
|
||
+ nl->timeout_count = 0;
|
||
+ }
|
||
+ queue_task(&nl->immediate, &tq_immediate);
|
||
+ mark_bh(IMMEDIATE_BH);
|
||
+ local_irq_restore(flags);
|
||
+
|
||
+ return 0;
|
||
+}
|
||
+
|
||
+/* Open/initialize the board. This is called (in the current kernel)
|
||
+ sometime after booting when the 'ifconfig' program is run.
|
||
+
|
||
+ */
|
||
+static int
|
||
+plip_open(struct device *dev)
|
||
+{
|
||
+ struct net_local *nl = (struct net_local *)dev->priv;
|
||
+ struct in_device *in_dev;
|
||
+
|
||
+#if defined(CONFIG_GVPIOEXT_LP) || defined(CONFIG_GVPIOEXT_LP_MODULE)
|
||
+ /* Yes, there is a race condition here. Fix it later */
|
||
+ if (PLIP_DEV(dev)->par_use & IOEXT_PAR_LP) {
|
||
+ /* Can't open if lp is in use */
|
||
+#if DEBUG
|
||
+ printk("par is in use by lp\n");
|
||
+#endif
|
||
+ return(-EBUSY);
|
||
+ }
|
||
+#endif
|
||
+ PLIP_DEV(dev)->par_use |= IOEXT_PAR_PLIP;
|
||
+
|
||
+#if DEBUG
|
||
+ printk("plip_open(): sending 00 to data port\n");
|
||
+#endif
|
||
+
|
||
+ /* Clear the data port. */
|
||
+ z_writeb (0x00, PAR_DATA(dev));
|
||
+
|
||
+#if DEBUG
|
||
+ printk("plip_open(): sent\n");
|
||
+#endif
|
||
+
|
||
+ /* Initialize the state machine. */
|
||
+ nl->rcv_data.state = nl->snd_data.state = PLIP_PK_DONE;
|
||
+ nl->rcv_data.skb = nl->snd_data.skb = NULL;
|
||
+ nl->connection = PLIP_CN_NONE;
|
||
+ nl->is_deferred = 0;
|
||
+
|
||
+ /* Fill in the MAC-level header.
|
||
+ (ab)Use "dev->broadcast" to store point-to-point MAC address.
|
||
+
|
||
+ PLIP doesn't have a real mac address, but we need to create one
|
||
+ to be DOS compatible. */
|
||
+ memset(dev->dev_addr, 0xfc, ETH_ALEN);
|
||
+ memset(dev->broadcast, 0xfc, ETH_ALEN);
|
||
+
|
||
+ if ((in_dev=dev->ip_ptr) != NULL) {
|
||
+ /*
|
||
+ * Any address will do - we take the first
|
||
+ */
|
||
+ struct in_ifaddr *ifa=in_dev->ifa_list;
|
||
+ if (ifa != NULL) {
|
||
+ memcpy(dev->dev_addr+2, &ifa->ifa_local, 4);
|
||
+ memcpy(dev->broadcast+2, &ifa->ifa_address, 4);
|
||
+ }
|
||
+ }
|
||
+
|
||
+ dev->interrupt = 0;
|
||
+ dev->start = 1;
|
||
+ dev->tbusy = 0;
|
||
+
|
||
+ MOD_INC_USE_COUNT;
|
||
+
|
||
+ /* Enable rx interrupt. */
|
||
+ enable_par_irq(dev, 1);
|
||
+
|
||
+ return 0;
|
||
+}
|
||
+
|
||
+/* The inverse routine to plip_open (). */
|
||
+static int
|
||
+plip_close(struct device *dev)
|
||
+{
|
||
+ struct net_local *nl = (struct net_local *)dev->priv;
|
||
+ struct plip_local *snd = &nl->snd_data;
|
||
+ struct plip_local *rcv = &nl->rcv_data;
|
||
+ unsigned long flags;
|
||
+
|
||
+ dev->tbusy = 1;
|
||
+ dev->start = 0;
|
||
+ local_irq_save(flags);
|
||
+ nl->is_deferred = 0;
|
||
+ nl->connection = PLIP_CN_NONE;
|
||
+ local_irq_restore(flags);
|
||
+ z_writeb(0x00, PAR_DATA(dev));
|
||
+
|
||
+ snd->state = PLIP_PK_DONE;
|
||
+ if (snd->skb) {
|
||
+ dev_kfree_skb(snd->skb);
|
||
+ snd->skb = NULL;
|
||
+ }
|
||
+ rcv->state = PLIP_PK_DONE;
|
||
+ if (rcv->skb) {
|
||
+ kfree_skb(rcv->skb);
|
||
+ rcv->skb = NULL;
|
||
+ }
|
||
+
|
||
+ PLIP_DEV(dev)->par_use &= ~IOEXT_PAR_PLIP;
|
||
+
|
||
+ MOD_DEC_USE_COUNT;
|
||
+ return 0;
|
||
+}
|
||
+
|
||
+static struct enet_statistics *
|
||
+plip_get_stats(struct device *dev)
|
||
+{
|
||
+ struct net_local *nl = (struct net_local *)dev->priv;
|
||
+ struct enet_statistics *r = &nl->enet_stats;
|
||
+
|
||
+ return r;
|
||
+}
|
||
+
|
||
+static int
|
||
+plip_config(struct device *dev, struct ifmap *map)
|
||
+{
|
||
+ if (dev->flags & IFF_UP)
|
||
+ return -EBUSY;
|
||
+
|
||
+ printk(KERN_INFO "%s: This interface is autodetected (ignored).\n",
|
||
+ dev->name);
|
||
+
|
||
+ return 0;
|
||
+}
|
||
+
|
||
+static int
|
||
+plip_ioctl(struct device *dev, struct ifreq *rq, int cmd)
|
||
+{
|
||
+ struct net_local *nl = (struct net_local *) dev->priv;
|
||
+ struct plipconf *pc = (struct plipconf *) &rq->ifr_data;
|
||
+
|
||
+ switch(pc->pcmd) {
|
||
+ case PLIP_GET_TIMEOUT:
|
||
+ pc->trigger = nl->trigger;
|
||
+ pc->nibble = nl->nibble;
|
||
+ break;
|
||
+ case PLIP_SET_TIMEOUT:
|
||
+ nl->trigger = pc->trigger;
|
||
+ nl->nibble = pc->nibble;
|
||
+ break;
|
||
+ default:
|
||
+ return -EOPNOTSUPP;
|
||
+ }
|
||
+ return 0;
|
||
+}
|
||
+
|
||
+/*
|
||
+ * Detect and initialize all IO-Extenders in this system.
|
||
+ *
|
||
+ * Both PLIP and serial devices are configured.
|
||
+ */
|
||
+int plip_init(struct device *dev)
|
||
+{
|
||
+ IOEXT_struct *board;
|
||
+ struct net_local *nl;
|
||
+
|
||
+ if (ioext_num == 0) {
|
||
+ printk(KERN_INFO "%s\n", version);
|
||
+ }
|
||
+
|
||
+ board = PLIP_DEV(dev)->board;
|
||
+ dev->base_addr = (unsigned long)&board->par.DATA;
|
||
+
|
||
+ /* Cheat and use irq to index into our table */
|
||
+ dev->irq = ioext_num;
|
||
+
|
||
+ printk(KERN_INFO "%s: IO-Extender parallel port at 0x%08lX\n", dev->name, dev->base_addr);
|
||
+
|
||
+ /* Fill in the generic fields of the device structure. */
|
||
+ ether_setup(dev);
|
||
+
|
||
+ /* Then, override parts of it */
|
||
+ dev->hard_start_xmit = plip_tx_packet;
|
||
+ dev->open = plip_open;
|
||
+ dev->stop = plip_close;
|
||
+ dev->get_stats = plip_get_stats;
|
||
+ dev->set_config = plip_config;
|
||
+ dev->do_ioctl = plip_ioctl;
|
||
+ dev->tx_queue_len = 10;
|
||
+ dev->flags = IFF_POINTOPOINT|IFF_NOARP;
|
||
+
|
||
+ /* Set the private structure */
|
||
+ dev->priv = kmalloc(sizeof (struct net_local), GFP_KERNEL);
|
||
+ if (dev->priv == NULL) {
|
||
+ printk(KERN_ERR "%s: out of memory\n", dev->name);
|
||
+ return -ENOMEM;
|
||
+ }
|
||
+ memset(dev->priv, 0, sizeof(struct net_local));
|
||
+ nl = (struct net_local *) dev->priv;
|
||
+
|
||
+ nl->orig_rebuild_header = dev->rebuild_header;
|
||
+ dev->rebuild_header = plip_rebuild_header;
|
||
+
|
||
+ /* Initialize constants */
|
||
+ nl->trigger = PLIP_TRIGGER_WAIT;
|
||
+ nl->nibble = PLIP_NIBBLE_WAIT;
|
||
+
|
||
+ /* Initialize task queue structures */
|
||
+ nl->immediate.next = NULL;
|
||
+ nl->immediate.sync = 0;
|
||
+ nl->immediate.routine = (void *)(void *)plip_bh;
|
||
+ nl->immediate.data = dev;
|
||
+
|
||
+ nl->deferred.next = NULL;
|
||
+ nl->deferred.sync = 0;
|
||
+ nl->deferred.routine = (void *)(void *)plip_kick_bh;
|
||
+ nl->deferred.data = dev;
|
||
+
|
||
+ /* Don't enable interrupts yet */
|
||
+
|
||
+ return 0;
|
||
+}
|
||
diff -urN --exclude-from=/usr/src/exclude-file linux-2.6.13-i386/drivers/char/serial167.c linux-2.6.13/drivers/char/serial167.c
|
||
--- linux-2.6.13-i386/drivers/char/serial167.c 2005-08-29 01:41:01.000000000 +0200
|
||
+++ linux-2.6.13/drivers/char/serial167.c 2005-10-12 16:32:10.000000000 +0200
|
||
@@ -1450,7 +1450,6 @@
|
||
volatile unsigned char *base_addr = (u_char *)BASE_ADDR;
|
||
unsigned long flags;
|
||
unsigned char status;
|
||
- unsigned int result;
|
||
|
||
channel = info->line;
|
||
|
||
@@ -1474,7 +1473,6 @@
|
||
int channel;
|
||
volatile unsigned char *base_addr = (u_char *)BASE_ADDR;
|
||
unsigned long flags;
|
||
- unsigned int arg;
|
||
|
||
channel = info->line;
|
||
|
||
diff -urN --exclude-from=/usr/src/exclude-file linux-2.6.13-i386/drivers/ide/ide-iops.c linux-2.6.13/drivers/ide/ide-iops.c
|
||
--- linux-2.6.13-i386/drivers/ide/ide-iops.c 2005-08-29 01:41:01.000000000 +0200
|
||
+++ linux-2.6.13/drivers/ide/ide-iops.c 2005-08-30 16:32:55.000000000 +0200
|
||
@@ -341,6 +341,23 @@
|
||
int i;
|
||
u16 *stringcast;
|
||
|
||
+#ifdef __mc68000__
|
||
+ if (!MACH_IS_AMIGA && !MACH_IS_MAC && !MACH_IS_Q40 && !MACH_IS_ATARI)
|
||
+ return;
|
||
+
|
||
+#ifdef M68K_IDE_SWAPW
|
||
+ if (M68K_IDE_SWAPW) { /* fix bus byteorder first */
|
||
+ u_char *p = (u_char *)id;
|
||
+ u_char t;
|
||
+ for (i = 0; i < 512; i += 2) {
|
||
+ t = p[i];
|
||
+ p[i] = p[i+1];
|
||
+ p[i+1] = t;
|
||
+ }
|
||
+ }
|
||
+#endif
|
||
+#endif /* __mc68000__ */
|
||
+
|
||
id->config = __le16_to_cpu(id->config);
|
||
id->cyls = __le16_to_cpu(id->cyls);
|
||
id->reserved2 = __le16_to_cpu(id->reserved2);
|
||
diff -urN --exclude-from=/usr/src/exclude-file linux-2.6.13-i386/drivers/ide/legacy/gayle.c linux-2.6.13/drivers/ide/legacy/gayle.c
|
||
--- linux-2.6.13-i386/drivers/ide/legacy/gayle.c 2005-08-29 01:41:01.000000000 +0200
|
||
+++ linux-2.6.13/drivers/ide/legacy/gayle.c 2005-09-02 16:32:11.000000000 +0200
|
||
@@ -161,6 +161,7 @@
|
||
base = (unsigned long)ZTWO_VADDR(phys_base);
|
||
ctrlport = GAYLE_HAS_CONTROL_REG ? (base + GAYLE_CONTROL) : 0;
|
||
|
||
+ memset(&hw, 0, sizeof(hw));
|
||
ide_setup_ports(&hw, base, gayle_offsets,
|
||
ctrlport, irqport, ack_intr,
|
||
// &gayle_iops,
|
||
diff -urN --exclude-from=/usr/src/exclude-file linux-2.6.13-i386/drivers/input/keyboard/Kconfig linux-2.6.13/drivers/input/keyboard/Kconfig
|
||
--- linux-2.6.13-i386/drivers/input/keyboard/Kconfig 2005-08-29 01:41:01.000000000 +0200
|
||
+++ linux-2.6.13/drivers/input/keyboard/Kconfig 2005-08-30 16:33:05.000000000 +0200
|
||
@@ -154,7 +154,7 @@
|
||
|
||
config KEYBOARD_HIL_OLD
|
||
tristate "HP HIL keyboard support (simple driver)"
|
||
- depends on GSC
|
||
+ depends on GSC || HP300
|
||
default y
|
||
help
|
||
The "Human Interface Loop" is a older, 8-channel USB-like
|
||
@@ -171,7 +171,7 @@
|
||
|
||
config KEYBOARD_HIL
|
||
tristate "HP HIL keyboard support"
|
||
- depends on GSC
|
||
+ depends on GSC || HP300
|
||
default y
|
||
select HP_SDC
|
||
select HIL_MLC
|
||
diff -urN --exclude-from=/usr/src/exclude-file linux-2.6.13-i386/drivers/input/misc/Kconfig linux-2.6.13/drivers/input/misc/Kconfig
|
||
--- linux-2.6.13-i386/drivers/input/misc/Kconfig 2005-08-29 01:41:01.000000000 +0200
|
||
+++ linux-2.6.13/drivers/input/misc/Kconfig 2005-08-30 16:33:05.000000000 +0200
|
||
@@ -51,7 +51,7 @@
|
||
|
||
config HP_SDC_RTC
|
||
tristate "HP SDC Real Time Clock"
|
||
- depends on GSC
|
||
+ depends on GSC || HP300
|
||
select HP_SDC
|
||
help
|
||
Say Y here if you want to support the built-in real time clock
|
||
diff -urN --exclude-from=/usr/src/exclude-file linux-2.6.13-i386/drivers/input/mouse/Kconfig linux-2.6.13/drivers/input/mouse/Kconfig
|
||
--- linux-2.6.13-i386/drivers/input/mouse/Kconfig 2005-08-29 01:41:01.000000000 +0200
|
||
+++ linux-2.6.13/drivers/input/mouse/Kconfig 2005-08-30 16:33:05.000000000 +0200
|
||
@@ -129,7 +129,7 @@
|
||
|
||
config MOUSE_HIL
|
||
tristate "HIL pointers (mice etc)."
|
||
- depends on GSC
|
||
+ depends on GSC || HP300
|
||
select HP_SDC
|
||
select HIL_MLC
|
||
help
|
||
diff -urN --exclude-from=/usr/src/exclude-file linux-2.6.13-i386/drivers/input/serio/Kconfig linux-2.6.13/drivers/input/serio/Kconfig
|
||
--- linux-2.6.13-i386/drivers/input/serio/Kconfig 2005-08-29 01:41:01.000000000 +0200
|
||
+++ linux-2.6.13/drivers/input/serio/Kconfig 2005-08-30 16:33:05.000000000 +0200
|
||
@@ -112,7 +112,7 @@
|
||
|
||
config HP_SDC
|
||
tristate "HP System Device Controller i8042 Support"
|
||
- depends on GSC && SERIO
|
||
+ depends on (GSC || HP300) && SERIO
|
||
default y
|
||
---help---
|
||
This option enables supports for the the "System Device
|
||
diff -urN --exclude-from=/usr/src/exclude-file linux-2.6.13-i386/drivers/macintosh/adb.c linux-2.6.13/drivers/macintosh/adb.c
|
||
--- linux-2.6.13-i386/drivers/macintosh/adb.c 2005-08-29 01:41:01.000000000 +0200
|
||
+++ linux-2.6.13/drivers/macintosh/adb.c 2005-08-30 16:33:07.000000000 +0200
|
||
@@ -476,13 +476,15 @@
|
||
use_sreq = 1;
|
||
} else
|
||
use_sreq = 0;
|
||
- req->nbytes = nbytes+1;
|
||
+ i = (flags & ADBREQ_RAW) ? 0 : 1;
|
||
+ req->nbytes = nbytes+i;
|
||
req->done = done;
|
||
req->reply_expected = flags & ADBREQ_REPLY;
|
||
req->data[0] = ADB_PACKET;
|
||
va_start(list, nbytes);
|
||
- for (i = 0; i < nbytes; ++i)
|
||
- req->data[i+1] = va_arg(list, int);
|
||
+ while (i < req->nbytes) {
|
||
+ req->data[i++] = va_arg(list, int);
|
||
+ }
|
||
va_end(list);
|
||
|
||
if (flags & ADBREQ_NOSEND)
|
||
diff -urN --exclude-from=/usr/src/exclude-file linux-2.6.13-i386/drivers/net/Kconfig linux-2.6.13/drivers/net/Kconfig
|
||
--- linux-2.6.13-i386/drivers/net/Kconfig 2005-08-29 01:41:01.000000000 +0200
|
||
+++ linux-2.6.13/drivers/net/Kconfig 2005-08-30 16:33:24.000000000 +0200
|
||
@@ -294,7 +294,7 @@
|
||
|
||
config MAC89x0
|
||
tristate "Macintosh CS89x0 based ethernet cards"
|
||
- depends on NET_ETHERNET && MAC && BROKEN
|
||
+ depends on NET_ETHERNET && MAC
|
||
---help---
|
||
Support for CS89x0 chipset based Ethernet cards. If you have a
|
||
Nubus or LC-PDS network (Ethernet) card of this type, say Y and
|
||
diff -urN --exclude-from=/usr/src/exclude-file linux-2.6.13-i386/drivers/net/hplance.c linux-2.6.13/drivers/net/hplance.c
|
||
--- linux-2.6.13-i386/drivers/net/hplance.c 2005-08-29 01:41:01.000000000 +0200
|
||
+++ linux-2.6.13/drivers/net/hplance.c 2005-10-12 16:32:43.000000000 +0200
|
||
@@ -77,6 +77,7 @@
|
||
{
|
||
struct net_device *dev;
|
||
int err = -ENOMEM;
|
||
+ int i;
|
||
|
||
dev = alloc_etherdev(sizeof(struct hplance_private));
|
||
if (!dev)
|
||
@@ -93,6 +94,15 @@
|
||
goto out_release_mem_region;
|
||
|
||
dio_set_drvdata(d, dev);
|
||
+
|
||
+ printk(KERN_INFO "%s: %s; select code %d, addr %2.2x", dev->name, d->name, d->scode, dev->dev_addr[0]);
|
||
+
|
||
+ for (i=1; i<6; i++) {
|
||
+ printk(":%2.2x", dev->dev_addr[i]);
|
||
+ }
|
||
+
|
||
+ printk(", irq %d\n", d->ipl);
|
||
+
|
||
return 0;
|
||
|
||
out_release_mem_region:
|
||
@@ -118,9 +128,7 @@
|
||
unsigned long va = (d->resource.start + DIO_VIRADDRBASE);
|
||
struct hplance_private *lp;
|
||
int i;
|
||
-
|
||
- printk(KERN_INFO "%s: %s; select code %d, addr", dev->name, d->name, d->scode);
|
||
-
|
||
+
|
||
/* reset the board */
|
||
out_8(va+DIO_IDOFF, 0xff);
|
||
udelay(100); /* ariba! ariba! udelay! udelay! */
|
||
@@ -143,7 +151,6 @@
|
||
*/
|
||
dev->dev_addr[i] = ((in_8(va + HPLANCE_NVRAMOFF + i*4 + 1) & 0xF) << 4)
|
||
| (in_8(va + HPLANCE_NVRAMOFF + i*4 + 3) & 0xF);
|
||
- printk("%c%2.2x", i == 0 ? ' ' : ':', dev->dev_addr[i]);
|
||
}
|
||
|
||
lp = netdev_priv(dev);
|
||
@@ -160,7 +167,6 @@
|
||
lp->lance.lance_log_tx_bufs = LANCE_LOG_TX_BUFFERS;
|
||
lp->lance.rx_ring_mod_mask = RX_RING_MOD_MASK;
|
||
lp->lance.tx_ring_mod_mask = TX_RING_MOD_MASK;
|
||
- printk(", irq %d\n", lp->lance.irq);
|
||
}
|
||
|
||
/* This is disgusting. We have to check the DIO status register for ack every
|
||
diff -urN --exclude-from=/usr/src/exclude-file linux-2.6.13-i386/drivers/net/mac89x0.c linux-2.6.13/drivers/net/mac89x0.c
|
||
--- linux-2.6.13-i386/drivers/net/mac89x0.c 2005-08-29 01:41:01.000000000 +0200
|
||
+++ linux-2.6.13/drivers/net/mac89x0.c 2004-12-30 16:38:24.000000000 +0100
|
||
@@ -128,7 +128,7 @@
|
||
extern void reset_chip(struct net_device *dev);
|
||
#endif
|
||
static int net_open(struct net_device *dev);
|
||
-static int net_send_packet(struct sk_buff *skb, struct net_device *dev);
|
||
+static int net_send_packet(struct sk_buff *skb, struct net_device *dev);
|
||
static irqreturn_t net_interrupt(int irq, void *dev_id, struct pt_regs *regs);
|
||
static void set_multicast_list(struct net_device *dev);
|
||
static void net_rx(struct net_device *dev);
|
||
@@ -374,56 +374,37 @@
|
||
static int
|
||
net_send_packet(struct sk_buff *skb, struct net_device *dev)
|
||
{
|
||
- if (dev->tbusy) {
|
||
- /* If we get here, some higher level has decided we are broken.
|
||
- There should really be a "kick me" function call instead. */
|
||
- int tickssofar = jiffies - dev->trans_start;
|
||
- if (tickssofar < 5)
|
||
- return 1;
|
||
- if (net_debug > 0) printk("%s: transmit timed out, %s?\n", dev->name,
|
||
- tx_done(dev) ? "IRQ conflict" : "network cable problem");
|
||
- /* Try to restart the adaptor. */
|
||
- dev->tbusy=0;
|
||
- dev->trans_start = jiffies;
|
||
- }
|
||
-
|
||
- /* Block a timer-based transmit from overlapping. This could better be
|
||
- done with atomic_swap(1, dev->tbusy), but set_bit() works as well. */
|
||
- if (test_and_set_bit(0, (void*)&dev->tbusy) != 0)
|
||
- printk("%s: Transmitter access conflict.\n", dev->name);
|
||
- else {
|
||
- struct net_local *lp = netdev_priv(dev);
|
||
- unsigned long flags;
|
||
-
|
||
- if (net_debug > 3)
|
||
- printk("%s: sent %d byte packet of type %x\n",
|
||
- dev->name, skb->len,
|
||
- (skb->data[ETH_ALEN+ETH_ALEN] << 8)
|
||
- | skb->data[ETH_ALEN+ETH_ALEN+1]);
|
||
-
|
||
- /* keep the upload from being interrupted, since we
|
||
- ask the chip to start transmitting before the
|
||
- whole packet has been completely uploaded. */
|
||
- local_irq_save(flags);
|
||
-
|
||
- /* initiate a transmit sequence */
|
||
- writereg(dev, PP_TxCMD, lp->send_cmd);
|
||
- writereg(dev, PP_TxLength, skb->len);
|
||
-
|
||
- /* Test to see if the chip has allocated memory for the packet */
|
||
- if ((readreg(dev, PP_BusST) & READY_FOR_TX_NOW) == 0) {
|
||
- /* Gasp! It hasn't. But that shouldn't happen since
|
||
- we're waiting for TxOk, so return 1 and requeue this packet. */
|
||
- local_irq_restore(flags);
|
||
- return 1;
|
||
- }
|
||
+ struct net_local *lp = netdev_priv(dev);
|
||
+ unsigned long flags;
|
||
|
||
- /* Write the contents of the packet */
|
||
- memcpy_toio(dev->mem_start + PP_TxFrame, skb->data, skb->len+1);
|
||
+ if (net_debug > 3)
|
||
+ printk("%s: sent %d byte packet of type %x\n",
|
||
+ dev->name, skb->len,
|
||
+ (skb->data[ETH_ALEN+ETH_ALEN] << 8)
|
||
+ | skb->data[ETH_ALEN+ETH_ALEN+1]);
|
||
+
|
||
+ /* keep the upload from being interrupted, since we
|
||
+ ask the chip to start transmitting before the
|
||
+ whole packet has been completely uploaded. */
|
||
+ local_irq_save(flags);
|
||
|
||
+ /* initiate a transmit sequence */
|
||
+ writereg(dev, PP_TxCMD, lp->send_cmd);
|
||
+ writereg(dev, PP_TxLength, skb->len);
|
||
+
|
||
+ /* Test to see if the chip has allocated memory for the packet */
|
||
+ if ((readreg(dev, PP_BusST) & READY_FOR_TX_NOW) == 0) {
|
||
+ /* Gasp! It hasn't. But that shouldn't happen since
|
||
+ we're waiting for TxOk, so return 1 and requeue this packet. */
|
||
local_irq_restore(flags);
|
||
- dev->trans_start = jiffies;
|
||
+ return 1;
|
||
}
|
||
+
|
||
+ /* Write the contents of the packet */
|
||
+ memcpy((void *)(dev->mem_start + PP_TxFrame), skb->data, skb->len+1);
|
||
+
|
||
+ local_irq_restore(flags);
|
||
+ dev->trans_start = jiffies;
|
||
dev_kfree_skb (skb);
|
||
|
||
return 0;
|
||
@@ -441,9 +422,6 @@
|
||
printk ("net_interrupt(): irq %d for unknown device.\n", irq);
|
||
return IRQ_NONE;
|
||
}
|
||
- if (dev->interrupt)
|
||
- printk("%s: Re-entering the interrupt handler.\n", dev->name);
|
||
- dev->interrupt = 1;
|
||
|
||
ioaddr = dev->base_addr;
|
||
lp = netdev_priv(dev);
|
||
@@ -464,8 +442,7 @@
|
||
break;
|
||
case ISQ_TRANSMITTER_EVENT:
|
||
lp->stats.tx_packets++;
|
||
- dev->tbusy = 0;
|
||
- mark_bh(NET_BH); /* Inform upper layers. */
|
||
+ netif_wake_queue(dev);
|
||
if ((status & TX_OK) == 0) lp->stats.tx_errors++;
|
||
if (status & TX_LOST_CRS) lp->stats.tx_carrier_errors++;
|
||
if (status & TX_SQE_ERROR) lp->stats.tx_heartbeat_errors++;
|
||
@@ -479,8 +456,7 @@
|
||
That shouldn't happen since we only ever
|
||
load one packet. Shrug. Do the right
|
||
thing anyway. */
|
||
- dev->tbusy = 0;
|
||
- mark_bh(NET_BH); /* Inform upper layers. */
|
||
+ netif_wake_queue(dev);
|
||
}
|
||
if (status & TX_UNDERRUN) {
|
||
if (net_debug > 0) printk("%s: transmit underrun\n", dev->name);
|
||
@@ -497,7 +473,6 @@
|
||
break;
|
||
}
|
||
}
|
||
- dev->interrupt = 0;
|
||
return IRQ_HANDLED;
|
||
}
|
||
|
||
@@ -532,7 +507,7 @@
|
||
skb_put(skb, length);
|
||
skb->dev = dev;
|
||
|
||
- memcpy_fromio(skb->data, dev->mem_start + PP_RxFrame, length);
|
||
+ memcpy(skb->data, (void *)(dev->mem_start + PP_RxFrame), length);
|
||
|
||
if (net_debug > 3)printk("%s: received %d byte packet of type %x\n",
|
||
dev->name, length,
|
||
@@ -611,8 +586,6 @@
|
||
static int set_mac_address(struct net_device *dev, void *addr)
|
||
{
|
||
int i;
|
||
- if (dev->start)
|
||
- return -EBUSY;
|
||
printk("%s: Setting MAC address to ", dev->name);
|
||
for (i = 0; i < 6; i++)
|
||
printk(" %2.2x", dev->dev_addr[i] = ((unsigned char *)addr)[i]);
|
||
diff -urN --exclude-from=/usr/src/exclude-file linux-2.6.13-i386/drivers/scsi/Kconfig linux-2.6.13/drivers/scsi/Kconfig
|
||
--- linux-2.6.13-i386/drivers/scsi/Kconfig 2005-08-29 01:41:01.000000000 +0200
|
||
+++ linux-2.6.13/drivers/scsi/Kconfig 2005-08-30 16:33:54.000000000 +0200
|
||
@@ -1627,7 +1627,7 @@
|
||
|
||
config SCSI_AMIGA7XX
|
||
bool "Amiga NCR53c710 SCSI support (EXPERIMENTAL)"
|
||
- depends on AMIGA && SCSI && EXPERIMENTAL && BROKEN
|
||
+ depends on AMIGA && SCSI && EXPERIMENTAL
|
||
help
|
||
Support for various NCR53c710-based SCSI controllers on the Amiga.
|
||
This includes:
|
||
@@ -1724,7 +1724,7 @@
|
||
|
||
config MVME16x_SCSI
|
||
bool "NCR53C710 SCSI driver for MVME16x"
|
||
- depends on MVME16x && SCSI && BROKEN
|
||
+ depends on MVME16x && SCSI
|
||
help
|
||
The Motorola MVME162, 166, 167, 172 and 177 boards use the NCR53C710
|
||
SCSI controller chip. Almost everyone using one of these boards
|
||
@@ -1732,7 +1732,7 @@
|
||
|
||
config BVME6000_SCSI
|
||
bool "NCR53C710 SCSI driver for BVME6000"
|
||
- depends on BVME6000 && SCSI && BROKEN
|
||
+ depends on BVME6000 && SCSI
|
||
help
|
||
The BVME4000 and BVME6000 boards from BVM Ltd use the NCR53C710
|
||
SCSI controller chip. Almost everyone using one of these boards
|
||
diff -urN --exclude-from=/usr/src/exclude-file linux-2.6.13-i386/drivers/scsi/amiga7xx.c linux-2.6.13/drivers/scsi/amiga7xx.c
|
||
--- linux-2.6.13-i386/drivers/scsi/amiga7xx.c 2005-08-29 01:41:01.000000000 +0200
|
||
+++ linux-2.6.13/drivers/scsi/amiga7xx.c 2004-10-30 16:35:43.000000000 +0200
|
||
@@ -27,8 +27,14 @@
|
||
#include "scsi.h"
|
||
#include <scsi/scsi_host.h>
|
||
#include "53c7xx.h"
|
||
-#include "amiga7xx.h"
|
||
|
||
+#ifndef CMD_PER_LUN
|
||
+#define CMD_PER_LUN 3
|
||
+#endif
|
||
+
|
||
+#ifndef CAN_QUEUE
|
||
+#define CAN_QUEUE 24
|
||
+#endif
|
||
|
||
static int amiga7xx_register_one(Scsi_Host_Template *tpnt,
|
||
unsigned long address)
|
||
@@ -115,8 +121,10 @@
|
||
{
|
||
if (shost->irq)
|
||
free_irq(shost->irq, NULL);
|
||
+#ifdef CONFIG_ISA
|
||
if (shost->dma_channel != 0xff)
|
||
free_dma(shost->dma_channel);
|
||
+#endif
|
||
if (shost->io_port && shost->n_io_port)
|
||
release_region(shost->io_port, shost->n_io_port);
|
||
scsi_unregister(shost);
|
||
@@ -128,8 +136,9 @@
|
||
.detect = amiga7xx_detect,
|
||
.release = amiga7xx_release,
|
||
.queuecommand = NCR53c7xx_queue_command,
|
||
- .abort = NCR53c7xx_abort,
|
||
- .reset = NCR53c7xx_reset,
|
||
+ .eh_abort_handler = NCR53c7xx_abort,
|
||
+ .eh_bus_reset_handler = NCR53c7xx_reset,
|
||
+ .slave_configure = NCR53c7xx_slave_configure,
|
||
.can_queue = 24,
|
||
.this_id = 7,
|
||
.sg_tablesize = 63,
|
||
diff -urN --exclude-from=/usr/src/exclude-file linux-2.6.13-i386/drivers/scsi/amiga7xx.h linux-2.6.13/drivers/scsi/amiga7xx.h
|
||
--- linux-2.6.13-i386/drivers/scsi/amiga7xx.h 2005-08-29 01:41:01.000000000 +0200
|
||
+++ linux-2.6.13/drivers/scsi/amiga7xx.h 1970-01-01 01:00:00.000000000 +0100
|
||
@@ -1,23 +0,0 @@
|
||
-#ifndef AMIGA7XX_H
|
||
-
|
||
-#include <linux/types.h>
|
||
-
|
||
-int amiga7xx_detect(Scsi_Host_Template *);
|
||
-const char *NCR53c7x0_info(void);
|
||
-int NCR53c7xx_queue_command(Scsi_Cmnd *, void (*done)(Scsi_Cmnd *));
|
||
-int NCR53c7xx_abort(Scsi_Cmnd *);
|
||
-int NCR53c7x0_release (struct Scsi_Host *);
|
||
-int NCR53c7xx_reset(Scsi_Cmnd *, unsigned int);
|
||
-void NCR53c7x0_intr(int irq, void *dev_id, struct pt_regs * regs);
|
||
-
|
||
-#ifndef CMD_PER_LUN
|
||
-#define CMD_PER_LUN 3
|
||
-#endif
|
||
-
|
||
-#ifndef CAN_QUEUE
|
||
-#define CAN_QUEUE 24
|
||
-#endif
|
||
-
|
||
-#include <scsi/scsicam.h>
|
||
-
|
||
-#endif /* AMIGA7XX_H */
|
||
diff -urN --exclude-from=/usr/src/exclude-file linux-2.6.13-i386/drivers/scsi/bvme6000.c linux-2.6.13/drivers/scsi/bvme6000.c
|
||
--- linux-2.6.13-i386/drivers/scsi/bvme6000.c 2005-08-29 01:41:01.000000000 +0200
|
||
+++ linux-2.6.13/drivers/scsi/bvme6000.c 2004-10-30 16:35:43.000000000 +0200
|
||
@@ -19,10 +19,16 @@
|
||
#include "scsi.h"
|
||
#include <scsi/scsi_host.h>
|
||
#include "53c7xx.h"
|
||
-#include "bvme6000.h"
|
||
|
||
#include<linux/stat.h>
|
||
|
||
+#ifndef CMD_PER_LUN
|
||
+#define CMD_PER_LUN 3
|
||
+#endif
|
||
+
|
||
+#ifndef CAN_QUEUE
|
||
+#define CAN_QUEUE 24
|
||
+#endif
|
||
|
||
int bvme6000_scsi_detect(Scsi_Host_Template *tpnt)
|
||
{
|
||
@@ -52,8 +58,10 @@
|
||
{
|
||
if (shost->irq)
|
||
free_irq(shost->irq, NULL);
|
||
+#ifdef CONFIG_ISA
|
||
if (shost->dma_channel != 0xff)
|
||
free_dma(shost->dma_channel);
|
||
+#endif
|
||
if (shost->io_port && shost->n_io_port)
|
||
release_region(shost->io_port, shost->n_io_port);
|
||
scsi_unregister(shost);
|
||
@@ -65,8 +73,9 @@
|
||
.detect = bvme6000_scsi_detect,
|
||
.release = bvme6000_scsi_release,
|
||
.queuecommand = NCR53c7xx_queue_command,
|
||
- .abort = NCR53c7xx_abort,
|
||
- .reset = NCR53c7xx_reset,
|
||
+ .eh_abort_handler = NCR53c7xx_abort,
|
||
+ .eh_bus_reset_handler = NCR53c7xx_reset,
|
||
+ .slave_configure = NCR53c7xx_slave_configure,
|
||
.can_queue = 24,
|
||
.this_id = 7,
|
||
.sg_tablesize = 63,
|
||
diff -urN --exclude-from=/usr/src/exclude-file linux-2.6.13-i386/drivers/scsi/bvme6000.h linux-2.6.13/drivers/scsi/bvme6000.h
|
||
--- linux-2.6.13-i386/drivers/scsi/bvme6000.h 2005-08-29 01:41:01.000000000 +0200
|
||
+++ linux-2.6.13/drivers/scsi/bvme6000.h 1970-01-01 01:00:00.000000000 +0100
|
||
@@ -1,24 +0,0 @@
|
||
-#ifndef BVME6000_SCSI_H
|
||
-#define BVME6000_SCSI_H
|
||
-
|
||
-#include <linux/types.h>
|
||
-
|
||
-int bvme6000_scsi_detect(Scsi_Host_Template *);
|
||
-const char *NCR53c7x0_info(void);
|
||
-int NCR53c7xx_queue_command(Scsi_Cmnd *, void (*done)(Scsi_Cmnd *));
|
||
-int NCR53c7xx_abort(Scsi_Cmnd *);
|
||
-int NCR53c7x0_release (struct Scsi_Host *);
|
||
-int NCR53c7xx_reset(Scsi_Cmnd *, unsigned int);
|
||
-void NCR53c7x0_intr(int irq, void *dev_id, struct pt_regs * regs);
|
||
-
|
||
-#ifndef CMD_PER_LUN
|
||
-#define CMD_PER_LUN 3
|
||
-#endif
|
||
-
|
||
-#ifndef CAN_QUEUE
|
||
-#define CAN_QUEUE 24
|
||
-#endif
|
||
-
|
||
-#include <scsi/scsicam.h>
|
||
-
|
||
-#endif /* BVME6000_SCSI_H */
|
||
diff -urN --exclude-from=/usr/src/exclude-file linux-2.6.13-i386/drivers/scsi/mvme16x.c linux-2.6.13/drivers/scsi/mvme16x.c
|
||
--- linux-2.6.13-i386/drivers/scsi/mvme16x.c 2005-08-29 01:41:01.000000000 +0200
|
||
+++ linux-2.6.13/drivers/scsi/mvme16x.c 2004-10-30 16:35:43.000000000 +0200
|
||
@@ -17,10 +17,16 @@
|
||
#include "scsi.h"
|
||
#include <scsi/scsi_host.h>
|
||
#include "53c7xx.h"
|
||
-#include "mvme16x.h"
|
||
|
||
#include<linux/stat.h>
|
||
|
||
+#ifndef CMD_PER_LUN
|
||
+#define CMD_PER_LUN 3
|
||
+#endif
|
||
+
|
||
+#ifndef CAN_QUEUE
|
||
+#define CAN_QUEUE 24
|
||
+#endif
|
||
|
||
int mvme16x_scsi_detect(Scsi_Host_Template *tpnt)
|
||
{
|
||
@@ -54,8 +60,10 @@
|
||
{
|
||
if (shost->irq)
|
||
free_irq(shost->irq, NULL);
|
||
+#ifdef CONFIG_ISA
|
||
if (shost->dma_channel != 0xff)
|
||
free_dma(shost->dma_channel);
|
||
+#endif
|
||
if (shost->io_port && shost->n_io_port)
|
||
release_region(shost->io_port, shost->n_io_port);
|
||
scsi_unregister(shost);
|
||
@@ -67,8 +75,9 @@
|
||
.detect = mvme16x_scsi_detect,
|
||
.release = mvme16x_scsi_release,
|
||
.queuecommand = NCR53c7xx_queue_command,
|
||
- .abort = NCR53c7xx_abort,
|
||
- .reset = NCR53c7xx_reset,
|
||
+ .eh_abort_handler = NCR53c7xx_abort,
|
||
+ .eh_bus_reset_handler = NCR53c7xx_reset,
|
||
+ .slave_configure = NCR53c7xx_slave_configure,
|
||
.can_queue = 24,
|
||
.this_id = 7,
|
||
.sg_tablesize = 63,
|
||
diff -urN --exclude-from=/usr/src/exclude-file linux-2.6.13-i386/drivers/scsi/mvme16x.h linux-2.6.13/drivers/scsi/mvme16x.h
|
||
--- linux-2.6.13-i386/drivers/scsi/mvme16x.h 2005-08-29 01:41:01.000000000 +0200
|
||
+++ linux-2.6.13/drivers/scsi/mvme16x.h 1970-01-01 01:00:00.000000000 +0100
|
||
@@ -1,24 +0,0 @@
|
||
-#ifndef MVME16x_SCSI_H
|
||
-#define MVME16x_SCSI_H
|
||
-
|
||
-#include <linux/types.h>
|
||
-
|
||
-int mvme16x_scsi_detect(Scsi_Host_Template *);
|
||
-const char *NCR53c7x0_info(void);
|
||
-int NCR53c7xx_queue_command(Scsi_Cmnd *, void (*done)(Scsi_Cmnd *));
|
||
-int NCR53c7xx_abort(Scsi_Cmnd *);
|
||
-int NCR53c7x0_release (struct Scsi_Host *);
|
||
-int NCR53c7xx_reset(Scsi_Cmnd *, unsigned int);
|
||
-void NCR53c7x0_intr(int irq, void *dev_id, struct pt_regs * regs);
|
||
-
|
||
-#ifndef CMD_PER_LUN
|
||
-#define CMD_PER_LUN 3
|
||
-#endif
|
||
-
|
||
-#ifndef CAN_QUEUE
|
||
-#define CAN_QUEUE 24
|
||
-#endif
|
||
-
|
||
-#include <scsi/scsicam.h>
|
||
-
|
||
-#endif /* MVME16x_SCSI_H */
|