Commit 2e8320f7 authored by Dave Jiang's avatar Dave Jiang Committed by Dan Williams

isci: Fixup for OEM parameter EFI variable retrieval

Updating the EFI variable OEM parameter retrieval after examining the EFI
variable exported via sysfs.
Signed-off-by: default avatarDave Jiang <dave.jiang@intel.com>
Signed-off-by: default avatarDan Williams <dan.j.williams@intel.com>
parent 3b67c1f3
...@@ -61,6 +61,11 @@ struct isci_orom *isci_request_oprom(struct pci_dev *pdev) ...@@ -61,6 +61,11 @@ struct isci_orom *isci_request_oprom(struct pci_dev *pdev)
len = pci_biosrom_size(pdev); len = pci_biosrom_size(pdev);
rom = devm_kzalloc(&pdev->dev, sizeof(*rom), GFP_KERNEL); rom = devm_kzalloc(&pdev->dev, sizeof(*rom), GFP_KERNEL);
if (!rom) {
dev_warn(&pdev->dev,
"Unable to allocate memory for orom\n");
return NULL;
}
for (i = 0; i < len && rom; i += ISCI_OEM_SIG_SIZE) { for (i = 0; i < len && rom; i += ISCI_OEM_SIG_SIZE) {
memcpy_fromio(oem_sig, oprom + i, ISCI_OEM_SIG_SIZE); memcpy_fromio(oem_sig, oprom + i, ISCI_OEM_SIG_SIZE);
...@@ -181,7 +186,11 @@ struct isci_orom *isci_get_efi_var(struct pci_dev *pdev) ...@@ -181,7 +186,11 @@ struct isci_orom *isci_get_efi_var(struct pci_dev *pdev)
{ {
struct efi_variable *evar; struct efi_variable *evar;
efi_status_t status; efi_status_t status;
struct isci_orom *orom = NULL; struct isci_orom *rom = NULL;
struct isci_oem_hdr *oem_hdr;
u8 *tmp, sum;
int j;
size_t copy_len;
evar = devm_kzalloc(&pdev->dev, evar = devm_kzalloc(&pdev->dev,
sizeof(struct efi_variable), sizeof(struct efi_variable),
...@@ -192,6 +201,16 @@ struct isci_orom *isci_get_efi_var(struct pci_dev *pdev) ...@@ -192,6 +201,16 @@ struct isci_orom *isci_get_efi_var(struct pci_dev *pdev)
return NULL; return NULL;
} }
rom = devm_kzalloc(&pdev->dev, sizeof(*rom), GFP_KERNEL);
if (!rom) {
dev_warn(&pdev->dev,
"Unable to allocate memory for orom\n");
return NULL;
}
for (j = 0; j < strlen(ISCI_EFI_VAR_NAME) + 1; j++)
evar->VariableName[j] = ISCI_EFI_VAR_NAME[j];
evar->DataSize = 1024; evar->DataSize = 1024;
evar->VendorGuid = ISCI_EFI_VENDOR_GUID; evar->VendorGuid = ISCI_EFI_VENDOR_GUID;
evar->Attributes = ISCI_EFI_ATTRIBUTES; evar->Attributes = ISCI_EFI_ATTRIBUTES;
...@@ -205,19 +224,48 @@ struct isci_orom *isci_get_efi_var(struct pci_dev *pdev) ...@@ -205,19 +224,48 @@ struct isci_orom *isci_get_efi_var(struct pci_dev *pdev)
else else
status = EFI_NOT_FOUND; status = EFI_NOT_FOUND;
if (status == EFI_SUCCESS) if (status != EFI_SUCCESS) {
orom = (struct isci_orom *)evar->Data;
else
dev_warn(&pdev->dev, dev_warn(&pdev->dev,
"Unable to obtain EFI variable for OEM parms\n"); "Unable to obtain EFI variable for OEM parms\n");
return NULL;
}
oem_hdr = (struct isci_oem_hdr *)evar->Data;
if (orom && memcmp(orom->hdr.signature, ISCI_ROM_SIG, if (memcmp(oem_hdr->sig, ISCI_OEM_SIG, ISCI_OEM_SIG_SIZE) != 0) {
strlen(ISCI_ROM_SIG)) != 0)
dev_warn(&pdev->dev, dev_warn(&pdev->dev,
"Verifying OROM signature failed\n"); "Invalid OEM header signature\n");
return NULL;
}
if (!orom) /* calculate checksum */
devm_kfree(&pdev->dev, evar); tmp = (u8 *)oem_hdr;
for (j = 0, sum = 0; j < sizeof(oem_hdr); j++, tmp++)
sum += *tmp;
return orom; tmp = (u8 *)rom;
for (j = 0; j < sizeof(*rom); j++, tmp++)
sum += *tmp;
if (sum != 0) {
dev_warn(&pdev->dev,
"OEM table checksum failed\n");
return NULL;
}
copy_len = min(evar->DataSize,
min(oem_hdr->len - sizeof(*oem_hdr),
sizeof(*rom)));
memcpy(rom, (char *)evar->Data + sizeof(*oem_hdr), copy_len);
if (memcmp(rom->hdr.signature,
ISCI_ROM_SIG,
ISCI_ROM_SIG_SIZE) != 0) {
dev_warn(&pdev->dev,
"Invalid OEM table signature\n");
return NULL;
}
return rom;
} }
...@@ -100,7 +100,7 @@ struct isci_oem_hdr { ...@@ -100,7 +100,7 @@ struct isci_oem_hdr {
EFI_GUID(0x193dfefa, 0xa445, 0x4302, 0x99, 0xd8, 0xef, 0x3a, 0xad, \ EFI_GUID(0x193dfefa, 0xa445, 0x4302, 0x99, 0xd8, 0xef, 0x3a, 0xad, \
0x1a, 0x04, 0xc6) 0x1a, 0x04, 0xc6)
#define ISCI_EFI_ATTRIBUTES 0 #define ISCI_EFI_ATTRIBUTES 0
#define ISCI_EFI_VAR_NAME "RST_SCU_OEM" #define ISCI_EFI_VAR_NAME "RstScuO"
/* Allowed PORT configuration modes APC Automatic PORT configuration mode is /* Allowed PORT configuration modes APC Automatic PORT configuration mode is
* defined by the OEM configuration parameters providing no PHY_MASK parameters * defined by the OEM configuration parameters providing no PHY_MASK parameters
......
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