|
@@ -127,6 +127,9 @@ static enum ixgbe_phy_type ixgbe_get_phy_type_from_id(u32 phy_id)
|
|
|
case QT2022_PHY_ID:
|
|
|
phy_type = ixgbe_phy_qt;
|
|
|
break;
|
|
|
+ case ATH_PHY_ID:
|
|
|
+ phy_type = ixgbe_phy_nl;
|
|
|
+ break;
|
|
|
default:
|
|
|
phy_type = ixgbe_phy_unknown;
|
|
|
break;
|
|
@@ -429,6 +432,261 @@ s32 ixgbe_setup_phy_link_speed_generic(struct ixgbe_hw *hw,
|
|
|
return 0;
|
|
|
}
|
|
|
|
|
|
+/**
|
|
|
+ * ixgbe_reset_phy_nl - Performs a PHY reset
|
|
|
+ * @hw: pointer to hardware structure
|
|
|
+ **/
|
|
|
+s32 ixgbe_reset_phy_nl(struct ixgbe_hw *hw)
|
|
|
+{
|
|
|
+ u16 phy_offset, control, eword, edata, block_crc;
|
|
|
+ bool end_data = false;
|
|
|
+ u16 list_offset, data_offset;
|
|
|
+ u16 phy_data = 0;
|
|
|
+ s32 ret_val = 0;
|
|
|
+ u32 i;
|
|
|
+
|
|
|
+ hw->phy.ops.read_reg(hw, IXGBE_MDIO_PHY_XS_CONTROL,
|
|
|
+ IXGBE_MDIO_PHY_XS_DEV_TYPE, &phy_data);
|
|
|
+
|
|
|
+ /* reset the PHY and poll for completion */
|
|
|
+ hw->phy.ops.write_reg(hw, IXGBE_MDIO_PHY_XS_CONTROL,
|
|
|
+ IXGBE_MDIO_PHY_XS_DEV_TYPE,
|
|
|
+ (phy_data | IXGBE_MDIO_PHY_XS_RESET));
|
|
|
+
|
|
|
+ for (i = 0; i < 100; i++) {
|
|
|
+ hw->phy.ops.read_reg(hw, IXGBE_MDIO_PHY_XS_CONTROL,
|
|
|
+ IXGBE_MDIO_PHY_XS_DEV_TYPE, &phy_data);
|
|
|
+ if ((phy_data & IXGBE_MDIO_PHY_XS_RESET) == 0)
|
|
|
+ break;
|
|
|
+ msleep(10);
|
|
|
+ }
|
|
|
+
|
|
|
+ if ((phy_data & IXGBE_MDIO_PHY_XS_RESET) != 0) {
|
|
|
+ hw_dbg(hw, "PHY reset did not complete.\n");
|
|
|
+ ret_val = IXGBE_ERR_PHY;
|
|
|
+ goto out;
|
|
|
+ }
|
|
|
+
|
|
|
+ /* Get init offsets */
|
|
|
+ ret_val = ixgbe_get_sfp_init_sequence_offsets(hw, &list_offset,
|
|
|
+ &data_offset);
|
|
|
+ if (ret_val != 0)
|
|
|
+ goto out;
|
|
|
+
|
|
|
+ ret_val = hw->eeprom.ops.read(hw, data_offset, &block_crc);
|
|
|
+ data_offset++;
|
|
|
+ while (!end_data) {
|
|
|
+ /*
|
|
|
+ * Read control word from PHY init contents offset
|
|
|
+ */
|
|
|
+ ret_val = hw->eeprom.ops.read(hw, data_offset, &eword);
|
|
|
+ control = (eword & IXGBE_CONTROL_MASK_NL) >>
|
|
|
+ IXGBE_CONTROL_SHIFT_NL;
|
|
|
+ edata = eword & IXGBE_DATA_MASK_NL;
|
|
|
+ switch (control) {
|
|
|
+ case IXGBE_DELAY_NL:
|
|
|
+ data_offset++;
|
|
|
+ hw_dbg(hw, "DELAY: %d MS\n", edata);
|
|
|
+ msleep(edata);
|
|
|
+ break;
|
|
|
+ case IXGBE_DATA_NL:
|
|
|
+ hw_dbg(hw, "DATA: \n");
|
|
|
+ data_offset++;
|
|
|
+ hw->eeprom.ops.read(hw, data_offset++,
|
|
|
+ &phy_offset);
|
|
|
+ for (i = 0; i < edata; i++) {
|
|
|
+ hw->eeprom.ops.read(hw, data_offset, &eword);
|
|
|
+ hw->phy.ops.write_reg(hw, phy_offset,
|
|
|
+ IXGBE_TWINAX_DEV, eword);
|
|
|
+ hw_dbg(hw, "Wrote %4.4x to %4.4x\n", eword,
|
|
|
+ phy_offset);
|
|
|
+ data_offset++;
|
|
|
+ phy_offset++;
|
|
|
+ }
|
|
|
+ break;
|
|
|
+ case IXGBE_CONTROL_NL:
|
|
|
+ data_offset++;
|
|
|
+ hw_dbg(hw, "CONTROL: \n");
|
|
|
+ if (edata == IXGBE_CONTROL_EOL_NL) {
|
|
|
+ hw_dbg(hw, "EOL\n");
|
|
|
+ end_data = true;
|
|
|
+ } else if (edata == IXGBE_CONTROL_SOL_NL) {
|
|
|
+ hw_dbg(hw, "SOL\n");
|
|
|
+ } else {
|
|
|
+ hw_dbg(hw, "Bad control value\n");
|
|
|
+ ret_val = IXGBE_ERR_PHY;
|
|
|
+ goto out;
|
|
|
+ }
|
|
|
+ break;
|
|
|
+ default:
|
|
|
+ hw_dbg(hw, "Bad control type\n");
|
|
|
+ ret_val = IXGBE_ERR_PHY;
|
|
|
+ goto out;
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+out:
|
|
|
+ return ret_val;
|
|
|
+}
|
|
|
+
|
|
|
+/**
|
|
|
+ * ixgbe_identify_sfp_module_generic - Identifies SFP module and assigns
|
|
|
+ * the PHY type.
|
|
|
+ * @hw: pointer to hardware structure
|
|
|
+ *
|
|
|
+ * Searches for and indentifies the SFP module. Assings appropriate PHY type.
|
|
|
+ **/
|
|
|
+s32 ixgbe_identify_sfp_module_generic(struct ixgbe_hw *hw)
|
|
|
+{
|
|
|
+ s32 status = IXGBE_ERR_PHY_ADDR_INVALID;
|
|
|
+ u32 vendor_oui = 0;
|
|
|
+ u8 identifier = 0;
|
|
|
+ u8 comp_codes_1g = 0;
|
|
|
+ u8 comp_codes_10g = 0;
|
|
|
+ u8 oui_bytes[4] = {0, 0, 0, 0};
|
|
|
+ u8 transmission_media = 0;
|
|
|
+
|
|
|
+ status = hw->phy.ops.read_i2c_eeprom(hw, IXGBE_SFF_IDENTIFIER,
|
|
|
+ &identifier);
|
|
|
+
|
|
|
+ if (status == IXGBE_ERR_SFP_NOT_PRESENT) {
|
|
|
+ hw->phy.sfp_type = ixgbe_sfp_type_not_present;
|
|
|
+ goto out;
|
|
|
+ }
|
|
|
+
|
|
|
+ if (identifier == IXGBE_SFF_IDENTIFIER_SFP) {
|
|
|
+ hw->phy.ops.read_i2c_eeprom(hw, IXGBE_SFF_1GBE_COMP_CODES,
|
|
|
+ &comp_codes_1g);
|
|
|
+ hw->phy.ops.read_i2c_eeprom(hw, IXGBE_SFF_10GBE_COMP_CODES,
|
|
|
+ &comp_codes_10g);
|
|
|
+ hw->phy.ops.read_i2c_eeprom(hw, IXGBE_SFF_TRANSMISSION_MEDIA,
|
|
|
+ &transmission_media);
|
|
|
+
|
|
|
+ /* ID Module
|
|
|
+ * =========
|
|
|
+ * 0 SFP_DA_CU
|
|
|
+ * 1 SFP_SR
|
|
|
+ * 2 SFP_LR
|
|
|
+ */
|
|
|
+ if (transmission_media & IXGBE_SFF_TWIN_AX_CAPABLE)
|
|
|
+ hw->phy.sfp_type = ixgbe_sfp_type_da_cu;
|
|
|
+ else if (comp_codes_10g & IXGBE_SFF_10GBASESR_CAPABLE)
|
|
|
+ hw->phy.sfp_type = ixgbe_sfp_type_sr;
|
|
|
+ else if (comp_codes_10g & IXGBE_SFF_10GBASELR_CAPABLE)
|
|
|
+ hw->phy.sfp_type = ixgbe_sfp_type_lr;
|
|
|
+ else
|
|
|
+ hw->phy.sfp_type = ixgbe_sfp_type_unknown;
|
|
|
+
|
|
|
+ /* Determine PHY vendor */
|
|
|
+ if (hw->phy.type == ixgbe_phy_unknown) {
|
|
|
+ hw->phy.id = identifier;
|
|
|
+ hw->phy.ops.read_i2c_eeprom(hw,
|
|
|
+ IXGBE_SFF_VENDOR_OUI_BYTE0,
|
|
|
+ &oui_bytes[0]);
|
|
|
+ hw->phy.ops.read_i2c_eeprom(hw,
|
|
|
+ IXGBE_SFF_VENDOR_OUI_BYTE1,
|
|
|
+ &oui_bytes[1]);
|
|
|
+ hw->phy.ops.read_i2c_eeprom(hw,
|
|
|
+ IXGBE_SFF_VENDOR_OUI_BYTE2,
|
|
|
+ &oui_bytes[2]);
|
|
|
+
|
|
|
+ vendor_oui =
|
|
|
+ ((oui_bytes[0] << IXGBE_SFF_VENDOR_OUI_BYTE0_SHIFT) |
|
|
|
+ (oui_bytes[1] << IXGBE_SFF_VENDOR_OUI_BYTE1_SHIFT) |
|
|
|
+ (oui_bytes[2] << IXGBE_SFF_VENDOR_OUI_BYTE2_SHIFT));
|
|
|
+
|
|
|
+ switch (vendor_oui) {
|
|
|
+ case IXGBE_SFF_VENDOR_OUI_TYCO:
|
|
|
+ if (transmission_media &
|
|
|
+ IXGBE_SFF_TWIN_AX_CAPABLE)
|
|
|
+ hw->phy.type = ixgbe_phy_tw_tyco;
|
|
|
+ break;
|
|
|
+ case IXGBE_SFF_VENDOR_OUI_FTL:
|
|
|
+ hw->phy.type = ixgbe_phy_sfp_ftl;
|
|
|
+ break;
|
|
|
+ case IXGBE_SFF_VENDOR_OUI_AVAGO:
|
|
|
+ hw->phy.type = ixgbe_phy_sfp_avago;
|
|
|
+ break;
|
|
|
+ default:
|
|
|
+ if (transmission_media &
|
|
|
+ IXGBE_SFF_TWIN_AX_CAPABLE)
|
|
|
+ hw->phy.type = ixgbe_phy_tw_unknown;
|
|
|
+ else
|
|
|
+ hw->phy.type = ixgbe_phy_sfp_unknown;
|
|
|
+ break;
|
|
|
+ }
|
|
|
+ }
|
|
|
+ status = 0;
|
|
|
+ }
|
|
|
+
|
|
|
+out:
|
|
|
+ return status;
|
|
|
+}
|
|
|
+
|
|
|
+/**
|
|
|
+ * ixgbe_get_sfp_init_sequence_offsets - Checks the MAC's EEPROM to see
|
|
|
+ * if it supports a given SFP+ module type, if so it returns the offsets to the
|
|
|
+ * phy init sequence block.
|
|
|
+ * @hw: pointer to hardware structure
|
|
|
+ * @list_offset: offset to the SFP ID list
|
|
|
+ * @data_offset: offset to the SFP data block
|
|
|
+ **/
|
|
|
+s32 ixgbe_get_sfp_init_sequence_offsets(struct ixgbe_hw *hw,
|
|
|
+ u16 *list_offset,
|
|
|
+ u16 *data_offset)
|
|
|
+{
|
|
|
+ u16 sfp_id;
|
|
|
+
|
|
|
+ if (hw->phy.sfp_type == ixgbe_sfp_type_unknown)
|
|
|
+ return IXGBE_ERR_SFP_NOT_SUPPORTED;
|
|
|
+
|
|
|
+ if (hw->phy.sfp_type == ixgbe_sfp_type_not_present)
|
|
|
+ return IXGBE_ERR_SFP_NOT_PRESENT;
|
|
|
+
|
|
|
+ if ((hw->device_id == IXGBE_DEV_ID_82598_SR_DUAL_PORT_EM) &&
|
|
|
+ (hw->phy.sfp_type == ixgbe_sfp_type_da_cu))
|
|
|
+ return IXGBE_ERR_SFP_NOT_SUPPORTED;
|
|
|
+
|
|
|
+ /* Read offset to PHY init contents */
|
|
|
+ hw->eeprom.ops.read(hw, IXGBE_PHY_INIT_OFFSET_NL, list_offset);
|
|
|
+
|
|
|
+ if ((!*list_offset) || (*list_offset == 0xFFFF))
|
|
|
+ return IXGBE_ERR_PHY;
|
|
|
+
|
|
|
+ /* Shift offset to first ID word */
|
|
|
+ (*list_offset)++;
|
|
|
+
|
|
|
+ /*
|
|
|
+ * Find the matching SFP ID in the EEPROM
|
|
|
+ * and program the init sequence
|
|
|
+ */
|
|
|
+ hw->eeprom.ops.read(hw, *list_offset, &sfp_id);
|
|
|
+
|
|
|
+ while (sfp_id != IXGBE_PHY_INIT_END_NL) {
|
|
|
+ if (sfp_id == hw->phy.sfp_type) {
|
|
|
+ (*list_offset)++;
|
|
|
+ hw->eeprom.ops.read(hw, *list_offset, data_offset);
|
|
|
+ if ((!*data_offset) || (*data_offset == 0xFFFF)) {
|
|
|
+ hw_dbg(hw, "SFP+ module not supported\n");
|
|
|
+ return IXGBE_ERR_SFP_NOT_SUPPORTED;
|
|
|
+ } else {
|
|
|
+ break;
|
|
|
+ }
|
|
|
+ } else {
|
|
|
+ (*list_offset) += 2;
|
|
|
+ if (hw->eeprom.ops.read(hw, *list_offset, &sfp_id))
|
|
|
+ return IXGBE_ERR_PHY;
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ if (sfp_id == IXGBE_PHY_INIT_END_NL) {
|
|
|
+ hw_dbg(hw, "No matching SFP+ module found\n");
|
|
|
+ return IXGBE_ERR_SFP_NOT_SUPPORTED;
|
|
|
+ }
|
|
|
+
|
|
|
+ return 0;
|
|
|
+}
|
|
|
+
|
|
|
/**
|
|
|
* ixgbe_check_phy_link_tnx - Determine link and speed status
|
|
|
* @hw: pointer to hardware structure
|