]> git.proxmox.com Git - mirror_edk2.git/blame - IntelFspWrapperPkg/FspNotifyDxe/LoadBelow4G.c
IntelFspWrapperPkg: Replace BSD License with BSD+Patent License
[mirror_edk2.git] / IntelFspWrapperPkg / FspNotifyDxe / LoadBelow4G.c
CommitLineData
00899456
JY
1/** @file\r
2\r
bcee1b9f 3Copyright (c) 2015 - 2017, Intel Corporation. All rights reserved.<BR>\r
00899456 4\r
19486360 5SPDX-License-Identifier: BSD-2-Clause-Patent\r
00899456
JY
6\r
7**/\r
8\r
9#include <Uefi.h>\r
10#include <Library/BaseLib.h>\r
11#include <Library/UefiDriverEntryPoint.h>\r
12#include <Library/BaseMemoryLib.h>\r
13#include <Library/DebugLib.h>\r
14#include <Library/PeCoffLib.h>\r
15#include <Library/UefiBootServicesTableLib.h>\r
16#include <Library/DxeServicesLib.h>\r
17#include <Library/CacheMaintenanceLib.h>\r
18#include <Library/UefiLib.h>\r
19\r
20/**\r
21 Relocate this image under 4G memory.\r
22\r
23 @param ImageHandle Handle of driver image.\r
24 @param SystemTable Pointer to system table.\r
25\r
26 @retval EFI_SUCCESS Image successfully relocated.\r
27 @retval EFI_ABORTED Failed to relocate image.\r
28\r
29**/\r
30EFI_STATUS\r
31RelocateImageUnder4GIfNeeded (\r
32 IN EFI_HANDLE ImageHandle,\r
33 IN EFI_SYSTEM_TABLE *SystemTable\r
34 )\r
35{\r
36 EFI_STATUS Status;\r
37 UINT8 *Buffer;\r
38 UINTN BufferSize;\r
39 EFI_HANDLE NewImageHandle;\r
40 UINTN Pages;\r
41 EFI_PHYSICAL_ADDRESS FfsBuffer;\r
42 PE_COFF_LOADER_IMAGE_CONTEXT ImageContext;\r
43 VOID *Interface;\r
44\r
45 //\r
46 // If it is already <4G, no need do relocate\r
47 //\r
48 if ((UINTN)RelocateImageUnder4GIfNeeded < 0xFFFFFFFF) {\r
49 return EFI_SUCCESS;\r
50 }\r
51\r
52 //\r
53 // If locate gEfiCallerIdGuid success, it means 2nd entry.\r
54 //\r
55 Status = gBS->LocateProtocol (&gEfiCallerIdGuid, NULL, &Interface);\r
56 if (!EFI_ERROR (Status)) {\r
57 DEBUG ((EFI_D_INFO, "FspNotifyDxe - 2nd entry\n"));\r
58 return EFI_SUCCESS;\r
59 }\r
60\r
61 DEBUG ((EFI_D_INFO, "FspNotifyDxe - 1st entry\n"));\r
62\r
63 //\r
64 // Here we install a dummy handle\r
65 //\r
66 NewImageHandle = NULL;\r
67 Status = gBS->InstallProtocolInterface (\r
68 &NewImageHandle,\r
69 &gEfiCallerIdGuid,\r
70 EFI_NATIVE_INTERFACE,\r
71 NULL\r
72 );\r
73 ASSERT_EFI_ERROR (Status);\r
74\r
75 //\r
76 // Reload image itself to <4G mem\r
77 //\r
78 Status = GetSectionFromAnyFv (\r
79 &gEfiCallerIdGuid,\r
80 EFI_SECTION_PE32,\r
81 0,\r
82 (VOID **) &Buffer,\r
83 &BufferSize\r
84 );\r
85 ASSERT_EFI_ERROR (Status);\r
86 ImageContext.Handle = Buffer;\r
87 ImageContext.ImageRead = PeCoffLoaderImageReadFromMemory;\r
88 //\r
89 // Get information about the image being loaded\r
90 //\r
91 Status = PeCoffLoaderGetImageInfo (&ImageContext);\r
92 ASSERT_EFI_ERROR (Status);\r
93 if (ImageContext.SectionAlignment > EFI_PAGE_SIZE) {\r
94 Pages = EFI_SIZE_TO_PAGES ((UINTN) (ImageContext.ImageSize + ImageContext.SectionAlignment));\r
95 } else {\r
96 Pages = EFI_SIZE_TO_PAGES ((UINTN) ImageContext.ImageSize);\r
97 }\r
98 FfsBuffer = 0xFFFFFFFF;\r
99 Status = gBS->AllocatePages (\r
100 AllocateMaxAddress,\r
101 EfiBootServicesCode,\r
102 Pages,\r
103 &FfsBuffer\r
104 );\r
105 ASSERT_EFI_ERROR (Status);\r
106 ImageContext.ImageAddress = (PHYSICAL_ADDRESS)(UINTN)FfsBuffer;\r
107 //\r
b9ca25cb 108 // Align buffer on section boundary\r
00899456
JY
109 //\r
110 ImageContext.ImageAddress += ImageContext.SectionAlignment - 1;\r
bcee1b9f 111 ImageContext.ImageAddress &= ~((EFI_PHYSICAL_ADDRESS)ImageContext.SectionAlignment - 1);\r
00899456
JY
112 //\r
113 // Load the image to our new buffer\r
114 //\r
115 Status = PeCoffLoaderLoadImage (&ImageContext);\r
116 ASSERT_EFI_ERROR (Status);\r
117\r
118 //\r
119 // Relocate the image in our new buffer\r
120 //\r
121 Status = PeCoffLoaderRelocateImage (&ImageContext);\r
122 ASSERT_EFI_ERROR (Status);\r
123\r
124 //\r
125 // Free the buffer allocated by ReadSection since the image has been relocated in the new buffer\r
126 //\r
127 gBS->FreePool (Buffer);\r
128\r
129 //\r
130 // Flush the instruction cache so the image data is written before we execute it\r
131 //\r
132 InvalidateInstructionCacheRange ((VOID *)(UINTN)ImageContext.ImageAddress, (UINTN)ImageContext.ImageSize);\r
133\r
134 DEBUG ((EFI_D_INFO, "Loading driver at 0x%08x EntryPoint=0x%08x\n", (UINTN)ImageContext.ImageAddress, (UINTN)ImageContext.EntryPoint));\r
135 Status = ((EFI_IMAGE_ENTRY_POINT)(UINTN)(ImageContext.EntryPoint)) (NewImageHandle, gST);\r
136 if (EFI_ERROR (Status)) {\r
137 DEBUG ((EFI_D_ERROR, "Error: Image at 0x%08x start failed: %r\n", ImageContext.ImageAddress, Status));\r
138 gBS->FreePages (FfsBuffer, Pages);\r
139 }\r
140\r
141 //\r
142 // return error to unload >4G copy, if we already relocate itself to <4G.\r
143 //\r
144 return EFI_ALREADY_STARTED;\r
145}\r