Quellcodebibliothek Statistik Leitseite products/Sources/formale Sprachen/C/Linux/drivers/net/ethernet/intel/ixgbe/   (Open Source Betriebssystem Version 6.17.9©)  Datei vom 24.10.2025 mit Größe 119 kB image not shown  

Quelle  ixgbe_common.c   Sprache: C

 
// SPDX-License-Identifier: GPL-2.0
/* Copyright(c) 1999 - 2024 Intel Corporation. */

#include <linux/pci.h>
#include <linux/delay.h>
#include <linux/sched.h>
#include <linux/netdevice.h>

#include "ixgbe.h"
#include "ixgbe_common.h"
#include "ixgbe_phy.h"

static int ixgbe_acquire_eeprom(struct ixgbe_hw *hw);
static int ixgbe_get_eeprom_semaphore(struct ixgbe_hw *hw);
static void ixgbe_release_eeprom_semaphore(struct ixgbe_hw *hw);
static int ixgbe_ready_eeprom(struct ixgbe_hw *hw);
static void ixgbe_standby_eeprom(struct ixgbe_hw *hw);
static void ixgbe_shift_out_eeprom_bits(struct ixgbe_hw *hw, u16 data,
     u16 count);
static u16 ixgbe_shift_in_eeprom_bits(struct ixgbe_hw *hw, u16 count);
static void ixgbe_raise_eeprom_clk(struct ixgbe_hw *hw, u32 *eec);
static void ixgbe_lower_eeprom_clk(struct ixgbe_hw *hw, u32 *eec);
static void ixgbe_release_eeprom(struct ixgbe_hw *hw);

static int ixgbe_mta_vector(struct ixgbe_hw *hw, u8 *mc_addr);
static int ixgbe_poll_eerd_eewr_done(struct ixgbe_hw *hw, u32 ee_reg);
static int ixgbe_read_eeprom_buffer_bit_bang(struct ixgbe_hw *hw, u16 offset,
          u16 words, u16 *data);
static int ixgbe_write_eeprom_buffer_bit_bang(struct ixgbe_hw *hw, u16 offset,
           u16 words, u16 *data);
static int ixgbe_detect_eeprom_page_size_generic(struct ixgbe_hw *hw,
       u16 offset);
static int ixgbe_disable_pcie_primary(struct ixgbe_hw *hw);

/* Base table for registers values that change by MAC */
const u32 ixgbe_mvals_8259X[IXGBE_MVALS_IDX_LIMIT] = {
 IXGBE_MVALS_INIT(8259X)
};

/**
 *  ixgbe_device_supports_autoneg_fc - Check if phy supports autoneg flow
 *  control
 *  @hw: pointer to hardware structure
 *
 *  There are several phys that do not support autoneg flow control. This
 *  function check the device id to see if the associated phy supports
 *  autoneg flow control.
 **/

bool ixgbe_device_supports_autoneg_fc(struct ixgbe_hw *hw)
{
 bool supported = false;
 ixgbe_link_speed speed;
 bool link_up;

 switch (hw->phy.media_type) {
 case ixgbe_media_type_fiber:
  /* flow control autoneg black list */
  switch (hw->device_id) {
  case IXGBE_DEV_ID_X550EM_A_SFP:
  case IXGBE_DEV_ID_X550EM_A_SFP_N:
  case IXGBE_DEV_ID_E610_SFP:
   supported = false;
   break;
  default:
   hw->mac.ops.check_link(hw, &speed, &link_up, false);
   /* if link is down, assume supported */
   if (link_up)
    supported = speed == IXGBE_LINK_SPEED_1GB_FULL;
   else
    supported = true;
  }

  break;
 case ixgbe_media_type_backplane:
  if (hw->device_id == IXGBE_DEV_ID_X550EM_X_XFI)
   supported = false;
  else
   supported = true;
  break;
 case ixgbe_media_type_copper:
  /* only some copper devices support flow control autoneg */
  switch (hw->device_id) {
  case IXGBE_DEV_ID_82599_T3_LOM:
  case IXGBE_DEV_ID_X540T:
  case IXGBE_DEV_ID_X540T1:
  case IXGBE_DEV_ID_X550T:
  case IXGBE_DEV_ID_X550T1:
  case IXGBE_DEV_ID_X550EM_X_10G_T:
  case IXGBE_DEV_ID_X550EM_A_10G_T:
  case IXGBE_DEV_ID_X550EM_A_1G_T:
  case IXGBE_DEV_ID_X550EM_A_1G_T_L:
  case IXGBE_DEV_ID_E610_10G_T:
  case IXGBE_DEV_ID_E610_2_5G_T:
   supported = true;
   break;
  default:
   break;
  }
  break;
 default:
  break;
 }

 if (!supported)
  hw_dbg(hw, "Device %x does not support flow control autoneg\n",
         hw->device_id);

 return supported;
}

/**
 *  ixgbe_setup_fc_generic - Set up flow control
 *  @hw: pointer to hardware structure
 *
 *  Called at init time to set up flow control.
 **/

int ixgbe_setup_fc_generic(struct ixgbe_hw *hw)
{
 u32 reg = 0, reg_bp = 0;
 bool locked = false;
 int ret_val = 0;
 u16 reg_cu = 0;

 /*
 * Validate the requested mode.  Strict IEEE mode does not allow
 * ixgbe_fc_rx_pause because it will cause us to fail at UNH.
 */

 if (hw->fc.strict_ieee && hw->fc.requested_mode == ixgbe_fc_rx_pause) {
  hw_dbg(hw, "ixgbe_fc_rx_pause not valid in strict IEEE mode\n");
  return -EINVAL;
 }

 /*
 * 10gig 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.requested_mode == ixgbe_fc_default)
  hw->fc.requested_mode = ixgbe_fc_full;

 /*
 * Set up the 1G and 10G flow control advertisement registers so the
 * HW will be able to do fc autoneg once the cable is plugged in.  If
 * we link at 10G, the 1G advertisement is harmless and vice versa.
 */

 switch (hw->phy.media_type) {
 case ixgbe_media_type_backplane:
  /* some MAC's need RMW protection on AUTOC */
  ret_val = hw->mac.ops.prot_autoc_read(hw, &locked, ®_bp);
  if (ret_val)
   return ret_val;

  fallthrough; /* only backplane uses autoc */
 case ixgbe_media_type_fiber:
  reg = IXGBE_READ_REG(hw, IXGBE_PCS1GANA);

  break;
 case ixgbe_media_type_copper:
  hw->phy.ops.read_reg(hw, MDIO_AN_ADVERTISE,
     MDIO_MMD_AN, ®_cu);
  break;
 default:
  break;
 }

 /*
 * The possible values of fc.requested_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.
 */

 switch (hw->fc.requested_mode) {
 case ixgbe_fc_none:
  /* Flow control completely disabled by software override. */
  reg &= ~(IXGBE_PCS1GANA_SYM_PAUSE | IXGBE_PCS1GANA_ASM_PAUSE);
  if (hw->phy.media_type == ixgbe_media_type_backplane)
   reg_bp &= ~(IXGBE_AUTOC_SYM_PAUSE |
        IXGBE_AUTOC_ASM_PAUSE);
  else if (hw->phy.media_type == ixgbe_media_type_copper)
   reg_cu &= ~(IXGBE_TAF_SYM_PAUSE | IXGBE_TAF_ASM_PAUSE);
  break;
 case ixgbe_fc_tx_pause:
  /*
 * Tx Flow control is enabled, and Rx Flow control is
 * disabled by software override.
 */

  reg |= IXGBE_PCS1GANA_ASM_PAUSE;
  reg &= ~IXGBE_PCS1GANA_SYM_PAUSE;
  if (hw->phy.media_type == ixgbe_media_type_backplane) {
   reg_bp |= IXGBE_AUTOC_ASM_PAUSE;
   reg_bp &= ~IXGBE_AUTOC_SYM_PAUSE;
  } else if (hw->phy.media_type == ixgbe_media_type_copper) {
   reg_cu |= IXGBE_TAF_ASM_PAUSE;
   reg_cu &= ~IXGBE_TAF_SYM_PAUSE;
  }
  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, as such we fall
 * through to the fc_full statement.  Later, we will
 * disable the adapter's ability to send PAUSE frames.
 */

 case ixgbe_fc_full:
  /* Flow control (both Rx and Tx) is enabled by SW override. */
  reg |= IXGBE_PCS1GANA_SYM_PAUSE | IXGBE_PCS1GANA_ASM_PAUSE;
  if (hw->phy.media_type == ixgbe_media_type_backplane)
   reg_bp |= IXGBE_AUTOC_SYM_PAUSE |
      IXGBE_AUTOC_ASM_PAUSE;
  else if (hw->phy.media_type == ixgbe_media_type_copper)
   reg_cu |= IXGBE_TAF_SYM_PAUSE | IXGBE_TAF_ASM_PAUSE;
  break;
 default:
  hw_dbg(hw, "Flow control param set incorrectly\n");
  return -EIO;
 }

 if (hw->mac.type != ixgbe_mac_X540) {
  /*
 * Enable auto-negotiation between the MAC & PHY;
 * the MAC will advertise clause 37 flow control.
 */

  IXGBE_WRITE_REG(hw, IXGBE_PCS1GANA, reg);
  reg = IXGBE_READ_REG(hw, IXGBE_PCS1GLCTL);

  /* Disable AN timeout */
  if (hw->fc.strict_ieee)
   reg &= ~IXGBE_PCS1GLCTL_AN_1G_TIMEOUT_EN;

  IXGBE_WRITE_REG(hw, IXGBE_PCS1GLCTL, reg);
  hw_dbg(hw, "Set up FC; PCS1GLCTL = 0x%08X\n", reg);
 }

 /*
 * AUTOC restart handles negotiation of 1G and 10G on backplane
 * and copper. There is no need to set the PCS1GCTL register.
 *
 */

 if (hw->phy.media_type == ixgbe_media_type_backplane) {
  /* Need the SW/FW semaphore around AUTOC writes if 82599 and
 * LESM is on, likewise reset_pipeline requires the lock as
 * it also writes AUTOC.
 */

  ret_val = hw->mac.ops.prot_autoc_write(hw, reg_bp, locked);
  if (ret_val)
   return ret_val;

 } else if ((hw->phy.media_type == ixgbe_media_type_copper) &&
     ixgbe_device_supports_autoneg_fc(hw)) {
  hw->phy.ops.write_reg(hw, MDIO_AN_ADVERTISE,
          MDIO_MMD_AN, reg_cu);
 }

 hw_dbg(hw, "Set up FC; IXGBE_AUTOC = 0x%08X\n", reg);
 return ret_val;
}

/**
 *  ixgbe_start_hw_generic - Prepare hardware for Tx/Rx
 *  @hw: pointer to hardware structure
 *
 *  Starts the hardware by filling the bus info structure and media type, clears
 *  all on chip counters, initializes receive address registers, multicast
 *  table, VLAN filter table, calls routine to set up link and flow control
 *  settings, and leaves transmit and receive units disabled and uninitialized
 **/

