r1000_n.c
来自「LINUX下面的RTL8169的驱动原码」· C语言 代码 · 共 2,072 行 · 第 1/4 页
C
2,072 行
free_netdev(dev);
#endif
pci_set_drvdata (pdev, NULL);
}
//======================================================================================================
static int r1000_open (struct net_device *dev)
{
struct r1000_private *priv = dev->priv;
struct pci_dev *pdev = priv->pci_dev;
int retval;
// u8 diff;
// u32 TxPhyAddr, RxPhyAddr;
if( priv->drvinit_fail == 1 ){
printk("%s: Gigabit driver open failed.\n", dev->name );
return -ENOMEM;
}
retval = request_irq (dev->irq, r1000_interrupt, SA_SHIRQ, dev->name, dev);
if (retval) {
return retval;
}
//2004-05-11
// Allocate tx/rx descriptor space
priv->sizeof_txdesc_space = NUM_TX_DESC * sizeof(struct TxDesc)+256;
priv->txdesc_space = pci_alloc_consistent( pdev, priv->sizeof_txdesc_space, &priv->txdesc_phy_dma_addr );
if( priv->txdesc_space == NULL ){
printk("%s: Gigabit driver alloc txdesc_space failed.\n", dev->name );
return -ENOMEM;
}
priv->sizeof_rxdesc_space = NUM_RX_DESC * sizeof(struct RxDesc)+256;
priv->rxdesc_space = pci_alloc_consistent( pdev, priv->sizeof_rxdesc_space, &priv->rxdesc_phy_dma_addr );
if( priv->rxdesc_space == NULL ){
printk("%s: Gigabit driver alloc rxdesc_space failed.\n", dev->name );
return -ENOMEM;
}
if(priv->txdesc_phy_dma_addr & 0xff){
printk("%s: Gigabit driver txdesc_phy_dma_addr is not 256-bytes-aligned.\n", dev->name );
}
if(priv->rxdesc_phy_dma_addr & 0xff){
printk("%s: Gigabit driver rxdesc_phy_dma_addr is not 256-bytes-aligned.\n", dev->name );
}
// Set tx/rx descriptor space
priv->TxDescArray = (struct TxDesc *)priv->txdesc_space;
priv->RxDescArray = (struct RxDesc *)priv->rxdesc_space;
{
int i;
struct sk_buff *skb = NULL;
for(i=0;i<NUM_RX_DESC;i++){
skb = R1000_ALLOC_RXSKB(MAX_RX_SKBDATA_SIZE);
if( skb != NULL ) {
skb_reserve (skb, 8); // 16 byte align the IP fields. //
priv->Rx_skbuff[i] = skb;
}
else{
printk("%s: Gigabit driver failed to allocate skbuff.\n", dev->name);
priv->drvinit_fail = 1;
}
}
}
//////////////////////////////////////////////////////////////////////////////
r1000_init_ring (dev);
r1000_hw_start (dev);
// ------------------------------------------------------
DBG_PRINT("FIX PCS -> r1000_request_timer\n");
priv->expire_time = R1000_TIMER_EXPIRE_TIME;
r1000_request_timer( (&priv->r1000_timer), priv->expire_time, r1000_timer_handler, ((void *)dev) ); //in open()
DBG_PRINT("%s: %s() alloc_rxskb_cnt = %d\n", dev->name, __FUNCTION__, alloc_rxskb_cnt );
return 0;
}//end of r1000_open (struct net_device *dev)
//======================================================================================================
static void r1000_hw_PHY_reset(struct net_device *dev)
{
int val, phy_reset_expiretime = 50;
struct r1000_private *priv = dev->priv;
unsigned long ioaddr = priv->ioaddr;
DBG_PRINT("%s: Reset RTL8169s PHY\n", dev->name);
val = ( R1000_READ_GMII_REG( ioaddr, 0 ) | 0x8000 ) & 0xffff;
R1000_WRITE_GMII_REG( ioaddr, 0, val );
do //waiting for phy reset
{
if( R1000_READ_GMII_REG( ioaddr, 0 ) & 0x8000 ){
phy_reset_expiretime --;
udelay(100);
}
else{
break;
}
}while( phy_reset_expiretime >= 0 );
assert( phy_reset_expiretime > 0 );
}
//======================================================================================================
static void r1000_hw_PHY_config (struct net_device *dev)
{
struct r1000_private *priv = dev->priv;
void *ioaddr = (void*)priv->ioaddr;
DBG_PRINT("priv->mcfg=%d, priv->pcfg=%d\n",priv->mcfg,priv->pcfg);
if( priv->mcfg == MCFG_METHOD_4 ){
/*
R1000_WRITE_GMII_REG( (unsigned long)ioaddr, 0x1F, 0x0001 );
R1000_WRITE_GMII_REG( (unsigned long)ioaddr, 0x1b, 0x841e );
R1000_WRITE_GMII_REG( (unsigned long)ioaddr, 0x0e, 0x7bfb );
R1000_WRITE_GMII_REG( (unsigned long)ioaddr, 0x09, 0x273a );
*/
R1000_WRITE_GMII_REG( (unsigned long)ioaddr, 0x1F, 0x0002 );
R1000_WRITE_GMII_REG( (unsigned long)ioaddr, 0x01, 0x90D0 );
R1000_WRITE_GMII_REG( (unsigned long)ioaddr, 0x1F, 0x0000 );
}else if((priv->mcfg == MCFG_METHOD_2)||(priv->mcfg == MCFG_METHOD_3)){
R1000_WRITE_GMII_REG( (unsigned long)ioaddr, 0x1F, 0x0001 );
R1000_WRITE_GMII_REG( (unsigned long)ioaddr, 0x15, 0x1000 );
R1000_WRITE_GMII_REG( (unsigned long)ioaddr, 0x18, 0x65C7 );
R1000_WRITE_GMII_REG( (unsigned long)ioaddr, 0x04, 0x0000 );
R1000_WRITE_GMII_REG( (unsigned long)ioaddr, 0x03, 0x00A1 );
R1000_WRITE_GMII_REG( (unsigned long)ioaddr, 0x02, 0x0008 );
R1000_WRITE_GMII_REG( (unsigned long)ioaddr, 0x01, 0x1020 );
R1000_WRITE_GMII_REG( (unsigned long)ioaddr, 0x00, 0x1000 );
R1000_WRITE_GMII_REG( (unsigned long)ioaddr, 0x04, 0x0800 );
R1000_WRITE_GMII_REG( (unsigned long)ioaddr, 0x04, 0x0000 );
R1000_WRITE_GMII_REG( (unsigned long)ioaddr, 0x04, 0x7000 );
R1000_WRITE_GMII_REG( (unsigned long)ioaddr, 0x03, 0xFF41 );
R1000_WRITE_GMII_REG( (unsigned long)ioaddr, 0x02, 0xDE60 );
R1000_WRITE_GMII_REG( (unsigned long)ioaddr, 0x01, 0x0140 );
R1000_WRITE_GMII_REG( (unsigned long)ioaddr, 0x00, 0x0077 );
R1000_WRITE_GMII_REG( (unsigned long)ioaddr, 0x04, 0x7800 );
R1000_WRITE_GMII_REG( (unsigned long)ioaddr, 0x04, 0x7000 );
R1000_WRITE_GMII_REG( (unsigned long)ioaddr, 0x04, 0xA000 );
R1000_WRITE_GMII_REG( (unsigned long)ioaddr, 0x03, 0xDF01 );
R1000_WRITE_GMII_REG( (unsigned long)ioaddr, 0x02, 0xDF20 );
R1000_WRITE_GMII_REG( (unsigned long)ioaddr, 0x01, 0xFF95 );
R1000_WRITE_GMII_REG( (unsigned long)ioaddr, 0x00, 0xFA00 );
R1000_WRITE_GMII_REG( (unsigned long)ioaddr, 0x04, 0xA800 );
R1000_WRITE_GMII_REG( (unsigned long)ioaddr, 0x04, 0xA000 );
R1000_WRITE_GMII_REG( (unsigned long)ioaddr, 0x04, 0xB000 );
R1000_WRITE_GMII_REG( (unsigned long)ioaddr, 0x03, 0xFF41 );
R1000_WRITE_GMII_REG( (unsigned long)ioaddr, 0x02, 0xDE20 );
R1000_WRITE_GMII_REG( (unsigned long)ioaddr, 0x01, 0x0140 );
R1000_WRITE_GMII_REG( (unsigned long)ioaddr, 0x00, 0x00BB );
R1000_WRITE_GMII_REG( (unsigned long)ioaddr, 0x04, 0xB800 );
R1000_WRITE_GMII_REG( (unsigned long)ioaddr, 0x04, 0xB000 );
R1000_WRITE_GMII_REG( (unsigned long)ioaddr, 0x04, 0xF000 );
R1000_WRITE_GMII_REG( (unsigned long)ioaddr, 0x03, 0xDF01 );
R1000_WRITE_GMII_REG( (unsigned long)ioaddr, 0x02, 0xDF20 );
R1000_WRITE_GMII_REG( (unsigned long)ioaddr, 0x01, 0xFF95 );
R1000_WRITE_GMII_REG( (unsigned long)ioaddr, 0x00, 0xBF00 );
R1000_WRITE_GMII_REG( (unsigned long)ioaddr, 0x04, 0xF800 );
R1000_WRITE_GMII_REG( (unsigned long)ioaddr, 0x04, 0xF000 );
R1000_WRITE_GMII_REG( (unsigned long)ioaddr, 0x04, 0x0000 );
R1000_WRITE_GMII_REG( (unsigned long)ioaddr, 0x1F, 0x0000 );
R1000_WRITE_GMII_REG( (unsigned long)ioaddr, 0x0B, 0x0000 );
}
else{
DBG_PRINT("priv->mcfg=%d. Discard hw PHY config.\n",priv->mcfg);
}
}
//======================================================================================================
static void r1000_hw_start (struct net_device *dev)
{
struct r1000_private *priv = dev->priv;
struct pci_dev *pdev = priv->pci_dev;
unsigned long ioaddr = priv->ioaddr;
u32 i;
u8 i8;
u16 i16;
if((priv->mcfg!=MCFG_METHOD_5)&&(priv->mcfg!=MCFG_METHOD_11)&&
(priv->mcfg!=MCFG_METHOD_12)&&(priv->mcfg!=MCFG_METHOD_13)&&
(priv->mcfg!=MCFG_METHOD_14)&&(priv->mcfg!=MCFG_METHOD_15)){
/* Soft reset the chip. */
RTL_W8 ( ChipCmd, CmdReset);
/* Check that the chip has finished the reset. */
for (i = 1000; i > 0; i--){
if ((RTL_R8( ChipCmd ) & CmdReset) == 0) break;
else udelay (10);
}
RTL_W8 ( Cfg9346, Cfg9346_Unlock);
RTL_W8 ( ChipCmd, CmdTxEnb | CmdRxEnb);
RTL_W8 ( ETThReg, ETTh);
// For gigabit rtl8169
RTL_W16 ( RxMaxSize, (unsigned short)priv->hw_rx_pkt_len );
// Set Rx Config register
i = r1000_rx_config | ( RTL_R32( RxConfig ) & rtl_chip_info[priv->chipset].RxConfigMask);
RTL_W32 ( RxConfig, i);
/* Set DMA burst size and Interframe Gap Time */
RTL_W32 ( TxConfig, (TX_DMA_BURST << TxDMAShift) | (InterFrameGap << TxInterFrameGapShift) );
RTL_W16( CPlusCmd, RTL_R16(CPlusCmd) );
if(priv->mcfg==MCFG_METHOD_2||priv->mcfg==MCFG_METHOD_3){
RTL_W16( CPlusCmd, (RTL_R16(CPlusCmd)|(1<<14)|(1<<3)) );
DBG_PRINT("Set MAC Reg C+CR Offset 0xE0: bit-3 and bit-14\n");
}else{
RTL_W16( CPlusCmd, (RTL_R16(CPlusCmd)|(1<<3)) );
DBG_PRINT("Set MAC Reg C+CR Offset 0xE0: bit-3.\n");
}
{
RTL_W16(0xE2,0x0000);
}
priv->cur_rx = 0;
RTL_W32 ( TxDescStartAddr, priv->txdesc_phy_dma_addr);
RTL_W32 ( TxDescStartAddr + 4, 0x00);
RTL_W32 ( RxDescStartAddr, priv->rxdesc_phy_dma_addr);
RTL_W32 ( RxDescStartAddr + 4, 0x00);
RTL_W8 ( Cfg9346, Cfg9346_Lock );
udelay (10);
RTL_W32 ( RxMissed, 0 );
r1000_set_rx_mode (dev);
RTL_W16 ( MultiIntr, RTL_R16(MultiIntr) & 0xF000);
RTL_W16 ( IntrMask, r1000_intr_mask);
}else{
/* Soft reset the chip. */
RTL_W8 ( ChipCmd, CmdReset);
/* Check that the chip has finished the reset. */
for (i = 1000; i > 0; i--){
if ((RTL_R8( ChipCmd ) & CmdReset) == 0) break;
else udelay (10);
}
if( priv->mcfg == MCFG_METHOD_13 ){
pci_write_config_word(pdev,0x68,0x00);
pci_write_config_word(pdev,0x69,0x08);
}
if( priv->mcfg == MCFG_METHOD_5 ){
i8=RTL_R8(Config2);
i8=i8&0x07;
if(i8&&0x01)
RTL_W32(Off7Ch,0x0007FFFF);
i=0x0007FF00;
RTL_W32(Off7Ch, i);
pci_read_config_word(pdev,0x04,&i16);
i16=i16&0xEF;
pci_write_config_word(pdev,0x04,i16);
}
RTL_W8 ( Cfg9346, Cfg9346_Unlock);
RTL_W8 ( ETThReg, ETTh);
// For gigabit rtl8169
RTL_W16 ( RxMaxSize, (unsigned short)priv->hw_rx_pkt_len );
RTL_W16( CPlusCmd, RTL_R16(CPlusCmd) );
if(priv->mcfg==MCFG_METHOD_2||priv->mcfg==MCFG_METHOD_3){
RTL_W16( CPlusCmd, (RTL_R16(CPlusCmd)|(1<<14)|(1<<3)) );
DBG_PRINT("Set MAC Reg C+CR Offset 0xE0: bit-3 and bit-14\n");
}else{
RTL_W16( CPlusCmd, (RTL_R16(CPlusCmd)|(1<<3)) );
DBG_PRINT("Set MAC Reg C+CR Offset 0xE0: bit-3.\n");
}
{
RTL_W16(0xE2,0x0000);
}
priv->cur_rx = 0;
RTL_W32 ( TxDescStartAddr, priv->txdesc_phy_dma_addr);
RTL_W32 ( TxDescStartAddr + 4, 0x00);
RTL_W32 ( RxDescStartAddr, priv->rxdesc_phy_dma_addr);
RTL_W32 ( RxDescStartAddr + 4, 0x00);
RTL_W8 ( ChipCmd, CmdTxEnb | CmdRxEnb);
// Set Rx Config register
i = r1000_rx_config | ( RTL_R32( RxConfig ) & rtl_chip_info[priv->chipset].RxConfigMask);
RTL_W32 ( RxConfig, i);
/* Set DMA burst size and Interframe Gap Time */
RTL_W32 ( TxConfig, (TX_DMA_BURST << TxDMAShift) | (InterFrameGap << TxInterFrameGapShift) );
RTL_W8 ( Cfg9346, Cfg9346_Lock );
udelay (10);
RTL_W32 ( RxMissed, 0 );
r1000_set_rx_mode (dev);
RTL_W16 ( MultiIntr, RTL_R16(MultiIntr) & 0xF000);
RTL_W16 ( IntrMask, r1000_intr_mask);
}
netif_start_queue (dev);
}//end of r1000_hw_start (struct net_device *dev)
//======================================================================================================
static void r1000_init_ring (struct net_device *dev)
{
struct r1000_private *priv = dev->priv;
struct pci_dev *pdev = priv->pci_dev;
int i;
struct sk_buff *skb;
priv->cur_rx = 0;
priv->cur_tx = 0;
priv->dirty_tx = 0;
memset(priv->TxDescArray, 0x0, NUM_TX_DESC*sizeof(struct TxDesc));
memset(priv->RxDescArray, 0x0, NUM_RX_DESC*sizeof(struct RxDesc));
for (i=0 ; i<NUM_TX_DESC ; i++){
priv->Tx_skbuff[i]=NULL;
priv->txdesc_array_dma_addr[i] = pci_map_single(pdev, &priv->TxDescArray[i], sizeof(struct TxDesc), PCI_DMA_TODEVICE);
}
for (i=0; i <NUM_RX_DESC; i++) {
if(i==(NUM_RX_DESC-1)){
priv->RxDescArray[i].status = cpu_to_le32((OWNbit | EORbit) | (unsigned long)priv->hw_rx_pkt_len);
}
else{
priv->RxDescArray[i].status = cpu_to_le32(OWNbit | (unsigned long)priv->hw_rx_pkt_len);
}
{//-----------------------------------------------------------------------
skb = priv->Rx_skbuff[i];
priv->rx_skbuff_dma_addr[i] = pci_map_single(pdev, skb->data, MAX_RX_SKBDATA_SIZE, PCI_DMA_FROMDEVICE);
if( skb != NULL ){
priv->RxDescArray[i].buf_addr = cpu_to_le32(priv->rx_skbuff_dma_addr[i]);
priv->RxDescArray[i].buf_Haddr = 0;
}
else{
DBG_PRINT("%s: %s() Rx_skbuff == NULL\n", dev->name, __FUNCTION__);
priv->drvinit_fail = 1;
}
}//-----------------------------------------------------------------------
priv->rxdesc_array_dma_addr[i] = pci_map_single(pdev, &priv->RxDescArray[i], sizeof(struct RxDesc), PCI_DMA_TODEVICE);
#if LINUX_VERSION_CODE < KERNEL_VERSION(2,6,10)
pci_dma_sync_single(pdev, priv->rxdesc_array_dma_addr[i], sizeof(struct RxDesc), PCI_DMA_TODEVICE);
#endif
}
}
//======================================================================================================
static void r1000_tx_clear (struct r1000_private *priv)
{
int i;
priv->cur_tx = 0;
for ( i = 0 ; i < NUM_TX_DESC ; i++ ){
if ( priv->Tx_skbuff[i] != NULL ) {
dev_kfree_skb ( priv->Tx_skbuff[i] );
priv->Tx_skbuff[i] = NULL;
priv->stats.tx_dropped++;
}
}
}
//======================================================================================================
static void r1000_tx_timeout (struct net_device *dev)
{
struct r1000_private *priv = dev->priv;
unsigned long ioaddr = priv->ioaddr;
u8 tmp8;
/* disable Tx, if not already */
tmp8 = RTL_R8( ChipCmd );
if (tmp8 & CmdTxEnb){
RTL_W8 ( ChipCmd, tmp8 & ~CmdTxEnb);
}
/* Disable interrupts by clearing the interrupt mask. */
RTL_W16 ( IntrMask, 0x0000);
/* Stop a shared interrupt from scavenging while we are. */
spin_lock_irq (&priv->lock);
r1000_tx_clear (priv);
spin_unlock_irq (&priv->lock);
r1000_hw_start (dev);
netif_wake_queue (dev);
}
//======================================================================================================
static int r1000_start_xmit (struct sk_buff *skb, struct net_device *dev)
{
struct r1000_private *priv = dev->priv;
unsigned long ioaddr = priv->ioaddr;
struct pci_dev *pdev = priv->pci_dev;
int entry = priv->cur_tx % NUM_TX_DESC;
int buf_len = 60;
dma_addr_t txbuf_dma_addr;
spin_lock_irq (&priv->lock);
if( (le32_to_cpu(priv->TxDescArray[entry].status) & OWNbit)==0 ){
priv->Tx_skbuff[entry] = skb;
txbuf_dma_addr = pci_map_single(pdev, skb->data, skb->len, PCI_DMA_TODEVICE);
priv->TxDescArray[entry].buf_addr = cpu_to_le32(txbuf_dma_addr);
DBG_PRINT("%s: TX pkt_size = %d\n", __FUNCTION__, skb->len);
if( skb->len <= priv->tx_pkt_len ){
buf_len = skb->len;
}
else{
printk("%s: Error -- Tx packet size(%d) > mtu(%d)+14\n", dev->name, skb->len, dev->mtu);
buf_len = priv->tx_pkt_len;
}
if( entry != (NUM_TX_DESC-1) ){
priv->TxDescArray[entry].status = cpu_to_le32((OWNbit | FSbit | LSbit) | buf_len);
}
else{
priv->TxDescArray[entry].status = cpu_to_le32((OWNbit | EORbit | FSbit | LSbit) | buf_len);
}
#if LINUX_VERSION_CODE < KERNEL_VERSION(2,6,10)
pci_dma_sync_single(pdev, priv->txdesc_array_dma_addr[entry], sizeof(struct TxDesc), PCI_DMA_TODEVICE);
#endif
RTL_W8 ( TxPoll, 0x40); //set polling bit
dev->trans_start = jiffies;
priv->stats.tx_bytes += ( (skb->len > ETH_ZLEN) ? skb->len : ETH_ZLEN);
priv->cur_tx++;
}//end of if( (priv->TxDescArray[entry].status & 0x80000000)==0 )
spin_unlock_irq (&priv->lock);
if ( (priv->cur_tx - NUM_TX_DESC) == priv->dirty_tx ){
netif_stop_queue (dev);
}
⌨️ 快捷键说明
复制代码Ctrl + C
搜索代码Ctrl + F
全屏模式F11
增大字号Ctrl + =
减小字号Ctrl + -
显示快捷键?