/* SPDX-License-Identifier: BSD-3-Clause * Copyright(c) 2015-2020 Beijing WangXun Technology Co., Ltd. * Copyright(c) 2010-2017 Intel Corporation */ #include "txgbe_type.h" #include "txgbe_mng.h" /** * txgbe_calculate_checksum - Calculate checksum for buffer * @buffer: pointer to EEPROM * @length: size of EEPROM to calculate a checksum for * Calculates the checksum for some buffer on a specified length. The * checksum calculated is returned. **/ static u8 txgbe_calculate_checksum(u8 *buffer, u32 length) { u32 i; u8 sum = 0; for (i = 0; i < length; i++) sum += buffer[i]; return (u8)(0 - sum); } /** * txgbe_hic_unlocked - Issue command to manageability block unlocked * @hw: pointer to the HW structure * @buffer: command to write and where the return status will be placed * @length: length of buffer, must be multiple of 4 bytes * @timeout: time in ms to wait for command completion * * Communicates with the manageability block. On success return 0 * else returns semaphore error when encountering an error acquiring * semaphore or TXGBE_ERR_HOST_INTERFACE_COMMAND when command fails. * * This function assumes that the TXGBE_MNGSEM_SWMBX semaphore is held * by the caller. **/ static s32 txgbe_hic_unlocked(struct txgbe_hw *hw, u32 *buffer, u32 length, u32 timeout) { u32 value, loop; u16 i, dword_len; if (!length || length > TXGBE_PMMBX_BSIZE) { DEBUGOUT("Buffer length failure buffersize=%d.", length); return TXGBE_ERR_HOST_INTERFACE_COMMAND; } /* Calculate length in DWORDs. We must be DWORD aligned */ if (length % sizeof(u32)) { DEBUGOUT("Buffer length failure, not aligned to dword"); return TXGBE_ERR_INVALID_ARGUMENT; } dword_len = length >> 2; /* The device driver writes the relevant command block * into the ram area. */ for (i = 0; i < dword_len; i++) { wr32a(hw, TXGBE_MNGMBX, i, cpu_to_le32(buffer[i])); buffer[i] = rd32a(hw, TXGBE_MNGMBX, i); } txgbe_flush(hw); /* Setting this bit tells the ARC that a new command is pending. */ wr32m(hw, TXGBE_MNGMBXCTL, TXGBE_MNGMBXCTL_SWRDY, TXGBE_MNGMBXCTL_SWRDY); /* Check command completion */ loop = po32m(hw, TXGBE_MNGMBXCTL, TXGBE_MNGMBXCTL_FWRDY, TXGBE_MNGMBXCTL_FWRDY, &value, timeout, 1000); if (!loop || !(value & TXGBE_MNGMBXCTL_FWACK)) { DEBUGOUT("Command has failed with no status valid."); return TXGBE_ERR_HOST_INTERFACE_COMMAND; } return 0; } /** * txgbe_host_interface_command - Issue command to manageability block * @hw: pointer to the HW structure * @buffer: contains the command to write and where the return status will * be placed * @length: length of buffer, must be multiple of 4 bytes * @timeout: time in ms to wait for command completion * @return_data: read and return data from the buffer (true) or not (false) * Needed because FW structures are big endian and decoding of * these fields can be 8 bit or 16 bit based on command. Decoding * is not easily understood without making a table of commands. * So we will leave this up to the caller to read back the data * in these cases. * * Communicates with the manageability block. On success return 0 * else returns semaphore error when encountering an error acquiring * semaphore or TXGBE_ERR_HOST_INTERFACE_COMMAND when command fails. **/ static s32 txgbe_host_interface_command(struct txgbe_hw *hw, u32 *buffer, u32 length, u32 timeout, bool return_data) { u32 hdr_size = sizeof(struct txgbe_hic_hdr); struct txgbe_hic_hdr *resp = (struct txgbe_hic_hdr *)buffer; u16 buf_len; s32 err; u32 bi; u32 dword_len; if (length == 0 || length > TXGBE_PMMBX_BSIZE) { DEBUGOUT("Buffer length failure buffersize=%d.", length); return TXGBE_ERR_HOST_INTERFACE_COMMAND; } /* Take management host interface semaphore */ err = hw->mac.acquire_swfw_sync(hw, TXGBE_MNGSEM_SWMBX); if (err) return err; err = txgbe_hic_unlocked(hw, buffer, length, timeout); if (err) goto rel_out; if (!return_data) goto rel_out; /* Calculate length in DWORDs */ dword_len = hdr_size >> 2; /* first pull in the header so we know the buffer length */ for (bi = 0; bi < dword_len; bi++) buffer[bi] = rd32a(hw, TXGBE_MNGMBX, bi); /* * If there is any thing in data position pull it in * Read Flash command requires reading buffer length from * two byes instead of one byte */ if (resp->cmd == 0x30) { for (; bi < dword_len + 2; bi++) buffer[bi] = rd32a(hw, TXGBE_MNGMBX, bi); buf_len = (((u16)(resp->cmd_or_resp.ret_status) << 3) & 0xF00) | resp->buf_len; hdr_size += (2 << 2); } else { buf_len = resp->buf_len; } if (!buf_len) goto rel_out; if (length < buf_len + hdr_size) { DEBUGOUT("Buffer not large enough for reply message."); err = TXGBE_ERR_HOST_INTERFACE_COMMAND; goto rel_out; } /* Calculate length in DWORDs, add 3 for odd lengths */ dword_len = (buf_len + 3) >> 2; /* Pull in the rest of the buffer (bi is where we left off) */ for (; bi <= dword_len; bi++) buffer[bi] = rd32a(hw, TXGBE_MNGMBX, bi); rel_out: hw->mac.release_swfw_sync(hw, TXGBE_MNGSEM_SWMBX); return err; } /** * txgbe_hic_sr_read - Read EEPROM word using a host interface cmd * assuming that the semaphore is already obtained. * @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 hostif. **/ s32 txgbe_hic_sr_read(struct txgbe_hw *hw, u32 addr, u8 *buf, int len) { struct txgbe_hic_read_shadow_ram command; u32 value; int err, i = 0, j = 0; if (len > TXGBE_PMMBX_DATA_SIZE) return TXGBE_ERR_HOST_INTERFACE_COMMAND; memset(&command, 0, sizeof(command)); command.hdr.req.cmd = FW_READ_SHADOW_RAM_CMD; command.hdr.req.buf_lenh = 0; command.hdr.req.buf_lenl = FW_READ_SHADOW_RAM_LEN; command.hdr.req.checksum = FW_DEFAULT_CHECKSUM; command.address = cpu_to_be32(addr); command.length = cpu_to_be16(len); err = txgbe_hic_unlocked(hw, (u32 *)&command, sizeof(command), TXGBE_HI_COMMAND_TIMEOUT); if (err) return err; while (i < (len >> 2)) { value = rd32a(hw, TXGBE_MNGMBX, FW_NVM_DATA_OFFSET + i); ((u32 *)buf)[i] = value; i++; } value = rd32a(hw, TXGBE_MNGMBX, FW_NVM_DATA_OFFSET + i); for (i <<= 2; i < len; i++) ((u8 *)buf)[i] = ((u8 *)&value)[j++]; return 0; } /** * txgbe_hic_sr_write - Write EEPROM word using hostif * @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 hostif. **/ s32 txgbe_hic_sr_write(struct txgbe_hw *hw, u32 addr, u8 *buf, int len) { struct txgbe_hic_write_shadow_ram command; u32 value; int err = 0, i = 0, j = 0; if (len > TXGBE_PMMBX_DATA_SIZE) return TXGBE_ERR_HOST_INTERFACE_COMMAND; memset(&command, 0, sizeof(command)); command.hdr.req.cmd = FW_WRITE_SHADOW_RAM_CMD; command.hdr.req.buf_lenh = 0; command.hdr.req.buf_lenl = FW_WRITE_SHADOW_RAM_LEN; command.hdr.req.checksum = FW_DEFAULT_CHECKSUM; command.address = cpu_to_be32(addr); command.length = cpu_to_be16(len); while (i < (len >> 2)) { value = ((u32 *)buf)[i]; wr32a(hw, TXGBE_MNGMBX, FW_NVM_DATA_OFFSET + i, value); i++; } for (i <<= 2; i < len; i++) ((u8 *)&value)[j++] = ((u8 *)buf)[i]; wr32a(hw, TXGBE_MNGMBX, FW_NVM_DATA_OFFSET + (i >> 2), value); UNREFERENCED_PARAMETER(&command); return err; } /** * txgbe_hic_set_drv_ver - Sends driver version to firmware * @hw: pointer to the HW structure * @maj: driver version major number * @min: driver version minor number * @build: driver version build number * @sub: driver version sub build number * @len: unused * @driver_ver: unused * * Sends driver version number to firmware through the manageability * block. On success return 0 * else returns TXGBE_ERR_SWFW_SYNC when encountering an error acquiring * semaphore or TXGBE_ERR_HOST_INTERFACE_COMMAND when command fails. **/ s32 txgbe_hic_set_drv_ver(struct txgbe_hw *hw, u8 maj, u8 min, u8 build, u8 sub, u16 len, const char *driver_ver) { struct txgbe_hic_drv_info fw_cmd; int i; s32 ret_val = 0; UNREFERENCED_PARAMETER(len, driver_ver); fw_cmd.hdr.cmd = FW_CEM_CMD_DRIVER_INFO; fw_cmd.hdr.buf_len = FW_CEM_CMD_DRIVER_INFO_LEN; fw_cmd.hdr.cmd_or_resp.cmd_resv = FW_CEM_CMD_RESERVED; fw_cmd.port_num = (u8)hw->bus.func; fw_cmd.ver_maj = maj; fw_cmd.ver_min = min; fw_cmd.ver_build = build; fw_cmd.ver_sub = sub; fw_cmd.hdr.checksum = 0; fw_cmd.pad = 0; fw_cmd.pad2 = 0; fw_cmd.hdr.checksum = txgbe_calculate_checksum((u8 *)&fw_cmd, (FW_CEM_HDR_LEN + fw_cmd.hdr.buf_len)); for (i = 0; i <= FW_CEM_MAX_RETRIES; i++) { ret_val = txgbe_host_interface_command(hw, (u32 *)&fw_cmd, sizeof(fw_cmd), TXGBE_HI_COMMAND_TIMEOUT, true); if (ret_val != 0) continue; if (fw_cmd.hdr.cmd_or_resp.ret_status == FW_CEM_RESP_STATUS_SUCCESS) ret_val = 0; else ret_val = TXGBE_ERR_HOST_INTERFACE_COMMAND; break; } return ret_val; } /** * txgbe_hic_reset - send reset cmd to fw * @hw: pointer to hardware structure * * Sends reset cmd to firmware through the manageability * block. On success return 0 * else returns TXGBE_ERR_SWFW_SYNC when encountering an error acquiring * semaphore or TXGBE_ERR_HOST_INTERFACE_COMMAND when command fails. **/ s32 txgbe_hic_reset(struct txgbe_hw *hw) { struct txgbe_hic_reset reset_cmd; int i; s32 err = 0; reset_cmd.hdr.cmd = FW_RESET_CMD; reset_cmd.hdr.buf_len = FW_RESET_LEN; reset_cmd.hdr.cmd_or_resp.cmd_resv = FW_CEM_CMD_RESERVED; reset_cmd.lan_id = hw->bus.lan_id; reset_cmd.reset_type = (u16)hw->reset_type; reset_cmd.hdr.checksum = 0; reset_cmd.hdr.checksum = txgbe_calculate_checksum((u8 *)&reset_cmd, (FW_CEM_HDR_LEN + reset_cmd.hdr.buf_len)); for (i = 0; i <= FW_CEM_MAX_RETRIES; i++) { err = txgbe_host_interface_command(hw, (u32 *)&reset_cmd, sizeof(reset_cmd), TXGBE_HI_COMMAND_TIMEOUT, true); if (err != 0) continue; if (reset_cmd.hdr.cmd_or_resp.ret_status == FW_CEM_RESP_STATUS_SUCCESS) err = 0; else err = TXGBE_ERR_HOST_INTERFACE_COMMAND; break; } return err; } /** * txgbe_mng_present - returns true when management capability is present * @hw: pointer to hardware structure */ bool txgbe_mng_present(struct txgbe_hw *hw) { if (hw->mac.type == txgbe_mac_unknown) return false; return !!rd32m(hw, TXGBE_STAT, TXGBE_STAT_MNGINIT); } /** * txgbe_mng_enabled - Is the manageability engine enabled? * @hw: pointer to hardware structure * * Returns true if the manageability engine is enabled. **/ bool txgbe_mng_enabled(struct txgbe_hw *hw) { UNREFERENCED_PARAMETER(hw); /* firmware does not control laser */ return false; }