int ixgbe_start_hw_generic(struct ixgbe_hw *hw)
{
 u16 device_caps;
 u32 ctrl_ext;
 int ret_val;

 /* Set the media type */
 hw->phy.media_type = hw->mac.ops.get_media_type(hw);

 /* Identify the PHY */
 hw->phy.ops.identify(hw);

 /* Clear the VLAN filter table */
 hw->mac.ops.clear_vfta(hw);

 /* Clear statistics registers */
 hw->mac.ops.clear_hw_cntrs(hw);

 /* Set No Snoop Disable */
 ctrl_ext = IXGBE_READ_REG(hw, IXGBE_CTRL_EXT);
 ctrl_ext |= IXGBE_CTRL_EXT_NS_DIS;
 IXGBE_WRITE_REG(hw, IXGBE_CTRL_EXT, ctrl_ext);
 IXGBE_WRITE_FLUSH(hw);

 /* Setup flow control if method for doing so */
 if (hw->mac.ops.setup_fc) {
  ret_val = hw->mac.ops.setup_fc(hw);
  if (ret_val)
   return ret_val;
 }

 /* Cache bit indicating need for crosstalk fix */
 switch (hw->mac.type) {
 case ixgbe_mac_82599EB:
 case ixgbe_mac_X550EM_x:
 case ixgbe_mac_x550em_a:
  hw->mac.ops.get_device_caps(hw, &device_caps);
  if (device_caps & IXGBE_DEVICE_CAPS_NO_CROSSTALK_WR)
   hw->need_crosstalk_fix = false;
  else
   hw->need_crosstalk_fix = true;
  break;
 default:
  hw->need_crosstalk_fix = false;
  break;
 }

 /* Clear adapter stopped flag */
 hw->adapter_stopped = false;

 return 0;
}

/**
 *  ixgbe_start_hw_gen2 - Init sequence for common device family
 *  @hw: pointer to hw structure
 *
 * Performs the init sequence common to the second generation
 * of 10 GbE devices.
 * Devices in the second generation:
 *     82599
 *     X540
 *     E610
 **/

int ixgbe_start_hw_gen2(struct ixgbe_hw *hw)
{
 u32 i;

 /* Clear the rate limiters */
 for (i = 0; i < hw->mac.max_tx_queues; i++) {
  IXGBE_WRITE_REG(hw, IXGBE_RTTDQSEL, i);
  IXGBE_WRITE_REG(hw, IXGBE_RTTBCNRC, 0);
 }
 IXGBE_WRITE_FLUSH(hw);

 return 0;
}

/**
 *  ixgbe_init_hw_generic - Generic hardware initialization
 *  @hw: pointer to hardware structure
 *
 *  Initialize the hardware by resetting the hardware, filling the bus info
 *  structure and media type, clears all on chip counters, initializes receive
 *  address registers, multicast table, VLAN filter table, calls routine to set
 *  up link and flow control settings, and leaves transmit and receive units
 *  disabled and uninitialized
 **/

int ixgbe_init_hw_generic(struct ixgbe_hw *hw)
{
 int status;

 /* Reset the hardware */
 status = hw->mac.ops.reset_hw(hw);

 if (status == 0) {
  /* Start the HW */
  status = hw->mac.ops.start_hw(hw);
 }

 /* Initialize the LED link active for LED blink support */
 if (hw->mac.ops.init_led_link_act)
  hw->mac.ops.init_led_link_act(hw);

 return status;
}

/**
 *  ixgbe_clear_hw_cntrs_generic - Generic clear hardware counters
 *  @hw: pointer to hardware structure
 *
 *  Clears all hardware statistics counters by reading them from the hardware
 *  Statistics counters are clear on read.
 **/

int ixgbe_clear_hw_cntrs_generic(struct ixgbe_hw *hw)
{
 u16 i = 0;

 IXGBE_READ_REG(hw, IXGBE_CRCERRS);
 IXGBE_READ_REG(hw, IXGBE_ILLERRC);
 IXGBE_READ_REG(hw, IXGBE_ERRBC);
 IXGBE_READ_REG(hw, IXGBE_MSPDC);
 for (i = 0; i < 8; i++)
  IXGBE_READ_REG(hw, IXGBE_MPC(i));

 IXGBE_READ_REG(hw, IXGBE_MLFC);
 IXGBE_READ_REG(hw, IXGBE_MRFC);
 IXGBE_READ_REG(hw, IXGBE_RLEC);
 IXGBE_READ_REG(hw, IXGBE_LXONTXC);
 IXGBE_READ_REG(hw, IXGBE_LXOFFTXC);
 if (hw->mac.type >= ixgbe_mac_82599EB) {
  IXGBE_READ_REG(hw, IXGBE_LXONRXCNT);
  IXGBE_READ_REG(hw, IXGBE_LXOFFRXCNT);
 } else {
  IXGBE_READ_REG(hw, IXGBE_LXONRXC);
  IXGBE_READ_REG(hw, IXGBE_LXOFFRXC);
 }

 for (i = 0; i < 8; i++) {
  IXGBE_READ_REG(hw, IXGBE_PXONTXC(i));
  IXGBE_READ_REG(hw, IXGBE_PXOFFTXC(i));
  if (hw->mac.type >= ixgbe_mac_82599EB) {
   IXGBE_READ_REG(hw, IXGBE_PXONRXCNT(i));
   IXGBE_READ_REG(hw, IXGBE_PXOFFRXCNT(i));
  } else {
   IXGBE_READ_REG(hw, IXGBE_PXONRXC(i));
   IXGBE_READ_REG(hw, IXGBE_PXOFFRXC(i));
  }
 }
 if (hw->mac.type >= ixgbe_mac_82599EB)
  for (i = 0; i < 8; i++)
   IXGBE_READ_REG(hw, IXGBE_PXON2OFFCNT(i));
 IXGBE_READ_REG(hw, IXGBE_PRC64);
 IXGBE_READ_REG(hw, IXGBE_PRC127);
 IXGBE_READ_REG(hw, IXGBE_PRC255);
 IXGBE_READ_REG(hw, IXGBE_PRC511);
 IXGBE_READ_REG(hw, IXGBE_PRC1023);
 IXGBE_READ_REG(hw, IXGBE_PRC1522);
 IXGBE_READ_REG(hw, IXGBE_GPRC);
 IXGBE_READ_REG(hw, IXGBE_BPRC);
 IXGBE_READ_REG(hw, IXGBE_MPRC);
 IXGBE_READ_REG(hw, IXGBE_GPTC);
 IXGBE_READ_REG(hw, IXGBE_GORCL);
 IXGBE_READ_REG(hw, IXGBE_GORCH);
 IXGBE_READ_REG(hw, IXGBE_GOTCL);
 IXGBE_READ_REG(hw, IXGBE_GOTCH);
 if (hw->mac.type == ixgbe_mac_82598EB)
  for (i = 0; i < 8; i++)
   IXGBE_READ_REG(hw, IXGBE_RNBC(i));
 IXGBE_READ_REG(hw, IXGBE_RUC);
 IXGBE_READ_REG(hw, IXGBE_RFC);
 IXGBE_READ_REG(hw, IXGBE_ROC);
 IXGBE_READ_REG(hw, IXGBE_RJC);
 IXGBE_READ_REG(hw, IXGBE_MNGPRC);
 IXGBE_READ_REG(hw, IXGBE_MNGPDC);
 IXGBE_READ_REG(hw, IXGBE_MNGPTC);
 IXGBE_READ_REG(hw, IXGBE_TORL);
 IXGBE_READ_REG(hw, IXGBE_TORH);
 IXGBE_READ_REG(hw, IXGBE_TPR);
 IXGBE_READ_REG(hw, IXGBE_TPT);
 IXGBE_READ_REG(hw, IXGBE_PTC64);
 IXGBE_READ_REG(hw, IXGBE_PTC127);
 IXGBE_READ_REG(hw, IXGBE_PTC255);
 IXGBE_READ_REG(hw, IXGBE_PTC511);
 IXGBE_READ_REG(hw, IXGBE_PTC1023);
 IXGBE_READ_REG(hw, IXGBE_PTC1522);
 IXGBE_READ_REG(hw, IXGBE_MPTC);
 IXGBE_READ_REG(hw, IXGBE_BPTC);
 for (i = 0; i < 16; i++) {
  IXGBE_READ_REG(hw, IXGBE_QPRC(i));
  IXGBE_READ_REG(hw, IXGBE_QPTC(i));
  if (hw->mac.type >= ixgbe_mac_82599EB) {
   IXGBE_READ_REG(hw, IXGBE_QBRC_L(i));
   IXGBE_READ_REG(hw, IXGBE_QBRC_H(i));
   IXGBE_READ_REG(hw, IXGBE_QBTC_L(i));
   IXGBE_READ_REG(hw, IXGBE_QBTC_H(i));
   IXGBE_READ_REG(hw, IXGBE_QPRDC(i));
  } else {
   IXGBE_READ_REG(hw, IXGBE_QBRC(i));
   IXGBE_READ_REG(hw, IXGBE_QBTC(i));
  }
 }

 if (hw->mac.type == ixgbe_mac_X550 ||
     hw->mac.type == ixgbe_mac_X540 ||
     hw->mac.type == ixgbe_mac_e610) {
  if (hw->phy.id == 0)
   hw->phy.ops.identify(hw);
 }

 if (hw->mac.type == ixgbe_mac_X550 || hw->mac.type == ixgbe_mac_X540) {
  hw->phy.ops.read_reg(hw, IXGBE_PCRC8ECL, MDIO_MMD_PCS, &i);
  hw->phy.ops.read_reg(hw, IXGBE_PCRC8ECH, MDIO_MMD_PCS, &i);
  hw->phy.ops.read_reg(hw, IXGBE_LDPCECL, MDIO_MMD_PCS, &i);
  hw->phy.ops.read_reg(hw, IXGBE_LDPCECH, MDIO_MMD_PCS, &i);
 }

 return 0;
}

/**
 *  ixgbe_read_pba_string_generic - Reads part number string from EEPROM
 *  @hw: pointer to hardware structure
 *  @pba_num: stores the part number string from the EEPROM
 *  @pba_num_size: part number string buffer length
 *
 *  Reads the part number string from the EEPROM.
 **/

