2007-06-28 09:00:39 +02:00
|
|
|
/*++
|
|
|
|
|
2010-11-26 02:54:49 +01:00
|
|
|
Copyright (c) 2004 - 2010, Intel Corporation. All rights reserved.<BR>
|
2010-04-28 13:21:53 +02:00
|
|
|
This program and the accompanying materials
|
2007-06-28 09:00:39 +02:00
|
|
|
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:
|
|
|
|
|
|
|
|
GenCRC32Section.c
|
|
|
|
|
|
|
|
Abstract:
|
|
|
|
|
|
|
|
This file contains functions required to generate a Firmware File System
|
|
|
|
file. The code is compliant with the Tiano C Coding standards.
|
|
|
|
|
|
|
|
--*/
|
|
|
|
|
|
|
|
#include "TianoCommon.h"
|
|
|
|
#include "EfiFirmwareFileSystem.h"
|
|
|
|
#include "EfiFirmwareVolumeHeader.h"
|
|
|
|
#include "ParseInf.h"
|
|
|
|
#include "crc32.h"
|
|
|
|
#include "EfiUtilityMsgs.h"
|
|
|
|
#include "GenCRC32Section.h"
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include <assert.h>
|
|
|
|
#include "CommonLib.h"
|
|
|
|
|
|
|
|
#include EFI_PROTOCOL_DEFINITION (GuidedSectionExtraction)
|
|
|
|
|
2010-11-26 02:54:49 +01:00
|
|
|
#define UTILITY_VERSION "v1.0"
|
2007-06-28 09:00:39 +02:00
|
|
|
|
2010-11-26 02:54:49 +01:00
|
|
|
#define UTILITY_NAME "GenCrc32Section"
|
2007-06-28 09:00:39 +02:00
|
|
|
|
|
|
|
EFI_GUID gEfiCrc32SectionGuid = EFI_CRC32_GUIDED_SECTION_EXTRACTION_PROTOCOL_GUID;
|
|
|
|
|
|
|
|
EFI_STATUS
|
|
|
|
SignSectionWithCrc32 (
|
|
|
|
IN OUT UINT8 *FileBuffer,
|
|
|
|
IN OUT UINT32 *BufferSize,
|
|
|
|
IN UINT32 DataSize
|
|
|
|
)
|
|
|
|
/*++
|
|
|
|
|
|
|
|
Routine Description:
|
|
|
|
|
|
|
|
Signs the section with CRC32 and add GUIDed section header for the
|
|
|
|
signed data. data stays in same location (overwrites source data).
|
|
|
|
|
|
|
|
Arguments:
|
|
|
|
|
|
|
|
FileBuffer - Buffer containing data to sign
|
|
|
|
|
|
|
|
BufferSize - On input, the size of FileBuffer. On output, the size of
|
|
|
|
actual section data (including added section header).
|
|
|
|
|
|
|
|
DataSize - Length of data to Sign
|
|
|
|
|
|
|
|
Key - Key to use when signing. Currently only CRC32 is supported.
|
|
|
|
|
|
|
|
Returns:
|
|
|
|
|
|
|
|
EFI_SUCCESS - Successful
|
|
|
|
EFI_OUT_OF_RESOURCES - Not enough resource to complete the operation.
|
|
|
|
|
|
|
|
--*/
|
|
|
|
{
|
|
|
|
|
|
|
|
UINT32 Crc32Checksum;
|
|
|
|
EFI_STATUS Status;
|
|
|
|
UINT32 TotalSize;
|
|
|
|
CRC32_SECTION_HEADER Crc32Header;
|
|
|
|
UINT8 *SwapBuffer;
|
|
|
|
|
|
|
|
Crc32Checksum = 0;
|
|
|
|
SwapBuffer = NULL;
|
|
|
|
|
|
|
|
if (DataSize == 0) {
|
|
|
|
*BufferSize = 0;
|
|
|
|
|
|
|
|
return EFI_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
Status = CalculateCrc32 (FileBuffer, DataSize, &Crc32Checksum);
|
|
|
|
if (EFI_ERROR (Status)) {
|
|
|
|
return Status;
|
|
|
|
}
|
|
|
|
|
|
|
|
TotalSize = DataSize + CRC32_SECTION_HEADER_SIZE;
|
|
|
|
Crc32Header.GuidSectionHeader.CommonHeader.Type = EFI_SECTION_GUID_DEFINED;
|
|
|
|
Crc32Header.GuidSectionHeader.CommonHeader.Size[0] = (UINT8) (TotalSize & 0xff);
|
|
|
|
Crc32Header.GuidSectionHeader.CommonHeader.Size[1] = (UINT8) ((TotalSize & 0xff00) >> 8);
|
|
|
|
Crc32Header.GuidSectionHeader.CommonHeader.Size[2] = (UINT8) ((TotalSize & 0xff0000) >> 16);
|
|
|
|
memcpy (&(Crc32Header.GuidSectionHeader.SectionDefinitionGuid), &gEfiCrc32SectionGuid, sizeof (EFI_GUID));
|
|
|
|
Crc32Header.GuidSectionHeader.Attributes = EFI_GUIDED_SECTION_AUTH_STATUS_VALID;
|
|
|
|
Crc32Header.GuidSectionHeader.DataOffset = CRC32_SECTION_HEADER_SIZE;
|
|
|
|
Crc32Header.CRC32Checksum = Crc32Checksum;
|
|
|
|
|
|
|
|
SwapBuffer = (UINT8 *) malloc (DataSize);
|
|
|
|
if (SwapBuffer == NULL) {
|
|
|
|
return EFI_OUT_OF_RESOURCES;
|
|
|
|
}
|
|
|
|
|
|
|
|
memcpy (SwapBuffer, FileBuffer, DataSize);
|
|
|
|
memcpy (FileBuffer, &Crc32Header, CRC32_SECTION_HEADER_SIZE);
|
|
|
|
memcpy (FileBuffer + CRC32_SECTION_HEADER_SIZE, SwapBuffer, DataSize);
|
|
|
|
|
|
|
|
//
|
|
|
|
// Make sure section ends on a DWORD boundary
|
|
|
|
//
|
|
|
|
while ((TotalSize & 0x03) != 0) {
|
|
|
|
FileBuffer[TotalSize] = 0;
|
|
|
|
TotalSize++;
|
|
|
|
}
|
|
|
|
|
|
|
|
*BufferSize = TotalSize;
|
|
|
|
|
|
|
|
if (SwapBuffer != NULL) {
|
|
|
|
free (SwapBuffer);
|
|
|
|
}
|
|
|
|
|
|
|
|
return EFI_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
VOID
|
|
|
|
PrintUsage (
|
|
|
|
VOID
|
|
|
|
)
|
|
|
|
{
|
2010-11-26 02:54:49 +01:00
|
|
|
int Index;
|
|
|
|
const char *Str[] = {
|
|
|
|
UTILITY_NAME" "UTILITY_VERSION" - Intel Generate CRC32 Section Utility",
|
|
|
|
" Copyright (C), 2004 - 2008 Intel Corporation",
|
|
|
|
|
|
|
|
#if ( defined(UTILITY_BUILD) && defined(UTILITY_VENDOR) )
|
|
|
|
" Built from "UTILITY_BUILD", project of "UTILITY_VENDOR,
|
|
|
|
#endif
|
|
|
|
"",
|
|
|
|
"Usage:",
|
|
|
|
" "UTILITY_NAME" [OPTION]",
|
|
|
|
"Options:",
|
|
|
|
" -i Input1 ... specifies the input file(s) that would be signed to CRC32",
|
|
|
|
" Guided section.",
|
|
|
|
" -o Output specifies the output file that is a CRC32 Guided section",
|
|
|
|
NULL
|
|
|
|
};
|
|
|
|
for (Index = 0; Str[Index] != NULL; Index++) {
|
|
|
|
fprintf (stdout, "%s\n", Str[Index]);
|
|
|
|
}
|
2007-06-28 09:00:39 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
INT32
|
|
|
|
ReadFilesContentsIntoBuffer (
|
|
|
|
IN CHAR8 *argv[],
|
|
|
|
IN INT32 Start,
|
|
|
|
IN OUT UINT8 **FileBuffer,
|
|
|
|
IN OUT UINT32 *BufferSize,
|
|
|
|
OUT UINT32 *ContentSize,
|
|
|
|
IN INT32 MaximumArguments
|
|
|
|
)
|
|
|
|
{
|
|
|
|
INT32 Index;
|
|
|
|
CHAR8 *FileName;
|
|
|
|
FILE *InputFile;
|
|
|
|
UINT8 Temp;
|
|
|
|
UINT32 Size;
|
|
|
|
|
|
|
|
FileName = NULL;
|
|
|
|
InputFile = NULL;
|
|
|
|
Size = 0;
|
|
|
|
Index = 0;
|
|
|
|
|
|
|
|
//
|
|
|
|
// read all input files into one file buffer
|
|
|
|
//
|
|
|
|
while (argv[Start + Index][0] != '-') {
|
|
|
|
|
|
|
|
FileName = argv[Start + Index];
|
|
|
|
InputFile = fopen (FileName, "rb");
|
|
|
|
if (InputFile == NULL) {
|
|
|
|
Error (NULL, 0, 0, FileName, "failed to open input binary file");
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
fread (&Temp, sizeof (UINT8), 1, InputFile);
|
|
|
|
while (!feof (InputFile)) {
|
|
|
|
(*FileBuffer)[Size++] = Temp;
|
|
|
|
fread (&Temp, sizeof (UINT8), 1, InputFile);
|
|
|
|
}
|
|
|
|
|
|
|
|
fclose (InputFile);
|
|
|
|
InputFile = NULL;
|
|
|
|
|
|
|
|
//
|
|
|
|
// Make sure section ends on a DWORD boundary
|
|
|
|
//
|
|
|
|
while ((Size & 0x03) != 0) {
|
|
|
|
(*FileBuffer)[Size] = 0;
|
|
|
|
Size++;
|
|
|
|
}
|
|
|
|
|
|
|
|
Index++;
|
|
|
|
if (Index == MaximumArguments) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
*ContentSize = Size;
|
|
|
|
return Index;
|
|
|
|
}
|
|
|
|
|
|
|
|
INT32
|
|
|
|
main (
|
|
|
|
INT32 argc,
|
|
|
|
CHAR8 *argv[]
|
|
|
|
)
|
|
|
|
{
|
|
|
|
FILE *OutputFile;
|
|
|
|
UINT8 *FileBuffer;
|
|
|
|
UINT32 BufferSize;
|
|
|
|
EFI_STATUS Status;
|
|
|
|
UINT32 ContentSize;
|
|
|
|
CHAR8 *OutputFileName;
|
|
|
|
INT32 ReturnValue;
|
|
|
|
INT32 Index;
|
|
|
|
|
|
|
|
OutputFile = NULL;
|
|
|
|
FileBuffer = NULL;
|
|
|
|
ContentSize = 0;
|
|
|
|
OutputFileName = NULL;
|
|
|
|
|
|
|
|
SetUtilityName (UTILITY_NAME);
|
|
|
|
|
|
|
|
if (argc == 1) {
|
|
|
|
PrintUsage ();
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
BufferSize = 1024 * 1024 * 16;
|
|
|
|
FileBuffer = (UINT8 *) malloc (BufferSize * sizeof (UINT8));
|
|
|
|
if (FileBuffer == NULL) {
|
|
|
|
Error (NULL, 0, 0, "memory allocation failed", NULL);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
ZeroMem (FileBuffer, BufferSize);
|
|
|
|
|
|
|
|
for (Index = 0; Index < argc; Index++) {
|
|
|
|
if (_strcmpi (argv[Index], "-i") == 0) {
|
|
|
|
ReturnValue = ReadFilesContentsIntoBuffer (
|
|
|
|
argv,
|
|
|
|
(Index + 1),
|
|
|
|
&FileBuffer,
|
|
|
|
&BufferSize,
|
|
|
|
&ContentSize,
|
|
|
|
(argc - (Index + 1))
|
|
|
|
);
|
|
|
|
if (ReturnValue == -1) {
|
|
|
|
Error (NULL, 0, 0, "failed to read file contents", NULL);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
Index += ReturnValue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (_strcmpi (argv[Index], "-o") == 0) {
|
|
|
|
OutputFileName = argv[Index + 1];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
OutputFile = fopen (OutputFileName, "wb");
|
|
|
|
if (OutputFile == NULL) {
|
|
|
|
Error (NULL, 0, 0, OutputFileName, "failed to open output binary file");
|
|
|
|
free (FileBuffer);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
//
|
|
|
|
// make sure section ends on a DWORD boundary ??
|
|
|
|
//
|
|
|
|
while ( (Size & 0x03) != 0 ) {
|
|
|
|
FileBuffer[Size] = 0;
|
|
|
|
Size ++;
|
|
|
|
}
|
|
|
|
*/
|
|
|
|
Status = SignSectionWithCrc32 (FileBuffer, &BufferSize, ContentSize);
|
|
|
|
if (EFI_ERROR (Status)) {
|
|
|
|
Error (NULL, 0, 0, "failed to sign section", NULL);
|
|
|
|
free (FileBuffer);
|
|
|
|
fclose (OutputFile);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
ContentSize = fwrite (FileBuffer, sizeof (UINT8), BufferSize, OutputFile);
|
|
|
|
if (ContentSize != BufferSize) {
|
|
|
|
Error (NULL, 0, 0, "failed to write output buffer", NULL);
|
|
|
|
ReturnValue = -1;
|
|
|
|
} else {
|
|
|
|
ReturnValue = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
free (FileBuffer);
|
|
|
|
fclose (OutputFile);
|
|
|
|
return ReturnValue;
|
|
|
|
}
|