soc/amd/common/block/pi: Fix AGESA heap deallocator

The deallocation was always subtracting the header, even when it
shouldn't. This caused problems for the allocator where buffer
sizes were incorrect and freed and used buffers could collide.
Fix the deallocation size.

Clear deallocated concatinated buffer header memory.

Fix the initial calculation of the total buffer size
available to be allocated.

BUG=b:71764350
TEST= Boot grunt.
BRANCH=none

Change-Id: I2789ddf72d662f24709dc5d9873741169cc4ef36
Signed-off-by: Marc Jones <marcj303@gmail.com>
Reviewed-on: https://review.coreboot.org/23284
Tested-by: build bot (Jenkins) <no-reply@coreboot.org>
Reviewed-by: Aaron Durbin <adurbin@chromium.org>
This commit is contained in:
Marc Jones 2018-01-15 18:47:24 -07:00 committed by Martin Roth
parent 2bd6939dc5
commit e6033ce179

View file

@ -77,8 +77,8 @@ AGESA_STATUS agesa_AllocateBuffer (UINT32 Func, UINTN Data, VOID *ConfigPtr)
FreedNodePtr = (BIOS_BUFFER_NODE *)(BiosHeapBaseAddr FreedNodePtr = (BIOS_BUFFER_NODE *)(BiosHeapBaseAddr
+ FreedNodeOffset); + FreedNodeOffset);
FreedNodePtr->BufferSize = AvailableHeapSize FreedNodePtr->BufferSize = AvailableHeapSize
- sizeof(BIOS_BUFFER_NODE) - (FreedNodeOffset - CurrNodeOffset)
- CurrNodePtr->BufferSize; - sizeof(BIOS_BUFFER_NODE);
FreedNodePtr->NextNodeOffset = 0; FreedNodePtr->NextNodeOffset = 0;
/* Update the offsets for Allocated and Freed nodes */ /* Update the offsets for Allocated and Freed nodes */
@ -250,13 +250,13 @@ AGESA_STATUS agesa_DeallocateBuffer (UINT32 Func, UINTN Data, VOID *ConfigPtr)
memset((UINT8 *)AllocNodePtr + sizeof(BIOS_BUFFER_NODE), 0, memset((UINT8 *)AllocNodePtr + sizeof(BIOS_BUFFER_NODE), 0,
AllocNodePtr->BufferSize); AllocNodePtr->BufferSize);
AllocNodePtr->BufferHandle = 0; AllocNodePtr->BufferHandle = 0;
AllocNodePtr->BufferSize += sizeof(BIOS_BUFFER_NODE);
/* Add deallocated node in order to the list of freed nodes */ /* Add deallocated node in order to the list of freed nodes */
FreedNodeOffset = BiosHeapBasePtr->StartOfFreedNodes; FreedNodeOffset = BiosHeapBasePtr->StartOfFreedNodes;
FreedNodePtr = (BIOS_BUFFER_NODE *)(BiosHeapBaseAddr + FreedNodeOffset); FreedNodePtr = (BIOS_BUFFER_NODE *)(BiosHeapBaseAddr + FreedNodeOffset);
EndNodeOffset = AllocNodeOffset + AllocNodePtr->BufferSize; EndNodeOffset = AllocNodeOffset + AllocNodePtr->BufferSize +
sizeof(BIOS_BUFFER_NODE);
if (AllocNodeOffset < FreedNodeOffset) { if (AllocNodeOffset < FreedNodeOffset) {
/* Add to the start of the freed list */ /* Add to the start of the freed list */
@ -264,16 +264,14 @@ AGESA_STATUS agesa_DeallocateBuffer (UINT32 Func, UINTN Data, VOID *ConfigPtr)
/* If the freed node is adjacent to the first node in /* If the freed node is adjacent to the first node in
* the list, concatenate both nodes * the list, concatenate both nodes
*/ */
AllocNodePtr->BufferSize += FreedNodePtr->BufferSize; AllocNodePtr->BufferSize += FreedNodePtr->BufferSize +
sizeof(BIOS_BUFFER_NODE);
AllocNodePtr->NextNodeOffset = AllocNodePtr->NextNodeOffset =
FreedNodePtr->NextNodeOffset; FreedNodePtr->NextNodeOffset;
/* Clear the BufferSize and NextNodeOffset of the /* Zero out the FreedNode header */
* previous first node memset((UINT8 *)FreedNodePtr, 0,
*/ sizeof(BIOS_BUFFER_NODE));
FreedNodePtr->BufferSize = 0;
FreedNodePtr->NextNodeOffset = 0;
} else { } else {
/* Otherwise, add freed node to the start of the list /* Otherwise, add freed node to the start of the list
* Update NextNodeOffset and BufferSize to include the * Update NextNodeOffset and BufferSize to include the
@ -304,12 +302,14 @@ AGESA_STATUS agesa_DeallocateBuffer (UINT32 Func, UINTN Data, VOID *ConfigPtr)
if (NextNodeOffset == EndNodeOffset) { if (NextNodeOffset == EndNodeOffset) {
NextNodePtr = (BIOS_BUFFER_NODE *)(BiosHeapBaseAddr NextNodePtr = (BIOS_BUFFER_NODE *)(BiosHeapBaseAddr
+ NextNodeOffset); + NextNodeOffset);
AllocNodePtr->BufferSize += NextNodePtr->BufferSize; AllocNodePtr->BufferSize += NextNodePtr->BufferSize +
sizeof(BIOS_BUFFER_NODE);
AllocNodePtr->NextNodeOffset = AllocNodePtr->NextNodeOffset =
NextNodePtr->NextNodeOffset; NextNodePtr->NextNodeOffset;
NextNodePtr->BufferSize = 0; /* Zero out the NextNode header */
NextNodePtr->NextNodeOffset = 0; memset((UINT8 *)NextNodePtr, 0,
sizeof(BIOS_BUFFER_NODE));
} else { } else {
/*AllocNodePtr->NextNodeOffset = /*AllocNodePtr->NextNodeOffset =
* FreedNodePtr->NextNodeOffset; */ * FreedNodePtr->NextNodeOffset; */
@ -321,14 +321,18 @@ AGESA_STATUS agesa_DeallocateBuffer (UINT32 Func, UINTN Data, VOID *ConfigPtr)
*/ */
PrevNodePtr = (BIOS_BUFFER_NODE *)(BiosHeapBaseAddr PrevNodePtr = (BIOS_BUFFER_NODE *)(BiosHeapBaseAddr
+ PrevNodeOffset); + PrevNodeOffset);
EndNodeOffset = PrevNodeOffset + PrevNodePtr->BufferSize; EndNodeOffset = PrevNodeOffset + PrevNodePtr->BufferSize +
sizeof(BIOS_BUFFER_NODE);
if (AllocNodeOffset == EndNodeOffset) { if (AllocNodeOffset == EndNodeOffset) {
PrevNodePtr->NextNodeOffset = PrevNodePtr->NextNodeOffset =
AllocNodePtr->NextNodeOffset; AllocNodePtr->NextNodeOffset;
PrevNodePtr->BufferSize += AllocNodePtr->BufferSize; PrevNodePtr->BufferSize += AllocNodePtr->BufferSize +
sizeof(BIOS_BUFFER_NODE);
AllocNodePtr->BufferSize = 0; /* Zero out the AllocNode header */
AllocNodePtr->NextNodeOffset = 0; memset((UINT8 *)AllocNodePtr, 0,
sizeof(BIOS_BUFFER_NODE));
} else { } else {
PrevNodePtr->NextNodeOffset = AllocNodeOffset; PrevNodePtr->NextNodeOffset = AllocNodeOffset;
} }