int ixgbe_read_pba_string_generic(struct ixgbe_hw *hw, u8 *pba_num,
      u32 pba_num_size)
{
 int ret_val;
 u16 pba_ptr;
 u16 offset;
 u16 length;
 u16 data;

 if (pba_num == NULL) {
  hw_dbg(hw, "PBA string buffer was null\n");
  return -EINVAL;
 }

 ret_val = hw->eeprom.ops.read(hw, IXGBE_PBANUM0_PTR, &data);
 if (ret_val) {
  hw_dbg(hw, "NVM Read Error\n");
  return ret_val;
 }

 ret_val = hw->eeprom.ops.read(hw, IXGBE_PBANUM1_PTR, &pba_ptr);
 if (ret_val) {
  hw_dbg(hw, "NVM Read Error\n");
  return ret_val;
 }

 /*
 * if data is not ptr guard the PBA must be in legacy format which
 * means pba_ptr is actually our second data word for the PBA number
 * and we can decode it into an ascii string
 */

 if (data != IXGBE_PBANUM_PTR_GUARD) {
  hw_dbg(hw, "NVM PBA number is not stored as string\n");

  /* we will need 11 characters to store the PBA */
  if (pba_num_size < 11) {
   hw_dbg(hw, "PBA string buffer too small\n");
   return -ENOSPC;
  }

  /* extract hex string from data and pba_ptr */
  pba_num[0] = (data >> 12) & 0xF;
  pba_num[1] = (data >> 8) & 0xF;
  pba_num[2] = (data >> 4) & 0xF;
  pba_num[3] = data & 0xF;
  pba_num[4] = (pba_ptr >> 12) & 0xF;
  pba_num[5] = (pba_ptr >> 8) & 0xF;
  pba_num[6] = '-';
  pba_num[7] = 0;
  pba_num[8] = (pba_ptr >> 4) & 0xF;
  pba_num[9] = pba_ptr & 0xF;

  /* put a null character on the end of our string */
  pba_num[10] = '\0';

  /* switch all the data but the '-' to hex char */
  for (offset = 0; offset < 10; offset++) {
   if (pba_num[offset] < 0xA)
    pba_num[offset] += '0';
   else if (pba_num[offset] < 0x10)
    pba_num[offset] += 'A' - 0xA;
  }

  return 0;
 }

 ret_val = hw->eeprom.ops.read(hw, pba_ptr, &length);
 if (ret_val) {
  hw_dbg(hw, "NVM Read Error\n");
  return ret_val;
 }

 if (length == 0xFFFF || length == 0) {
  hw_dbg(hw, "NVM PBA number section invalid length\n");
  return -EIO;
 }

 /* check if pba_num buffer is big enough */
 if (pba_num_size  < (((u32)length * 2) - 1)) {
  hw_dbg(hw, "PBA string buffer too small\n");
  return -ENOSPC;
 }

 /* trim pba length from start of string */
 pba_ptr++;
 length--;

 for (offset = 0; offset < length; offset++) {
  ret_val = hw->eeprom.ops.read(hw, pba_ptr + offset, &data);
  if (ret_val) {
   hw_dbg(hw, "NVM Read Error\n");
   return ret_val;
  }
  pba_num[offset * 2] = (u8)(data >> 8);
  pba_num[(offset * 2) + 1] = (u8)(data & 0xFF);
 }
 pba_num[offset * 2] = '\0';

 return 0;
}

/**
 *  ixgbe_get_mac_addr_generic - Generic get MAC address
 *  @hw: pointer to hardware structure
 *  @mac_addr: Adapter MAC address
 *
 *  Reads the adapter's MAC address from first Receive Address Register (RAR0)
 *  A reset of the adapter must be performed prior to calling this function
 *  in order for the MAC address to have been loaded from the EEPROM into RAR0
 **/

int ixgbe_get_mac_addr_generic(struct ixgbe_hw *hw, u8 *mac_addr)
{
 u32 rar_high;
 u32 rar_low;
 u16 i;

 rar_high = IXGBE_READ_REG(hw, IXGBE_RAH(0));
 rar_low = IXGBE_READ_REG(hw, IXGBE_RAL(0));

 for (i = 0; i < 4; i++)
  mac_addr[i] = (u8)(rar_low >> (i*8));

 for (i = 0; i < 2; i++)
  mac_addr[i+4] = (u8)(rar_high >> (i*8));

 return 0;
}

enum ixgbe_bus_width ixgbe_convert_bus_width(u16 link_status)
{
 switch (link_status & IXGBE_PCI_LINK_WIDTH) {
 case IXGBE_PCI_LINK_WIDTH_1:
  return ixgbe_bus_width_pcie_x1;
 case IXGBE_PCI_LINK_WIDTH_2:
  return ixgbe_bus_width_pcie_x2;
 case IXGBE_PCI_LINK_WIDTH_4:
  return ixgbe_bus_width_pcie_x4;
 case IXGBE_PCI_LINK_WIDTH_8:
  return ixgbe_bus_width_pcie_x8;
 default:
  return ixgbe_bus_width_unknown;
 }
}

enum ixgbe_bus_speed ixgbe_convert_bus_speed(u16 link_status)
{
 switch (link_status & IXGBE_PCI_LINK_SPEED) {
 case IXGBE_PCI_LINK_SPEED_2500:
  return ixgbe_bus_speed_2500;
 case IXGBE_PCI_LINK_SPEED_5000:
  return ixgbe_bus_speed_5000;
 case IXGBE_PCI_LINK_SPEED_8000:
  return ixgbe_bus_speed_8000;
 default:
  return ixgbe_bus_speed_unknown;
 }
}

/**
 *  ixgbe_get_bus_info_generic - Generic set PCI bus info
 *  @hw: pointer to hardware structure
 *
 *  Sets the PCI bus info (speed, width, type) within the ixgbe_hw structure
 **/

int ixgbe_get_bus_info_generic(struct ixgbe_hw *hw)
{
 u16 link_status;

 hw->bus.type = ixgbe_bus_type_pci_express;

 /* Get the negotiated link width and speed from PCI config space */
 if (hw->mac.type == ixgbe_mac_e610)
  link_status = ixgbe_read_pci_cfg_word(hw, IXGBE_PCI_LINK_STATUS_E610);
 else
  link_status = ixgbe_read_pci_cfg_word(hw,
            IXGBE_PCI_LINK_STATUS);

 hw->bus.width = ixgbe_convert_bus_width(link_status);
 hw->bus.speed = ixgbe_convert_bus_speed(link_status);

 hw->mac.ops.set_lan_id(hw);

 return 0;
}

/**
 *  ixgbe_set_lan_id_multi_port_pcie - Set LAN id for PCIe multiple port devices
 *  @hw: pointer to the HW structure
 *
 *  Determines the LAN function id by reading memory-mapped registers
 *  and swaps the port value if requested.
 **/

void ixgbe_set_lan_id_multi_port_pcie(struct ixgbe_hw *hw)
{
 struct ixgbe_bus_info *bus = &hw->bus;
 u16 ee_ctrl_4;
 u32 reg;

 reg = IXGBE_READ_REG(hw, IXGBE_STATUS);
 bus->func = FIELD_GET(IXGBE_STATUS_LAN_ID, reg);
 bus->lan_id = bus->func;

 /* check for a port swap */
 reg = IXGBE_READ_REG(hw, IXGBE_FACTPS(hw));
 if (reg & IXGBE_FACTPS_LFS)
  bus->func ^= 0x1;

 /* Get MAC instance from EEPROM for configuring CS4227 */
 if (hw->device_id == IXGBE_DEV_ID_X550EM_A_SFP) {
  hw->eeprom.ops.read(hw, IXGBE_EEPROM_CTRL_4, &ee_ctrl_4);
  bus->instance_id = FIELD_GET(IXGBE_EE_CTRL_4_INST_ID,
          ee_ctrl_4);
 }
}

/**
 *  ixgbe_stop_adapter_generic - Generic stop Tx/Rx units
 *  @hw: pointer to hardware structure
 *
 *  Sets the adapter_stopped flag within ixgbe_hw struct. Clears interrupts,
 *  disables transmit and receive units. The adapter_stopped flag is used by
 *  the shared code and drivers to determine if the adapter is in a stopped
 *  state and should not touch the hardware.
 **/

int ixgbe_stop_adapter_generic(struct ixgbe_hw *hw)
{
 u32 reg_val;
 u16 i;

 /*
 * Set the adapter_stopped flag so other driver functions stop touching
 * the hardware
 */

 hw->adapter_stopped = true;

 /* Disable the receive unit */
 hw->mac.ops.disable_rx(hw);

 /* Clear interrupt mask to stop interrupts from being generated */
 IXGBE_WRITE_REG(hw, IXGBE_EIMC, IXGBE_IRQ_CLEAR_MASK);

 /* Clear any pending interrupts, flush previous writes */
 IXGBE_READ_REG(hw, IXGBE_EICR);

 /* Disable the transmit unit.  Each queue must be disabled. */
 for (i = 0; i < hw->mac.max_tx_queues; i++)
  IXGBE_WRITE_REG(hw, IXGBE_TXDCTL(i), IXGBE_TXDCTL_SWFLSH);

 /* Disable the receive unit by stopping each queue */
 for (i = 0; i < hw->mac.max_rx_queues; i++) {
  reg_val = IXGBE_READ_REG(hw, IXGBE_RXDCTL(i));
  reg_val &= ~IXGBE_RXDCTL_ENABLE;
  reg_val |= IXGBE_RXDCTL_SWFLSH;
  IXGBE_WRITE_REG(hw, IXGBE_RXDCTL(i), reg_val);
 }

 /* flush all queues disables */
 IXGBE_WRITE_FLUSH(hw);
 usleep_range(1000, 2000);

 /*
 * Prevent the PCI-E bus from hanging by disabling PCI-E primary
 * access and verify no pending requests
 */

 return ixgbe_disable_pcie_primary(hw);
}

/**
 *  ixgbe_init_led_link_act_generic - Store the LED index link/activity.
 *  @hw: pointer to hardware structure
 *
 *  Store the index for the link active LED. This will be used to support
 *  blinking the LED.
 **/

int ixgbe_init_led_link_act_generic(struct ixgbe_hw *hw)
{
 struct ixgbe_mac_info *mac = &hw->mac;
 u32 led_reg, led_mode;
 u16 i;

 led_reg = IXGBE_READ_REG(hw, IXGBE_LEDCTL);

 /* Get LED link active from the LEDCTL register */
 for (i = 0; i < 4; i++) {
  led_mode = led_reg >> IXGBE_LED_MODE_SHIFT(i);

  if ((led_mode & IXGBE_LED_MODE_MASK_BASE) ==
      IXGBE_LED_LINK_ACTIVE) {
   mac->led_link_act = i;
   return 0;
  }
 }

 /* If LEDCTL register does not have the LED link active set, then use
 * known MAC defaults.
 */

 switch (hw->mac.type) {
 case ixgbe_mac_x550em_a:
  mac->led_link_act = 0;
  break;
 case ixgbe_mac_X550EM_x:
  mac->led_link_act = 1;
  break;
 default:
  mac->led_link_act = 2;
 }

 return 0;
}

/**
 *  ixgbe_led_on_generic - Turns on the software controllable LEDs.
 *  @hw: pointer to hardware structure
 *  @index: led number to turn on
 **/

int ixgbe_led_on_generic(struct ixgbe_hw *hw, u32 index)
{
 u32 led_reg = IXGBE_READ_REG(hw, IXGBE_LEDCTL);

 if (index > 3)
  return -EINVAL;

 /* To turn on the LED, set mode to ON. */
 led_reg &= ~IXGBE_LED_MODE_MASK(index);
 led_reg |= IXGBE_LED_ON << IXGBE_LED_MODE_SHIFT(index);
 IXGBE_WRITE_REG(hw, IXGBE_LEDCTL, led_reg);
 IXGBE_WRITE_FLUSH(hw);

 return 0;
}

/**
 *  ixgbe_led_off_generic - Turns off the software controllable LEDs.
 *  @hw: pointer to hardware structure
 *  @index: led number to turn off
 **/

