⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 8390.c

📁 linux1.1源代码
💻 C
📖 第 1 页 / 共 2 页
字号:
				ei_local->lasttx = 10, ei_local->txing = 0;		} else			printk("%s: unexpected TX-done interrupt, lasttx=%d.\n",				   dev->name, ei_local->lasttx);    } else {		ei_local->txing = 0;		dev->tbusy = 0;    }    /* Minimize Tx latency: update the statistics after we restart TXing. */	if (status & ENTSR_COL) ei_local->stat.collisions++;    if (status & ENTSR_PTX)		ei_local->stat.tx_packets++;    else {		ei_local->stat.tx_errors++;		if (status & ENTSR_ABT) ei_local->stat.tx_aborted_errors++;		if (status & ENTSR_CRS) ei_local->stat.tx_carrier_errors++;		if (status & ENTSR_FU)  ei_local->stat.tx_fifo_errors++;		if (status & ENTSR_CDH) ei_local->stat.tx_heartbeat_errors++;		if (status & ENTSR_OWC) ei_local->stat.tx_window_errors++;	}        mark_bh (INET_BH);}/* We have a good packet(s), get it/them out of the buffers. */static void ei_receive(struct device *dev){    int e8390_base = dev->base_addr;    struct ei_device *ei_local = (struct ei_device *) dev->priv;    int rxing_page, this_frame, next_frame, current_offset;    int rx_pkt_count = 0;    struct e8390_pkt_hdr rx_frame;    int num_rx_pages = ei_local->stop_page-ei_local->rx_start_page;        while (++rx_pkt_count < 10) {		int pkt_len;				/* Get the rx page (incoming packet pointer). */		outb_p(E8390_NODMA+E8390_PAGE1, e8390_base + E8390_CMD);		rxing_page = inb_p(e8390_base + EN1_CURPAG);		outb_p(E8390_NODMA+E8390_PAGE0, e8390_base + E8390_CMD);				/* Remove one frame from the ring.  Boundary is alway a page behind. */		this_frame = inb_p(e8390_base + EN0_BOUNDARY) + 1;		if (this_frame >= ei_local->stop_page)			this_frame = ei_local->rx_start_page;				/* Someday we'll omit the previous, iff we never get this message.		   (There is at least one clone claimed to have a problem.)  */		if (ei_debug > 0  &&  this_frame != ei_local->current_page)			printk("%s: mismatched read page pointers %2x vs %2x.\n",				   dev->name, this_frame, ei_local->current_page);				if (this_frame == rxing_page)	/* Read all the frames? */			break;				/* Done for now */				current_offset = this_frame << 8;		ei_block_input(dev, sizeof(rx_frame), (char *)&rx_frame,					   current_offset);				pkt_len = rx_frame.count - sizeof(rx_frame);				next_frame = this_frame + 1 + ((pkt_len+4)>>8);				/* Check for bogosity warned by 3c503 book: the status byte is never		   written.  This happened a lot during testing! This code should be		   cleaned up someday. */		if (rx_frame.next != next_frame			&& rx_frame.next != next_frame + 1			&& rx_frame.next != next_frame - num_rx_pages			&& rx_frame.next != next_frame + 1 - num_rx_pages) {			ei_local->current_page = rxing_page;			outb(ei_local->current_page-1, e8390_base+EN0_BOUNDARY);			ei_local->stat.rx_errors++;			continue;		}		if (pkt_len < 60  ||  pkt_len > 1518) {			if (ei_debug)				printk("%s: bogus packet size: %d, status=%#2x nxpg=%#2x.\n",					   dev->name, rx_frame.count, rx_frame.status,					   rx_frame.next);			ei_local->stat.rx_errors++;		} else if ((rx_frame.status & 0x0F) == ENRSR_RXOK) {			int sksize = sizeof(struct sk_buff) + pkt_len;			struct sk_buff *skb;						skb = alloc_skb(sksize, GFP_ATOMIC);			if (skb == NULL) {				if (ei_debug)					printk("%s: Couldn't allocate a sk_buff of size %d.\n",						   dev->name, sksize);				ei_local->stat.rx_dropped++;				break;			} else {				skb->mem_len = sksize;				skb->mem_addr = skb;				skb->len = pkt_len;				skb->dev = dev;								ei_block_input(dev, pkt_len, (char *) skb->data,							   current_offset + sizeof(rx_frame));				netif_rx(skb);				ei_local->stat.rx_packets++;			}		} else {			int errs = rx_frame.status;			if (ei_debug)				printk("%s: bogus packet: status=%#2x nxpg=%#2x size=%d\n",					   dev->name, rx_frame.status, rx_frame.next,					   rx_frame.count);			if (errs & ENRSR_FO)				ei_local->stat.rx_fifo_errors++;		}		next_frame = rx_frame.next;				/* This _should_ never happen: it's here for avoiding bad clones. */		if (next_frame >= ei_local->stop_page) {			printk("%s: next frame inconsistency, %#2x..", dev->name,				   next_frame);			next_frame = ei_local->rx_start_page;		}		ei_local->current_page = next_frame;		outb(next_frame-1, e8390_base+EN0_BOUNDARY);    }    /* If any worth-while packets have been received, dev_rint()       has done a mark_bh(INET_BH) for us and will work on them       when we get to the bottom-half routine. */	/* Record the maximum Rx packet queue. */	if (rx_pkt_count > high_water_mark)		high_water_mark = rx_pkt_count;    /* Bug alert!  Reset ENISR_OVER to avoid spurious overruns! */    outb_p(ENISR_RX+ENISR_RX_ERR+ENISR_OVER, e8390_base+EN0_ISR);    return;}/* We have a receiver overrun: we have to kick the 8390 to get it started   again.*/static void ei_rx_overrun(struct device *dev){    int e8390_base = dev->base_addr;    int reset_start_time = jiffies;    struct ei_device *ei_local = (struct ei_device *) dev->priv;        /* We should already be stopped and in page0.  Remove after testing. */    outb_p(E8390_NODMA+E8390_PAGE0+E8390_STOP, e8390_base+E8390_CMD);        if (ei_debug)		printk("%s: Receiver overrun.\n", dev->name);    ei_local->stat.rx_over_errors++;        /* The old Biro driver does dummy = inb_p( RBCR[01] ); at this point.       It might mean something -- magic to speed up a reset?  A 8390 bug?*/        /* Wait for the reset to complete.	This should happen almost instantly,	   but could take up to 1.5msec in certain rare instances.  There is no	   easy way of timing something in that range, so we use 'jiffies' as	   a sanity check. */    while ((inb_p(e8390_base+EN0_ISR) & ENISR_RESET) == 0)		if (jiffies - reset_start_time > 1) {			printk("%s: reset did not complete at ei_rx_overrun.\n",				   dev->name);			NS8390_init(dev, 1);			return;		}        /* Remove packets right away. */    ei_receive(dev);        outb_p(0xff, e8390_base+EN0_ISR);    /* Generic 8390 insns to start up again, same as in open_8390(). */    outb_p(E8390_NODMA + E8390_PAGE0 + E8390_START, e8390_base + E8390_CMD);    outb_p(E8390_TXCONFIG, e8390_base + EN0_TXCR); /* xmit on. */}static struct enet_statistics *get_stats(struct device *dev){    short ioaddr = dev->base_addr;    struct ei_device *ei_local = (struct ei_device *) dev->priv;        /* Read the counter registers, assuming we are in page 0. */    ei_local->stat.rx_frame_errors += inb_p(ioaddr + EN0_COUNTER0);    ei_local->stat.rx_crc_errors   += inb_p(ioaddr + EN0_COUNTER1);    ei_local->stat.rx_missed_errors+= inb_p(ioaddr + EN0_COUNTER2);        return &ei_local->stat;}#ifdef HAVE_MULTICAST/* Set or clear the multicast filter for this adaptor.   num_addrs == -1	Promiscuous mode, receive all packets   num_addrs == 0	Normal mode, clear multicast list   num_addrs > 0	Multicast mode, receive normal and MC packets, and do   .   				best-effort filtering.   */static void set_multicast_list(struct device *dev, int num_addrs, void *addrs){    short ioaddr = dev->base_addr;        if (num_addrs > 0) {		/* The multicast-accept list is initialized to accept-all, and we		   rely on higher-level filtering for now. */		outb_p(E8390_RXCONFIG | 0x08, ioaddr + EN0_RXCR);    } else if (num_addrs < 0)		outb_p(E8390_RXCONFIG | 0x10, ioaddr + EN0_RXCR);    else		outb_p(E8390_RXCONFIG, ioaddr + EN0_RXCR);}#endif/* Initialize the rest of the 8390 device structure. */int ethdev_init(struct device *dev){    int i;        if (ei_debug > 1)		printk(version);        if (dev->priv == NULL) {		struct ei_device *ei_local;				dev->priv = kmalloc(sizeof(struct ei_device), GFP_KERNEL);		memset(dev->priv, 0, sizeof(struct ei_device));		ei_local = (struct ei_device *)dev->priv;#ifndef NO_PINGPONG		ei_local->pingpong = 1;#endif    }        /* The open call may be overridden by the card-specific code. */    if (dev->open == NULL)		dev->open = &ei_open;    /* We should have a dev->stop entry also. */    dev->hard_start_xmit = &ei_start_xmit;    dev->get_stats	= get_stats;#ifdef HAVE_MULTICAST    dev->set_multicast_list = &set_multicast_list;#endif        for (i = 0; i < DEV_NUMBUFFS; i++)		dev->buffs[i] = NULL;        dev->hard_header	= eth_header;    dev->add_arp		= eth_add_arp;    dev->queue_xmit		= dev_queue_xmit;    dev->rebuild_header	= eth_rebuild_header;    dev->type_trans		= eth_type_trans;        dev->type		= ARPHRD_ETHER;    dev->hard_header_len = ETH_HLEN;    dev->mtu		= 1500; /* eth_mtu */    dev->addr_len	= ETH_ALEN;    for (i = 0; i < ETH_ALEN; i++) {		dev->broadcast[i]=0xff;    }        /* New-style flags. */    dev->flags		= IFF_BROADCAST;    dev->family		= AF_INET;    dev->pa_addr	= 0;    dev->pa_brdaddr	= 0;    dev->pa_mask	= 0;    dev->pa_alen	= sizeof(unsigned long);        return 0;}/* This page of functions should be 8390 generic *//* Follow National Semi's recommendations for initializing the "NIC". */void NS8390_init(struct device *dev, int startp){    int e8390_base = dev->base_addr;    struct ei_device *ei_local = (struct ei_device *) dev->priv;    int i;    int endcfg = ei_local->word16 ? (0x48 | ENDCFG_WTS) : 0x48;        /* Follow National Semi's recommendations for initing the DP83902. */    outb_p(E8390_NODMA+E8390_PAGE0+E8390_STOP, e8390_base); /* 0x21 */    outb_p(endcfg, e8390_base + EN0_DCFG);	/* 0x48 or 0x49 */    /* Clear the remote byte count registers. */    outb_p(0x00,  e8390_base + EN0_RCNTLO);    outb_p(0x00,  e8390_base + EN0_RCNTHI);    /* Set to monitor and loopback mode -- this is vital!. */    outb_p(E8390_RXOFF, e8390_base + EN0_RXCR); /* 0x20 */    outb_p(E8390_TXOFF, e8390_base + EN0_TXCR); /* 0x02 */    /* Set the transmit page and receive ring. */    outb_p(ei_local->tx_start_page,	 e8390_base + EN0_TPSR);    ei_local->tx1 = ei_local->tx2 = 0;    outb_p(ei_local->rx_start_page,	 e8390_base + EN0_STARTPG);    outb_p(ei_local->stop_page-1, e8390_base + EN0_BOUNDARY); /* 3c503 says 0x3f,NS0x26*/    ei_local->current_page = ei_local->rx_start_page;		/* assert boundary+1 */    outb_p(ei_local->stop_page,	  e8390_base + EN0_STOPPG);    /* Clear the pending interrupts and mask. */    outb_p(0xFF, e8390_base + EN0_ISR);    outb_p(0x00,  e8390_base + EN0_IMR);        /* Copy the station address into the DS8390 registers,       and set the multicast hash bitmap to receive all multicasts. */    cli();    outb_p(E8390_NODMA + E8390_PAGE1 + E8390_STOP, e8390_base); /* 0x61 */    for(i = 0; i < 6; i++) {		outb_p(dev->dev_addr[i], e8390_base + EN1_PHYS + i);    }    /* Initialize the multicast list to accept-all.  If we enable multicast       the higher levels can do the filtering. */    for(i = 0; i < 8; i++)		outb_p(0xff, e8390_base + EN1_MULT + i);        outb_p(ei_local->rx_start_page,	 e8390_base + EN1_CURPAG);    outb_p(E8390_NODMA+E8390_PAGE0+E8390_STOP, e8390_base);    sti();    dev->tbusy = 0;    dev->interrupt = 0;    ei_local->tx1 = ei_local->tx2 = 0;    ei_local->txing = 0;    if (startp) {		outb_p(0xff,  e8390_base + EN0_ISR);		outb_p(ENISR_ALL,  e8390_base + EN0_IMR);		outb_p(E8390_NODMA+E8390_PAGE0+E8390_START, e8390_base);		outb_p(E8390_TXCONFIG, e8390_base + EN0_TXCR); /* xmit on. */		/* 3c503 TechMan says rxconfig only after the NIC is started. */		outb_p(E8390_RXCONFIG,	e8390_base + EN0_RXCR); /* rx on,  */    }    return;}/* Trigger a transmit start, assuming the length is valid. */static void NS8390_trigger_send(struct device *dev, unsigned int length,								int start_page){    int e8390_base = dev->base_addr;        ei_status.txing = 1;    outb_p(E8390_NODMA+E8390_PAGE0, e8390_base);        if (inb_p(e8390_base) & E8390_TRANS) {		printk("%s: trigger_send() called with the transmitter busy.\n",			   dev->name);		return;    }    outb_p(length & 0xff, e8390_base + EN0_TCNTLO);    outb_p(length >> 8, e8390_base + EN0_TCNTHI);    outb_p(start_page, e8390_base + EN0_TPSR);    outb_p(E8390_NODMA+E8390_TRANS+E8390_START, e8390_base);    return;}/* * Local variables: *  compile-command: "gcc -D__KERNEL__ -I/usr/src/linux/net/inet -Wall -Wstrict-prototypes -O6 -m486 -c 8390.c" *  version-control: t *  kept-new-versions: 5 *  c-indent-level: 4 *  tab-width: 4 * End: */

⌨️ 快捷键说明

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