e100_phy.c
来自「是关于linux2.5.1的完全源码」· C语言 代码 · 共 1,130 行 · 第 1/3 页
C
1,130 行
/*******************************************************************************This software program is available to you under a choice of one of two licenses. You may choose to be licensed under either the GNU General Public License 2.0, June 1991, available at http://www.fsf.org/copyleft/gpl.html, or the Intel BSD + Patent License, the text of which follows:Recipient has requested a license and Intel Corporation ("Intel") is willingto grant a license for the software entitled Linux Base Driver for the Intel(R) PRO/100 Family of Adapters (e100) (the "Software") being provided by Intel Corporation. The following definitions apply to this license:"Licensed Patents" means patent claims licensable by Intel Corporation which are necessarily infringed by the use of sale of the Software alone or when combined with the operating system referred to below."Recipient" means the party to whom Intel delivers this Software."Licensee" means Recipient and those third parties that receive a license to any operating system available under the GNU General Public License 2.0 or later.Copyright (c) 1999 - 2002 Intel Corporation.All rights reserved.The license is provided to Recipient and Recipient's Licensees under the following terms.Redistribution and use in source and binary forms of the Software, with or without modification, are permitted provided that the following conditions are met:Redistributions of source code of the Software may retain the above copyright notice, this list of conditions and the following disclaimer.Redistributions in binary form of the Software may reproduce the above copyright notice, this list of conditions and the following disclaimer in the documentation and/or materials provided with the distribution.Neither the name of Intel Corporation nor the names of its contributors shall be used to endorse or promote products derived from this Software without specific prior written permission.Intel hereby grants Recipient and Licensees a non-exclusive, worldwide, royalty-free patent license under Licensed Patents to make, use, sell, offer to sell, import and otherwise transfer the Software, if any, in source code and object code form. This license shall include changes to the Software that are error corrections or other minor changes to the Software that do not add functionality or features when the Software is incorporated in any version of an operating system that has been distributed under the GNU General Public License 2.0 or later. This patent license shall apply to the combination of the Software and any operating system licensed under the GNU General Public License 2.0 or later if, at the time Intel provides the Software to Recipient, such addition of the Software to the then publicly available versions of such operating systems available under the GNU General Public License 2.0 or later (whether in gold, beta or alpha form) causes such combination to be covered by the Licensed Patents. The patent license shall not apply to any other combinations which include the Software. NO hardware per se is licensed hereunder.THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MECHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL INTEL OR IT CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; ANY LOSS OF USE; DATA, OR PROFITS; OR BUSINESS INTERUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.*******************************************************************************/#include "e100_phy.h"void e100_handle_zlock(struct e100_private *bdp);/* * Procedure: e100_mdi_write * * Description: This routine will write a value to the specified MII register * of an external MDI compliant device (e.g. PHY 100). The * command will execute in polled mode. * * Arguments: * bdp - Ptr to this card's e100_bdconfig structure * reg_addr - The MII register that we are writing to * phy_addr - The MDI address of the Phy component. * data - The value that we are writing to the MII register. * * Returns: * NOTHING */voide100_mdi_write(struct e100_private *bdp, u32 reg_addr, u32 phy_addr, u16 data){ int e100_retry; u32 temp_val; temp_val = (((u32) data) | (reg_addr << 16) | (phy_addr << 21) | (MDI_WRITE << 26)); writel(temp_val, &bdp->scb->scb_mdi_cntrl); /* wait 20usec before checking status */ udelay(20); /* poll for the mdi write to complete */ e100_retry = E100_CMD_WAIT; while ((!(readl(&bdp->scb->scb_mdi_cntrl) & MDI_PHY_READY)) && (e100_retry)) { udelay(20); e100_retry--; }}/* * Procedure: e100_mdi_read * * Description: This routine will read a value from the specified MII register * of an external MDI compliant device (e.g. PHY 100), and return * it to the calling routine. The command will execute in polled * mode. * * Arguments: * bdp - Ptr to this card's e100_bdconfig structure * reg_addr - The MII register that we are reading from * phy_addr - The MDI address of the Phy component. * * Results: * data - The value that we read from the MII register. * * Returns: * NOTHING */voide100_mdi_read(struct e100_private *bdp, u32 reg_addr, u32 phy_addr, u16 *data){ int e100_retry; u32 temp_val; /* Issue the read command to the MDI control register. */ temp_val = ((reg_addr << 16) | (phy_addr << 21) | (MDI_READ << 26)); writel(temp_val, &bdp->scb->scb_mdi_cntrl); /* wait 20usec before checking status */ udelay(20); /* poll for the mdi read to complete */ e100_retry = E100_CMD_WAIT; while ((!(readl(&bdp->scb->scb_mdi_cntrl) & MDI_PHY_READY)) && (e100_retry)) { udelay(20); e100_retry--; } // return the lower word *data = (u16) readl(&bdp->scb->scb_mdi_cntrl);}static unsigned char __devinite100_phy_valid(struct e100_private *bdp, unsigned int phy_address){ u16 ctrl_reg, stat_reg; /* Read the MDI control register */ e100_mdi_read(bdp, MII_BMCR, phy_address, &ctrl_reg); /* Read the status register twice, bacause of sticky bits */ e100_mdi_read(bdp, MII_BMSR, phy_address, &stat_reg); e100_mdi_read(bdp, MII_BMSR, phy_address, &stat_reg); if ((ctrl_reg == 0xffff) || ((stat_reg == 0) && (ctrl_reg == 0))) return false; return true;}static void __devinite100_phy_address_detect(struct e100_private *bdp){ unsigned int addr; unsigned char valid_phy_found = false; if (IS_NC3133(bdp)) { bdp->phy_addr = 0; return; } if (e100_phy_valid(bdp, PHY_DEFAULT_ADDRESS)) { bdp->phy_addr = PHY_DEFAULT_ADDRESS; valid_phy_found = true; } else { for (addr = MIN_PHY_ADDR; addr <= MAX_PHY_ADDR; addr++) { if (e100_phy_valid(bdp, addr)) { bdp->phy_addr = addr; valid_phy_found = true; break; } } } if (!valid_phy_found) { bdp->phy_addr = PHY_ADDRESS_503; }}static void __devinite100_phy_id_detect(struct e100_private *bdp){ u16 low_id_reg, high_id_reg; if (bdp->phy_addr == PHY_ADDRESS_503) { bdp->PhyId = PHY_503; return; } if (!(bdp->flags & IS_ICH)) { if (bdp->rev_id >= D102_REV_ID) { bdp->PhyId = PHY_82562ET; return; } } /* Read phy id from the MII register */ e100_mdi_read(bdp, MII_PHYSID1, bdp->phy_addr, &low_id_reg); e100_mdi_read(bdp, MII_PHYSID2, bdp->phy_addr, &high_id_reg); bdp->PhyId = ((unsigned int) low_id_reg | ((unsigned int) high_id_reg << 16));}static void __devinite100_phy_isolate(struct e100_private *bdp){ unsigned int phy_address; u16 ctrl_reg; /* Go over all phy addresses. Deisolate the selected one, and isolate * all the rest */ for (phy_address = 0; phy_address <= MAX_PHY_ADDR; phy_address++) { if (phy_address != bdp->phy_addr) { e100_mdi_write(bdp, MII_BMCR, phy_address, BMCR_ISOLATE); } else { e100_mdi_read(bdp, MII_BMCR, bdp->phy_addr, &ctrl_reg); ctrl_reg &= ~BMCR_ISOLATE; e100_mdi_write(bdp, MII_BMCR, bdp->phy_addr, ctrl_reg); } udelay(100); }}static unsigned char __devinite100_phy_specific_setup(struct e100_private *bdp){ u16 misc_reg; if (bdp->phy_addr == PHY_ADDRESS_503) { switch (bdp->params.e100_speed_duplex) { case E100_AUTONEG: /* The adapter can't autoneg. so set to 10/HALF */ printk(KERN_INFO "503 serial component detected which " "cannot autonegotiate\n"); printk(KERN_INFO "speed/duplex forced to 10Mbps / Half duplex\n"); bdp->params.e100_speed_duplex = E100_SPEED_10_HALF; break; case E100_SPEED_100_HALF: case E100_SPEED_100_FULL: printk(KERN_ERR "503 serial component detected which does not " "support 100Mbps\n"); printk(KERN_ERR "Change the forced speed/duplex to a supported " "setting\n"); return false; } return true; } if (IS_NC3133(bdp)) { u16 int_reg; /* enable 100BASE fiber interface */ e100_mdi_write(bdp, MDI_NC3133_CONFIG_REG, bdp->phy_addr, MDI_NC3133_100FX_ENABLE); if ((bdp->params.e100_speed_duplex != E100_AUTONEG) && (bdp->params.e100_speed_duplex != E100_SPEED_100_FULL)) { /* just inform user about 100 full */ printk(KERN_ERR "NC3133 NIC can only run " "at 100Mbps full duplex\n"); } bdp->params.e100_speed_duplex = E100_SPEED_100_FULL; /* enable interrupts */ e100_mdi_read(bdp, MDI_NC3133_INT_ENABLE_REG, bdp->phy_addr, &int_reg); int_reg |= MDI_NC3133_INT_ENABLE; e100_mdi_write(bdp, MDI_NC3133_INT_ENABLE_REG, bdp->phy_addr, int_reg); } /* Handle the National TX */ if ((bdp->PhyId & PHY_MODEL_REV_ID_MASK) == PHY_NSC_TX) { e100_mdi_read(bdp, NSC_CONG_CONTROL_REG, bdp->phy_addr, &misc_reg); misc_reg |= NSC_TX_CONG_TXREADY; /* disable the congestion control bit in the National Phy */ misc_reg &= ~NSC_TX_CONG_ENABLE; e100_mdi_write(bdp, NSC_CONG_CONTROL_REG, bdp->phy_addr, misc_reg); } return true;}/* * Procedure: e100_phy_fix_squelch * * Description: * Help find link on certain rare scenarios. * NOTE: This routine must be called once per watchdog, * and *after* setting the current link state. * * Arguments: * bdp - Ptr to this card's e100_bdconfig structure * * Returns: * NOTHING */static voide100_phy_fix_squelch(struct e100_private *bdp){ if ((bdp->PhyId != PHY_82555_TX) || (bdp->flags & DF_SPEED_FORCED)) return; if (netif_carrier_ok(bdp->device)) { switch (bdp->PhyState) { case 0: break; case 1: e100_mdi_write(bdp, PHY_82555_SPECIAL_CONTROL, bdp->phy_addr, 0x0000); break; case 2: e100_mdi_write(bdp, PHY_82555_MDI_EQUALIZER_CSR, bdp->phy_addr, 0x3000); break; } bdp->PhyState = 0; bdp->PhyDelay = 0; } else if (!bdp->PhyDelay--) { switch (bdp->PhyState) { case 0: e100_mdi_write(bdp, PHY_82555_SPECIAL_CONTROL, bdp->phy_addr, EXTENDED_SQUELCH_BIT); bdp->PhyState = 1; break; case 1: e100_mdi_write(bdp, PHY_82555_SPECIAL_CONTROL, bdp->phy_addr, 0x0000); e100_mdi_write(bdp, PHY_82555_MDI_EQUALIZER_CSR, bdp->phy_addr, 0x2010); bdp->PhyState = 2; break;
⌨️ 快捷键说明
复制代码Ctrl + C
搜索代码Ctrl + F
全屏模式F11
增大字号Ctrl + =
减小字号Ctrl + -
显示快捷键?