int ixgbe_led_off_generic(struct ixgbe_hw *hw, u32 index)
{
 u32 led_reg = IXGBE_READ_REG(hw, IXGBE_LEDCTL);

 if (index > 3)
  return -EINVAL;

 /* To turn off the LED, set mode to OFF. */
 led_reg &= ~IXGBE_LED_MODE_MASK(index);
 led_reg |= IXGBE_LED_OFF << IXGBE_LED_MODE_SHIFT(index);
 IXGBE_WRITE_REG(hw, IXGBE_LEDCTL, led_reg);
 IXGBE_WRITE_FLUSH(hw);

 return 0;
}

/**
 *  ixgbe_init_eeprom_params_generic - Initialize EEPROM params
 *  @hw: pointer to hardware structure
 *
 *  Initializes the EEPROM parameters ixgbe_eeprom_info within the
 *  ixgbe_hw struct in order to set up EEPROM access.
 **/

int ixgbe_init_eeprom_params_generic(struct ixgbe_hw *hw)
{
 struct ixgbe_eeprom_info *eeprom = &hw->eeprom;
 u32 eec;
 u16 eeprom_size;

 if (eeprom->type == ixgbe_eeprom_uninitialized) {
  eeprom->type = ixgbe_eeprom_none;
  /* Set default semaphore delay to 10ms which is a well
 * tested value */

  eeprom->semaphore_delay = 10;
  /* Clear EEPROM page size, it will be initialized as needed */
  eeprom->word_page_size = 0;

  /*
 * Check for EEPROM present first.
 * If not present leave as none
 */

  eec = IXGBE_READ_REG(hw, IXGBE_EEC(hw));
  if (eec & IXGBE_EEC_PRES) {
   eeprom->type = ixgbe_eeprom_spi;

   /*
 * SPI EEPROM is assumed here.  This code would need to
 * change if a future EEPROM is not SPI.
 */

   eeprom_size = FIELD_GET(IXGBE_EEC_SIZE, eec);
   eeprom->word_size = BIT(eeprom_size +
      IXGBE_EEPROM_WORD_SIZE_SHIFT);
  }

  if (eec & IXGBE_EEC_ADDR_SIZE)
   eeprom->address_bits = 16;
  else
   eeprom->address_bits = 8;
  hw_dbg(hw, "Eeprom params: type = %d, size = %d, address bits: %d\n",
         eeprom->type, eeprom->word_size, eeprom->address_bits);
 }

 return 0;
}

/**
 *  ixgbe_write_eeprom_buffer_bit_bang_generic - Write EEPROM using bit-bang
 *  @hw: pointer to hardware structure
 *  @offset: offset within the EEPROM to write
 *  @words: number of words
 *  @data: 16 bit word(s) to write to EEPROM
 *
 *  Reads 16 bit word(s) from EEPROM through bit-bang method
 **/

int ixgbe_write_eeprom_buffer_bit_bang_generic(struct ixgbe_hw *hw, u16 offset,
            u16 words, u16 *data)
{
 u16 i, count;
 int status;

 hw->eeprom.ops.init_params(hw);

 if (words == 0 || (offset + words > hw->eeprom.word_size))
  return -EINVAL;

 /*
 * The EEPROM page size cannot be queried from the chip. We do lazy
 * initialization. It is worth to do that when we write large buffer.
 */

 if ((hw->eeprom.word_page_size == 0) &&
     (words > IXGBE_EEPROM_PAGE_SIZE_MAX))
  ixgbe_detect_eeprom_page_size_generic(hw, offset);

 /*
 * We cannot hold synchronization semaphores for too long
 * to avoid other entity starvation. However it is more efficient
 * to read in bursts than synchronizing access for each word.
 */

 for (i = 0; i < words; i += IXGBE_EEPROM_RD_BUFFER_MAX_COUNT) {
  count = (words - i) / IXGBE_EEPROM_RD_BUFFER_MAX_COUNT > 0 ?
    IXGBE_EEPROM_RD_BUFFER_MAX_COUNT : (words - i);
  status = ixgbe_write_eeprom_buffer_bit_bang(hw, offset + i,
           count, &data[i]);

  if (status != 0)
   break;
 }

 return status;
}

/**
 *  ixgbe_write_eeprom_buffer_bit_bang - Writes 16 bit word(s) to EEPROM
 *  @hw: pointer to hardware structure
 *  @offset: offset within the EEPROM to be written to
 *  @words: number of word(s)
 *  @data: 16 bit word(s) to be written to the EEPROM
 *
 *  If ixgbe_eeprom_update_checksum is not called after this function, the
 *  EEPROM will most likely contain an invalid checksum.
 **/

static int ixgbe_write_eeprom_buffer_bit_bang(struct ixgbe_hw *hw, u16 offset,
           u16 words, u16 *data)
{
 u8 write_opcode = IXGBE_EEPROM_WRITE_OPCODE_SPI;
 u16 page_size;
 int status;
 u16 word;
 u16 i;

 /* Prepare the EEPROM for writing  */
 status = ixgbe_acquire_eeprom(hw);
 if (status)
  return status;

 if (ixgbe_ready_eeprom(hw) != 0) {
  ixgbe_release_eeprom(hw);
  return -EIO;
 }

 for (i = 0; i < words; i++) {
  ixgbe_standby_eeprom(hw);

  /* Send the WRITE ENABLE command (8 bit opcode) */
  ixgbe_shift_out_eeprom_bits(hw,
         IXGBE_EEPROM_WREN_OPCODE_SPI,
         IXGBE_EEPROM_OPCODE_BITS);

  ixgbe_standby_eeprom(hw);

  /* Some SPI eeproms use the 8th address bit embedded
 * in the opcode
 */

  if ((hw->eeprom.address_bits == 8) &&
      ((offset + i) >= 128))
   write_opcode |= IXGBE_EEPROM_A8_OPCODE_SPI;

  /* Send the Write command (8-bit opcode + addr) */
  ixgbe_shift_out_eeprom_bits(hw, write_opcode,
         IXGBE_EEPROM_OPCODE_BITS);
  ixgbe_shift_out_eeprom_bits(hw, (u16)((offset + i) * 2),
         hw->eeprom.address_bits);

  page_size = hw->eeprom.word_page_size;

  /* Send the data in burst via SPI */
  do {
   word = data[i];
   word = (word >> 8) | (word << 8);
   ixgbe_shift_out_eeprom_bits(hw, word, 16);

   if (page_size == 0)
    break;

   /* do not wrap around page */
   if (((offset + i) & (page_size - 1)) ==
       (page_size - 1))
    break;
  } while (++i < words);

  ixgbe_standby_eeprom(hw);
  usleep_range(10000, 20000);
 }
 /* Done with writing - release the EEPROM */
 ixgbe_release_eeprom(hw);

 return 0;
}

/**
 *  ixgbe_write_eeprom_generic - Writes 16 bit value to EEPROM
 *  @hw: pointer to hardware structure
 *  @offset: offset within the EEPROM to be written to
 *  @data: 16 bit word to be written to the EEPROM
 *
 *  If ixgbe_eeprom_update_checksum is not called after this function, the
 *  EEPROM will most likely contain an invalid checksum.
 **/

int ixgbe_write_eeprom_generic(struct ixgbe_hw *hw, u16 offset, u16 data)
{
 hw->eeprom.ops.init_params(hw);

 if (offset >= hw->eeprom.word_size)
  return -EINVAL;

 return ixgbe_write_eeprom_buffer_bit_bang(hw, offset, 1, &data);
}

/**
 *  ixgbe_read_eeprom_buffer_bit_bang_generic - Read EEPROM using bit-bang
 *  @hw: pointer to hardware structure
 *  @offset: offset within the EEPROM to be read
 *  @words: number of word(s)
 *  @data: read 16 bit words(s) from EEPROM
 *
 *  Reads 16 bit word(s) from EEPROM through bit-bang method
 **/

int ixgbe_read_eeprom_buffer_bit_bang_generic(struct ixgbe_hw *hw, u16 offset,
           u16 words, u16 *data)
{
 u16 i, count;
 int status;

 hw->eeprom.ops.init_params(hw);

 if (words == 0 || (offset + words > hw->eeprom.word_size))
  return -EINVAL;

 /*
 * We cannot hold synchronization semaphores for too long
 * to avoid other entity starvation. However it is more efficient
 * to read in bursts than synchronizing access for each word.
 */

 for (i = 0; i < words; i += IXGBE_EEPROM_RD_BUFFER_MAX_COUNT) {
  count = (words - i) / IXGBE_EEPROM_RD_BUFFER_MAX_COUNT > 0 ?
    IXGBE_EEPROM_RD_BUFFER_MAX_COUNT : (words - i);

  status = ixgbe_read_eeprom_buffer_bit_bang(hw, offset + i,
          count, &data[i]);

  if (status)
   return status;
 }

 return 0;
}

/**
 *  ixgbe_read_eeprom_buffer_bit_bang - Read EEPROM using bit-bang
 *  @hw: pointer to hardware structure
 *  @offset: offset within the EEPROM to be read
 *  @words: number of word(s)
 *  @data: read 16 bit word(s) from EEPROM
 *
 *  Reads 16 bit word(s) from EEPROM through bit-bang method
 **/

static int ixgbe_read_eeprom_buffer_bit_bang(struct ixgbe_hw *hw, u16 offset,
          u16 words, u16 *data)
{
 u8 read_opcode = IXGBE_EEPROM_READ_OPCODE_SPI;
 u16 word_in;
 int status;
 u16 i;

 /* Prepare the EEPROM for reading  */
 status = ixgbe_acquire_eeprom(hw);
 if (status)
  return status;

 if (ixgbe_ready_eeprom(hw) != 0) {
  ixgbe_release_eeprom(hw);
  return -EIO;
 }

 for (i = 0; i < words; i++) {
  ixgbe_standby_eeprom(hw);
  /* Some SPI eeproms use the 8th address bit embedded
 * in the opcode
 */

  if ((hw->eeprom.address_bits == 8) &&
      ((offset + i) >= 128))
   read_opcode |= IXGBE_EEPROM_A8_OPCODE_SPI;

  /* Send the READ command (opcode + addr) */
  ixgbe_shift_out_eeprom_bits(hw, read_opcode,
         IXGBE_EEPROM_OPCODE_BITS);
  ixgbe_shift_out_eeprom_bits(hw, (u16)((offset + i) * 2),
         hw->eeprom.address_bits);

  /* Read the data. */
  word_in = ixgbe_shift_in_eeprom_bits(hw, 16);
  data[i] = (word_in >> 8) | (word_in << 8);
 }

 /* End this read operation */
 ixgbe_release_eeprom(hw);

 return 0;
}

/**
 *  ixgbe_read_eeprom_bit_bang_generic - Read EEPROM word using bit-bang
 *  @hw: pointer to hardware structure
 *  @offset: offset within the EEPROM to be read
 *  @data: read 16 bit value from EEPROM
 *
 *  Reads 16 bit value from EEPROM through bit-bang method
 **/

int ixgbe_read_eeprom_bit_bang_generic(struct ixgbe_hw *hw, u16 offset,
           u16 *data)
{
 hw->eeprom.ops.init_params(hw);

 if (offset >= hw->eeprom.word_size)
  return -EINVAL;

 return ixgbe_read_eeprom_buffer_bit_bang(hw, offset, 1, data);
}

