]> git.proxmox.com Git - mirror_edk2.git/blame - MdeModulePkg/Core/Dxe/Misc/PropertiesTable.c
MdeModulePkg: Replace BSD License with BSD+Patent License
[mirror_edk2.git] / MdeModulePkg / Core / Dxe / Misc / PropertiesTable.c
CommitLineData
03d486b2
JY
1/** @file\r
2 UEFI PropertiesTable support\r
3\r
63ebde8e 4Copyright (c) 2015 - 2018, Intel Corporation. All rights reserved.<BR>\r
9d510e61 5SPDX-License-Identifier: BSD-2-Clause-Patent\r
03d486b2
JY
6\r
7**/\r
8\r
9#include <PiDxe.h>\r
10#include <Library/BaseLib.h>\r
11#include <Library/BaseMemoryLib.h>\r
12#include <Library/MemoryAllocationLib.h>\r
13#include <Library/UefiBootServicesTableLib.h>\r
14#include <Library/DxeServicesTableLib.h>\r
15#include <Library/DebugLib.h>\r
16#include <Library/UefiLib.h>\r
17#include <Library/PcdLib.h>\r
18\r
19#include <Guid/EventGroup.h>\r
20#include <Protocol/DxeSmmReadyToLock.h>\r
21\r
22#include <Library/PeCoffLib.h>\r
23#include <Library/PeCoffGetEntryPointLib.h>\r
24#include <Protocol/Runtime.h>\r
25\r
26#include <Guid/PropertiesTable.h>\r
27\r
28#include "DxeMain.h"\r
63ebde8e 29#include "HeapGuard.h"\r
03d486b2
JY
30\r
31#define PREVIOUS_MEMORY_DESCRIPTOR(MemoryDescriptor, Size) \\r
32 ((EFI_MEMORY_DESCRIPTOR *)((UINT8 *)(MemoryDescriptor) - (Size)))\r
33\r
03d486b2
JY
34#define IMAGE_PROPERTIES_PRIVATE_DATA_SIGNATURE SIGNATURE_32 ('I','P','P','D')\r
35\r
36typedef struct {\r
37 UINT32 Signature;\r
38 UINTN ImageRecordCount;\r
39 UINTN CodeSegmentCountMax;\r
40 LIST_ENTRY ImageRecordList;\r
41} IMAGE_PROPERTIES_PRIVATE_DATA;\r
42\r
43IMAGE_PROPERTIES_PRIVATE_DATA mImagePropertiesPrivateData = {\r
44 IMAGE_PROPERTIES_PRIVATE_DATA_SIGNATURE,\r
45 0,\r
46 0,\r
47 INITIALIZE_LIST_HEAD_VARIABLE (mImagePropertiesPrivateData.ImageRecordList)\r
48};\r
49\r
50EFI_PROPERTIES_TABLE mPropertiesTable = {\r
51 EFI_PROPERTIES_TABLE_VERSION,\r
52 sizeof(EFI_PROPERTIES_TABLE),\r
53 EFI_PROPERTIES_RUNTIME_MEMORY_PROTECTION_NON_EXECUTABLE_PE_DATA\r
54};\r
55\r
56EFI_LOCK mPropertiesTableLock = EFI_INITIALIZE_LOCK_VARIABLE (TPL_NOTIFY);\r
57\r
82f0f411 58BOOLEAN mPropertiesTableEnable;\r
f5a91bba 59\r
4a723ed2
JY
60BOOLEAN mPropertiesTableEndOfDxe = FALSE;\r
61\r
03d486b2
JY
62//\r
63// Below functions are for MemoryMap\r
64//\r
65\r
66/**\r
67 Converts a number of EFI_PAGEs to a size in bytes.\r
68\r
69 NOTE: Do not use EFI_PAGES_TO_SIZE because it handles UINTN only.\r
70\r
71 @param Pages The number of EFI_PAGES.\r
72\r
73 @return The number of bytes associated with the number of EFI_PAGEs specified\r
74 by Pages.\r
75**/\r
2caab142 76STATIC\r
03d486b2
JY
77UINT64\r
78EfiPagesToSize (\r
79 IN UINT64 Pages\r
80 )\r
81{\r
82 return LShiftU64 (Pages, EFI_PAGE_SHIFT);\r
83}\r
84\r
85/**\r
86 Converts a size, in bytes, to a number of EFI_PAGESs.\r
87\r
88 NOTE: Do not use EFI_SIZE_TO_PAGES because it handles UINTN only.\r
89\r
90 @param Size A size in bytes.\r
91\r
92 @return The number of EFI_PAGESs associated with the number of bytes specified\r
93 by Size.\r
94\r
95**/\r
2caab142 96STATIC\r
03d486b2
JY
97UINT64\r
98EfiSizeToPages (\r
99 IN UINT64 Size\r
100 )\r
101{\r
102 return RShiftU64 (Size, EFI_PAGE_SHIFT) + ((((UINTN)Size) & EFI_PAGE_MASK) ? 1 : 0);\r
103}\r
104\r
105/**\r
106 Acquire memory lock on mPropertiesTableLock.\r
107**/\r
2caab142 108STATIC\r
03d486b2
JY
109VOID\r
110CoreAcquirePropertiesTableLock (\r
111 VOID\r
112 )\r
113{\r
114 CoreAcquireLock (&mPropertiesTableLock);\r
115}\r
116\r
117/**\r
118 Release memory lock on mPropertiesTableLock.\r
119**/\r
2caab142 120STATIC\r
03d486b2
JY
121VOID\r
122CoreReleasePropertiesTableLock (\r
123 VOID\r
124 )\r
125{\r
126 CoreReleaseLock (&mPropertiesTableLock);\r
127}\r
128\r
03d486b2
JY
129/**\r
130 Sort memory map entries based upon PhysicalStart, from low to high.\r
131\r
132 @param MemoryMap A pointer to the buffer in which firmware places\r
133 the current memory map.\r
134 @param MemoryMapSize Size, in bytes, of the MemoryMap buffer.\r
135 @param DescriptorSize Size, in bytes, of an individual EFI_MEMORY_DESCRIPTOR.\r
136**/\r
2caab142 137STATIC\r
03d486b2
JY
138VOID\r
139SortMemoryMap (\r
140 IN OUT EFI_MEMORY_DESCRIPTOR *MemoryMap,\r
141 IN UINTN MemoryMapSize,\r
142 IN UINTN DescriptorSize\r
143 )\r
144{\r
145 EFI_MEMORY_DESCRIPTOR *MemoryMapEntry;\r
146 EFI_MEMORY_DESCRIPTOR *NextMemoryMapEntry;\r
147 EFI_MEMORY_DESCRIPTOR *MemoryMapEnd;\r
148 EFI_MEMORY_DESCRIPTOR TempMemoryMap;\r
149\r
150 MemoryMapEntry = MemoryMap;\r
151 NextMemoryMapEntry = NEXT_MEMORY_DESCRIPTOR (MemoryMapEntry, DescriptorSize);\r
152 MemoryMapEnd = (EFI_MEMORY_DESCRIPTOR *) ((UINT8 *) MemoryMap + MemoryMapSize);\r
153 while (MemoryMapEntry < MemoryMapEnd) {\r
154 while (NextMemoryMapEntry < MemoryMapEnd) {\r
155 if (MemoryMapEntry->PhysicalStart > NextMemoryMapEntry->PhysicalStart) {\r
156 CopyMem (&TempMemoryMap, MemoryMapEntry, sizeof(EFI_MEMORY_DESCRIPTOR));\r
157 CopyMem (MemoryMapEntry, NextMemoryMapEntry, sizeof(EFI_MEMORY_DESCRIPTOR));\r
158 CopyMem (NextMemoryMapEntry, &TempMemoryMap, sizeof(EFI_MEMORY_DESCRIPTOR));\r
159 }\r
160\r
161 NextMemoryMapEntry = NEXT_MEMORY_DESCRIPTOR (NextMemoryMapEntry, DescriptorSize);\r
162 }\r
163\r
164 MemoryMapEntry = NEXT_MEMORY_DESCRIPTOR (MemoryMapEntry, DescriptorSize);\r
165 NextMemoryMapEntry = NEXT_MEMORY_DESCRIPTOR (MemoryMapEntry, DescriptorSize);\r
166 }\r
167\r
168 return ;\r
169}\r
170\r
171/**\r
172 Merge continous memory map entries whose have same attributes.\r
173\r
174 @param MemoryMap A pointer to the buffer in which firmware places\r
175 the current memory map.\r
176 @param MemoryMapSize A pointer to the size, in bytes, of the\r
177 MemoryMap buffer. On input, this is the size of\r
178 the current memory map. On output,\r
179 it is the size of new memory map after merge.\r
180 @param DescriptorSize Size, in bytes, of an individual EFI_MEMORY_DESCRIPTOR.\r
181**/\r
182VOID\r
183MergeMemoryMap (\r
184 IN OUT EFI_MEMORY_DESCRIPTOR *MemoryMap,\r
185 IN OUT UINTN *MemoryMapSize,\r
186 IN UINTN DescriptorSize\r
187 )\r
188{\r
189 EFI_MEMORY_DESCRIPTOR *MemoryMapEntry;\r
190 EFI_MEMORY_DESCRIPTOR *MemoryMapEnd;\r
191 UINT64 MemoryBlockLength;\r
192 EFI_MEMORY_DESCRIPTOR *NewMemoryMapEntry;\r
193 EFI_MEMORY_DESCRIPTOR *NextMemoryMapEntry;\r
194\r
195 MemoryMapEntry = MemoryMap;\r
196 NewMemoryMapEntry = MemoryMap;\r
197 MemoryMapEnd = (EFI_MEMORY_DESCRIPTOR *) ((UINT8 *) MemoryMap + *MemoryMapSize);\r
198 while ((UINTN)MemoryMapEntry < (UINTN)MemoryMapEnd) {\r
199 CopyMem (NewMemoryMapEntry, MemoryMapEntry, sizeof(EFI_MEMORY_DESCRIPTOR));\r
200 NextMemoryMapEntry = NEXT_MEMORY_DESCRIPTOR (MemoryMapEntry, DescriptorSize);\r
201\r
f5a91bba 202 do {\r
63ebde8e
JW
203 MergeGuardPages (NewMemoryMapEntry, NextMemoryMapEntry->PhysicalStart);\r
204 MemoryBlockLength = (UINT64) (EfiPagesToSize (NewMemoryMapEntry->NumberOfPages));\r
f5a91bba 205 if (((UINTN)NextMemoryMapEntry < (UINTN)MemoryMapEnd) &&\r
63ebde8e
JW
206 (NewMemoryMapEntry->Type == NextMemoryMapEntry->Type) &&\r
207 (NewMemoryMapEntry->Attribute == NextMemoryMapEntry->Attribute) &&\r
208 ((NewMemoryMapEntry->PhysicalStart + MemoryBlockLength) == NextMemoryMapEntry->PhysicalStart)) {\r
209 NewMemoryMapEntry->NumberOfPages += NextMemoryMapEntry->NumberOfPages;\r
f5a91bba
JY
210 NextMemoryMapEntry = NEXT_MEMORY_DESCRIPTOR (NextMemoryMapEntry, DescriptorSize);\r
211 continue;\r
212 } else {\r
213 MemoryMapEntry = PREVIOUS_MEMORY_DESCRIPTOR (NextMemoryMapEntry, DescriptorSize);\r
214 break;\r
215 }\r
216 } while (TRUE);\r
03d486b2
JY
217\r
218 MemoryMapEntry = NEXT_MEMORY_DESCRIPTOR (MemoryMapEntry, DescriptorSize);\r
219 NewMemoryMapEntry = NEXT_MEMORY_DESCRIPTOR (NewMemoryMapEntry, DescriptorSize);\r
220 }\r
221\r
222 *MemoryMapSize = (UINTN)NewMemoryMapEntry - (UINTN)MemoryMap;\r
223\r
224 return ;\r
225}\r
226\r
227/**\r
228 Enforce memory map attributes.\r
229 This function will set EfiRuntimeServicesData/EfiMemoryMappedIO/EfiMemoryMappedIOPortSpace to be EFI_MEMORY_XP.\r
230\r
231 @param MemoryMap A pointer to the buffer in which firmware places\r
232 the current memory map.\r
233 @param MemoryMapSize Size, in bytes, of the MemoryMap buffer.\r
234 @param DescriptorSize Size, in bytes, of an individual EFI_MEMORY_DESCRIPTOR.\r
235**/\r
2caab142 236STATIC\r
03d486b2
JY
237VOID\r
238EnforceMemoryMapAttribute (\r
239 IN OUT EFI_MEMORY_DESCRIPTOR *MemoryMap,\r
240 IN UINTN MemoryMapSize,\r
241 IN UINTN DescriptorSize\r
242 )\r
243{\r
244 EFI_MEMORY_DESCRIPTOR *MemoryMapEntry;\r
245 EFI_MEMORY_DESCRIPTOR *MemoryMapEnd;\r
246\r
247 MemoryMapEntry = MemoryMap;\r
248 MemoryMapEnd = (EFI_MEMORY_DESCRIPTOR *) ((UINT8 *) MemoryMap + MemoryMapSize);\r
249 while ((UINTN)MemoryMapEntry < (UINTN)MemoryMapEnd) {\r
250 switch (MemoryMapEntry->Type) {\r
251 case EfiRuntimeServicesCode:\r
252 // do nothing\r
253 break;\r
254 case EfiRuntimeServicesData:\r
255 case EfiMemoryMappedIO:\r
256 case EfiMemoryMappedIOPortSpace:\r
257 MemoryMapEntry->Attribute |= EFI_MEMORY_XP;\r
258 break;\r
259 case EfiReservedMemoryType:\r
260 case EfiACPIMemoryNVS:\r
261 break;\r
262 }\r
263\r
264 MemoryMapEntry = NEXT_MEMORY_DESCRIPTOR (MemoryMapEntry, DescriptorSize);\r
265 }\r
266\r
267 return ;\r
268}\r
269\r
03d486b2
JY
270/**\r
271 Return the first image record, whose [ImageBase, ImageSize] covered by [Buffer, Length].\r
272\r
273 @param Buffer Start Address\r
274 @param Length Address length\r
275\r
276 @return first image record covered by [buffer, length]\r
277**/\r
2caab142 278STATIC\r
03d486b2
JY
279IMAGE_PROPERTIES_RECORD *\r
280GetImageRecordByAddress (\r
281 IN EFI_PHYSICAL_ADDRESS Buffer,\r
282 IN UINT64 Length\r
283 )\r
284{\r
285 IMAGE_PROPERTIES_RECORD *ImageRecord;\r
286 LIST_ENTRY *ImageRecordLink;\r
287 LIST_ENTRY *ImageRecordList;\r
288\r
289 ImageRecordList = &mImagePropertiesPrivateData.ImageRecordList;\r
290\r
291 for (ImageRecordLink = ImageRecordList->ForwardLink;\r
292 ImageRecordLink != ImageRecordList;\r
293 ImageRecordLink = ImageRecordLink->ForwardLink) {\r
294 ImageRecord = CR (\r
295 ImageRecordLink,\r
296 IMAGE_PROPERTIES_RECORD,\r
297 Link,\r
298 IMAGE_PROPERTIES_RECORD_SIGNATURE\r
299 );\r
300\r
301 if ((Buffer <= ImageRecord->ImageBase) &&\r
302 (Buffer + Length >= ImageRecord->ImageBase + ImageRecord->ImageSize)) {\r
303 return ImageRecord;\r
304 }\r
305 }\r
306\r
307 return NULL;\r
308}\r
309\r
310/**\r
311 Set the memory map to new entries, according to one old entry,\r
312 based upon PE code section and data section in image record\r
313\r
314 @param ImageRecord An image record whose [ImageBase, ImageSize] covered\r
315 by old memory map entry.\r
316 @param NewRecord A pointer to several new memory map entries.\r
317 The caller gurantee the buffer size be 1 +\r
318 (SplitRecordCount * DescriptorSize) calculated\r
319 below.\r
320 @param OldRecord A pointer to one old memory map entry.\r
321 @param DescriptorSize Size, in bytes, of an individual EFI_MEMORY_DESCRIPTOR.\r
322**/\r
2caab142 323STATIC\r
03d486b2
JY
324UINTN\r
325SetNewRecord (\r
326 IN IMAGE_PROPERTIES_RECORD *ImageRecord,\r
327 IN OUT EFI_MEMORY_DESCRIPTOR *NewRecord,\r
328 IN EFI_MEMORY_DESCRIPTOR *OldRecord,\r
329 IN UINTN DescriptorSize\r
330 )\r
331{\r
332 EFI_MEMORY_DESCRIPTOR TempRecord;\r
333 IMAGE_PROPERTIES_RECORD_CODE_SECTION *ImageRecordCodeSection;\r
334 LIST_ENTRY *ImageRecordCodeSectionLink;\r
335 LIST_ENTRY *ImageRecordCodeSectionEndLink;\r
336 LIST_ENTRY *ImageRecordCodeSectionList;\r
337 UINTN NewRecordCount;\r
338 UINT64 PhysicalEnd;\r
339 UINT64 ImageEnd;\r
340\r
341 CopyMem (&TempRecord, OldRecord, sizeof(EFI_MEMORY_DESCRIPTOR));\r
342 PhysicalEnd = TempRecord.PhysicalStart + EfiPagesToSize(TempRecord.NumberOfPages);\r
343 NewRecordCount = 0;\r
344\r
345 ImageRecordCodeSectionList = &ImageRecord->CodeSegmentList;\r
346\r
347 ImageRecordCodeSectionLink = ImageRecordCodeSectionList->ForwardLink;\r
348 ImageRecordCodeSectionEndLink = ImageRecordCodeSectionList;\r
349 while (ImageRecordCodeSectionLink != ImageRecordCodeSectionEndLink) {\r
350 ImageRecordCodeSection = CR (\r
351 ImageRecordCodeSectionLink,\r
352 IMAGE_PROPERTIES_RECORD_CODE_SECTION,\r
353 Link,\r
354 IMAGE_PROPERTIES_RECORD_CODE_SECTION_SIGNATURE\r
355 );\r
356 ImageRecordCodeSectionLink = ImageRecordCodeSectionLink->ForwardLink;\r
357\r
358 if (TempRecord.PhysicalStart <= ImageRecordCodeSection->CodeSegmentBase) {\r
359 //\r
360 // DATA\r
361 //\r
82f0f411
JY
362 if (!mPropertiesTableEnable) {\r
363 NewRecord->Type = TempRecord.Type;\r
364 } else {\r
365 NewRecord->Type = EfiRuntimeServicesData;\r
366 }\r
03d486b2
JY
367 NewRecord->PhysicalStart = TempRecord.PhysicalStart;\r
368 NewRecord->VirtualStart = 0;\r
369 NewRecord->NumberOfPages = EfiSizeToPages(ImageRecordCodeSection->CodeSegmentBase - NewRecord->PhysicalStart);\r
370 NewRecord->Attribute = TempRecord.Attribute | EFI_MEMORY_XP;\r
371 if (NewRecord->NumberOfPages != 0) {\r
372 NewRecord = NEXT_MEMORY_DESCRIPTOR (NewRecord, DescriptorSize);\r
373 NewRecordCount ++;\r
374 }\r
375\r
376 //\r
377 // CODE\r
378 //\r
82f0f411
JY
379 if (!mPropertiesTableEnable) {\r
380 NewRecord->Type = TempRecord.Type;\r
381 } else {\r
382 NewRecord->Type = EfiRuntimeServicesCode;\r
383 }\r
03d486b2
JY
384 NewRecord->PhysicalStart = ImageRecordCodeSection->CodeSegmentBase;\r
385 NewRecord->VirtualStart = 0;\r
386 NewRecord->NumberOfPages = EfiSizeToPages(ImageRecordCodeSection->CodeSegmentSize);\r
387 NewRecord->Attribute = (TempRecord.Attribute & (~EFI_MEMORY_XP)) | EFI_MEMORY_RO;\r
388 if (NewRecord->NumberOfPages != 0) {\r
389 NewRecord = NEXT_MEMORY_DESCRIPTOR (NewRecord, DescriptorSize);\r
390 NewRecordCount ++;\r
391 }\r
392\r
393 TempRecord.PhysicalStart = ImageRecordCodeSection->CodeSegmentBase + EfiPagesToSize (EfiSizeToPages(ImageRecordCodeSection->CodeSegmentSize));\r
394 TempRecord.NumberOfPages = EfiSizeToPages(PhysicalEnd - TempRecord.PhysicalStart);\r
395 if (TempRecord.NumberOfPages == 0) {\r
396 break;\r
397 }\r
398 }\r
399 }\r
400\r
401 ImageEnd = ImageRecord->ImageBase + ImageRecord->ImageSize;\r
402\r
403 //\r
404 // Final DATA\r
405 //\r
406 if (TempRecord.PhysicalStart < ImageEnd) {\r
82f0f411
JY
407 if (!mPropertiesTableEnable) {\r
408 NewRecord->Type = TempRecord.Type;\r
409 } else {\r
410 NewRecord->Type = EfiRuntimeServicesData;\r
411 }\r
03d486b2
JY
412 NewRecord->PhysicalStart = TempRecord.PhysicalStart;\r
413 NewRecord->VirtualStart = 0;\r
414 NewRecord->NumberOfPages = EfiSizeToPages (ImageEnd - TempRecord.PhysicalStart);\r
415 NewRecord->Attribute = TempRecord.Attribute | EFI_MEMORY_XP;\r
416 NewRecordCount ++;\r
417 }\r
418\r
419 return NewRecordCount;\r
420}\r
421\r
422/**\r
423 Return the max number of new splitted entries, according to one old entry,\r
424 based upon PE code section and data section.\r
425\r
426 @param OldRecord A pointer to one old memory map entry.\r
427\r
428 @retval 0 no entry need to be splitted.\r
429 @return the max number of new splitted entries\r
430**/\r
2caab142 431STATIC\r
03d486b2
JY
432UINTN\r
433GetMaxSplitRecordCount (\r
434 IN EFI_MEMORY_DESCRIPTOR *OldRecord\r
435 )\r
436{\r
437 IMAGE_PROPERTIES_RECORD *ImageRecord;\r
438 UINTN SplitRecordCount;\r
439 UINT64 PhysicalStart;\r
440 UINT64 PhysicalEnd;\r
441\r
442 SplitRecordCount = 0;\r
443 PhysicalStart = OldRecord->PhysicalStart;\r
444 PhysicalEnd = OldRecord->PhysicalStart + EfiPagesToSize(OldRecord->NumberOfPages);\r
445\r
446 do {\r
447 ImageRecord = GetImageRecordByAddress (PhysicalStart, PhysicalEnd - PhysicalStart);\r
448 if (ImageRecord == NULL) {\r
449 break;\r
450 }\r
451 SplitRecordCount += (2 * ImageRecord->CodeSegmentCount + 1);\r
452 PhysicalStart = ImageRecord->ImageBase + ImageRecord->ImageSize;\r
453 } while ((ImageRecord != NULL) && (PhysicalStart < PhysicalEnd));\r
454\r
455 if (SplitRecordCount != 0) {\r
456 SplitRecordCount--;\r
457 }\r
458\r
459 return SplitRecordCount;\r
460}\r
461\r
462/**\r
463 Split the memory map to new entries, according to one old entry,\r
464 based upon PE code section and data section.\r
465\r
466 @param OldRecord A pointer to one old memory map entry.\r
467 @param NewRecord A pointer to several new memory map entries.\r
468 The caller gurantee the buffer size be 1 +\r
469 (SplitRecordCount * DescriptorSize) calculated\r
470 below.\r
9d58ab09 471 @param MaxSplitRecordCount The max number of splitted entries\r
03d486b2
JY
472 @param DescriptorSize Size, in bytes, of an individual EFI_MEMORY_DESCRIPTOR.\r
473\r
474 @retval 0 no entry is splitted.\r
475 @return the real number of splitted record.\r
476**/\r
2caab142 477STATIC\r
03d486b2
JY
478UINTN\r
479SplitRecord (\r
480 IN EFI_MEMORY_DESCRIPTOR *OldRecord,\r
481 IN OUT EFI_MEMORY_DESCRIPTOR *NewRecord,\r
482 IN UINTN MaxSplitRecordCount,\r
483 IN UINTN DescriptorSize\r
484 )\r
485{\r
486 EFI_MEMORY_DESCRIPTOR TempRecord;\r
487 IMAGE_PROPERTIES_RECORD *ImageRecord;\r
488 IMAGE_PROPERTIES_RECORD *NewImageRecord;\r
489 UINT64 PhysicalStart;\r
490 UINT64 PhysicalEnd;\r
491 UINTN NewRecordCount;\r
492 UINTN TotalNewRecordCount;\r
82f0f411 493 BOOLEAN IsLastRecordData;\r
03d486b2
JY
494\r
495 if (MaxSplitRecordCount == 0) {\r
496 CopyMem (NewRecord, OldRecord, DescriptorSize);\r
497 return 0;\r
498 }\r
499\r
500 TotalNewRecordCount = 0;\r
501\r
502 //\r
503 // Override previous record\r
504 //\r
505 CopyMem (&TempRecord, OldRecord, sizeof(EFI_MEMORY_DESCRIPTOR));\r
506 PhysicalStart = TempRecord.PhysicalStart;\r
507 PhysicalEnd = TempRecord.PhysicalStart + EfiPagesToSize(TempRecord.NumberOfPages);\r
508\r
509 ImageRecord = NULL;\r
510 do {\r
511 NewImageRecord = GetImageRecordByAddress (PhysicalStart, PhysicalEnd - PhysicalStart);\r
512 if (NewImageRecord == NULL) {\r
513 //\r
514 // No more image covered by this range, stop\r
515 //\r
516 if ((PhysicalEnd > PhysicalStart) && (ImageRecord != NULL)) {\r
517 //\r
518 // If this is still address in this record, need record.\r
519 //\r
520 NewRecord = PREVIOUS_MEMORY_DESCRIPTOR (NewRecord, DescriptorSize);\r
82f0f411
JY
521 IsLastRecordData = FALSE;\r
522 if (!mPropertiesTableEnable) {\r
523 if ((NewRecord->Attribute & EFI_MEMORY_XP) != 0) {\r
524 IsLastRecordData = TRUE;\r
525 }\r
526 } else {\r
527 if (NewRecord->Type == EfiRuntimeServicesData) {\r
528 IsLastRecordData = TRUE;\r
529 }\r
530 }\r
531 if (IsLastRecordData) {\r
03d486b2
JY
532 //\r
533 // Last record is DATA, just merge it.\r
534 //\r
535 NewRecord->NumberOfPages = EfiSizeToPages(PhysicalEnd - NewRecord->PhysicalStart);\r
536 } else {\r
537 //\r
538 // Last record is CODE, create a new DATA entry.\r
539 //\r
540 NewRecord = NEXT_MEMORY_DESCRIPTOR (NewRecord, DescriptorSize);\r
82f0f411
JY
541 if (!mPropertiesTableEnable) {\r
542 NewRecord->Type = TempRecord.Type;\r
543 } else {\r
544 NewRecord->Type = EfiRuntimeServicesData;\r
545 }\r
03d486b2
JY
546 NewRecord->PhysicalStart = TempRecord.PhysicalStart;\r
547 NewRecord->VirtualStart = 0;\r
548 NewRecord->NumberOfPages = TempRecord.NumberOfPages;\r
549 NewRecord->Attribute = TempRecord.Attribute | EFI_MEMORY_XP;\r
550 TotalNewRecordCount ++;\r
551 }\r
552 }\r
553 break;\r
554 }\r
555 ImageRecord = NewImageRecord;\r
556\r
557 //\r
558 // Set new record\r
559 //\r
560 NewRecordCount = SetNewRecord (ImageRecord, NewRecord, &TempRecord, DescriptorSize);\r
561 TotalNewRecordCount += NewRecordCount;\r
562 NewRecord = (EFI_MEMORY_DESCRIPTOR *)((UINT8 *)NewRecord + NewRecordCount * DescriptorSize);\r
563\r
564 //\r
565 // Update PhysicalStart, in order to exclude the image buffer already splitted.\r
566 //\r
567 PhysicalStart = ImageRecord->ImageBase + ImageRecord->ImageSize;\r
568 TempRecord.PhysicalStart = PhysicalStart;\r
569 TempRecord.NumberOfPages = EfiSizeToPages (PhysicalEnd - PhysicalStart);\r
570 } while ((ImageRecord != NULL) && (PhysicalStart < PhysicalEnd));\r
571\r
1860cb00
HW
572 //\r
573 // The logic in function SplitTable() ensures that TotalNewRecordCount will not be zero if the\r
574 // code reaches here.\r
575 //\r
576 ASSERT (TotalNewRecordCount != 0);\r
03d486b2
JY
577 return TotalNewRecordCount - 1;\r
578}\r
579\r
580/**\r
581 Split the original memory map, and add more entries to describe PE code section and data section.\r
582 This function will set EfiRuntimeServicesData to be EFI_MEMORY_XP.\r
583 This function will merge entries with same attributes finally.\r
584\r
585 NOTE: It assumes PE code/data section are page aligned.\r
586 NOTE: It assumes enough entry is prepared for new memory map.\r
587\r
588 Split table:\r
589 +---------------+\r
590 | Record X |\r
591 +---------------+\r
592 | Record RtCode |\r
593 +---------------+\r
594 | Record Y |\r
595 +---------------+\r
596 ==>\r
597 +---------------+\r
598 | Record X |\r
599 +---------------+ ----\r
600 | Record RtData | |\r
601 +---------------+ |\r
602 | Record RtCode | |-> PE/COFF1\r
603 +---------------+ |\r
604 | Record RtData | |\r
605 +---------------+ ----\r
606 | Record RtData | |\r
607 +---------------+ |\r
608 | Record RtCode | |-> PE/COFF2\r
609 +---------------+ |\r
610 | Record RtData | |\r
611 +---------------+ ----\r
612 | Record Y |\r
613 +---------------+\r
614\r
615 @param MemoryMapSize A pointer to the size, in bytes, of the\r
616 MemoryMap buffer. On input, this is the size of\r
617 old MemoryMap before split. The actual buffer\r
618 size of MemoryMap is MemoryMapSize +\r
619 (AdditionalRecordCount * DescriptorSize) calculated\r
620 below. On output, it is the size of new MemoryMap\r
621 after split.\r
622 @param MemoryMap A pointer to the buffer in which firmware places\r
623 the current memory map.\r
624 @param DescriptorSize Size, in bytes, of an individual EFI_MEMORY_DESCRIPTOR.\r
625**/\r
2caab142 626STATIC\r
03d486b2
JY
627VOID\r
628SplitTable (\r
629 IN OUT UINTN *MemoryMapSize,\r
630 IN OUT EFI_MEMORY_DESCRIPTOR *MemoryMap,\r
631 IN UINTN DescriptorSize\r
632 )\r
633{\r
634 INTN IndexOld;\r
635 INTN IndexNew;\r
636 UINTN MaxSplitRecordCount;\r
637 UINTN RealSplitRecordCount;\r
638 UINTN TotalSplitRecordCount;\r
639 UINTN AdditionalRecordCount;\r
640\r
641 AdditionalRecordCount = (2 * mImagePropertiesPrivateData.CodeSegmentCountMax + 1) * mImagePropertiesPrivateData.ImageRecordCount;\r
642\r
643 TotalSplitRecordCount = 0;\r
644 //\r
645 // Let old record point to end of valid MemoryMap buffer.\r
646 //\r
647 IndexOld = ((*MemoryMapSize) / DescriptorSize) - 1;\r
648 //\r
649 // Let new record point to end of full MemoryMap buffer.\r
650 //\r
651 IndexNew = ((*MemoryMapSize) / DescriptorSize) - 1 + AdditionalRecordCount;\r
652 for (; IndexOld >= 0; IndexOld--) {\r
653 MaxSplitRecordCount = GetMaxSplitRecordCount ((EFI_MEMORY_DESCRIPTOR *)((UINT8 *)MemoryMap + IndexOld * DescriptorSize));\r
654 //\r
655 // Split this MemoryMap record\r
656 //\r
657 IndexNew -= MaxSplitRecordCount;\r
658 RealSplitRecordCount = SplitRecord (\r
659 (EFI_MEMORY_DESCRIPTOR *)((UINT8 *)MemoryMap + IndexOld * DescriptorSize),\r
660 (EFI_MEMORY_DESCRIPTOR *)((UINT8 *)MemoryMap + IndexNew * DescriptorSize),\r
661 MaxSplitRecordCount,\r
662 DescriptorSize\r
663 );\r
664 //\r
665 // Adjust IndexNew according to real split.\r
666 //\r
667 CopyMem (\r
668 ((UINT8 *)MemoryMap + (IndexNew + MaxSplitRecordCount - RealSplitRecordCount) * DescriptorSize),\r
669 ((UINT8 *)MemoryMap + IndexNew * DescriptorSize),\r
670 RealSplitRecordCount * DescriptorSize\r
671 );\r
672 IndexNew = IndexNew + MaxSplitRecordCount - RealSplitRecordCount;\r
673 TotalSplitRecordCount += RealSplitRecordCount;\r
674 IndexNew --;\r
675 }\r
676 //\r
677 // Move all records to the beginning.\r
678 //\r
679 CopyMem (\r
680 MemoryMap,\r
681 (UINT8 *)MemoryMap + (AdditionalRecordCount - TotalSplitRecordCount) * DescriptorSize,\r
682 (*MemoryMapSize) + TotalSplitRecordCount * DescriptorSize\r
683 );\r
684\r
685 *MemoryMapSize = (*MemoryMapSize) + DescriptorSize * TotalSplitRecordCount;\r
686\r
687 //\r
688 // Sort from low to high (Just in case)\r
689 //\r
690 SortMemoryMap (MemoryMap, *MemoryMapSize, DescriptorSize);\r
691\r
692 //\r
693 // Set RuntimeData to XP\r
694 //\r
695 EnforceMemoryMapAttribute (MemoryMap, *MemoryMapSize, DescriptorSize);\r
696\r
697 //\r
698 // Merge same type to save entry size\r
699 //\r
700 MergeMemoryMap (MemoryMap, MemoryMapSize, DescriptorSize);\r
701\r
702 return ;\r
703}\r
704\r
705/**\r
f5a91bba 706 This function for GetMemoryMap() with properties table capability.\r
03d486b2
JY
707\r
708 It calls original GetMemoryMap() to get the original memory map information. Then\r
709 plus the additional memory map entries for PE Code/Data seperation.\r
710\r
711 @param MemoryMapSize A pointer to the size, in bytes, of the\r
712 MemoryMap buffer. On input, this is the size of\r
713 the buffer allocated by the caller. On output,\r
714 it is the size of the buffer returned by the\r
715 firmware if the buffer was large enough, or the\r
716 size of the buffer needed to contain the map if\r
717 the buffer was too small.\r
718 @param MemoryMap A pointer to the buffer in which firmware places\r
719 the current memory map.\r
720 @param MapKey A pointer to the location in which firmware\r
721 returns the key for the current memory map.\r
722 @param DescriptorSize A pointer to the location in which firmware\r
723 returns the size, in bytes, of an individual\r
724 EFI_MEMORY_DESCRIPTOR.\r
725 @param DescriptorVersion A pointer to the location in which firmware\r
726 returns the version number associated with the\r
727 EFI_MEMORY_DESCRIPTOR.\r
728\r
729 @retval EFI_SUCCESS The memory map was returned in the MemoryMap\r
730 buffer.\r
731 @retval EFI_BUFFER_TOO_SMALL The MemoryMap buffer was too small. The current\r
732 buffer size needed to hold the memory map is\r
733 returned in MemoryMapSize.\r
734 @retval EFI_INVALID_PARAMETER One of the parameters has an invalid value.\r
735\r
736**/\r
737EFI_STATUS\r
738EFIAPI\r
0ab90add 739CoreGetMemoryMapWithSeparatedImageSection (\r
03d486b2
JY
740 IN OUT UINTN *MemoryMapSize,\r
741 IN OUT EFI_MEMORY_DESCRIPTOR *MemoryMap,\r
742 OUT UINTN *MapKey,\r
743 OUT UINTN *DescriptorSize,\r
744 OUT UINT32 *DescriptorVersion\r
745 )\r
746{\r
747 EFI_STATUS Status;\r
748 UINTN OldMemoryMapSize;\r
749 UINTN AdditionalRecordCount;\r
750\r
751 //\r
752 // If PE code/data is not aligned, just return.\r
753 //\r
754 if ((mPropertiesTable.MemoryProtectionAttribute & EFI_PROPERTIES_RUNTIME_MEMORY_PROTECTION_NON_EXECUTABLE_PE_DATA) == 0) {\r
755 return CoreGetMemoryMap (MemoryMapSize, MemoryMap, MapKey, DescriptorSize, DescriptorVersion);\r
756 }\r
757\r
758 if (MemoryMapSize == NULL) {\r
759 return EFI_INVALID_PARAMETER;\r
760 }\r
761\r
762 CoreAcquirePropertiesTableLock ();\r
763\r
764 AdditionalRecordCount = (2 * mImagePropertiesPrivateData.CodeSegmentCountMax + 1) * mImagePropertiesPrivateData.ImageRecordCount;\r
765\r
766 OldMemoryMapSize = *MemoryMapSize;\r
767 Status = CoreGetMemoryMap (MemoryMapSize, MemoryMap, MapKey, DescriptorSize, DescriptorVersion);\r
768 if (Status == EFI_BUFFER_TOO_SMALL) {\r
769 *MemoryMapSize = *MemoryMapSize + (*DescriptorSize) * AdditionalRecordCount;\r
770 } else if (Status == EFI_SUCCESS) {\r
4cb6375c 771 ASSERT (MemoryMap != NULL);\r
03d486b2
JY
772 if (OldMemoryMapSize - *MemoryMapSize < (*DescriptorSize) * AdditionalRecordCount) {\r
773 *MemoryMapSize = *MemoryMapSize + (*DescriptorSize) * AdditionalRecordCount;\r
774 //\r
775 // Need update status to buffer too small\r
776 //\r
777 Status = EFI_BUFFER_TOO_SMALL;\r
778 } else {\r
779 //\r
780 // Split PE code/data\r
781 //\r
782 SplitTable (MemoryMapSize, MemoryMap, *DescriptorSize);\r
783 }\r
784 }\r
785\r
786 CoreReleasePropertiesTableLock ();\r
787 return Status;\r
788}\r
789\r
790//\r
791// Below functions are for ImageRecord\r
792//\r
793\r
794/**\r
ab91c9a8 795 Set PropertiesTable according to PE/COFF image section alignment.\r
03d486b2
JY
796\r
797 @param SectionAlignment PE/COFF section alignment\r
798**/\r
2caab142 799STATIC\r
03d486b2
JY
800VOID\r
801SetPropertiesTableSectionAlignment (\r
802 IN UINT32 SectionAlignment\r
803 )\r
804{\r
d4731a98 805 if (((SectionAlignment & (RUNTIME_PAGE_ALLOCATION_GRANULARITY - 1)) != 0) &&\r
03d486b2
JY
806 ((mPropertiesTable.MemoryProtectionAttribute & EFI_PROPERTIES_RUNTIME_MEMORY_PROTECTION_NON_EXECUTABLE_PE_DATA) != 0)) {\r
807 DEBUG ((EFI_D_VERBOSE, "SetPropertiesTableSectionAlignment - Clear\n"));\r
a6451806 808 mPropertiesTable.MemoryProtectionAttribute &= ~((UINT64)EFI_PROPERTIES_RUNTIME_MEMORY_PROTECTION_NON_EXECUTABLE_PE_DATA);\r
03d486b2
JY
809 gBS->GetMemoryMap = CoreGetMemoryMap;\r
810 gBS->Hdr.CRC32 = 0;\r
811 gBS->CalculateCrc32 ((UINT8 *)gBS, gBS->Hdr.HeaderSize, &gBS->Hdr.CRC32);\r
812 }\r
813}\r
814\r
815/**\r
816 Swap two code sections in image record.\r
817\r
818 @param FirstImageRecordCodeSection first code section in image record\r
819 @param SecondImageRecordCodeSection second code section in image record\r
820**/\r
2caab142 821STATIC\r
03d486b2
JY
822VOID\r
823SwapImageRecordCodeSection (\r
824 IN IMAGE_PROPERTIES_RECORD_CODE_SECTION *FirstImageRecordCodeSection,\r
825 IN IMAGE_PROPERTIES_RECORD_CODE_SECTION *SecondImageRecordCodeSection\r
826 )\r
827{\r
828 IMAGE_PROPERTIES_RECORD_CODE_SECTION TempImageRecordCodeSection;\r
829\r
830 TempImageRecordCodeSection.CodeSegmentBase = FirstImageRecordCodeSection->CodeSegmentBase;\r
831 TempImageRecordCodeSection.CodeSegmentSize = FirstImageRecordCodeSection->CodeSegmentSize;\r
832\r
833 FirstImageRecordCodeSection->CodeSegmentBase = SecondImageRecordCodeSection->CodeSegmentBase;\r
834 FirstImageRecordCodeSection->CodeSegmentSize = SecondImageRecordCodeSection->CodeSegmentSize;\r
835\r
836 SecondImageRecordCodeSection->CodeSegmentBase = TempImageRecordCodeSection.CodeSegmentBase;\r
837 SecondImageRecordCodeSection->CodeSegmentSize = TempImageRecordCodeSection.CodeSegmentSize;\r
838}\r
839\r
840/**\r
841 Sort code section in image record, based upon CodeSegmentBase from low to high.\r
842\r
843 @param ImageRecord image record to be sorted\r
844**/\r
845VOID\r
846SortImageRecordCodeSection (\r
847 IN IMAGE_PROPERTIES_RECORD *ImageRecord\r
848 )\r
849{\r
850 IMAGE_PROPERTIES_RECORD_CODE_SECTION *ImageRecordCodeSection;\r
851 IMAGE_PROPERTIES_RECORD_CODE_SECTION *NextImageRecordCodeSection;\r
852 LIST_ENTRY *ImageRecordCodeSectionLink;\r
853 LIST_ENTRY *NextImageRecordCodeSectionLink;\r
854 LIST_ENTRY *ImageRecordCodeSectionEndLink;\r
855 LIST_ENTRY *ImageRecordCodeSectionList;\r
856\r
857 ImageRecordCodeSectionList = &ImageRecord->CodeSegmentList;\r
858\r
859 ImageRecordCodeSectionLink = ImageRecordCodeSectionList->ForwardLink;\r
860 NextImageRecordCodeSectionLink = ImageRecordCodeSectionLink->ForwardLink;\r
861 ImageRecordCodeSectionEndLink = ImageRecordCodeSectionList;\r
862 while (ImageRecordCodeSectionLink != ImageRecordCodeSectionEndLink) {\r
863 ImageRecordCodeSection = CR (\r
864 ImageRecordCodeSectionLink,\r
865 IMAGE_PROPERTIES_RECORD_CODE_SECTION,\r
866 Link,\r
867 IMAGE_PROPERTIES_RECORD_CODE_SECTION_SIGNATURE\r
868 );\r
869 while (NextImageRecordCodeSectionLink != ImageRecordCodeSectionEndLink) {\r
870 NextImageRecordCodeSection = CR (\r
871 NextImageRecordCodeSectionLink,\r
872 IMAGE_PROPERTIES_RECORD_CODE_SECTION,\r
873 Link,\r
874 IMAGE_PROPERTIES_RECORD_CODE_SECTION_SIGNATURE\r
875 );\r
876 if (ImageRecordCodeSection->CodeSegmentBase > NextImageRecordCodeSection->CodeSegmentBase) {\r
877 SwapImageRecordCodeSection (ImageRecordCodeSection, NextImageRecordCodeSection);\r
878 }\r
879 NextImageRecordCodeSectionLink = NextImageRecordCodeSectionLink->ForwardLink;\r
880 }\r
881\r
882 ImageRecordCodeSectionLink = ImageRecordCodeSectionLink->ForwardLink;\r
883 NextImageRecordCodeSectionLink = ImageRecordCodeSectionLink->ForwardLink;\r
884 }\r
885}\r
886\r
887/**\r
9d58ab09 888 Check if code section in image record is valid.\r
03d486b2
JY
889\r
890 @param ImageRecord image record to be checked\r
891\r
892 @retval TRUE image record is valid\r
893 @retval FALSE image record is invalid\r
894**/\r
895BOOLEAN\r
896IsImageRecordCodeSectionValid (\r
897 IN IMAGE_PROPERTIES_RECORD *ImageRecord\r
898 )\r
899{\r
900 IMAGE_PROPERTIES_RECORD_CODE_SECTION *ImageRecordCodeSection;\r
901 IMAGE_PROPERTIES_RECORD_CODE_SECTION *LastImageRecordCodeSection;\r
902 LIST_ENTRY *ImageRecordCodeSectionLink;\r
903 LIST_ENTRY *ImageRecordCodeSectionEndLink;\r
904 LIST_ENTRY *ImageRecordCodeSectionList;\r
905\r
906 DEBUG ((EFI_D_VERBOSE, "ImageCode SegmentCount - 0x%x\n", ImageRecord->CodeSegmentCount));\r
907\r
908 ImageRecordCodeSectionList = &ImageRecord->CodeSegmentList;\r
909\r
910 ImageRecordCodeSectionLink = ImageRecordCodeSectionList->ForwardLink;\r
911 ImageRecordCodeSectionEndLink = ImageRecordCodeSectionList;\r
912 LastImageRecordCodeSection = NULL;\r
913 while (ImageRecordCodeSectionLink != ImageRecordCodeSectionEndLink) {\r
914 ImageRecordCodeSection = CR (\r
915 ImageRecordCodeSectionLink,\r
916 IMAGE_PROPERTIES_RECORD_CODE_SECTION,\r
917 Link,\r
918 IMAGE_PROPERTIES_RECORD_CODE_SECTION_SIGNATURE\r
919 );\r
920 if (ImageRecordCodeSection->CodeSegmentSize == 0) {\r
921 return FALSE;\r
922 }\r
923 if (ImageRecordCodeSection->CodeSegmentBase < ImageRecord->ImageBase) {\r
924 return FALSE;\r
925 }\r
926 if (ImageRecordCodeSection->CodeSegmentBase >= MAX_ADDRESS - ImageRecordCodeSection->CodeSegmentSize) {\r
927 return FALSE;\r
928 }\r
929 if ((ImageRecordCodeSection->CodeSegmentBase + ImageRecordCodeSection->CodeSegmentSize) > (ImageRecord->ImageBase + ImageRecord->ImageSize)) {\r
930 return FALSE;\r
931 }\r
932 if (LastImageRecordCodeSection != NULL) {\r
933 if ((LastImageRecordCodeSection->CodeSegmentBase + LastImageRecordCodeSection->CodeSegmentSize) > ImageRecordCodeSection->CodeSegmentBase) {\r
934 return FALSE;\r
935 }\r
936 }\r
937\r
938 LastImageRecordCodeSection = ImageRecordCodeSection;\r
939 ImageRecordCodeSectionLink = ImageRecordCodeSectionLink->ForwardLink;\r
940 }\r
941\r
942 return TRUE;\r
943}\r
944\r
945/**\r
946 Swap two image records.\r
947\r
948 @param FirstImageRecord first image record.\r
949 @param SecondImageRecord second image record.\r
950**/\r
2caab142 951STATIC\r
03d486b2
JY
952VOID\r
953SwapImageRecord (\r
954 IN IMAGE_PROPERTIES_RECORD *FirstImageRecord,\r
955 IN IMAGE_PROPERTIES_RECORD *SecondImageRecord\r
956 )\r
957{\r
958 IMAGE_PROPERTIES_RECORD TempImageRecord;\r
959\r
960 TempImageRecord.ImageBase = FirstImageRecord->ImageBase;\r
961 TempImageRecord.ImageSize = FirstImageRecord->ImageSize;\r
962 TempImageRecord.CodeSegmentCount = FirstImageRecord->CodeSegmentCount;\r
963\r
964 FirstImageRecord->ImageBase = SecondImageRecord->ImageBase;\r
965 FirstImageRecord->ImageSize = SecondImageRecord->ImageSize;\r
966 FirstImageRecord->CodeSegmentCount = SecondImageRecord->CodeSegmentCount;\r
967\r
968 SecondImageRecord->ImageBase = TempImageRecord.ImageBase;\r
969 SecondImageRecord->ImageSize = TempImageRecord.ImageSize;\r
970 SecondImageRecord->CodeSegmentCount = TempImageRecord.CodeSegmentCount;\r
971\r
972 SwapListEntries (&FirstImageRecord->CodeSegmentList, &SecondImageRecord->CodeSegmentList);\r
973}\r
974\r
975/**\r
976 Sort image record based upon the ImageBase from low to high.\r
977**/\r
2caab142 978STATIC\r
03d486b2
JY
979VOID\r
980SortImageRecord (\r
981 VOID\r
982 )\r
983{\r
984 IMAGE_PROPERTIES_RECORD *ImageRecord;\r
985 IMAGE_PROPERTIES_RECORD *NextImageRecord;\r
986 LIST_ENTRY *ImageRecordLink;\r
987 LIST_ENTRY *NextImageRecordLink;\r
988 LIST_ENTRY *ImageRecordEndLink;\r
989 LIST_ENTRY *ImageRecordList;\r
990\r
991 ImageRecordList = &mImagePropertiesPrivateData.ImageRecordList;\r
992\r
993 ImageRecordLink = ImageRecordList->ForwardLink;\r
994 NextImageRecordLink = ImageRecordLink->ForwardLink;\r
995 ImageRecordEndLink = ImageRecordList;\r
996 while (ImageRecordLink != ImageRecordEndLink) {\r
997 ImageRecord = CR (\r
998 ImageRecordLink,\r
999 IMAGE_PROPERTIES_RECORD,\r
1000 Link,\r
1001 IMAGE_PROPERTIES_RECORD_SIGNATURE\r
1002 );\r
1003 while (NextImageRecordLink != ImageRecordEndLink) {\r
1004 NextImageRecord = CR (\r
1005 NextImageRecordLink,\r
1006 IMAGE_PROPERTIES_RECORD,\r
1007 Link,\r
1008 IMAGE_PROPERTIES_RECORD_SIGNATURE\r
1009 );\r
1010 if (ImageRecord->ImageBase > NextImageRecord->ImageBase) {\r
1011 SwapImageRecord (ImageRecord, NextImageRecord);\r
1012 }\r
1013 NextImageRecordLink = NextImageRecordLink->ForwardLink;\r
1014 }\r
1015\r
1016 ImageRecordLink = ImageRecordLink->ForwardLink;\r
1017 NextImageRecordLink = ImageRecordLink->ForwardLink;\r
1018 }\r
1019}\r
1020\r
1021/**\r
9d58ab09 1022 Dump image record.\r
03d486b2 1023**/\r
2caab142 1024STATIC\r
03d486b2
JY
1025VOID\r
1026DumpImageRecord (\r
1027 VOID\r
1028 )\r
1029{\r
1030 IMAGE_PROPERTIES_RECORD *ImageRecord;\r
1031 LIST_ENTRY *ImageRecordLink;\r
1032 LIST_ENTRY *ImageRecordList;\r
1033 UINTN Index;\r
1034\r
1035 ImageRecordList = &mImagePropertiesPrivateData.ImageRecordList;\r
1036\r
1037 for (ImageRecordLink = ImageRecordList->ForwardLink, Index= 0;\r
1038 ImageRecordLink != ImageRecordList;\r
1039 ImageRecordLink = ImageRecordLink->ForwardLink, Index++) {\r
1040 ImageRecord = CR (\r
1041 ImageRecordLink,\r
1042 IMAGE_PROPERTIES_RECORD,\r
1043 Link,\r
1044 IMAGE_PROPERTIES_RECORD_SIGNATURE\r
1045 );\r
1046 DEBUG ((EFI_D_VERBOSE, " Image[%d]: 0x%016lx - 0x%016lx\n", Index, ImageRecord->ImageBase, ImageRecord->ImageSize));\r
1047 }\r
1048}\r
1049\r
1050/**\r
1051 Insert image record.\r
1052\r
1053 @param RuntimeImage Runtime image information\r
1054**/\r
1055VOID\r
1056InsertImageRecord (\r
1057 IN EFI_RUNTIME_IMAGE_ENTRY *RuntimeImage\r
1058 )\r
1059{\r
1060 VOID *ImageAddress;\r
1061 EFI_IMAGE_DOS_HEADER *DosHdr;\r
1062 UINT32 PeCoffHeaderOffset;\r
1063 UINT32 SectionAlignment;\r
03d486b2
JY
1064 EFI_IMAGE_SECTION_HEADER *Section;\r
1065 EFI_IMAGE_OPTIONAL_HEADER_PTR_UNION Hdr;\r
03d486b2
JY
1066 UINT8 *Name;\r
1067 UINTN Index;\r
1068 IMAGE_PROPERTIES_RECORD *ImageRecord;\r
1069 CHAR8 *PdbPointer;\r
1070 IMAGE_PROPERTIES_RECORD_CODE_SECTION *ImageRecordCodeSection;\r
1071\r
1072 DEBUG ((EFI_D_VERBOSE, "InsertImageRecord - 0x%x\n", RuntimeImage));\r
1073 DEBUG ((EFI_D_VERBOSE, "InsertImageRecord - 0x%016lx - 0x%016lx\n", (EFI_PHYSICAL_ADDRESS)(UINTN)RuntimeImage->ImageBase, RuntimeImage->ImageSize));\r
1074\r
4a723ed2
JY
1075 if (mPropertiesTableEndOfDxe) {\r
1076 DEBUG ((DEBUG_INFO, "Do not insert runtime image record after EndOfDxe\n"));\r
1077 return ;\r
1078 }\r
1079\r
03d486b2
JY
1080 ImageRecord = AllocatePool (sizeof(*ImageRecord));\r
1081 if (ImageRecord == NULL) {\r
1082 return ;\r
1083 }\r
1084 ImageRecord->Signature = IMAGE_PROPERTIES_RECORD_SIGNATURE;\r
1085\r
1086 DEBUG ((EFI_D_VERBOSE, "ImageRecordCount - 0x%x\n", mImagePropertiesPrivateData.ImageRecordCount));\r
1087\r
1088 //\r
1089 // Step 1: record whole region\r
1090 //\r
1091 ImageRecord->ImageBase = (EFI_PHYSICAL_ADDRESS)(UINTN)RuntimeImage->ImageBase;\r
1092 ImageRecord->ImageSize = RuntimeImage->ImageSize;\r
1093\r
1094 ImageAddress = RuntimeImage->ImageBase;\r
1095\r
1096 PdbPointer = PeCoffLoaderGetPdbPointer ((VOID*) (UINTN) ImageAddress);\r
1097 if (PdbPointer != NULL) {\r
1098 DEBUG ((EFI_D_VERBOSE, " Image - %a\n", PdbPointer));\r
1099 }\r
1100\r
1101 //\r
1102 // Check PE/COFF image\r
1103 //\r
1104 DosHdr = (EFI_IMAGE_DOS_HEADER *) (UINTN) ImageAddress;\r
1105 PeCoffHeaderOffset = 0;\r
1106 if (DosHdr->e_magic == EFI_IMAGE_DOS_SIGNATURE) {\r
1107 PeCoffHeaderOffset = DosHdr->e_lfanew;\r
1108 }\r
1109\r
1110 Hdr.Pe32 = (EFI_IMAGE_NT_HEADERS32 *)((UINT8 *) (UINTN) ImageAddress + PeCoffHeaderOffset);\r
1111 if (Hdr.Pe32->Signature != EFI_IMAGE_NT_SIGNATURE) {\r
1112 DEBUG ((EFI_D_VERBOSE, "Hdr.Pe32->Signature invalid - 0x%x\n", Hdr.Pe32->Signature));\r
81338070 1113 // It might be image in SMM.\r
03d486b2
JY
1114 goto Finish;\r
1115 }\r
1116\r
1117 //\r
81338070 1118 // Get SectionAlignment\r
03d486b2 1119 //\r
05cfd73f 1120 if (Hdr.Pe32->OptionalHeader.Magic == EFI_IMAGE_NT_OPTIONAL_HDR32_MAGIC) {\r
81338070
JY
1121 SectionAlignment = Hdr.Pe32->OptionalHeader.SectionAlignment;\r
1122 } else {\r
03d486b2
JY
1123 SectionAlignment = Hdr.Pe32Plus->OptionalHeader.SectionAlignment;\r
1124 }\r
1125\r
1126 SetPropertiesTableSectionAlignment (SectionAlignment);\r
d4731a98 1127 if ((SectionAlignment & (RUNTIME_PAGE_ALLOCATION_GRANULARITY - 1)) != 0) {\r
2495c9bf 1128 DEBUG ((EFI_D_WARN, "!!!!!!!! InsertImageRecord - Section Alignment(0x%x) is not %dK !!!!!!!!\n",\r
d4731a98 1129 SectionAlignment, RUNTIME_PAGE_ALLOCATION_GRANULARITY >> 10));\r
03d486b2
JY
1130 PdbPointer = PeCoffLoaderGetPdbPointer ((VOID*) (UINTN) ImageAddress);\r
1131 if (PdbPointer != NULL) {\r
2495c9bf 1132 DEBUG ((EFI_D_WARN, "!!!!!!!! Image - %a !!!!!!!!\n", PdbPointer));\r
03d486b2
JY
1133 }\r
1134 goto Finish;\r
1135 }\r
1136\r
1137 Section = (EFI_IMAGE_SECTION_HEADER *) (\r
1138 (UINT8 *) (UINTN) ImageAddress +\r
1139 PeCoffHeaderOffset +\r
1140 sizeof(UINT32) +\r
1141 sizeof(EFI_IMAGE_FILE_HEADER) +\r
1142 Hdr.Pe32->FileHeader.SizeOfOptionalHeader\r
1143 );\r
1144 ImageRecord->CodeSegmentCount = 0;\r
1145 InitializeListHead (&ImageRecord->CodeSegmentList);\r
1146 for (Index = 0; Index < Hdr.Pe32->FileHeader.NumberOfSections; Index++) {\r
1147 Name = Section[Index].Name;\r
1148 DEBUG ((\r
1149 EFI_D_VERBOSE,\r
1150 " Section - '%c%c%c%c%c%c%c%c'\n",\r
1151 Name[0],\r
1152 Name[1],\r
1153 Name[2],\r
1154 Name[3],\r
1155 Name[4],\r
1156 Name[5],\r
1157 Name[6],\r
1158 Name[7]\r
1159 ));\r
1160\r
1161 if ((Section[Index].Characteristics & EFI_IMAGE_SCN_CNT_CODE) != 0) {\r
1162 DEBUG ((EFI_D_VERBOSE, " VirtualSize - 0x%08x\n", Section[Index].Misc.VirtualSize));\r
1163 DEBUG ((EFI_D_VERBOSE, " VirtualAddress - 0x%08x\n", Section[Index].VirtualAddress));\r
1164 DEBUG ((EFI_D_VERBOSE, " SizeOfRawData - 0x%08x\n", Section[Index].SizeOfRawData));\r
1165 DEBUG ((EFI_D_VERBOSE, " PointerToRawData - 0x%08x\n", Section[Index].PointerToRawData));\r
1166 DEBUG ((EFI_D_VERBOSE, " PointerToRelocations - 0x%08x\n", Section[Index].PointerToRelocations));\r
1167 DEBUG ((EFI_D_VERBOSE, " PointerToLinenumbers - 0x%08x\n", Section[Index].PointerToLinenumbers));\r
1168 DEBUG ((EFI_D_VERBOSE, " NumberOfRelocations - 0x%08x\n", Section[Index].NumberOfRelocations));\r
1169 DEBUG ((EFI_D_VERBOSE, " NumberOfLinenumbers - 0x%08x\n", Section[Index].NumberOfLinenumbers));\r
1170 DEBUG ((EFI_D_VERBOSE, " Characteristics - 0x%08x\n", Section[Index].Characteristics));\r
1171\r
1172 //\r
1173 // Step 2: record code section\r
1174 //\r
1175 ImageRecordCodeSection = AllocatePool (sizeof(*ImageRecordCodeSection));\r
1176 if (ImageRecordCodeSection == NULL) {\r
1177 return ;\r
1178 }\r
1179 ImageRecordCodeSection->Signature = IMAGE_PROPERTIES_RECORD_CODE_SECTION_SIGNATURE;\r
1180\r
1181 ImageRecordCodeSection->CodeSegmentBase = (UINTN)ImageAddress + Section[Index].VirtualAddress;\r
1182 ImageRecordCodeSection->CodeSegmentSize = Section[Index].SizeOfRawData;\r
1183\r
1184 DEBUG ((EFI_D_VERBOSE, "ImageCode: 0x%016lx - 0x%016lx\n", ImageRecordCodeSection->CodeSegmentBase, ImageRecordCodeSection->CodeSegmentSize));\r
1185\r
1186 InsertTailList (&ImageRecord->CodeSegmentList, &ImageRecordCodeSection->Link);\r
1187 ImageRecord->CodeSegmentCount++;\r
1188 }\r
1189 }\r
1190\r
1191 if (ImageRecord->CodeSegmentCount == 0) {\r
1192 SetPropertiesTableSectionAlignment (1);\r
1193 DEBUG ((EFI_D_ERROR, "!!!!!!!! InsertImageRecord - CodeSegmentCount is 0 !!!!!!!!\n"));\r
1194 PdbPointer = PeCoffLoaderGetPdbPointer ((VOID*) (UINTN) ImageAddress);\r
1195 if (PdbPointer != NULL) {\r
1196 DEBUG ((EFI_D_ERROR, "!!!!!!!! Image - %a !!!!!!!!\n", PdbPointer));\r
1197 }\r
1198 goto Finish;\r
1199 }\r
1200\r
1201 //\r
1202 // Final\r
1203 //\r
1204 SortImageRecordCodeSection (ImageRecord);\r
1205 //\r
1206 // Check overlap all section in ImageBase/Size\r
1207 //\r
1208 if (!IsImageRecordCodeSectionValid (ImageRecord)) {\r
1209 DEBUG ((EFI_D_ERROR, "IsImageRecordCodeSectionValid - FAIL\n"));\r
1210 goto Finish;\r
1211 }\r
1212\r
1213 InsertTailList (&mImagePropertiesPrivateData.ImageRecordList, &ImageRecord->Link);\r
1214 mImagePropertiesPrivateData.ImageRecordCount++;\r
1215\r
03d486b2
JY
1216 if (mImagePropertiesPrivateData.CodeSegmentCountMax < ImageRecord->CodeSegmentCount) {\r
1217 mImagePropertiesPrivateData.CodeSegmentCountMax = ImageRecord->CodeSegmentCount;\r
1218 }\r
1219\r
3d817fd1
JW
1220 SortImageRecord ();\r
1221\r
03d486b2
JY
1222Finish:\r
1223 return ;\r
1224}\r
1225\r
1226/**\r
ab91c9a8 1227 Find image record according to image base and size.\r
03d486b2
JY
1228\r
1229 @param ImageBase Base of PE image\r
1230 @param ImageSize Size of PE image\r
1231\r
1232 @return image record\r
1233**/\r
2caab142 1234STATIC\r
03d486b2
JY
1235IMAGE_PROPERTIES_RECORD *\r
1236FindImageRecord (\r
1237 IN EFI_PHYSICAL_ADDRESS ImageBase,\r
1238 IN UINT64 ImageSize\r
1239 )\r
1240{\r
1241 IMAGE_PROPERTIES_RECORD *ImageRecord;\r
1242 LIST_ENTRY *ImageRecordLink;\r
1243 LIST_ENTRY *ImageRecordList;\r
1244\r
1245 ImageRecordList = &mImagePropertiesPrivateData.ImageRecordList;\r
1246\r
1247 for (ImageRecordLink = ImageRecordList->ForwardLink;\r
1248 ImageRecordLink != ImageRecordList;\r
1249 ImageRecordLink = ImageRecordLink->ForwardLink) {\r
1250 ImageRecord = CR (\r
1251 ImageRecordLink,\r
1252 IMAGE_PROPERTIES_RECORD,\r
1253 Link,\r
1254 IMAGE_PROPERTIES_RECORD_SIGNATURE\r
1255 );\r
1256\r
1257 if ((ImageBase == ImageRecord->ImageBase) &&\r
1258 (ImageSize == ImageRecord->ImageSize)) {\r
1259 return ImageRecord;\r
1260 }\r
1261 }\r
1262\r
1263 return NULL;\r
1264}\r
1265\r
1266/**\r
1267 Remove Image record.\r
1268\r
1269 @param RuntimeImage Runtime image information\r
1270**/\r
1271VOID\r
1272RemoveImageRecord (\r
1273 IN EFI_RUNTIME_IMAGE_ENTRY *RuntimeImage\r
1274 )\r
1275{\r
1276 IMAGE_PROPERTIES_RECORD *ImageRecord;\r
1277 LIST_ENTRY *CodeSegmentListHead;\r
1278 IMAGE_PROPERTIES_RECORD_CODE_SECTION *ImageRecordCodeSection;\r
1279\r
1280 DEBUG ((EFI_D_VERBOSE, "RemoveImageRecord - 0x%x\n", RuntimeImage));\r
1281 DEBUG ((EFI_D_VERBOSE, "RemoveImageRecord - 0x%016lx - 0x%016lx\n", (EFI_PHYSICAL_ADDRESS)(UINTN)RuntimeImage->ImageBase, RuntimeImage->ImageSize));\r
1282\r
4a723ed2
JY
1283 if (mPropertiesTableEndOfDxe) {\r
1284 DEBUG ((DEBUG_INFO, "Do not remove runtime image record after EndOfDxe\n"));\r
1285 return ;\r
1286 }\r
1287\r
03d486b2
JY
1288 ImageRecord = FindImageRecord ((EFI_PHYSICAL_ADDRESS)(UINTN)RuntimeImage->ImageBase, RuntimeImage->ImageSize);\r
1289 if (ImageRecord == NULL) {\r
1290 DEBUG ((EFI_D_ERROR, "!!!!!!!! ImageRecord not found !!!!!!!!\n"));\r
1291 return ;\r
1292 }\r
1293\r
1294 CodeSegmentListHead = &ImageRecord->CodeSegmentList;\r
1295 while (!IsListEmpty (CodeSegmentListHead)) {\r
1296 ImageRecordCodeSection = CR (\r
1297 CodeSegmentListHead->ForwardLink,\r
1298 IMAGE_PROPERTIES_RECORD_CODE_SECTION,\r
1299 Link,\r
1300 IMAGE_PROPERTIES_RECORD_CODE_SECTION_SIGNATURE\r
1301 );\r
1302 RemoveEntryList (&ImageRecordCodeSection->Link);\r
1303 FreePool (ImageRecordCodeSection);\r
1304 }\r
1305\r
1306 RemoveEntryList (&ImageRecord->Link);\r
1307 FreePool (ImageRecord);\r
1308 mImagePropertiesPrivateData.ImageRecordCount--;\r
1309}\r
1310\r
1311\r
1312/**\r
1313 Install PropertiesTable.\r
1314\r
1315 @param[in] Event The Event this notify function registered to.\r
1316 @param[in] Context Pointer to the context data registered to the Event.\r
1317**/\r
1318VOID\r
1319EFIAPI\r
1320InstallPropertiesTable (\r
1321 EFI_EVENT Event,\r
1322 VOID *Context\r
1323 )\r
1324{\r
4a723ed2 1325 mPropertiesTableEndOfDxe = TRUE;\r
bc31c0c6 1326 if (PcdGetBool (PcdPropertiesTableEnable)) {\r
03d486b2
JY
1327 EFI_STATUS Status;\r
1328\r
1329 Status = gBS->InstallConfigurationTable (&gEfiPropertiesTableGuid, &mPropertiesTable);\r
1330 ASSERT_EFI_ERROR (Status);\r
1331\r
1332 DEBUG ((EFI_D_INFO, "MemoryProtectionAttribute - 0x%016lx\n", mPropertiesTable.MemoryProtectionAttribute));\r
1333 if ((mPropertiesTable.MemoryProtectionAttribute & EFI_PROPERTIES_RUNTIME_MEMORY_PROTECTION_NON_EXECUTABLE_PE_DATA) == 0) {\r
2495c9bf 1334 DEBUG ((EFI_D_ERROR, "MemoryProtectionAttribute NON_EXECUTABLE_PE_DATA is not set, "));\r
d4731a98 1335 DEBUG ((EFI_D_ERROR, "because Runtime Driver Section Alignment is not %dK.\n", RUNTIME_PAGE_ALLOCATION_GRANULARITY >> 10));\r
03d486b2
JY
1336 return ;\r
1337 }\r
1338\r
0ab90add 1339 gBS->GetMemoryMap = CoreGetMemoryMapWithSeparatedImageSection;\r
03d486b2
JY
1340 gBS->Hdr.CRC32 = 0;\r
1341 gBS->CalculateCrc32 ((UINT8 *)gBS, gBS->Hdr.HeaderSize, &gBS->Hdr.CRC32);\r
1342\r
1343 DEBUG ((EFI_D_VERBOSE, "Total Image Count - 0x%x\n", mImagePropertiesPrivateData.ImageRecordCount));\r
1344 DEBUG ((EFI_D_VERBOSE, "Dump ImageRecord:\n"));\r
1345 DumpImageRecord ();\r
82f0f411
JY
1346\r
1347 mPropertiesTableEnable = TRUE;\r
03d486b2
JY
1348 }\r
1349}\r
1350\r
1351/**\r
1352 Initialize PropertiesTable support.\r
1353**/\r
1354VOID\r
1355EFIAPI\r
1356CoreInitializePropertiesTable (\r
1357 VOID\r
1358 )\r
1359{\r
1360 EFI_STATUS Status;\r
1361 EFI_EVENT EndOfDxeEvent;\r
1362\r
1363 Status = gBS->CreateEventEx (\r
1364 EVT_NOTIFY_SIGNAL,\r
1365 TPL_NOTIFY,\r
1366 InstallPropertiesTable,\r
1367 NULL,\r
1368 &gEfiEndOfDxeEventGroupGuid,\r
1369 &EndOfDxeEvent\r
1370 );\r
1371 ASSERT_EFI_ERROR (Status);\r
1372 return ;\r
1373}\r