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

📄 atl1_main.c

📁 linux 内核源代码
💻 C
📖 第 1 页 / 共 5 页
字号:
/* * Copyright(c) 2005 - 2006 Attansic Corporation. All rights reserved. * Copyright(c) 2006 Chris Snook <csnook@redhat.com> * Copyright(c) 2006 Jay Cliburn <jcliburn@gmail.com> * * Derived from Intel e1000 driver * Copyright(c) 1999 - 2005 Intel Corporation. All rights reserved. * * This program is free software; you can redistribute it and/or modify it * under the terms of the GNU General Public License as published by the Free * Software Foundation; either version 2 of the License, or (at your option) * any later version. * * This program is distributed in the hope that it will be useful, but WITHOUT * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or * FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License for * more details. * * You should have received a copy of the GNU General Public License along with * this program; if not, write to the Free Software Foundation, Inc., 59 * Temple Place - Suite 330, Boston, MA  02111-1307, USA. * * The full GNU General Public License is included in this distribution in the * file called COPYING. * * Contact Information: * Xiong Huang <xiong_huang@attansic.com> * Attansic Technology Corp. 3F 147, Xianzheng 9th Road, Zhubei, * Xinzhu  302, TAIWAN, REPUBLIC OF CHINA * * Chris Snook <csnook@redhat.com> * Jay Cliburn <jcliburn@gmail.com> * * This version is adapted from the Attansic reference driver for * inclusion in the Linux kernel.  It is currently under heavy development. * A very incomplete list of things that need to be dealt with: * * TODO: * Fix TSO; tx performance is horrible with TSO enabled. * Wake on LAN. * Add more ethtool functions. * Fix abstruse irq enable/disable condition described here: *	http://marc.theaimsgroup.com/?l=linux-netdev&m=116398508500553&w=2 * * NEEDS TESTING: * VLAN * multicast * promiscuous mode * interrupt coalescing * SMP torture testing */#include <linux/types.h>#include <linux/netdevice.h>#include <linux/pci.h>#include <linux/spinlock.h>#include <linux/slab.h>#include <linux/string.h>#include <linux/skbuff.h>#include <linux/etherdevice.h>#include <linux/if_vlan.h>#include <linux/if_ether.h>#include <linux/irqreturn.h>#include <linux/workqueue.h>#include <linux/timer.h>#include <linux/jiffies.h>#include <linux/hardirq.h>#include <linux/interrupt.h>#include <linux/irqflags.h>#include <linux/dma-mapping.h>#include <linux/net.h>#include <linux/pm.h>#include <linux/in.h>#include <linux/ip.h>#include <linux/tcp.h>#include <linux/compiler.h>#include <linux/delay.h>#include <linux/mii.h>#include <net/checksum.h>#include <asm/atomic.h>#include <asm/byteorder.h>#include "atl1.h"#define DRIVER_VERSION "2.0.7"char atl1_driver_name[] = "atl1";static const char atl1_driver_string[] = "Attansic L1 Ethernet Network Driver";static const char atl1_copyright[] = "Copyright(c) 2005-2006 Attansic Corporation.";char atl1_driver_version[] = DRIVER_VERSION;MODULE_AUTHOR    ("Attansic Corporation <xiong_huang@attansic.com>, Chris Snook <csnook@redhat.com>, Jay Cliburn <jcliburn@gmail.com>");MODULE_DESCRIPTION("Attansic 1000M Ethernet Network Driver");MODULE_LICENSE("GPL");MODULE_VERSION(DRIVER_VERSION);/* * atl1_pci_tbl - PCI Device ID Table */static const struct pci_device_id atl1_pci_tbl[] = {	{PCI_DEVICE(PCI_VENDOR_ID_ATTANSIC, PCI_DEVICE_ID_ATTANSIC_L1)},	/* required last entry */	{0,}};MODULE_DEVICE_TABLE(pci, atl1_pci_tbl);/* * atl1_sw_init - Initialize general software structures (struct atl1_adapter) * @adapter: board private structure to initialize * * atl1_sw_init initializes the Adapter private data structure. * Fields are initialized based on PCI device information and * OS network device settings (MTU size). */static int __devinit atl1_sw_init(struct atl1_adapter *adapter){	struct atl1_hw *hw = &adapter->hw;	struct net_device *netdev = adapter->netdev;	hw->max_frame_size = netdev->mtu + ETH_HLEN + ETH_FCS_LEN + VLAN_HLEN;	hw->min_frame_size = ETH_ZLEN + ETH_FCS_LEN;	adapter->wol = 0;	adapter->rx_buffer_len = (hw->max_frame_size + 7) & ~7;	adapter->ict = 50000;	/* 100ms */	adapter->link_speed = SPEED_0;	/* hardware init */	adapter->link_duplex = FULL_DUPLEX;	hw->phy_configured = false;	hw->preamble_len = 7;	hw->ipgt = 0x60;	hw->min_ifg = 0x50;	hw->ipgr1 = 0x40;	hw->ipgr2 = 0x60;	hw->max_retry = 0xf;	hw->lcol = 0x37;	hw->jam_ipg = 7;	hw->rfd_burst = 8;	hw->rrd_burst = 8;	hw->rfd_fetch_gap = 1;	hw->rx_jumbo_th = adapter->rx_buffer_len / 8;	hw->rx_jumbo_lkah = 1;	hw->rrd_ret_timer = 16;	hw->tpd_burst = 4;	hw->tpd_fetch_th = 16;	hw->txf_burst = 0x100;	hw->tx_jumbo_task_th = (hw->max_frame_size + 7) >> 3;	hw->tpd_fetch_gap = 1;	hw->rcb_value = atl1_rcb_64;	hw->dma_ord = atl1_dma_ord_enh;	hw->dmar_block = atl1_dma_req_256;	hw->dmaw_block = atl1_dma_req_256;	hw->cmb_rrd = 4;	hw->cmb_tpd = 4;	hw->cmb_rx_timer = 1;	/* about 2us */	hw->cmb_tx_timer = 1;	/* about 2us */	hw->smb_timer = 100000;	/* about 200ms */	spin_lock_init(&adapter->lock);	spin_lock_init(&adapter->mb_lock);	return 0;}static int mdio_read(struct net_device *netdev, int phy_id, int reg_num){	struct atl1_adapter *adapter = netdev_priv(netdev);	u16 result;	atl1_read_phy_reg(&adapter->hw, reg_num & 0x1f, &result);	return result;}static void mdio_write(struct net_device *netdev, int phy_id, int reg_num,	int val){	struct atl1_adapter *adapter = netdev_priv(netdev);	atl1_write_phy_reg(&adapter->hw, reg_num, val);}/* * atl1_mii_ioctl - * @netdev: * @ifreq: * @cmd: */static int atl1_mii_ioctl(struct net_device *netdev, struct ifreq *ifr, int cmd){	struct atl1_adapter *adapter = netdev_priv(netdev);	unsigned long flags;	int retval;	if (!netif_running(netdev))		return -EINVAL;	spin_lock_irqsave(&adapter->lock, flags);	retval = generic_mii_ioctl(&adapter->mii, if_mii(ifr), cmd, NULL);	spin_unlock_irqrestore(&adapter->lock, flags);	return retval;}/* * atl1_ioctl - * @netdev: * @ifreq: * @cmd: */static int atl1_ioctl(struct net_device *netdev, struct ifreq *ifr, int cmd){	switch (cmd) {	case SIOCGMIIPHY:	case SIOCGMIIREG:	case SIOCSMIIREG:		return atl1_mii_ioctl(netdev, ifr, cmd);	default:		return -EOPNOTSUPP;	}}/* * atl1_setup_mem_resources - allocate Tx / RX descriptor resources * @adapter: board private structure * * Return 0 on success, negative on failure */s32 atl1_setup_ring_resources(struct atl1_adapter *adapter){	struct atl1_tpd_ring *tpd_ring = &adapter->tpd_ring;	struct atl1_rfd_ring *rfd_ring = &adapter->rfd_ring;	struct atl1_rrd_ring *rrd_ring = &adapter->rrd_ring;	struct atl1_ring_header *ring_header = &adapter->ring_header;	struct pci_dev *pdev = adapter->pdev;	int size;	u8 offset = 0;	size = sizeof(struct atl1_buffer) * (tpd_ring->count + rfd_ring->count);	tpd_ring->buffer_info = kzalloc(size, GFP_KERNEL);	if (unlikely(!tpd_ring->buffer_info)) {		dev_err(&pdev->dev, "kzalloc failed , size = D%d\n", size);		goto err_nomem;	}	rfd_ring->buffer_info =		(struct atl1_buffer *)(tpd_ring->buffer_info + tpd_ring->count);	/* real ring DMA buffer	 * each ring/block may need up to 8 bytes for alignment, hence the	 * additional 40 bytes tacked onto the end.	 */	ring_header->size = size =		sizeof(struct tx_packet_desc) * tpd_ring->count		+ sizeof(struct rx_free_desc) * rfd_ring->count		+ sizeof(struct rx_return_desc) * rrd_ring->count		+ sizeof(struct coals_msg_block)		+ sizeof(struct stats_msg_block)		+ 40;	ring_header->desc = pci_alloc_consistent(pdev, ring_header->size,		&ring_header->dma);	if (unlikely(!ring_header->desc)) {		dev_err(&pdev->dev, "pci_alloc_consistent failed\n");		goto err_nomem;	}	memset(ring_header->desc, 0, ring_header->size);	/* init TPD ring */	tpd_ring->dma = ring_header->dma;	offset = (tpd_ring->dma & 0x7) ? (8 - (ring_header->dma & 0x7)) : 0;	tpd_ring->dma += offset;	tpd_ring->desc = (u8 *) ring_header->desc + offset;	tpd_ring->size = sizeof(struct tx_packet_desc) * tpd_ring->count;	/* init RFD ring */	rfd_ring->dma = tpd_ring->dma + tpd_ring->size;	offset = (rfd_ring->dma & 0x7) ? (8 - (rfd_ring->dma & 0x7)) : 0;	rfd_ring->dma += offset;	rfd_ring->desc = (u8 *) tpd_ring->desc + (tpd_ring->size + offset);	rfd_ring->size = sizeof(struct rx_free_desc) * rfd_ring->count;	/* init RRD ring */	rrd_ring->dma = rfd_ring->dma + rfd_ring->size;	offset = (rrd_ring->dma & 0x7) ? (8 - (rrd_ring->dma & 0x7)) : 0;	rrd_ring->dma += offset;	rrd_ring->desc = (u8 *) rfd_ring->desc + (rfd_ring->size + offset);	rrd_ring->size = sizeof(struct rx_return_desc) * rrd_ring->count;	/* init CMB */	adapter->cmb.dma = rrd_ring->dma + rrd_ring->size;	offset = (adapter->cmb.dma & 0x7) ? (8 - (adapter->cmb.dma & 0x7)) : 0;	adapter->cmb.dma += offset;	adapter->cmb.cmb = (struct coals_msg_block *)		((u8 *) rrd_ring->desc + (rrd_ring->size + offset));	/* init SMB */	adapter->smb.dma = adapter->cmb.dma + sizeof(struct coals_msg_block);	offset = (adapter->smb.dma & 0x7) ? (8 - (adapter->smb.dma & 0x7)) : 0;	adapter->smb.dma += offset;	adapter->smb.smb = (struct stats_msg_block *)		((u8 *) adapter->cmb.cmb +		(sizeof(struct coals_msg_block) + offset));	return ATL1_SUCCESS;err_nomem:	kfree(tpd_ring->buffer_info);	return -ENOMEM;}static void atl1_init_ring_ptrs(struct atl1_adapter *adapter){	struct atl1_tpd_ring *tpd_ring = &adapter->tpd_ring;	struct atl1_rfd_ring *rfd_ring = &adapter->rfd_ring;	struct atl1_rrd_ring *rrd_ring = &adapter->rrd_ring;	atomic_set(&tpd_ring->next_to_use, 0);	atomic_set(&tpd_ring->next_to_clean, 0);	rfd_ring->next_to_clean = 0;	atomic_set(&rfd_ring->next_to_use, 0);	rrd_ring->next_to_use = 0;	atomic_set(&rrd_ring->next_to_clean, 0);}/* * atl1_clean_rx_ring - Free RFD Buffers * @adapter: board private structure */static void atl1_clean_rx_ring(struct atl1_adapter *adapter){	struct atl1_rfd_ring *rfd_ring = &adapter->rfd_ring;	struct atl1_rrd_ring *rrd_ring = &adapter->rrd_ring;	struct atl1_buffer *buffer_info;	struct pci_dev *pdev = adapter->pdev;	unsigned long size;	unsigned int i;	/* Free all the Rx ring sk_buffs */	for (i = 0; i < rfd_ring->count; i++) {		buffer_info = &rfd_ring->buffer_info[i];		if (buffer_info->dma) {			pci_unmap_page(pdev, buffer_info->dma,				buffer_info->length, PCI_DMA_FROMDEVICE);			buffer_info->dma = 0;		}		if (buffer_info->skb) {			dev_kfree_skb(buffer_info->skb);			buffer_info->skb = NULL;		}	}	size = sizeof(struct atl1_buffer) * rfd_ring->count;	memset(rfd_ring->buffer_info, 0, size);	/* Zero out the descriptor ring */	memset(rfd_ring->desc, 0, rfd_ring->size);	rfd_ring->next_to_clean = 0;	atomic_set(&rfd_ring->next_to_use, 0);	rrd_ring->next_to_use = 0;	atomic_set(&rrd_ring->next_to_clean, 0);}/* * atl1_clean_tx_ring - Free Tx Buffers * @adapter: board private structure */static void atl1_clean_tx_ring(struct atl1_adapter *adapter){	struct atl1_tpd_ring *tpd_ring = &adapter->tpd_ring;	struct atl1_buffer *buffer_info;	struct pci_dev *pdev = adapter->pdev;	unsigned long size;	unsigned int i;	/* Free all the Tx ring sk_buffs */	for (i = 0; i < tpd_ring->count; i++) {		buffer_info = &tpd_ring->buffer_info[i];		if (buffer_info->dma) {			pci_unmap_page(pdev, buffer_info->dma,				buffer_info->length, PCI_DMA_TODEVICE);			buffer_info->dma = 0;		}	}	for (i = 0; i < tpd_ring->count; i++) {		buffer_info = &tpd_ring->buffer_info[i];		if (buffer_info->skb) {			dev_kfree_skb_any(buffer_info->skb);			buffer_info->skb = NULL;		}	}	size = sizeof(struct atl1_buffer) * tpd_ring->count;	memset(tpd_ring->buffer_info, 0, size);	/* Zero out the descriptor ring */	memset(tpd_ring->desc, 0, tpd_ring->size);	atomic_set(&tpd_ring->next_to_use, 0);	atomic_set(&tpd_ring->next_to_clean, 0);}/* * atl1_free_ring_resources - Free Tx / RX descriptor Resources * @adapter: board private structure * * Free all transmit software resources */void atl1_free_ring_resources(struct atl1_adapter *adapter){	struct pci_dev *pdev = adapter->pdev;	struct atl1_tpd_ring *tpd_ring = &adapter->tpd_ring;	struct atl1_rfd_ring *rfd_ring = &adapter->rfd_ring;	struct atl1_rrd_ring *rrd_ring = &adapter->rrd_ring;	struct atl1_ring_header *ring_header = &adapter->ring_header;	atl1_clean_tx_ring(adapter);	atl1_clean_rx_ring(adapter);	kfree(tpd_ring->buffer_info);	pci_free_consistent(pdev, ring_header->size, ring_header->desc,		ring_header->dma);	tpd_ring->buffer_info = NULL;	tpd_ring->desc = NULL;	tpd_ring->dma = 0;	rfd_ring->buffer_info = NULL;	rfd_ring->desc = NULL;	rfd_ring->dma = 0;	rrd_ring->desc = NULL;	rrd_ring->dma = 0;}static void atl1_setup_mac_ctrl(struct atl1_adapter *adapter){	u32 value;	struct atl1_hw *hw = &adapter->hw;	struct net_device *netdev = adapter->netdev;	/* Config MAC CTRL Register */	value = MAC_CTRL_TX_EN | MAC_CTRL_RX_EN;	/* duplex */	if (FULL_DUPLEX == adapter->link_duplex)		value |= MAC_CTRL_DUPLX;	/* speed */	value |= ((u32) ((SPEED_1000 == adapter->link_speed) ?

⌨️ 快捷键说明

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