/**
 *  ixgbe_read_eerd_buffer_generic - Read EEPROM word(s) using EERD
 *  @hw: pointer to hardware structure
 *  @offset: offset of word in the EEPROM to read
 *  @words: number of word(s)
 *  @data: 16 bit word(s) from the EEPROM
 *
 *  Reads a 16 bit word(s) from the EEPROM using the EERD register.
 **/

int ixgbe_read_eerd_buffer_generic(struct ixgbe_hw *hw, u16 offset,
       u16 words, u16 *data)
{
 int status;
 u32 eerd;
 u32 i;

 hw->eeprom.ops.init_params(hw);

 if (words == 0 || offset >= hw->eeprom.word_size)
  return -EINVAL;

 for (i = 0; i < words; i++) {
  eerd = ((offset + i) << IXGBE_EEPROM_RW_ADDR_SHIFT) |
         IXGBE_EEPROM_RW_REG_START;

  IXGBE_WRITE_REG(hw, IXGBE_EERD, eerd);
  status = ixgbe_poll_eerd_eewr_done(hw, IXGBE_NVM_POLL_READ);

  if (status == 0) {
   data[i] = (IXGBE_READ_REG(hw, IXGBE_EERD) >>
       IXGBE_EEPROM_RW_REG_DATA);
  } else {
   hw_dbg(hw, "Eeprom read timed out\n");
   return status;
  }
 }

 return 0;
}

/**
 *  ixgbe_detect_eeprom_page_size_generic - Detect EEPROM page size
 *  @hw: pointer to hardware structure
 *  @offset: offset within the EEPROM to be used as a scratch pad
 *
 *  Discover EEPROM page size by writing marching data at given offset.
 *  This function is called only when we are writing a new large buffer
 *  at given offset so the data would be overwritten anyway.
 **/

static int ixgbe_detect_eeprom_page_size_generic(struct ixgbe_hw *hw,
       u16 offset)
{
 u16 data[IXGBE_EEPROM_PAGE_SIZE_MAX];
 int status;
 u16 i;

 for (i = 0; i < IXGBE_EEPROM_PAGE_SIZE_MAX; i++)
  data[i] = i;

 hw->eeprom.word_page_size = IXGBE_EEPROM_PAGE_SIZE_MAX;
 status = ixgbe_write_eeprom_buffer_bit_bang(hw, offset,
          IXGBE_EEPROM_PAGE_SIZE_MAX, data);
 hw->eeprom.word_page_size = 0;
 if (status)
  return status;

 status = ixgbe_read_eeprom_buffer_bit_bang(hw, offset, 1, data);
 if (status)
  return status;

 /*
 * When writing in burst more than the actual page size
 * EEPROM address wraps around current page.
 */

 hw->eeprom.word_page_size = IXGBE_EEPROM_PAGE_SIZE_MAX - data[0];

 hw_dbg(hw, "Detected EEPROM page size = %d words.\n",
        hw->eeprom.word_page_size);
 return 0;
}

/**
 *  ixgbe_read_eerd_generic - Read EEPROM word using EERD
 *  @hw: pointer to hardware structure
 *  @offset: offset of  word in the EEPROM to read
 *  @data: word read from the EEPROM
 *
 *  Reads a 16 bit word from the EEPROM using the EERD register.
 **/

int ixgbe_read_eerd_generic(struct ixgbe_hw *hw, u16 offset, u16 *data)
{
 return ixgbe_read_eerd_buffer_generic(hw, offset, 1, data);
}

/**
 *  ixgbe_write_eewr_buffer_generic - Write EEPROM word(s) using EEWR
 *  @hw: pointer to hardware structure
 *  @offset: offset of  word in the EEPROM to write
 *  @words: number of words
 *  @data: word(s) write to the EEPROM
 *
 *  Write a 16 bit word(s) to the EEPROM using the EEWR register.
 **/

int ixgbe_write_eewr_buffer_generic(struct ixgbe_hw *hw, u16 offset,
        u16 words, u16 *data)
{
 int status;
 u32 eewr;
 u16 i;

 hw->eeprom.ops.init_params(hw);

 if (words == 0 || offset >= hw->eeprom.word_size)
  return -EINVAL;

 for (i = 0; i < words; i++) {
  eewr = ((offset + i) << IXGBE_EEPROM_RW_ADDR_SHIFT) |
         (data[i] << IXGBE_EEPROM_RW_REG_DATA) |
         IXGBE_EEPROM_RW_REG_START;

  status = ixgbe_poll_eerd_eewr_done(hw, IXGBE_NVM_POLL_WRITE);
  if (status) {
   hw_dbg(hw, "Eeprom write EEWR timed out\n");
   return status;
  }

  IXGBE_WRITE_REG(hw, IXGBE_EEWR, eewr);

  status = ixgbe_poll_eerd_eewr_done(hw, IXGBE_NVM_POLL_WRITE);
  if (status) {
   hw_dbg(hw, "Eeprom write EEWR timed out\n");
   return status;
  }
 }

 return 0;
}

/**
 *  ixgbe_write_eewr_generic - Write EEPROM word using EEWR
 *  @hw: pointer to hardware structure
 *  @offset: offset of  word in the EEPROM to write
 *  @data: word write to the EEPROM
 *
 *  Write a 16 bit word to the EEPROM using the EEWR register.
 **/

int ixgbe_write_eewr_generic(struct ixgbe_hw *hw, u16 offset, u16 data)
{
 return ixgbe_write_eewr_buffer_generic(hw, offset, 1, &data);
}

/**
 *  ixgbe_poll_eerd_eewr_done - Poll EERD read or EEWR write status
 *  @hw: pointer to hardware structure
 *  @ee_reg: EEPROM flag for polling
 *
 *  Polls the status bit (bit 1) of the EERD or EEWR to determine when the
 *  read or write is done respectively.
 **/

static int ixgbe_poll_eerd_eewr_done(struct ixgbe_hw *hw, u32 ee_reg)
{
 u32 i;
 u32 reg;

 for (i = 0; i < IXGBE_EERD_EEWR_ATTEMPTS; i++) {
  if (ee_reg == IXGBE_NVM_POLL_READ)
   reg = IXGBE_READ_REG(hw, IXGBE_EERD);
  else
   reg = IXGBE_READ_REG(hw, IXGBE_EEWR);

  if (reg & IXGBE_EEPROM_RW_REG_DONE) {
   return 0;
  }
  udelay(5);
 }
 return -EIO;
}

/**
 *  ixgbe_acquire_eeprom - Acquire EEPROM using bit-bang
 *  @hw: pointer to hardware structure
 *
 *  Prepares EEPROM for access using bit-bang method. This function should
 *  be called before issuing a command to the EEPROM.
 **/

static int ixgbe_acquire_eeprom(struct ixgbe_hw *hw)
{
 u32 eec;
 u32 i;

 if (hw->mac.ops.acquire_swfw_sync(hw, IXGBE_GSSR_EEP_SM) != 0)
  return -EBUSY;

 eec = IXGBE_READ_REG(hw, IXGBE_EEC(hw));

 /* Request EEPROM Access */
 eec |= IXGBE_EEC_REQ;
 IXGBE_WRITE_REG(hw, IXGBE_EEC(hw), eec);

 for (i = 0; i < IXGBE_EEPROM_GRANT_ATTEMPTS; i++) {
  eec = IXGBE_READ_REG(hw, IXGBE_EEC(hw));
  if (eec & IXGBE_EEC_GNT)
   break;
  udelay(5);
 }

 /* Release if grant not acquired */
 if (!(eec & IXGBE_EEC_GNT)) {
  eec &= ~IXGBE_EEC_REQ;
  IXGBE_WRITE_REG(hw, IXGBE_EEC(hw), eec);
  hw_dbg(hw, "Could not acquire EEPROM grant\n");

  hw->mac.ops.release_swfw_sync(hw, IXGBE_GSSR_EEP_SM);
  return -EIO;
 }

 /* Setup EEPROM for Read/Write */
 /* Clear CS and SK */
 eec &= ~(IXGBE_EEC_CS | IXGBE_EEC_SK);
 IXGBE_WRITE_REG(hw, IXGBE_EEC(hw), eec);
 IXGBE_WRITE_FLUSH(hw);
 udelay(1);
 return 0;
}

/**
 *  ixgbe_get_eeprom_semaphore - Get hardware semaphore
 *  @hw: pointer to hardware structure
 *
 *  Sets the hardware semaphores so EEPROM access can occur for bit-bang method
 **/

static int ixgbe_get_eeprom_semaphore(struct ixgbe_hw *hw)
{
 u32 timeout = 2000;
 u32 i;
 u32 swsm;

 /* Get SMBI software semaphore between device drivers first */
 for (i = 0; i < timeout; i++) {
  /*
 * If the SMBI bit is 0 when we read it, then the bit will be
 * set and we have the semaphore
 */

  swsm = IXGBE_READ_REG(hw, IXGBE_SWSM(hw));
  if (!(swsm & IXGBE_SWSM_SMBI))
   break;
  usleep_range(50, 100);
 }

 if (i == timeout) {
  hw_dbg(hw, "Driver can't access the Eeprom - SMBI Semaphore not granted.\n");
  /* this release is particularly important because our attempts
 * above to get the semaphore may have succeeded, and if there
 * was a timeout, we should unconditionally clear the semaphore
 * bits to free the driver to make progress
 */

  ixgbe_release_eeprom_semaphore(hw);

  usleep_range(50, 100);
  /* one last try
 * If the SMBI bit is 0 when we read it, then the bit will be
 * set and we have the semaphore
 */

  swsm = IXGBE_READ_REG(hw, IXGBE_SWSM(hw));
  if (swsm & IXGBE_SWSM_SMBI) {
   hw_dbg(hw, "Software semaphore SMBI between device drivers not granted.\n");
   return -EIO;
  }
 }

 /* Now get the semaphore between SW/FW through the SWESMBI bit */
 for (i = 0; i < timeout; i++) {
  swsm = IXGBE_READ_REG(hw, IXGBE_SWSM(hw));

  /* Set the SW EEPROM semaphore bit to request access */
  swsm |= IXGBE_SWSM_SWESMBI;
  IXGBE_WRITE_REG(hw, IXGBE_SWSM(hw), swsm);

  /* If we set the bit successfully then we got the
 * semaphore.
 */

  swsm = IXGBE_READ_REG(hw, IXGBE_SWSM(hw));
  if (swsm & IXGBE_SWSM_SWESMBI)
   break;

  usleep_range(50, 100);
 }

 /* Release semaphores and return error if SW EEPROM semaphore
 * was not granted because we don't have access to the EEPROM
 */

 if (i >= timeout) {
  hw_dbg(hw, "SWESMBI Software EEPROM semaphore not granted.\n");
  ixgbe_release_eeprom_semaphore(hw);
  return -EIO;
 }

 return 0;
}

/**
 *  ixgbe_release_eeprom_semaphore - Release hardware semaphore
 *  @hw: pointer to hardware structure
 *
 *  This function clears hardware semaphore bits.
 **/

