ixgbe_82598.c revision 63b3bba89578684727721dabdbefc6c50888394e
/******************************************************************************
Copyright (c) 2001-2010, Intel Corporation
All rights reserved.
Redistribution and use in source and binary forms, with or without
modification, are permitted provided that the following conditions are met:
1. Redistributions of source code must retain the above copyright notice,
this list of conditions and the following disclaimer.
2. Redistributions in binary form must reproduce the above copyright
notice, this list of conditions and the following disclaimer in the
3. Neither the name of the Intel Corporation nor the names of its
contributors may be used to endorse or promote products derived from
this software without specific prior written permission.
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 MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE
LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
INTERRUPTION) 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.
******************************************************************************/
/*$FreeBSD$*/
#include "ixgbe_type.h"
#include "ixgbe_api.h"
#include "ixgbe_common.h"
#include "ixgbe_phy.h"
bool *autoneg);
bool autoneg_wait_to_complete);
bool link_up_wait_to_complete);
bool autoneg,
bool autoneg_wait_to_complete);
bool autoneg,
bool autoneg_wait_to_complete);
u8 *eeprom_data);
/**
* ixgbe_set_pcie_completion_timeout - set pci-e completion timeout
* @hw: pointer to the HW structure
*
* The defaults for 82598 should be in the range of 50us to 50ms,
* however the hardware default for these parts is 500us to 1ms which is less
* than the 10ms recommended by the pci-e spec. To address this we need to
* increase the value to either 10ms to 250ms for capability version 1 config,
* or 16ms to 55ms for version 2.
**/
{
/* only take action if timeout value is defaulted to 0 */
if (gcr & IXGBE_GCR_CMPL_TMOUT_MASK)
goto out;
/*
* if capababilities version is type 1 we can write the
* timeout of 10ms to 250ms through the GCR register
*/
if (!(gcr & IXGBE_GCR_CAP_VER2)) {
goto out;
}
/*
* for version 2 capabilities we need to write the config space
* directly in order to set the completion timeout value for
* 16ms to 55ms
*/
out:
/* disable completion timeout resend */
}
/**
* ixgbe_get_pcie_msix_count_82598 - Gets MSI-X vector count
* @hw: pointer to hardware structure
*
* Read PCIe configuration space, and get the MSI-X vector count from
* the capabilities table.
**/
{
DEBUGFUNC("ixgbe_get_pcie_msix_count_82598");
/* MSI-X count is zero-based in HW, so increment to give
* proper value */
msix_count++;
}
return msix_count;
}
/**
* ixgbe_init_ops_82598 - Inits func ptrs and MAC type
* @hw: pointer to hardware structure
*
* Initialize the function pointers and assign the MAC type for 82598.
* Does not touch the hardware.
**/
{
DEBUGFUNC("ixgbe_init_ops_82598");
/* PHY */
/* MAC */
/* RAR, Multicast, VLAN */
/* Flow Control */
/* SFP+ Module */
/* Link */
return ret_val;
}
/**
* @hw: pointer to hardware structure
*
* Initialize any function pointers that were not able to be
* not known. Perform the SFP init if necessary.
*
**/
{
DEBUGFUNC("ixgbe_init_phy_ops_82598");
/* Identify the PHY */
/* Overwrite the link function pointers if copper PHY */
}
case ixgbe_phy_tn:
break;
case ixgbe_phy_aq:
break;
case ixgbe_phy_nl:
/* Call SFP+ identify routine to get the SFP+ module type */
if (ret_val != IXGBE_SUCCESS)
goto out;
goto out;
}
/* Check to see if SFP+ module is supported */
&data_offset);
if (ret_val != IXGBE_SUCCESS) {
goto out;
}
break;
default:
break;
}
out:
return ret_val;
}
/**
* @hw: pointer to hardware structure
*
* Starts the hardware using the generic start_hw function.
* Disables relaxed ordering Then set pcie completion timeout
*
**/
{
u32 i;
DEBUGFUNC("ixgbe_start_hw_82598");
/* Disable relaxed ordering */
(i < IXGBE_DCA_MAX_QUEUES_82598)); i++) {
}
(i < IXGBE_DCA_MAX_QUEUES_82598)); i++) {
}
/* set the completion timeout for interface */
if (ret_val == IXGBE_SUCCESS)
return ret_val;
}
/**
* ixgbe_get_link_capabilities_82598 - Determines link capabilities
* @hw: pointer to hardware structure
* @speed: pointer to link speed
* @autoneg: boolean auto-negotiation value
*
* Determines the link capabilities by reading the AUTOC register.
**/
bool *autoneg)
{
DEBUGFUNC("ixgbe_get_link_capabilities_82598");
/*
* Determine link capabilities based on the stored value of AUTOC,
* which represents EEPROM defaults. If AUTOC value has not been
* stored, use the current register value.
*/
else
switch (autoc & IXGBE_AUTOC_LMS_MASK) {
break;
break;
case IXGBE_AUTOC_LMS_1G_AN:
break;
case IXGBE_AUTOC_LMS_KX4_AN:
if (autoc & IXGBE_AUTOC_KX4_SUPP)
if (autoc & IXGBE_AUTOC_KX_SUPP)
break;
default:
break;
}
return status;
}
/**
* ixgbe_get_media_type_82598 - Determines media type
* @hw: pointer to hardware structure
*
* Returns the media type (fiber, copper, backplane)
**/
{
enum ixgbe_media_type media_type;
DEBUGFUNC("ixgbe_get_media_type_82598");
/* Detect if there is a copper PHY attached. */
case ixgbe_phy_cu_unknown:
case ixgbe_phy_tn:
case ixgbe_phy_aq:
goto out;
default:
break;
}
/* Media type for I82598 is based on device ID */
case IXGBE_DEV_ID_82598:
case IXGBE_DEV_ID_82598_BX:
break;
break;
case IXGBE_DEV_ID_82598EB_CX4:
break;
case IXGBE_DEV_ID_82598AT:
case IXGBE_DEV_ID_82598AT2:
break;
default:
break;
}
out:
return media_type;
}
/**
* ixgbe_fc_enable_82598 - Enable flow control
* @hw: pointer to hardware structure
* @packetbuf_num: packet buffer number (0-7)
*
* Enable flow control according to the current settings.
**/
{
u32 link_speed = 0;
bool link_up;
DEBUGFUNC("ixgbe_fc_enable_82598");
/*
* On 82598 having Rx FC on causes resets while doing 1G
* so if it's on turn it off once we know link_speed. For
* more details see 82598 Specification update.
*/
case ixgbe_fc_full:
break;
case ixgbe_fc_rx_pause:
break;
default:
/* no change */
break;
}
}
/* Negotiate the fc mode to use */
if (ret_val == IXGBE_ERR_FLOW_CONTROL)
goto out;
/* Disable any previous flow control settings */
/*
* The possible values of fc.current_mode are:
* 0: Flow control is completely disabled
* 1: Rx flow control is enabled (we can receive pause frames,
* but not send pause frames).
* 2: Tx flow control is enabled (we can send pause frames but
* we do not support receiving pause frames).
* 3: Both Rx and Tx flow control (symmetric) are enabled.
* other: Invalid.
*/
case ixgbe_fc_none:
/*
* Flow control is disabled by software override or autoneg.
* The code below will actually disable it in the HW.
*/
break;
case ixgbe_fc_rx_pause:
/*
* Rx Flow control is enabled and Tx Flow control is
* disabled by software override. Since there really
* isn't a way to advertise that we are capable of RX
* Pause ONLY, we will advertise that we support both
* symmetric and asymmetric Rx PAUSE. Later, we will
* disable the adapter's ability to send PAUSE frames.
*/
break;
case ixgbe_fc_tx_pause:
/*
* Tx Flow control is enabled, and Rx Flow control is
* disabled by software override.
*/
break;
case ixgbe_fc_full:
/* Flow control (both Rx and Tx) is enabled by SW override. */
break;
default:
DEBUGOUT("Flow control param set incorrectly\n");
goto out;
}
/* Set 802.3x based flow control settings. */
reg |= IXGBE_FCRTL_XONE;
reg |= IXGBE_FCRTH_FCEN;
}
/* Configure pause time (2 TCs per register) */
if ((packetbuf_num & 1) == 0)
else
out:
return ret_val;
}
/**
* ixgbe_start_mac_link_82598 - Configures MAC link settings
* @hw: pointer to hardware structure
*
* Configures link settings based on values in the ixgbe_hw struct.
* Restarts the link. Performs autonegotiation if needed.
**/
bool autoneg_wait_to_complete)
{
u32 i;
DEBUGFUNC("ixgbe_start_mac_link_82598");
/* Restart link */
/* Only poll for autoneg to complete if specified to do so */
if (autoneg_wait_to_complete) {
if ((autoc_reg & IXGBE_AUTOC_LMS_MASK) ==
(autoc_reg & IXGBE_AUTOC_LMS_MASK) ==
links_reg = 0; /* Just in case Autoneg time = 0 */
for (i = 0; i < IXGBE_AUTO_NEG_TIME; i++) {
if (links_reg & IXGBE_LINKS_KX_AN_COMP)
break;
msec_delay(100);
}
if (!(links_reg & IXGBE_LINKS_KX_AN_COMP)) {
DEBUGOUT("Autonegotiation did not complete.\n");
}
}
}
/* Add delay to filter out noises during initial link setup */
msec_delay(50);
return status;
}
/**
* ixgbe_validate_link_ready - Function looks for phy link
* @hw: pointer to hardware structure
*
* Function indicates success when phy link is available. If phy is not ready
* within 5 seconds of MAC indicating link, the function returns error.
**/
{
return IXGBE_SUCCESS;
for (timeout = 0;
if ((an_reg & IXGBE_MII_AUTONEG_COMPLETE) &&
break;
msec_delay(100);
}
if (timeout == IXGBE_VALIDATE_LINK_READY_TIMEOUT) {
DEBUGOUT("Link was indicated but link is down\n");
return IXGBE_ERR_LINK_SETUP;
}
return IXGBE_SUCCESS;
}
/**
* @hw: pointer to hardware structure
* @speed: pointer to link speed
* @link_up: TRUE is link is up, FALSE otherwise
* @link_up_wait_to_complete: bool used to wait for link up or not
*
* Reads the links register to determine if link is up and the current speed
**/
bool link_up_wait_to_complete)
{
u32 i;
DEBUGFUNC("ixgbe_check_mac_link_82598");
/*
* SERDES PHY requires us to read link status from undocumented
* indicates link down. OxC00C is read to check that the XAUI lanes
* are active. Bit 0 clear indicates active; set indicates inactive.
*/
if (link_up_wait_to_complete) {
for (i = 0; i < IXGBE_LINK_UP_TIME; i++) {
if ((link_reg & 1) &&
((adapt_comp_reg & 1) == 0)) {
break;
} else {
}
msec_delay(100);
&link_reg);
}
} else {
else
}
goto out;
}
if (link_up_wait_to_complete) {
for (i = 0; i < IXGBE_LINK_UP_TIME; i++) {
if (links_reg & IXGBE_LINKS_UP) {
break;
} else {
}
msec_delay(100);
}
} else {
if (links_reg & IXGBE_LINKS_UP)
else
}
if (links_reg & IXGBE_LINKS_SPEED)
else
/* if link is down, zero out the current_mode */
}
out:
return IXGBE_SUCCESS;
}
/**
* ixgbe_setup_mac_link_82598 - Set MAC link speed
* @hw: pointer to hardware structure
* @speed: new link speed
* @autoneg: TRUE if autonegotiation enabled
* @autoneg_wait_to_complete: TRUE when waiting for completion is needed
*
* Set the link speed in the AUTOC register and restarts link.
**/
bool autoneg_wait_to_complete)
{
DEBUGFUNC("ixgbe_setup_mac_link_82598");
/* Check to see if speed passed in is supported. */
if (speed == IXGBE_LINK_SPEED_UNKNOWN)
else if (link_mode == IXGBE_AUTOC_LMS_KX4_AN ||
if (speed & IXGBE_LINK_SPEED_10GB_FULL)
if (speed & IXGBE_LINK_SPEED_1GB_FULL)
if (autoc != curr_autoc)
}
if (status == IXGBE_SUCCESS) {
/*
* Setup and restart the link based on the new values in
* ixgbe_hw This will write the AUTOC register based on the new
* stored values
*/
}
return status;
}
/**
* ixgbe_setup_copper_link_82598 - Set the PHY autoneg advertised field
* @hw: pointer to hardware structure
* @speed: new link speed
* @autoneg: TRUE if autonegotiation enabled
* @autoneg_wait_to_complete: TRUE if waiting is needed to complete
*
* Sets the link speed in the AUTOC register in the MAC and restarts link.
**/
bool autoneg,
bool autoneg_wait_to_complete)
{
DEBUGFUNC("ixgbe_setup_copper_link_82598");
/* Setup the PHY according to input speed */
/* Set up MAC */
return status;
}
/**
* ixgbe_reset_hw_82598 - Performs hardware reset
* @hw: pointer to hardware structure
*
* Resets the hardware by resetting the transmit and receive units, masks and
* clears all interrupts, performing a PHY reset, and performing a link (MAC)
* reset.
**/
{
u32 i;
DEBUGFUNC("ixgbe_reset_hw_82598");
/*
* Power up the Atlas Tx lanes if they are currently powered down.
* Atlas Tx lanes are powered down for MAC loopback tests, but
* they are not automatically restored on reset.
*/
if (analog_val & IXGBE_ATLAS_PDN_TX_REG_EN) {
/* Enable Tx Atlas so packets can be transmitted again */
&analog_val);
&analog_val);
&analog_val);
&analog_val);
}
/* Reset PHY */
/* PHY ops must be identified and initialized prior to reset */
/* Init PHY and function pointers, perform SFP setup */
goto reset_hw_out;
else if (phy_status == IXGBE_ERR_SFP_NOT_PRESENT)
goto no_phy_reset;
}
/*
* Prevent the PCI-E bus from from hanging by disabling PCI-E master
* access and verify no pending requests before reset
*/
(void) ixgbe_disable_pcie_master(hw);
/*
* Issue global reset to the MAC. This needs to be a SW reset.
* If link reset is used, it might reset the MAC when mng is using it
*/
/* Poll for reset bit to self-clear indicating reset is complete */
for (i = 0; i < 10; i++) {
usec_delay(1);
if (!(ctrl & IXGBE_CTRL_RST))
break;
}
if (ctrl & IXGBE_CTRL_RST) {
DEBUGOUT("Reset polling failed to complete.\n");
}
/*
* Double resets are required for recovery from certain error
* conditions. Between resets, it is necessary to stall to allow time
* for any pending HW events to complete. We use 1usec since that is
* what is needed for ixgbe_disable_pcie_master(). The second reset
* then clears out any effects of those events.
*/
usec_delay(1);
goto mac_reset_top;
}
msec_delay(50);
/*
* Store the original AUTOC value if it has not been
* stored off yet. Otherwise restore the stored original
* AUTOC value since the reset operation sets back to deaults.
*/
}
/* Store the permanent mac address */
/*
* Store MAC address from RAR0, clear receive address registers, and
* clear the multicast table
*/
if (phy_status != IXGBE_SUCCESS)
status = phy_status;
return status;
}
/**
* ixgbe_set_vmdq_82598 - Associate a VMDq set index with a rx address
* @hw: pointer to hardware struct
* @rar: receive address register index to associate with a VMDq index
* @vmdq: VMDq set index
**/
{
DEBUGFUNC("ixgbe_set_vmdq_82598");
/* Make sure we are using a valid rar index range */
if (rar >= rar_entries) {
return IXGBE_ERR_INVALID_ARGUMENT;
}
return IXGBE_SUCCESS;
}
/**
* ixgbe_clear_vmdq_82598 - Disassociate a VMDq set index from an rx address
* @hw: pointer to hardware struct
* @rar: receive address register index to associate with a VMDq index
* @vmdq: VMDq clear index (not used in 82598, but elsewhere)
**/
{
/* Make sure we are using a valid rar index range */
if (rar >= rar_entries) {
return IXGBE_ERR_INVALID_ARGUMENT;
}
if (rar_high & IXGBE_RAH_VIND_MASK) {
}
return IXGBE_SUCCESS;
}
/**
* ixgbe_set_vfta_82598 - Set VLAN filter table
* @hw: pointer to hardware structure
* @vlan: VLAN id to write to VLAN filter
* @vind: VMDq output index that maps queue to VLAN id in VFTA
*
**/
bool vlan_on)
{
DEBUGFUNC("ixgbe_set_vfta_82598");
if (vlan > 4095)
return IXGBE_ERR_PARAM;
/* Determine 32-bit word position in array */
/* Determine the location of the (VMD) queue index */
/* Set the nibble for VMD queue index */
/* Determine the location of the bit for this VLAN id */
if (vlan_on)
/* Turn on this VLAN id */
else
/* Turn off this VLAN id */
return IXGBE_SUCCESS;
}
/**
* ixgbe_clear_vfta_82598 - Clear VLAN filter table
* @hw: pointer to hardware structure
*
* Clears the VLAN filer table, and the VMDq index associated with the filter
**/
{
DEBUGFUNC("ixgbe_clear_vfta_82598");
0);
return IXGBE_SUCCESS;
}
/**
* ixgbe_read_analog_reg8_82598 - Reads 8 bit Atlas analog register
* @hw: pointer to hardware structure
* @reg: analog register to read
* @val: read value
*
* Performs read operation to Atlas analog register specified.
**/
{
DEBUGFUNC("ixgbe_read_analog_reg8_82598");
usec_delay(10);
return IXGBE_SUCCESS;
}
/**
* ixgbe_write_analog_reg8_82598 - Writes 8 bit Atlas analog register
* @hw: pointer to hardware structure
* @reg: atlas register to write
* @val: value to write
*
* Performs write operation to Atlas analog register specified.
**/
{
DEBUGFUNC("ixgbe_write_analog_reg8_82598");
usec_delay(10);
return IXGBE_SUCCESS;
}
/**
* ixgbe_read_i2c_eeprom_82598 - Reads 8 bit word over I2C interface.
* @hw: pointer to hardware structure
* @byte_offset: EEPROM byte offset to read
* @eeprom_data: value read
*
* Performs 8 byte read operation to SFP module's EEPROM over I2C interface.
**/
{
u32 i;
DEBUGFUNC("ixgbe_read_i2c_eeprom_82598");
/*
* 0xC30D. These registers are used to talk to the SFP+
*/
sfp_addr);
/* Poll status */
for (i = 0; i < 100; i++) {
&sfp_stat);
break;
msec_delay(10);
}
if (sfp_stat != IXGBE_I2C_EEPROM_STATUS_PASS) {
DEBUGOUT("EEPROM read did not pass.\n");
goto out;
}
/* Read data */
} else {
goto out;
}
out:
return status;
}
/**
* ixgbe_get_supported_physical_layer_82598 - Returns physical layer type
* @hw: pointer to hardware structure
*
* Determines physical layer capabilities of the current configuration.
**/
{
u16 ext_ability = 0;
DEBUGFUNC("ixgbe_get_supported_physical_layer_82598");
/* Copper PHY must be checked before AUTOC LMS to determine correct
case ixgbe_phy_tn:
case ixgbe_phy_aq:
case ixgbe_phy_cu_unknown:
goto out;
default:
break;
}
switch (autoc & IXGBE_AUTOC_LMS_MASK) {
case IXGBE_AUTOC_LMS_1G_AN:
if (pma_pmd_1g == IXGBE_AUTOC_1G_KX)
else
break;
if (pma_pmd_10g == IXGBE_AUTOC_10G_CX4)
else if (pma_pmd_10g == IXGBE_AUTOC_10G_KX4)
else /* XAUI */
break;
case IXGBE_AUTOC_LMS_KX4_AN:
if (autoc & IXGBE_AUTOC_KX_SUPP)
if (autoc & IXGBE_AUTOC_KX4_SUPP)
break;
default:
break;
}
case ixgbe_sfp_type_da_cu:
break;
case ixgbe_sfp_type_sr:
break;
case ixgbe_sfp_type_lr:
break;
default:
break;
}
}
break;
break;
break;
default:
break;
}
out:
return physical_layer;
}
/**
* ixgbe_set_lan_id_multi_port_pcie_82598 - Set LAN id for PCIe multiple
* port devices.
* @hw: pointer to the HW structure
*
* Calls common function and corrects issue with some single port devices
* that enable LAN1 but not LAN0.
**/
{
DEBUGFUNC("ixgbe_set_lan_id_multi_port_pcie_82598");
/* check if LAN0 is disabled */
/* if LAN0 is completely disabled force function to 0 */
if ((pci_ctrl2 & IXGBE_PCIE_CTRL2_LAN_DISABLE) &&
}
}
}
/**
* ixgbe_enable_relaxed_ordering_82598 - enable relaxed ordering
* @hw: pointer to hardware structure
*
**/
{
u32 i;
DEBUGFUNC("ixgbe_enable_relaxed_ordering_82598");
/* Enable relaxed ordering */
(i < IXGBE_DCA_MAX_QUEUES_82598)); i++) {
}
(i < IXGBE_DCA_MAX_QUEUES_82598)); i++) {
}
}