Marc Jones | 21cde8b | 2017-05-07 16:47:36 -0600 | [diff] [blame] | 1 | /* |
| 2 | * This file is part of the coreboot project. |
| 3 | * |
| 4 | * This program is free software; you can redistribute it and/or modify |
| 5 | * it under the terms of the GNU General Public License as published by |
| 6 | * the Free Software Foundation; version 2 of the License. |
| 7 | * |
| 8 | * This program is distributed in the hope that it will be useful, |
| 9 | * but WITHOUT ANY WARRANTY; without even the implied warranty of |
| 10 | * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the |
| 11 | * GNU General Public License for more details. |
| 12 | */ |
| 13 | |
| 14 | |
Richard Spiegel | 0ad74ac | 2017-12-08 16:53:29 -0700 | [diff] [blame^] | 15 | #include <amdblocks/agesawrapper.h> |
Marc Jones | 21cde8b | 2017-05-07 16:47:36 -0600 | [diff] [blame] | 16 | #include <amdlib.h> |
| 17 | #include <arch/acpi.h> |
Richard Spiegel | 0ad74ac | 2017-12-08 16:53:29 -0700 | [diff] [blame^] | 18 | #include <amdblocks/BiosCallOuts.h> |
Marc Jones | 21cde8b | 2017-05-07 16:47:36 -0600 | [diff] [blame] | 19 | #include <cbmem.h> |
Marc Jones | 21cde8b | 2017-05-07 16:47:36 -0600 | [diff] [blame] | 20 | #include <string.h> |
| 21 | |
Marshall Dawson | ed501d5b | 2017-09-18 16:13:17 -0600 | [diff] [blame] | 22 | static void *GetHeapBase(void) |
Marc Jones | 21cde8b | 2017-05-07 16:47:36 -0600 | [diff] [blame] | 23 | { |
Marshall Dawson | 6c74706 | 2017-09-21 12:21:14 -0600 | [diff] [blame] | 24 | return cbmem_add(CBMEM_ID_RESUME_SCRATCH, BIOS_HEAP_SIZE); |
Marc Jones | 21cde8b | 2017-05-07 16:47:36 -0600 | [diff] [blame] | 25 | } |
| 26 | |
Marshall Dawson | 6c74706 | 2017-09-21 12:21:14 -0600 | [diff] [blame] | 27 | static void EmptyHeap(int unused) |
Marc Jones | 21cde8b | 2017-05-07 16:47:36 -0600 | [diff] [blame] | 28 | { |
Marshall Dawson | ed501d5b | 2017-09-18 16:13:17 -0600 | [diff] [blame] | 29 | void *BiosManagerPtr = GetHeapBase(); |
Marc Jones | 21cde8b | 2017-05-07 16:47:36 -0600 | [diff] [blame] | 30 | memset(BiosManagerPtr, 0, BIOS_HEAP_SIZE); |
| 31 | } |
| 32 | |
Marshall Dawson | 6c74706 | 2017-09-21 12:21:14 -0600 | [diff] [blame] | 33 | #if IS_ENABLED(CONFIG_LATE_CBMEM_INIT) |
| 34 | #error "Only EARLY_CBMEM_INIT is supported." |
| 35 | #endif |
| 36 | ROMSTAGE_CBMEM_INIT_HOOK(EmptyHeap) |
| 37 | |
Stefan Reinauer | 8d29dd1 | 2017-06-26 14:30:39 -0700 | [diff] [blame] | 38 | AGESA_STATUS agesa_AllocateBuffer (UINT32 Func, UINTN Data, VOID *ConfigPtr) |
Marc Jones | 21cde8b | 2017-05-07 16:47:36 -0600 | [diff] [blame] | 39 | { |
| 40 | UINT32 AvailableHeapSize; |
| 41 | UINT8 *BiosHeapBaseAddr; |
| 42 | UINT32 CurrNodeOffset; |
| 43 | UINT32 PrevNodeOffset; |
| 44 | UINT32 FreedNodeOffset; |
| 45 | UINT32 BestFitNodeOffset; |
| 46 | UINT32 BestFitPrevNodeOffset; |
| 47 | UINT32 NextFreeOffset; |
| 48 | BIOS_BUFFER_NODE *CurrNodePtr; |
| 49 | BIOS_BUFFER_NODE *FreedNodePtr; |
| 50 | BIOS_BUFFER_NODE *BestFitNodePtr; |
| 51 | BIOS_BUFFER_NODE *BestFitPrevNodePtr; |
| 52 | BIOS_BUFFER_NODE *NextFreePtr; |
| 53 | BIOS_HEAP_MANAGER *BiosHeapBasePtr; |
| 54 | AGESA_BUFFER_PARAMS *AllocParams; |
| 55 | |
Marshall Dawson | f3dc71e | 2017-06-14 16:22:07 -0600 | [diff] [blame] | 56 | AllocParams = ((AGESA_BUFFER_PARAMS *)ConfigPtr); |
Marc Jones | 21cde8b | 2017-05-07 16:47:36 -0600 | [diff] [blame] | 57 | AllocParams->BufferPointer = NULL; |
| 58 | |
| 59 | AvailableHeapSize = BIOS_HEAP_SIZE - sizeof(BIOS_HEAP_MANAGER); |
Marshall Dawson | ed501d5b | 2017-09-18 16:13:17 -0600 | [diff] [blame] | 60 | BiosHeapBaseAddr = GetHeapBase(); |
Marshall Dawson | f3dc71e | 2017-06-14 16:22:07 -0600 | [diff] [blame] | 61 | BiosHeapBasePtr = (BIOS_HEAP_MANAGER *)BiosHeapBaseAddr; |
Marc Jones | 21cde8b | 2017-05-07 16:47:36 -0600 | [diff] [blame] | 62 | |
| 63 | if (BiosHeapBasePtr->StartOfAllocatedNodes == 0) { |
| 64 | /* First allocation */ |
| 65 | CurrNodeOffset = sizeof(BIOS_HEAP_MANAGER); |
Marshall Dawson | f3dc71e | 2017-06-14 16:22:07 -0600 | [diff] [blame] | 66 | CurrNodePtr = (BIOS_BUFFER_NODE *)(BiosHeapBaseAddr |
| 67 | + CurrNodeOffset); |
Marc Jones | 21cde8b | 2017-05-07 16:47:36 -0600 | [diff] [blame] | 68 | CurrNodePtr->BufferHandle = AllocParams->BufferHandle; |
| 69 | CurrNodePtr->BufferSize = AllocParams->BufferLength; |
| 70 | CurrNodePtr->NextNodeOffset = 0; |
Marshall Dawson | f3dc71e | 2017-06-14 16:22:07 -0600 | [diff] [blame] | 71 | AllocParams->BufferPointer = (UINT8 *)CurrNodePtr |
| 72 | + sizeof(BIOS_BUFFER_NODE); |
Marc Jones | 21cde8b | 2017-05-07 16:47:36 -0600 | [diff] [blame] | 73 | |
| 74 | /* Update the remaining free space */ |
Marshall Dawson | f3dc71e | 2017-06-14 16:22:07 -0600 | [diff] [blame] | 75 | FreedNodeOffset = CurrNodeOffset + CurrNodePtr->BufferSize |
| 76 | + sizeof(BIOS_BUFFER_NODE); |
| 77 | FreedNodePtr = (BIOS_BUFFER_NODE *)(BiosHeapBaseAddr |
| 78 | + FreedNodeOffset); |
| 79 | FreedNodePtr->BufferSize = AvailableHeapSize |
| 80 | - sizeof(BIOS_BUFFER_NODE) |
| 81 | - CurrNodePtr->BufferSize; |
Marc Jones | 21cde8b | 2017-05-07 16:47:36 -0600 | [diff] [blame] | 82 | FreedNodePtr->NextNodeOffset = 0; |
| 83 | |
| 84 | /* Update the offsets for Allocated and Freed nodes */ |
| 85 | BiosHeapBasePtr->StartOfAllocatedNodes = CurrNodeOffset; |
| 86 | BiosHeapBasePtr->StartOfFreedNodes = FreedNodeOffset; |
| 87 | } else { |
| 88 | /* Find out whether BufferHandle has been allocated on the heap. |
| 89 | * If it has, return AGESA_BOUNDS_CHK. |
| 90 | */ |
| 91 | CurrNodeOffset = BiosHeapBasePtr->StartOfAllocatedNodes; |
Marshall Dawson | f3dc71e | 2017-06-14 16:22:07 -0600 | [diff] [blame] | 92 | CurrNodePtr = (BIOS_BUFFER_NODE *)(BiosHeapBaseAddr |
| 93 | + CurrNodeOffset); |
Marc Jones | 21cde8b | 2017-05-07 16:47:36 -0600 | [diff] [blame] | 94 | |
| 95 | while (CurrNodeOffset != 0) { |
Marshall Dawson | f3dc71e | 2017-06-14 16:22:07 -0600 | [diff] [blame] | 96 | CurrNodePtr = (BIOS_BUFFER_NODE *)(BiosHeapBaseAddr |
| 97 | + CurrNodeOffset); |
| 98 | if (CurrNodePtr->BufferHandle == |
| 99 | AllocParams->BufferHandle) { |
Marc Jones | 21cde8b | 2017-05-07 16:47:36 -0600 | [diff] [blame] | 100 | return AGESA_BOUNDS_CHK; |
| 101 | } |
| 102 | CurrNodeOffset = CurrNodePtr->NextNodeOffset; |
Marshall Dawson | f3dc71e | 2017-06-14 16:22:07 -0600 | [diff] [blame] | 103 | /* If BufferHandle has not been allocated on the heap, |
| 104 | * CurrNodePtr here points to the end of the allocated |
| 105 | * nodes list. |
Marc Jones | 21cde8b | 2017-05-07 16:47:36 -0600 | [diff] [blame] | 106 | */ |
| 107 | } |
| 108 | /* Find the node that best fits the requested buffer size */ |
| 109 | FreedNodeOffset = BiosHeapBasePtr->StartOfFreedNodes; |
| 110 | PrevNodeOffset = FreedNodeOffset; |
| 111 | BestFitNodeOffset = 0; |
| 112 | BestFitPrevNodeOffset = 0; |
Marshall Dawson | f3dc71e | 2017-06-14 16:22:07 -0600 | [diff] [blame] | 113 | while (FreedNodeOffset != 0) { /* todo: simplify this */ |
| 114 | FreedNodePtr = (BIOS_BUFFER_NODE *)(BiosHeapBaseAddr |
| 115 | + FreedNodeOffset); |
Marc Jones | 21cde8b | 2017-05-07 16:47:36 -0600 | [diff] [blame] | 116 | if (FreedNodePtr->BufferSize >= (AllocParams->BufferLength + sizeof(BIOS_BUFFER_NODE))) { |
| 117 | if (BestFitNodeOffset == 0) { |
| 118 | /* First node that fits the requested buffer size */ |
| 119 | BestFitNodeOffset = FreedNodeOffset; |
| 120 | BestFitPrevNodeOffset = PrevNodeOffset; |
| 121 | } else { |
| 122 | /* Find out whether current node is a better fit than the previous nodes */ |
Marshall Dawson | f3dc71e | 2017-06-14 16:22:07 -0600 | [diff] [blame] | 123 | BestFitNodePtr = (BIOS_BUFFER_NODE *)(BiosHeapBaseAddr + BestFitNodeOffset); |
Marc Jones | 21cde8b | 2017-05-07 16:47:36 -0600 | [diff] [blame] | 124 | if (BestFitNodePtr->BufferSize > FreedNodePtr->BufferSize) { |
| 125 | BestFitNodeOffset = FreedNodeOffset; |
| 126 | BestFitPrevNodeOffset = PrevNodeOffset; |
| 127 | } |
| 128 | } |
| 129 | } |
| 130 | PrevNodeOffset = FreedNodeOffset; |
| 131 | FreedNodeOffset = FreedNodePtr->NextNodeOffset; |
| 132 | } /* end of while loop */ |
| 133 | |
| 134 | if (BestFitNodeOffset == 0) { |
Marshall Dawson | f3dc71e | 2017-06-14 16:22:07 -0600 | [diff] [blame] | 135 | /* If we could not find a node that fits the requested |
| 136 | * buffer size, return AGESA_BOUNDS_CHK. |
Marc Jones | 21cde8b | 2017-05-07 16:47:36 -0600 | [diff] [blame] | 137 | */ |
| 138 | return AGESA_BOUNDS_CHK; |
| 139 | } else { |
Marshall Dawson | f3dc71e | 2017-06-14 16:22:07 -0600 | [diff] [blame] | 140 | BestFitNodePtr = (BIOS_BUFFER_NODE *)(BiosHeapBaseAddr |
| 141 | + BestFitNodeOffset); |
| 142 | BestFitPrevNodePtr = (BIOS_BUFFER_NODE *) |
| 143 | (BiosHeapBaseAddr |
| 144 | + BestFitPrevNodeOffset); |
Marc Jones | 21cde8b | 2017-05-07 16:47:36 -0600 | [diff] [blame] | 145 | |
Marshall Dawson | f3dc71e | 2017-06-14 16:22:07 -0600 | [diff] [blame] | 146 | /* If BestFitNode is larger than the requested buffer, |
| 147 | * fragment the node further |
| 148 | */ |
| 149 | if (BestFitNodePtr->BufferSize > |
| 150 | (AllocParams->BufferLength |
| 151 | + sizeof(BIOS_BUFFER_NODE))) { |
| 152 | NextFreeOffset = BestFitNodeOffset |
| 153 | + AllocParams->BufferLength |
| 154 | + sizeof(BIOS_BUFFER_NODE); |
Marc Jones | 21cde8b | 2017-05-07 16:47:36 -0600 | [diff] [blame] | 155 | |
Marshall Dawson | f3dc71e | 2017-06-14 16:22:07 -0600 | [diff] [blame] | 156 | NextFreePtr = (BIOS_BUFFER_NODE *) |
| 157 | (BiosHeapBaseAddr |
| 158 | + NextFreeOffset); |
| 159 | NextFreePtr->BufferSize = |
| 160 | BestFitNodePtr->BufferSize |
| 161 | - (AllocParams->BufferLength |
| 162 | + sizeof(BIOS_BUFFER_NODE)); |
| 163 | NextFreePtr->NextNodeOffset = |
| 164 | BestFitNodePtr->NextNodeOffset; |
Marc Jones | 21cde8b | 2017-05-07 16:47:36 -0600 | [diff] [blame] | 165 | } else { |
Marshall Dawson | f3dc71e | 2017-06-14 16:22:07 -0600 | [diff] [blame] | 166 | /* Otherwise, next free node is |
| 167 | * NextNodeOffset of BestFitNode |
| 168 | */ |
Marc Jones | 21cde8b | 2017-05-07 16:47:36 -0600 | [diff] [blame] | 169 | NextFreeOffset = BestFitNodePtr->NextNodeOffset; |
| 170 | } |
| 171 | |
Marshall Dawson | f3dc71e | 2017-06-14 16:22:07 -0600 | [diff] [blame] | 172 | /* If BestFitNode is the first buffer in the list, then |
| 173 | * update StartOfFreedNodes to reflect new free node. |
Marc Jones | 21cde8b | 2017-05-07 16:47:36 -0600 | [diff] [blame] | 174 | */ |
Marshall Dawson | f3dc71e | 2017-06-14 16:22:07 -0600 | [diff] [blame] | 175 | if (BestFitNodeOffset == |
| 176 | BiosHeapBasePtr->StartOfFreedNodes) |
| 177 | BiosHeapBasePtr->StartOfFreedNodes = |
| 178 | NextFreeOffset; |
| 179 | else |
| 180 | BestFitPrevNodePtr->NextNodeOffset = |
| 181 | NextFreeOffset; |
Marc Jones | 21cde8b | 2017-05-07 16:47:36 -0600 | [diff] [blame] | 182 | |
| 183 | /* Add BestFitNode to the list of Allocated nodes */ |
| 184 | CurrNodePtr->NextNodeOffset = BestFitNodeOffset; |
| 185 | BestFitNodePtr->BufferSize = AllocParams->BufferLength; |
Marshall Dawson | f3dc71e | 2017-06-14 16:22:07 -0600 | [diff] [blame] | 186 | BestFitNodePtr->BufferHandle = |
| 187 | AllocParams->BufferHandle; |
Marc Jones | 21cde8b | 2017-05-07 16:47:36 -0600 | [diff] [blame] | 188 | BestFitNodePtr->NextNodeOffset = 0; |
| 189 | |
| 190 | /* Remove BestFitNode from list of Freed nodes */ |
Marshall Dawson | f3dc71e | 2017-06-14 16:22:07 -0600 | [diff] [blame] | 191 | AllocParams->BufferPointer = (UINT8 *)BestFitNodePtr |
| 192 | + sizeof(BIOS_BUFFER_NODE); |
Marc Jones | 21cde8b | 2017-05-07 16:47:36 -0600 | [diff] [blame] | 193 | } |
| 194 | } |
| 195 | |
| 196 | return AGESA_SUCCESS; |
| 197 | } |
| 198 | |
Stefan Reinauer | 8d29dd1 | 2017-06-26 14:30:39 -0700 | [diff] [blame] | 199 | AGESA_STATUS agesa_DeallocateBuffer (UINT32 Func, UINTN Data, VOID *ConfigPtr) |
Marc Jones | 21cde8b | 2017-05-07 16:47:36 -0600 | [diff] [blame] | 200 | { |
| 201 | |
| 202 | UINT8 *BiosHeapBaseAddr; |
| 203 | UINT32 AllocNodeOffset; |
| 204 | UINT32 PrevNodeOffset; |
| 205 | UINT32 NextNodeOffset; |
| 206 | UINT32 FreedNodeOffset; |
| 207 | UINT32 EndNodeOffset; |
| 208 | BIOS_BUFFER_NODE *AllocNodePtr; |
| 209 | BIOS_BUFFER_NODE *PrevNodePtr; |
| 210 | BIOS_BUFFER_NODE *FreedNodePtr; |
| 211 | BIOS_BUFFER_NODE *NextNodePtr; |
| 212 | BIOS_HEAP_MANAGER *BiosHeapBasePtr; |
| 213 | AGESA_BUFFER_PARAMS *AllocParams; |
| 214 | |
Marshall Dawson | f3dc71e | 2017-06-14 16:22:07 -0600 | [diff] [blame] | 215 | AllocParams = (AGESA_BUFFER_PARAMS *)ConfigPtr; |
Marc Jones | 21cde8b | 2017-05-07 16:47:36 -0600 | [diff] [blame] | 216 | |
Marshall Dawson | ed501d5b | 2017-09-18 16:13:17 -0600 | [diff] [blame] | 217 | BiosHeapBaseAddr = GetHeapBase(); |
Marshall Dawson | f3dc71e | 2017-06-14 16:22:07 -0600 | [diff] [blame] | 218 | BiosHeapBasePtr = (BIOS_HEAP_MANAGER *)BiosHeapBaseAddr; |
Marc Jones | 21cde8b | 2017-05-07 16:47:36 -0600 | [diff] [blame] | 219 | |
| 220 | /* Find target node to deallocate in list of allocated nodes. |
| 221 | * Return AGESA_BOUNDS_CHK if the BufferHandle is not found. |
| 222 | */ |
| 223 | AllocNodeOffset = BiosHeapBasePtr->StartOfAllocatedNodes; |
Marshall Dawson | f3dc71e | 2017-06-14 16:22:07 -0600 | [diff] [blame] | 224 | AllocNodePtr = (BIOS_BUFFER_NODE *)(BiosHeapBaseAddr + AllocNodeOffset); |
Marc Jones | 21cde8b | 2017-05-07 16:47:36 -0600 | [diff] [blame] | 225 | PrevNodeOffset = AllocNodeOffset; |
| 226 | |
Marshall Dawson | f3dc71e | 2017-06-14 16:22:07 -0600 | [diff] [blame] | 227 | while (AllocNodePtr->BufferHandle != AllocParams->BufferHandle) { |
| 228 | if (AllocNodePtr->NextNodeOffset == 0) |
Marc Jones | 21cde8b | 2017-05-07 16:47:36 -0600 | [diff] [blame] | 229 | return AGESA_BOUNDS_CHK; |
Marc Jones | 21cde8b | 2017-05-07 16:47:36 -0600 | [diff] [blame] | 230 | PrevNodeOffset = AllocNodeOffset; |
| 231 | AllocNodeOffset = AllocNodePtr->NextNodeOffset; |
Marshall Dawson | f3dc71e | 2017-06-14 16:22:07 -0600 | [diff] [blame] | 232 | AllocNodePtr = (BIOS_BUFFER_NODE *)(BiosHeapBaseAddr |
| 233 | + AllocNodeOffset); |
Marc Jones | 21cde8b | 2017-05-07 16:47:36 -0600 | [diff] [blame] | 234 | } |
| 235 | |
| 236 | /* Remove target node from list of allocated nodes */ |
Marshall Dawson | f3dc71e | 2017-06-14 16:22:07 -0600 | [diff] [blame] | 237 | PrevNodePtr = (BIOS_BUFFER_NODE *)(BiosHeapBaseAddr + PrevNodeOffset); |
Marc Jones | 21cde8b | 2017-05-07 16:47:36 -0600 | [diff] [blame] | 238 | PrevNodePtr->NextNodeOffset = AllocNodePtr->NextNodeOffset; |
| 239 | |
| 240 | /* Zero out the buffer, and clear the BufferHandle */ |
Aaron Durbin | e3f7d44 | 2017-11-03 11:44:10 -0600 | [diff] [blame] | 241 | memset((UINT8 *)AllocNodePtr + sizeof(BIOS_BUFFER_NODE), 0, |
| 242 | AllocNodePtr->BufferSize); |
Marc Jones | 21cde8b | 2017-05-07 16:47:36 -0600 | [diff] [blame] | 243 | AllocNodePtr->BufferHandle = 0; |
| 244 | AllocNodePtr->BufferSize += sizeof(BIOS_BUFFER_NODE); |
| 245 | |
| 246 | /* Add deallocated node in order to the list of freed nodes */ |
| 247 | FreedNodeOffset = BiosHeapBasePtr->StartOfFreedNodes; |
Marshall Dawson | f3dc71e | 2017-06-14 16:22:07 -0600 | [diff] [blame] | 248 | FreedNodePtr = (BIOS_BUFFER_NODE *)(BiosHeapBaseAddr + FreedNodeOffset); |
Marc Jones | 21cde8b | 2017-05-07 16:47:36 -0600 | [diff] [blame] | 249 | |
| 250 | EndNodeOffset = AllocNodeOffset + AllocNodePtr->BufferSize; |
| 251 | |
| 252 | if (AllocNodeOffset < FreedNodeOffset) { |
| 253 | /* Add to the start of the freed list */ |
| 254 | if (EndNodeOffset == FreedNodeOffset) { |
Marshall Dawson | f3dc71e | 2017-06-14 16:22:07 -0600 | [diff] [blame] | 255 | /* If the freed node is adjacent to the first node in |
| 256 | * the list, concatenate both nodes |
| 257 | */ |
Marc Jones | 21cde8b | 2017-05-07 16:47:36 -0600 | [diff] [blame] | 258 | AllocNodePtr->BufferSize += FreedNodePtr->BufferSize; |
Marshall Dawson | f3dc71e | 2017-06-14 16:22:07 -0600 | [diff] [blame] | 259 | AllocNodePtr->NextNodeOffset = |
| 260 | FreedNodePtr->NextNodeOffset; |
Marc Jones | 21cde8b | 2017-05-07 16:47:36 -0600 | [diff] [blame] | 261 | |
Marshall Dawson | f3dc71e | 2017-06-14 16:22:07 -0600 | [diff] [blame] | 262 | /* Clear the BufferSize and NextNodeOffset of the |
| 263 | * previous first node |
| 264 | */ |
Marc Jones | 21cde8b | 2017-05-07 16:47:36 -0600 | [diff] [blame] | 265 | FreedNodePtr->BufferSize = 0; |
| 266 | FreedNodePtr->NextNodeOffset = 0; |
| 267 | |
| 268 | } else { |
| 269 | /* Otherwise, add freed node to the start of the list |
| 270 | * Update NextNodeOffset and BufferSize to include the |
| 271 | * size of BIOS_BUFFER_NODE. |
| 272 | */ |
| 273 | AllocNodePtr->NextNodeOffset = FreedNodeOffset; |
| 274 | } |
| 275 | /* Update StartOfFreedNodes to the new first node */ |
| 276 | BiosHeapBasePtr->StartOfFreedNodes = AllocNodeOffset; |
| 277 | } else { |
Marshall Dawson | f3dc71e | 2017-06-14 16:22:07 -0600 | [diff] [blame] | 278 | /* Traverse list of freed nodes to find where the deallocated |
| 279 | * node should be placed. |
Marc Jones | 21cde8b | 2017-05-07 16:47:36 -0600 | [diff] [blame] | 280 | */ |
| 281 | NextNodeOffset = FreedNodeOffset; |
| 282 | NextNodePtr = FreedNodePtr; |
| 283 | while (AllocNodeOffset > NextNodeOffset) { |
| 284 | PrevNodeOffset = NextNodeOffset; |
Marshall Dawson | f3dc71e | 2017-06-14 16:22:07 -0600 | [diff] [blame] | 285 | if (NextNodePtr->NextNodeOffset == 0) |
Marc Jones | 21cde8b | 2017-05-07 16:47:36 -0600 | [diff] [blame] | 286 | break; |
Marc Jones | 21cde8b | 2017-05-07 16:47:36 -0600 | [diff] [blame] | 287 | NextNodeOffset = NextNodePtr->NextNodeOffset; |
Marshall Dawson | f3dc71e | 2017-06-14 16:22:07 -0600 | [diff] [blame] | 288 | NextNodePtr = (BIOS_BUFFER_NODE *)(BiosHeapBaseAddr |
| 289 | + NextNodeOffset); |
Marc Jones | 21cde8b | 2017-05-07 16:47:36 -0600 | [diff] [blame] | 290 | } |
| 291 | |
| 292 | /* If deallocated node is adjacent to the next node, |
| 293 | * concatenate both nodes. |
| 294 | */ |
| 295 | if (NextNodeOffset == EndNodeOffset) { |
Marshall Dawson | f3dc71e | 2017-06-14 16:22:07 -0600 | [diff] [blame] | 296 | NextNodePtr = (BIOS_BUFFER_NODE *)(BiosHeapBaseAddr |
| 297 | + NextNodeOffset); |
Marc Jones | 21cde8b | 2017-05-07 16:47:36 -0600 | [diff] [blame] | 298 | AllocNodePtr->BufferSize += NextNodePtr->BufferSize; |
Marshall Dawson | f3dc71e | 2017-06-14 16:22:07 -0600 | [diff] [blame] | 299 | AllocNodePtr->NextNodeOffset = |
| 300 | NextNodePtr->NextNodeOffset; |
Marc Jones | 21cde8b | 2017-05-07 16:47:36 -0600 | [diff] [blame] | 301 | |
| 302 | NextNodePtr->BufferSize = 0; |
| 303 | NextNodePtr->NextNodeOffset = 0; |
| 304 | } else { |
Marshall Dawson | f3dc71e | 2017-06-14 16:22:07 -0600 | [diff] [blame] | 305 | /*AllocNodePtr->NextNodeOffset = |
| 306 | * FreedNodePtr->NextNodeOffset; */ |
Marc Jones | 21cde8b | 2017-05-07 16:47:36 -0600 | [diff] [blame] | 307 | AllocNodePtr->NextNodeOffset = NextNodeOffset; |
| 308 | } |
| 309 | /* If deallocated node is adjacent to the previous node, |
| 310 | * concatenate both nodes. |
| 311 | */ |
Marshall Dawson | f3dc71e | 2017-06-14 16:22:07 -0600 | [diff] [blame] | 312 | PrevNodePtr = (BIOS_BUFFER_NODE *)(BiosHeapBaseAddr |
| 313 | + PrevNodeOffset); |
Marc Jones | 21cde8b | 2017-05-07 16:47:36 -0600 | [diff] [blame] | 314 | EndNodeOffset = PrevNodeOffset + PrevNodePtr->BufferSize; |
| 315 | if (AllocNodeOffset == EndNodeOffset) { |
Marshall Dawson | f3dc71e | 2017-06-14 16:22:07 -0600 | [diff] [blame] | 316 | PrevNodePtr->NextNodeOffset = |
| 317 | AllocNodePtr->NextNodeOffset; |
Marc Jones | 21cde8b | 2017-05-07 16:47:36 -0600 | [diff] [blame] | 318 | PrevNodePtr->BufferSize += AllocNodePtr->BufferSize; |
| 319 | |
| 320 | AllocNodePtr->BufferSize = 0; |
| 321 | AllocNodePtr->NextNodeOffset = 0; |
| 322 | } else { |
| 323 | PrevNodePtr->NextNodeOffset = AllocNodeOffset; |
| 324 | } |
| 325 | } |
| 326 | return AGESA_SUCCESS; |
| 327 | } |
| 328 | |
Stefan Reinauer | 8d29dd1 | 2017-06-26 14:30:39 -0700 | [diff] [blame] | 329 | AGESA_STATUS agesa_LocateBuffer (UINT32 Func, UINTN Data, VOID *ConfigPtr) |
Marc Jones | 21cde8b | 2017-05-07 16:47:36 -0600 | [diff] [blame] | 330 | { |
| 331 | UINT32 AllocNodeOffset; |
| 332 | UINT8 *BiosHeapBaseAddr; |
Marshall Dawson | f3dc71e | 2017-06-14 16:22:07 -0600 | [diff] [blame] | 333 | BIOS_BUFFER_NODE *AllocNodePtr; |
| 334 | BIOS_HEAP_MANAGER *BiosHeapBasePtr; |
Marc Jones | 21cde8b | 2017-05-07 16:47:36 -0600 | [diff] [blame] | 335 | AGESA_BUFFER_PARAMS *AllocParams; |
| 336 | |
Marshall Dawson | f3dc71e | 2017-06-14 16:22:07 -0600 | [diff] [blame] | 337 | AllocParams = (AGESA_BUFFER_PARAMS *)ConfigPtr; |
Marc Jones | 21cde8b | 2017-05-07 16:47:36 -0600 | [diff] [blame] | 338 | |
Marshall Dawson | ed501d5b | 2017-09-18 16:13:17 -0600 | [diff] [blame] | 339 | BiosHeapBaseAddr = GetHeapBase(); |
Marshall Dawson | f3dc71e | 2017-06-14 16:22:07 -0600 | [diff] [blame] | 340 | BiosHeapBasePtr = (BIOS_HEAP_MANAGER *)BiosHeapBaseAddr; |
Marc Jones | 21cde8b | 2017-05-07 16:47:36 -0600 | [diff] [blame] | 341 | |
| 342 | AllocNodeOffset = BiosHeapBasePtr->StartOfAllocatedNodes; |
Marshall Dawson | f3dc71e | 2017-06-14 16:22:07 -0600 | [diff] [blame] | 343 | AllocNodePtr = (BIOS_BUFFER_NODE *)(BiosHeapBaseAddr + AllocNodeOffset); |
Marc Jones | 21cde8b | 2017-05-07 16:47:36 -0600 | [diff] [blame] | 344 | |
| 345 | while (AllocParams->BufferHandle != AllocNodePtr->BufferHandle) { |
| 346 | if (AllocNodePtr->NextNodeOffset == 0) { |
| 347 | AllocParams->BufferPointer = NULL; |
| 348 | AllocParams->BufferLength = 0; |
| 349 | return AGESA_BOUNDS_CHK; |
| 350 | } else { |
| 351 | AllocNodeOffset = AllocNodePtr->NextNodeOffset; |
Marshall Dawson | f3dc71e | 2017-06-14 16:22:07 -0600 | [diff] [blame] | 352 | AllocNodePtr = (BIOS_BUFFER_NODE *)(BiosHeapBaseAddr |
| 353 | + AllocNodeOffset); |
Marc Jones | 21cde8b | 2017-05-07 16:47:36 -0600 | [diff] [blame] | 354 | } |
| 355 | } |
| 356 | |
Marshall Dawson | f3dc71e | 2017-06-14 16:22:07 -0600 | [diff] [blame] | 357 | AllocParams->BufferPointer = (UINT8 *)((UINT8 *)AllocNodePtr |
| 358 | + sizeof(BIOS_BUFFER_NODE)); |
Marc Jones | 21cde8b | 2017-05-07 16:47:36 -0600 | [diff] [blame] | 359 | AllocParams->BufferLength = AllocNodePtr->BufferSize; |
| 360 | |
| 361 | return AGESA_SUCCESS; |
| 362 | |
| 363 | } |