static void ixgbe_release_eeprom_semaphore(struct ixgbe_hw *hw)
{
 u32 swsm;

 swsm = IXGBE_READ_REG(hw, IXGBE_SWSM(hw));

 /* Release both semaphores by writing 0 to the bits SWESMBI and SMBI */
 swsm &= ~(IXGBE_SWSM_SWESMBI | IXGBE_SWSM_SMBI);
 IXGBE_WRITE_REG(hw, IXGBE_SWSM(hw), swsm);
 IXGBE_WRITE_FLUSH(hw);
}

/**
 *  ixgbe_ready_eeprom - Polls for EEPROM ready
 *  @hw: pointer to hardware structure
 **/

static int ixgbe_ready_eeprom(struct ixgbe_hw *hw)
{
 u16 i;
 u8 spi_stat_reg;

 /*
 * Read "Status Register" repeatedly until the LSB is cleared.  The
 * EEPROM will signal that the command has been completed by clearing
 * bit 0 of the internal status register.  If it's not cleared within
 * 5 milliseconds, then error out.
 */

 for (i = 0; i < IXGBE_EEPROM_MAX_RETRY_SPI; i += 5) {
  ixgbe_shift_out_eeprom_bits(hw, IXGBE_EEPROM_RDSR_OPCODE_SPI,
         IXGBE_EEPROM_OPCODE_BITS);
  spi_stat_reg = (u8)ixgbe_shift_in_eeprom_bits(hw, 8);
  if (!(spi_stat_reg & IXGBE_EEPROM_STATUS_RDY_SPI))
   break;

  udelay(5);
  ixgbe_standby_eeprom(hw);
 }

 /*
 * On some parts, SPI write time could vary from 0-20mSec on 3.3V
 * devices (and only 0-5mSec on 5V devices)
 */

 if (i >= IXGBE_EEPROM_MAX_RETRY_SPI) {
  hw_dbg(hw, "SPI EEPROM Status error\n");
  return -EIO;
 }

 return 0;
}

/**
 *  ixgbe_standby_eeprom - Returns EEPROM to a "standby" state
 *  @hw: pointer to hardware structure
 **/

static void ixgbe_standby_eeprom(struct ixgbe_hw *hw)
{
 u32 eec;

 eec = IXGBE_READ_REG(hw, IXGBE_EEC(hw));

 /* Toggle CS to flush commands */
 eec |= IXGBE_EEC_CS;
 IXGBE_WRITE_REG(hw, IXGBE_EEC(hw), eec);
 IXGBE_WRITE_FLUSH(hw);
 udelay(1);
 eec &= ~IXGBE_EEC_CS;
 IXGBE_WRITE_REG(hw, IXGBE_EEC(hw), eec);
 IXGBE_WRITE_FLUSH(hw);
 udelay(1);
}

/**
 *  ixgbe_shift_out_eeprom_bits - Shift data bits out to the EEPROM.
 *  @hw: pointer to hardware structure
 *  @data: data to send to the EEPROM
 *  @count: number of bits to shift out
 **/

static void ixgbe_shift_out_eeprom_bits(struct ixgbe_hw *hw, u16 data,
     u16 count)
{
 u32 eec;
 u32 mask;
 u32 i;

 eec = IXGBE_READ_REG(hw, IXGBE_EEC(hw));

 /*
 * Mask is used to shift "count" bits of "data" out to the EEPROM
 * one bit at a time.  Determine the starting bit based on count
 */

 mask = BIT(count - 1);

 for (i = 0; i < count; i++) {
  /*
 * A "1" is shifted out to the EEPROM by setting bit "DI" to a
 * "1", and then raising and then lowering the clock (the SK
 * bit controls the clock input to the EEPROM).  A "0" is
 * shifted out to the EEPROM by setting "DI" to "0" and then
 * raising and then lowering the clock.
 */

  if (data & mask)
   eec |= IXGBE_EEC_DI;
  else
   eec &= ~IXGBE_EEC_DI;

  IXGBE_WRITE_REG(hw, IXGBE_EEC(hw), eec);
  IXGBE_WRITE_FLUSH(hw);

  udelay(1);

  ixgbe_raise_eeprom_clk(hw, &eec);
  ixgbe_lower_eeprom_clk(hw, &eec);

  /*
 * Shift mask to signify next bit of data to shift in to the
 * EEPROM
 */

  mask = mask >> 1;
 }

 /* We leave the "DI" bit set to "0" when we leave this routine. */
 eec &= ~IXGBE_EEC_DI;
 IXGBE_WRITE_REG(hw, IXGBE_EEC(hw), eec);
 IXGBE_WRITE_FLUSH(hw);
}

/**
 *  ixgbe_shift_in_eeprom_bits - Shift data bits in from the EEPROM
 *  @hw: pointer to hardware structure
 *  @count: number of bits to shift
 **/

static u16 ixgbe_shift_in_eeprom_bits(struct ixgbe_hw *hw, u16 count)
{
 u32 eec;
 u32 i;
 u16 data = 0;

 /*
 * In order to read a register from the EEPROM, we need to shift
 * 'count' bits in from the EEPROM. Bits are "shifted in" by raising
 * the clock input to the EEPROM (setting the SK bit), and then reading
 * the value of the "DO" bit.  During this "shifting in" process the
 * "DI" bit should always be clear.
 */

 eec = IXGBE_READ_REG(hw, IXGBE_EEC(hw));

 eec &= ~(IXGBE_EEC_DO | IXGBE_EEC_DI);

 for (i = 0; i < count; i++) {
  data = data << 1;
  ixgbe_raise_eeprom_clk(hw, &eec);

  eec = IXGBE_READ_REG(hw, IXGBE_EEC(hw));

  eec &= ~(IXGBE_EEC_DI);
  if (eec & IXGBE_EEC_DO)
   data |= 1;

  ixgbe_lower_eeprom_clk(hw, &eec);
 }

 return data;
}

/**
 *  ixgbe_raise_eeprom_clk - Raises the EEPROM's clock input.
 *  @hw: pointer to hardware structure
 *  @eec: EEC register's current value
 **/

static void ixgbe_raise_eeprom_clk(struct ixgbe_hw *hw, u32 *eec)
{
 /*
 * Raise the clock input to the EEPROM
 * (setting the SK bit), then delay
 */

 *eec = *eec | IXGBE_EEC_SK;
 IXGBE_WRITE_REG(hw, IXGBE_EEC(hw), *eec);
 IXGBE_WRITE_FLUSH(hw);
 udelay(1);
}

/**
 *  ixgbe_lower_eeprom_clk - Lowers the EEPROM's clock input.
 *  @hw: pointer to hardware structure
 *  @eec: EEC's current value
 **/

static void ixgbe_lower_eeprom_clk(struct ixgbe_hw *hw, u32 *eec)
{
 /*
 * Lower the clock input to the EEPROM (clearing the SK bit), then
 * delay
 */

 *eec = *eec & ~IXGBE_EEC_SK;
 IXGBE_WRITE_REG(hw, IXGBE_EEC(hw), *eec);
 IXGBE_WRITE_FLUSH(hw);
 udelay(1);
}

/**
 *  ixgbe_release_eeprom - Release EEPROM, release semaphores
 *  @hw: pointer to hardware structure
 **/

static void ixgbe_release_eeprom(struct ixgbe_hw *hw)
{
 u32 eec;

 eec = IXGBE_READ_REG(hw, IXGBE_EEC(hw));

 eec |= IXGBE_EEC_CS;  /* Pull CS high */
 eec &= ~IXGBE_EEC_SK; /* Lower SCK */

 IXGBE_WRITE_REG(hw, IXGBE_EEC(hw), eec);
 IXGBE_WRITE_FLUSH(hw);

 udelay(1);

 /* Stop requesting EEPROM access */
 eec &= ~IXGBE_EEC_REQ;
 IXGBE_WRITE_REG(hw, IXGBE_EEC(hw), eec);

 hw->mac.ops.release_swfw_sync(hw, IXGBE_GSSR_EEP_SM);

 /*
 * Delay before attempt to obtain semaphore again to allow FW
 * access. semaphore_delay is in ms we need us for usleep_range
 */

 usleep_range(hw->eeprom.semaphore_delay * 1000,
       hw->eeprom.semaphore_delay * 2000);
}

/**
 *  ixgbe_calc_eeprom_checksum_generic - Calculates and returns the checksum
 *  @hw: pointer to hardware structure
 **/

int ixgbe_calc_eeprom_checksum_generic(struct ixgbe_hw *hw)
{
 u16 i;
 u16 j;
 u16 checksum = 0;
 u16 length = 0;
 u16 pointer = 0;
 u16 word = 0;

 /* Include 0x0-0x3F in the checksum */
 for (i = 0; i < IXGBE_EEPROM_CHECKSUM; i++) {
  if (hw->eeprom.ops.read(hw, i, &word)) {
   hw_dbg(hw, "EEPROM read failed\n");
   break;
  }
  checksum += word;
 }

 /* Include all data from pointers except for the fw pointer */
 for (i = IXGBE_PCIE_ANALOG_PTR; i < IXGBE_FW_PTR; i++) {
  if (hw->eeprom.ops.read(hw, i, &pointer)) {
   hw_dbg(hw, "EEPROM read failed\n");
   return -EIO;
  }

  /* If the pointer seems invalid */
  if (pointer == 0xFFFF || pointer == 0)
   continue;

  if (hw->eeprom.ops.read(hw, pointer, &length)) {
   hw_dbg(hw, "EEPROM read failed\n");
   return -EIO;
  }

  if (length == 0xFFFF || length == 0)
   continue;

  for (j = pointer + 1; j <= pointer + length; j++) {
   if (hw->eeprom.ops.read(hw, j, &word)) {
    hw_dbg(hw, "EEPROM read failed\n");
    return -EIO;
   }
   checksum += word;
  }
 }

 checksum = (u16)IXGBE_EEPROM_SUM - checksum;

 return (int)checksum;
}

/**
 *  ixgbe_validate_eeprom_checksum_generic - Validate EEPROM checksum
 *  @hw: pointer to hardware structure
 *  @checksum_val: calculated checksum
 *
 *  Performs checksum calculation and validates the EEPROM checksum.  If the
 *  caller does not need checksum_val, the value can be NULL.
 **/

int ixgbe_validate_eeprom_checksum_generic(struct ixgbe_hw *hw,
        u16 *checksum_val)
{
 u16 read_checksum = 0;
 u16 checksum;
 int status;

 /*
 * Read the first word from the EEPROM. If this times out or fails, do
 * not continue or we could be in for a very long wait while every
 * EEPROM read fails
 */

 status = hw->eeprom.ops.read(hw, 0, &checksum);
 if (status) {
  hw_dbg(hw, "EEPROM read failed\n");
  return status;
 }

 status = hw->eeprom.ops.calc_checksum(hw);
 if (status < 0)
  return status;

 checksum = (u16)(status & 0xffff);

 status = hw->eeprom.ops.read(hw, IXGBE_EEPROM_CHECKSUM, &read_checksum);
 if (status) {
  hw_dbg(hw, "EEPROM read failed\n");
  return status;
 }

 /* Verify read checksum from EEPROM is the same as
 * calculated checksum
 */

 if (read_checksum != checksum)
  status = -EIO;

 /* If the user cares, return the calculated checksum */
 if (checksum_val)
  *checksum_val = checksum;

 return status;
}

