AGESA f15 f15tn f16kb: Add extra checks for incorrect SPD data

Make DMI data calculation fail-safe to incorrect SPD data.

Change-Id: Ica92850cc77e1f7cbf3e7e44717de42a03b93bbe
Signed-off-by: Konstantin Aladyshev <aladyshev22@gmail.com>
Reviewed-on: https://review.coreboot.org/20839
Reviewed-by: Kyösti Mälkki <kyosti.malkki@gmail.com>
Tested-by: build bot (Jenkins) <no-reply@coreboot.org>
Reviewed-by: Paul Menzel <paulepanter@users.sourceforge.net>
This commit is contained in:
Konstantin Aladyshev 2017-08-01 15:26:30 +03:00 committed by Martin Roth
parent 1cda0d0e50
commit c4f60f33bd
3 changed files with 90 additions and 48 deletions

View file

@ -315,6 +315,9 @@ MemFDMISupport3 (
DmiTable[DimmIndex].Attributes = 4; // Quad Rank Dimm DmiTable[DimmIndex].Attributes = 4; // Quad Rank Dimm
} }
if ((!BusWidth) || (!Width) || (!Rank))
DimmSize = 0xFFFF; // size is unknown
else
DimmSize = (UINT32) (Capacity / 8 * BusWidth / Width * Rank); DimmSize = (UINT32) (Capacity / 8 * BusWidth / Width * Rank);
if (DimmSize < 0x7FFF) { if (DimmSize < 0x7FFF) {
DmiTable[DimmIndex].MemorySize = (UINT16) DimmSize; DmiTable[DimmIndex].MemorySize = (UINT16) DimmSize;
@ -335,9 +338,19 @@ MemFDMISupport3 (
DmiTable[DimmIndex].DimmPresent = 1; DmiTable[DimmIndex].DimmPresent = 1;
// Speed (offset 15h) // Speed (offset 15h)
MTB_ps = ((INT32) SpdDataStructure[DimmIndex].Data[10] * 1000) / SpdDataStructure[DimmIndex].Data[11];
FTB_ps = (SpdDataStructure[DimmIndex].Data[9] >> 4) / (SpdDataStructure[DimmIndex].Data[9] & 0xF); // If SPD is wrong, division by 0 in DIMM speed calculation could reboot CPU
Value32 = (MTB_ps * SpdDataStructure[DimmIndex].Data[12]) + (FTB_ps * (INT8) SpdDataStructure[DimmIndex].Data[34]) ; // So avoid it by this check
if ((SpdDataStructure[DimmIndex].Data[11]==0) ||
((SpdDataStructure[DimmIndex].Data[9] & 0xF) == 0)) {
DmiTable[DimmIndex].Speed = 0; // Unknown
} else {
MTB_ps = ((INT32) SpdDataStructure[DimmIndex].Data[10] * 1000) /
SpdDataStructure[DimmIndex].Data[11];
FTB_ps = (SpdDataStructure[DimmIndex].Data[9] >> 4) /
(SpdDataStructure[DimmIndex].Data[9] & 0xF);
Value32 = (MTB_ps * SpdDataStructure[DimmIndex].Data[12]) +
(FTB_ps * (INT8) SpdDataStructure[DimmIndex].Data[34]);
if (Value32 <= 938) { if (Value32 <= 938) {
DmiTable[DimmIndex].Speed = 1067; // DDR3-2133 DmiTable[DimmIndex].Speed = 1067; // DDR3-2133
} else if (Value32 <= 1071) { } else if (Value32 <= 1071) {
@ -351,6 +364,7 @@ MemFDMISupport3 (
} else if (Value32 <= 2500) { } else if (Value32 <= 2500) {
DmiTable[DimmIndex].Speed = 400; // DDR3-800 DmiTable[DimmIndex].Speed = 400; // DDR3-800
} }
}
// Manufacturer (offset 17h) // Manufacturer (offset 17h)
ManufacturerIdCode = (UINT64) SpdDataStructure[DimmIndex].Data[118]; ManufacturerIdCode = (UINT64) SpdDataStructure[DimmIndex].Data[118];

View file

@ -330,6 +330,9 @@ MemFDMISupport3 (
DmiTable[DimmIndex].Attributes = 4; // Quad Rank Dimm DmiTable[DimmIndex].Attributes = 4; // Quad Rank Dimm
} }
if ((!BusWidth) || (!Width) || (!Rank))
DimmSize = 0xFFFF; // size is unknown
else
DimmSize = (UINT32) (Capacity / 8 * BusWidth / Width * Rank); DimmSize = (UINT32) (Capacity / 8 * BusWidth / Width * Rank);
if (DimmSize < 0x7FFF) { if (DimmSize < 0x7FFF) {
DmiTable[DimmIndex].MemorySize = (UINT16) DimmSize; DmiTable[DimmIndex].MemorySize = (UINT16) DimmSize;
@ -354,9 +357,19 @@ MemFDMISupport3 (
} }
// Speed (offset 15h) // Speed (offset 15h)
MTB_ps = ((INT32) SpdDataStructure[DimmIndex].Data[10] * 1000) / SpdDataStructure[DimmIndex].Data[11];
FTB_ps = (SpdDataStructure[DimmIndex].Data[9] >> 4) / (SpdDataStructure[DimmIndex].Data[9] & 0xF); // If SPD is wrong, division by 0 in DIMM speed calculation could reboot CPU
Value32 = (MTB_ps * SpdDataStructure[DimmIndex].Data[12]) + (FTB_ps * (INT8) SpdDataStructure[DimmIndex].Data[34]) ; // So avoid it by this check
if ((SpdDataStructure[DimmIndex].Data[11]==0) ||
((SpdDataStructure[DimmIndex].Data[9] & 0xF) == 0)) {
DmiTable[DimmIndex].Speed = 0; // Unknown
} else {
MTB_ps = ((INT32) SpdDataStructure[DimmIndex].Data[10] * 1000) /
SpdDataStructure[DimmIndex].Data[11];
FTB_ps = (SpdDataStructure[DimmIndex].Data[9] >> 4) /
(SpdDataStructure[DimmIndex].Data[9] & 0xF);
Value32 = (MTB_ps * SpdDataStructure[DimmIndex].Data[12]) +
(FTB_ps * (INT8) SpdDataStructure[DimmIndex].Data[34]);
if (Value32 <= 938) { if (Value32 <= 938) {
DmiTable[DimmIndex].Speed = 1067; // DDR3-2133 DmiTable[DimmIndex].Speed = 1067; // DDR3-2133
} else if (Value32 <= 1071) { } else if (Value32 <= 1071) {
@ -370,6 +383,7 @@ MemFDMISupport3 (
} else if (Value32 <= 2500) { } else if (Value32 <= 2500) {
DmiTable[DimmIndex].Speed = 400; // DDR3-800 DmiTable[DimmIndex].Speed = 400; // DDR3-800
} }
}
// Manufacturer (offset 17h) // Manufacturer (offset 17h)
ManufacturerIdCode = (UINT64) SpdDataStructure[DimmIndex].Data[118]; ManufacturerIdCode = (UINT64) SpdDataStructure[DimmIndex].Data[118];

View file

@ -380,6 +380,9 @@ MemFDMISupport3 (
DmiPhysicalDimmInfoTable->Attributes = 4; // Quad Rank Dimm DmiPhysicalDimmInfoTable->Attributes = 4; // Quad Rank Dimm
} }
if ((!BusWidth) || (!Width) || (!Rank))
DimmSize = 0xFFFF; // size is unknown
else
DimmSize = (UINT32) (Capacity / 8 * BusWidth / Width * Rank); DimmSize = (UINT32) (Capacity / 8 * BusWidth / Width * Rank);
if (DimmSize < 0x7FFF) { if (DimmSize < 0x7FFF) {
DmiPhysicalDimmInfoTable->MemorySize = (UINT16) DimmSize; DmiPhysicalDimmInfoTable->MemorySize = (UINT16) DimmSize;
@ -409,9 +412,19 @@ MemFDMISupport3 (
} }
// Speed (offset 15h) // Speed (offset 15h)
MTB_ps = ((INT32) SpdDataStructure->Data[10] * 1000) / SpdDataStructure->Data[11];
FTB_ps = (SpdDataStructure->Data[9] >> 4) / (SpdDataStructure->Data[9] & 0xF); // If SPD is wrong, division by 0 in DIMM speed calculation could reboot CPU
Value32 = (MTB_ps * SpdDataStructure->Data[12]) + (FTB_ps * (INT8) SpdDataStructure->Data[34]) ; // So avoid it by this check
if ((SpdDataStructure->Data[11]==0) ||
((SpdDataStructure->Data[9] & 0xF) == 0)) {
DmiPhysicalDimmInfoTable->Speed = 0; // Unknown
} else {
MTB_ps = ((INT32) SpdDataStructure->Data[10] * 1000) /
SpdDataStructure->Data[11];
FTB_ps = (SpdDataStructure->Data[9] >> 4) /
(SpdDataStructure->Data[9] & 0xF);
Value32 = (MTB_ps * SpdDataStructure->Data[12]) +
(FTB_ps * (INT8) SpdDataStructure->Data[34]);
if (Value32 <= 938) { if (Value32 <= 938) {
DmiPhysicalDimmInfoTable->Speed = 1067; // DDR3-2133 DmiPhysicalDimmInfoTable->Speed = 1067; // DDR3-2133
} else if (Value32 <= 1071) { } else if (Value32 <= 1071) {
@ -425,6 +438,7 @@ MemFDMISupport3 (
} else if (Value32 <= 2500) { } else if (Value32 <= 2500) {
DmiPhysicalDimmInfoTable->Speed = 400; // DDR3-800 DmiPhysicalDimmInfoTable->Speed = 400; // DDR3-800
} }
}
// Manufacturer (offset 17h) // Manufacturer (offset 17h)
ManufacturerIdCode = (UINT64) SpdDataStructure->Data[118]; ManufacturerIdCode = (UINT64) SpdDataStructure->Data[118];