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

📄 wanxl.c

📁 linux-2.6.15.6
💻 C
📖 第 1 页 / 共 2 页
字号:
	writel(1 << (DOORBELL_TO_CARD_CLOSE_0 + port->node), dbr);	return -EFAULT;}static int wanxl_close(struct net_device *dev){	port_t *port = dev_to_port(dev);	unsigned long timeout;	int i;	hdlc_close(dev);	/* signal the card */	writel(1 << (DOORBELL_TO_CARD_CLOSE_0 + port->node),	       port->card->plx + PLX_DOORBELL_TO_CARD);	timeout = jiffies + HZ;	do		if (!get_status(port)->open)			break;	while (time_after(timeout, jiffies));	if (get_status(port)->open)		printk(KERN_ERR "%s: unable to close port\n", dev->name);	netif_stop_queue(dev);	for (i = 0; i < TX_BUFFERS; i++) {		desc_t *desc = &get_status(port)->tx_descs[i];		if (desc->stat != PACKET_EMPTY) {			desc->stat = PACKET_EMPTY;			pci_unmap_single(port->card->pdev, desc->address,					 port->tx_skbs[i]->len,					 PCI_DMA_TODEVICE);			dev_kfree_skb(port->tx_skbs[i]);		}	}	return 0;}static struct net_device_stats *wanxl_get_stats(struct net_device *dev){	struct net_device_stats *stats = hdlc_stats(dev);	port_t *port = dev_to_port(dev);	stats->rx_over_errors = get_status(port)->rx_overruns;	stats->rx_frame_errors = get_status(port)->rx_frame_errors;	stats->rx_errors = stats->rx_over_errors + stats->rx_frame_errors;        return stats;}static int wanxl_puts_command(card_t *card, u32 cmd){	unsigned long timeout = jiffies + 5 * HZ;	writel(cmd, card->plx + PLX_MAILBOX_1);	do {		if (readl(card->plx + PLX_MAILBOX_1) == 0)			return 0;		schedule();	}while (time_after(timeout, jiffies));	return -1;}static void wanxl_reset(card_t *card){	u32 old_value = readl(card->plx + PLX_CONTROL) & ~PLX_CTL_RESET;	writel(0x80, card->plx + PLX_MAILBOX_0);	writel(old_value | PLX_CTL_RESET, card->plx + PLX_CONTROL);	readl(card->plx + PLX_CONTROL); /* wait for posted write */	udelay(1);	writel(old_value, card->plx + PLX_CONTROL);	readl(card->plx + PLX_CONTROL); /* wait for posted write */}static void wanxl_pci_remove_one(struct pci_dev *pdev){	card_t *card = pci_get_drvdata(pdev);	int i;	for (i = 0; i < card->n_ports; i++) {		unregister_hdlc_device(card->ports[i].dev);		free_netdev(card->ports[i].dev);	}	/* unregister and free all host resources */	if (card->irq)		free_irq(card->irq, card);	wanxl_reset(card);	for (i = 0; i < RX_QUEUE_LENGTH; i++)		if (card->rx_skbs[i]) {			pci_unmap_single(card->pdev,					 card->status->rx_descs[i].address,					 BUFFER_LENGTH, PCI_DMA_FROMDEVICE);			dev_kfree_skb(card->rx_skbs[i]);		}	if (card->plx)		iounmap(card->plx);	if (card->status)		pci_free_consistent(pdev, sizeof(card_status_t),				    card->status, card->status_address);	pci_release_regions(pdev);	pci_disable_device(pdev);	pci_set_drvdata(pdev, NULL);	kfree(card);}#include "wanxlfw.inc"static int __devinit wanxl_pci_init_one(struct pci_dev *pdev,					const struct pci_device_id *ent){	card_t *card;	u32 ramsize, stat;	unsigned long timeout;	u32 plx_phy;		/* PLX PCI base address */	u32 mem_phy;		/* memory PCI base addr */	u8 __iomem *mem;	/* memory virtual base addr */	int i, ports, alloc_size;#ifndef MODULE	static int printed_version;	if (!printed_version) {		printed_version++;		printk(KERN_INFO "%s\n", version);	}#endif	i = pci_enable_device(pdev);	if (i)		return i;	/* QUICC can only access first 256 MB of host RAM directly,	   but PLX9060 DMA does 32-bits for actual packet data transfers */	/* FIXME when PCI/DMA subsystems are fixed.	   We set both dma_mask and consistent_dma_mask to 28 bits	   and pray pci_alloc_consistent() will use this info. It should	   work on most platforms */	if (pci_set_consistent_dma_mask(pdev, 0x0FFFFFFF) ||	    pci_set_dma_mask(pdev, 0x0FFFFFFF)) {		printk(KERN_ERR "wanXL: No usable DMA configuration\n");		return -EIO;	}	i = pci_request_regions(pdev, "wanXL");	if (i) {		pci_disable_device(pdev);		return i;	}	switch (pdev->device) {	case PCI_DEVICE_ID_SBE_WANXL100: ports = 1; break;	case PCI_DEVICE_ID_SBE_WANXL200: ports = 2; break;	default: ports = 4;	}	alloc_size = sizeof(card_t) + ports * sizeof(port_t);	card = kmalloc(alloc_size, GFP_KERNEL);	if (card == NULL) {		printk(KERN_ERR "wanXL %s: unable to allocate memory\n",		       pci_name(pdev));		pci_release_regions(pdev);		pci_disable_device(pdev);		return -ENOBUFS;	}	memset(card, 0, alloc_size);	pci_set_drvdata(pdev, card);	card->pdev = pdev;	card->status = pci_alloc_consistent(pdev, sizeof(card_status_t),					    &card->status_address);	if (card->status == NULL) {		wanxl_pci_remove_one(pdev);		return -ENOBUFS;	}#ifdef DEBUG_PCI	printk(KERN_DEBUG "wanXL %s: pci_alloc_consistent() returned memory"	       " at 0x%LX\n", pci_name(pdev),	       (unsigned long long)card->status_address);#endif	/* FIXME when PCI/DMA subsystems are fixed.	   We set both dma_mask and consistent_dma_mask back to 32 bits	   to indicate the card can do 32-bit DMA addressing */	if (pci_set_consistent_dma_mask(pdev, DMA_32BIT_MASK) ||	    pci_set_dma_mask(pdev, DMA_32BIT_MASK)) {		printk(KERN_ERR "wanXL: No usable DMA configuration\n");		wanxl_pci_remove_one(pdev);		return -EIO;	}	/* set up PLX mapping */	plx_phy = pci_resource_start(pdev, 0);	card->plx = ioremap_nocache(plx_phy, 0x70);#if RESET_WHILE_LOADING	wanxl_reset(card);#endif	timeout = jiffies + 20 * HZ;	while ((stat = readl(card->plx + PLX_MAILBOX_0)) != 0) {		if (time_before(timeout, jiffies)) {			printk(KERN_WARNING "wanXL %s: timeout waiting for"			       " PUTS to complete\n", pci_name(pdev));			wanxl_pci_remove_one(pdev);			return -ENODEV;		}		switch(stat & 0xC0) {		case 0x00:	/* hmm - PUTS completed with non-zero code? */		case 0x80:	/* PUTS still testing the hardware */			break;		default:			printk(KERN_WARNING "wanXL %s: PUTS test 0x%X"			       " failed\n", pci_name(pdev), stat & 0x30);			wanxl_pci_remove_one(pdev);			return -ENODEV;		}		schedule();	}	/* get on-board memory size (PUTS detects no more than 4 MB) */	ramsize = readl(card->plx + PLX_MAILBOX_2) & MBX2_MEMSZ_MASK;	/* set up on-board RAM mapping */	mem_phy = pci_resource_start(pdev, 2);	/* sanity check the board's reported memory size */	if (ramsize < BUFFERS_ADDR +	    (TX_BUFFERS + RX_BUFFERS) * BUFFER_LENGTH * ports) {		printk(KERN_WARNING "wanXL %s: no enough on-board RAM"		       " (%u bytes detected, %u bytes required)\n",		       pci_name(pdev), ramsize, BUFFERS_ADDR +		       (TX_BUFFERS + RX_BUFFERS) * BUFFER_LENGTH * ports);		wanxl_pci_remove_one(pdev);		return -ENODEV;	}	if (wanxl_puts_command(card, MBX1_CMD_BSWAP)) {		printk(KERN_WARNING "wanXL %s: unable to Set Byte Swap"		       " Mode\n", pci_name(pdev));		wanxl_pci_remove_one(pdev);		return -ENODEV;	}	for (i = 0; i < RX_QUEUE_LENGTH; i++) {		struct sk_buff *skb = dev_alloc_skb(BUFFER_LENGTH);		card->rx_skbs[i] = skb;		if (skb)			card->status->rx_descs[i].address =				pci_map_single(card->pdev, skb->data,					       BUFFER_LENGTH,					       PCI_DMA_FROMDEVICE);	}	mem = ioremap_nocache(mem_phy, PDM_OFFSET + sizeof(firmware));	for (i = 0; i < sizeof(firmware); i += 4)		writel(htonl(*(u32*)(firmware + i)), mem + PDM_OFFSET + i);	for (i = 0; i < ports; i++)		writel(card->status_address +		       (void *)&card->status->port_status[i] -		       (void *)card->status, mem + PDM_OFFSET + 4 + i * 4);	writel(card->status_address, mem + PDM_OFFSET + 20);	writel(PDM_OFFSET, mem);	iounmap(mem);	writel(0, card->plx + PLX_MAILBOX_5);	if (wanxl_puts_command(card, MBX1_CMD_ABORTJ)) {		printk(KERN_WARNING "wanXL %s: unable to Abort and Jump\n",		       pci_name(pdev));		wanxl_pci_remove_one(pdev);		return -ENODEV;	}	stat = 0;	timeout = jiffies + 5 * HZ;	do {		if ((stat = readl(card->plx + PLX_MAILBOX_5)) != 0)			break;		schedule();	}while (time_after(timeout, jiffies));	if (!stat) {		printk(KERN_WARNING "wanXL %s: timeout while initializing card"		       "firmware\n", pci_name(pdev));		wanxl_pci_remove_one(pdev);		return -ENODEV;	}#if DETECT_RAM	ramsize = stat;#endif	printk(KERN_INFO "wanXL %s: at 0x%X, %u KB of RAM at 0x%X, irq %u\n",	       pci_name(pdev), plx_phy, ramsize / 1024, mem_phy, pdev->irq);	/* Allocate IRQ */	if (request_irq(pdev->irq, wanxl_intr, SA_SHIRQ, "wanXL", card)) {		printk(KERN_WARNING "wanXL %s: could not allocate IRQ%i.\n",		       pci_name(pdev), pdev->irq);		wanxl_pci_remove_one(pdev);		return -EBUSY;	}	card->irq = pdev->irq;	for (i = 0; i < ports; i++) {		hdlc_device *hdlc;		port_t *port = &card->ports[i];		struct net_device *dev = alloc_hdlcdev(port);		if (!dev) {			printk(KERN_ERR "wanXL %s: unable to allocate"			       " memory\n", pci_name(pdev));			wanxl_pci_remove_one(pdev);			return -ENOMEM;		}		port->dev = dev;		hdlc = dev_to_hdlc(dev);		spin_lock_init(&port->lock);		SET_MODULE_OWNER(dev);		dev->tx_queue_len = 50;		dev->do_ioctl = wanxl_ioctl;		dev->open = wanxl_open;		dev->stop = wanxl_close;		hdlc->attach = wanxl_attach;		hdlc->xmit = wanxl_xmit;		dev->get_stats = wanxl_get_stats;		port->card = card;		port->node = i;		get_status(port)->clocking = CLOCK_EXT;		if (register_hdlc_device(dev)) {			printk(KERN_ERR "wanXL %s: unable to register hdlc"			       " device\n", pci_name(pdev));			free_netdev(dev);			wanxl_pci_remove_one(pdev);			return -ENOBUFS;		}		card->n_ports++;	}	printk(KERN_INFO "wanXL %s: port", pci_name(pdev));	for (i = 0; i < ports; i++)		printk("%s #%i: %s", i ? "," : "", i,		       card->ports[i].dev->name);	printk("\n");	for (i = 0; i < ports; i++)		wanxl_cable_intr(&card->ports[i]); /* get carrier status etc.*/	return 0;}static struct pci_device_id wanxl_pci_tbl[] __devinitdata = {	{ PCI_VENDOR_ID_SBE, PCI_DEVICE_ID_SBE_WANXL100, PCI_ANY_ID,	  PCI_ANY_ID, 0, 0, 0 },	{ PCI_VENDOR_ID_SBE, PCI_DEVICE_ID_SBE_WANXL200, PCI_ANY_ID,	  PCI_ANY_ID, 0, 0, 0 },	{ PCI_VENDOR_ID_SBE, PCI_DEVICE_ID_SBE_WANXL400, PCI_ANY_ID,	  PCI_ANY_ID, 0, 0, 0 },	{ 0, }};static struct pci_driver wanxl_pci_driver = {	.name		= "wanXL",	.id_table	= wanxl_pci_tbl,	.probe		= wanxl_pci_init_one,	.remove		= wanxl_pci_remove_one,};static int __init wanxl_init_module(void){#ifdef MODULE	printk(KERN_INFO "%s\n", version);#endif	return pci_module_init(&wanxl_pci_driver);}static void __exit wanxl_cleanup_module(void){	pci_unregister_driver(&wanxl_pci_driver);}MODULE_AUTHOR("Krzysztof Halasa <khc@pm.waw.pl>");MODULE_DESCRIPTION("SBE Inc. wanXL serial port driver");MODULE_LICENSE("GPL v2");MODULE_DEVICE_TABLE(pci, wanxl_pci_tbl);module_init(wanxl_init_module);module_exit(wanxl_cleanup_module);

⌨️ 快捷键说明

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