nightloader/libs/posix-uefi/utils/efiffs.c

270 lines
8.5 KiB
C

/*
* utils/efiffs.c
*
* Copyright (C) 2022 bzt (bztsrc@gitlab)
*
* Permission is hereby granted, free of charge, to any person
* obtaining a copy of this software and associated documentation
* files (the "Software"), to deal in the Software without
* restriction, including without limitation the rights to use, copy,
* modify, merge, publish, distribute, sublicense, and/or sell copies
* of the Software, and to permit persons to whom the Software is
* furnished to do so, subject to the following conditions:
*
* The above copyright notice and this permission notice shall be
* included in all copies or substantial portions of the Software.
*
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
* EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
* NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT
* HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY,
* WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
* DEALINGS IN THE SOFTWARE.
*
* This file is part of the POSIX-UEFI package.
* @brief small tool to convert an .efi file to an .ffs file
*
*/
#include <stdint.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <time.h>
typedef struct {
uint32_t Data1;
uint16_t Data2;
uint16_t Data3;
uint8_t Data4[8];
} __attribute__((packed)) guid_t;
/**
* Print usage
*/
void usage(char *cmd)
{
printf("POSIX-UEFI utils - efiffs by bztsrc@gitlab MIT\r\n\r\n");
printf("%s [-g <guid>] [-n <name>] [-v <ver>] [-t <type>] [-p <type>] infile [outfile]\r\n\r\n", cmd);
printf(" -g <guid> specify the GUID (defaults to random)\r\n");
printf(" -n <name> specify the driver's name (eg 'FAT')\r\n");
printf(" -v <ver> specify the driver's version (eg '1.0')\r\n");
printf(" -t <type> specify the ffs type (defaults to 7, EFI_FV_FILETYPE_DRIVER)\r\n");
printf(" -p <type> specify the pe section type (defaults to 16, EFI_SECTION_PE32)\r\n");
printf(" infile input .efi file\r\n");
printf(" outfile output file name (default generated from infile)\r\n");
exit(1);
}
/**
* Convert hex string into integer
*/
unsigned int gethex(char *ptr, int len)
{
unsigned int ret = 0;
for(;len--;ptr++) {
if(*ptr>='0' && *ptr<='9') { ret <<= 4; ret += (unsigned int)(*ptr-'0'); }
else if(*ptr >= 'a' && *ptr <= 'f') { ret <<= 4; ret += (unsigned int)(*ptr-'a'+10); }
else if(*ptr >= 'A' && *ptr <= 'F') { ret <<= 4; ret += (unsigned int)(*ptr-'A'+10); }
else break;
}
return ret;
}
/**
* Parse a GUID in string into binary representation
*/
void getguid(char *ptr, guid_t *guid)
{
int i;
if(!ptr || !*ptr || ptr[8] != '-' || ptr[13] != '-' || ptr[18] != '-') {
fprintf(stderr, "efiffs: bad GUID format\r\n");
return;
}
guid->Data1 = gethex(ptr, 8); ptr += 9;
guid->Data2 = gethex(ptr, 4); ptr += 5;
guid->Data3 = gethex(ptr, 4); ptr += 5;
guid->Data4[0] = gethex(ptr, 2); ptr += 2;
guid->Data4[1] = gethex(ptr, 2); ptr += 2; if(*ptr == '-') ptr++;
for(i = 2; i < 8; i++, ptr += 2) guid->Data4[i] = gethex(ptr, 2);
}
/**
* Parse ffs type
*/
int gettype(char *str)
{
static const char *types[] = {
NULL, "RAW", "FREEFORM", "SECURITY_CORE", "PEI_CORE", "DXE_CORE", "PEIM", "DRIVER", "COMBINED_PEIM_DRIVER",
"APPLICATION", "SMM", "FIRMWARE_VOLUME_IMAGE", "COMBINED_SMM_DXE", "SMM_CORE", NULL };
int i;
i = atoi(str);
if(i > 0 && i < (int)(sizeof(types)/sizeof(types[0]))) return i;
if(!memcmp(str, "EFI_FV_FILETYPE_", 16)) str += 16;
for(i = 1; types[i] && strcmp(str, types[i]); i++);
if(!types[i]) {
fprintf(stderr, "efiffs: invalid ffs type, available values:\r\n");
for(i = 1; types[i]; i++)
fprintf(stderr, " EFI_FV_FILETYPE_%s\r\n", types[i]);
return 7; /* EFI_FV_FILETYPE_DRIVER */
}
return i;
}
/**
* Parse section type
*/
int getsec(char *str)
{
static const char *types[] = {
"PE32", "PIC", "TE", "DXE_DEPEX", "VERSION", "USER_INTERFACE", "COMPATIBILITY16",
"FIRMWARE_VOLUME_IMAGE", "FREEFORM_SUBTYPE_GUID", "RAW", "RESERVED", "PEI_DEPEX", "SMM_DEPEX", NULL };
int i;
i = atoi(str);
if(i == 1 || i == 2 || i >= 16) return i;
if(!memcmp(str, "EFI_SECTION_", 12)) str += 12;
if(!strcmp(str, "COMPRESSION")) return 1;
if(!strcmp(str, "GUID_DEFINED")) return 2;
for(i = 0; types[i] && strcmp(str, types[i]); i++);
if(!types[i]) {
fprintf(stderr, "efiffs: invalid section type, available values:\r\n");
fprintf(stderr, " EFI_SECTION_COMPRESSION\r\n EFI_SECTION_GUID_DEFINED\r\n");
for(i = 0; types[i]; i++)
fprintf(stderr, " EFI_SECTION_%s\r\n", types[i]);
return 16; /* EFI_SECTION_PE32 */
}
return i + 16;
}
/**
* Calculate checksum
*/
uint8_t checksum8(uint8_t *buff, int len)
{
uint8_t ret = 0;
int i;
for(i = 0; i < len; i++)
ret = (uint8_t)(ret + buff[i]);
return (uint8_t)(0x100 - ret);
}
/**
* Main function
*/
int main(int argc, char **argv)
{
FILE *f;
int i, size, len;
int filetype = 7; /* EFI_FV_FILETYPE_DRIVER */
int sectype = 16; /* EFI_SECTION_PE32 */
char *in = NULL, *out = NULL, *name = NULL, *ver = NULL;
uint8_t *buff = NULL, *sec = NULL;
guid_t guid;
/* get a random GUID */
srand(time(NULL));
i = rand(); memcpy(&((uint8_t*)&guid)[0], &i, 4);
i = rand(); memcpy(&((uint8_t*)&guid)[4], &i, 4);
i = rand(); memcpy(&((uint8_t*)&guid)[8], &i, 4);
i = rand(); memcpy(&((uint8_t*)&guid)[12], &i, 4);
/* parse command line */
for(i = 1; i < argc && argv[i]; i++)
if(argv[i][0] == '-') {
switch(argv[i][1]) {
case 'g': getguid(argv[++i], &guid); break;
case 'n': name = argv[++i]; break;
case 'v': ver = argv[++i]; break;
case 't': filetype = gettype(argv[++i]); break;
case 'p': sectype = getsec(argv[++i]); break;
default: usage(argv[0]); break;
}
} else
if(!in) in = argv[i]; else
if(!out) out = argv[i]; else
usage(argv[0]);
if(!in) usage(argv[0]);
/* get input data */
f = fopen(in, "rb");
if(!f) {
fprintf(stderr, "efiffs: unable to read '%s'\r\n", in);
return 2;
}
fseek(f, 0L, SEEK_END);
size = (int)ftell(f);
fseek(f, 0L, SEEK_SET);
len = 24 + 12 + 4 + size + (name ? 6 + 2 * strlen(name) : 0) + (ver ? 6 + 2 * strlen(ver) : 0);
buff = (uint8_t*)malloc(len);
if(!buff) { fprintf(stderr, "efiffs: unable to allocate memory\r\n"); return 2; }
memset(buff, 0, len);
sec = buff + 24;
/* add the PE section */
sec[0] = size & 0xff;
sec[1] = (size >> 8) & 0xff;
sec[2] = (size >> 16) & 0xff;
sec[3] = sectype;
fread(sec + 4, 1, size, f);
fclose(f);
sec += 4 + ((size + 3) & ~3);
/* add the name section */
if(name) {
size = 6 + 2 * strlen(name);
sec[0] = size & 0xff;
sec[1] = (size >> 8) & 0xff;
sec[2] = (size >> 16) & 0xff;
sec[3] = 0x15; /* EFI_SECTION_USER_INTERFACE */
for(i = 0; name[i]; i++)
sec[4 + 2 * i] = name[i];
sec += ((size + 3) & ~3);
}
/* add the version section */
if(ver) {
size = 6 + 2 * strlen(ver);
sec[0] = size & 0xff;
sec[1] = (size >> 8) & 0xff;
sec[2] = (size >> 16) & 0xff;
sec[3] = 0x14; /* EFI_SECTION_VERSION */
for(i = 0; ver[i]; i++)
sec[4 + 2 * i] = ver[i];
sec += ((size + 3) & ~3);
}
/* calculate ffs header fields */
len = (int)((uintptr_t)sec - (uintptr_t)buff);
memcpy(buff, &guid, 16);
buff[0x12] = filetype;
buff[0x14] = len & 0xff;
buff[0x15] = (len >> 8) & 0xff;
buff[0x16] = (len >> 16) & 0xff;
buff[0x11] = 0xAA;
buff[0x10] = checksum8(buff, 24);
/* write out image */
if(!out) {
i = strlen(in);
out = (char*)malloc(i + 5);
if(!out) { fprintf(stderr, "efiffs: unable to allocate memory\r\n"); return 2; }
strcpy(out, in);
strcpy(!strcmp(out + i - 4, ".efi") ? out + i - 4 : out + i, ".ffs");
}
f = fopen(out, "wb");
if(!f) {
fprintf(stderr, "efiffs: unable to write '%s'\r\n", out);
return 2;
}
fwrite(buff, 1, len, f);
fclose(f);
free(buff);
return 0;
}