WindowsXP/Source/XPSP1/NT/base/efiutil/launcher/launcher.c
2024-08-03 16:30:48 +02:00

559 lines
11 KiB
C

#include <efi.h>
#include <efilib.h>
//
// Prototype
//
void ParseArgs (EFI_LOADED_IMAGE *ImageHdl);
void Launch (CHAR16 *exePath);
EFI_STATUS OpenCreateFile (UINT64 OCFlags,EFI_FILE_HANDLE *StartHdl,CHAR16 *FileName);
void ParseBootFile (EFI_FILE_HANDLE BootFile);
void PopulateStartFile (EFI_FILE_HANDLE StartFile);
void TrimNonPrint(CHAR16 * str);
CHAR16 * __cdecl mystrstr (const CHAR16 * str1,const CHAR16 * str2);
CHAR16 * ParseLine (CHAR16 *optCopy);
//
//Globals
//
EFI_HANDLE ExeHdl;
EFI_LOADED_IMAGE *ExeImage;
//
// Defines
//
#define REGISTER1 L"*register"
#define REGISTER2 L"*register*"
#define STARTFILE L"startup.nsh"
#define BOOTOFILE L"boot.nvr"
#define OSLOADOPT L"OSLOADER"
#define PARTENT L"partition"
#define PARTENTRE L"*partition*"
EFI_STATUS
EfiMain ( IN EFI_HANDLE ImageHandle,
IN EFI_SYSTEM_TABLE *SystemTable)
{
EFI_STATUS Status;
EFI_FILE_HANDLE bootFile;
InitializeLib (ImageHandle, SystemTable);
ExeHdl = ImageHandle;
BS->HandleProtocol (ImageHandle, &LoadedImageProtocol, &ExeImage);
ParseArgs(ExeImage);
//
// Read the bootfile
//
Status = OpenCreateFile (EFI_FILE_MODE_READ,&bootFile,BOOTOFILE);
ParseBootFile (bootFile);
//
// If we get here, we failed to load the OS
//
return EFI_SUCCESS;
}
void
ParseArgs (EFI_LOADED_IMAGE *ImageInfo)
{
BOOLEAN optFound;
EFI_STATUS Status;
EFI_FILE_HANDLE startFile;
if (MetaiMatch (ImageInfo->LoadOptions,REGISTER1) ||
MetaiMatch (ImageInfo->LoadOptions,REGISTER2)) {
Status = OpenCreateFile (EFI_FILE_MODE_WRITE|EFI_FILE_MODE_CREATE,&startFile,STARTFILE);
if (!(EFI_ERROR (Status))) {
PopulateStartFile (startFile);
BS->Exit (ExeHdl,EFI_SUCCESS,0,NULL);
}
}
}
void
Launch (CHAR16 *exePath)
{
EFI_HANDLE exeHdl=NULL;
UINTN i;
EFI_DEVICE_PATH *ldrDevPath;
EFI_STATUS Status;
EFI_FILE_IO_INTERFACE *Vol;
EFI_FILE_HANDLE RootFs;
EFI_FILE_HANDLE CurDir;
EFI_FILE_HANDLE FileHandle;
CHAR16 FileName[100],*DevicePathAsString;
//
// Open the volume for the device where the exe was loaded from.
//
Status = BS->HandleProtocol (ExeImage->DeviceHandle,
&FileSystemProtocol,
&Vol
);
if (EFI_ERROR(Status)) {
Print(L"Can not get a FileSystem handle for ExeImage->DeviceHandle\n");
BS->Exit(ExeHdl,EFI_SUCCESS,0,NULL);
}
Status = Vol->OpenVolume (Vol, &RootFs);
if (EFI_ERROR(Status)) {
Print(L"Can not open the volume for the file system\n");
BS->Exit(ExeHdl,EFI_SUCCESS,0,NULL);
}
CurDir = RootFs;
//
// Open the file relative to the root.
//
DevicePathAsString = DevicePathToStr(ExeImage->FilePath);
if (DevicePathAsString!=NULL) {
StrCpy(FileName,DevicePathAsString);
FreePool(DevicePathAsString);
}
FileName[0] = 0;
StrCat(FileName,exePath);
// size = StrLen(FileName);
// Print(L"Length of filename is %d\n", size);
// DumpHex(4, 0, 10, &FileName[size - 4]);
//
// Get rid of trailing spaces, new lines, whatever
//
TrimNonPrint(FileName);
Status = CurDir->Open (CurDir,
&FileHandle,
FileName,
EFI_FILE_MODE_READ,
0
);
if (EFI_ERROR(Status)) {
Print(L"Can not open the file ->%s<-, error was %X\n",FileName, Status);
BS->Exit(ExeHdl,EFI_SUCCESS,0,NULL);
} else {
// Print(L"Opened %s\n",FileName);
}
ldrDevPath = FileDevicePath (ExeImage->DeviceHandle,FileName);
/*
if (ldrDevPath) {
Print (L"Type: %d\nSub-Type: %d\nLength[0][1]: [%d][%d]\n",ldrDevPath->Type,
ldrDevPath->SubType,ldrDevPath->Length[0],ldrDevPath->Length[1]);
}else {
Print (L"bad dev path\n");
}
*/
// DumpHex (4,0,ldrDevPath->Length[0],ldrDevPath);
Status = BS->LoadImage (FALSE,ExeHdl,ldrDevPath,NULL,0,&exeHdl);
if (!(EFI_ERROR (Status))) {
// Print (L"Image loaded!\n");
}else {
Print (L"Load Error: %X\n",Status);
BS->Exit(ExeHdl,EFI_SUCCESS,0,NULL);
}
FreePool (ldrDevPath);
BS->StartImage (exeHdl,&i,NULL);
return;
}
EFI_STATUS
OpenCreateFile (UINT64 OCFlags,EFI_FILE_HANDLE *StartHdl,CHAR16 *Name)
{
EFI_FILE_IO_INTERFACE *Vol;
EFI_FILE_HANDLE RootFs;
EFI_FILE_HANDLE CurDir;
EFI_FILE_HANDLE FileHandle;
CHAR16 FileName[100],*DevicePathAsString;
UINTN i;
EFI_STATUS Status;
//
// Open the volume for the device where the EFI OS Loader was loaded from.
//
Status = BS->HandleProtocol (ExeImage->DeviceHandle,
&FileSystemProtocol,
&Vol
);
if (EFI_ERROR(Status)) {
Print(L"Can not get a FileSystem handle for %s DeviceHandle\n",ExeImage->FilePath);
BS->Exit(ExeHdl,EFI_SUCCESS,0,NULL);
}
Status = Vol->OpenVolume (Vol, &RootFs);
if (EFI_ERROR(Status)) {
Print(L"Can not open the volume for the file system\n");
BS->Exit(ExeHdl,EFI_SUCCESS,0,NULL);
}
CurDir = RootFs;
//
// Open the startup options file in the same path as the launcher
//
DevicePathAsString = DevicePathToStr(ExeImage->FilePath);
if (DevicePathAsString!=NULL) {
StrCpy(FileName,DevicePathAsString);
FreePool(DevicePathAsString);
}
for(i=StrLen(FileName);i>0 && FileName[i]!='\\';i--);
FileName[i] = 0;
StrCat(FileName,Name);
Status = CurDir->Open (CurDir,
&FileHandle,
FileName,
OCFlags,
0
);
if (EFI_ERROR(Status)) {
Print(L"Can not open the file %s\n",FileName);
BS->Exit(ExeHdl,EFI_SUCCESS,0,NULL);
}
*StartHdl=FileHandle;
// Print(L"Opened %s\n",FileName);
return Status;
}
void ParseBootFile (EFI_FILE_HANDLE BootFile)
{
EFI_STATUS Status;
char *buffer,*t;
CHAR16 *uniBuf,*optBegin,*optEnd,*optCopy;
UINTN i,size;
EFI_FILE_INFO *bootInfo;
size= SIZE_OF_EFI_FILE_INFO+255*sizeof (CHAR16);
bootInfo = AllocatePool (size);
if (bootInfo == NULL) {
Print (L"Failed to allocate memory for File Info buffer!\n");
BS->Exit(ExeHdl,EFI_SUCCESS,0,NULL);
}
Status = BootFile->GetInfo(BootFile,&GenericFileInfo,&size,bootInfo);
size=(UINTN) bootInfo->FileSize;
FreePool (bootInfo);
buffer = AllocatePool ((size+1));
if (buffer == NULL) {
Print (L"Failed to allocate memory for File buffer!\n");
BS->Exit(ExeHdl,EFI_SUCCESS,0,NULL);
}
Status = BootFile->Read(BootFile,&size,buffer);
BootFile->Close (BootFile);
if (EFI_ERROR (Status)) {
Print (L"Failed to read bootfile!\n");
FreePool (buffer);
BS->Exit(ExeHdl,EFI_SUCCESS,0,NULL);
}
//
// allocate FileSize's space worth of unicode chars.
// (the file is in ASCII)
//
uniBuf = AllocateZeroPool ((size+1) * sizeof (CHAR16));
if (uniBuf == NULL) {
Print (L"Failed to allocate memory for Unicode buffer!\n");
FreePool (buffer);
BS->Exit(ExeHdl,EFI_SUCCESS,0,NULL);
}
t=(char *)uniBuf;
//
// Convert the buffer to a hacked unicode.
//
for (i=0;i<size;i++) {
*(t+i*2)=*(buffer+i);
}
//
//find the option we care about
//
optBegin = mystrstr (uniBuf,OSLOADOPT);
//
// find the end
//
optEnd = optBegin;
while (*(optEnd++) != '\n');
optCopy = AllocateZeroPool (((optEnd-optBegin)+2)*sizeof (CHAR16));
if (optCopy == NULL) {
Print (L"Failed to allocate memory for Unicode buffer!\n");
FreePool (buffer);
FreePool (uniBuf);
BS->Exit(ExeHdl,EFI_SUCCESS,0,NULL);
}
CopyMem (optCopy,optBegin,(optEnd-optBegin)*sizeof (CHAR16));
// Print (L"copied to unicode:%d bytes\n%lX\n%lX\n%s\n",(optEnd-optBegin)*sizeof(CHAR16),optEnd,optBegin,optCopy);
FreePool (uniBuf);
FreePool (buffer);
//
// re-using uniBuf;
//
uniBuf=optBegin=optCopy;
uniBuf =ParseLine (optCopy);
#if 0
do {
uniBuf = mystrstr (uniBuf,PARTENT);
if (uniBuf) {
uniBuf+= StrLen (PARTENT);
optBegin = uniBuf;
}
} while ( uniBuf );
//
// optBegin points to the last partition(n) value
//
while (*(optBegin++) != ')');
optEnd = ++optBegin;
while ((*optEnd != ';') && (*optEnd != '\n')) {
optEnd++;
}
uniBuf = AllocateZeroPool (((optEnd-optBegin)+1)*sizeof (CHAR16));
CopyMem (uniBuf,optBegin,(optEnd-optBegin)*sizeof (CHAR16));
#endif
Print (L"Will launch... %s\n",uniBuf);
Print (L"\nPress any key to abort autoload\n");
Status = WaitForSingleEvent (ST->ConIn->WaitForKey,5*10000000);
if (Status != EFI_TIMEOUT){
BS->Exit(ExeHdl,EFI_SUCCESS,0,NULL);
}
Launch (uniBuf);
FreePool (optCopy);
}
//
// fill out startup.nsh with the name of this program
//
void PopulateStartFile (EFI_FILE_HANDLE StartFile)
{
CHAR16 *NameBuf;
EFI_STATUS Status;
UINTN i,size;
EFI_FILE_INFO *bootInfo;
CHAR16 UnicodeMarker = UNICODE_BYTE_ORDER_MARK;
size= SIZE_OF_EFI_FILE_INFO+255*sizeof (CHAR16);
bootInfo = AllocatePool (size);
if (bootInfo == NULL) {
Print (L"Failed to allocate memory for File Info buffer!\n");
BS->Exit(ExeHdl,EFI_SUCCESS,0,NULL);
}
size = sizeof(UnicodeMarker);
StartFile->Write(StartFile, &size, &UnicodeMarker);
NameBuf=DevicePathToStr (ExeImage->FilePath);
while (*(++NameBuf) != '\\');
//
// take off 4 for the '.efi' extension, which hangs the shell!!!!!
//
size= (StrLen (NameBuf)+2)*sizeof (CHAR16);
StartFile->Write (StartFile,&size,NameBuf);
size = sizeof (CHAR16);
StartFile->Write (StartFile,&size,&L"\n");
StartFile->Close (StartFile);
FreePool (bootInfo);
}
CHAR16*
ParseLine (CHAR16 *optCopy)
{
EFI_STATUS Status;
UINTN i,len,count=0;
CHAR16 *p;
// Print (L"ParseLine: working on %s\n",optCopy);
len=StrLen (optCopy);
//
// Figure out how many tokens there are in the option line
// it will be: TOKENNAME=a;b;c
// (watch for a;b;c;)
//
for (i=0;i<len-1;i++) {
if (*(optCopy+i) == ';') {
count++;
}
}
while (*(++optCopy) != '=');
//
// strip the arc info
//
while (*(++optCopy) != '\\');
p = ++optCopy;
while (*optCopy != '\0' && *optCopy != ';') {
optCopy++;
}
*optCopy='\0';
return (p);
}
/***
*CHAR16 *mystrstr(string1, string2) - search for string2 in string1
*
*Purpose:
* finds the first occurrence of string2 in string1
*
*Entry:
* CHAR16 *string1 - string to search in
* CHAR16 *string2 - string to search for
*
*Exit:
* returns a pointer to the first occurrence of string2 in
* string1, or NULL if string2 does not occur in string1
*
*Uses:
*
*Exceptions:
*
*******************************************************************************/
CHAR16 * __cdecl mystrstr (
const CHAR16 * str1,
const CHAR16 * str2
)
{
CHAR16 *cp = (CHAR16 *) str1;
CHAR16 *s1, *s2;
if ( !*str2 )
return((CHAR16 *)str1);
while (*cp)
{
s1 = cp;
s2 = (CHAR16 *) str2;
while ( *s1 && *s2 && !(*s1-*s2) )
s1++, s2++;
if (!*s2)
return(cp);
cp++;
}
return(NULL);
}
void
TrimNonPrint(
CHAR16 * str
)
{
INTN i,size;
if ((NULL == str) || (L'\0' == *str)) {
return;
}
size = (INTN) StrLen(str);
// Print(L"Size is %d\n", size);
// DumpHex(4, 0, 2, &str[size]);
for (i = size; i > 0; i--) {
if (str[i] <= 0x20) {
str[i] = L'\0';
}
else {
// Leave when we hit a legit character
break;
}
}
}