Commit c165541e authored by Padmanabh Ratnakar's avatar Padmanabh Ratnakar Committed by David S. Miller

be2net: Fix FW download for BE

Skip flashing a FW component if that component is not present in a
particular FW UFI image.
Signed-off-by: default avatarSomnath Kotur <somnath.kotur@emulex.com>
Signed-off-by: default avatarPadmanabh Ratnakar <padmanabh.ratnakar@emulex.com>
Signed-off-by: default avatarDavid S. Miller <davem@davemloft.net>
parent 652bf646
...@@ -1979,7 +1979,7 @@ int be_cmd_get_flash_crc(struct be_adapter *adapter, u8 *flashed_crc, ...@@ -1979,7 +1979,7 @@ int be_cmd_get_flash_crc(struct be_adapter *adapter, u8 *flashed_crc,
be_wrb_cmd_hdr_prepare(&req->hdr, CMD_SUBSYSTEM_COMMON, be_wrb_cmd_hdr_prepare(&req->hdr, CMD_SUBSYSTEM_COMMON,
OPCODE_COMMON_READ_FLASHROM, sizeof(*req)+4, wrb, NULL); OPCODE_COMMON_READ_FLASHROM, sizeof(*req)+4, wrb, NULL);
req->params.op_type = cpu_to_le32(IMG_TYPE_REDBOOT); req->params.op_type = cpu_to_le32(OPTYPE_REDBOOT);
req->params.op_code = cpu_to_le32(FLASHROM_OPER_REPORT); req->params.op_code = cpu_to_le32(FLASHROM_OPER_REPORT);
req->params.offset = cpu_to_le32(offset); req->params.offset = cpu_to_le32(offset);
req->params.data_buf_size = cpu_to_le32(0x4); req->params.data_buf_size = cpu_to_le32(0x4);
......
...@@ -162,22 +162,23 @@ ...@@ -162,22 +162,23 @@
#define QUERY_FAT 1 #define QUERY_FAT 1
/* Flashrom related descriptors */ /* Flashrom related descriptors */
#define MAX_FLASH_COMP 32
#define IMAGE_TYPE_FIRMWARE 160 #define IMAGE_TYPE_FIRMWARE 160
#define IMAGE_TYPE_BOOTCODE 224 #define IMAGE_TYPE_BOOTCODE 224
#define IMAGE_TYPE_OPTIONROM 32 #define IMAGE_TYPE_OPTIONROM 32
#define NUM_FLASHDIR_ENTRIES 32 #define NUM_FLASHDIR_ENTRIES 32
#define IMG_TYPE_ISCSI_ACTIVE 0 #define OPTYPE_ISCSI_ACTIVE 0
#define IMG_TYPE_REDBOOT 1 #define OPTYPE_REDBOOT 1
#define IMG_TYPE_BIOS 2 #define OPTYPE_BIOS 2
#define IMG_TYPE_PXE_BIOS 3 #define OPTYPE_PXE_BIOS 3
#define IMG_TYPE_FCOE_BIOS 8 #define OPTYPE_FCOE_BIOS 8
#define IMG_TYPE_ISCSI_BACKUP 9 #define OPTYPE_ISCSI_BACKUP 9
#define IMG_TYPE_FCOE_FW_ACTIVE 10 #define OPTYPE_FCOE_FW_ACTIVE 10
#define IMG_TYPE_FCOE_FW_BACKUP 11 #define OPTYPE_FCOE_FW_BACKUP 11
#define IMG_TYPE_NCSI_FW 13 #define OPTYPE_NCSI_FW 13
#define IMG_TYPE_PHY_FW 99 #define OPTYPE_PHY_FW 99
#define TN_8022 13 #define TN_8022 13
#define ILLEGAL_IOCTL_REQ 2 #define ILLEGAL_IOCTL_REQ 2
...@@ -223,6 +224,24 @@ ...@@ -223,6 +224,24 @@
#define FLASH_REDBOOT_START_g3 (262144) #define FLASH_REDBOOT_START_g3 (262144)
#define FLASH_PHY_FW_START_g3 1310720 #define FLASH_PHY_FW_START_g3 1310720
#define IMAGE_NCSI 16
#define IMAGE_OPTION_ROM_PXE 32
#define IMAGE_OPTION_ROM_FCoE 33
#define IMAGE_OPTION_ROM_ISCSI 34
#define IMAGE_FLASHISM_JUMPVECTOR 48
#define IMAGE_FLASH_ISM 49
#define IMAGE_JUMP_VECTOR 50
#define IMAGE_FIRMWARE_iSCSI 160
#define IMAGE_FIRMWARE_COMP_iSCSI 161
#define IMAGE_FIRMWARE_FCoE 162
#define IMAGE_FIRMWARE_COMP_FCoE 163
#define IMAGE_FIRMWARE_BACKUP_iSCSI 176
#define IMAGE_FIRMWARE_BACKUP_COMP_iSCSI 177
#define IMAGE_FIRMWARE_BACKUP_FCoE 178
#define IMAGE_FIRMWARE_BACKUP_COMP_FCoE 179
#define IMAGE_FIRMWARE_PHY 192
#define IMAGE_BOOT_CODE 224
/************* Rx Packet Type Encoding **************/ /************* Rx Packet Type Encoding **************/
#define BE_UNICAST_PACKET 0 #define BE_UNICAST_PACKET 0
#define BE_MULTICAST_PACKET 1 #define BE_MULTICAST_PACKET 1
...@@ -445,6 +464,7 @@ struct flash_comp { ...@@ -445,6 +464,7 @@ struct flash_comp {
unsigned long offset; unsigned long offset;
int optype; int optype;
int size; int size;
int img_type;
}; };
struct image_hdr { struct image_hdr {
...@@ -481,17 +501,19 @@ struct flash_section_hdr { ...@@ -481,17 +501,19 @@ struct flash_section_hdr {
u32 format_rev; u32 format_rev;
u32 cksum; u32 cksum;
u32 antidote; u32 antidote;
u32 build_no; u32 num_images;
u8 id_string[64]; u8 id_string[128];
u32 active_entry_mask; u32 rsvd[4];
u32 valid_entry_mask; } __packed;
u32 org_content_mask;
u32 rsvd0; struct flash_section_hdr_g2 {
u32 rsvd1; u32 format_rev;
u32 rsvd2; u32 cksum;
u32 rsvd3; u32 antidote;
u32 rsvd4; u32 build_num;
}; u8 id_string[128];
u32 rsvd[8];
} __packed;
struct flash_section_entry { struct flash_section_entry {
u32 type; u32 type;
...@@ -503,10 +525,16 @@ struct flash_section_entry { ...@@ -503,10 +525,16 @@ struct flash_section_entry {
u32 rsvd0; u32 rsvd0;
u32 rsvd1; u32 rsvd1;
u8 ver_data[32]; u8 ver_data[32];
}; } __packed;
struct flash_section_info { struct flash_section_info {
u8 cookie[32]; u8 cookie[32];
struct flash_section_hdr fsec_hdr; struct flash_section_hdr fsec_hdr;
struct flash_section_entry fsec_entry[32]; struct flash_section_entry fsec_entry[32];
}; } __packed;
struct flash_section_info_g2 {
u8 cookie[32];
struct flash_section_hdr_g2 fsec_hdr;
struct flash_section_entry fsec_entry[32];
} __packed;
...@@ -2759,6 +2759,8 @@ static void be_netpoll(struct net_device *netdev) ...@@ -2759,6 +2759,8 @@ static void be_netpoll(struct net_device *netdev)
#endif #endif
#define FW_FILE_HDR_SIGN "ServerEngines Corp. " #define FW_FILE_HDR_SIGN "ServerEngines Corp. "
char flash_cookie[2][16] = {"*** SE FLAS", "H DIRECTORY *** "};
static bool be_flash_redboot(struct be_adapter *adapter, static bool be_flash_redboot(struct be_adapter *adapter,
const u8 *p, u32 img_start, int image_size, const u8 *p, u32 img_start, int image_size,
int hdr_size) int hdr_size)
...@@ -2792,58 +2794,101 @@ static bool phy_flashing_required(struct be_adapter *adapter) ...@@ -2792,58 +2794,101 @@ static bool phy_flashing_required(struct be_adapter *adapter)
adapter->phy.interface_type == PHY_TYPE_BASET_10GB); adapter->phy.interface_type == PHY_TYPE_BASET_10GB);
} }
static bool is_comp_in_ufi(struct be_adapter *adapter,
struct flash_section_info *fsec, int type)
{
int i = 0, img_type = 0;
struct flash_section_info_g2 *fsec_g2 = NULL;
if (adapter->generation != BE_GEN3)
fsec_g2 = (struct flash_section_info_g2 *)fsec;
for (i = 0; i < MAX_FLASH_COMP; i++) {
if (fsec_g2)
img_type = le32_to_cpu(fsec_g2->fsec_entry[i].type);
else
img_type = le32_to_cpu(fsec->fsec_entry[i].type);
if (img_type == type)
return true;
}
return false;
}
struct flash_section_info *get_fsec_info(struct be_adapter *adapter,
int header_size,
const struct firmware *fw)
{
struct flash_section_info *fsec = NULL;
const u8 *p = fw->data;
p += header_size;
while (p < (fw->data + fw->size)) {
fsec = (struct flash_section_info *)p;
if (!memcmp(flash_cookie, fsec->cookie, sizeof(flash_cookie)))
return fsec;
p += 32;
}
return NULL;
}
static int be_flash_data(struct be_adapter *adapter, static int be_flash_data(struct be_adapter *adapter,
const struct firmware *fw, const struct firmware *fw,
struct be_dma_mem *flash_cmd, int num_of_images) struct be_dma_mem *flash_cmd,
int num_of_images)
{ {
int status = 0, i, filehdr_size = 0; int status = 0, i, filehdr_size = 0;
int img_hdrs_size = (num_of_images * sizeof(struct image_hdr));
u32 total_bytes = 0, flash_op; u32 total_bytes = 0, flash_op;
int num_bytes; int num_bytes;
const u8 *p = fw->data; const u8 *p = fw->data;
struct be_cmd_write_flashrom *req = flash_cmd->va; struct be_cmd_write_flashrom *req = flash_cmd->va;
const struct flash_comp *pflashcomp; const struct flash_comp *pflashcomp;
int num_comp; int num_comp, hdr_size;
struct flash_section_info *fsec = NULL;
static const struct flash_comp gen3_flash_types[10] = {
{ FLASH_iSCSI_PRIMARY_IMAGE_START_g3, IMG_TYPE_ISCSI_ACTIVE, struct flash_comp gen3_flash_types[] = {
FLASH_IMAGE_MAX_SIZE_g3}, { FLASH_iSCSI_PRIMARY_IMAGE_START_g3, OPTYPE_ISCSI_ACTIVE,
{ FLASH_REDBOOT_START_g3, IMG_TYPE_REDBOOT, FLASH_IMAGE_MAX_SIZE_g3, IMAGE_FIRMWARE_iSCSI},
FLASH_REDBOOT_IMAGE_MAX_SIZE_g3}, { FLASH_REDBOOT_START_g3, OPTYPE_REDBOOT,
{ FLASH_iSCSI_BIOS_START_g3, IMG_TYPE_BIOS, FLASH_REDBOOT_IMAGE_MAX_SIZE_g3, IMAGE_BOOT_CODE},
FLASH_BIOS_IMAGE_MAX_SIZE_g3}, { FLASH_iSCSI_BIOS_START_g3, OPTYPE_BIOS,
{ FLASH_PXE_BIOS_START_g3, IMG_TYPE_PXE_BIOS, FLASH_BIOS_IMAGE_MAX_SIZE_g3, IMAGE_OPTION_ROM_ISCSI},
FLASH_BIOS_IMAGE_MAX_SIZE_g3}, { FLASH_PXE_BIOS_START_g3, OPTYPE_PXE_BIOS,
{ FLASH_FCoE_BIOS_START_g3, IMG_TYPE_FCOE_BIOS, FLASH_BIOS_IMAGE_MAX_SIZE_g3, IMAGE_OPTION_ROM_PXE},
FLASH_BIOS_IMAGE_MAX_SIZE_g3}, { FLASH_FCoE_BIOS_START_g3, OPTYPE_FCOE_BIOS,
{ FLASH_iSCSI_BACKUP_IMAGE_START_g3, IMG_TYPE_ISCSI_BACKUP, FLASH_BIOS_IMAGE_MAX_SIZE_g3, IMAGE_OPTION_ROM_FCoE},
FLASH_IMAGE_MAX_SIZE_g3}, { FLASH_iSCSI_BACKUP_IMAGE_START_g3, OPTYPE_ISCSI_BACKUP,
{ FLASH_FCoE_PRIMARY_IMAGE_START_g3, IMG_TYPE_FCOE_FW_ACTIVE, FLASH_IMAGE_MAX_SIZE_g3, IMAGE_FIRMWARE_BACKUP_iSCSI},
FLASH_IMAGE_MAX_SIZE_g3}, { FLASH_FCoE_PRIMARY_IMAGE_START_g3, OPTYPE_FCOE_FW_ACTIVE,
{ FLASH_FCoE_BACKUP_IMAGE_START_g3, IMG_TYPE_FCOE_FW_BACKUP, FLASH_IMAGE_MAX_SIZE_g3, IMAGE_FIRMWARE_FCoE},
FLASH_IMAGE_MAX_SIZE_g3}, { FLASH_FCoE_BACKUP_IMAGE_START_g3, OPTYPE_FCOE_FW_BACKUP,
{ FLASH_NCSI_START_g3, IMG_TYPE_NCSI_FW, FLASH_IMAGE_MAX_SIZE_g3, IMAGE_FIRMWARE_BACKUP_FCoE},
FLASH_NCSI_IMAGE_MAX_SIZE_g3}, { FLASH_NCSI_START_g3, OPTYPE_NCSI_FW,
{ FLASH_PHY_FW_START_g3, IMG_TYPE_PHY_FW, FLASH_NCSI_IMAGE_MAX_SIZE_g3, IMAGE_NCSI},
FLASH_PHY_FW_IMAGE_MAX_SIZE_g3} { FLASH_PHY_FW_START_g3, OPTYPE_PHY_FW,
FLASH_PHY_FW_IMAGE_MAX_SIZE_g3, IMAGE_FIRMWARE_PHY}
}; };
static const struct flash_comp gen2_flash_types[8] = {
{ FLASH_iSCSI_PRIMARY_IMAGE_START_g2, IMG_TYPE_ISCSI_ACTIVE, struct flash_comp gen2_flash_types[] = {
FLASH_IMAGE_MAX_SIZE_g2}, { FLASH_iSCSI_PRIMARY_IMAGE_START_g2, OPTYPE_ISCSI_ACTIVE,
{ FLASH_REDBOOT_START_g2, IMG_TYPE_REDBOOT, FLASH_IMAGE_MAX_SIZE_g2, IMAGE_FIRMWARE_iSCSI},
FLASH_REDBOOT_IMAGE_MAX_SIZE_g2}, { FLASH_REDBOOT_START_g2, OPTYPE_REDBOOT,
{ FLASH_iSCSI_BIOS_START_g2, IMG_TYPE_BIOS, FLASH_REDBOOT_IMAGE_MAX_SIZE_g2, IMAGE_BOOT_CODE},
FLASH_BIOS_IMAGE_MAX_SIZE_g2}, { FLASH_iSCSI_BIOS_START_g2, OPTYPE_BIOS,
{ FLASH_PXE_BIOS_START_g2, IMG_TYPE_PXE_BIOS, FLASH_BIOS_IMAGE_MAX_SIZE_g2, IMAGE_OPTION_ROM_ISCSI},
FLASH_BIOS_IMAGE_MAX_SIZE_g2}, { FLASH_PXE_BIOS_START_g2, OPTYPE_PXE_BIOS,
{ FLASH_FCoE_BIOS_START_g2, IMG_TYPE_FCOE_BIOS, FLASH_BIOS_IMAGE_MAX_SIZE_g2, IMAGE_OPTION_ROM_PXE},
FLASH_BIOS_IMAGE_MAX_SIZE_g2}, { FLASH_FCoE_BIOS_START_g2, OPTYPE_FCOE_BIOS,
{ FLASH_iSCSI_BACKUP_IMAGE_START_g2, IMG_TYPE_ISCSI_BACKUP, FLASH_BIOS_IMAGE_MAX_SIZE_g2, IMAGE_OPTION_ROM_FCoE},
FLASH_IMAGE_MAX_SIZE_g2}, { FLASH_iSCSI_BACKUP_IMAGE_START_g2, OPTYPE_ISCSI_BACKUP,
{ FLASH_FCoE_PRIMARY_IMAGE_START_g2, IMG_TYPE_FCOE_FW_ACTIVE, FLASH_IMAGE_MAX_SIZE_g2, IMAGE_FIRMWARE_BACKUP_iSCSI},
FLASH_IMAGE_MAX_SIZE_g2}, { FLASH_FCoE_PRIMARY_IMAGE_START_g2, OPTYPE_FCOE_FW_ACTIVE,
{ FLASH_FCoE_BACKUP_IMAGE_START_g2, IMG_TYPE_FCOE_FW_BACKUP, FLASH_IMAGE_MAX_SIZE_g2, IMAGE_FIRMWARE_FCoE},
FLASH_IMAGE_MAX_SIZE_g2} { FLASH_FCoE_BACKUP_IMAGE_START_g2, OPTYPE_FCOE_FW_BACKUP,
FLASH_IMAGE_MAX_SIZE_g2, IMAGE_FIRMWARE_BACKUP_FCoE}
}; };
if (adapter->generation == BE_GEN3) { if (adapter->generation == BE_GEN3) {
...@@ -2855,22 +2900,37 @@ static int be_flash_data(struct be_adapter *adapter, ...@@ -2855,22 +2900,37 @@ static int be_flash_data(struct be_adapter *adapter,
filehdr_size = sizeof(struct flash_file_hdr_g2); filehdr_size = sizeof(struct flash_file_hdr_g2);
num_comp = ARRAY_SIZE(gen2_flash_types); num_comp = ARRAY_SIZE(gen2_flash_types);
} }
/* Get flash section info*/
fsec = get_fsec_info(adapter, filehdr_size + img_hdrs_size, fw);
if (!fsec) {
dev_err(&adapter->pdev->dev,
"Invalid Cookie. UFI corrupted ?\n");
return -1;
}
for (i = 0; i < num_comp; i++) { for (i = 0; i < num_comp; i++) {
if ((pflashcomp[i].optype == IMG_TYPE_NCSI_FW) && if (!is_comp_in_ufi(adapter, fsec, pflashcomp[i].img_type))
continue;
if ((pflashcomp[i].optype == OPTYPE_NCSI_FW) &&
memcmp(adapter->fw_ver, "3.102.148.0", 11) < 0) memcmp(adapter->fw_ver, "3.102.148.0", 11) < 0)
continue; continue;
if (pflashcomp[i].optype == IMG_TYPE_PHY_FW) {
if (pflashcomp[i].optype == OPTYPE_PHY_FW) {
if (!phy_flashing_required(adapter)) if (!phy_flashing_required(adapter))
continue; continue;
} }
if ((pflashcomp[i].optype == IMG_TYPE_REDBOOT) &&
(!be_flash_redboot(adapter, fw->data, hdr_size = filehdr_size +
pflashcomp[i].offset, pflashcomp[i].size, filehdr_size + (num_of_images * sizeof(struct image_hdr));
(num_of_images * sizeof(struct image_hdr)))))
if ((pflashcomp[i].optype == OPTYPE_REDBOOT) &&
(!be_flash_redboot(adapter, fw->data, pflashcomp[i].offset,
pflashcomp[i].size, hdr_size)))
continue; continue;
/* Flash the component */
p = fw->data; p = fw->data;
p += filehdr_size + pflashcomp[i].offset p += filehdr_size + pflashcomp[i].offset + img_hdrs_size;
+ (num_of_images * sizeof(struct image_hdr));
if (p + pflashcomp[i].size > fw->data + fw->size) if (p + pflashcomp[i].size > fw->data + fw->size)
return -1; return -1;
total_bytes = pflashcomp[i].size; total_bytes = pflashcomp[i].size;
...@@ -2881,12 +2941,12 @@ static int be_flash_data(struct be_adapter *adapter, ...@@ -2881,12 +2941,12 @@ static int be_flash_data(struct be_adapter *adapter,
num_bytes = total_bytes; num_bytes = total_bytes;
total_bytes -= num_bytes; total_bytes -= num_bytes;
if (!total_bytes) { if (!total_bytes) {
if (pflashcomp[i].optype == IMG_TYPE_PHY_FW) if (pflashcomp[i].optype == OPTYPE_PHY_FW)
flash_op = FLASHROM_OPER_PHY_FLASH; flash_op = FLASHROM_OPER_PHY_FLASH;
else else
flash_op = FLASHROM_OPER_FLASH; flash_op = FLASHROM_OPER_FLASH;
} else { } else {
if (pflashcomp[i].optype == IMG_TYPE_PHY_FW) if (pflashcomp[i].optype == OPTYPE_PHY_FW)
flash_op = FLASHROM_OPER_PHY_SAVE; flash_op = FLASHROM_OPER_PHY_SAVE;
else else
flash_op = FLASHROM_OPER_SAVE; flash_op = FLASHROM_OPER_SAVE;
...@@ -2898,7 +2958,7 @@ static int be_flash_data(struct be_adapter *adapter, ...@@ -2898,7 +2958,7 @@ static int be_flash_data(struct be_adapter *adapter,
if (status) { if (status) {
if ((status == ILLEGAL_IOCTL_REQ) && if ((status == ILLEGAL_IOCTL_REQ) &&
(pflashcomp[i].optype == (pflashcomp[i].optype ==
IMG_TYPE_PHY_FW)) OPTYPE_PHY_FW))
break; break;
dev_err(&adapter->pdev->dev, dev_err(&adapter->pdev->dev,
"cmd to write to flash rom failed.\n"); "cmd to write to flash rom failed.\n");
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment