// SPDX-License-Identifier: GPL-2.0 /* Copyright (c) 2018 Intel Corporation */ #include "igc_mac.h" #include "igc_nvm.h" /** * igc_poll_eerd_eewr_done - Poll for EEPROM read/write completion * @hw: pointer to the HW structure * @ee_reg: EEPROM flag for polling * * Polls the EEPROM status bit for either read or write completion based * upon the value of 'ee_reg'. */ static s32 igc_poll_eerd_eewr_done(struct igc_hw *hw, int ee_reg) { s32 ret_val = -IGC_ERR_NVM; u32 attempts = 100000; u32 i, reg = 0; for (i = 0; i < attempts; i++) { if (ee_reg == IGC_NVM_POLL_READ) reg = rd32(IGC_EERD); else reg = rd32(IGC_EEWR); if (reg & IGC_NVM_RW_REG_DONE) { ret_val = 0; break; } udelay(5); } return ret_val; } /** * igc_acquire_nvm - Generic request for access to EEPROM * @hw: pointer to the HW structure * * Set the EEPROM access request bit and wait for EEPROM access grant bit. * Return successful if access grant bit set, else clear the request for * EEPROM access and return -IGC_ERR_NVM (-1). */ s32 igc_acquire_nvm(struct igc_hw *hw) { s32 timeout = IGC_NVM_GRANT_ATTEMPTS; u32 eecd = rd32(IGC_EECD); s32 ret_val = 0; wr32(IGC_EECD, eecd | IGC_EECD_REQ); eecd = rd32(IGC_EECD); while (timeout) { if (eecd & IGC_EECD_GNT) break; udelay(5); eecd = rd32(IGC_EECD); timeout--; } if (!timeout) { eecd &= ~IGC_EECD_REQ; wr32(IGC_EECD, eecd); hw_dbg("Could not acquire NVM grant\n"); ret_val = -IGC_ERR_NVM; } return ret_val; } /** * igc_release_nvm - Release exclusive access to EEPROM * @hw: pointer to the HW structure * * Stop any current commands to the EEPROM and clear the EEPROM request bit. */ void igc_release_nvm(struct igc_hw *hw) { u32 eecd; eecd = rd32(IGC_EECD); eecd &= ~IGC_EECD_REQ; wr32(IGC_EECD, eecd); } /** * igc_read_nvm_eerd - Reads EEPROM using EERD register * @hw: pointer to the HW structure * @offset: offset of word in the EEPROM to read * @words: number of words to read * @data: word read from the EEPROM * * Reads a 16 bit word from the EEPROM using the EERD register. */ s32 igc_read_nvm_eerd(struct igc_hw *hw, u16 offset, u16 words, u16 *data) { struct igc_nvm_info *nvm = &hw->nvm; u32 i, eerd = 0; s32 ret_val = 0; /* A check for invalid values: offset too large, too many words, * and not enough words. */ if (offset >= nvm->word_size || (words > (nvm->word_size - offset)) || words == 0) { hw_dbg("nvm parameter(s) out of bounds\n"); ret_val = -IGC_ERR_NVM; goto out; } for (i = 0; i < words; i++) { eerd = ((offset + i) << IGC_NVM_RW_ADDR_SHIFT) + IGC_NVM_RW_REG_START; wr32(IGC_EERD, eerd); ret_val = igc_poll_eerd_eewr_done(hw, IGC_NVM_POLL_READ); if (ret_val) break; data[i] = (rd32(IGC_EERD) >> IGC_NVM_RW_REG_DATA); } out: return ret_val; } /** * igc_read_mac_addr - Read device MAC address * @hw: pointer to the HW structure */ s32 igc_read_mac_addr(struct igc_hw *hw) { u32 rar_high; u32 rar_low; u16 i; rar_high = rd32(IGC_RAH(0)); rar_low = rd32(IGC_RAL(0)); for (i = 0; i < IGC_RAL_MAC_ADDR_LEN; i++) hw->mac.perm_addr[i] = (u8)(rar_low >> (i * 8)); for (i = 0; i < IGC_RAH_MAC_ADDR_LEN; i++) hw->mac.perm_addr[i + 4] = (u8)(rar_high >> (i * 8)); for (i = 0; i < ETH_ALEN; i++) hw->mac.addr[i] = hw->mac.perm_addr[i]; return 0; } /** * igc_validate_nvm_checksum - Validate EEPROM checksum * @hw: pointer to the HW structure * * Calculates the EEPROM checksum by reading/adding each word of the EEPROM * and then verifies that the sum of the EEPROM is equal to 0xBABA. */ s32 igc_validate_nvm_checksum(struct igc_hw *hw) { u16 checksum = 0; u16 i, nvm_data; s32 ret_val = 0; for (i = 0; i < (NVM_CHECKSUM_REG + 1); i++) { ret_val = hw->nvm.ops.read(hw, i, 1, &nvm_data); if (ret_val) { hw_dbg("NVM Read Error\n"); goto out; } checksum += nvm_data; } if (checksum != (u16)NVM_SUM) { hw_dbg("NVM Checksum Invalid\n"); ret_val = -IGC_ERR_NVM; goto out; } out: return ret_val; } /** * igc_update_nvm_checksum - Update EEPROM checksum * @hw: pointer to the HW structure * * Updates the EEPROM checksum by reading/adding each word of the EEPROM * up to the checksum. Then calculates the EEPROM checksum and writes the * value to the EEPROM. */ s32 igc_update_nvm_checksum(struct igc_hw *hw) { u16 checksum = 0; u16 i, nvm_data; s32 ret_val; for (i = 0; i < NVM_CHECKSUM_REG; i++) { ret_val = hw->nvm.ops.read(hw, i, 1, &nvm_data); if (ret_val) { hw_dbg("NVM Read Error while updating checksum.\n"); goto out; } checksum += nvm_data; } checksum = (u16)NVM_SUM - checksum; ret_val = hw->nvm.ops.write(hw, NVM_CHECKSUM_REG, 1, &checksum); if (ret_val) hw_dbg("NVM Write Error while updating checksum.\n"); out: return ret_val; }