]> git.proxmox.com Git - mirror_edk2.git/blame - MdeModulePkg/Core/Pei/FwVol/FwVol.c
Remove the tab characters.
[mirror_edk2.git] / MdeModulePkg / Core / Pei / FwVol / FwVol.c
CommitLineData
615c6dd0 1/** @file\r
b1f6a7c6 2 Pei Core Firmware File System service routines.\r
3 \r
3b428ade 4Copyright (c) 2006 - 2009, Intel Corporation \r
192f6d4c 5All rights reserved. This program and the accompanying materials \r
6are licensed and made available under the terms and conditions of the BSD License \r
7which accompanies this distribution. The full text of the license may be found at \r
8http://opensource.org/licenses/bsd-license.php \r
9 \r
10THE PROGRAM IS DISTRIBUTED UNDER THE BSD LICENSE ON AN "AS IS" BASIS, \r
11WITHOUT WARRANTIES OR REPRESENTATIONS OF ANY KIND, EITHER EXPRESS OR IMPLIED. \r
12\r
615c6dd0 13**/\r
192f6d4c 14\r
3b428ade 15#include "FwVol.h"\r
192f6d4c 16\r
fe1e36e5 17EFI_PEI_NOTIFY_DESCRIPTOR mNotifyOnFvInfoList = {\r
b0d803fe 18 (EFI_PEI_PPI_DESCRIPTOR_NOTIFY_CALLBACK | EFI_PEI_PPI_DESCRIPTOR_TERMINATE_LIST),\r
19 &gEfiPeiFirmwareVolumeInfoPpiGuid,\r
20 FirmwareVolmeInfoPpiNotifyCallback \r
21};\r
22\r
3b428ade 23EFI_PEI_FIRMWARE_VOLUME_PPI mPeiFfs2FvPpi = {\r
24 PeiFfs2FvPpiProcessVolume,\r
25 PeiFfs2FvPpiFindFileByType,\r
26 PeiFfs2FvPpiFindFileByName,\r
27 PeiFfs2FvPpiGetFileInfo,\r
28 PeiFfs2FvPpiGetVolumeInfo,\r
29 PeiFfs2FvPpiFindSectionByType\r
30};\r
31 \r
32EFI_PEI_PPI_DESCRIPTOR mPeiFfs2FvPpiList = {\r
33 (EFI_PEI_PPI_DESCRIPTOR_PPI | EFI_PEI_PPI_DESCRIPTOR_TERMINATE_LIST),\r
34 &gEfiFirmwareFileSystem2Guid,\r
35 &mPeiFfs2FvPpi\r
36};\r
37 \r
b1f6a7c6 38/**\r
ed299e3c 39 Returns the file state set by the highest zero bit in the State field\r
192f6d4c 40\r
b1f6a7c6 41 @param ErasePolarity Erase Polarity as defined by EFI_FVB2_ERASE_POLARITY\r
42 in the Attributes field.\r
43 @param FfsHeader Pointer to FFS File Header.\r
192f6d4c 44\r
ed299e3c
LG
45 @retval EFI_FFS_FILE_STATE File state is set by the highest none zero bit \r
46 in the header State field.\r
b1f6a7c6 47**/\r
48EFI_FFS_FILE_STATE\r
49GetFileState(\r
50 IN UINT8 ErasePolarity,\r
51 IN EFI_FFS_FILE_HEADER *FfsHeader\r
52 )\r
192f6d4c 53{\r
54 EFI_FFS_FILE_STATE FileState;\r
55 EFI_FFS_FILE_STATE HighestBit;\r
56\r
57 FileState = FfsHeader->State;\r
58\r
59 if (ErasePolarity != 0) {\r
60 FileState = (EFI_FFS_FILE_STATE)~FileState;\r
61 }\r
ed299e3c
LG
62 \r
63 //\r
64 // Get file state set by its highest none zero bit.\r
65 //\r
192f6d4c 66 HighestBit = 0x80;\r
67 while (HighestBit != 0 && (HighestBit & FileState) == 0) {\r
68 HighestBit >>= 1;\r
69 }\r
70\r
71 return HighestBit;\r
72} \r
73\r
b1f6a7c6 74/**\r
75 Calculates the checksum of the header of a file.\r
76\r
77 @param FileHeader Pointer to FFS File Header.\r
78\r
79 @return Checksum of the header.\r
b1f6a7c6 80 Zero means the header is good.\r
81 Non-zero means the header is bad.\r
82**/\r
192f6d4c 83UINT8\r
84CalculateHeaderChecksum (\r
85 IN EFI_FFS_FILE_HEADER *FileHeader\r
86 )\r
192f6d4c 87{\r
ed299e3c 88 EFI_FFS_FILE_HEADER TestFileHeader;\r
192f6d4c 89 \r
ed299e3c 90 CopyMem (&TestFileHeader, FileHeader, sizeof (EFI_FFS_FILE_HEADER));\r
192f6d4c 91 //\r
ed299e3c 92 // Ingore State and File field in FFS header.\r
192f6d4c 93 //\r
ed299e3c
LG
94 TestFileHeader.State = 0;\r
95 TestFileHeader.IntegrityCheck.Checksum.File = 0;\r
192f6d4c 96\r
ed299e3c 97 return CalculateSum8 ((CONST UINT8 *) &TestFileHeader, sizeof (EFI_FFS_FILE_HEADER));\r
192f6d4c 98}\r
99\r
b1f6a7c6 100/**\r
3b428ade 101 Find FV handler according to FileHandle in that FV.\r
b1f6a7c6 102\r
103 @param FileHandle Handle of file image\r
3b428ade 104 \r
105 @return Pointer to instance of PEI_CORE_FV_HANDLE.\r
b1f6a7c6 106**/\r
3b428ade 107PEI_CORE_FV_HANDLE*\r
108FileHandleToVolume (\r
109 IN EFI_PEI_FILE_HANDLE FileHandle\r
b0d803fe 110 )\r
111{\r
112 UINTN Index;\r
113 PEI_CORE_INSTANCE *PrivateData;\r
114 EFI_FIRMWARE_VOLUME_HEADER *FwVolHeader;\r
115\r
116 PrivateData = PEI_CORE_INSTANCE_FROM_PS_THIS (GetPeiServicesTablePointer ());\r
3b428ade 117 \r
b0d803fe 118 for (Index = 0; Index < PrivateData->FvCount; Index++) {\r
119 FwVolHeader = PrivateData->Fv[Index].FvHeader;\r
67a58d0f 120 if (((UINT64) (UINTN) FileHandle > (UINT64) (UINTN) FwVolHeader ) && \\r
121 ((UINT64) (UINTN) FileHandle <= ((UINT64) (UINTN) FwVolHeader + FwVolHeader->FvLength - 1))) {\r
3b428ade 122 return &PrivateData->Fv[Index];\r
b0d803fe 123 }\r
124 }\r
3b428ade 125 return NULL;\r
b0d803fe 126}\r
127\r
b1f6a7c6 128/**\r
ed299e3c 129 Given the input file pointer, search for the first matching file in the\r
b1f6a7c6 130 FFS volume as defined by SearchType. The search starts from FileHeader inside\r
131 the Firmware Volume defined by FwVolHeader.\r
ed299e3c
LG
132 If SearchType is EFI_FV_FILETYPE_ALL, the first FFS file will return without check its file type.\r
133 If SearchType is PEI_CORE_INTERNAL_FFS_FILE_DISPATCH_TYPE, \r
134 the first PEIM, or COMBINED PEIM or FV file type FFS file will return. \r
b1f6a7c6 135\r
136 @param FvHandle Pointer to the FV header of the volume to search\r
137 @param FileName File name\r
138 @param SearchType Filter to find only files of this type.\r
139 Type EFI_FV_FILETYPE_ALL causes no filtering to be done.\r
140 @param FileHandle This parameter must point to a valid FFS volume.\r
141 @param AprioriFile Pointer to AprioriFile image in this FV if has\r
b0d803fe 142\r
b1f6a7c6 143 @return EFI_NOT_FOUND No files matching the search criteria were found\r
144 @retval EFI_SUCCESS Success to search given file\r
145\r
146**/\r
192f6d4c 147EFI_STATUS\r
3b428ade 148FindFileEx (\r
b0d803fe 149 IN CONST EFI_PEI_FV_HANDLE FvHandle,\r
150 IN CONST EFI_GUID *FileName, OPTIONAL\r
151 IN EFI_FV_FILETYPE SearchType,\r
152 IN OUT EFI_PEI_FILE_HANDLE *FileHandle,\r
153 IN OUT EFI_PEI_FV_HANDLE *AprioriFile OPTIONAL\r
192f6d4c 154 )\r
192f6d4c 155{\r
b0d803fe 156 EFI_FIRMWARE_VOLUME_HEADER *FwVolHeader;\r
157 EFI_FFS_FILE_HEADER **FileHeader;\r
158 EFI_FFS_FILE_HEADER *FfsFileHeader;\r
b0d803fe 159 UINT32 FileLength;\r
160 UINT32 FileOccupiedSize;\r
161 UINT32 FileOffset;\r
162 UINT64 FvLength;\r
163 UINT8 ErasePolarity;\r
164 UINT8 FileState;\r
3b428ade 165 \r
166 //\r
167 // Convert the handle of FV to FV header for memory-mapped firmware volume\r
168 //\r
169 FwVolHeader = (EFI_FIRMWARE_VOLUME_HEADER *) FvHandle;\r
b0d803fe 170 FileHeader = (EFI_FFS_FILE_HEADER **)FileHandle;\r
192f6d4c 171\r
172 FvLength = FwVolHeader->FvLength;\r
d976bf31 173 if ((FwVolHeader->Attributes & EFI_FVB2_ERASE_POLARITY) != 0) {\r
192f6d4c 174 ErasePolarity = 1;\r
175 } else {\r
176 ErasePolarity = 0;\r
177 }\r
178 \r
179 //\r
b0d803fe 180 // If FileHeader is not specified (NULL) or FileName is not NULL,\r
181 // start with the first file in the firmware volume. Otherwise,\r
182 // start from the FileHeader.\r
192f6d4c 183 //\r
b0d803fe 184 if ((*FileHeader == NULL) || (FileName != NULL)) {\r
192f6d4c 185 FfsFileHeader = (EFI_FFS_FILE_HEADER *)((UINT8 *)FwVolHeader + FwVolHeader->HeaderLength);\r
186 } else {\r
187 //\r
188 // Length is 24 bits wide so mask upper 8 bits\r
189 // FileLength is adjusted to FileOccupiedSize as it is 8 byte aligned.\r
190 //\r
191 FileLength = *(UINT32 *)(*FileHeader)->Size & 0x00FFFFFF;\r
b0d803fe 192 FileOccupiedSize = GET_OCCUPIED_SIZE (FileLength, 8);\r
192f6d4c 193 FfsFileHeader = (EFI_FFS_FILE_HEADER *)((UINT8 *)*FileHeader + FileOccupiedSize);\r
194 }\r
e98cd821 195 \r
192f6d4c 196 FileOffset = (UINT32) ((UINT8 *)FfsFileHeader - (UINT8 *)FwVolHeader);\r
197 ASSERT (FileOffset <= 0xFFFFFFFF);\r
e98cd821 198\r
b0d803fe 199 while (FileOffset < (FvLength - sizeof (EFI_FFS_FILE_HEADER))) {\r
192f6d4c 200 //\r
201 // Get FileState which is the highest bit of the State \r
202 //\r
203 FileState = GetFileState (ErasePolarity, FfsFileHeader);\r
192f6d4c 204 switch (FileState) {\r
205\r
206 case EFI_FILE_HEADER_INVALID:\r
40f26b8f 207 FileOffset += sizeof(EFI_FFS_FILE_HEADER);\r
208 FfsFileHeader = (EFI_FFS_FILE_HEADER *)((UINT8 *)FfsFileHeader + sizeof(EFI_FFS_FILE_HEADER));\r
192f6d4c 209 break;\r
210 \r
211 case EFI_FILE_DATA_VALID:\r
212 case EFI_FILE_MARKED_FOR_UPDATE:\r
b0d803fe 213 if (CalculateHeaderChecksum (FfsFileHeader) != 0) {\r
192f6d4c 214 ASSERT (FALSE);\r
e98cd821 215 *FileHeader = NULL;\r
192f6d4c 216 return EFI_NOT_FOUND;\r
217 }\r
b0d803fe 218\r
40f26b8f 219 FileLength = *(UINT32 *)(FfsFileHeader->Size) & 0x00FFFFFF;\r
b0d803fe 220 FileOccupiedSize = GET_OCCUPIED_SIZE(FileLength, 8);\r
221\r
222 if (FileName != NULL) {\r
223 if (CompareGuid (&FfsFileHeader->Name, (EFI_GUID*)FileName)) {\r
224 *FileHeader = FfsFileHeader;\r
225 return EFI_SUCCESS;\r
226 }\r
227 } else if (SearchType == PEI_CORE_INTERNAL_FFS_FILE_DISPATCH_TYPE) {\r
228 if ((FfsFileHeader->Type == EFI_FV_FILETYPE_PEIM) || \r
288f9b38
LG
229 (FfsFileHeader->Type == EFI_FV_FILETYPE_COMBINED_PEIM_DRIVER) ||\r
230 (FfsFileHeader->Type == EFI_FV_FILETYPE_FIRMWARE_VOLUME_IMAGE)) { \r
b0d803fe 231 \r
232 *FileHeader = FfsFileHeader;\r
233 return EFI_SUCCESS;\r
234 } else if (AprioriFile != NULL) {\r
235 if (FfsFileHeader->Type == EFI_FV_FILETYPE_FREEFORM) {\r
236 if (CompareGuid (&FfsFileHeader->Name, &gPeiAprioriFileNameGuid)) {\r
237 *AprioriFile = FfsFileHeader;\r
238 } \r
239 } \r
240 }\r
7e181f2f 241 } else if (((SearchType == FfsFileHeader->Type) || (SearchType == EFI_FV_FILETYPE_ALL)) && \r
242 (FfsFileHeader->Type != EFI_FV_FILETYPE_FFS_PAD)) { \r
b0d803fe 243 *FileHeader = FfsFileHeader;\r
244 return EFI_SUCCESS;\r
245 }\r
246\r
40f26b8f 247 FileOffset += FileOccupiedSize; \r
248 FfsFileHeader = (EFI_FFS_FILE_HEADER *)((UINT8 *)FfsFileHeader + FileOccupiedSize);\r
192f6d4c 249 break;\r
250 \r
251 case EFI_FILE_DELETED:\r
40f26b8f 252 FileLength = *(UINT32 *)(FfsFileHeader->Size) & 0x00FFFFFF;\r
253 FileOccupiedSize = GET_OCCUPIED_SIZE(FileLength, 8);\r
254 FileOffset += FileOccupiedSize;\r
255 FfsFileHeader = (EFI_FFS_FILE_HEADER *)((UINT8 *)FfsFileHeader + FileOccupiedSize);\r
192f6d4c 256 break;\r
257\r
258 default:\r
e98cd821 259 *FileHeader = NULL;\r
192f6d4c 260 return EFI_NOT_FOUND;\r
192f6d4c 261 } \r
262 }\r
e98cd821
LG
263 \r
264 *FileHeader = NULL;\r
192f6d4c 265 return EFI_NOT_FOUND; \r
266}\r
267\r
b1f6a7c6 268/**\r
b1f6a7c6 269 Initialize PeiCore Fv List.\r
270\r
b1f6a7c6 271 @param PrivateData - Pointer to PEI_CORE_INSTANCE.\r
272 @param SecCoreData - Pointer to EFI_SEC_PEI_HAND_OFF.\r
b1f6a7c6 273**/\r
b0d803fe 274VOID \r
275PeiInitializeFv (\r
276 IN PEI_CORE_INSTANCE *PrivateData,\r
277 IN CONST EFI_SEC_PEI_HAND_OFF *SecCoreData\r
278 )\r
b0d803fe 279{\r
3b428ade 280 EFI_STATUS Status;\r
281 EFI_PEI_FIRMWARE_VOLUME_PPI *FvPpi;\r
282 EFI_PEI_FV_HANDLE FvHandle;\r
283 EFI_FIRMWARE_VOLUME_HEADER *BfvHeader;\r
284 \r
b0d803fe 285 //\r
3b428ade 286 // Install FV_PPI for FFS2 file system.\r
b0d803fe 287 //\r
3b428ade 288 PeiServicesInstallPpi (&mPeiFfs2FvPpiList);\r
289 \r
290 BfvHeader = (EFI_FIRMWARE_VOLUME_HEADER *)SecCoreData->BootFirmwareVolumeBase;\r
291 \r
292 //\r
293 // The FV_PPI in BFV's format should be installed.\r
294 //\r
295 Status = PeiServicesLocatePpi (\r
296 &BfvHeader->FileSystemGuid,\r
297 0,\r
298 NULL,\r
299 (VOID**)&FvPpi\r
300 );\r
301 ASSERT_EFI_ERROR (Status);\r
302 \r
303 //\r
304 // Get handle of BFV\r
305 //\r
306 FvPpi->ProcessVolume (\r
307 FvPpi, \r
308 SecCoreData->BootFirmwareVolumeBase,\r
309 (UINTN)BfvHeader->FvLength,\r
310 &FvHandle\r
311 );\r
312 \r
b0d803fe 313 //\r
314 // Post a call-back for the FvInfoPPI services to expose\r
315 // additional Fvs to PeiCore.\r
316 //\r
317 Status = PeiServicesNotifyPpi (&mNotifyOnFvInfoList);\r
318 ASSERT_EFI_ERROR (Status);\r
319\r
320}\r
3b428ade 321 \r
b1f6a7c6 322/**\r
323 Process Firmware Volum Information once FvInfoPPI install.\r
ed299e3c
LG
324 The FV Info will be registered into PeiCore private data structure.\r
325 And search the inside FV image, if found, the new FV INFO PPI will be installed.\r
b1f6a7c6 326\r
ed299e3c 327 @param PeiServices An indirect pointer to the EFI_PEI_SERVICES table published by the PEI Foundation\r
731bd38e 328 @param NotifyDescriptor Address of the notification descriptor data structure.\r
329 @param Ppi Address of the PPI that was installed.\r
b1f6a7c6 330\r
ed299e3c 331 @retval EFI_SUCCESS The FV Info is registered into PeiCore private data structure.\r
6a1ae84a 332 @return if not EFI_SUCESS, fail to verify FV.\r
b1f6a7c6 333\r
334**/\r
b0d803fe 335EFI_STATUS\r
336EFIAPI\r
337FirmwareVolmeInfoPpiNotifyCallback (\r
338 IN EFI_PEI_SERVICES **PeiServices,\r
339 IN EFI_PEI_NOTIFY_DESCRIPTOR *NotifyDescriptor,\r
340 IN VOID *Ppi\r
341 )\r
b0d803fe 342{\r
3b428ade 343 EFI_PEI_FIRMWARE_VOLUME_INFO_PPI *FvInfoPpi;\r
344 EFI_PEI_FIRMWARE_VOLUME_PPI *FvPpi;\r
b0d803fe 345 PEI_CORE_INSTANCE *PrivateData;\r
288f9b38 346 EFI_STATUS Status;\r
3b428ade 347 EFI_PEI_FV_HANDLE FvHandle;\r
348 UINTN FvIndex;\r
b0d803fe 349 \r
288f9b38
LG
350 Status = EFI_SUCCESS;\r
351 PrivateData = PEI_CORE_INSTANCE_FROM_PS_THIS (PeiServices);\r
b0d803fe 352\r
177aabe6 353 if (PrivateData->FvCount >= FixedPcdGet32 (PcdPeiCoreMaxFvSupported)) {\r
97b2c9b5
LG
354 DEBUG ((EFI_D_ERROR, "The number of Fv Images (%d) exceed the max supported FVs (%d) in Pei", PrivateData->FvCount + 1, FixedPcdGet32 (PcdPeiCoreMaxFvSupported)));\r
355 DEBUG ((EFI_D_ERROR, "PcdPeiCoreMaxFvSupported value need be reconfigurated in DSC"));\r
b0d803fe 356 ASSERT (FALSE);\r
357 }\r
358\r
3b428ade 359 FvInfoPpi = (EFI_PEI_FIRMWARE_VOLUME_INFO_PPI *)Ppi;\r
b0d803fe 360\r
97b2c9b5 361 //\r
3b428ade 362 // Locate the corresponding FV_PPI according to founded FV's format guid\r
97b2c9b5 363 //\r
3b428ade 364 Status = PeiServicesLocatePpi (\r
365 &FvInfoPpi->FvFormat, \r
366 0, \r
367 NULL,\r
368 (VOID**)&FvPpi\r
369 );\r
370 if (!EFI_ERROR (Status)) {\r
371 //\r
372 // Process new found FV and get FV handle.\r
373 //\r
374 Status = FvPpi->ProcessVolume (FvPpi, FvInfoPpi->FvInfo, FvInfoPpi->FvInfoSize, &FvHandle);\r
375 if (EFI_ERROR (Status)) {\r
376 DEBUG ((EFI_D_ERROR, "Fail to process new found FV, FV may be corrupted!"));\r
6a1ae84a 377 return Status;\r
378 }\r
3b428ade 379 DEBUG ((EFI_D_INFO, "Found and process new FV %p, all fv's count is %d\n", FvHandle, PrivateData->FvCount));\r
380 } else {\r
381 DEBUG ((EFI_D_ERROR, "Fail to process FV %p because no corresponding EFI_FIRMWARE_VOLUME_PPI is found!\n", FvInfoPpi->FvInfo));\r
6a1ae84a 382 \r
3b428ade 383 //\r
384 // If can not find EFI_FIRMWARE_VOLUME_PPI to process firmware to get FvHandle, \r
385 // use the address of FV buffer as its handle.\r
386 //\r
387 FvHandle = FvInfoPpi->FvInfo;\r
288f9b38 388 \r
288f9b38 389 //\r
3b428ade 390 // Check whether the FV has already been processed.\r
288f9b38 391 //\r
3b428ade 392 for (FvIndex = 0; FvIndex < PrivateData->FvCount; FvIndex ++) {\r
393 if (PrivateData->Fv[FvIndex].FvHandle == FvHandle) {\r
394 DEBUG ((EFI_D_INFO, "The Fv %p has already been processed!\n", FvHandle));\r
395 return EFI_SUCCESS;\r
288f9b38 396 }\r
3b428ade 397 } \r
398 \r
399 PrivateData->Fv[PrivateData->FvCount].FvHeader = (EFI_FIRMWARE_VOLUME_HEADER*) FvInfoPpi->FvInfo;\r
400 PrivateData->Fv[PrivateData->FvCount].FvPpi = NULL;\r
401 PrivateData->Fv[PrivateData->FvCount].FvHandle = FvHandle;\r
402 PrivateData->FvCount ++;\r
288f9b38 403 }\r
3b428ade 404 \r
b0d803fe 405 return EFI_SUCCESS;\r
406}\r
407\r
b1f6a7c6 408/**\r
ed299e3c
LG
409 Go through the file to search SectionType section. \r
410 Search within encapsulation sections (compression and GUIDed) recursively, \r
411 until the match section is found.\r
412 \r
3b428ade 413 @param PeiServices An indirect pointer to the EFI_PEI_SERVICES table published by the PEI Foundation.\r
414 @param SectionType Filter to find only section of this type.\r
415 @param Section From where to search.\r
416 @param SectionSize The file size to search.\r
417 @param OutputBuffer A pointer to the discovered section, if successful.\r
ed299e3c 418 NULL if section not found\r
b1f6a7c6 419\r
ed299e3c
LG
420 @return EFI_NOT_FOUND The match section is not found.\r
421 @return EFI_SUCCESS The match section is found.\r
b1f6a7c6 422\r
423**/\r
b0d803fe 424EFI_STATUS\r
3b428ade 425ProcessSection (\r
b0d803fe 426 IN CONST EFI_PEI_SERVICES **PeiServices,\r
427 IN EFI_SECTION_TYPE SectionType,\r
428 IN EFI_COMMON_SECTION_HEADER *Section,\r
429 IN UINTN SectionSize,\r
288f9b38 430 OUT VOID **OutputBuffer\r
b0d803fe 431 )\r
b0d803fe 432{\r
433 EFI_STATUS Status;\r
434 UINT32 SectionLength;\r
435 UINT32 ParsedLength;\r
b0d803fe 436 EFI_PEI_GUIDED_SECTION_EXTRACTION_PPI *GuidSectionPpi;\r
b0d803fe 437 EFI_PEI_DECOMPRESS_PPI *DecompressPpi;\r
438 VOID *PpiOutput;\r
439 UINTN PpiOutputSize;\r
288f9b38
LG
440 UINTN Index;\r
441 UINT32 Authentication;\r
442 PEI_CORE_INSTANCE *PrivateData;\r
b0d803fe 443\r
288f9b38 444 PrivateData = PEI_CORE_INSTANCE_FROM_PS_THIS (PeiServices);\r
b0d803fe 445 *OutputBuffer = NULL;\r
288f9b38
LG
446 ParsedLength = 0;\r
447 Index = 0;\r
448 Status = EFI_NOT_FOUND;\r
449 PpiOutput = NULL;\r
450 PpiOutputSize = 0;\r
b0d803fe 451 while (ParsedLength < SectionSize) {\r
452 if (Section->Type == SectionType) {\r
453 *OutputBuffer = (VOID *)(Section + 1);\r
454 return EFI_SUCCESS;\r
288f9b38
LG
455 } else if ((Section->Type == EFI_SECTION_GUID_DEFINED) || (Section->Type == EFI_SECTION_COMPRESSION)) {\r
456 //\r
457 // Check the encapsulated section is extracted into the cache data.\r
458 //\r
459 for (Index = 0; Index < PrivateData->CacheSection.AllSectionCount; Index ++) {\r
460 if (Section == PrivateData->CacheSection.Section[Index]) {\r
461 PpiOutput = PrivateData->CacheSection.SectionData[Index];\r
462 PpiOutputSize = PrivateData->CacheSection.SectionSize[Index];\r
463 //\r
464 // Search section directly from the cache data.\r
465 //\r
3b428ade 466 return ProcessSection (\r
467 PeiServices,\r
468 SectionType, \r
469 PpiOutput, \r
470 PpiOutputSize, \r
471 OutputBuffer \r
472 );\r
b0d803fe 473 }\r
474 }\r
288f9b38
LG
475 \r
476 Status = EFI_NOT_FOUND;\r
477 if (Section->Type == EFI_SECTION_GUID_DEFINED) {\r
478 Status = PeiServicesLocatePpi (\r
479 &((EFI_GUID_DEFINED_SECTION *)Section)->SectionDefinitionGuid, \r
480 0, \r
481 NULL, \r
482 (VOID **) &GuidSectionPpi\r
483 );\r
b0d803fe 484 if (!EFI_ERROR (Status)) {\r
288f9b38 485 Status = GuidSectionPpi->ExtractSection (\r
3b428ade 486 GuidSectionPpi,\r
487 Section,\r
488 &PpiOutput,\r
489 &PpiOutputSize,\r
490 &Authentication\r
491 );\r
288f9b38
LG
492 }\r
493 } else if (Section->Type == EFI_SECTION_COMPRESSION) {\r
494 Status = PeiServicesLocatePpi (&gEfiPeiDecompressPpiGuid, 0, NULL, (VOID **) &DecompressPpi);\r
495 if (!EFI_ERROR (Status)) {\r
496 Status = DecompressPpi->Decompress (\r
497 DecompressPpi,\r
498 (CONST EFI_COMPRESSION_SECTION*) Section,\r
499 &PpiOutput,\r
500 &PpiOutputSize\r
501 );\r
b0d803fe 502 }\r
503 }\r
288f9b38
LG
504 \r
505 if (!EFI_ERROR (Status)) {\r
506 //\r
507 // Update cache section data.\r
508 //\r
509 if (PrivateData->CacheSection.AllSectionCount < CACHE_SETION_MAX_NUMBER) {\r
510 PrivateData->CacheSection.AllSectionCount ++;\r
511 }\r
512 PrivateData->CacheSection.Section [PrivateData->CacheSection.SectionIndex] = Section;\r
513 PrivateData->CacheSection.SectionData [PrivateData->CacheSection.SectionIndex] = PpiOutput;\r
514 PrivateData->CacheSection.SectionSize [PrivateData->CacheSection.SectionIndex] = PpiOutputSize;\r
515 PrivateData->CacheSection.SectionIndex = (PrivateData->CacheSection.SectionIndex + 1)%CACHE_SETION_MAX_NUMBER;\r
516 \r
3b428ade 517 return ProcessSection (\r
518 PeiServices,\r
519 SectionType, \r
520 PpiOutput, \r
521 PpiOutputSize, \r
522 OutputBuffer \r
523 );\r
288f9b38 524 }\r
b0d803fe 525 }\r
526\r
527 //\r
528 // Size is 24 bits wide so mask upper 8 bits. \r
529 // SectionLength is adjusted it is 4 byte aligned.\r
530 // Go to the next section\r
531 //\r
532 SectionLength = *(UINT32 *)Section->Size & 0x00FFFFFF;\r
533 SectionLength = GET_OCCUPIED_SIZE (SectionLength, 4);\r
534 ASSERT (SectionLength != 0);\r
535 ParsedLength += SectionLength;\r
536 Section = (EFI_COMMON_SECTION_HEADER *)((UINT8 *)Section + SectionLength);\r
537 }\r
538 \r
539 return EFI_NOT_FOUND;\r
540}\r
541\r
192f6d4c 542\r
b1f6a7c6 543/**\r
3b428ade 544 Searches for the next matching section within the specified file.\r
b1f6a7c6 545\r
ed299e3c 546 @param PeiServices An indirect pointer to the EFI_PEI_SERVICES table published by the PEI Foundation\r
b1f6a7c6 547 @param SectionType Filter to find only sections of this type.\r
548 @param FileHandle Pointer to the current file to search.\r
ed299e3c 549 @param SectionData A pointer to the discovered section, if successful.\r
b1f6a7c6 550 NULL if section not found\r
551\r
3b428ade 552 @retval EFI_NOT_FOUND The section was not found.\r
553 @retval EFI_SUCCESS The section was found.\r
b1f6a7c6 554\r
555**/\r
192f6d4c 556EFI_STATUS\r
557EFIAPI\r
558PeiFfsFindSectionData (\r
3b428ade 559 IN CONST EFI_PEI_SERVICES **PeiServices,\r
192f6d4c 560 IN EFI_SECTION_TYPE SectionType,\r
0c2b5da8 561 IN EFI_PEI_FILE_HANDLE FileHandle,\r
3b428ade 562 OUT VOID **SectionData\r
192f6d4c 563 )\r
192f6d4c 564{\r
3b428ade 565 PEI_CORE_FV_HANDLE *CoreFvHandle;\r
566 \r
567 CoreFvHandle = FileHandleToVolume (FileHandle);\r
568 if ((CoreFvHandle == NULL) || (CoreFvHandle->FvPpi == NULL)) {\r
569 return EFI_NOT_FOUND;\r
570 }\r
571 \r
572 return CoreFvHandle->FvPpi->FindSectionByType (CoreFvHandle->FvPpi, SectionType, FileHandle, SectionData);\r
192f6d4c 573}\r
574\r
b1f6a7c6 575/**\r
3b428ade 576 Searches for the next matching file in the firmware volume.\r
b1f6a7c6 577\r
d73d93c3 578 @param PeiServices An indirect pointer to the EFI_PEI_SERVICES table published by the PEI Foundation.\r
b1f6a7c6 579 @param SearchType Filter to find only files of this type.\r
580 Type EFI_FV_FILETYPE_ALL causes no filtering to be done.\r
3b428ade 581 @param VolumeHandle Handle of firmware volume in which to search.\r
582 @param FileHandle On entry, points to the current handle from which to begin searching or NULL to start\r
583 at the beginning of the firmware volume. On exit, points the file handle of the next file\r
584 in the volume or NULL if there are no more files.\r
585\r
586 @retval EFI_NOT_FOUND The file was not found.\r
587 @retval EFI_NOT_FOUND The header checksum was not zero.\r
588 @retval EFI_SUCCESS The file was found.\r
b1f6a7c6 589\r
590**/\r
192f6d4c 591EFI_STATUS\r
592EFIAPI\r
593PeiFfsFindNextFile (\r
0c2b5da8 594 IN CONST EFI_PEI_SERVICES **PeiServices,\r
595 IN UINT8 SearchType,\r
3b428ade 596 IN EFI_PEI_FV_HANDLE FvHandle,\r
0c2b5da8 597 IN OUT EFI_PEI_FILE_HANDLE *FileHandle\r
192f6d4c 598 )\r
192f6d4c 599{\r
3b428ade 600 PEI_CORE_FV_HANDLE *CoreFvHandle;\r
601 \r
602 CoreFvHandle = FvHandleToCoreHandle (FvHandle);\r
603 \r
2f016ba8 604 //\r
605 // To make backward compatiblity, if can not find corresponding the handle of FV\r
606 // then treat FV as build-in FFS2 format and memory mapped FV that FV handle is pointed\r
607 // to the address of first byte of FV.\r
608 //\r
609 if ((CoreFvHandle == NULL) && FeaturePcdGet (PcdFrameworkCompatibilitySupport)) {\r
610 return FindFileEx (FvHandle, NULL, SearchType, FileHandle, NULL);\r
611 } \r
612 \r
3b428ade 613 if ((CoreFvHandle == NULL) || CoreFvHandle->FvPpi == NULL) {\r
614 return EFI_NOT_FOUND;\r
615 }\r
616 \r
617 return CoreFvHandle->FvPpi->FindFileByType (CoreFvHandle->FvPpi, SearchType, FvHandle, FileHandle);\r
192f6d4c 618}\r
619\r
b0d803fe 620\r
b1f6a7c6 621/**\r
ed299e3c 622 Search the firmware volumes by index\r
b1f6a7c6 623\r
ed299e3c 624 @param PeiServices An indirect pointer to the EFI_PEI_SERVICES table published by the PEI Foundation\r
3b428ade 625 @param Instance This instance of the firmware volume to find. The value 0 is the Boot Firmware\r
626 Volume (BFV).\r
627 @param VolumeHandle On exit, points to the next volume handle or NULL if it does not exist.\r
b1f6a7c6 628\r
3b428ade 629 @retval EFI_INVALID_PARAMETER VolumeHandle is NULL\r
630 @retval EFI_NOT_FOUND The volume was not found.\r
631 @retval EFI_SUCCESS The volume was found.\r
b1f6a7c6 632\r
633**/\r
192f6d4c 634EFI_STATUS \r
635EFIAPI\r
3b428ade 636PeiFfsFindNextVolume (\r
2a00326e 637 IN CONST EFI_PEI_SERVICES **PeiServices,\r
192f6d4c 638 IN UINTN Instance,\r
0c2b5da8 639 IN OUT EFI_PEI_FV_HANDLE *VolumeHandle\r
192f6d4c 640 )\r
192f6d4c 641{\r
3b428ade 642 PEI_CORE_INSTANCE *Private;\r
643 PEI_CORE_FV_HANDLE *CoreFvHandle;\r
644 \r
3e4c5b49 645 if (VolumeHandle == NULL) {\r
646 return EFI_INVALID_PARAMETER;\r
647 }\r
648 \r
b0d803fe 649 Private = PEI_CORE_INSTANCE_FROM_PS_THIS (PeiServices);\r
2a00326e 650 \r
3b428ade 651 CoreFvHandle = FindNextCoreFvHandle (Private, Instance);\r
652 if (CoreFvHandle == NULL) {\r
653 *VolumeHandle = NULL;\r
654 return EFI_NOT_FOUND;\r
2a00326e 655 }\r
3b428ade 656 \r
657 *VolumeHandle = CoreFvHandle->FvHandle;\r
658 \r
b0d803fe 659 return EFI_SUCCESS;\r
660}\r
192f6d4c 661\r
192f6d4c 662\r
b1f6a7c6 663/**\r
ed299e3c 664 Find a file within a volume by its name.\r
b1f6a7c6 665\r
ed299e3c
LG
666 @param FileName A pointer to the name of the file to find within the firmware volume.\r
667 @param VolumeHandle The firmware volume to search\r
668 @param FileHandle Upon exit, points to the found file's handle \r
669 or NULL if it could not be found.\r
b1f6a7c6 670\r
ed299e3c
LG
671 @retval EFI_SUCCESS File was found.\r
672 @retval EFI_NOT_FOUND File was not found.\r
673 @retval EFI_INVALID_PARAMETER VolumeHandle or FileHandle or FileName was NULL.\r
b1f6a7c6 674\r
b1f6a7c6 675**/\r
b0d803fe 676EFI_STATUS\r
677EFIAPI \r
678PeiFfsFindFileByName (\r
679 IN CONST EFI_GUID *FileName,\r
680 IN EFI_PEI_FV_HANDLE VolumeHandle,\r
681 OUT EFI_PEI_FILE_HANDLE *FileHandle\r
682 )\r
b0d803fe 683{\r
3b428ade 684 PEI_CORE_FV_HANDLE *CoreFvHandle;\r
685 \r
b0d803fe 686 if ((VolumeHandle == NULL) || (FileName == NULL) || (FileHandle == NULL)) {\r
192f6d4c 687 return EFI_INVALID_PARAMETER;\r
688 }\r
3b428ade 689 \r
690 CoreFvHandle = FvHandleToCoreHandle (VolumeHandle);\r
691 if ((CoreFvHandle == NULL) || (CoreFvHandle->FvPpi == NULL)) {\r
692 return EFI_NOT_FOUND;\r
b0d803fe 693 }\r
3b428ade 694 \r
695 return CoreFvHandle->FvPpi->FindFileByName (CoreFvHandle->FvPpi, FileName, &VolumeHandle, FileHandle);\r
b0d803fe 696}\r
697\r
b1f6a7c6 698/**\r
b1f6a7c6 699 Returns information about a specific file.\r
700\r
3b428ade 701 @param FileHandle Handle of the file.\r
702 @param FileInfo Upon exit, points to the file’s information.\r
b1f6a7c6 703\r
3b428ade 704 @retval EFI_INVALID_PARAMETER If FileInfo is NULL.\r
705 @retval EFI_INVALID_PARAMETER If FileHandle does not represent a valid file.\r
706 @retval EFI_SUCCESS File information returned.\r
b1f6a7c6 707\r
708**/\r
b0d803fe 709EFI_STATUS\r
710EFIAPI \r
711PeiFfsGetFileInfo (\r
712 IN EFI_PEI_FILE_HANDLE FileHandle,\r
713 OUT EFI_FV_FILE_INFO *FileInfo\r
714 )\r
b0d803fe 715{\r
3b428ade 716 PEI_CORE_FV_HANDLE *CoreFvHandle;\r
717 \r
b0d803fe 718 if ((FileHandle == NULL) || (FileInfo == NULL)) {\r
719 return EFI_INVALID_PARAMETER;\r
720 }\r
721\r
722 //\r
723 // Retrieve the FirmwareVolume which the file resides in.\r
724 //\r
3b428ade 725 CoreFvHandle = FileHandleToVolume (FileHandle);\r
726 if ((CoreFvHandle == NULL) || (CoreFvHandle->FvPpi == NULL)) {\r
b0d803fe 727 return EFI_INVALID_PARAMETER;\r
728 }\r
729\r
3b428ade 730 return CoreFvHandle->FvPpi->GetFileInfo (CoreFvHandle->FvPpi, FileHandle, FileInfo);\r
b0d803fe 731}\r
732\r
733\r
b1f6a7c6 734/**\r
3b428ade 735 Returns information about the specified volume.\r
b1f6a7c6 736\r
5a9403b8 737 This function returns information about a specific firmware\r
738 volume, including its name, type, attributes, starting address\r
739 and size.\r
740\r
741 @param VolumeHandle Handle of the volume.\r
742 @param VolumeInfo Upon exit, points to the volume's information.\r
743\r
744 @retval EFI_SUCCESS Volume information returned.\r
745 @retval EFI_INVALID_PARAMETER If VolumeHandle does not represent a valid volume.\r
746 @retval EFI_INVALID_PARAMETER If VolumeHandle is NULL.\r
747 @retval EFI_SUCCESS Information successfully returned.\r
748 @retval EFI_INVALID_PARAMETER The volume designated by the VolumeHandle is not available.\r
b1f6a7c6 749\r
b1f6a7c6 750**/\r
b0d803fe 751EFI_STATUS\r
752EFIAPI \r
753PeiFfsGetVolumeInfo (\r
754 IN EFI_PEI_FV_HANDLE VolumeHandle,\r
755 OUT EFI_FV_INFO *VolumeInfo\r
756 )\r
b0d803fe 757{\r
3b428ade 758 PEI_CORE_FV_HANDLE *CoreHandle;\r
759 \r
b0d803fe 760 if (VolumeInfo == NULL) {\r
761 return EFI_INVALID_PARAMETER;\r
192f6d4c 762 }\r
288f9b38 763 \r
3b428ade 764 CoreHandle = FvHandleToCoreHandle (VolumeHandle);\r
765 \r
766 if ((CoreHandle == NULL) || (CoreHandle->FvPpi == NULL)) {\r
288f9b38
LG
767 return EFI_INVALID_PARAMETER;\r
768 }\r
3b428ade 769 \r
770 return CoreHandle->FvPpi->GetVolumeInfo (CoreHandle->FvPpi, VolumeHandle, VolumeInfo);\r
192f6d4c 771}\r
b0d803fe 772\r
4c6486b9 773/**\r
774 Get Fv image from the FV type file, then install FV INFO ppi, Build FV hob.\r
775\r
3b428ade 776 @param ParentFvCoreHandle Pointer of EFI_CORE_FV_HANDLE to parent Fv image that contain this Fv image.\r
97b2c9b5 777 @param ParentFvFileHandle File handle of a Fv type file that contain this Fv image.\r
4c6486b9 778\r
779 @retval EFI_NOT_FOUND FV image can't be found.\r
780 @retval EFI_SUCCESS Successfully to process it.\r
781 @retval EFI_OUT_OF_RESOURCES Can not allocate page when aligning FV image\r
782 @retval Others Can not find EFI_SECTION_FIRMWARE_VOLUME_IMAGE section\r
783 \r
784**/\r
785EFI_STATUS\r
786ProcessFvFile (\r
3b428ade 787 IN PEI_CORE_FV_HANDLE *ParentFvCoreHandle,\r
788 IN EFI_PEI_FILE_HANDLE ParentFvFileHandle\r
4c6486b9 789 )\r
790{\r
3b428ade 791 EFI_STATUS Status;\r
792 EFI_FV_INFO ParentFvImageInfo;\r
793 UINT32 FvAlignment;\r
794 VOID *NewFvBuffer;\r
795 EFI_PEI_HOB_POINTERS HobPtr;\r
796 EFI_PEI_FIRMWARE_VOLUME_PPI *ParentFvPpi;\r
797 EFI_PEI_FV_HANDLE ParentFvHandle;\r
798 EFI_FIRMWARE_VOLUME_HEADER *FvHeader;\r
799 EFI_FV_FILE_INFO FileInfo;\r
800 \r
4c6486b9 801 //\r
802 // Check if this EFI_FV_FILETYPE_FIRMWARE_VOLUME_IMAGE file has already\r
803 // been extracted.\r
804 //\r
805 HobPtr.Raw = GetHobList ();\r
806 while ((HobPtr.Raw = GetNextHob (EFI_HOB_TYPE_FV2, HobPtr.Raw)) != NULL) {\r
97b2c9b5 807 if (CompareGuid (&(((EFI_FFS_FILE_HEADER *)ParentFvFileHandle)->Name), &HobPtr.FirmwareVolume2->FileName)) {\r
4c6486b9 808 //\r
809 // this FILE has been dispatched, it will not be dispatched again.\r
810 //\r
3b428ade 811 DEBUG ((EFI_D_INFO, "FV file %p has been dispatched!\r\n", ParentFvFileHandle));\r
4c6486b9 812 return EFI_SUCCESS;\r
813 }\r
814 HobPtr.Raw = GET_NEXT_HOB (HobPtr);\r
815 }\r
816\r
3b428ade 817 ParentFvHandle = ParentFvCoreHandle->FvHandle;\r
818 ParentFvPpi = ParentFvCoreHandle->FvPpi;\r
819 \r
4c6486b9 820 //\r
821 // Find FvImage in FvFile\r
822 //\r
3b428ade 823 Status = ParentFvPpi->FindSectionByType (\r
824 ParentFvPpi,\r
825 EFI_SECTION_FIRMWARE_VOLUME_IMAGE,\r
826 ParentFvFileHandle,\r
827 (VOID **)&FvHeader\r
828 );\r
829 \r
4c6486b9 830 if (EFI_ERROR (Status)) {\r
831 return Status;\r
832 }\r
97b2c9b5
LG
833\r
834 //\r
3b428ade 835 // FvAlignment must be more than 8 bytes required by FvHeader structure.\r
4c6486b9 836 //\r
3b428ade 837 FvAlignment = 1 << ((FvHeader->Attributes & EFI_FVB2_ALIGNMENT) >> 16);\r
4c6486b9 838 if (FvAlignment < 8) {\r
839 FvAlignment = 8;\r
840 }\r
841 \r
842 //\r
843 // Check FvImage\r
844 //\r
3b428ade 845 if ((UINTN) FvHeader % FvAlignment != 0) {\r
846 NewFvBuffer = AllocateAlignedPages (EFI_SIZE_TO_PAGES ((UINT32) FvHeader->FvLength), FvAlignment);\r
847 if (NewFvBuffer == NULL) {\r
4c6486b9 848 return EFI_OUT_OF_RESOURCES;\r
849 }\r
3b428ade 850 CopyMem (NewFvBuffer, FvHeader, (UINTN) FvHeader->FvLength);\r
851 FvHeader = (EFI_FIRMWARE_VOLUME_HEADER*) NewFvBuffer;\r
4c6486b9 852 }\r
3b428ade 853 \r
854 Status = ParentFvPpi->GetVolumeInfo (ParentFvPpi, ParentFvHandle, &ParentFvImageInfo);\r
855 ASSERT_EFI_ERROR (Status);\r
856 \r
857 Status = ParentFvPpi->GetFileInfo (ParentFvPpi, ParentFvFileHandle, &FileInfo);\r
858 ASSERT_EFI_ERROR (Status);\r
859 \r
4c6486b9 860 //\r
861 // Install FvPpi and Build FvHob\r
862 //\r
187f1a5b 863 PeiServicesInstallFvInfoPpi (\r
3b428ade 864 &FvHeader->FileSystemGuid,\r
865 (VOID**) FvHeader,\r
866 (UINT32) FvHeader->FvLength,\r
97b2c9b5 867 &ParentFvImageInfo.FvName,\r
3b428ade 868 &FileInfo.FileName\r
4c6486b9 869 );\r
870\r
871 //\r
872 // Inform the extracted FvImage to Fv HOB consumer phase, i.e. DXE phase\r
4c6486b9 873 //\r
223c22d8 874 BuildFvHob (\r
3b428ade 875 (EFI_PHYSICAL_ADDRESS) (UINTN) FvHeader,\r
876 FvHeader->FvLength\r
18fcd6a7 877 );\r
4c6486b9 878\r
879 //\r
880 // Makes the encapsulated volume show up in DXE phase to skip processing of\r
881 // encapsulated file again.\r
882 //\r
883 BuildFv2Hob (\r
3b428ade 884 (EFI_PHYSICAL_ADDRESS) (UINTN) FvHeader,\r
885 FvHeader->FvLength,\r
97b2c9b5 886 &ParentFvImageInfo.FvName,\r
3b428ade 887 &FileInfo.FileName\r
4c6486b9 888 );\r
889\r
890 return EFI_SUCCESS;\r
891}\r
892\r
3b428ade 893/**\r
894 Process a firmware volume and create a volume handle.\r
895\r
896 Create a volume handle from the information in the buffer. For\r
897 memory-mapped firmware volumes, Buffer and BufferSize refer to\r
898 the start of the firmware volume and the firmware volume size.\r
899 For non memory-mapped firmware volumes, this points to a\r
900 buffer which contains the necessary information for creating\r
901 the firmware volume handle. Normally, these values are derived\r
902 from the EFI_FIRMWARE_VOLUME_INFO_PPI.\r
903 \r
904 \r
905 @param This Points to this instance of the\r
906 EFI_PEI_FIRMWARE_VOLUME_PPI.\r
907 @param Buffer Points to the start of the buffer.\r
908 @param BufferSize Size of the buffer.\r
909 @param FvHandle Points to the returned firmware volume\r
910 handle. The firmware volume handle must\r
911 be unique within the system. \r
912\r
913 @retval EFI_SUCCESS Firmware volume handle created.\r
914 @retval EFI_VOLUME_CORRUPTED Volume was corrupt.\r
915\r
916**/\r
917EFI_STATUS\r
918EFIAPI\r
919PeiFfs2FvPpiProcessVolume (\r
920 IN CONST EFI_PEI_FIRMWARE_VOLUME_PPI *This,\r
921 IN VOID *Buffer,\r
922 IN UINTN BufferSize,\r
923 OUT EFI_PEI_FV_HANDLE *FvHandle\r
924 )\r
925{\r
926 EFI_STATUS Status;\r
927 PEI_CORE_INSTANCE *PrivateData;\r
928 EFI_PEI_FILE_HANDLE FileHandle;\r
929 VOID *DepexData;\r
930 EFI_PEI_SERVICES **PeiServices;\r
931 UINTN FvIndex;\r
932 \r
933 PeiServices = (EFI_PEI_SERVICES**) GetPeiServicesTablePointer ();\r
934 PrivateData = PEI_CORE_INSTANCE_FROM_PS_THIS (PeiServices);\r
935 \r
936 //\r
937 // The build-in EFI_PEI_FIRMWARE_VOLUME_PPI for FFS2 support memory-mapped\r
938 // FV image and the handle is pointed to Fv image's buffer.\r
939 //\r
940 *FvHandle = (EFI_PEI_FV_HANDLE) Buffer;\r
941 \r
942 //\r
943 // Do verify for given FV buffer.\r
944 //\r
945 Status = VerifyFv ((EFI_FIRMWARE_VOLUME_HEADER*) Buffer);\r
946 if (EFI_ERROR(Status)) {\r
947 DEBUG ((EFI_D_ERROR, "Fail to verify FV which address is 0x%11p", Buffer));\r
948 return EFI_VOLUME_CORRUPTED;\r
949 }\r
950\r
951 //\r
952 // Check whether the FV has already been processed.\r
953 //\r
954 for (FvIndex = 0; FvIndex < PrivateData->FvCount; FvIndex ++) {\r
955 if (PrivateData->Fv[FvIndex].FvHandle == *FvHandle) {\r
956 DEBUG ((EFI_D_INFO, "The Fv %p has already been processed!", Buffer));\r
957 return EFI_SUCCESS;\r
958 }\r
959 }\r
960 \r
961 //\r
962 // Update internal PEI_CORE_FV array.\r
963 //\r
964 PrivateData->Fv[PrivateData->FvCount].FvHeader = (EFI_FIRMWARE_VOLUME_HEADER*) Buffer;\r
965 PrivateData->Fv[PrivateData->FvCount].FvPpi = (EFI_PEI_FIRMWARE_VOLUME_PPI*) This;\r
966 PrivateData->Fv[PrivateData->FvCount].FvHandle = *FvHandle;\r
967 \r
968 DEBUG ((EFI_D_INFO, \r
969 "The %dth FV start address is 0x%11p and size is 0x%08x\n", \r
970 (UINT32) PrivateData->FvCount, \r
971 (VOID *) Buffer, \r
972 BufferSize\r
973 ));\r
974 PrivateData->FvCount ++;\r
975 \r
ca577976 976 FileHandle = NULL;\r
977 \r
3b428ade 978 do {\r
979 Status = This->FindFileByType (\r
980 This,\r
981 EFI_FV_FILETYPE_FIRMWARE_VOLUME_IMAGE,\r
982 *FvHandle,\r
983 &FileHandle\r
984 );\r
985 if (!EFI_ERROR (Status)) {\r
986 Status = This->FindSectionByType (\r
987 This,\r
988 EFI_SECTION_PEI_DEPEX,\r
989 FileHandle,\r
990 (VOID**)&DepexData\r
991 );\r
992 if (!EFI_ERROR (Status)) {\r
993 if (!PeimDispatchReadiness (PeiServices, DepexData)) {\r
994 //\r
995 // Dependency is not satisfied.\r
996 //\r
997 continue;\r
998 }\r
999 }\r
1000 \r
1001 DEBUG ((EFI_D_INFO, "Found firmware volume Image File %p in FV[%d] %p", FileHandle, PrivateData->FvCount - 1, *FvHandle));\r
1002 ProcessFvFile (&PrivateData->Fv[PrivateData->FvCount - 1], FileHandle);\r
1003 }\r
1004 } while (FileHandle != NULL);\r
1005\r
1006 return EFI_SUCCESS;\r
1007} \r
1008\r
1009/**\r
1010 Finds the next file of the specified type.\r
1011\r
1012 This service enables PEI modules to discover additional firmware files. \r
1013 The FileHandle must be unique within the system.\r
1014\r
1015 @param This Points to this instance of the\r
1016 EFI_PEI_FIRMWARE_VOLUME_PPI.\r
1017 @param SearchType A filter to find only files of this type. Type\r
1018 EFI_FV_FILETYPE_ALL causes no filtering to be\r
1019 done.\r
1020 @param FvHandle Handle of firmware volume in which to\r
1021 search.\r
1022 @param FileHandle Points to the current handle from which to\r
1023 begin searching or NULL to start at the\r
1024 beginning of the firmware volume. Updated\r
1025 upon return to reflect the file found.\r
1026\r
1027 @retval EFI_SUCCESS The file was found.\r
1028 @retval EFI_NOT_FOUND The file was not found. FileHandle contains NULL.\r
1029\r
1030**/\r
1031EFI_STATUS\r
1032EFIAPI\r
1033PeiFfs2FvPpiFindFileByType (\r
1034 IN CONST EFI_PEI_FIRMWARE_VOLUME_PPI *This,\r
1035 IN EFI_FV_FILETYPE SearchType,\r
1036 IN EFI_PEI_FV_HANDLE FvHandle,\r
1037 IN OUT EFI_PEI_FILE_HANDLE *FileHandle\r
1038 )\r
1039{ \r
1040 return FindFileEx (FvHandle, NULL, SearchType, FileHandle, NULL);\r
1041}\r
1042\r
1043/**\r
1044 Find a file within a volume by its name. \r
1045 \r
1046 This service searches for files with a specific name, within\r
1047 either the specified firmware volume or all firmware volumes.\r
1048\r
1049 @param This Points to this instance of the\r
1050 EFI_PEI_FIRMWARE_VOLUME_PPI.\r
1051 @param FileName A pointer to the name of the file to find\r
1052 within the firmware volume.\r
1053 @param FvHandle Upon entry, the pointer to the firmware\r
1054 volume to search or NULL if all firmware\r
1055 volumes should be searched. Upon exit, the\r
1056 actual firmware volume in which the file was\r
1057 found.\r
1058 @param FileHandle Upon exit, points to the found file's\r
1059 handle or NULL if it could not be found.\r
1060\r
1061 @retval EFI_SUCCESS File was found.\r
1062 @retval EFI_NOT_FOUND File was not found.\r
1063 @retval EFI_INVALID_PARAMETER FvHandle or FileHandle or\r
1064 FileName was NULL.\r
1065\r
1066\r
1067**/\r
1068EFI_STATUS\r
1069EFIAPI\r
1070PeiFfs2FvPpiFindFileByName (\r
1071 IN CONST EFI_PEI_FIRMWARE_VOLUME_PPI *This,\r
1072 IN CONST EFI_GUID *FileName,\r
1073 IN EFI_PEI_FV_HANDLE *FvHandle,\r
1074 OUT EFI_PEI_FILE_HANDLE *FileHandle \r
1075 )\r
1076{\r
1077 EFI_STATUS Status;\r
1078 PEI_CORE_INSTANCE *PrivateData;\r
1079 UINTN Index;\r
1080 \r
1081 if ((FvHandle == NULL) || (FileName == NULL) || (FileHandle == NULL)) {\r
1082 return EFI_INVALID_PARAMETER;\r
1083 }\r
1084 \r
1085 if (*FvHandle != NULL) {\r
1086 Status = FindFileEx (*FvHandle, FileName, 0, FileHandle, NULL);\r
1087 if (Status == EFI_NOT_FOUND) {\r
1088 *FileHandle = NULL;\r
1089 }\r
1090 } else { \r
1091 //\r
1092 // If *FvHandle = NULL, so search all FV for given filename\r
1093 //\r
1094 Status = EFI_NOT_FOUND;\r
1095 \r
1096 PrivateData = PEI_CORE_INSTANCE_FROM_PS_THIS (GetPeiServicesTablePointer());\r
1097 for (Index = 0; Index < PrivateData->FvCount; Index ++) {\r
1098 //\r
1099 // Only search the FV which is associated with a EFI_PEI_FIRMWARE_VOLUME_PPI instance.\r
1100 //\r
1101 if (PrivateData->Fv[Index].FvPpi != NULL) {\r
1102 Status = FindFileEx (PrivateData->Fv[Index].FvHandle, FileName, 0, FileHandle, NULL);\r
1103 if (!EFI_ERROR (Status)) {\r
1104 *FvHandle = PrivateData->Fv[Index].FvHandle;\r
1105 }\r
1106 }\r
1107 }\r
1108 }\r
1109 \r
1110 return Status; \r
1111} \r
1112\r
1113/**\r
1114 Returns information about a specific file.\r
1115\r
1116 This function returns information about a specific\r
1117 file, including its file name, type, attributes, starting\r
1118 address and size. \r
1119 \r
1120 @param This Points to this instance of the\r
1121 EFI_PEI_FIRMWARE_VOLUME_PPI.\r
1122 @param FileHandle Handle of the file.\r
1123 @param FileInfo Upon exit, points to the file's\r
1124 information.\r
1125\r
1126 @retval EFI_SUCCESS File information returned.\r
1127 @retval EFI_INVALID_PARAMETER If FileHandle does not\r
1128 represent a valid file.\r
1129 @retval EFI_INVALID_PARAMETER If FileInfo is NULL.\r
1130 \r
1131**/ \r
1132EFI_STATUS\r
1133EFIAPI\r
1134PeiFfs2FvPpiGetFileInfo (\r
1135 IN CONST EFI_PEI_FIRMWARE_VOLUME_PPI *This, \r
1136 IN EFI_PEI_FILE_HANDLE FileHandle, \r
1137 OUT EFI_FV_FILE_INFO *FileInfo\r
1138 )\r
1139{\r
1140 UINT8 FileState;\r
1141 UINT8 ErasePolarity;\r
1142 EFI_FFS_FILE_HEADER *FileHeader;\r
1143 PEI_CORE_FV_HANDLE *CoreFvHandle;\r
1144 \r
1145 if ((FileHandle == NULL) || (FileInfo == NULL)) {\r
1146 return EFI_INVALID_PARAMETER;\r
1147 }\r
1148\r
1149 //\r
1150 // Retrieve the FirmwareVolume which the file resides in.\r
1151 //\r
1152 CoreFvHandle = FileHandleToVolume (FileHandle);\r
1153 if (CoreFvHandle == NULL) {\r
1154 return EFI_INVALID_PARAMETER;\r
1155 }\r
1156\r
1157 if (CoreFvHandle->FvHeader->Attributes & EFI_FVB2_ERASE_POLARITY) {\r
1158 ErasePolarity = 1;\r
1159 } else {\r
1160 ErasePolarity = 0;\r
1161 }\r
1162\r
1163 //\r
1164 // Get FileState which is the highest bit of the State \r
1165 //\r
1166 FileState = GetFileState (ErasePolarity, (EFI_FFS_FILE_HEADER*)FileHandle);\r
1167\r
1168 switch (FileState) {\r
1169 case EFI_FILE_DATA_VALID:\r
1170 case EFI_FILE_MARKED_FOR_UPDATE:\r
1171 break; \r
1172 default:\r
1173 return EFI_INVALID_PARAMETER;\r
1174 }\r
1175\r
1176 FileHeader = (EFI_FFS_FILE_HEADER *)FileHandle;\r
1177 CopyMem (&FileInfo->FileName, &FileHeader->Name, sizeof(EFI_GUID));\r
1178 FileInfo->FileType = FileHeader->Type;\r
1179 FileInfo->FileAttributes = FileHeader->Attributes;\r
1180 FileInfo->BufferSize = ((*(UINT32 *)FileHeader->Size) & 0x00FFFFFF) - sizeof (EFI_FFS_FILE_HEADER);\r
1181 FileInfo->Buffer = (FileHeader + 1);\r
1182 return EFI_SUCCESS; \r
1183} \r
1184 \r
1185/**\r
1186 This function returns information about the firmware volume.\r
1187 \r
1188 @param This Points to this instance of the\r
1189 EFI_PEI_FIRMWARE_VOLUME_PPI.\r
1190 @param FvHandle Handle to the firmware handle.\r
1191 @param VolumeInfo Points to the returned firmware volume\r
1192 information.\r
1193\r
1194 @retval EFI_SUCCESS Information returned successfully.\r
1195 @retval EFI_INVALID_PARAMETER FvHandle does not indicate a valid\r
1196 firmware volume or VolumeInfo is NULL.\r
1197\r
1198**/ \r
1199EFI_STATUS\r
1200EFIAPI\r
1201PeiFfs2FvPpiGetVolumeInfo (\r
1202 IN CONST EFI_PEI_FIRMWARE_VOLUME_PPI *This, \r
1203 IN EFI_PEI_FV_HANDLE FvHandle, \r
1204 OUT EFI_FV_INFO *VolumeInfo\r
1205 )\r
1206{\r
1207 EFI_FIRMWARE_VOLUME_HEADER FwVolHeader;\r
1208 EFI_FIRMWARE_VOLUME_EXT_HEADER *FwVolExHeaderInfo;\r
1209\r
1210 if (VolumeInfo == NULL) {\r
1211 return EFI_INVALID_PARAMETER;\r
1212 }\r
1213 \r
1214 //\r
1215 // VolumeHandle may not align at 8 byte, \r
1216 // but FvLength is UINT64 type, which requires FvHeader align at least 8 byte. \r
1217 // So, Copy FvHeader into the local FvHeader structure.\r
1218 //\r
1219 CopyMem (&FwVolHeader, FvHandle, sizeof (EFI_FIRMWARE_VOLUME_HEADER));\r
1220\r
1221 //\r
1222 // Check Fv Image Signature\r
1223 //\r
1224 if (FwVolHeader.Signature != EFI_FVH_SIGNATURE) {\r
1225 return EFI_INVALID_PARAMETER;\r
1226 }\r
1227\r
1228 ZeroMem (VolumeInfo, sizeof (EFI_FV_INFO));\r
1229 VolumeInfo->FvAttributes = FwVolHeader.Attributes;\r
1230 VolumeInfo->FvStart = (VOID *) FvHandle;\r
1231 VolumeInfo->FvSize = FwVolHeader.FvLength;\r
1232 CopyMem (&VolumeInfo->FvFormat, &FwVolHeader.FileSystemGuid, sizeof(EFI_GUID));\r
1233\r
1234 if (FwVolHeader.ExtHeaderOffset != 0) {\r
1235 FwVolExHeaderInfo = (EFI_FIRMWARE_VOLUME_EXT_HEADER*)(((UINT8 *)FvHandle) + FwVolHeader.ExtHeaderOffset);\r
1236 CopyMem (&VolumeInfo->FvName, &FwVolExHeaderInfo->FvName, sizeof(EFI_GUID));\r
1237 }\r
1238 \r
1239 return EFI_SUCCESS; \r
1240} \r
1241\r
1242/**\r
1243 Find the next matching section in the firmware file.\r
1244 \r
1245 This service enables PEI modules to discover sections\r
1246 of a given type within a valid file.\r
1247 \r
1248 @param This Points to this instance of the\r
1249 EFI_PEI_FIRMWARE_VOLUME_PPI.\r
1250 @param SearchType A filter to find only sections of this\r
1251 type.\r
1252 @param FileHandle Handle of firmware file in which to\r
1253 search.\r
1254 @param SectionData Updated upon return to point to the\r
1255 section found.\r
1256 \r
1257 @retval EFI_SUCCESS Section was found.\r
1258 @retval EFI_NOT_FOUND Section of the specified type was not\r
1259 found. SectionData contains NULL.\r
1260**/\r
1261EFI_STATUS\r
1262EFIAPI\r
1263PeiFfs2FvPpiFindSectionByType (\r
1264 IN CONST EFI_PEI_FIRMWARE_VOLUME_PPI *This,\r
1265 IN EFI_SECTION_TYPE SearchType,\r
1266 IN EFI_PEI_FILE_HANDLE FileHandle,\r
1267 OUT VOID **SectionData\r
1268 )\r
1269{\r
1270 EFI_FFS_FILE_HEADER *FfsFileHeader;\r
1271 UINT32 FileSize;\r
1272 EFI_COMMON_SECTION_HEADER *Section;\r
1273 \r
1274 FfsFileHeader = (EFI_FFS_FILE_HEADER *)(FileHandle);\r
1275\r
1276 //\r
1277 // Size is 24 bits wide so mask upper 8 bits. \r
1278 // Does not include FfsFileHeader header size\r
1279 // FileSize is adjusted to FileOccupiedSize as it is 8 byte aligned.\r
1280 //\r
1281 Section = (EFI_COMMON_SECTION_HEADER *)(FfsFileHeader + 1);\r
1282 FileSize = *(UINT32 *)(FfsFileHeader->Size) & 0x00FFFFFF;\r
1283 FileSize -= sizeof (EFI_FFS_FILE_HEADER);\r
1284\r
1285 return ProcessSection (\r
1286 GetPeiServicesTablePointer (),\r
1287 SearchType, \r
1288 Section, \r
1289 FileSize, \r
1290 SectionData\r
1291 ); \r
1292} \r
1293\r
1294/**\r
1295 Convert the handle of FV to pointer of corresponding PEI_CORE_FV_HANDLE.\r
1296 \r
1297 @param FvHandle The handle of a FV.\r
1298 \r
1299 @retval NULL if can not find.\r
1300 @return Pointer of corresponding PEI_CORE_FV_HANDLE. \r
1301**/\r
1302PEI_CORE_FV_HANDLE *\r
1303FvHandleToCoreHandle (\r
1304 IN EFI_PEI_FV_HANDLE FvHandle\r
1305 )\r
1306{\r
1307 UINTN Index;\r
1308 PEI_CORE_INSTANCE *PrivateData;\r
1309 \r
1310 PrivateData = PEI_CORE_INSTANCE_FROM_PS_THIS (GetPeiServicesTablePointer()); \r
1311 for (Index = 0; Index < PrivateData->FvCount; Index ++) {\r
1312 if (FvHandle == PrivateData->Fv[Index].FvHandle) {\r
1313 return &PrivateData->Fv[Index];\r
1314 }\r
1315 }\r
1316 \r
1317 return NULL;\r
1318} \r
1319\r
1320/**\r
1321 Get instance of PEI_CORE_FV_HANDLE for next volume according to given index.\r
1322 \r
1323 This routine also will install FvInfo ppi for FV hob in PI ways.\r
1324 \r
1325 @param Private Pointer of PEI_CORE_INSTANCE\r
1326 @param Instance The index of FV want to be searched.\r
1327 \r
1328 @return Instance of PEI_CORE_FV_HANDLE.\r
1329**/\r
1330PEI_CORE_FV_HANDLE *\r
1331FindNextCoreFvHandle (\r
1332 IN PEI_CORE_INSTANCE *Private,\r
1333 IN UINTN Instance\r
1334 )\r
1335{\r
1336 UINTN Index;\r
1337 BOOLEAN Match;\r
1338 EFI_HOB_FIRMWARE_VOLUME *FvHob;\r
1339 \r
1340 //\r
1341 // Handle Framework FvHob and Install FvInfo Ppi for it.\r
1342 //\r
1343 if (FeaturePcdGet (PcdFrameworkCompatibilitySupport)) {\r
1344 //\r
1345 // Loop to search the wanted FirmwareVolume which supports FFS\r
1346 //\r
1347 FvHob = (EFI_HOB_FIRMWARE_VOLUME *)GetFirstHob (EFI_HOB_TYPE_FV);\r
1348 while (FvHob != NULL) {\r
1349 for (Index = 0, Match = FALSE; Index < Private->FvCount; Index++) {\r
1350 if ((EFI_PEI_FV_HANDLE)(UINTN)FvHob->BaseAddress == Private->Fv[Index].FvHeader) {\r
1351 Match = TRUE;\r
1352 break;\r
1353 }\r
1354 }\r
1355 //\r
1356 // If Not Found, Install FvInfo Ppi for it.\r
1357 //\r
1358 if (!Match) {\r
1359 PeiServicesInstallFvInfoPpi (\r
1360 &(((EFI_FIRMWARE_VOLUME_HEADER *)(UINTN)FvHob->BaseAddress)->FileSystemGuid),\r
1361 (VOID *)(UINTN)FvHob->BaseAddress,\r
1362 (UINT32)FvHob->Length,\r
1363 NULL,\r
1364 NULL\r
1365 );\r
1366 }\r
1367 FvHob = (EFI_HOB_FIRMWARE_VOLUME *)GetNextHob (EFI_HOB_TYPE_FV, (VOID *)((UINTN)FvHob + FvHob->Header.HobLength)); \r
1368 }\r
1369 }\r
1370\r
1371 if (Instance >= Private->FvCount) {\r
1372 return NULL;\r
1373 }\r
4c6486b9 1374\r
3b428ade 1375 return &Private->Fv[Instance];\r
1376} \r
8e0e40ed 1377\r
1378/**\r
1379 After PeiCore image is shadowed into permanent memory, all build-in FvPpi should\r
1380 be re-installed with the instance in permanent memory and all cached FvPpi pointers in \r
1381 PrivateData->Fv[] array should be fixed up to be pointed to the one in permenant\r
1382 memory.\r
1383 \r
1384 @param PrivateData Pointer to PEI_CORE_INSTANCE.\r
1385**/ \r
1386VOID\r
1387PeiReinitializeFv (\r
1388 IN PEI_CORE_INSTANCE *PrivateData\r
1389 )\r
1390{\r
7ec93917 1391 VOID *OldFfs2FvPpi;\r
1392 EFI_PEI_PPI_DESCRIPTOR *OldDescriptor;\r
1393 UINTN Index;\r
1394 EFI_STATUS Status;\r
1395\r
1396 //\r
1397 // Locate old build-in Ffs2 EFI_PEI_FIRMWARE_VOLUME_PPI which\r
1398 // in flash.\r
1399 //\r
1400 Status = PeiServicesLocatePpi (\r
1401 &gEfiFirmwareFileSystem2Guid,\r
1402 0,\r
1403 &OldDescriptor,\r
1404 &OldFfs2FvPpi\r
1405 );\r
1406 ASSERT_EFI_ERROR (Status);\r
1407\r
1408 //\r
1409 // Re-install the EFI_PEI_FIRMWARE_VOLUME_PPI for build-in Ffs2\r
1410 // which is shadowed from flash to permanent memory within PeiCore image.\r
1411 //\r
1412 Status = PeiServicesReInstallPpi (OldDescriptor, &mPeiFfs2FvPpiList);\r
1413 ASSERT_EFI_ERROR (Status);\r
1414 \r
1415 //\r
1416 // Fixup all FvPpi pointers for the implementation in flash to permanent memory.\r
1417 //\r
1418 for (Index = 0; Index < FixedPcdGet32 (PcdPeiCoreMaxFvSupported); Index ++) {\r
1419 if (PrivateData->Fv[Index].FvPpi == OldFfs2FvPpi) {\r
1420 PrivateData->Fv[Index].FvPpi = &mPeiFfs2FvPpi;\r
1421 }\r
1422 }\r
8e0e40ed 1423} \r
1424\r