/**
 *  ixgbe_update_eeprom_checksum_generic - Updates the EEPROM checksum
 *  @hw: pointer to hardware structure
 **/

int ixgbe_update_eeprom_checksum_generic(struct ixgbe_hw *hw)
{
 u16 checksum;
 int status;

 /*
 * Read the first word from the EEPROM. If this times out or fails, do
 * not continue or we could be in for a very long wait while every
 * EEPROM read fails
 */

 status = hw->eeprom.ops.read(hw, 0, &checksum);
 if (status) {
  hw_dbg(hw, "EEPROM read failed\n");
  return status;
 }

 status = hw->eeprom.ops.calc_checksum(hw);
 if (status < 0)
  return status;

 checksum = (u16)(status & 0xffff);

 status = hw->eeprom.ops.write(hw, IXGBE_EEPROM_CHECKSUM, checksum);

 return status;
}

/**
 *  ixgbe_set_rar_generic - Set Rx address register
 *  @hw: pointer to hardware structure
 *  @index: Receive address register to write
 *  @addr: Address to put into receive address register
 *  @vmdq: VMDq "set" or "pool" index
 *  @enable_addr: set flag that address is active
 *
 *  Puts an ethernet address into a receive address register.
 **/

int ixgbe_set_rar_generic(struct ixgbe_hw *hw, u32 index, u8 *addr, u32 vmdq,
     u32 enable_addr)
{
 u32 rar_low, rar_high;
 u32 rar_entries = hw->mac.num_rar_entries;

 /* Make sure we are using a valid rar index range */
 if (index >= rar_entries) {
  hw_dbg(hw, "RAR index %d is out of range.\n", index);
  return -EINVAL;
 }

 /* setup VMDq pool selection before this RAR gets enabled */
 hw->mac.ops.set_vmdq(hw, index, vmdq);

 /*
 * HW expects these in little endian so we reverse the byte
 * order from network order (big endian) to little endian
 */

 rar_low = ((u32)addr[0] |
     ((u32)addr[1] << 8) |
     ((u32)addr[2] << 16) |
     ((u32)addr[3] << 24));
 /*
 * Some parts put the VMDq setting in the extra RAH bits,
 * so save everything except the lower 16 bits that hold part
 * of the address and the address valid bit.
 */

 rar_high = IXGBE_READ_REG(hw, IXGBE_RAH(index));
 rar_high &= ~(0x0000FFFF | IXGBE_RAH_AV);
 rar_high |= ((u32)addr[4] | ((u32)addr[5] << 8));

 if (enable_addr != 0)
  rar_high |= IXGBE_RAH_AV;

 /* Record lower 32 bits of MAC address and then make
 * sure that write is flushed to hardware before writing
 * the upper 16 bits and setting the valid bit.
 */

 IXGBE_WRITE_REG(hw, IXGBE_RAL(index), rar_low);
 IXGBE_WRITE_FLUSH(hw);
 IXGBE_WRITE_REG(hw, IXGBE_RAH(index), rar_high);

 return 0;
}

/**
 *  ixgbe_clear_rar_generic - Remove Rx address register
 *  @hw: pointer to hardware structure
 *  @index: Receive address register to write
 *
 *  Clears an ethernet address from a receive address register.
 **/

int ixgbe_clear_rar_generic(struct ixgbe_hw *hw, u32 index)
{
 u32 rar_high;
 u32 rar_entries = hw->mac.num_rar_entries;

 /* Make sure we are using a valid rar index range */
 if (index >= rar_entries) {
  hw_dbg(hw, "RAR index %d is out of range.\n", index);
  return -EINVAL;
 }

 /*
 * Some parts put the VMDq setting in the extra RAH bits,
 * so save everything except the lower 16 bits that hold part
 * of the address and the address valid bit.
 */

 rar_high = IXGBE_READ_REG(hw, IXGBE_RAH(index));
 rar_high &= ~(0x0000FFFF | IXGBE_RAH_AV);

 /* Clear the address valid bit and upper 16 bits of the address
 * before clearing the lower bits. This way we aren't updating
 * a live filter.
 */

 IXGBE_WRITE_REG(hw, IXGBE_RAH(index), rar_high);
 IXGBE_WRITE_FLUSH(hw);
 IXGBE_WRITE_REG(hw, IXGBE_RAL(index), 0);

 /* clear VMDq pool/queue selection for this RAR */
 hw->mac.ops.clear_vmdq(hw, index, IXGBE_CLEAR_VMDQ_ALL);

 return 0;
}

/**
 *  ixgbe_init_rx_addrs_generic - Initializes receive address filters.
 *  @hw: pointer to hardware structure
 *
 *  Places the MAC address in receive address register 0 and clears the rest
 *  of the receive address registers. Clears the multicast table. Assumes
 *  the receiver is in reset when the routine is called.
 **/

int ixgbe_init_rx_addrs_generic(struct ixgbe_hw *hw)
{
 u32 i;
 u32 rar_entries = hw->mac.num_rar_entries;

 /*
 * If the current mac address is valid, assume it is a software override
 * to the permanent address.
 * Otherwise, use the permanent address from the eeprom.
 */

 if (!is_valid_ether_addr(hw->mac.addr)) {
  /* Get the MAC address from the RAR0 for later reference */
  hw->mac.ops.get_mac_addr(hw, hw->mac.addr);

  hw_dbg(hw, " Keeping Current RAR0 Addr =%pM\n", hw->mac.addr);
 } else {
  /* Setup the receive address. */
  hw_dbg(hw, "Overriding MAC Address in RAR[0]\n");
  hw_dbg(hw, " New MAC Addr =%pM\n", hw->mac.addr);

  hw->mac.ops.set_rar(hw, 0, hw->mac.addr, 0, IXGBE_RAH_AV);
 }

 /*  clear VMDq pool/queue selection for RAR 0 */
 hw->mac.ops.clear_vmdq(hw, 0, IXGBE_CLEAR_VMDQ_ALL);

 hw->addr_ctrl.overflow_promisc = 0;

 hw->addr_ctrl.rar_used_count = 1;

 /* Zero out the other receive addresses. */
 hw_dbg(hw, "Clearing RAR[1-%d]\n", rar_entries - 1);
 for (i = 1; i < rar_entries; i++) {
  IXGBE_WRITE_REG(hw, IXGBE_RAL(i), 0);
  IXGBE_WRITE_REG(hw, IXGBE_RAH(i), 0);
 }

 /* Clear the MTA */
 hw->addr_ctrl.mta_in_use = 0;
 IXGBE_WRITE_REG(hw, IXGBE_MCSTCTRL, hw->mac.mc_filter_type);

 hw_dbg(hw, " Clearing MTA\n");
 for (i = 0; i < hw->mac.mcft_size; i++)
  IXGBE_WRITE_REG(hw, IXGBE_MTA(i), 0);

 if (hw->mac.ops.init_uta_tables)
  hw->mac.ops.init_uta_tables(hw);

 return 0;
}

/**
 *  ixgbe_mta_vector - Determines bit-vector in multicast table to set
 *  @hw: pointer to hardware structure
 *  @mc_addr: the multicast address
 *
 *  Extracts the 12 bits, from a multicast address, to determine which
 *  bit-vector to set in the multicast table. The hardware uses 12 bits, from
 *  incoming rx multicast addresses, to determine the bit-vector to check in
 *  the MTA. Which of the 4 combination, of 12-bits, the hardware uses is set
 *  by the MO field of the MCSTCTRL. The MO field is set during initialization
 *  to mc_filter_type.
 **/

static int ixgbe_mta_vector(struct ixgbe_hw *hw, u8 *mc_addr)
{
 u32 vector = 0;

 switch (hw->mac.mc_filter_type) {
 case 0:   /* use bits [47:36] of the address */
  vector = ((mc_addr[4] >> 4) | (((u16)mc_addr[5]) << 4));
  break;
 case 1:   /* use bits [46:35] of the address */
  vector = ((mc_addr[4] >> 3) | (((u16)mc_addr[5]) << 5));
  break;
 case 2:   /* use bits [45:34] of the address */
  vector = ((mc_addr[4] >> 2) | (((u16)mc_addr[5]) << 6));
  break;
 case 3:   /* use bits [43:32] of the address */
  vector = ((mc_addr[4]) | (((u16)mc_addr[5]) << 8));
  break;
 default:  /* Invalid mc_filter_type */
  hw_dbg(hw, "MC filter type param set incorrectly\n");
  break;
 }

 /* vector can only be 12-bits or boundary will be exceeded */
 vector &= 0xFFF;
 return vector;
}

/**
 *  ixgbe_set_mta - Set bit-vector in multicast table
 *  @hw: pointer to hardware structure
 *  @mc_addr: Multicast address
 *
 *  Sets the bit-vector in the multicast table.
 **/

static void ixgbe_set_mta(struct ixgbe_hw *hw, u8 *mc_addr)
{
 u32 vector;
 u32 vector_bit;
 u32 vector_reg;

 hw->addr_ctrl.mta_in_use++;

 vector = ixgbe_mta_vector(hw, mc_addr);
 hw_dbg(hw, " bit-vector = 0x%03X\n", vector);

 /*
 * The MTA is a register array of 128 32-bit registers. It is treated
 * like an array of 4096 bits.  We want to set bit
 * BitArray[vector_value]. So we figure out what register the bit is
 * in, read it, OR in the new bit, then write back the new value.  The
 * register is determined by the upper 7 bits of the vector value and
 * the bit within that register are determined by the lower 5 bits of
 * the value.
 */

 vector_reg = (vector >> 5) & 0x7F;
 vector_bit = vector & 0x1F;
 hw->mac.mta_shadow[vector_reg] |= BIT(vector_bit);
}

/**
 *  ixgbe_update_mc_addr_list_generic - Updates MAC list of multicast addresses
 *  @hw: pointer to hardware structure
 *  @netdev: pointer to net device structure
 *
 *  The given list replaces any existing list. Clears the MC addrs from receive
 *  address registers and the multicast table. Uses unused receive address
 *  registers for the first multicast addresses, and hashes the rest into the
 *  multicast table.
 **/

int ixgbe_update_mc_addr_list_generic(struct ixgbe_hw *hw,
          struct net_device *netdev)
{
 struct netdev_hw_addr *ha;
 u32 i;

 /*
 * Set the new number of MC addresses that we are being requested to
 * use.
 */

 hw->addr_ctrl.num_mc_addrs = netdev_mc_count(netdev);
 hw->addr_ctrl.mta_in_use = 0;

 /* Clear mta_shadow */
 hw_dbg(hw, " Clearing MTA\n");
 memset(&hw->mac.mta_shadow, 0, sizeof(hw->mac.mta_shadow));

 /* Update mta shadow */
 netdev_for_each_mc_addr(ha, netdev) {
  hw_dbg(hw, " Adding the multicast addresses:\n");
  ixgbe_set_mta(hw, ha->addr);
 }

 /* Enable mta */
 for (i = 0; i < hw->mac.mcft_size; i++)
  IXGBE_WRITE_REG_ARRAY(hw, IXGBE_MTA(0), i,
          hw->mac.mta_shadow[i]);

