/*++ Copyright (c) 2002-2006 Intel Corporation. All rights reserved This program and the accompanying materials are licensed and made available under the terms and conditions of the BSD License which accompanies this distribution. The full text of the license may be found at http://opensource.org/licenses/bsd-license.php THE PROGRAM IS DISTRIBUTED UNDER THE BSD LICENSE ON AN "AS IS" BASIS, WITHOUT WARRANTIES OR REPRESENTATIONS OF ANY KIND, EITHER EXPRESS OR IMPLIED. Module Name: GenCapsuleHdr.c Abstract: Generate a capsule header for a file, and optionally prepend the header to a file or list of files. --*/ #define _UNICODE #include <stdio.h> #include <string.h> #include <stdlib.h> #include <ctype.h> #include <Common/UefiBaseTypes.h> #include <Common/MultiPhase.h> #include <Common/Capsule.h> #include <Common/FirmwareVolumeImageFormat.h> #include <Common/FirmwareVolumeHeader.h> #include <Common/FirmwareFileSystem.h> // for FV header GUID #include <Guid/Capsule.h> #include <Guid/FirmwareFileSystem.h> // for FV header GUID #include "CommonLib.h" #include "EfiUtilityMsgs.h" #define MAX_PATH 256 #define UTILITY_NAME "GenCapsuleHdr" #define UTILITY_MAJOR_VERSION 1 #define UTILITY_MINOR_VERSION 0 #define UNICODE_BACKSLASH L'\\' #define UNICODE_FILE_START 0xFEFF #define UNICODE_CR 0x000D #define UNICODE_LF 0x000A #define UNICODE_NULL 0x0000 #define UNICODE_SPACE L' ' #define UNICODE_SLASH L'/' #define UNICODE_DOUBLE_QUOTE L'"' #define UNICODE_A L'A' #define UNICODE_F L'F' #define UNICODE_Z L'Z' #define UNICODE_a L'a' #define UNICODE_f L'f' #define UNICODE_z L'z' #define UNICODE_0 L'0' #define UNICODE_9 L'9' #define UNICODE_TAB L'\t' #define OEM_HEADER_STRING L"OemHeader" #define AUTHOR_INFO_STRING L"AuthorInfo" #define REVISION_INFO_STRING L"RevisionInfo" #define SHORT_DESCRIPTION_STRING L"ShortDescription" #define LONG_DESCRIPTION_STRING L"LongDescription" #define EQUAL_STRING L"=" #define OPEN_BRACE_STRING L"{" #define CLOSE_BRACE_STRING L"}" #define GUID_STRING L"GUID" #define DATA_STRING L"DATA" #if (EFI_SPECIFICATION_VERSION >= 0x00020000) #define UEFI_CAPSULE_HEADER_NO_FALAGS 0 #define UEFI_CAPSULE_HEADER_RESET_FALAGS CAPSULE_FLAGS_PERSIST_ACROSS_RESET #define UEFI_CAPSULE_HEADER_ALL_FALAGS (CAPSULE_FLAGS_PERSIST_ACROSS_RESET | CAPSULE_FLAGS_POPULATE_SYSTEM_TABLE) #endif typedef wchar_t WCHAR; typedef struct _FILE_LIST { struct _FILE_LIST *Next; INT8 FileName[MAX_PATH]; } FILE_LIST; typedef struct _SIZE_LIST { struct _SIZE_LIST *Next; UINT32 Size; } SIZE_LIST; typedef struct { INT8 FileName[MAX_PATH]; WCHAR *FileBuffer; WCHAR *FileBufferPtr; UINT32 FileSize; FILE *FilePtr; BOOLEAN EndOfFile; UINT32 LineNum; } SOURCE_FILE; // // Here's all our globals. // static struct { BOOLEAN Dump; BOOLEAN Verbose; BOOLEAN JoinMode; INT8 ScriptFileName[MAX_PATH]; INT8 OutputFileName[MAX_PATH]; FILE_LIST *FileList; FILE *OutFptr; SIZE_LIST *SizeList; SIZE_LIST *LastSize; SIZE_LIST *CurrentSize; } mOptions; static EFI_GUID mEfiCapsuleHeaderGuid = EFI_CAPSULE_GUID; void CreateGuid ( EFI_GUID *Guid ); static STATUS ProcessArgs ( int Argc, char *Argv[] ); static void SkipWhiteSpace ( SOURCE_FILE *SourceFile ); static STATUS GetHexValue ( SOURCE_FILE *SourceFile, UINT32 *Value, UINT32 NumDigits ); static BOOLEAN GetSplitFileName ( INT8 *BaseFileName, INT8 *NewFileName, UINT32 SequenceNumber ); static STATUS SplitCapsule ( INT8 *CapsuleFileName ); static void Version ( VOID ); static void Usage ( VOID ); static void DumpCapsule ( VOID ); static STATUS JoinCapsule ( VOID ); static STATUS DumpCapsuleHeaderStrings ( UINT8 *SectionName, WCHAR *Buffer ); static STATUS CheckFirmwareVolumeHeader ( INT8 *FileName, INT8 *Buffer, UINT32 BufferSize ); static BOOLEAN IsToken ( SOURCE_FILE *File, WCHAR *Token ); static BOOLEAN GetNumber ( INT8 *Str, UINT32 *Value ); static STATUS ProcessScriptFile ( INT8 *ScriptFileName, FILE *OutFptr, EFI_CAPSULE_HEADER *CapsuleHeader ); static STATUS ParseCapsuleInfo ( SOURCE_FILE *SourceFile, FILE *OutFptr, WCHAR *SectionName ); static STATUS CreateCapsule ( VOID ); static STATUS ParseOemInfo ( SOURCE_FILE *SourceFile, FILE *OutFptr ); static BOOLEAN IsWhiteSpace ( WCHAR Char ); static BOOLEAN EndOfFile ( SOURCE_FILE *File ); int main ( int Argc, char *Argv[] ) /*++ Routine Description: Call the routine to process the command-line arguments, then dispatch to the appropriate function. Arguments: Standard C main() argc and argv. Returns: 0 if successful nonzero otherwise --*/ // GC_TODO: Argc - add argument and description to function comment // GC_TODO: ] - add argument and description to function comment { STATUS Status; FILE_LIST *NextFile; // // Specify our program name to the error printing routines. // SetUtilityName (UTILITY_NAME); // // Process the command-line arguments // Status = ProcessArgs (Argc, Argv); if (Status == STATUS_SUCCESS) { if (mOptions.Dump) { DumpCapsule (); } else if (mOptions.JoinMode) { JoinCapsule (); } else { CreateCapsule (); } } // // Cleanup // while (mOptions.FileList != NULL) { NextFile = mOptions.FileList->Next; free (mOptions.FileList); mOptions.FileList = NextFile; } while (mOptions.SizeList != NULL) { mOptions.CurrentSize = mOptions.SizeList->Next; free (mOptions.SizeList); mOptions.SizeList = mOptions.CurrentSize; } return GetUtilityStatus (); } static STATUS CreateCapsule ( VOID ) /*++ Routine Description: GC_TODO: Add function description Arguments: None Returns: GC_TODO: add return values --*/ { FILE *InFptr; FILE_LIST *FileList; INT8 *Buffer; UINT32 Size; EFI_CAPSULE_HEADER CapsuleHeader; UINT8 Zero; UINT8 FirstFile; UINT32 CapsuleHeaderSize; long InsertedBlockMapEntryOffset; EFI_FV_BLOCK_MAP_ENTRY InsertedBlockMapEntry; UINT64 FirmwareVolumeSize; long FileSize; EFI_FIRMWARE_VOLUME_HEADER FVHeader; Buffer = NULL; InFptr = NULL; FirmwareVolumeSize = 0; CapsuleHeaderSize = 0; InsertedBlockMapEntryOffset = 0; memset (&InsertedBlockMapEntry, 0, sizeof (EFI_FV_BLOCK_MAP_ENTRY)); memset (&FVHeader, 0, sizeof (EFI_FIRMWARE_VOLUME_HEADER)); if ((mOptions.OutFptr = fopen (mOptions.OutputFileName, "wb")) == NULL) { Error (NULL, 0, 0, mOptions.OutputFileName, "failed to open output file for writing"); return STATUS_ERROR; } memset ((char *) &CapsuleHeader, 0, sizeof (CapsuleHeader)); memcpy ((void *) &CapsuleHeader.CapsuleGuid, (void *) &mEfiCapsuleHeaderGuid, sizeof (EFI_GUID)); CapsuleHeader.HeaderSize = sizeof (EFI_CAPSULE_HEADER); CapsuleHeader.CapsuleImageSize = sizeof (EFI_CAPSULE_HEADER); if (mOptions.ScriptFileName[0] != 0) { if (ProcessScriptFile (mOptions.ScriptFileName, mOptions.OutFptr, &CapsuleHeader) != STATUS_SUCCESS) { goto Done; } } else { // // Insert a default capsule header #if (EFI_SPECIFICATION_VERSION >= 0x00020000) CapsuleHeader.HeaderSize = sizeof (EFI_CAPSULE_HEADER); CapsuleHeader.Flags = UEFI_CAPSULE_HEADER_ALL_FALAGS; #endif CapsuleHeader.OffsetToCapsuleBody = sizeof (EFI_CAPSULE_HEADER); if (fwrite ((void *) &CapsuleHeader, sizeof (CapsuleHeader), 1, mOptions.OutFptr) != 1) { Error (NULL, 0, 0, mOptions.OutputFileName, "failed to write to file"); goto Done; } } CapsuleHeaderSize = CapsuleHeader.OffsetToCapsuleBody; // // Now copy the contents of any other files specified on the command // line to the output file. Files must be FFS files, which are aligned // on 8-byte boundaries. Don't align the first file, since it's the start // of the image once the capsule header has been removed. // FileList = mOptions.FileList; FirstFile = 1; Zero = 0; while (FileList != NULL) { if ((InFptr = fopen (FileList->FileName, "rb")) == NULL) { Error (NULL, 0, 0, FileList->FileName, "failed to open file for reading"); goto Done; } // // Allocate a buffer into which we can read the file. // fseek (InFptr, 0, SEEK_END); Size = ftell (InFptr); rewind (InFptr); Buffer = (char *) malloc (Size); if (Buffer == NULL) { Error (__FILE__, __LINE__, 0, FileList->FileName, "failed to allocate buffer to read file into"); goto Done; } if (fread ((void *) Buffer, Size, 1, InFptr) != 1) { Error (NULL, 0, 0, FileList->FileName, "failed to read file contents"); goto Done; } if (Size > 0) { // // Align the write of the first bytes from the file if not the first file // if (FirstFile) { // // First file must be a firmware volume. Double-check, and then insert // an additional block map entry so we can add more files from the command line // if (CheckFirmwareVolumeHeader (FileList->FileName, Buffer, Size) != STATUS_SUCCESS) { goto Done; } // // Save a copy of the firmware volume header for later // memcpy (&FVHeader, Buffer, sizeof (EFI_FIRMWARE_VOLUME_HEADER)); FirmwareVolumeSize = FVHeader.FvLength; if (FileList->Next != NULL) { // // Copy the firmware volume header // InsertedBlockMapEntryOffset = CapsuleHeaderSize + FVHeader.HeaderLength; if (fwrite (Buffer, FVHeader.HeaderLength, 1, mOptions.OutFptr) != 1) { Error (NULL, 0, 0, mOptions.OutputFileName, "failed to write to file"); goto Done; } if (fwrite (&InsertedBlockMapEntry, sizeof (EFI_FV_BLOCK_MAP_ENTRY), 1, mOptions.OutFptr) != 1) { Error (NULL, 0, 0, mOptions.OutputFileName, "failed to write to file"); goto Done; } if (fwrite ( Buffer + FVHeader.HeaderLength, Size - FVHeader.HeaderLength, 1, mOptions.OutFptr ) != 1) { Error (NULL, 0, 0, mOptions.OutputFileName, "failed to write to file"); goto Done; } } else { // // Copy the file contents as-is // if (fwrite ((void *) Buffer, Size, 1, mOptions.OutFptr) != 1) { Error (NULL, 0, 0, mOptions.OutputFileName, "failed to write to file"); goto Done; } } } else { while ((ftell (mOptions.OutFptr) - CapsuleHeaderSize) & 0x07) { if (fwrite ((void *) &Zero, 1, 1, mOptions.OutFptr) != 1) { Error (NULL, 0, 0, mOptions.OutputFileName, "failed to write to file"); goto Done; } } if (fwrite ((void *) Buffer, Size, 1, mOptions.OutFptr) != 1) { Error (NULL, 0, 0, mOptions.OutputFileName, "failed to write to file"); goto Done; } } FirstFile = 0; } free (Buffer); Buffer = NULL; fclose (InFptr); InFptr = NULL; FileList = FileList->Next; } Done: if (Buffer != NULL) { free (Buffer); } if (InFptr != NULL) { fclose (InFptr); } // // If we inserted an additional block map entry, then fix it up. Fix up the // FV header as well to reflect our new size. // if (InsertedBlockMapEntryOffset != 0) { FileSize = ftell (mOptions.OutFptr); InsertedBlockMapEntry.NumBlocks = 1; InsertedBlockMapEntry.BlockLength = (UINT32) ((UINT64) FileSize - (UINT64) CapsuleHeaderSize - FirmwareVolumeSize - sizeof (EFI_FV_BLOCK_MAP_ENTRY)); fseek (mOptions.OutFptr, InsertedBlockMapEntryOffset, SEEK_SET); fwrite (&InsertedBlockMapEntry, sizeof (EFI_FV_BLOCK_MAP_ENTRY), 1, mOptions.OutFptr); // // Fix up the firmware volume header and write it out // fseek (mOptions.OutFptr, CapsuleHeaderSize, SEEK_SET); FVHeader.FvLength = FileSize - CapsuleHeaderSize; FVHeader.HeaderLength += sizeof (EFI_FV_BLOCK_MAP_ENTRY); fwrite (&FVHeader, sizeof (EFI_FIRMWARE_VOLUME_HEADER) - sizeof (EFI_FV_BLOCK_MAP_ENTRY), 1, mOptions.OutFptr); // // Reposition to the end of the file // } // // Close files and free the global string lists we allocated memory for // if (mOptions.OutFptr != NULL) { // // We should now know the full capsule image size. Update the header and write it again. // fseek (mOptions.OutFptr, 0, SEEK_END); Size = ftell (mOptions.OutFptr); CapsuleHeader.CapsuleImageSize = Size; fseek (mOptions.OutFptr, 0, SEEK_SET); if (fwrite ((void *) &CapsuleHeader, sizeof (CapsuleHeader), 1, mOptions.OutFptr) != 1) { Error (NULL, 0, 0, mOptions.OutputFileName, "failed to write to file"); } fseek (mOptions.OutFptr, 0, SEEK_END); fclose (mOptions.OutFptr); mOptions.OutFptr = NULL; } // // If they are doing split capsule output, then split it up now. // if ((mOptions.Dump == 0) && (GetUtilityStatus () == STATUS_SUCCESS) && (mOptions.SizeList != NULL)) { SplitCapsule (mOptions.OutputFileName); } return STATUS_SUCCESS; } static STATUS ProcessScriptFile ( INT8 *ScriptFileName, FILE *OutFptr, EFI_CAPSULE_HEADER *CapsuleHeader ) /*++ Routine Description: Parse a capsule header script file. Arguments: ScriptFileName - name of script file to parse OutFptr - output to dump binary data CapsuleHeader - capsule header to update with size info of parsed fields in the script file Returns: STATUS_SUCCESS - if all went well --*/ { #if 0 STATUS Status; SOURCE_FILE SourceFile; WCHAR *WScriptFileName; BOOLEAN InComment; if (fwrite (CapsuleHeader, sizeof (EFI_CAPSULE_HEADER), 1, OutFptr) != 1) { Error (NULL, 0, 0, "failed to write capsule header to output file", NULL); return STATUS_ERROR; } memset (&SourceFile, 0, sizeof (SOURCE_FILE)); strcpy (SourceFile.FileName, ScriptFileName); Status = STATUS_ERROR; // // Open the input unicode script file and read it into a buffer // WScriptFileName = (WCHAR *) malloc ((strlen (ScriptFileName) + 1) * sizeof (WCHAR)); if (WScriptFileName == NULL) { Error (__FILE__, __LINE__, 0, "failed to allocate memory", NULL); return STATUS_ERROR; } swprintf (WScriptFileName, L"%S", ScriptFileName); if ((SourceFile.FilePtr = _wfopen (WScriptFileName, L"r")) == NULL) { free (WScriptFileName); Error (NULL, 0, 0, ScriptFileName, "failed to open script file for reading"); goto Done; } free (WScriptFileName); fseek (SourceFile.FilePtr, 0, SEEK_END); SourceFile.FileSize = ftell (SourceFile.FilePtr); rewind (SourceFile.FilePtr); SourceFile.FileBuffer = (WCHAR *) malloc (SourceFile.FileSize + sizeof (WCHAR)); if (SourceFile.FileBuffer == NULL) { Error (__FILE__, __LINE__, 0, ScriptFileName, "failed to allocate memory to read in file contents"); goto Done; } if (fread (SourceFile.FileBuffer, SourceFile.FileSize, 1, SourceFile.FilePtr) != 1) { Error (NULL, 0, 0, ScriptFileName, "failed to read file contents"); goto Done; } SourceFile.FileBufferPtr = SourceFile.FileBuffer; SourceFile.LineNum = 1; if (SourceFile.FileBuffer[0] != UNICODE_FILE_START) { Error (ScriptFileName, 1, 0, "file does not appear to be a unicode file", NULL); goto Done; } SourceFile.FileBufferPtr++; SourceFile.FileBuffer[SourceFile.FileSize / sizeof (WCHAR)] = 0; // // Walk the source file buffer and replace all carriage returns with 0 so // we can print from the file contents on parse errors. // InComment = 0; while (!EndOfFile (&SourceFile)) { if (SourceFile.FileBufferPtr[0] == UNICODE_CR) { SourceFile.FileBufferPtr[0] = 0; InComment = 0; } else if (SourceFile.FileBufferPtr[0] == UNICODE_LF) { InComment = 0; } else if (InComment) { SourceFile.FileBufferPtr[0] = UNICODE_SPACE; } else if ((SourceFile.FileBufferPtr[0] == UNICODE_SLASH) && (SourceFile.FileBufferPtr[1] == UNICODE_SLASH)) { InComment = 1; SourceFile.FileBufferPtr[0] = UNICODE_SPACE; } SourceFile.FileBufferPtr++; } // // Reposition to the start of the file, but skip over the unicode file start // SourceFile.FileBufferPtr = SourceFile.FileBuffer; SourceFile.FileBufferPtr++; SourceFile.EndOfFile = 0; CapsuleHeader->OffsetToOemDefinedHeader = ftell (OutFptr); // // Parse the OEM bytes // if (ParseOemInfo (&SourceFile, OutFptr) != STATUS_SUCCESS) { goto Done; } // // Parse the author information // CapsuleHeader->OffsetToAuthorInformation = ftell (OutFptr); if (ParseCapsuleInfo (&SourceFile, OutFptr, AUTHOR_INFO_STRING) != STATUS_SUCCESS) { goto Done; } // // Parse the revision information // CapsuleHeader->OffsetToRevisionInformation = ftell (OutFptr); if (ParseCapsuleInfo (&SourceFile, OutFptr, REVISION_INFO_STRING) != STATUS_SUCCESS) { goto Done; } // // Parse the short description // CapsuleHeader->OffsetToShortDescription = ftell (OutFptr); if (ParseCapsuleInfo (&SourceFile, OutFptr, SHORT_DESCRIPTION_STRING) != STATUS_SUCCESS) { goto Done; } // // Parse the long description // CapsuleHeader->OffsetToLongDescription = ftell (OutFptr); if (ParseCapsuleInfo (&SourceFile, OutFptr, LONG_DESCRIPTION_STRING) != STATUS_SUCCESS) { goto Done; } // // Better be end of contents // SkipWhiteSpace (&SourceFile); if (!EndOfFile (&SourceFile)) { Error (ScriptFileName, SourceFile.LineNum, 0, NULL, "expected end-of-file, not %.20S", SourceFile.FileBufferPtr); goto Done; } CapsuleHeader->OffsetToCapsuleBody = ftell (OutFptr); rewind (OutFptr); fwrite (CapsuleHeader, sizeof (EFI_CAPSULE_HEADER), 1, OutFptr); fseek (OutFptr, 0, SEEK_END); Status = STATUS_SUCCESS; Done: if (SourceFile.FilePtr != NULL) { fclose (SourceFile.FilePtr); } if (SourceFile.FileBuffer != NULL) { free (SourceFile.FileBuffer); } return Status; #endif return STATUS_SUCCESS; } // // Parse the OEM data of format: // OemInfo { // GUID = 12345676-1234-1234-123456789ABC // DATA = 0x01, 0x02, 0x03... // } // static STATUS ParseOemInfo ( SOURCE_FILE *SourceFile, FILE *OutFptr ) /*++ Routine Description: GC_TODO: Add function description Arguments: SourceFile - GC_TODO: add argument description OutFptr - GC_TODO: add argument description Returns: GC_TODO: add return values --*/ { long OemHeaderOffset; UINT32 Data; EFI_CAPSULE_OEM_HEADER OemHeader; STATUS Status; UINT32 DigitCount; WCHAR *SaveFilePos; UINT8 ByteData; Status = STATUS_ERROR; memset (&OemHeader, 0, sizeof (EFI_CAPSULE_OEM_HEADER)); OemHeaderOffset = ftell (OutFptr); OemHeader.HeaderSize = sizeof (EFI_CAPSULE_OEM_HEADER); if (fwrite (&OemHeader, sizeof (EFI_CAPSULE_OEM_HEADER), 1, OutFptr) != 1) { Error (NULL, 0, 0, "failed to write OEM header to output file", NULL); goto Done; } if (!IsToken (SourceFile, OEM_HEADER_STRING)) { Error ( SourceFile->FileName, SourceFile->LineNum, 0, NULL, "expected %S, not %.20S", OEM_HEADER_STRING, SourceFile->FileBufferPtr ); goto Done; } if (!IsToken (SourceFile, EQUAL_STRING)) { Error ( SourceFile->FileName, SourceFile->LineNum, 0, NULL, "expected %S, not %.20S", EQUAL_STRING, SourceFile->FileBufferPtr ); goto Done; } if (!IsToken (SourceFile, OPEN_BRACE_STRING)) { Error ( SourceFile->FileName, SourceFile->LineNum, 0, NULL, "expected %S, not %.20S", OPEN_BRACE_STRING, SourceFile->FileBufferPtr ); goto Done; } // // Look for: GUID = xxxxxxx-xxxx-xxxx-xxxxxxxxxxxxx // if (!IsToken (SourceFile, GUID_STRING)) { Error ( SourceFile->FileName, SourceFile->LineNum, 0, NULL, "expected %S, not %.20S", GUID_STRING, SourceFile->FileBufferPtr ); goto Done; } if (!IsToken (SourceFile, EQUAL_STRING)) { Error ( SourceFile->FileName, SourceFile->LineNum, 0, NULL, "expected %S, not %.20S", EQUAL_STRING, SourceFile->FileBufferPtr ); goto Done; } // // Parse the xxxxxxxx-xxxx-xxxx-xxxx portion of the GUID // SkipWhiteSpace (SourceFile); if (GetHexValue (SourceFile, &Data, 8) != STATUS_SUCCESS) { Error (SourceFile->FileName, SourceFile->LineNum, 0, "invalid GUID", NULL); goto Done; } OemHeader.OemGuid.Data1 = Data; if (!IsToken (SourceFile, L"-")) { Error ( SourceFile->FileName, SourceFile->LineNum, 0, NULL, "expected dash in GUID, not %S", SourceFile->FileBufferPtr ); goto Done; } // // Get 3 word values // for (DigitCount = 0; DigitCount < 3; DigitCount++) { if (GetHexValue (SourceFile, &Data, 4) != STATUS_SUCCESS) { Error (SourceFile->FileName, SourceFile->LineNum, 0, "invalid GUID", NULL); goto Done; } switch (DigitCount) { case 0: OemHeader.OemGuid.Data2 = (UINT16) Data; break; case 1: OemHeader.OemGuid.Data3 = (UINT16) Data; break; case 2: OemHeader.OemGuid.Data4[1] = (UINT8) Data; OemHeader.OemGuid.Data4[0] = (UINT8) (Data >> 8); break; } if (!IsToken (SourceFile, L"-")) { Error ( SourceFile->FileName, SourceFile->LineNum, 0, NULL, "expected dash in GUID, not %S", SourceFile->FileBufferPtr ); goto Done; } } // // Pick up the last 6 bytes of the GUID // SaveFilePos = SourceFile->FileBufferPtr; for (DigitCount = 0; DigitCount < 6; DigitCount++) { if (GetHexValue (SourceFile, &Data, 2) != STATUS_SUCCESS) { Error (SourceFile->FileName, SourceFile->LineNum, 0, "invalid GUID", NULL); goto Done; } OemHeader.OemGuid.Data4[DigitCount + 2] = (UINT8) Data; } // // Now read raw OEM data bytes. May or may not be present. // DATA = 0x01, 0x02, 0x02... // if (IsToken (SourceFile, CLOSE_BRACE_STRING)) { Status = STATUS_SUCCESS; goto Done; } if (!IsToken (SourceFile, DATA_STRING)) { Error ( SourceFile->FileName, SourceFile->LineNum, 0, NULL, "expected %S, not %.20S", DATA_STRING, SourceFile->FileBufferPtr ); goto Done; } if (!IsToken (SourceFile, EQUAL_STRING)) { Error ( SourceFile->FileName, SourceFile->LineNum, 0, NULL, "expected %S, not %.20S", EQUAL_STRING, SourceFile->FileBufferPtr ); goto Done; } while (!EndOfFile (SourceFile)) { if (IsToken (SourceFile, CLOSE_BRACE_STRING)) { Status = STATUS_SUCCESS; goto Done; } if (IsToken (SourceFile, L"0x")) { if (swscanf (SourceFile->FileBufferPtr, L"%x", &Data) != 1) { Error ( SourceFile->FileName, SourceFile->LineNum, 0, NULL, "expected hex byte value, not %.20S", SourceFile->FileBufferPtr ); goto Done; } if (Data &~0xFF) { Error ( SourceFile->FileName, SourceFile->LineNum, 0, NULL, "expected byte hex byte value at %.20S", SourceFile->FileBufferPtr ); goto Done; } // // Skip over the hex digits, then write the data // while (iswxdigit (SourceFile->FileBufferPtr[0])) { SourceFile->FileBufferPtr++; } ByteData = (UINT8) Data; if (fwrite (&ByteData, 1, 1, OutFptr) != 1) { Error (NULL, 0, 0, "failed to write OEM data to output file", NULL); goto Done; } OemHeader.HeaderSize++; // // Optional comma // IsToken (SourceFile, L","); } else { Error ( SourceFile->FileName, SourceFile->LineNum, 0, NULL, "expected hex OEM data, not %.20S", SourceFile->FileBufferPtr ); goto Done; } } if (EndOfFile (SourceFile)) { Error ( SourceFile->FileName, SourceFile->LineNum, 0, NULL, "expected %S close to OEM header data", CLOSE_BRACE_STRING ); goto Done; } Status = STATUS_SUCCESS; Done: // // re-write the oem header if no errors // if (Status == STATUS_SUCCESS) { fseek (OutFptr, OemHeaderOffset, SEEK_SET); if (fwrite (&OemHeader, sizeof (EFI_CAPSULE_OEM_HEADER), 1, OutFptr) != 1) { Error (NULL, 0, 0, "failed to write OEM header to output file", NULL); goto Done; } fseek (OutFptr, 0, SEEK_END); } return Status; } static STATUS ParseCapsuleInfo ( SOURCE_FILE *SourceFile, FILE *OutFptr, WCHAR *SectionName ) // GC_TODO: function comment should start with '/*++' // // GC_TODO: function comment is missing 'Routine Description:' // GC_TODO: function comment is missing 'Arguments:' // GC_TODO: function comment is missing 'Returns:' // GC_TODO: SourceFile - add argument and description to function comment // GC_TODO: OutFptr - add argument and description to function comment // GC_TODO: SectionName - add argument and description to function comment // Parse: eng "string " "parts" // spa "string " "parts" // Write out: "eng string parts\0spa string parts\0\0 // { STATUS Status; int StringCount; WCHAR Zero; WCHAR Spacebar; Status = STATUS_ERROR; Zero = 0; Spacebar = UNICODE_SPACE; if (!IsToken (SourceFile, SectionName)) { Error ( SourceFile->FileName, SourceFile->LineNum, 0, NULL, "expected %S, not %.20S", SectionName, SourceFile->FileBufferPtr ); goto Done; } if (!IsToken (SourceFile, EQUAL_STRING)) { Error ( SourceFile->FileName, SourceFile->LineNum, 0, NULL, "expected %S, not %.20S", EQUAL_STRING, SourceFile->FileBufferPtr ); goto Done; } if (!IsToken (SourceFile, OPEN_BRACE_STRING)) { Error ( SourceFile->FileName, SourceFile->LineNum, 0, NULL, "expected %S, not %.20S", OPEN_BRACE_STRING, SourceFile->FileBufferPtr ); goto Done; } while (!EndOfFile (SourceFile)) { if (IsToken (SourceFile, CLOSE_BRACE_STRING)) { break; } // // Look for language identifier (3 lowercase chars) // if ((SourceFile->FileBufferPtr[0] >= UNICODE_a) && (SourceFile->FileBufferPtr[0] <= UNICODE_z) && (SourceFile->FileBufferPtr[1] >= UNICODE_a) && (SourceFile->FileBufferPtr[1] <= UNICODE_z) && (SourceFile->FileBufferPtr[2] >= UNICODE_a) && (SourceFile->FileBufferPtr[2] <= UNICODE_z) && IsWhiteSpace (SourceFile->FileBufferPtr[3]) ) { // // Write the 3 chars followed by a spacebar, and then look for opening quote // fwrite (SourceFile->FileBufferPtr, sizeof (WCHAR), 1, OutFptr); SourceFile->FileBufferPtr++; fwrite (SourceFile->FileBufferPtr, sizeof (WCHAR), 1, OutFptr); SourceFile->FileBufferPtr++; fwrite (SourceFile->FileBufferPtr, sizeof (WCHAR), 1, OutFptr); SourceFile->FileBufferPtr++; fwrite (&Spacebar, sizeof (WCHAR), 1, OutFptr); StringCount = 0; while (IsToken (SourceFile, L"\"")) { StringCount++; while (!EndOfFile (SourceFile)) { if (SourceFile->FileBufferPtr[0] == UNICODE_DOUBLE_QUOTE) { SourceFile->FileBufferPtr++; break; } else if ((SourceFile->FileBufferPtr[0] == UNICODE_LF) || (SourceFile->FileBufferPtr[0] == 0)) { Error (SourceFile->FileName, SourceFile->LineNum, 0, "missing closing quote on string", NULL); goto Done; } else { fwrite (SourceFile->FileBufferPtr, sizeof (WCHAR), 1, OutFptr); SourceFile->FileBufferPtr++; } } } if (StringCount == 0) { Error ( SourceFile->FileName, SourceFile->LineNum, 0, NULL, "expected quoted string, not %.20S", SourceFile->FileBufferPtr ); goto Done; } // // This string's null terminator // fwrite (&Zero, sizeof (WCHAR), 1, OutFptr); } else { Error ( SourceFile->FileName, SourceFile->LineNum, 0, NULL, "expected valid language identifer, not %.20S", SourceFile->FileBufferPtr ); goto Done; } } // // Double null terminator // fwrite (&Zero, sizeof (WCHAR), 1, OutFptr); Status = STATUS_SUCCESS; Done: return Status; } static STATUS SplitCapsule ( INT8 *CapsuleFileName ) /*++ Routine Description: We've created an entire capsule image. Now split it up into the size pieces they requested. Arguments: CapsuleFileName - name of an existing capsule file on disk Returns: STATUS_SUCCESS - if no problems Notes: This implementation reads in the entire capsule image from disk, then overwrites the original file with the first in the series. --*/ { #if 0 EFI_CAPSULE_HEADER *CapHdr; EFI_CAPSULE_HEADER Hdr; FILE *CapFptr; FILE *OutFptr; UINT32 SizeLeft; UINT32 CurrentSize; UINT32 DataSize; UINT32 SequenceNumber; INT8 *Buffer; INT8 FileName[MAX_PATH]; STATUS Status; UINT32 FileSize; // // Figure out the total size, then rewind the input file and // read the entire thing in // if ((CapFptr = fopen (CapsuleFileName, "rb")) == NULL) { Error (NULL, 0, 0, CapsuleFileName, "failed to open capsule image for reading"); return STATUS_ERROR; } OutFptr = NULL; Status = STATUS_SUCCESS; fseek (CapFptr, 0, SEEK_END); SizeLeft = ftell (CapFptr); fseek (CapFptr, 0, SEEK_SET); CapHdr = (EFI_CAPSULE_HEADER *) malloc (SizeLeft); if (CapHdr == NULL) { Error (NULL, 0, 0, "memory allocation failure", NULL); goto FailDone; } if (fread (CapHdr, SizeLeft, 1, CapFptr) != 1) { Error (NULL, 0, 0, "failed to read capsule contents", "split failed"); goto FailDone; } fclose (CapFptr); CapFptr = NULL; // // Get a GUID to fill in the InstanceId GUID in the header // CreateGuid (&CapHdr->InstanceId); SequenceNumber = 0; // // If the split size is larger than the original capsule image, then // we're done. // if (mOptions.SizeList->Size >= SizeLeft) { mOptions.SizeList->Size = SizeLeft; goto Done; } // // First size has to be big enough for the original header // if (mOptions.SizeList->Size < CapHdr->OffsetToCapsuleBody) { Error (NULL, 0, 0, "first split size is insufficient for the original capsule header", NULL); goto FailDone; } // // Initialize the header we'll use on all but the first part // memset (&Hdr, 0, sizeof (Hdr)); Hdr.CapsuleGuid = CapHdr->CapsuleGuid; Hdr.HeaderSize = sizeof (Hdr); Hdr.Flags = CapHdr->Flags; Hdr.InstanceId = CapHdr->InstanceId; Hdr.CapsuleImageSize = CapHdr->CapsuleImageSize; Hdr.OffsetToCapsuleBody = Hdr.HeaderSize; Hdr.SequenceNumber = 1; // // printf ("Created %s - 0x%X bytes\n", CapsuleFileName, mOptions.SizeList->Size); // Buffer = (UINT8 *) CapHdr; // // Walk the list of sizes and write out a capsule header, and // then the raw capsule data. // // SizeLeft -= mOptions.SizeList->Size; // mOptions.CurrentSize = mOptions.SizeList; while (SizeLeft) { CurrentSize = mOptions.CurrentSize->Size; GetSplitFileName (mOptions.OutputFileName, FileName, SequenceNumber); if ((OutFptr = fopen (FileName, "wb")) == NULL) { Error (NULL, 0, 0, FileName, "failed to open split file for writing"); goto FailDone; } if (Buffer == (UINT8 *) CapHdr) { // // First part -- write out original header and data // if (fwrite (Buffer, CurrentSize, 1, OutFptr) != 1) { Error (NULL, 0, 0, FileName, "failed to write to split image file"); goto FailDone; } SizeLeft -= CurrentSize; Buffer += CurrentSize; DataSize = CurrentSize; FileSize = CurrentSize; } else { // // Not the first part. Write the default header, and then the raw bytes from the // original image. // if (CurrentSize <= sizeof (Hdr)) { Error (NULL, 0, 0, "split size too small for capsule header + data", "0x%X", CurrentSize); goto FailDone; } DataSize = CurrentSize - sizeof (Hdr); if (DataSize > SizeLeft) { DataSize = SizeLeft; } if (fwrite (&Hdr, sizeof (Hdr), 1, OutFptr) != 1) { Error (NULL, 0, 0, FileName, "failed to write capsule header to output file"); fclose (OutFptr); goto FailDone; } if (fwrite (Buffer, DataSize, 1, OutFptr) != 1) { Error (NULL, 0, 0, FileName, "failed to write capsule data to output file"); fclose (OutFptr); goto FailDone; } Hdr.SequenceNumber++; Buffer += DataSize; SizeLeft -= DataSize; FileSize = DataSize + sizeof (Hdr); } // // Next size in list if there is one // if (mOptions.CurrentSize->Next != NULL) { mOptions.CurrentSize = mOptions.CurrentSize->Next; } SequenceNumber++; fclose (OutFptr); OutFptr = NULL; printf ("Created %s - 0x%X bytes (0x%X bytes of data)\n", FileName, FileSize, DataSize); } goto Done; FailDone: Status = STATUS_ERROR; Done: if (CapHdr != NULL) { free (CapHdr); } if (CapFptr != NULL) { fclose (CapFptr); } if (OutFptr != NULL) { fclose (OutFptr); } return Status; #endif return STATUS_SUCCESS; } static BOOLEAN GetSplitFileName ( INT8 *BaseFileName, INT8 *NewFileName, UINT32 SequenceNumber ) /*++ Routine Description: GC_TODO: Add function description Arguments: BaseFileName - GC_TODO: add argument description NewFileName - GC_TODO: add argument description SequenceNumber - GC_TODO: add argument description Returns: GC_TODO: add return values --*/ { /*++ Routine Description: Given an initial split capsule file name and a sequence number, create an appropriate file name for this split of a capsule image. Arguments: BaseFileName - name of of the first split file in the series NewFileName - output name of the split file SequenceNumber - 0-based sequence number of split images Returns: TRUE - name created successfully FALSE - otherwise --*/ INT8 *Ptr; INT8 *Part2Start; UINT32 Digits; UINT32 Len; UINT32 BaseOffset; // // Work back from the end of the file name and see if there is a number somewhere // for (Ptr = BaseFileName + strlen (BaseFileName) - 1; (Ptr > BaseFileName) && !isdigit (*Ptr); Ptr--) ; if ((Ptr == BaseFileName) && (!isdigit (*Ptr))) { // // Found no number, so just add it to the end // sprintf (NewFileName, "%s%d", BaseFileName, SequenceNumber); return TRUE; } else { // // Found a number. Look back to find the first digit. // Part2Start = Ptr + 1; for (Digits = 1; isdigit (*Ptr) && (Ptr > BaseFileName); Ptr--, Digits++) ; if (!isdigit (*Ptr)) { Ptr++; Digits--; } BaseOffset = atoi (Ptr); SequenceNumber = SequenceNumber + BaseOffset; if (Digits > 1) { // // Copy the first part of the original file name to the new filename // This is the path for filenames with format path\name001.cap // Len = (UINT32) Ptr - (UINT32) BaseFileName; strncpy (NewFileName, BaseFileName, Len); sprintf (NewFileName + Len, "%0*d", Digits, SequenceNumber); strcat (NewFileName, Part2Start); return TRUE; } else { // // Only one digit found. This is the path for filenames with // format path\name1.cap // Len = (UINT32) Ptr - (UINT32) BaseFileName + 1; strncpy (NewFileName, BaseFileName, Len); sprintf (NewFileName + Len - 1, "%d", SequenceNumber); strcat (NewFileName, Part2Start); return TRUE; } } } static BOOLEAN IsWhiteSpace ( WCHAR Char ) /*++ Routine Description: GC_TODO: Add function description Arguments: Char - GC_TODO: add argument description Returns: GC_TODO: add return values --*/ { switch (Char) { case UNICODE_SPACE: case UNICODE_TAB: case UNICODE_NULL: case UNICODE_CR: case UNICODE_LF: return TRUE; default: return FALSE; } } static BOOLEAN IsToken ( SOURCE_FILE *File, WCHAR *Token ) /*++ Routine Description: GC_TODO: Add function description Arguments: File - GC_TODO: add argument description Token - GC_TODO: add argument description Returns: GC_TODO: add return values --*/ { SkipWhiteSpace (File); if (EndOfFile (File)) { return FALSE; } if (wcsncmp (Token, File->FileBufferPtr, wcslen (Token)) == 0) { File->FileBufferPtr += wcslen (Token); return TRUE; } return FALSE; } static STATUS CheckFirmwareVolumeHeader ( INT8 *FileName, INT8 *Buffer, UINT32 BufferSize ) /*++ Routine Description: GC_TODO: Add function description Arguments: FileName - GC_TODO: add argument description Buffer - GC_TODO: add argument description BufferSize - GC_TODO: add argument description Returns: GC_TODO: add return values --*/ { EFI_FIRMWARE_VOLUME_HEADER *Hdr; EFI_GUID FVHeaderGuid = EFI_FIRMWARE_FILE_SYSTEM_GUID; Hdr = (EFI_FIRMWARE_VOLUME_HEADER *) Buffer; if (Hdr->Signature != EFI_FVH_SIGNATURE) { Error (NULL, 0, 0, FileName, "file does not appear to be a firmware volume (bad signature)"); return STATUS_ERROR; } if (Hdr->Revision != EFI_FVH_REVISION) { Error (NULL, 0, 0, FileName, "unsupported firmware volume header version"); return STATUS_ERROR; } if (Hdr->FvLength > BufferSize) { Error (NULL, 0, 0, FileName, "malformed firmware volume -- FvLength > file size"); return STATUS_ERROR; } if (memcmp (&Hdr->FileSystemGuid, &FVHeaderGuid, sizeof (EFI_GUID)) != 0) { Error (NULL, 0, 0, FileName, "invalid FFS GUID in firmware volume header"); return STATUS_ERROR; } return STATUS_SUCCESS; } static void DumpCapsule ( VOID ) /*++ Routine Description: GC_TODO: Add function description Arguments: None Returns: GC_TODO: add return values --*/ { #if 0 FILE *InFptr; FILE_LIST *FileList; EFI_CAPSULE_HEADER CapsuleHeader; EFI_FIRMWARE_VOLUME_HEADER FVHeader; EFI_CAPSULE_OEM_HEADER *OemHeader; UINT8 *BPtr; UINT32 FileSize; UINT32 CapsuleHeaderDataSize; UINT8 ByteCount; UINT8 *CapsuleHeaderData; BOOLEAN SplitImage; InFptr = NULL; CapsuleHeaderData = NULL; FileList = mOptions.FileList; while (FileList != NULL) { if ((InFptr = fopen (FileList->FileName, "rb")) == NULL) { Error (NULL, 0, 0, FileList->FileName, "failed to open file for reading"); goto Done; } if (fread (&CapsuleHeader, sizeof (EFI_CAPSULE_HEADER), 1, InFptr) != 1) { Error (NULL, 0, 0, FileList->FileName, "failed to read capsule header"); goto Done; } fseek (InFptr, 0, SEEK_END); FileSize = ftell (InFptr); if (CapsuleHeader.CapsuleImageSize > FileSize) { SplitImage = TRUE; } else { SplitImage = FALSE; } printf ( "Capsule %s Size=0x%X CargoSize=0x%X\n", FileList->FileName, FileSize, FileSize - CapsuleHeader.OffsetToCapsuleBody ); printf ( " GUID %08X-%04X-%04X-%02X%02X-%02X%02X%02X%02X%02X%02X", CapsuleHeader.CapsuleGuid.Data1, (UINT32) CapsuleHeader.CapsuleGuid.Data2, (UINT32) CapsuleHeader.CapsuleGuid.Data3, (UINT32) CapsuleHeader.CapsuleGuid.Data4[0], (UINT32) CapsuleHeader.CapsuleGuid.Data4[1], (UINT32) CapsuleHeader.CapsuleGuid.Data4[2], (UINT32) CapsuleHeader.CapsuleGuid.Data4[3], (UINT32) CapsuleHeader.CapsuleGuid.Data4[4], (UINT32) CapsuleHeader.CapsuleGuid.Data4[5], (UINT32) CapsuleHeader.CapsuleGuid.Data4[6], (UINT32) CapsuleHeader.CapsuleGuid.Data4[7] ); if (memcmp (&CapsuleHeader.CapsuleGuid, &mEfiCapsuleHeaderGuid, sizeof (EFI_GUID)) != 0) { printf (" INVALID GUID"); } printf ("\n"); printf (" Header size 0x%08X\n", CapsuleHeader.HeaderSize); printf (" Flags 0x%08X\n", CapsuleHeader.Flags); if (!SplitImage) { printf (" Capsule image size 0x%08X\n", CapsuleHeader.CapsuleImageSize); } else { printf (" Capsule image size 0x%08X (split)\n", CapsuleHeader.CapsuleImageSize); } printf (" Sequence number %d\n", CapsuleHeader.SequenceNumber); printf ( " InstanceId %08X-%04X-%04X-%02X%02X-%02X%02X%02X%02X%02X%02X\n", CapsuleHeader.InstanceId.Data1, (UINT32) CapsuleHeader.InstanceId.Data2, (UINT32) CapsuleHeader.InstanceId.Data3, (UINT32) CapsuleHeader.InstanceId.Data4[0], (UINT32) CapsuleHeader.InstanceId.Data4[1], (UINT32) CapsuleHeader.InstanceId.Data4[2], (UINT32) CapsuleHeader.InstanceId.Data4[3], (UINT32) CapsuleHeader.InstanceId.Data4[4], (UINT32) CapsuleHeader.InstanceId.Data4[5], (UINT32) CapsuleHeader.InstanceId.Data4[6], (UINT32) CapsuleHeader.InstanceId.Data4[7] ); printf (" Offset to capsule 0x%X\n", CapsuleHeader.OffsetToCapsuleBody); // // Dump header data if there // CapsuleHeaderDataSize = CapsuleHeader.OffsetToCapsuleBody - CapsuleHeader.HeaderSize; if (CapsuleHeaderDataSize != 0) { CapsuleHeaderData = (UINT8 *) malloc (CapsuleHeaderDataSize); if (CapsuleHeaderData == NULL) { Error ( NULL, 0, 0, "failed to allocate memory to read in capsule header data", "0x%X bytes", CapsuleHeaderDataSize ); goto Done; } fseek (InFptr, CapsuleHeader.HeaderSize, SEEK_SET); if (fread (CapsuleHeaderData, CapsuleHeaderDataSize, 1, InFptr) != 1) { Error ( NULL, 0, 0, "failed to read capsule header data contents from file", "0x%X bytes", CapsuleHeaderDataSize ); goto Done; } // // ************************************************************************ // // OEM HEADER // // ************************************************************************ // if (CapsuleHeader.OffsetToOemDefinedHeader != 0) { OemHeader = (EFI_CAPSULE_OEM_HEADER *) (CapsuleHeaderData + CapsuleHeader.OffsetToOemDefinedHeader - CapsuleHeader.HeaderSize); printf (" OEM Header\n"); printf ( " GUID %08X-%04X-%04X-%02X%02X-%02X%02X%02X%02X%02X%02X\n", OemHeader->OemGuid.Data1, (UINT32) OemHeader->OemGuid.Data2, (UINT32) OemHeader->OemGuid.Data3, (UINT32) OemHeader->OemGuid.Data4[0], (UINT32) OemHeader->OemGuid.Data4[1], (UINT32) OemHeader->OemGuid.Data4[2], (UINT32) OemHeader->OemGuid.Data4[3], (UINT32) OemHeader->OemGuid.Data4[4], (UINT32) OemHeader->OemGuid.Data4[5], (UINT32) OemHeader->OemGuid.Data4[6], (UINT32) OemHeader->OemGuid.Data4[7] ); printf (" Header size: 0x%X\n", OemHeader->HeaderSize); printf (" OEM data"); BPtr = (UINT8 *) (OemHeader + 1); for (ByteCount = 0; ByteCount < OemHeader->HeaderSize - sizeof (EFI_CAPSULE_OEM_HEADER); ByteCount++) { if ((ByteCount & 0x7) == 0) { printf ("\n "); } printf ("%02X ", (UINT32) *BPtr); BPtr++; } printf ("\n"); } // // ************************************************************************ // // Author, revision, short description, and long description information // // ************************************************************************ // if (CapsuleHeader.OffsetToAuthorInformation != 0) { if (DumpCapsuleHeaderStrings ( "Author information", (WCHAR *) (CapsuleHeaderData + CapsuleHeader.OffsetToAuthorInformation - CapsuleHeader.HeaderSize) ) != STATUS_SUCCESS) { goto Done; } } if (CapsuleHeader.OffsetToRevisionInformation != 0) { if (DumpCapsuleHeaderStrings ( "Revision information", (WCHAR *) (CapsuleHeaderData + CapsuleHeader.OffsetToRevisionInformation - CapsuleHeader.HeaderSize) ) != STATUS_SUCCESS) { goto Done; } } if (CapsuleHeader.OffsetToShortDescription != 0) { if (DumpCapsuleHeaderStrings ( "Short description", (WCHAR *) (CapsuleHeaderData + CapsuleHeader.OffsetToShortDescription - CapsuleHeader.HeaderSize) ) != STATUS_SUCCESS) { goto Done; } } if (CapsuleHeader.OffsetToLongDescription != 0) { if (DumpCapsuleHeaderStrings ( "Long description", (WCHAR *) (CapsuleHeaderData + CapsuleHeader.OffsetToLongDescription - CapsuleHeader.HeaderSize) ) != STATUS_SUCCESS) { goto Done; } } } // // If it's not a split image, or it is a split image and this is the first in the series, then // dump the cargo volume. // if ((!SplitImage) || (CapsuleHeader.SequenceNumber == 0)) { printf (" Cargo FV dump\n"); fseek (InFptr, CapsuleHeader.OffsetToCapsuleBody, SEEK_SET); if (fread (&FVHeader, sizeof (EFI_FIRMWARE_VOLUME_HEADER), 1, InFptr) != 1) { Error (NULL, 0, 0, FileList->FileName, "failed to read cargo FV header"); goto Done; } printf (" FV length 0x%X", FVHeader.FvLength); if (FileSize - CapsuleHeader.OffsetToCapsuleBody != FVHeader.FvLength) { if (!SplitImage) { printf (" ERROR: expected 0x%X to jive with file size on disk", FileSize - CapsuleHeader.OffsetToCapsuleBody); } } printf ("\n"); printf (" Signature 0x%X ", FVHeader.Signature); if (FVHeader.Signature == EFI_FVH_SIGNATURE) { printf ("_FVH\n"); } else { printf ("INVALID\n"); } printf (" FV header length 0x%X\n", (UINT32) FVHeader.HeaderLength); printf (" Revision 0x%X\n", (UINT32) FVHeader.Revision); printf ("\n"); } FileList = FileList->Next; } Done: if (InFptr != NULL) { fclose (InFptr); } if (CapsuleHeaderData != NULL) { free (CapsuleHeaderData); } #endif } static STATUS JoinCapsule ( VOID ) /*++ Routine Description: Join split capsule images into a single image. This is the support function for the -j command-line option. Arguments: None. Returns: STATUS_SUCCESS - no problems encountered --*/ { #if 0 UINT32 Size; FILE *InFptr; FILE *OutFptr; INT8 *Buffer; FILE_LIST *FileList; STATUS Status; EFI_CAPSULE_HEADER CapHdr; EFI_CAPSULE_HEADER *CapHdrPtr; UINT32 SizeLeft; UINT32 SequenceNumber; // // Must have at least two files for join mode // if ((mOptions.FileList == NULL) || (mOptions.FileList->Next == NULL)) { Error (NULL, 0, 0, "must specify at least two file names to join", NULL); return STATUS_ERROR; } // // Open the output file // if ((OutFptr = fopen (mOptions.OutputFileName, "wb")) == NULL) { Error (NULL, 0, 0, mOptions.OutputFileName, "failed to open output file for writing"); return STATUS_ERROR; } FileList = mOptions.FileList; Buffer = NULL; SequenceNumber = 0; InFptr = NULL; SizeLeft = 0; while (FileList != NULL) { if ((InFptr = fopen (FileList->FileName, "rb")) == NULL) { Error (NULL, 0, 0, FileList->FileName, "failed to open file for reading"); goto FailDone; } // // Allocate a buffer into which we can read the file. // fseek (InFptr, 0, SEEK_END); Size = ftell (InFptr); rewind (InFptr); Buffer = (char *) malloc (Size); if (Buffer == NULL) { Error (__FILE__, __LINE__, 0, FileList->FileName, "failed to allocate buffer to read file into"); goto FailDone; } CapHdrPtr = (EFI_CAPSULE_HEADER *) Buffer; if (fread ((void *) Buffer, Size, 1, InFptr) != 1) { Error (NULL, 0, 0, FileList->FileName, "failed to read file contents"); goto FailDone; } // // Check the header for validity. Check size first. // if (Size < sizeof (EFI_CAPSULE_HEADER)) { Error (NULL, 0, 0, FileList->FileName, "file size is insufficient for a capsule header"); goto FailDone; } // // Check GUID // if (memcmp (&CapHdrPtr->CapsuleGuid, &mEfiCapsuleHeaderGuid, sizeof (EFI_GUID)) != 0) { Error (NULL, 0, 0, FileList->FileName, "invalid capsule GUID"); goto FailDone; } // // Check sequence number // if (CapHdrPtr->SequenceNumber != SequenceNumber) { Error ( NULL, 0, 0, FileList->FileName, "invalid sequence number %d (expected %d)", CapHdrPtr->SequenceNumber, SequenceNumber ); goto FailDone; } // // If the first file, read save the capsule header // if (SequenceNumber == 0) { memcpy (&CapHdr, CapHdrPtr, sizeof (EFI_CAPSULE_HEADER)); // // Erase the InstanceId GUID // memset (&CapHdrPtr->InstanceId, 0, sizeof (EFI_GUID)); if (fwrite (Buffer, Size, 1, OutFptr) != 1) { Error (NULL, 0, 0, FileList->FileName, "failed to write contents to output file"); goto FailDone; } if (CapHdr.CapsuleImageSize < Size) { Error (NULL, 0, 0, FileList->FileName, "capsule image size in capsule header < image size"); goto FailDone; } SizeLeft = CapHdr.CapsuleImageSize - Size; } else { // // Check the GUID against the first file's GUID // if (memcmp (&CapHdr.CapsuleGuid, &CapHdrPtr->CapsuleGuid, sizeof (EFI_GUID)) != 0) { Error (NULL, 0, 0, FileList->FileName, "GUID does not match first file's GUID"); goto FailDone; } // // Make sure we're not throwing out any header info // if (CapHdrPtr->OffsetToCapsuleBody > sizeof (EFI_CAPSULE_HEADER)) { // // Could be the split information, so just emit a warning // Warning ( NULL, 0, 0, FileList->FileName, "image appears to have additional capsule header information -- ignoring" ); } else if (CapHdrPtr->OffsetToCapsuleBody < sizeof (EFI_CAPSULE_HEADER)) { Error (NULL, 0, 0, FileList->FileName, "offset to capsule body in capsule header is insufficient"); goto FailDone; } if (fwrite (Buffer + CapHdrPtr->OffsetToCapsuleBody, Size - CapHdrPtr->OffsetToCapsuleBody, 1, OutFptr) != 1) { Error (NULL, 0, 0, mOptions.OutputFileName, "failed to write to file"); goto FailDone; } if (SizeLeft < (Size - CapHdrPtr->OffsetToCapsuleBody)) { Error (NULL, 0, 0, "sum of image sizes exceeds size specified in initial capsule header", NULL); goto FailDone; } // // printf ("FILE: %s OffsetToCapsuleBody=0x%X\n", FileList->FileName, CapHdrPtr->OffsetToCapsuleBody); // SizeLeft = SizeLeft - (Size - CapHdrPtr->OffsetToCapsuleBody); } // // printf ("FILE: %s sizeleft=0x%X\n", FileList->FileName, SizeLeft); // fclose (InFptr); InFptr = NULL; free (Buffer); Buffer = NULL; FileList = FileList->Next; SequenceNumber++; } if (SizeLeft) { Error (NULL, 0, 0, "sum of capsule images is insufficient", "0x%X bytes missing", SizeLeft); goto FailDone; } Status = STATUS_SUCCESS; goto Done; FailDone: Status = STATUS_ERROR; Done: if (InFptr != NULL) { fclose (InFptr); } if (OutFptr != NULL) { fclose (OutFptr); } if (Buffer != NULL) { free (Buffer); } return Status; #endif return STATUS_SUCCESS; } static STATUS DumpCapsuleHeaderStrings ( UINT8 *SectionName, WCHAR *Buffer ) /*++ Routine Description: Given a pointer to string data from a capsule header, dump the strings. Arguments: SectionName - name of the capsule header section to which the string data pertains Buffer - pointer to string data from a capsule header Returns: STATUS_SUCCESS - all went well --*/ { printf (" %s\n", SectionName); while (*Buffer) { printf (" Language: %S\n", Buffer); while (*Buffer) { Buffer++; } Buffer++; while (*Buffer) { if (wcslen (Buffer) > 60) { printf (" %.60S\n", Buffer); Buffer += 60; } else { printf (" %S\n", Buffer); Buffer += wcslen (Buffer); } } Buffer++; } return STATUS_SUCCESS; } static STATUS GetHexValue ( SOURCE_FILE *SourceFile, UINT32 *Value, UINT32 NumDigits ) /*++ Routine Description: Scan a hex value from the input stream. Arguments: SourceFile - input file contents Value - returned value NumDigits - number of digits to read Returns: STATUS_SUCCESS - if NumDigits were read from the file STATUS_ERROR - otherwise --*/ { WCHAR *SaveFilePos; UINT32 Digits; WCHAR Nibble; SaveFilePos = SourceFile->FileBufferPtr; *Value = 0; Digits = NumDigits; while (Digits > 0) { Nibble = SourceFile->FileBufferPtr[0]; if ((Nibble >= UNICODE_0) && (Nibble <= UNICODE_9)) { *Value = (*Value << 4) | (Nibble - UNICODE_0); } else if ((Nibble >= UNICODE_A) && (Nibble <= UNICODE_F)) { *Value = (*Value << 4) | (Nibble - UNICODE_A + 0x10); } else if ((Nibble >= UNICODE_a) && (Nibble <= UNICODE_f)) { *Value = (*Value << 4) | (Nibble - UNICODE_a + 0x10); } else { Error ( SourceFile->FileName, SourceFile->LineNum, 0, NULL, "expected %d valid hex nibbles at %.20S", NumDigits, SaveFilePos ); return STATUS_ERROR; } SourceFile->FileBufferPtr++; Digits--; } return STATUS_SUCCESS; } static BOOLEAN EndOfFile ( SOURCE_FILE *File ) /*++ Routine Description: GC_TODO: Add function description Arguments: File - GC_TODO: add argument description Returns: GC_TODO: add return values --*/ { if ((UINT32) File->FileBufferPtr - (UINT32) File->FileBuffer >= File->FileSize) { File->EndOfFile = TRUE; } // // Reposition to the end of the file if we went beyond // if (File->EndOfFile) { File->FileBufferPtr = File->FileBuffer + File->FileSize / sizeof (WCHAR); } return File->EndOfFile; } static void SkipWhiteSpace ( SOURCE_FILE *SourceFile ) /*++ Routine Description: GC_TODO: Add function description Arguments: SourceFile - GC_TODO: add argument description Returns: GC_TODO: add return values --*/ { while (!EndOfFile (SourceFile)) { switch (*SourceFile->FileBufferPtr) { case UNICODE_NULL: case UNICODE_CR: case UNICODE_SPACE: case UNICODE_TAB: SourceFile->FileBufferPtr++; break; case UNICODE_LF: SourceFile->FileBufferPtr++; SourceFile->LineNum++; break; default: return ; } } } // // Parse a number. Possible format: // 1234 // 1234k // 1234K // 1M // 1m // 0x100 // static BOOLEAN GetNumber ( INT8 *Str, UINT32 *Value ) /*++ Routine Description: GC_TODO: Add function description Arguments: Str - GC_TODO: add argument description Value - GC_TODO: add argument description Returns: GC_TODO: add return values --*/ { UINT32 LValue; *Value = 0; LValue = 0; if (!isdigit (Str[0])) { return FALSE; } // // Look for hex number // if ((Str[0] == '0') && (tolower (Str[1]) == 'x')) { Str += 2; if (Str[0] == 0) { return FALSE; } while (Str[0]) { if ((Str[0] >= '0') && (Str[0] <= '9')) { LValue = (LValue << 4) | (Str[0] - '0'); } else if ((Str[0] >= 'A') && (Str[0] <= 'F')) { LValue = (LValue << 4) | (Str[0] - 'A' + 0x10); } else if ((Str[0] >= 'a') && (Str[0] <= 'f')) { LValue = (LValue << 4) | (Str[0] - 'a' + 0x10); } else { break; } Str++; } } else { LValue = atoi (Str); while (isdigit (*Str)) { Str++; } } // // If string left over, better be one character we recognize // if (Str[0]) { if (Str[1]) { return FALSE; } switch (Str[0]) { case 'k': case 'K': LValue *= 1024; break; case 'm': case 'M': LValue *= 1024 * 1024; break; default: return FALSE; } } *Value = LValue; return TRUE; } // // Process the command-line arguments // static STATUS ProcessArgs ( int Argc, char *Argv[] ) /*++ Routine Description: Processes command line arguments. Arguments: Argc - Number of command line arguments Argv[] - Array of files input on command line Returns: STATUS_ERROR - Function exited with an error STATUS_SUCCESS - Function executed successfully --*/ { FILE_LIST *NewFile; FILE_LIST *LastFile; SIZE_LIST *NewSize; NewFile = NULL; NewSize = NULL; // // Clear our globals // memset ((char *) &mOptions, 0, sizeof (mOptions)); // // Skip program name // Argc--; Argv++; if (Argc == 0) { Usage (); return STATUS_ERROR; } if ((strcmp(Argv[0], "-h") == 0) || (strcmp(Argv[0], "--help") == 0) || (strcmp(Argv[0], "-?") == 0) || (strcmp(Argv[0], "/?") == 0)) { Usage(); return STATUS_ERROR; } if ((strcmp(Argv[0], "-V") == 0) || (strcmp(Argv[0], "--version") == 0)) { Version(); return STATUS_ERROR; } if (Argc == 1) { Usage (); return STATUS_ERROR; } // // Process until no more options // while ((Argc > 0) && (Argv[0][0] == '-')) { if (stricmp (Argv[0], "-script") == 0) { // // Check for one more arg // if (Argc > 1) { // // Save the file name // if (strlen (Argv[1]) >= sizeof (mOptions.ScriptFileName)) { Error (NULL, 0, 0, NULL, "input script file name length exceeds internal buffer size"); if (NewFile != NULL) { free (NewFile); } if (NewSize != NULL) { free (NewSize); } return STATUS_ERROR; } strcpy (mOptions.ScriptFileName, Argv[1]); } else { Error (NULL, 0, 0, Argv[0], "missing script file name with option"); Usage (); if (NewFile != NULL) { free (NewFile); } if (NewSize != NULL) { free (NewSize); } return STATUS_ERROR; } Argc--; Argv++; // // -o outfilename -- specify output file name (required) // } else if (stricmp (Argv[0], "-o") == 0) { // // check for one more arg // if (Argc > 1) { // // Try to open the file // // if ((mOptions.OutFptr = fopen (Argv[1], "wb")) == NULL) { // Error (NULL, 0, 0, Argv[1], "failed to open output file for writing"); // return STATUS_ERROR; // } // strcpy (mOptions.OutputFileName, Argv[1]); } else { Error (NULL, 0, 0, Argv[0], "missing output filename with option"); Usage (); if (NewFile != NULL) { free (NewFile); } if (NewSize != NULL) { free (NewSize); } return STATUS_ERROR; } Argc--; Argv++; } else if (stricmp (Argv[0], "-j") == 0) { mOptions.JoinMode = TRUE; // // -split <size> option (multiple allowed) // } else if (stricmp (Argv[0], "-split") == 0) { if (Argc > 1) { NewSize = (SIZE_LIST *) malloc (sizeof (SIZE_LIST)); if (NewSize == NULL) { Error (NULL, 0, 0, "memory allocation failure", NULL); if (NewFile != NULL) { free (NewFile); } if (NewSize != NULL) { free (NewSize); } return STATUS_ERROR; } memset (NewSize, 0, sizeof (SIZE_LIST)); // // Get the size from the next arg, and then add this size // to our size list // if (!GetNumber (Argv[1], &NewSize->Size)) { Error (NULL, 0, 0, Argv[1], "invalid split size argument"); if (NewFile != NULL) { free (NewFile); } if (NewSize != NULL) { free (NewSize); } return STATUS_ERROR; } if (mOptions.SizeList == NULL) { mOptions.SizeList = NewSize; mOptions.CurrentSize = NewSize; } else { mOptions.LastSize->Next = NewSize; } mOptions.LastSize = NewSize; free (NewSize); } else { Error (NULL, 0, 0, Argv[0], "missing size parameter with option"); Usage (); if (NewFile != NULL) { free (NewFile); } if (NewSize != NULL) { free (NewSize); } return STATUS_ERROR; } Argc--; Argv++; } else if ((stricmp (Argv[0], "-h") == 0) || (strcmp (Argv[0], "-?") == 0)) { Usage (); if (NewFile != NULL) { free (NewFile); } if (NewSize != NULL) { free (NewSize); } return STATUS_ERROR; // // Default minimum header // } else if (stricmp (Argv[0], "-dump") == 0) { mOptions.Dump = TRUE; } else if (stricmp (Argv[0], "-v") == 0) { mOptions.Verbose = TRUE; } else { Error (NULL, 0, 0, Argv[0], "unrecognized option"); Usage (); if (NewFile != NULL) { free (NewFile); } if (NewSize != NULL) { free (NewSize); } return STATUS_ERROR; } Argc--; Argv++; } // // Can't -j join files and -s split output capsule // if ((mOptions.SizeList != NULL) && (mOptions.JoinMode)) { Error (NULL, 0, 0, "cannot specify both -j and -size", NULL); if (NewFile != NULL) { free (NewFile); } if (NewSize != NULL) { free (NewSize); } return STATUS_ERROR; } // // Must have specified an output file name if not -dump // if ((mOptions.Dump == 0) && (mOptions.OutputFileName[0] == 0)) { Error (NULL, 0, 0, NULL, "-o OutputFileName must be specified"); Usage (); if (NewFile != NULL) { free (NewFile); } if (NewSize != NULL) { free (NewSize); } return STATUS_ERROR; } // // Rest of arguments are input files. The first one is a firmware // volume image, and the rest are FFS files that are to be inserted // into the firmware volume. // LastFile = NULL; while (Argc > 0) { NewFile = (FILE_LIST *) malloc (sizeof (FILE_LIST)); if (NewFile == NULL) { Error (NULL, 0, 0, "memory allocation failure", NULL); if (NewFile != NULL) { free (NewFile); } if (NewSize != NULL) { free (NewSize); } return STATUS_ERROR; } memset ((char *) NewFile, 0, sizeof (FILE_LIST)); strcpy (NewFile->FileName, Argv[0]); if (mOptions.FileList == NULL) { mOptions.FileList = NewFile; } else { if (LastFile == NULL) { LastFile = NewFile; } else { LastFile->Next = NewFile; } } LastFile = NewFile; Argc--; Argv++; } // // Must have provided at least one file name // if (mOptions.FileList == NULL) { Error (NULL, 0, 0, "must specify at least one file name", NULL); Usage (); if (NewFile != NULL) { free (NewFile); } if (NewSize != NULL) { free (NewSize); } return STATUS_ERROR; } return STATUS_SUCCESS; } static void Version( void ) /*++ Routine Description: Print out version information for this utility. Arguments: None Returns: None --*/ { printf ("%s v%d.%d -EDK utility to create a capsule header.\n", UTILITY_NAME, UTILITY_MAJOR_VERSION, UTILITY_MINOR_VERSION); printf ("Copyright (c) 1999-2006 Intel Corporation. All rights reserved.\n"); } static void Usage ( VOID ) /*++ Routine Description: Print usage information for this utility. Arguments: None. Returns: Nothing. --*/ { int Index; static const char *Str[] = { "\nUsage: "UTILITY_NAME " {options} [CapsuleFV]", // // {FfsFileNames}", // " Options include:", " -h,--help,-?,/? to display help messages", " -V,--version to display version information", " -script fname to take capsule header info from unicode script", " file fname", " -o fname write output to file fname (required)", " -split size split capsule image into multiple output files", " -dump to dump a capsule header", " -v for verbose output\n", " -j to join split capsule images into a single image", "", " CapsuleFV is the name of an existing well-formed Tiano firmware", " volume file.", // // FfsFileNames are the names of one or more Tiano FFS files to", // " insert into the output capsule image.", // NULL }; Version(); for (Index = 0; Str[Index] != NULL; Index++) { fprintf (stdout, "%s\n", Str[Index]); } }