declance.c

来自「linux 内核源代码」· C语言 代码 · 共 1,382 行 · 第 1/3 页

C
1,382
字号
/* *    Lance ethernet driver for the MIPS processor based *      DECstation family * * *      adopted from sunlance.c by Richard van den Berg * *      Copyright (C) 2002, 2003, 2005, 2006  Maciej W. Rozycki * *      additional sources: *      - PMAD-AA TURBOchannel Ethernet Module Functional Specification, *        Revision 1.2 * *      History: * *      v0.001: The kernel accepts the code and it shows the hardware address. * *      v0.002: Removed most sparc stuff, left only some module and dma stuff. * *      v0.003: Enhanced base address calculation from proposals by *              Harald Koerfgen and Thomas Riemer. * *      v0.004: lance-regs is pointing at the right addresses, added prom *              check. First start of address mapping and DMA. * *      v0.005: started to play around with LANCE-DMA. This driver will not *              work for non IOASIC lances. HK * *      v0.006: added pointer arrays to lance_private and setup routine for *              them in dec_lance_init. HK * *      v0.007: Big shit. The LANCE seems to use a different DMA mechanism to *              access the init block. This looks like one (short) word at a *              time, but the smallest amount the IOASIC can transfer is a *              (long) word. So we have a 2-2 padding here. Changed *              lance_init_block accordingly. The 16-16 padding for the buffers *              seems to be correct. HK * *      v0.008: mods to make PMAX_LANCE work. 01/09/1999 triemer * *      v0.009: Module support fixes, multiple interfaces support, various *              bits. macro * *      v0.010: Fixes for the PMAD mapping of the LANCE buffer and for the *              PMAX requirement to only use halfword accesses to the *              buffer. macro * *      v0.011: Converted the PMAD to the driver model. macro */#include <linux/crc32.h>#include <linux/delay.h>#include <linux/errno.h>#include <linux/if_ether.h>#include <linux/init.h>#include <linux/kernel.h>#include <linux/module.h>#include <linux/netdevice.h>#include <linux/etherdevice.h>#include <linux/spinlock.h>#include <linux/stddef.h>#include <linux/string.h>#include <linux/tc.h>#include <linux/types.h>#include <asm/addrspace.h>#include <asm/system.h>#include <asm/dec/interrupts.h>#include <asm/dec/ioasic.h>#include <asm/dec/ioasic_addrs.h>#include <asm/dec/kn01.h>#include <asm/dec/machtype.h>#include <asm/dec/system.h>static char version[] __devinitdata ="declance.c: v0.011 by Linux MIPS DECstation task force\n";MODULE_AUTHOR("Linux MIPS DECstation task force");MODULE_DESCRIPTION("DEC LANCE (DECstation onboard, PMAD-xx) driver");MODULE_LICENSE("GPL");#define __unused __attribute__ ((unused))/* * card types */#define ASIC_LANCE 1#define PMAD_LANCE 2#define PMAX_LANCE 3#define LE_CSR0 0#define LE_CSR1 1#define LE_CSR2 2#define LE_CSR3 3#define LE_MO_PROM      0x8000	/* Enable promiscuous mode */#define	LE_C0_ERR	0x8000	/* Error: set if BAB, SQE, MISS or ME is set */#define	LE_C0_BABL	0x4000	/* BAB:  Babble: tx timeout. */#define	LE_C0_CERR	0x2000	/* SQE:  Signal quality error */#define	LE_C0_MISS	0x1000	/* MISS: Missed a packet */#define	LE_C0_MERR	0x0800	/* ME:   Memory error */#define	LE_C0_RINT	0x0400	/* Received interrupt */#define	LE_C0_TINT	0x0200	/* Transmitter Interrupt */#define	LE_C0_IDON	0x0100	/* IFIN: Init finished. */#define	LE_C0_INTR	0x0080	/* Interrupt or error */#define	LE_C0_INEA	0x0040	/* Interrupt enable */#define	LE_C0_RXON	0x0020	/* Receiver on */#define	LE_C0_TXON	0x0010	/* Transmitter on */#define	LE_C0_TDMD	0x0008	/* Transmitter demand */#define	LE_C0_STOP	0x0004	/* Stop the card */#define	LE_C0_STRT	0x0002	/* Start the card */#define	LE_C0_INIT	0x0001	/* Init the card */#define	LE_C3_BSWP	0x4	/* SWAP */#define	LE_C3_ACON	0x2	/* ALE Control */#define	LE_C3_BCON	0x1	/* Byte control *//* Receive message descriptor 1 */#define LE_R1_OWN	0x8000	/* Who owns the entry */#define LE_R1_ERR	0x4000	/* Error: if FRA, OFL, CRC or BUF is set */#define LE_R1_FRA	0x2000	/* FRA: Frame error */#define LE_R1_OFL	0x1000	/* OFL: Frame overflow */#define LE_R1_CRC	0x0800	/* CRC error */#define LE_R1_BUF	0x0400	/* BUF: Buffer error */#define LE_R1_SOP	0x0200	/* Start of packet */#define LE_R1_EOP	0x0100	/* End of packet */#define LE_R1_POK	0x0300	/* Packet is complete: SOP + EOP *//* Transmit message descriptor 1 */#define LE_T1_OWN	0x8000	/* Lance owns the packet */#define LE_T1_ERR	0x4000	/* Error summary */#define LE_T1_EMORE	0x1000	/* Error: more than one retry needed */#define LE_T1_EONE	0x0800	/* Error: one retry needed */#define LE_T1_EDEF	0x0400	/* Error: deferred */#define LE_T1_SOP	0x0200	/* Start of packet */#define LE_T1_EOP	0x0100	/* End of packet */#define LE_T1_POK	0x0300	/* Packet is complete: SOP + EOP */#define LE_T3_BUF       0x8000	/* Buffer error */#define LE_T3_UFL       0x4000	/* Error underflow */#define LE_T3_LCOL      0x1000	/* Error late collision */#define LE_T3_CLOS      0x0800	/* Error carrier loss */#define LE_T3_RTY       0x0400	/* Error retry */#define LE_T3_TDR       0x03ff	/* Time Domain Reflectometry counter *//* Define: 2^4 Tx buffers and 2^4 Rx buffers */#ifndef LANCE_LOG_TX_BUFFERS#define LANCE_LOG_TX_BUFFERS 4#define LANCE_LOG_RX_BUFFERS 4#endif#define TX_RING_SIZE			(1 << (LANCE_LOG_TX_BUFFERS))#define TX_RING_MOD_MASK		(TX_RING_SIZE - 1)#define RX_RING_SIZE			(1 << (LANCE_LOG_RX_BUFFERS))#define RX_RING_MOD_MASK		(RX_RING_SIZE - 1)#define PKT_BUF_SZ		1536#define RX_BUFF_SIZE            PKT_BUF_SZ#define TX_BUFF_SIZE            PKT_BUF_SZ#undef TEST_HITS#define ZERO 0/* * The DS2100/3100 have a linear 64 kB buffer which supports halfword * accesses only.  Each halfword of the buffer is word-aligned in the * CPU address space. * * The PMAD-AA has a 128 kB buffer on-board. * * The IOASIC LANCE devices use a shared memory region.  This region * as seen from the CPU is (max) 128 kB long and has to be on an 128 kB * boundary.  The LANCE sees this as a 64 kB long continuous memory * region. * * The LANCE's DMA address is used as an index in this buffer and DMA * takes place in bursts of eight 16-bit words which are packed into * four 32-bit words by the IOASIC.  This leads to a strange padding: * 16 bytes of valid data followed by a 16 byte gap :-(. */struct lance_rx_desc {	unsigned short rmd0;		/* low address of packet */	unsigned short rmd1;		/* high address of packet					   and descriptor bits */	short length;			/* 2s complement (negative!)					   of buffer length */	unsigned short mblength;	/* actual number of bytes received */};struct lance_tx_desc {	unsigned short tmd0;		/* low address of packet */	unsigned short tmd1;		/* high address of packet					   and descriptor bits */	short length;			/* 2s complement (negative!)					   of buffer length */	unsigned short misc;};/* First part of the LANCE initialization block, described in databook. */struct lance_init_block {	unsigned short mode;		/* pre-set mode (reg. 15) */	unsigned short phys_addr[3];	/* physical ethernet address */	unsigned short filter[4];	/* multicast filter */	/* Receive and transmit ring base, along with extra bits. */	unsigned short rx_ptr;		/* receive descriptor addr */	unsigned short rx_len;		/* receive len and high addr */	unsigned short tx_ptr;		/* transmit descriptor addr */	unsigned short tx_len;		/* transmit len and high addr */	short gap[4];	/* The buffer descriptors */	struct lance_rx_desc brx_ring[RX_RING_SIZE];	struct lance_tx_desc btx_ring[TX_RING_SIZE];};#define BUF_OFFSET_CPU sizeof(struct lance_init_block)#define BUF_OFFSET_LNC sizeof(struct lance_init_block)#define shift_off(off, type)						\	(type == ASIC_LANCE || type == PMAX_LANCE ? off << 1 : off)#define lib_off(rt, type)						\	shift_off(offsetof(struct lance_init_block, rt), type)#define lib_ptr(ib, rt, type) 						\	((volatile u16 *)((u8 *)(ib) + lib_off(rt, type)))#define rds_off(rt, type)						\	shift_off(offsetof(struct lance_rx_desc, rt), type)#define rds_ptr(rd, rt, type) 						\	((volatile u16 *)((u8 *)(rd) + rds_off(rt, type)))#define tds_off(rt, type)						\	shift_off(offsetof(struct lance_tx_desc, rt), type)#define tds_ptr(td, rt, type) 						\	((volatile u16 *)((u8 *)(td) + tds_off(rt, type)))struct lance_private {	struct net_device *next;	int type;	int dma_irq;	volatile struct lance_regs *ll;	spinlock_t	lock;	int rx_new, tx_new;	int rx_old, tx_old;	unsigned short busmaster_regval;	struct timer_list       multicast_timer;	/* Pointers to the ring buffers as seen from the CPU */	char *rx_buf_ptr_cpu[RX_RING_SIZE];	char *tx_buf_ptr_cpu[TX_RING_SIZE];	/* Pointers to the ring buffers as seen from the LANCE */	uint rx_buf_ptr_lnc[RX_RING_SIZE];	uint tx_buf_ptr_lnc[TX_RING_SIZE];};#define TX_BUFFS_AVAIL ((lp->tx_old<=lp->tx_new)?\			lp->tx_old+TX_RING_MOD_MASK-lp->tx_new:\			lp->tx_old - lp->tx_new-1)/* The lance control ports are at an absolute address, machine and tc-slot * dependent. * DECstations do only 32-bit access and the LANCE uses 16 bit addresses, * so we have to give the structure an extra member making rap pointing * at the right address */struct lance_regs {	volatile unsigned short rdp;	/* register data port */	unsigned short pad;	volatile unsigned short rap;	/* register address port */};int dec_lance_debug = 2;static struct tc_driver dec_lance_tc_driver;static struct net_device *root_lance_dev;static inline void writereg(volatile unsigned short *regptr, short value){	*regptr = value;	iob();}/* Load the CSR registers */static void load_csrs(struct lance_private *lp){	volatile struct lance_regs *ll = lp->ll;	uint leptr;	/* The address space as seen from the LANCE	 * begins at address 0. HK	 */	leptr = 0;	writereg(&ll->rap, LE_CSR1);	writereg(&ll->rdp, (leptr & 0xFFFF));	writereg(&ll->rap, LE_CSR2);	writereg(&ll->rdp, leptr >> 16);	writereg(&ll->rap, LE_CSR3);	writereg(&ll->rdp, lp->busmaster_regval);	/* Point back to csr0 */	writereg(&ll->rap, LE_CSR0);}/* * Our specialized copy routines * */static void cp_to_buf(const int type, void *to, const void *from, int len){	unsigned short *tp, *fp, clen;	unsigned char *rtp, *rfp;	if (type == PMAD_LANCE) {		memcpy(to, from, len);	} else if (type == PMAX_LANCE) {		clen = len >> 1;		tp = (unsigned short *) to;		fp = (unsigned short *) from;		while (clen--) {			*tp++ = *fp++;			tp++;		}		clen = len & 1;		rtp = (unsigned char *) tp;		rfp = (unsigned char *) fp;		while (clen--) {			*rtp++ = *rfp++;		}	} else {		/*		 * copy 16 Byte chunks		 */		clen = len >> 4;		tp = (unsigned short *) to;		fp = (unsigned short *) from;		while (clen--) {			*tp++ = *fp++;			*tp++ = *fp++;			*tp++ = *fp++;			*tp++ = *fp++;			*tp++ = *fp++;			*tp++ = *fp++;			*tp++ = *fp++;			*tp++ = *fp++;			tp += 8;		}		/*		 * do the rest, if any.		 */		clen = len & 15;		rtp = (unsigned char *) tp;		rfp = (unsigned char *) fp;		while (clen--) {			*rtp++ = *rfp++;		}	}	iob();}static void cp_from_buf(const int type, void *to, const void *from, int len){	unsigned short *tp, *fp, clen;	unsigned char *rtp, *rfp;	if (type == PMAD_LANCE) {		memcpy(to, from, len);	} else if (type == PMAX_LANCE) {		clen = len >> 1;		tp = (unsigned short *) to;		fp = (unsigned short *) from;		while (clen--) {			*tp++ = *fp++;			fp++;		}		clen = len & 1;		rtp = (unsigned char *) tp;		rfp = (unsigned char *) fp;		while (clen--) {			*rtp++ = *rfp++;		}	} else {		/*		 * copy 16 Byte chunks		 */		clen = len >> 4;		tp = (unsigned short *) to;		fp = (unsigned short *) from;		while (clen--) {			*tp++ = *fp++;			*tp++ = *fp++;			*tp++ = *fp++;			*tp++ = *fp++;			*tp++ = *fp++;			*tp++ = *fp++;			*tp++ = *fp++;			*tp++ = *fp++;			fp += 8;		}		/*		 * do the rest, if any.		 */		clen = len & 15;		rtp = (unsigned char *) tp;		rfp = (unsigned char *) fp;		while (clen--) {			*rtp++ = *rfp++;		}	}}/* Setup the Lance Rx and Tx rings */static void lance_init_ring(struct net_device *dev){	struct lance_private *lp = netdev_priv(dev);	volatile u16 *ib = (volatile u16 *)dev->mem_start;	uint leptr;	int i;	/* Lock out other processes while setting up hardware */	netif_stop_queue(dev);	lp->rx_new = lp->tx_new = 0;	lp->rx_old = lp->tx_old = 0;	/* Copy the ethernet address to the lance init block.	 * XXX bit 0 of the physical address registers has to be zero	 */	*lib_ptr(ib, phys_addr[0], lp->type) = (dev->dev_addr[1] << 8) |				     dev->dev_addr[0];	*lib_ptr(ib, phys_addr[1], lp->type) = (dev->dev_addr[3] << 8) |				     dev->dev_addr[2];

⌨️ 快捷键说明

复制代码Ctrl + C
搜索代码Ctrl + F
全屏模式F11
增大字号Ctrl + =
减小字号Ctrl + -
显示快捷键?