 if (hw->addr_ctrl.mta_in_use > 0)
  IXGBE_WRITE_REG(hw, IXGBE_MCSTCTRL,
    IXGBE_MCSTCTRL_MFE | hw->mac.mc_filter_type);

 hw_dbg(hw, "ixgbe_update_mc_addr_list_generic Complete\n");
 return 0;
}

/**
 *  ixgbe_enable_mc_generic - Enable multicast address in RAR
 *  @hw: pointer to hardware structure
 *
 *  Enables multicast address in RAR and the use of the multicast hash table.
 **/

int ixgbe_enable_mc_generic(struct ixgbe_hw *hw)
{
 struct ixgbe_addr_filter_info *a = &hw->addr_ctrl;

 if (a->mta_in_use > 0)
  IXGBE_WRITE_REG(hw, IXGBE_MCSTCTRL, IXGBE_MCSTCTRL_MFE |
    hw->mac.mc_filter_type);

 return 0;
}

/**
 *  ixgbe_disable_mc_generic - Disable multicast address in RAR
 *  @hw: pointer to hardware structure
 *
 *  Disables multicast address in RAR and the use of the multicast hash table.
 **/

int ixgbe_disable_mc_generic(struct ixgbe_hw *hw)
{
 struct ixgbe_addr_filter_info *a = &hw->addr_ctrl;

 if (a->mta_in_use > 0)
  IXGBE_WRITE_REG(hw, IXGBE_MCSTCTRL, hw->mac.mc_filter_type);

 return 0;
}

/**
 *  ixgbe_fc_enable_generic - Enable flow control
 *  @hw: pointer to hardware structure
 *
 *  Enable flow control according to the current settings.
 **/

int ixgbe_fc_enable_generic(struct ixgbe_hw *hw)
{
 u32 mflcn_reg, fccfg_reg;
 u32 reg;
 u32 fcrtl, fcrth;
 int i;

 /* Validate the water mark configuration. */
 if (!hw->fc.pause_time)
  return -EINVAL;

 /* Low water mark of zero causes XOFF floods */
 for (i = 0; i < MAX_TRAFFIC_CLASS; i++) {
  if ((hw->fc.current_mode & ixgbe_fc_tx_pause) &&
      hw->fc.high_water[i]) {
   if (!hw->fc.low_water[i] ||
       hw->fc.low_water[i] >= hw->fc.high_water[i]) {
    hw_dbg(hw, "Invalid water mark configuration\n");
    return -EINVAL;
   }
  }
 }

 /* Negotiate the fc mode to use */
 hw->mac.ops.fc_autoneg(hw);

 /* Disable any previous flow control settings */
 mflcn_reg = IXGBE_READ_REG(hw, IXGBE_MFLCN);
 mflcn_reg &= ~(IXGBE_MFLCN_RPFCE_MASK | IXGBE_MFLCN_RFCE);

 fccfg_reg = IXGBE_READ_REG(hw, IXGBE_FCCFG);
 fccfg_reg &= ~(IXGBE_FCCFG_TFCE_802_3X | IXGBE_FCCFG_TFCE_PRIORITY);

 /*
 * 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.
 */

 switch (hw->fc.current_mode) {
 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.
 */

  mflcn_reg |= IXGBE_MFLCN_RFCE;
  break;
 case ixgbe_fc_tx_pause:
  /*
 * Tx Flow control is enabled, and Rx Flow control is
 * disabled by software override.
 */

  fccfg_reg |= IXGBE_FCCFG_TFCE_802_3X;
  break;
 case ixgbe_fc_full:
  /* Flow control (both Rx and Tx) is enabled by SW override. */
  mflcn_reg |= IXGBE_MFLCN_RFCE;
  fccfg_reg |= IXGBE_FCCFG_TFCE_802_3X;
  break;
 default:
  hw_dbg(hw, "Flow control param set incorrectly\n");
  return -EIO;
 }

 /* Set 802.3x based flow control settings. */
 mflcn_reg |= IXGBE_MFLCN_DPF;
 IXGBE_WRITE_REG(hw, IXGBE_MFLCN, mflcn_reg);
 IXGBE_WRITE_REG(hw, IXGBE_FCCFG, fccfg_reg);

 /* Set up and enable Rx high/low water mark thresholds, enable XON. */
 for (i = 0; i < MAX_TRAFFIC_CLASS; i++) {
  if ((hw->fc.current_mode & ixgbe_fc_tx_pause) &&
      hw->fc.high_water[i]) {
   fcrtl = (hw->fc.low_water[i] << 10) | IXGBE_FCRTL_XONE;
   IXGBE_WRITE_REG(hw, IXGBE_FCRTL_82599(i), fcrtl);
   fcrth = (hw->fc.high_water[i] << 10) | IXGBE_FCRTH_FCEN;
  } else {
   IXGBE_WRITE_REG(hw, IXGBE_FCRTL_82599(i), 0);
   /*
 * In order to prevent Tx hangs when the internal Tx
 * switch is enabled we must set the high water mark
 * to the Rx packet buffer size - 24KB.  This allows
 * the Tx switch to function even under heavy Rx
 * workloads.
 */

   fcrth = IXGBE_READ_REG(hw, IXGBE_RXPBSIZE(i)) - 24576;
  }

  IXGBE_WRITE_REG(hw, IXGBE_FCRTH_82599(i), fcrth);
 }

 /* Configure pause time (2 TCs per register) */
 reg = hw->fc.pause_time * 0x00010001U;
 for (i = 0; i < (MAX_TRAFFIC_CLASS / 2); i++)
  IXGBE_WRITE_REG(hw, IXGBE_FCTTV(i), reg);

 IXGBE_WRITE_REG(hw, IXGBE_FCRTV, hw->fc.pause_time / 2);

 return 0;
}

/**
 *  ixgbe_negotiate_fc - Negotiate flow control
 *  @hw: pointer to hardware structure
 *  @adv_reg: flow control advertised settings
 *  @lp_reg: link partner's flow control settings
 *  @adv_sym: symmetric pause bit in advertisement
 *  @adv_asm: asymmetric pause bit in advertisement
 *  @lp_sym: symmetric pause bit in link partner advertisement
 *  @lp_asm: asymmetric pause bit in link partner advertisement
 *
 *  Find the intersection between advertised settings and link partner's
 *  advertised settings
 **/

int ixgbe_negotiate_fc(struct ixgbe_hw *hw, u32 adv_reg, u32 lp_reg,
         u32 adv_sym, u32 adv_asm, u32 lp_sym, u32 lp_asm)
{
 if ((!(adv_reg)) ||  (!(lp_reg)))
  return -EINVAL;

 if ((adv_reg & adv_sym) && (lp_reg & lp_sym)) {
  /*
 * 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.requested_mode == ixgbe_fc_full) {
   hw->fc.current_mode = ixgbe_fc_full;
   hw_dbg(hw, "Flow Control = FULL.\n");
  } else {
   hw->fc.current_mode = ixgbe_fc_rx_pause;
   hw_dbg(hw, "Flow Control=RX PAUSE frames only\n");
  }
 } else if (!(adv_reg & adv_sym) && (adv_reg & adv_asm) &&
     (lp_reg & lp_sym) && (lp_reg & lp_asm)) {
  hw->fc.current_mode = ixgbe_fc_tx_pause;
  hw_dbg(hw, "Flow Control = TX PAUSE frames only.\n");
 } else if ((adv_reg & adv_sym) && (adv_reg & adv_asm) &&
     !(lp_reg & lp_sym) && (lp_reg & lp_asm)) {
  hw->fc.current_mode = ixgbe_fc_rx_pause;
  hw_dbg(hw, "Flow Control = RX PAUSE frames only.\n");
 } else {
  hw->fc.current_mode = ixgbe_fc_none;
  hw_dbg(hw, "Flow Control = NONE.\n");
 }
 return 0;
}

/**
 *  ixgbe_fc_autoneg_fiber - Enable flow control on 1 gig fiber
 *  @hw: pointer to hardware structure
 *
 *  Enable flow control according on 1 gig fiber.
 **/

static int ixgbe_fc_autoneg_fiber(struct ixgbe_hw *hw)
{
 u32 pcs_anadv_reg, pcs_lpab_reg, linkstat;
 int ret_val;

 /*
 * On multispeed fiber at 1g, bail out if
 * - link is up but AN did not complete, or if
 * - link is up and AN completed but timed out
 */


 linkstat = IXGBE_READ_REG(hw, IXGBE_PCS1GLSTA);
 if ((!!(linkstat & IXGBE_PCS1GLSTA_AN_COMPLETE) == 0) ||
     (!!(linkstat & IXGBE_PCS1GLSTA_AN_TIMED_OUT) == 1))
  return -EIO;

 pcs_anadv_reg = IXGBE_READ_REG(hw, IXGBE_PCS1GANA);
 pcs_lpab_reg = IXGBE_READ_REG(hw, IXGBE_PCS1GANLP);

 ret_val =  ixgbe_negotiate_fc(hw, pcs_anadv_reg,
          pcs_lpab_reg, IXGBE_PCS1GANA_SYM_PAUSE,
          IXGBE_PCS1GANA_ASM_PAUSE,
          IXGBE_PCS1GANA_SYM_PAUSE,
          IXGBE_PCS1GANA_ASM_PAUSE);

 return ret_val;
}

/**
 *  ixgbe_fc_autoneg_backplane - Enable flow control IEEE clause 37
 *  @hw: pointer to hardware structure
 *
 *  Enable flow control according to IEEE clause 37.
 **/

static int ixgbe_fc_autoneg_backplane(struct ixgbe_hw *hw)
{
 u32 links2, anlp1_reg, autoc_reg, links;
 int ret_val;

 /*
 * On backplane, bail out if
 * - backplane autoneg was not completed, or if
 * - we are 82599 and link partner is not AN enabled
 */

 links = IXGBE_READ_REG(hw, IXGBE_LINKS);
 if ((links & IXGBE_LINKS_KX_AN_COMP) == 0)
  return -EIO;

 if (hw->mac.type == ixgbe_mac_82599EB) {
  links2 = IXGBE_READ_REG(hw, IXGBE_LINKS2);
  if ((links2 & IXGBE_LINKS2_AN_SUPPORTED) == 0)
   return -EIO;
 }
 /*
 * Read the 10g AN autoc and LP ability registers and resolve
--> --------------------

--> maximum size reached

--> --------------------

Messung V0.5
C=93 H=90 G=91

¤ Dauer der Verarbeitung: 0.19 Sekunden  ¤

*© Formatika GbR, Deutschland






Wurzel

Suchen

Beweissystem der NASA

Beweissystem Isabelle

NIST Cobol Testsuite

Cephes Mathematical Library

Wiener Entwicklungsmethode

Haftungshinweis

Die Informationen auf dieser Webseite wurden nach bestem Wissen sorgfältig zusammengestellt. Es wird jedoch weder Vollständigkeit, noch Richtigkeit, noch Qualität der bereit gestellten Informationen zugesichert.

Bemerkung:

Die farbliche Syntaxdarstellung und die Messung sind noch experimentell.