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