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

📄 starfire.c

📁 Linux下各种网卡的驱动程序
💻 C
📖 第 1 页 / 共 4 页
字号:
	if ((np->msg_level & NETIF_MSG_RX_STATUS)		|| memcmp(np->pad0, np->pad0 + 1, sizeof(np->pad0) -1))		printk(KERN_DEBUG "  exiting netdev_rx() status of %d was %8.8x %d.\n",			   np->rx_done, desc_status,			   memcmp(np->pad0, np->pad0 + 1, sizeof(np->pad0) -1));	return 0;}static void netdev_error(struct net_device *dev, int intr_status){	struct netdev_private *np = (struct netdev_private *)dev->priv;	if (intr_status & LinkChange) {		int phy_num = np->phys[0];		if (np->msg_level & NETIF_MSG_LINK)			printk(KERN_NOTICE "%s: Link changed: Autonegotiation advertising"				   " %4.4x  partner %4.4x.\n", dev->name,				   mdio_read(dev, phy_num, 4),				   mdio_read(dev, phy_num, 5));		/* Clear sticky bit. */		mdio_read(dev, phy_num, 1);		/* If link beat has returned... */		if (mdio_read(dev, phy_num, 1) & 0x0004)			netif_link_up(dev);		else			netif_link_down(dev);		check_duplex(dev);	}	if (intr_status & StatsMax) {		get_stats(dev);	}	/* Came close to underrunning the Tx FIFO, increase threshold. */	if (intr_status & IntrTxDataLow)		writel(++np->tx_threshold, dev->base_addr + TxThreshold);	/* Ingore expected normal events, and handled abnormal events. */	if ((intr_status &		 ~(IntrAbnormalSummary|LinkChange|StatsMax|IntrTxDataLow| 0xFF01))		&& (np->msg_level & NETIF_MSG_DRV))		printk(KERN_ERR "%s: Something Wicked happened! %4.4x.\n",			   dev->name, intr_status);	/* Hmmmmm, it's not clear how to recover from PCI faults. */	if (intr_status & IntrTxPCIErr)		np->stats.tx_fifo_errors++;	if (intr_status & IntrRxPCIErr)		np->stats.rx_fifo_errors++;}static struct net_device_stats *get_stats(struct net_device *dev){	long ioaddr = dev->base_addr;	struct netdev_private *np = (struct netdev_private *)dev->priv;	/* This adapter architecture needs no SMP locks. */#if LINUX_VERSION_CODE > 0x20119	np->stats.tx_bytes = readl(ioaddr + 0x57010);	np->stats.rx_bytes = readl(ioaddr + 0x57044);#endif	np->stats.tx_packets = readl(ioaddr + 0x57000);	np->stats.tx_aborted_errors =		readl(ioaddr + 0x57024) + readl(ioaddr + 0x57028);	np->stats.tx_window_errors = readl(ioaddr + 0x57018);	np->stats.collisions = readl(ioaddr + 0x57004) + readl(ioaddr + 0x57008);	/* The chip only need report frame silently dropped. */	np->stats.rx_dropped	   += readw(ioaddr + RxDMAStatus);	writew(0, ioaddr + RxDMAStatus);	np->stats.rx_crc_errors	   = readl(ioaddr + 0x5703C);	np->stats.rx_frame_errors = readl(ioaddr + 0x57040);	np->stats.rx_length_errors = readl(ioaddr + 0x57058);	np->stats.rx_missed_errors = readl(ioaddr + 0x5707C);	return &np->stats;}/* The little-endian AUTODIN II ethernet CRC calculations.   A big-endian version is also available.   This is slow but compact code.  Do not use this routine for bulk data,   use a table-based routine instead.   This is common code and should be moved to net/core/crc.c.   Chips may use the upper or lower CRC bits, and may reverse and/or invert   them.  Select the endian-ness that results in minimal calculations.*/static unsigned const ethernet_polynomial_le = 0xedb88320U;static inline unsigned ether_crc_le(int length, unsigned char *data){	unsigned int crc = ~0;	/* Initial value. */	while(--length >= 0) {		unsigned char current_octet = *data++;		int bit;		for (bit = 8; --bit >= 0; current_octet >>= 1) {			if ((crc ^ current_octet) & 1) {				crc >>= 1;				crc ^= ethernet_polynomial_le;			} else				crc >>= 1;		}	}	return crc;}static void set_rx_mode(struct net_device *dev){	struct netdev_private *np = (struct netdev_private *)dev->priv;	long ioaddr = dev->base_addr;	u32 rx_mode;	struct dev_mc_list *mclist;	int i;	if (dev->flags & IFF_PROMISC) {			/* Set promiscuous. */		/* Unconditionally log net taps. */		printk(KERN_NOTICE "%s: Promiscuous mode enabled.\n", dev->name);		rx_mode = AcceptBroadcast|AcceptAllMulticast|AcceptAll|AcceptMyPhys;	} else if ((dev->mc_count > np->multicast_filter_limit)			   ||  (dev->flags & IFF_ALLMULTI)) {		/* Too many to match, or accept all multicasts. */		rx_mode = AcceptBroadcast|AcceptAllMulticast|AcceptMyPhys;	} else if (dev->mc_count <= 15) {		/* Use the 16 element perfect filter. */		long filter_addr = ioaddr + 0x56000 + 1*16;		for (i = 1, mclist = dev->mc_list; mclist  &&  i <= dev->mc_count;			 i++, mclist = mclist->next) {			u16 *eaddrs = (u16 *)mclist->dmi_addr;			writew(cpu_to_be16(eaddrs[2]), filter_addr); filter_addr += 4;			writew(cpu_to_be16(eaddrs[1]), filter_addr); filter_addr += 4;			writew(cpu_to_be16(eaddrs[0]), filter_addr); filter_addr += 8;		}		while (i++ < 16) {			writew(0xffff, filter_addr); filter_addr += 4;			writew(0xffff, filter_addr); filter_addr += 4;			writew(0xffff, filter_addr); filter_addr += 8;		}		rx_mode = AcceptBroadcast | AcceptMyPhys;	} else {		/* Must use a multicast hash table. */		long filter_addr;		u16 mc_filter[32];			/* Multicast hash filter */		memset(mc_filter, 0, sizeof(mc_filter));		for (i = 0, mclist = dev->mc_list; mclist && i < dev->mc_count;			 i++, mclist = mclist->next) {			set_bit(ether_crc_le(ETH_ALEN, mclist->dmi_addr) >> 23, mc_filter);		}		/* Clear the perfect filter list. */		filter_addr = ioaddr + 0x56000 + 1*16;		for (i = 1; i < 16; i++) {			writew(0xffff, filter_addr); filter_addr += 4;			writew(0xffff, filter_addr); filter_addr += 4;			writew(0xffff, filter_addr); filter_addr += 8;		}		for (filter_addr=ioaddr + 0x56100, i=0; i < 32; filter_addr+= 16, i++){			np->mc_filter[i] = mc_filter[i];			writew(mc_filter[i], filter_addr);		}		rx_mode = AcceptBroadcast | AcceptMulticast | AcceptMyPhys;	}	writel(rx_mode, ioaddr + RxFilterMode);}/*  Handle user-level ioctl() calls.  We must use two numeric constants as the key because some clueless person  changed the value for the symbolic name.*/static int mii_ioctl(struct net_device *dev, struct ifreq *rq, int cmd){	struct netdev_private *np = (struct netdev_private *)dev->priv;	u16 *data = (u16 *)&rq->ifr_data;	u32 *data32 = (void *)&rq->ifr_data;	switch(cmd) {	case 0x8947: case 0x89F0:		/* SIOCGMIIPHY: Get the address of the PHY in use. */		data[0] = np->phys[0] & 0x1f;		/* Fall Through */	case 0x8948: case 0x89F1:		/* SIOCGMIIREG: Read the specified MII register. */		data[3] = mdio_read(dev, data[0] & 0x1f, data[1] & 0x1f);		return 0;	case 0x8949: case 0x89F2:		/* SIOCSMIIREG: Write the specified MII register */		if (!capable(CAP_NET_ADMIN))			return -EPERM;		if (data[0] == np->phys[0]) {			u16 value = data[2];			switch (data[1]) {			case 0:				/* Check for autonegotiation on or reset. */				np->medialock = (value & 0x9000) ? 0 : 1;				if (np->medialock)					np->full_duplex = (value & 0x0100) ? 1 : 0;				break;			case 4: np->advertising = value; break;			}			check_duplex(dev);		}		mdio_write(dev, data[0] & 0x1f, data[1] & 0x1f, data[2]);		return 0;	case SIOCGPARAMS:		data32[0] = np->msg_level;		data32[1] = np->multicast_filter_limit;		data32[2] = np->max_interrupt_work;		data32[3] = np->rx_copybreak;		return 0;	case SIOCSPARAMS:		if (!capable(CAP_NET_ADMIN))			return -EPERM;		np->msg_level = data32[0];		np->multicast_filter_limit = data32[1];		np->max_interrupt_work = data32[2];		np->rx_copybreak = data32[3];		return 0;	default:		return -EOPNOTSUPP;	}}static int netdev_close(struct net_device *dev){	long ioaddr = dev->base_addr;	struct netdev_private *np = (struct netdev_private *)dev->priv;	int i;	netif_stop_tx_queue(dev);	if (np->msg_level & NETIF_MSG_IFDOWN) {		printk(KERN_DEBUG "%s: Shutting down ethercard, Intr status %4.4x.\n",			   dev->name, (int)readl(ioaddr + IntrStatus));		printk(KERN_DEBUG "%s: Queue pointers were Tx %d / %d,  Rx %d / %d.\n",			   dev->name, np->cur_tx, np->dirty_tx, np->cur_rx, np->dirty_rx);	}	/* Disable interrupts by clearing the interrupt mask. */	writel(0, ioaddr + IntrEnable);	/* Stop the chip's Tx and Rx processes. */	writel(0, ioaddr + GenCtrl);	del_timer(&np->timer);#ifdef __i386__	if (np->msg_level & NETIF_MSG_IFDOWN) {		printk("\n"KERN_DEBUG"  Tx ring at %8.8x:\n",			   (int)virt_to_bus(np->tx_ring));		for (i = 0; i < 8 /* TX_RING_SIZE is huge! */; i++)			printk(KERN_DEBUG " #%d desc. %8.8x %8.8x -> %8.8x.\n",				   i, np->tx_ring[i].status, np->tx_ring[i].addr,				   np->tx_done_q[i].status);		printk(KERN_DEBUG "  Rx ring at %8.8x -> %p:\n",			   (int)virt_to_bus(np->rx_ring), np->rx_done_q);		if (np->rx_done_q)			for (i = 0; i < 8 /* RX_RING_SIZE */; i++) {				printk(KERN_DEBUG " #%d desc. %8.8x -> %8.8x\n",					   i, np->rx_ring[i].rxaddr, np->rx_done_q[i].status);		}	}#endif /* __i386__ debugging only */	free_irq(dev->irq, dev);	/* Free all the skbuffs in the Rx queue. */	for (i = 0; i < RX_RING_SIZE; i++) {		np->rx_ring[i].rxaddr = 0xBADF00D0; /* An invalid address. */		if (np->rx_skbuff[i]) {#if LINUX_VERSION_CODE < 0x20100			np->rx_skbuff[i]->free = 1;#endif			dev_free_skb(np->rx_skbuff[i]);		}		np->rx_skbuff[i] = 0;	}	for (i = 0; i < TX_RING_SIZE; i++) {		if (np->tx_skbuff[i])			dev_free_skb(np->tx_skbuff[i]);		np->tx_skbuff[i] = 0;	}	MOD_DEC_USE_COUNT;	return 0;}static int starfire_pwr_event(void *dev_instance, int event){	struct net_device *dev = dev_instance;	struct netdev_private *np = (struct netdev_private *)dev->priv;	long ioaddr = dev->base_addr;	if (np->msg_level & NETIF_MSG_LINK)		printk(KERN_DEBUG "%s: Handling power event %d.\n", dev->name, event);	switch(event) {	case DRV_ATTACH:		MOD_INC_USE_COUNT;		break;	case DRV_SUSPEND:		/* Disable interrupts, stop Tx and Rx. */		writel(0x0000, ioaddr + IntrEnable);		writel(0, ioaddr + GenCtrl);		break;	case DRV_RESUME:		/* This is incomplete: we must factor start_chip() out of open(). */		writel(np->tx_threshold, ioaddr + TxThreshold);		writel(interrupt_mitigation, ioaddr + IntrTimerCtrl);		set_rx_mode(dev);		writel(np->intr_enable, ioaddr + IntrEnable);		writel(TxEnable|RxEnable, ioaddr + GenCtrl);		break;	case DRV_DETACH: {		struct net_device **devp, **next;		if (dev->flags & IFF_UP) {			/* Some, but not all, kernel versions close automatically. */			dev_close(dev);			dev->flags &= ~(IFF_UP|IFF_RUNNING);		}		unregister_netdev(dev);		release_region(dev->base_addr, pci_id_tbl[np->chip_id].io_size);#ifndef USE_IO_OPS		iounmap((char *)dev->base_addr);#endif		for (devp = &root_net_dev; *devp; devp = next) {			next = &((struct netdev_private *)(*devp)->priv)->next_module;			if (*devp == dev) {				*devp = *next;				break;			}		}		if (np->priv_addr)			kfree(np->priv_addr);		kfree(dev);		MOD_DEC_USE_COUNT;		break;	}	}	return 0;}#ifdef MODULEint init_module(void){	if (debug >= NETIF_MSG_DRV)	/* Emit version even if no cards detected. */		printk(KERN_INFO "%s" KERN_INFO "%s", version1, version2);	if (pci_drv_register(&starfire_drv_id, NULL)) {		printk(KERN_INFO " No Starfire adapters detected, driver not loaded.\n");		return -ENODEV;	}	return 0;}void cleanup_module(void){	struct net_device *next_dev;	pci_drv_unregister(&starfire_drv_id);	/* No need to check MOD_IN_USE, as sys_delete_module() checks. */	while (root_net_dev) {		struct netdev_private *np = (void *)(root_net_dev->priv);		unregister_netdev(root_net_dev);		iounmap((char *)(root_net_dev->base_addr));		next_dev = np->next_module;		if (np->tx_done_q) free_page((long)np->tx_done_q);		if (np->rx_done_q) free_page((long)np->rx_done_q);		if (np->priv_addr) kfree(np->priv_addr);		kfree(root_net_dev);		root_net_dev = next_dev;	}}#endif  /* MODULE *//* * Local variables: *  compile-command: "make KERNVER=`uname -r` starfire.o" *  compile-cmd: "gcc -DMODULE -Wall -Wstrict-prototypes -O6 -c starfire.c" *  simple-compile-command: "gcc -DMODULE -O6 -c starfire.c" *  c-indent-level: 4 *  c-basic-offset: 4 *  tab-width: 4 * End: */

⌨️ 快捷键说明

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