TianoCore EDK2 master
Loading...
Searching...
No Matches
SdHcMem.c
Go to the documentation of this file.
1
9#include "SdBlockIoPei.h"
10
21 IN UINTN Pages
22 )
23{
24 SD_PEIM_MEM_BLOCK *Block;
25 VOID *BufHost;
26 VOID *Mapping;
27 EFI_PHYSICAL_ADDRESS MappedAddr;
28 EFI_STATUS Status;
29 VOID *TempPtr;
30
31 TempPtr = NULL;
32 Block = NULL;
33
34 Status = PeiServicesAllocatePool (sizeof (SD_PEIM_MEM_BLOCK), &TempPtr);
35 if (EFI_ERROR (Status)) {
36 return NULL;
37 }
38
39 ZeroMem ((VOID *)(UINTN)TempPtr, sizeof (SD_PEIM_MEM_BLOCK));
40
41 //
42 // each bit in the bit array represents SD_PEIM_MEM_UNIT
43 // bytes of memory in the memory block.
44 //
45 ASSERT (SD_PEIM_MEM_UNIT * 8 <= EFI_PAGE_SIZE);
46
47 Block = (SD_PEIM_MEM_BLOCK *)(UINTN)TempPtr;
48 Block->BufLen = EFI_PAGES_TO_SIZE (Pages);
49 Block->BitsLen = Block->BufLen / (SD_PEIM_MEM_UNIT * 8);
50
51 Status = PeiServicesAllocatePool (Block->BitsLen, &TempPtr);
52 if (EFI_ERROR (Status)) {
53 return NULL;
54 }
55
56 ZeroMem ((VOID *)(UINTN)TempPtr, Block->BitsLen);
57
58 Block->Bits = (UINT8 *)(UINTN)TempPtr;
59
60 Status = IoMmuAllocateBuffer (
61 Pages,
62 &BufHost,
63 &MappedAddr,
64 &Mapping
65 );
66 if (EFI_ERROR (Status)) {
67 return NULL;
68 }
69
70 ZeroMem ((VOID *)(UINTN)BufHost, EFI_PAGES_TO_SIZE (Pages));
71
72 Block->BufHost = (UINT8 *)(UINTN)BufHost;
73 Block->Buf = (UINT8 *)(UINTN)MappedAddr;
74 Block->Mapping = Mapping;
75 Block->Next = NULL;
76
77 return Block;
78}
79
87VOID
89 IN SD_PEIM_MEM_POOL *Pool,
91 )
92{
93 ASSERT ((Pool != NULL) && (Block != NULL));
94
95 IoMmuFreeBuffer (EFI_SIZE_TO_PAGES (Block->BufLen), Block->BufHost, Block->Mapping);
96}
97
108VOID *
110 IN SD_PEIM_MEM_BLOCK *Block,
111 IN UINTN Units
112 )
113{
114 UINTN Byte;
115 UINT8 Bit;
116 UINTN StartByte;
117 UINT8 StartBit;
118 UINTN Available;
119 UINTN Count;
120
121 ASSERT ((Block != 0) && (Units != 0));
122
123 StartByte = 0;
124 StartBit = 0;
125 Available = 0;
126
127 for (Byte = 0, Bit = 0; Byte < Block->BitsLen;) {
128 //
129 // If current bit is zero, the corresponding memory unit is
130 // available, otherwise we need to restart our searching.
131 // Available counts the consecutive number of zero bit.
132 //
133 if (!SD_PEIM_MEM_BIT_IS_SET (Block->Bits[Byte], Bit)) {
134 Available++;
135
136 if (Available >= Units) {
137 break;
138 }
139
140 SD_PEIM_NEXT_BIT (Byte, Bit);
141 } else {
142 SD_PEIM_NEXT_BIT (Byte, Bit);
143
144 Available = 0;
145 StartByte = Byte;
146 StartBit = Bit;
147 }
148 }
149
150 if (Available < Units) {
151 return NULL;
152 }
153
154 //
155 // Mark the memory as allocated
156 //
157 Byte = StartByte;
158 Bit = StartBit;
159
160 for (Count = 0; Count < Units; Count++) {
161 ASSERT (!SD_PEIM_MEM_BIT_IS_SET (Block->Bits[Byte], Bit));
162
163 Block->Bits[Byte] = (UINT8)(Block->Bits[Byte] | (UINT8)SD_PEIM_MEM_BIT (Bit));
164 SD_PEIM_NEXT_BIT (Byte, Bit);
165 }
166
167 return Block->Buf + (StartByte * 8 + StartBit) * SD_PEIM_MEM_UNIT;
168}
169
177VOID
179 IN SD_PEIM_MEM_BLOCK *Head,
180 IN SD_PEIM_MEM_BLOCK *Block
181 )
182{
183 ASSERT ((Head != NULL) && (Block != NULL));
184 Block->Next = Head->Next;
185 Head->Next = Block;
186}
187
197BOOLEAN
199 IN SD_PEIM_MEM_BLOCK *Block
200 )
201{
202 UINTN Index;
203
204 for (Index = 0; Index < Block->BitsLen; Index++) {
205 if (Block->Bits[Index] != 0) {
206 return FALSE;
207 }
208 }
209
210 return TRUE;
211}
212
225 )
226{
227 SD_PEIM_MEM_POOL *Pool;
228 EFI_STATUS Status;
229 VOID *TempPtr;
230
231 TempPtr = NULL;
232 Pool = NULL;
233
234 Status = PeiServicesAllocatePool (sizeof (SD_PEIM_MEM_POOL), &TempPtr);
235 if (EFI_ERROR (Status)) {
236 return EFI_OUT_OF_RESOURCES;
237 }
238
239 ZeroMem ((VOID *)(UINTN)TempPtr, sizeof (SD_PEIM_MEM_POOL));
240
241 Pool = (SD_PEIM_MEM_POOL *)((UINTN)TempPtr);
242
243 Pool->Head = SdPeimAllocMemBlock (SD_PEIM_MEM_DEFAULT_PAGES);
244
245 if (Pool->Head == NULL) {
246 return EFI_OUT_OF_RESOURCES;
247 }
248
249 Private->Pool = Pool;
250 return EFI_SUCCESS;
251}
252
264 IN SD_PEIM_MEM_POOL *Pool
265 )
266{
267 SD_PEIM_MEM_BLOCK *Block;
268
269 ASSERT (Pool->Head != NULL);
270
271 //
272 // Unlink all the memory blocks from the pool, then free them.
273 //
274 for (Block = Pool->Head->Next; Block != NULL; Block = Pool->Head->Next) {
275 SdPeimFreeMemBlock (Pool, Block);
276 }
277
278 SdPeimFreeMemBlock (Pool, Pool->Head);
279
280 return EFI_SUCCESS;
281}
282
293VOID *
295 IN SD_PEIM_MEM_POOL *Pool,
296 IN UINTN Size
297 )
298{
299 SD_PEIM_MEM_BLOCK *Head;
300 SD_PEIM_MEM_BLOCK *Block;
301 SD_PEIM_MEM_BLOCK *NewBlock;
302 VOID *Mem;
303 UINTN AllocSize;
304 UINTN Pages;
305
306 Mem = NULL;
307 AllocSize = SD_PEIM_MEM_ROUND (Size);
308 Head = Pool->Head;
309 ASSERT (Head != NULL);
310
311 //
312 // First check whether current memory blocks can satisfy the allocation.
313 //
314 for (Block = Head; Block != NULL; Block = Block->Next) {
315 Mem = SdPeimAllocMemFromBlock (Block, AllocSize / SD_PEIM_MEM_UNIT);
316
317 if (Mem != NULL) {
318 ZeroMem (Mem, Size);
319 break;
320 }
321 }
322
323 if (Mem != NULL) {
324 return Mem;
325 }
326
327 //
328 // Create a new memory block if there is not enough memory
329 // in the pool. If the allocation size is larger than the
330 // default page number, just allocate a large enough memory
331 // block. Otherwise allocate default pages.
332 //
333 if (AllocSize > EFI_PAGES_TO_SIZE (SD_PEIM_MEM_DEFAULT_PAGES)) {
334 Pages = EFI_SIZE_TO_PAGES (AllocSize) + 1;
335 } else {
336 Pages = SD_PEIM_MEM_DEFAULT_PAGES;
337 }
338
339 NewBlock = SdPeimAllocMemBlock (Pages);
340 if (NewBlock == NULL) {
341 return NULL;
342 }
343
344 //
345 // Add the new memory block to the pool, then allocate memory from it
346 //
347 SdPeimInsertMemBlockToPool (Head, NewBlock);
348 Mem = SdPeimAllocMemFromBlock (NewBlock, AllocSize / SD_PEIM_MEM_UNIT);
349
350 if (Mem != NULL) {
351 ZeroMem (Mem, Size);
352 }
353
354 return Mem;
355}
356
365VOID
367 IN SD_PEIM_MEM_POOL *Pool,
368 IN VOID *Mem,
369 IN UINTN Size
370 )
371{
372 SD_PEIM_MEM_BLOCK *Head;
373 SD_PEIM_MEM_BLOCK *Block;
374 UINT8 *ToFree;
375 UINTN AllocSize;
376 UINTN Byte;
377 UINTN Bit;
378 UINTN Count;
379
380 Head = Pool->Head;
381 AllocSize = SD_PEIM_MEM_ROUND (Size);
382 ToFree = (UINT8 *)Mem;
383
384 for (Block = Head; Block != NULL; Block = Block->Next) {
385 //
386 // scan the memory block list for the memory block that
387 // completely contains the memory to free.
388 //
389 if ((Block->Buf <= ToFree) && ((ToFree + AllocSize) <= (Block->Buf + Block->BufLen))) {
390 //
391 // compute the start byte and bit in the bit array
392 //
393 Byte = ((ToFree - Block->Buf) / SD_PEIM_MEM_UNIT) / 8;
394 Bit = ((ToFree - Block->Buf) / SD_PEIM_MEM_UNIT) % 8;
395
396 //
397 // reset associated bits in bit array
398 //
399 for (Count = 0; Count < (AllocSize / SD_PEIM_MEM_UNIT); Count++) {
400 ASSERT (SD_PEIM_MEM_BIT_IS_SET (Block->Bits[Byte], Bit));
401
402 Block->Bits[Byte] = (UINT8)(Block->Bits[Byte] ^ SD_PEIM_MEM_BIT (Bit));
403 SD_PEIM_NEXT_BIT (Byte, Bit);
404 }
405
406 break;
407 }
408 }
409
410 //
411 // If Block == NULL, it means that the current memory isn't
412 // in the host controller's pool. This is critical because
413 // the caller has passed in a wrong memory point
414 //
415 ASSERT (Block != NULL);
416
417 //
418 // Release the current memory block if it is empty and not the head
419 //
420 if ((Block != Head) && SdPeimIsMemBlockEmpty (Block)) {
421 SdPeimFreeMemBlock (Pool, Block);
422 }
423
424 return;
425}
UINT64 UINTN
EFI_STATUS IoMmuAllocateBuffer(IN UINTN Pages, OUT VOID **HostAddress, OUT EFI_PHYSICAL_ADDRESS *DeviceAddress, OUT VOID **Mapping)
Definition: DmaMem.c:170
EFI_STATUS IoMmuFreeBuffer(IN UINTN Pages, IN VOID *HostAddress, IN VOID *Mapping)
Definition: DmaMem.c:251
VOID *EFIAPI ZeroMem(OUT VOID *Buffer, IN UINTN Length)
EFI_STATUS EFIAPI PeiServicesAllocatePool(IN UINTN Size, OUT VOID **Buffer)
#define NULL
Definition: Base.h:319
#define TRUE
Definition: Base.h:301
#define FALSE
Definition: Base.h:307
#define IN
Definition: Base.h:279
BOOLEAN SdPeimIsMemBlockEmpty(IN SD_PEIM_MEM_BLOCK *Block)
Definition: SdHcMem.c:198
VOID SdPeimFreeMem(IN SD_PEIM_MEM_POOL *Pool, IN VOID *Mem, IN UINTN Size)
Definition: SdHcMem.c:366
VOID SdPeimInsertMemBlockToPool(IN SD_PEIM_MEM_BLOCK *Head, IN SD_PEIM_MEM_BLOCK *Block)
Definition: SdHcMem.c:178
EFI_STATUS SdPeimFreeMemPool(IN SD_PEIM_MEM_POOL *Pool)
Definition: SdHcMem.c:263
SD_PEIM_MEM_BLOCK * SdPeimAllocMemBlock(IN UINTN Pages)
Definition: SdHcMem.c:20
EFI_STATUS SdPeimInitMemPool(IN SD_PEIM_HC_PRIVATE_DATA *Private)
Definition: SdHcMem.c:223
VOID * SdPeimAllocMemFromBlock(IN SD_PEIM_MEM_BLOCK *Block, IN UINTN Units)
Definition: SdHcMem.c:109
VOID * SdPeimAllocateMem(IN SD_PEIM_MEM_POOL *Pool, IN UINTN Size)
Definition: SdHcMem.c:294
VOID SdPeimFreeMemBlock(IN SD_PEIM_MEM_POOL *Pool, IN SD_PEIM_MEM_BLOCK *Block)
Definition: SdHcMem.c:88
UINT64 EFI_PHYSICAL_ADDRESS
Definition: UefiBaseType.h:50
#define EFI_PAGES_TO_SIZE(Pages)
Definition: UefiBaseType.h:213
RETURN_STATUS EFI_STATUS
Definition: UefiBaseType.h:29
#define EFI_SIZE_TO_PAGES(Size)
Definition: UefiBaseType.h:200
#define EFI_SUCCESS
Definition: UefiBaseType.h:112