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

📄 cs89x0.c

📁 linux环境下CS8900以太网设备驱动
💻 C
📖 第 1 页 / 共 4 页
字号:
}#endif/* This is the real probe routine.  Linux has a history of friendly device   probes on the ISA bus.  A good device probes avoids doing writes, and   verifies that the correct device exists and functions.   Return 0 on success. */static int __initcs89x0_probe1(struct net_device *dev, int ioaddr, int modular){	struct net_local *lp = netdev_priv(dev);	static unsigned version_printed;	int i;	int tmp;	unsigned rev_type = 0;	int eeprom_buff[CHKSUM_LEN];	int retval;	SET_MODULE_OWNER(dev);	/* Initialize the device structure. */	if (!modular) {		memset(lp, 0, sizeof(*lp));		spin_lock_init(&lp->lock);#ifndef MODULE#if ALLOW_DMA		if (g_cs89x0_dma) {			lp->use_dma = 1;			lp->dma = g_cs89x0_dma;			lp->dmasize = 16;	/* Could make this an option... */		}#endif		lp->force = g_cs89x0_media__force;#endif        }#ifdef CONFIG_ARCH_PNX010X	initialize_ebi();	/* Map GPIO registers for the pins connected to the CS8900a. */	if (map_cirrus_gpio() < 0)		return -ENODEV;	reset_cirrus();	/* Map event-router registers. */	if (map_event_router() < 0)		return -ENODEV;	enable_cirrus_irq();	unmap_cirrus_gpio();	unmap_event_router();	dev->base_addr = ioaddr;	for (i = 0 ; i < 3 ; i++)		readreg(dev, 0);#endif	/* Grab the region so we can find another board if autoIRQ fails. */	/* WTF is going on here? */	if (!request_region(ioaddr & ~3, NETCARD_IO_EXTENT, DRV_NAME)) {		printk(KERN_ERR "%s: request_region(0x%x, 0x%x) failed\n",				DRV_NAME, ioaddr, NETCARD_IO_EXTENT);		retval = -EBUSY;		goto out1;	}#ifdef CONFIG_SH_HICOSH4	/* truely reset the chip */	writeword(ioaddr, ADD_PORT, 0x0114);	writeword(ioaddr, DATA_PORT, 0x0040);#endif	/* if they give us an odd I/O address, then do ONE write to           the address port, to get it back to address zero, where we           expect to find the EISA signature word. An IO with a base of 0x3	   will skip the test for the ADD_PORT. */	if (ioaddr & 1) {		if (net_debug > 1)			printk(KERN_INFO "%s: odd ioaddr 0x%x\n", dev->name, ioaddr);	        if ((ioaddr & 2) != 2)	        	if ((readword(ioaddr & ~3, ADD_PORT) & ADD_MASK) != ADD_SIG) {				printk(KERN_ERR "%s: bad signature 0x%x\n",					dev->name, readword(ioaddr & ~3, ADD_PORT));		        	retval = -ENODEV;				goto out2;			}	}	printk(KERN_DEBUG "PP_addr at %x[%x]: 0x%x\n",			ioaddr, ADD_PORT, readword(ioaddr, ADD_PORT));	ioaddr &= ~3;	writeword(ioaddr, ADD_PORT, PP_ChipID);	tmp = readword(ioaddr, DATA_PORT);	if (tmp != CHIP_EISA_ID_SIG) {		printk(KERN_DEBUG "%s: incorrect signature at %x[%x]: 0x%x!="			CHIP_EISA_ID_SIG_STR "\n",			dev->name, ioaddr, DATA_PORT, tmp);  		retval = -ENODEV;  		goto out2;	}	/* Fill in the 'dev' fields. */	dev->base_addr = ioaddr;	/* get the chip type */	rev_type = readreg(dev, PRODUCT_ID_ADD);	lp->chip_type = rev_type &~ REVISON_BITS;	lp->chip_revision = ((rev_type & REVISON_BITS) >> 8) + 'A';	/* Check the chip type and revision in order to set the correct send command	CS8920 revision C and CS8900 revision F can use the faster send. */	lp->send_cmd = TX_AFTER_381;	if (lp->chip_type == CS8900 && lp->chip_revision >= 'F')		lp->send_cmd = TX_NOW;	if (lp->chip_type != CS8900 && lp->chip_revision >= 'C')		lp->send_cmd = TX_NOW;	if (net_debug  &&  version_printed++ == 0)		printk(version);	printk(KERN_INFO "%s: cs89%c0%s rev %c found at %#3lx ",	       dev->name,	       lp->chip_type==CS8900?'0':'2',	       lp->chip_type==CS8920M?"M":"",	       lp->chip_revision,	       dev->base_addr);	reset_chip(dev);           /* Here we read the current configuration of the chip. If there	   is no Extended EEPROM then the idea is to not disturb the chip	   configuration, it should have been correctly setup by automatic	   EEPROM read on reset. So, if the chip says it read the EEPROM	   the driver will always do *something* instead of complain that	   adapter_cnf is 0. */#ifdef CONFIG_SH_HICOSH4	if (1) {		/* For the HiCO.SH4 board, things are different: we don't		   have EEPROM, but there is some data in flash, so we go		   get it there directly (MAC). */		__u16 *confd;		short cnt;		if (((* (volatile __u32 *) 0xa0013ff0) & 0x00ffffff)			== 0x006c3000) {			confd = (__u16*) 0xa0013fc0;		} else {			confd = (__u16*) 0xa001ffc0;		}		cnt = (*confd++ & 0x00ff) >> 1;		while (--cnt > 0) {			__u16 j = *confd++;						switch (j & 0x0fff) {			case PP_IA:				for (i = 0; i < ETH_ALEN/2; i++) {					dev->dev_addr[i*2] = confd[i] & 0xFF;					dev->dev_addr[i*2+1] = confd[i] >> 8;				}				break;			}			j = (j >> 12) + 1;			confd += j;			cnt -= j;		}	} else#endif        if ((readreg(dev, PP_SelfST) & (EEPROM_OK | EEPROM_PRESENT)) == 	      (EEPROM_OK|EEPROM_PRESENT)) {	        /* Load the MAC. */		for (i=0; i < ETH_ALEN/2; i++) {	                unsigned int Addr;			Addr = readreg(dev, PP_IA+i*2);		        dev->dev_addr[i*2] = Addr & 0xFF;		        dev->dev_addr[i*2+1] = Addr >> 8;		}   	   	/* Load the Adapter Configuration. 		   Note:  Barring any more specific information from some 		   other source (ie EEPROM+Schematics), we would not know 		   how to operate a 10Base2 interface on the AUI port. 		   However, since we  do read the status of HCB1 and use 		   settings that always result in calls to control_dc_dc(dev,0) 		   a BNC interface should work if the enable pin 		   (dc/dc converter) is on HCB1. It will be called AUI 		   however. */	   		lp->adapter_cnf = 0;		i = readreg(dev, PP_LineCTL);		/* Preserve the setting of the HCB1 pin. */		if ((i & (HCB1 | HCB1_ENBL)) ==  (HCB1 | HCB1_ENBL))			lp->adapter_cnf |= A_CNF_DC_DC_POLARITY;		/* Save the sqelch bit */		if ((i & LOW_RX_SQUELCH) == LOW_RX_SQUELCH)			lp->adapter_cnf |= A_CNF_EXTND_10B_2 | A_CNF_LOW_RX_SQUELCH;		/* Check if the card is in 10Base-t only mode */		if ((i & (AUI_ONLY | AUTO_AUI_10BASET)) == 0)			lp->adapter_cnf |=  A_CNF_10B_T | A_CNF_MEDIA_10B_T;		/* Check if the card is in AUI only mode */		if ((i & (AUI_ONLY | AUTO_AUI_10BASET)) == AUI_ONLY)			lp->adapter_cnf |=  A_CNF_AUI | A_CNF_MEDIA_AUI;		/* Check if the card is in Auto mode. */		if ((i & (AUI_ONLY | AUTO_AUI_10BASET)) == AUTO_AUI_10BASET)			lp->adapter_cnf |=  A_CNF_AUI | A_CNF_10B_T | 			A_CNF_MEDIA_AUI | A_CNF_MEDIA_10B_T | A_CNF_MEDIA_AUTO;				if (net_debug > 1)			printk(KERN_INFO "%s: PP_LineCTL=0x%x, adapter_cnf=0x%x\n",					dev->name, i, lp->adapter_cnf);		/* IRQ. Other chips already probe, see below. */		if (lp->chip_type == CS8900) 			lp->isa_config = readreg(dev, PP_CS8900_ISAINT) & INT_NO_MASK;	   		printk( "[Cirrus EEPROM] ");	}        printk("\n");   	/* First check to see if an EEPROM is attached. */#ifdef CONFIG_SH_HICOSH4 /* no EEPROM on HiCO, don't hazzle with it here */	if (1) {		printk(KERN_NOTICE "cs89x0: No EEPROM on HiCO.SH4\n");	} else#endif	if ((readreg(dev, PP_SelfST) & EEPROM_PRESENT) == 0)		printk(KERN_WARNING "cs89x0: No EEPROM, relying on command line....\n");	else if (get_eeprom_data(dev, START_EEPROM_DATA,CHKSUM_LEN,eeprom_buff) < 0) {		printk(KERN_WARNING "\ncs89x0: EEPROM read failed, relying on command line.\n");        } else if (get_eeprom_cksum(START_EEPROM_DATA,CHKSUM_LEN,eeprom_buff) < 0) {		/* Check if the chip was able to read its own configuration starting		   at 0 in the EEPROM*/		if ((readreg(dev, PP_SelfST) & (EEPROM_OK | EEPROM_PRESENT)) !=		    (EEPROM_OK|EEPROM_PRESENT))                 	printk(KERN_WARNING "cs89x0: Extended EEPROM checksum bad and no Cirrus EEPROM, relying on command line\n");		           } else {		/* This reads an extended EEPROM that is not documented		   in the CS8900 datasheet. */		                /* get transmission control word  but keep the autonegotiation bits */                if (!lp->auto_neg_cnf) lp->auto_neg_cnf = eeprom_buff[AUTO_NEG_CNF_OFFSET/2];                /* Store adapter configuration */                if (!lp->adapter_cnf) lp->adapter_cnf = eeprom_buff[ADAPTER_CNF_OFFSET/2];                /* Store ISA configuration */                lp->isa_config = eeprom_buff[ISA_CNF_OFFSET/2];                dev->mem_start = eeprom_buff[PACKET_PAGE_OFFSET/2] << 8;                /* eeprom_buff has 32-bit ints, so we can't just memcpy it */                /* store the initial memory base address */                for (i = 0; i < ETH_ALEN/2; i++) {                        dev->dev_addr[i*2] = eeprom_buff[i];                        dev->dev_addr[i*2+1] = eeprom_buff[i] >> 8;                }		if (net_debug > 1)			printk(KERN_DEBUG "%s: new adapter_cnf: 0x%x\n",				dev->name, lp->adapter_cnf);        }        /* allow them to force multiple transceivers.  If they force multiple, autosense */        {		int count = 0;		if (lp->force & FORCE_RJ45)	{lp->adapter_cnf |= A_CNF_10B_T; count++; }		if (lp->force & FORCE_AUI) 	{lp->adapter_cnf |= A_CNF_AUI; count++; }		if (lp->force & FORCE_BNC)	{lp->adapter_cnf |= A_CNF_10B_2; count++; }		if (count > 1)			{lp->adapter_cnf |= A_CNF_MEDIA_AUTO; }		else if (lp->force & FORCE_RJ45){lp->adapter_cnf |= A_CNF_MEDIA_10B_T; }		else if (lp->force & FORCE_AUI)	{lp->adapter_cnf |= A_CNF_MEDIA_AUI; }		else if (lp->force & FORCE_BNC)	{lp->adapter_cnf |= A_CNF_MEDIA_10B_2; }        }	if (net_debug > 1)		printk(KERN_DEBUG "%s: after force 0x%x, adapter_cnf=0x%x\n",			dev->name, lp->force, lp->adapter_cnf);        /* FIXME: We don't let you set dc-dc polarity or low RX squelch from the command line: add it here */        /* FIXME: We don't let you set the IMM bit from the command line: add it to lp->auto_neg_cnf here */        /* FIXME: we don't set the Ethernet address on the command line.  Use           ifconfig IFACE hw ether AABBCCDDEEFF */	printk(KERN_INFO "cs89x0 media %s%s%s",	       (lp->adapter_cnf & A_CNF_10B_T)?"RJ-45,":"",	       (lp->adapter_cnf & A_CNF_AUI)?"AUI,":"",	       (lp->adapter_cnf & A_CNF_10B_2)?"BNC,":"");	lp->irq_map = 0xffff;	/* If this is a CS8900 then no pnp soft */	if (lp->chip_type != CS8900 &&	    /* Check if the ISA IRQ has been set  */		(i = readreg(dev, PP_CS8920_ISAINT) & 0xff,		 (i != 0 && i < CS8920_NO_INTS))) {		if (!dev->irq)			dev->irq = i;	} else {		i = lp->isa_config & INT_NO_MASK;		if (lp->chip_type == CS8900) {#if defined(CONFIG_MACH_IXDP2351) || defined(CONFIG_ARCH_IXDP2X01) || defined(CONFIG_ARCH_PNX010X)		        i = cs8900_irq_map[0];#else			/* Translate the IRQ using the IRQ mapping table. */			if (i >= sizeof(cs8900_irq_map)/sizeof(cs8900_irq_map[0]))				printk("\ncs89x0: invalid ISA interrupt number %d\n", i);			else				i = cs8900_irq_map[i];						lp->irq_map = CS8900_IRQ_MAP; /* fixed IRQ map for CS8900 */		} else {			int irq_map_buff[IRQ_MAP_LEN/2];			if (get_eeprom_data(dev, IRQ_MAP_EEPROM_DATA,					    IRQ_MAP_LEN/2,					    irq_map_buff) >= 0) {				if ((irq_map_buff[0] & 0xff) == PNP_IRQ_FRMT)					lp->irq_map = (irq_map_buff[0]>>8) | (irq_map_buff[1] << 8);			}#endif		}		if (!dev->irq)			dev->irq = i;	}	printk(" IRQ %d", dev->irq);#if ALLOW_DMA	if (lp->use_dma) {		get_dma_channel(dev);		printk(", DMA %d", dev->dma);	}	else#endif	{		printk(", programmed I/O");	}	/* print the ethernet address. */	printk(", MAC");	for (i = 0; i < ETH_ALEN; i++)	{		printk("%c%02x", i ? ':' : ' ', dev->dev_addr[i]);	}	dev->open		= net_open;	dev->stop		= net_close;	dev->tx_timeout		= net_timeout;	dev->watchdog_timeo	= HZ;	dev->hard_start_xmit 	= net_send_packet;	dev->get_stats		= net_get_stats;	dev->set_multicast_list = set_multicast_list;	dev->set_mac_address 	= set_mac_address;#ifdef CONFIG_NET_POLL_CONTROLLER	dev->poll_controller	= net_poll_controller;#endif	printk("\n");	if (net_debug)		printk("cs89x0_probe1() successful\n");	retval = register_netdev(dev);	if (retval)		goto out3;	return 0;out3:	writeword(dev->base_addr, ADD_PORT, PP_ChipID);out2:	release_region(ioaddr & ~3, NETCARD_IO_EXTENT);out1:	return retval;}/********************************* * This page contains DMA routines**********************************/#if ALLOW_DMA#define dma_page_eq(ptr1, ptr2) ((long)(ptr1)>>17 == (long)(ptr2)>>17)static voidget_dma_channel(struct net_device *dev){	struct net_local *lp = netdev_priv(dev);	if (lp->dma) {		dev->dma = lp->dma;		lp->isa_config |= ISA_RxDMA;	} else {		if ((lp->isa_config & ANY_ISA_DMA) == 0)			return;		dev->dma = lp->isa_config & DMA_NO_MASK;		if (lp->chip_type == CS8900)			dev->dma += 5;		if (dev->dma < 5 || dev->dma > 7) {			lp->isa_config &= ~ANY_ISA_DMA;			return;		}	}	return;}static voidwrite_dma(struct net_device *dev, int chip_type, int dma){	struct net_local *lp = netdev_priv(dev);	if ((lp->isa_config & ANY_ISA_DMA) == 0)		return;	if (chip_type == CS8900) {		writereg(dev, PP_CS8900_ISADMA, dma-5);	} else {		writereg(dev, PP_CS8920_ISADMA, dma);	}}static voidset_dma_cfg(struct net_device *dev){	struct net_local *lp = netdev_priv(dev);	if (lp->use_dma) {		if ((lp->isa_config & ANY_ISA_DMA) == 0) {			if (net_debug > 3)				printk("set_dma_cfg(): no DMA\n");			return;		}		if (lp->isa_config & ISA_RxDMA) {			lp->curr_rx_cfg |= RX_DMA_ONLY;			if (net_debug > 3)				printk("set_dma_cfg(): RX_DMA_ONLY\n");		} else {			lp->curr_rx_cfg |= AUTO_RX_DMA;	/* not that we support it... */			if (net_debug > 3)				printk("set_dma_cfg(): AUTO_RX_DMA\n");		}	}}static intdma_bufcfg(struct net_device *dev){	struct net_local *lp = netdev_priv(dev);	if (lp->use_dma)		return (lp->isa_config & ANY_ISA_DMA)? RX_DMA_ENBL : 0;	else		return 0;}static intdma_busctl(struct net_device *dev){	int retval = 0;	struct net_local *lp = netdev_priv(dev);	if (lp->use_dma) {		if (lp->isa_config & ANY_ISA_DMA)			retval |= RESET_RX_DMA; /* Reset the DMA pointer */		if (lp->isa_config & DMA_BURST)			retval |= DMA_BURST_MODE; /* Does ISA config specify DMA burst ? */		if (lp->dmasize == 64)			retval |= RX_DMA_SIZE_64K; /* did they ask for 64K? */		retval |= MEMORY_ON;	/* we need memory enabled to use DMA. */	}	return retval;}static voiddma_rx(struct net_device *dev){	struct net_local *lp = netdev_priv(dev);	struct sk_buff *skb;	int status, length;	unsigned char *bp = lp->rx_dma_ptr;	status = bp[0] + (bp[1]<<8);	length = bp[2] + (bp[3]<<8);	bp += 4;	if (net_debug > 5) {		printk(	"%s: receiving DMA packet at %lx, status %x, length %x\n",			dev->name, (unsigned long)bp, status, length);	}	if ((status & RX_OK) == 0) {		count_rx_errors(status, lp);		goto skip_this_frame;	}	/* Malloc up new buffer. */	skb = dev_alloc_skb(length + 2);	if (skb == NULL) {		if (net_debug)	/* I don't think we want to do this to a stressed system */			printk("%s: Memory squeeze, dropping packet.\n", dev->name);		lp->stats.rx_dropped++;		/* AKPM: advance bp to the next frame */

⌨️ 快捷键说明

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