57de9f8139
-It has new hardware support -It uses a new method of TX cleanup called Head Write Back -It includes the provisional generic TCP LRO feature contributed by Myricom and made general purpose by me. This should move into the stack upon approval but for this driver drop its in here. -Also bug fixes and etc... MFC in a week if no serious issues arise.
1117 lines
34 KiB
C
1117 lines
34 KiB
C
/******************************************************************************
|
|
|
|
Copyright (c) 2001-2008, 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
|
|
documentation and/or other materials provided with the distribution.
|
|
|
|
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"
|
|
|
|
s32 ixgbe_init_ops_82598(struct ixgbe_hw *hw);
|
|
s32 ixgbe_get_link_capabilities_82598(struct ixgbe_hw *hw,
|
|
ixgbe_link_speed *speed,
|
|
bool *autoneg);
|
|
s32 ixgbe_get_copper_link_capabilities_82598(struct ixgbe_hw *hw,
|
|
ixgbe_link_speed *speed,
|
|
bool *autoneg);
|
|
enum ixgbe_media_type ixgbe_get_media_type_82598(struct ixgbe_hw *hw);
|
|
s32 ixgbe_setup_fc_82598(struct ixgbe_hw *hw, s32 packetbuf_num);
|
|
s32 ixgbe_setup_mac_link_82598(struct ixgbe_hw *hw);
|
|
s32 ixgbe_check_mac_link_82598(struct ixgbe_hw *hw,
|
|
ixgbe_link_speed *speed,
|
|
bool *link_up, bool link_up_wait_to_complete);
|
|
s32 ixgbe_setup_mac_link_speed_82598(struct ixgbe_hw *hw,
|
|
ixgbe_link_speed speed,
|
|
bool autoneg,
|
|
bool autoneg_wait_to_complete);
|
|
s32 ixgbe_setup_copper_link_82598(struct ixgbe_hw *hw);
|
|
s32 ixgbe_setup_copper_link_speed_82598(struct ixgbe_hw *hw,
|
|
ixgbe_link_speed speed,
|
|
bool autoneg,
|
|
bool autoneg_wait_to_complete);
|
|
#ifndef NO_82598_A0_SUPPORT
|
|
s32 ixgbe_reset_hw_rev_0_82598(struct ixgbe_hw *hw);
|
|
#endif
|
|
s32 ixgbe_reset_hw_82598(struct ixgbe_hw *hw);
|
|
s32 ixgbe_configure_fiber_serdes_fc_82598(struct ixgbe_hw *hw);
|
|
s32 ixgbe_setup_fiber_serdes_link_82598(struct ixgbe_hw *hw);
|
|
s32 ixgbe_set_vmdq_82598(struct ixgbe_hw *hw, u32 rar, u32 vmdq);
|
|
s32 ixgbe_blink_led_stop_82598(struct ixgbe_hw *hw, u32 index);
|
|
s32 ixgbe_blink_led_start_82598(struct ixgbe_hw *hw, u32 index);
|
|
|
|
/**
|
|
* 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.
|
|
**/
|
|
s32 ixgbe_init_ops_82598(struct ixgbe_hw *hw)
|
|
{
|
|
struct ixgbe_mac_info *mac = &hw->mac;
|
|
struct ixgbe_phy_info *phy = &hw->phy;
|
|
s32 ret_val;
|
|
|
|
ret_val = ixgbe_init_phy_ops_generic(hw);
|
|
ret_val = ixgbe_init_ops_generic(hw);
|
|
|
|
/* MAC */
|
|
#ifndef NO_82598_A0_SUPPORT
|
|
if (hw->revision_id == 0)
|
|
mac->ops.reset_hw = &ixgbe_reset_hw_rev_0_82598;
|
|
else
|
|
mac->ops.reset_hw = &ixgbe_reset_hw_82598;
|
|
#else
|
|
mac->ops.reset_hw = &ixgbe_reset_hw_82598;
|
|
#endif
|
|
mac->ops.get_media_type = &ixgbe_get_media_type_82598;
|
|
|
|
/* LEDs */
|
|
mac->ops.blink_led_start = &ixgbe_blink_led_start_82598;
|
|
mac->ops.blink_led_stop = &ixgbe_blink_led_stop_82598;
|
|
|
|
/* RAR, Multicast, VLAN */
|
|
mac->ops.set_vmdq = &ixgbe_set_vmdq_82598;
|
|
|
|
/* Flow Control */
|
|
mac->ops.setup_fc = &ixgbe_setup_fc_82598;
|
|
|
|
/* Call PHY identify routine to get the phy type */
|
|
phy->ops.identify(hw);
|
|
|
|
/* PHY Init */
|
|
switch (hw->phy.type) {
|
|
case ixgbe_phy_tn:
|
|
phy->ops.check_link = &ixgbe_check_phy_link_tnx;
|
|
phy->ops.get_firmware_version =
|
|
&ixgbe_get_phy_firmware_version_tnx;
|
|
break;
|
|
case ixgbe_phy_nl:
|
|
phy->ops.reset = &ixgbe_reset_phy_nl;
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
|
|
/* Link */
|
|
mac->ops.check_link = &ixgbe_check_mac_link_82598;
|
|
if (mac->ops.get_media_type(hw) == ixgbe_media_type_copper) {
|
|
mac->ops.setup_link = &ixgbe_setup_copper_link_82598;
|
|
mac->ops.setup_link_speed =
|
|
&ixgbe_setup_copper_link_speed_82598;
|
|
mac->ops.get_link_capabilities =
|
|
&ixgbe_get_copper_link_capabilities_82598;
|
|
} else {
|
|
mac->ops.setup_link = &ixgbe_setup_mac_link_82598;
|
|
mac->ops.setup_link_speed = &ixgbe_setup_mac_link_speed_82598;
|
|
mac->ops.get_link_capabilities =
|
|
&ixgbe_get_link_capabilities_82598;
|
|
}
|
|
|
|
mac->mcft_size = 128;
|
|
mac->vft_size = 128;
|
|
mac->num_rar_entries = 16;
|
|
mac->max_tx_queues = 32;
|
|
mac->max_rx_queues = 64;
|
|
|
|
return IXGBE_SUCCESS;
|
|
}
|
|
|
|
/**
|
|
* 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.
|
|
**/
|
|
s32 ixgbe_get_link_capabilities_82598(struct ixgbe_hw *hw,
|
|
ixgbe_link_speed *speed,
|
|
bool *autoneg)
|
|
{
|
|
s32 status = IXGBE_SUCCESS;
|
|
s32 autoc_reg;
|
|
|
|
autoc_reg = IXGBE_READ_REG(hw, IXGBE_AUTOC);
|
|
|
|
if (hw->mac.link_settings_loaded) {
|
|
autoc_reg &= ~IXGBE_AUTOC_LMS_ATTACH_TYPE;
|
|
autoc_reg &= ~IXGBE_AUTOC_LMS_MASK;
|
|
autoc_reg |= hw->mac.link_attach_type;
|
|
autoc_reg |= hw->mac.link_mode_select;
|
|
}
|
|
|
|
switch (autoc_reg & IXGBE_AUTOC_LMS_MASK) {
|
|
case IXGBE_AUTOC_LMS_1G_LINK_NO_AN:
|
|
*speed = IXGBE_LINK_SPEED_1GB_FULL;
|
|
*autoneg = FALSE;
|
|
break;
|
|
|
|
case IXGBE_AUTOC_LMS_10G_LINK_NO_AN:
|
|
*speed = IXGBE_LINK_SPEED_10GB_FULL;
|
|
*autoneg = FALSE;
|
|
break;
|
|
|
|
case IXGBE_AUTOC_LMS_1G_AN:
|
|
*speed = IXGBE_LINK_SPEED_1GB_FULL;
|
|
*autoneg = TRUE;
|
|
break;
|
|
|
|
case IXGBE_AUTOC_LMS_KX4_AN:
|
|
case IXGBE_AUTOC_LMS_KX4_AN_1G_AN:
|
|
*speed = IXGBE_LINK_SPEED_UNKNOWN;
|
|
if (autoc_reg & IXGBE_AUTOC_KX4_SUPP)
|
|
*speed |= IXGBE_LINK_SPEED_10GB_FULL;
|
|
if (autoc_reg & IXGBE_AUTOC_KX_SUPP)
|
|
*speed |= IXGBE_LINK_SPEED_1GB_FULL;
|
|
*autoneg = TRUE;
|
|
break;
|
|
|
|
default:
|
|
status = IXGBE_ERR_LINK_SETUP;
|
|
break;
|
|
}
|
|
|
|
return status;
|
|
}
|
|
|
|
/**
|
|
* ixgbe_get_copper_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.
|
|
**/
|
|
s32 ixgbe_get_copper_link_capabilities_82598(struct ixgbe_hw *hw,
|
|
ixgbe_link_speed *speed,
|
|
bool *autoneg)
|
|
{
|
|
s32 status = IXGBE_ERR_LINK_SETUP;
|
|
u16 speed_ability;
|
|
|
|
*speed = 0;
|
|
*autoneg = TRUE;
|
|
|
|
status = hw->phy.ops.read_reg(hw, IXGBE_MDIO_PHY_SPEED_ABILITY,
|
|
IXGBE_MDIO_PMA_PMD_DEV_TYPE,
|
|
&speed_ability);
|
|
|
|
if (status == IXGBE_SUCCESS) {
|
|
if (speed_ability & IXGBE_MDIO_PHY_SPEED_10G)
|
|
*speed |= IXGBE_LINK_SPEED_10GB_FULL;
|
|
if (speed_ability & IXGBE_MDIO_PHY_SPEED_1G)
|
|
*speed |= IXGBE_LINK_SPEED_1GB_FULL;
|
|
}
|
|
|
|
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 ixgbe_get_media_type_82598(struct ixgbe_hw *hw)
|
|
{
|
|
enum ixgbe_media_type media_type;
|
|
|
|
/* Media type for I82598 is based on device ID */
|
|
switch (hw->device_id) {
|
|
case IXGBE_DEV_ID_82598AF_DUAL_PORT:
|
|
case IXGBE_DEV_ID_82598AF_SINGLE_PORT:
|
|
case IXGBE_DEV_ID_82598EB_CX4:
|
|
case IXGBE_DEV_ID_82598_CX4_DUAL_PORT:
|
|
case IXGBE_DEV_ID_82598_DA_DUAL_PORT:
|
|
case IXGBE_DEV_ID_82598_SR_DUAL_PORT_EM:
|
|
case IXGBE_DEV_ID_82598EB_XF_LR:
|
|
media_type = ixgbe_media_type_fiber;
|
|
break;
|
|
case IXGBE_DEV_ID_82598AT:
|
|
media_type = ixgbe_media_type_copper;
|
|
break;
|
|
case IXGBE_DEV_ID_82598AT_DUAL_PORT:
|
|
media_type = ixgbe_media_type_copper;
|
|
break;
|
|
default:
|
|
media_type = ixgbe_media_type_unknown;
|
|
break;
|
|
}
|
|
|
|
return media_type;
|
|
}
|
|
|
|
/**
|
|
* ixgbe_setup_fc_82598 - Configure flow control settings
|
|
* @hw: pointer to hardware structure
|
|
* @packetbuf_num: packet buffer number (0-7)
|
|
*
|
|
* Configures the flow control settings based on SW configuration. This
|
|
* function is used for 802.3x flow control configuration only.
|
|
**/
|
|
s32 ixgbe_setup_fc_82598(struct ixgbe_hw *hw, s32 packetbuf_num)
|
|
{
|
|
u32 frctl_reg;
|
|
u32 rmcs_reg;
|
|
|
|
if (packetbuf_num < 0 || packetbuf_num > 7) {
|
|
DEBUGOUT1("Invalid packet buffer number [%d], expected range is"
|
|
" 0-7\n", packetbuf_num);
|
|
ASSERT(0);
|
|
}
|
|
|
|
frctl_reg = IXGBE_READ_REG(hw, IXGBE_FCTRL);
|
|
frctl_reg &= ~(IXGBE_FCTRL_RFCE | IXGBE_FCTRL_RPFCE);
|
|
|
|
rmcs_reg = IXGBE_READ_REG(hw, IXGBE_RMCS);
|
|
rmcs_reg &= ~(IXGBE_RMCS_TFCE_PRIORITY | IXGBE_RMCS_TFCE_802_3X);
|
|
|
|
/*
|
|
* 10 gig parts do not have a word in the EEPROM to determine the
|
|
* default flow control setting, so we explicitly set it to full.
|
|
*/
|
|
if (hw->fc.type == ixgbe_fc_default)
|
|
hw->fc.type = ixgbe_fc_full;
|
|
|
|
/*
|
|
* We want to save off the original Flow Control configuration just in
|
|
* case we get disconnected and then reconnected into a different hub
|
|
* or switch with different Flow Control capabilities.
|
|
*/
|
|
hw->fc.original_type = hw->fc.type;
|
|
|
|
/*
|
|
* The possible values of the "flow_control" parameter 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.
|
|
*/
|
|
switch (hw->fc.type) {
|
|
case ixgbe_fc_none:
|
|
break;
|
|
case ixgbe_fc_rx_pause:
|
|
/*
|
|
* Rx Flow control is enabled,
|
|
* and Tx Flow control is disabled.
|
|
*/
|
|
frctl_reg |= IXGBE_FCTRL_RFCE;
|
|
break;
|
|
case ixgbe_fc_tx_pause:
|
|
/*
|
|
* Tx Flow control is enabled, and Rx Flow control is disabled,
|
|
* by a software over-ride.
|
|
*/
|
|
rmcs_reg |= IXGBE_RMCS_TFCE_802_3X;
|
|
break;
|
|
case ixgbe_fc_full:
|
|
/*
|
|
* Flow control (both Rx and Tx) is enabled by a software
|
|
* over-ride.
|
|
*/
|
|
frctl_reg |= IXGBE_FCTRL_RFCE;
|
|
rmcs_reg |= IXGBE_RMCS_TFCE_802_3X;
|
|
break;
|
|
default:
|
|
/* We should never get here. The value should be 0-3. */
|
|
DEBUGOUT("Flow control param set incorrectly\n");
|
|
ASSERT(0);
|
|
break;
|
|
}
|
|
|
|
/* Enable 802.3x based flow control settings. */
|
|
IXGBE_WRITE_REG(hw, IXGBE_FCTRL, frctl_reg);
|
|
IXGBE_WRITE_REG(hw, IXGBE_RMCS, rmcs_reg);
|
|
|
|
/*
|
|
* Check for invalid software configuration, zeros are completely
|
|
* invalid for all parameters used past this point, and if we enable
|
|
* flow control with zero water marks, we blast flow control packets.
|
|
*/
|
|
if (!hw->fc.low_water || !hw->fc.high_water || !hw->fc.pause_time) {
|
|
DEBUGOUT("Flow control structure initialized incorrectly\n");
|
|
return IXGBE_ERR_INVALID_LINK_SETTINGS;
|
|
}
|
|
|
|
/*
|
|
* We need to set up the Receive Threshold high and low water
|
|
* marks as well as (optionally) enabling the transmission of
|
|
* XON frames.
|
|
*/
|
|
if (hw->fc.type & ixgbe_fc_tx_pause) {
|
|
if (hw->fc.send_xon) {
|
|
IXGBE_WRITE_REG(hw, IXGBE_FCRTL(packetbuf_num),
|
|
(hw->fc.low_water | IXGBE_FCRTL_XONE));
|
|
} else {
|
|
IXGBE_WRITE_REG(hw, IXGBE_FCRTL(packetbuf_num),
|
|
hw->fc.low_water);
|
|
}
|
|
IXGBE_WRITE_REG(hw, IXGBE_FCRTH(packetbuf_num),
|
|
(hw->fc.high_water)|IXGBE_FCRTH_FCEN);
|
|
}
|
|
|
|
IXGBE_WRITE_REG(hw, IXGBE_FCTTV(0), hw->fc.pause_time);
|
|
IXGBE_WRITE_REG(hw, IXGBE_FCRTV, (hw->fc.pause_time >> 1));
|
|
|
|
return IXGBE_SUCCESS;
|
|
}
|
|
|
|
/**
|
|
* ixgbe_setup_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.
|
|
**/
|
|
s32 ixgbe_setup_mac_link_82598(struct ixgbe_hw *hw)
|
|
{
|
|
ixgbe_link_speed speed;
|
|
bool link_up;
|
|
u32 autoc_reg;
|
|
u32 links_reg;
|
|
u32 i;
|
|
s32 status = IXGBE_SUCCESS;
|
|
|
|
autoc_reg = IXGBE_READ_REG(hw, IXGBE_AUTOC);
|
|
|
|
if (hw->mac.link_settings_loaded) {
|
|
autoc_reg &= ~IXGBE_AUTOC_LMS_ATTACH_TYPE;
|
|
autoc_reg &= ~IXGBE_AUTOC_LMS_MASK;
|
|
autoc_reg |= hw->mac.link_attach_type;
|
|
autoc_reg |= hw->mac.link_mode_select;
|
|
|
|
IXGBE_WRITE_REG(hw, IXGBE_AUTOC, autoc_reg);
|
|
IXGBE_WRITE_FLUSH(hw);
|
|
msec_delay(50);
|
|
}
|
|
|
|
/* Restart link */
|
|
autoc_reg |= IXGBE_AUTOC_AN_RESTART;
|
|
IXGBE_WRITE_REG(hw, IXGBE_AUTOC, autoc_reg);
|
|
|
|
/* Only poll for autoneg to complete if specified to do so */
|
|
if (hw->phy.autoneg_wait_to_complete) {
|
|
if (hw->mac.link_mode_select == IXGBE_AUTOC_LMS_KX4_AN ||
|
|
hw->mac.link_mode_select == IXGBE_AUTOC_LMS_KX4_AN_1G_AN) {
|
|
links_reg = 0; /* Just in case Autoneg time = 0 */
|
|
for (i = 0; i < IXGBE_AUTO_NEG_TIME; i++) {
|
|
links_reg = IXGBE_READ_REG(hw, IXGBE_LINKS);
|
|
if (links_reg & IXGBE_LINKS_KX_AN_COMP)
|
|
break;
|
|
msec_delay(100);
|
|
}
|
|
if (!(links_reg & IXGBE_LINKS_KX_AN_COMP)) {
|
|
status = IXGBE_ERR_AUTONEG_NOT_COMPLETE;
|
|
DEBUGOUT("Autonegotiation did not complete.\n");
|
|
}
|
|
}
|
|
}
|
|
|
|
/*
|
|
* We want to save off the original Flow Control configuration just in
|
|
* case we get disconnected and then reconnected into a different hub
|
|
* or switch with different Flow Control capabilities.
|
|
*/
|
|
hw->fc.original_type = hw->fc.type;
|
|
/*
|
|
* Set up the SerDes link if in 1Gb mode, otherwise just set up
|
|
* 10Gb flow control.
|
|
*/
|
|
hw->mac.ops.check_link(hw, &speed, &link_up, FALSE);
|
|
if (speed == IXGBE_LINK_SPEED_1GB_FULL)
|
|
status = ixgbe_setup_fiber_serdes_link_82598(hw);
|
|
else
|
|
ixgbe_setup_fc_82598(hw, 0);
|
|
|
|
/* Add delay to filter out noises during initial link setup */
|
|
msec_delay(50);
|
|
|
|
return status;
|
|
}
|
|
|
|
/**
|
|
* ixgbe_check_mac_link_82598 - Get link/speed status
|
|
* @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
|
|
**/
|
|
s32 ixgbe_check_mac_link_82598(struct ixgbe_hw *hw, ixgbe_link_speed *speed,
|
|
bool *link_up, bool link_up_wait_to_complete)
|
|
{
|
|
u32 links_reg;
|
|
u32 i;
|
|
u16 link_reg, adapt_comp_reg;
|
|
|
|
if (hw->phy.type == ixgbe_phy_nl) {
|
|
hw->phy.ops.read_reg(hw, 1, IXGBE_TWINAX_DEV, &link_reg);
|
|
hw->phy.ops.read_reg(hw, 1, IXGBE_TWINAX_DEV, &link_reg);
|
|
hw->phy.ops.read_reg(hw, 0xC00C, IXGBE_TWINAX_DEV,
|
|
&adapt_comp_reg);
|
|
if (link_up_wait_to_complete) {
|
|
for (i = 0; i < IXGBE_LINK_UP_TIME; i++) {
|
|
if ((link_reg & (1 << 2)) &&
|
|
((adapt_comp_reg & 1) == 0)) {
|
|
*link_up = TRUE;
|
|
break;
|
|
} else {
|
|
*link_up = FALSE;
|
|
}
|
|
msec_delay(100);
|
|
hw->phy.ops.read_reg(hw, 1, IXGBE_TWINAX_DEV,
|
|
&link_reg);
|
|
hw->phy.ops.read_reg(hw, 0xC00C,
|
|
IXGBE_TWINAX_DEV,
|
|
&adapt_comp_reg);
|
|
}
|
|
} else {
|
|
if ((link_reg & (1 << 2)) &&
|
|
((adapt_comp_reg & 1) == 0))
|
|
*link_up = TRUE;
|
|
else
|
|
*link_up = FALSE;
|
|
}
|
|
|
|
if (*link_up == FALSE)
|
|
goto out;
|
|
}
|
|
|
|
links_reg = IXGBE_READ_REG(hw, IXGBE_LINKS);
|
|
if (link_up_wait_to_complete) {
|
|
for (i = 0; i < IXGBE_LINK_UP_TIME; i++) {
|
|
if (links_reg & IXGBE_LINKS_UP) {
|
|
*link_up = TRUE;
|
|
break;
|
|
} else {
|
|
*link_up = FALSE;
|
|
}
|
|
msec_delay(100);
|
|
links_reg = IXGBE_READ_REG(hw, IXGBE_LINKS);
|
|
}
|
|
} else {
|
|
if (links_reg & IXGBE_LINKS_UP)
|
|
*link_up = TRUE;
|
|
else
|
|
*link_up = FALSE;
|
|
}
|
|
|
|
if (links_reg & IXGBE_LINKS_SPEED)
|
|
*speed = IXGBE_LINK_SPEED_10GB_FULL;
|
|
else
|
|
*speed = IXGBE_LINK_SPEED_1GB_FULL;
|
|
|
|
out:
|
|
return IXGBE_SUCCESS;
|
|
}
|
|
|
|
/**
|
|
* ixgbe_configure_fiber_serdes_fc_82598 - Configure fiber flow control
|
|
* @hw: pointer to hardware structure
|
|
*
|
|
* Reads PCS registers and sets flow control settings, based on
|
|
* link-partner's abilities.
|
|
**/
|
|
s32 ixgbe_configure_fiber_serdes_fc_82598(struct ixgbe_hw *hw)
|
|
{
|
|
s32 ret_val = IXGBE_SUCCESS;
|
|
u32 pcs_anadv_reg, pcs_lpab_reg, pcs_lstat_reg, i;
|
|
DEBUGFUNC("ixgbe_configure_fiber_serdes_fc_82598");
|
|
|
|
/* Check that autonegotiation has completed */
|
|
for (i = 0; i < FIBER_LINK_UP_LIMIT; i++) {
|
|
msec_delay(10);
|
|
pcs_lstat_reg = IXGBE_READ_REG(hw, IXGBE_PCS1GLSTA);
|
|
if (pcs_lstat_reg & IXGBE_PCS1GLSTA_LINK_OK) {
|
|
if (pcs_lstat_reg & IXGBE_PCS1GLSTA_AN_COMPLETE) {
|
|
if (!(pcs_lstat_reg &
|
|
(IXGBE_PCS1GLSTA_AN_TIMED_OUT)))
|
|
hw->mac.autoneg_failed = 0;
|
|
else
|
|
hw->mac.autoneg_failed = 1;
|
|
break;
|
|
} else {
|
|
hw->mac.autoneg_failed = 1;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (hw->mac.autoneg_failed) {
|
|
/*
|
|
* AutoNeg failed to achieve a link, so we will turn
|
|
* flow control off.
|
|
*/
|
|
hw->fc.type = ixgbe_fc_none;
|
|
DEBUGOUT("Flow Control = NONE.\n");
|
|
ret_val = ixgbe_setup_fc_82598(hw, 0);
|
|
goto out;
|
|
}
|
|
|
|
/*
|
|
* Read the AN advertisement and LP ability registers and resolve
|
|
* local flow control settings accordingly
|
|
*/
|
|
pcs_anadv_reg = IXGBE_READ_REG(hw, IXGBE_PCS1GANA);
|
|
pcs_lpab_reg = IXGBE_READ_REG(hw, IXGBE_PCS1GANLP);
|
|
if ((pcs_anadv_reg & IXGBE_PCS1GANA_SYM_PAUSE) &&
|
|
(pcs_lpab_reg & IXGBE_PCS1GANA_SYM_PAUSE)) {
|
|
/*
|
|
* Now we need to check if the user selected Rx ONLY
|
|
* of pause frames. In this case, we had to advertise
|
|
* FULL flow control because we could not advertise RX
|
|
* ONLY. Hence, we must now check to see if we need to
|
|
* turn OFF the TRANSMISSION of PAUSE frames.
|
|
*/
|
|
if (hw->fc.original_type == ixgbe_fc_full) {
|
|
hw->fc.type = ixgbe_fc_full;
|
|
DEBUGOUT("Flow Control = FULL.\n");
|
|
} else {
|
|
hw->fc.type = ixgbe_fc_rx_pause;
|
|
DEBUGOUT("Flow Control = RX PAUSE frames only.\n");
|
|
}
|
|
} else if (!(pcs_anadv_reg & IXGBE_PCS1GANA_SYM_PAUSE) &&
|
|
(pcs_anadv_reg & IXGBE_PCS1GANA_ASM_PAUSE) &&
|
|
(pcs_lpab_reg & IXGBE_PCS1GANA_SYM_PAUSE) &&
|
|
(pcs_lpab_reg & IXGBE_PCS1GANA_ASM_PAUSE)) {
|
|
hw->fc.type = ixgbe_fc_tx_pause;
|
|
DEBUGOUT("Flow Control = TX PAUSE frames only.\n");
|
|
} else if ((pcs_anadv_reg & IXGBE_PCS1GANA_SYM_PAUSE) &&
|
|
(pcs_anadv_reg & IXGBE_PCS1GANA_ASM_PAUSE) &&
|
|
!(pcs_lpab_reg & IXGBE_PCS1GANA_SYM_PAUSE) &&
|
|
(pcs_lpab_reg & IXGBE_PCS1GANA_ASM_PAUSE)) {
|
|
hw->fc.type = ixgbe_fc_rx_pause;
|
|
DEBUGOUT("Flow Control = RX PAUSE frames only.\n");
|
|
} else {
|
|
hw->fc.type = ixgbe_fc_none;
|
|
DEBUGOUT("Flow Control = NONE.\n");
|
|
}
|
|
|
|
ret_val = ixgbe_setup_fc_82598(hw, 0);
|
|
if (ret_val) {
|
|
DEBUGOUT("Error forcing flow control settings\n");
|
|
goto out;
|
|
}
|
|
|
|
out:
|
|
return ret_val;
|
|
}
|
|
|
|
/**
|
|
* ixgbe_setup_fiber_serdes_link_82598 - Configure fiber serdes link
|
|
* @hw: pointer to hardware structure
|
|
*
|
|
* Sets up PCS registers and sets flow control settings, based on
|
|
* link-partner's abilities.
|
|
**/
|
|
s32 ixgbe_setup_fiber_serdes_link_82598(struct ixgbe_hw *hw)
|
|
{
|
|
u32 reg;
|
|
s32 ret_val;
|
|
|
|
DEBUGFUNC("ixgbe_setup_fiber_serdes_link_82598");
|
|
|
|
/*
|
|
* 10 gig parts do not have a word in the EEPROM to determine the
|
|
* default flow control setting, so we explicitly set it to full.
|
|
*/
|
|
if (hw->fc.type == ixgbe_fc_default)
|
|
hw->fc.type = ixgbe_fc_full;
|
|
|
|
/*
|
|
* 82598 fiber/serdes devices require that flow control be resolved in
|
|
* software.
|
|
*/
|
|
reg = IXGBE_READ_REG(hw, IXGBE_PCS1GANA);
|
|
|
|
/*
|
|
* The possible values of the "fc" parameter 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.
|
|
*/
|
|
switch (hw->fc.type) {
|
|
case ixgbe_fc_none:
|
|
/*
|
|
* Flow control completely disabled by a software
|
|
* over-ride.
|
|
*/
|
|
reg &= ~(IXGBE_PCS1GANA_SYM_PAUSE | IXGBE_PCS1GANA_ASM_PAUSE);
|
|
break;
|
|
case ixgbe_fc_rx_pause:
|
|
/*
|
|
* Rx Flow control is enabled and Tx Flow control is
|
|
* disabled by a software over-ride. 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.
|
|
*/
|
|
reg |= (IXGBE_PCS1GANA_SYM_PAUSE | IXGBE_PCS1GANA_ASM_PAUSE);
|
|
break;
|
|
case ixgbe_fc_tx_pause:
|
|
/*
|
|
* Tx Flow control is enabled, and Rx Flow control is
|
|
* disabled, by a software over-ride.
|
|
*/
|
|
reg |= (IXGBE_PCS1GANA_ASM_PAUSE);
|
|
reg &= ~(IXGBE_PCS1GANA_SYM_PAUSE);
|
|
break;
|
|
case ixgbe_fc_full:
|
|
/*
|
|
* Flow control (both Rx and Tx) is enabled by a
|
|
* software over-ride.
|
|
*/
|
|
reg |= (IXGBE_PCS1GANA_SYM_PAUSE | IXGBE_PCS1GANA_ASM_PAUSE);
|
|
break;
|
|
default:
|
|
DEBUGOUT("Flow control param set incorrectly\n");
|
|
ret_val = -IXGBE_ERR_CONFIG;
|
|
goto out;
|
|
break;
|
|
}
|
|
|
|
IXGBE_WRITE_REG(hw, IXGBE_PCS1GANA, reg);
|
|
reg = IXGBE_READ_REG(hw, IXGBE_PCS1GLCTL);
|
|
|
|
/* Set PCS register for autoneg */
|
|
/* Enable and restart autoneg */
|
|
reg |= IXGBE_PCS1GLCTL_AN_ENABLE | IXGBE_PCS1GLCTL_AN_RESTART;
|
|
|
|
reg &= ~IXGBE_PCS1GLCTL_AN_1G_TIMEOUT_EN; /* Disable AN timeout */
|
|
DEBUGOUT1("Configuring Autoneg; PCS_LCTL = 0x%08X\n", reg);
|
|
IXGBE_WRITE_REG(hw, IXGBE_PCS1GLCTL, reg);
|
|
|
|
/*
|
|
* Configure flow control. If we aren't auto-negotiating,
|
|
* just setup the flow control and do not worry about PCS autoneg.
|
|
*/
|
|
ixgbe_configure_fiber_serdes_fc_82598(hw);
|
|
|
|
out:
|
|
return IXGBE_SUCCESS;
|
|
}
|
|
|
|
/**
|
|
* ixgbe_setup_mac_link_speed_82598 - Set MAC link speed
|
|
* @hw: pointer to hardware structure
|
|
* @speed: new link speed
|
|
* @autoneg: TRUE if auto-negotiation enabled
|
|
* @autoneg_wait_to_complete: TRUE if waiting is needed to complete
|
|
*
|
|
* Set the link speed in the AUTOC register and restarts link.
|
|
**/
|
|
s32 ixgbe_setup_mac_link_speed_82598(struct ixgbe_hw *hw,
|
|
ixgbe_link_speed speed, bool autoneg,
|
|
bool autoneg_wait_to_complete)
|
|
{
|
|
s32 status = IXGBE_SUCCESS;
|
|
|
|
/* If speed is 10G, then check for CX4 or XAUI. */
|
|
if ((speed == IXGBE_LINK_SPEED_10GB_FULL) &&
|
|
(!(hw->mac.link_attach_type & IXGBE_AUTOC_10G_KX4))) {
|
|
hw->mac.link_mode_select = IXGBE_AUTOC_LMS_10G_LINK_NO_AN;
|
|
} else if ((speed == IXGBE_LINK_SPEED_1GB_FULL) && (!autoneg)) {
|
|
hw->mac.link_mode_select = IXGBE_AUTOC_LMS_1G_LINK_NO_AN;
|
|
} else if (autoneg) {
|
|
/* BX mode - Autonegotiate 1G */
|
|
if (!(hw->mac.link_attach_type & IXGBE_AUTOC_1G_PMA_PMD))
|
|
hw->mac.link_mode_select = IXGBE_AUTOC_LMS_1G_AN;
|
|
else /* KX/KX4 mode */
|
|
hw->mac.link_mode_select = IXGBE_AUTOC_LMS_KX4_AN_1G_AN;
|
|
} else {
|
|
status = IXGBE_ERR_LINK_SETUP;
|
|
}
|
|
|
|
if (status == IXGBE_SUCCESS) {
|
|
hw->phy.autoneg_wait_to_complete = autoneg_wait_to_complete;
|
|
|
|
hw->mac.link_settings_loaded = TRUE;
|
|
/*
|
|
* 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
|
|
*/
|
|
ixgbe_setup_mac_link_82598(hw);
|
|
}
|
|
|
|
return status;
|
|
}
|
|
|
|
|
|
/**
|
|
* ixgbe_setup_copper_link_82598 - Setup copper 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. Restart
|
|
* phy and wait for autonegotiate to finish. Then synchronize the
|
|
* MAC and PHY.
|
|
**/
|
|
s32 ixgbe_setup_copper_link_82598(struct ixgbe_hw *hw)
|
|
{
|
|
s32 status;
|
|
|
|
/* Restart autonegotiation on PHY */
|
|
status = hw->phy.ops.setup_link(hw);
|
|
|
|
/* Set MAC to KX/KX4 autoneg, which defaults to Parallel detection */
|
|
hw->mac.link_attach_type = (IXGBE_AUTOC_10G_KX4 | IXGBE_AUTOC_1G_KX);
|
|
hw->mac.link_mode_select = IXGBE_AUTOC_LMS_KX4_AN;
|
|
|
|
/* Set up MAC */
|
|
ixgbe_setup_mac_link_82598(hw);
|
|
|
|
return status;
|
|
}
|
|
|
|
/**
|
|
* ixgbe_setup_copper_link_speed_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.
|
|
**/
|
|
s32 ixgbe_setup_copper_link_speed_82598(struct ixgbe_hw *hw,
|
|
ixgbe_link_speed speed,
|
|
bool autoneg,
|
|
bool autoneg_wait_to_complete)
|
|
{
|
|
s32 status;
|
|
|
|
/* Setup the PHY according to input speed */
|
|
status = hw->phy.ops.setup_link_speed(hw, speed, autoneg,
|
|
autoneg_wait_to_complete);
|
|
|
|
/* Set MAC to KX/KX4 autoneg, which defaults to Parallel detection */
|
|
hw->mac.link_attach_type = (IXGBE_AUTOC_10G_KX4 | IXGBE_AUTOC_1G_KX);
|
|
hw->mac.link_mode_select = IXGBE_AUTOC_LMS_KX4_AN;
|
|
|
|
/* Set up MAC */
|
|
ixgbe_setup_mac_link_82598(hw);
|
|
|
|
return status;
|
|
}
|
|
|
|
#ifndef NO_82598_A0_SUPPORT
|
|
/**
|
|
* ixgbe_reset_hw_rev_0_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.
|
|
**/
|
|
s32 ixgbe_reset_hw_rev_0_82598(struct ixgbe_hw *hw)
|
|
{
|
|
s32 status = IXGBE_SUCCESS;
|
|
u32 ctrl;
|
|
u32 gheccr;
|
|
u32 autoc;
|
|
u32 i;
|
|
u32 resets;
|
|
|
|
/* Call adapter stop to disable tx/rx and clear interrupts */
|
|
hw->mac.ops.stop_adapter(hw);
|
|
|
|
/* Reset PHY */
|
|
hw->phy.ops.reset(hw);
|
|
|
|
for (resets = 0; resets < 10; resets++) {
|
|
/*
|
|
* Prevent the PCI-E bus from from hanging by disabling PCI-E
|
|
* master access and verify no pending requests before reset
|
|
*/
|
|
if (ixgbe_disable_pcie_master(hw) != IXGBE_SUCCESS) {
|
|
status = IXGBE_ERR_MASTER_REQUESTS_PENDING;
|
|
DEBUGOUT("PCI-E Master disable polling has failed.\n");
|
|
}
|
|
|
|
/*
|
|
* 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.
|
|
*/
|
|
ctrl = IXGBE_READ_REG(hw, IXGBE_CTRL);
|
|
IXGBE_WRITE_REG(hw, IXGBE_CTRL, (ctrl | IXGBE_CTRL_RST));
|
|
IXGBE_WRITE_FLUSH(hw);
|
|
|
|
/*
|
|
* Poll for reset bit to self-clear indicating reset is
|
|
* complete
|
|
*/
|
|
for (i = 0; i < 10; i++) {
|
|
usec_delay(1);
|
|
ctrl = IXGBE_READ_REG(hw, IXGBE_CTRL);
|
|
if (!(ctrl & IXGBE_CTRL_RST))
|
|
break;
|
|
}
|
|
if (ctrl & IXGBE_CTRL_RST) {
|
|
status = IXGBE_ERR_RESET_FAILED;
|
|
DEBUGOUT("Reset polling failed to complete.\n");
|
|
}
|
|
}
|
|
|
|
msec_delay(50);
|
|
|
|
gheccr = IXGBE_READ_REG(hw, IXGBE_GHECCR);
|
|
gheccr &= ~((1 << 21) | (1 << 18) | (1 << 9) | (1 << 6));
|
|
IXGBE_WRITE_REG(hw, IXGBE_GHECCR, gheccr);
|
|
|
|
/*
|
|
* AUTOC register which stores link settings gets cleared
|
|
* and reloaded from EEPROM after reset. We need to restore
|
|
* our stored value from init in case SW changed the attach
|
|
* type or speed. If this is the first time and link settings
|
|
* have not been stored, store default settings from AUTOC.
|
|
*/
|
|
autoc = IXGBE_READ_REG(hw, IXGBE_AUTOC);
|
|
if (hw->mac.link_settings_loaded) {
|
|
autoc &= ~(IXGBE_AUTOC_LMS_ATTACH_TYPE);
|
|
autoc &= ~(IXGBE_AUTOC_LMS_MASK);
|
|
autoc |= hw->mac.link_attach_type;
|
|
autoc |= hw->mac.link_mode_select;
|
|
IXGBE_WRITE_REG(hw, IXGBE_AUTOC, autoc);
|
|
} else {
|
|
hw->mac.link_attach_type =
|
|
(autoc & IXGBE_AUTOC_LMS_ATTACH_TYPE);
|
|
hw->mac.link_mode_select = (autoc & IXGBE_AUTOC_LMS_MASK);
|
|
hw->mac.link_settings_loaded = TRUE;
|
|
}
|
|
|
|
/* Store the permanent mac address */
|
|
hw->mac.ops.get_mac_addr(hw, hw->mac.perm_addr);
|
|
|
|
return status;
|
|
}
|
|
|
|
#endif /* NO_A0_SUPPORT */
|
|
/**
|
|
* 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.
|
|
**/
|
|
s32 ixgbe_reset_hw_82598(struct ixgbe_hw *hw)
|
|
{
|
|
s32 status = IXGBE_SUCCESS;
|
|
u32 ctrl;
|
|
u32 gheccr;
|
|
u32 i;
|
|
u32 autoc;
|
|
u8 analog_val;
|
|
|
|
/* Call adapter stop to disable tx/rx and clear interrupts */
|
|
hw->mac.ops.stop_adapter(hw);
|
|
|
|
/*
|
|
* 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.
|
|
*/
|
|
hw->mac.ops.read_analog_reg8(hw, IXGBE_ATLAS_PDN_LPBK, &analog_val);
|
|
if (analog_val & IXGBE_ATLAS_PDN_TX_REG_EN) {
|
|
/* Enable Tx Atlas so packets can be transmitted again */
|
|
hw->mac.ops.read_analog_reg8(hw, IXGBE_ATLAS_PDN_LPBK,
|
|
&analog_val);
|
|
analog_val &= ~IXGBE_ATLAS_PDN_TX_REG_EN;
|
|
hw->mac.ops.write_analog_reg8(hw, IXGBE_ATLAS_PDN_LPBK,
|
|
analog_val);
|
|
|
|
hw->mac.ops.read_analog_reg8(hw, IXGBE_ATLAS_PDN_10G,
|
|
&analog_val);
|
|
analog_val &= ~ IXGBE_ATLAS_PDN_TX_10G_QL_ALL;
|
|
hw->mac.ops.write_analog_reg8(hw, IXGBE_ATLAS_PDN_10G,
|
|
analog_val);
|
|
|
|
hw->mac.ops.read_analog_reg8(hw, IXGBE_ATLAS_PDN_1G,
|
|
&analog_val);
|
|
analog_val &= ~IXGBE_ATLAS_PDN_TX_1G_QL_ALL;
|
|
hw->mac.ops.write_analog_reg8(hw, IXGBE_ATLAS_PDN_1G,
|
|
analog_val);
|
|
|
|
hw->mac.ops.read_analog_reg8(hw, IXGBE_ATLAS_PDN_AN,
|
|
&analog_val);
|
|
analog_val &= ~IXGBE_ATLAS_PDN_TX_AN_QL_ALL;
|
|
hw->mac.ops.write_analog_reg8(hw, IXGBE_ATLAS_PDN_AN,
|
|
analog_val);
|
|
}
|
|
|
|
/* Reset PHY */
|
|
if (hw->phy.reset_disable == FALSE)
|
|
hw->phy.ops.reset(hw);
|
|
|
|
/*
|
|
* Prevent the PCI-E bus from from hanging by disabling PCI-E master
|
|
* access and verify no pending requests before reset
|
|
*/
|
|
if (ixgbe_disable_pcie_master(hw) != IXGBE_SUCCESS) {
|
|
status = IXGBE_ERR_MASTER_REQUESTS_PENDING;
|
|
DEBUGOUT("PCI-E Master disable polling has failed.\n");
|
|
}
|
|
|
|
/*
|
|
* 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
|
|
*/
|
|
ctrl = IXGBE_READ_REG(hw, IXGBE_CTRL);
|
|
IXGBE_WRITE_REG(hw, IXGBE_CTRL, (ctrl | IXGBE_CTRL_RST));
|
|
IXGBE_WRITE_FLUSH(hw);
|
|
|
|
/* Poll for reset bit to self-clear indicating reset is complete */
|
|
for (i = 0; i < 10; i++) {
|
|
usec_delay(1);
|
|
ctrl = IXGBE_READ_REG(hw, IXGBE_CTRL);
|
|
if (!(ctrl & IXGBE_CTRL_RST))
|
|
break;
|
|
}
|
|
if (ctrl & IXGBE_CTRL_RST) {
|
|
status = IXGBE_ERR_RESET_FAILED;
|
|
DEBUGOUT("Reset polling failed to complete.\n");
|
|
}
|
|
|
|
msec_delay(50);
|
|
|
|
gheccr = IXGBE_READ_REG(hw, IXGBE_GHECCR);
|
|
gheccr &= ~((1 << 21) | (1 << 18) | (1 << 9) | (1 << 6));
|
|
IXGBE_WRITE_REG(hw, IXGBE_GHECCR, gheccr);
|
|
|
|
/*
|
|
* AUTOC register which stores link settings gets cleared
|
|
* and reloaded from EEPROM after reset. We need to restore
|
|
* our stored value from init in case SW changed the attach
|
|
* type or speed. If this is the first time and link settings
|
|
* have not been stored, store default settings from AUTOC.
|
|
*/
|
|
autoc = IXGBE_READ_REG(hw, IXGBE_AUTOC);
|
|
if (hw->mac.link_settings_loaded) {
|
|
autoc &= ~(IXGBE_AUTOC_LMS_ATTACH_TYPE);
|
|
autoc &= ~(IXGBE_AUTOC_LMS_MASK);
|
|
autoc |= hw->mac.link_attach_type;
|
|
autoc |= hw->mac.link_mode_select;
|
|
IXGBE_WRITE_REG(hw, IXGBE_AUTOC, autoc);
|
|
} else {
|
|
hw->mac.link_attach_type =
|
|
(autoc & IXGBE_AUTOC_LMS_ATTACH_TYPE);
|
|
hw->mac.link_mode_select = (autoc & IXGBE_AUTOC_LMS_MASK);
|
|
hw->mac.link_settings_loaded = TRUE;
|
|
}
|
|
|
|
/* Store the permanent mac address */
|
|
hw->mac.ops.get_mac_addr(hw, hw->mac.perm_addr);
|
|
|
|
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
|
|
**/
|
|
s32 ixgbe_set_vmdq_82598(struct ixgbe_hw *hw, u32 rar, u32 vmdq)
|
|
{
|
|
u32 rar_high;
|
|
|
|
rar_high = IXGBE_READ_REG(hw, IXGBE_RAH(rar));
|
|
rar_high &= ~IXGBE_RAH_VIND_MASK;
|
|
rar_high |= ((vmdq << IXGBE_RAH_VIND_SHIFT) & IXGBE_RAH_VIND_MASK);
|
|
IXGBE_WRITE_REG(hw, IXGBE_RAH(rar), rar_high);
|
|
return IXGBE_SUCCESS;
|
|
}
|
|
|
|
/**
|
|
* ixgbe_blink_led_start_82598 - Blink LED based on index.
|
|
* @hw: pointer to hardware structure
|
|
* @index: led number to blink
|
|
**/
|
|
s32 ixgbe_blink_led_start_82598(struct ixgbe_hw *hw, u32 index)
|
|
{
|
|
ixgbe_link_speed speed = 0;
|
|
bool link_up = 0;
|
|
u32 autoc_reg = IXGBE_READ_REG(hw, IXGBE_AUTOC);
|
|
u32 led_reg = IXGBE_READ_REG(hw, IXGBE_LEDCTL);
|
|
|
|
/*
|
|
* Link must be up to auto-blink the LEDs on the 82598EB MAC;
|
|
* force it if link is down.
|
|
*/
|
|
hw->mac.ops.check_link(hw, &speed, &link_up, FALSE);
|
|
|
|
if (!link_up) {
|
|
autoc_reg |= IXGBE_AUTOC_FLU;
|
|
IXGBE_WRITE_REG(hw, IXGBE_AUTOC, autoc_reg);
|
|
msec_delay(10);
|
|
}
|
|
|
|
led_reg &= ~IXGBE_LED_MODE_MASK(index);
|
|
led_reg |= IXGBE_LED_BLINK(index);
|
|
IXGBE_WRITE_REG(hw, IXGBE_LEDCTL, led_reg);
|
|
IXGBE_WRITE_FLUSH(hw);
|
|
|
|
return IXGBE_SUCCESS;
|
|
}
|
|
|
|
/**
|
|
* ixgbe_blink_led_stop_82598 - Stop blinking LED based on index.
|
|
* @hw: pointer to hardware structure
|
|
* @index: led number to stop blinking
|
|
**/
|
|
s32 ixgbe_blink_led_stop_82598(struct ixgbe_hw *hw, u32 index)
|
|
{
|
|
u32 autoc_reg = IXGBE_READ_REG(hw, IXGBE_AUTOC);
|
|
u32 led_reg = IXGBE_READ_REG(hw, IXGBE_LEDCTL);
|
|
|
|
autoc_reg &= ~IXGBE_AUTOC_FLU;
|
|
IXGBE_WRITE_REG(hw, IXGBE_AUTOC, autoc_reg);
|
|
|
|
led_reg &= ~IXGBE_LED_MODE_MASK(index);
|
|
led_reg &= ~IXGBE_LED_BLINK(index);
|
|
IXGBE_WRITE_REG(hw, IXGBE_LEDCTL, led_reg);
|
|
IXGBE_WRITE_FLUSH(hw);
|
|
|
|
return IXGBE_SUCCESS;
|
|
}
|