mirror of https://github.com/acidanthera/audk.git
460 lines
15 KiB
C
460 lines
15 KiB
C
/** @file
|
||
Main file for Parse shell level 2 function.
|
||
|
||
(C) Copyright 2013-2015 Hewlett-Packard Development Company, L.P.<BR>
|
||
Copyright (c) 2009 - 2012, Intel Corporation. All rights reserved.<BR>
|
||
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.
|
||
|
||
**/
|
||
|
||
#include "UefiShellLevel2CommandsLib.h"
|
||
|
||
/**
|
||
Check if data is coming from StdIn output.
|
||
|
||
@param[in] None
|
||
|
||
@retval TRUE StdIn stream data available to parse
|
||
@retval FALSE StdIn stream data is not available to parse.
|
||
**/
|
||
BOOLEAN
|
||
IsStdInDataAvailable (
|
||
VOID
|
||
)
|
||
{
|
||
SHELL_FILE_HANDLE FileHandle;
|
||
EFI_STATUS Status;
|
||
CHAR16 CharBuffer;
|
||
UINTN CharSize;
|
||
UINT64 OriginalFilePosition;
|
||
|
||
Status = EFI_SUCCESS;
|
||
FileHandle = NULL;
|
||
OriginalFilePosition = 0;
|
||
|
||
if (ShellOpenFileByName (L">i", &FileHandle, EFI_FILE_MODE_READ, 0) == EFI_SUCCESS) {
|
||
CharSize = sizeof(CHAR16);
|
||
gEfiShellProtocol->GetFilePosition (FileHandle, &OriginalFilePosition);
|
||
Status = gEfiShellProtocol->ReadFile (FileHandle, &CharSize, &CharBuffer);
|
||
if (EFI_ERROR (Status) || (CharSize != sizeof(CHAR16))) {
|
||
return FALSE;
|
||
}
|
||
gEfiShellProtocol->SetFilePosition(FileHandle, OriginalFilePosition);
|
||
}
|
||
|
||
if (FileHandle == NULL) {
|
||
return FALSE;
|
||
} else {
|
||
return TRUE;
|
||
}
|
||
}
|
||
|
||
/**
|
||
Function to read a single line (up to but not including the \n) using StdIn data from a SHELL_FILE_HANDLE.
|
||
|
||
If the position upon start is 0, then the Ascii Boolean will be set. This should be
|
||
maintained and not changed for all operations with the same file.
|
||
|
||
@param[in] Handle SHELL_FILE_HANDLE to read from.
|
||
@param[in, out] Buffer The pointer to buffer to read into.
|
||
@param[in, out] Size The pointer to number of bytes in Buffer.
|
||
@param[in] Truncate If the buffer is large enough, this has no effect.
|
||
If the buffer is is too small and Truncate is TRUE,
|
||
the line will be truncated.
|
||
If the buffer is is too small and Truncate is FALSE,
|
||
then no read will occur.
|
||
|
||
@retval EFI_SUCCESS The operation was successful. The line is stored in
|
||
Buffer.
|
||
@retval EFI_INVALID_PARAMETER Handle was NULL.
|
||
@retval EFI_INVALID_PARAMETER Size was NULL.
|
||
@retval EFI_BUFFER_TOO_SMALL Size was not large enough to store the line.
|
||
Size was updated to the minimum space required.
|
||
**/
|
||
EFI_STATUS
|
||
EFIAPI
|
||
ShellFileHandleReadStdInLine(
|
||
IN SHELL_FILE_HANDLE Handle,
|
||
IN OUT CHAR16 *Buffer,
|
||
IN OUT UINTN *Size,
|
||
IN BOOLEAN Truncate
|
||
)
|
||
{
|
||
EFI_STATUS Status;
|
||
CHAR16 CharBuffer;
|
||
UINTN CharSize;
|
||
UINTN CountSoFar;
|
||
UINT64 OriginalFilePosition;
|
||
|
||
|
||
if (Handle == NULL
|
||
||Size == NULL
|
||
){
|
||
return (EFI_INVALID_PARAMETER);
|
||
}
|
||
if (Buffer == NULL) {
|
||
ASSERT(*Size == 0);
|
||
} else {
|
||
*Buffer = CHAR_NULL;
|
||
}
|
||
gEfiShellProtocol->GetFilePosition (Handle, &OriginalFilePosition);
|
||
|
||
for (CountSoFar = 0;;CountSoFar++){
|
||
CharBuffer = 0;
|
||
CharSize = sizeof(CHAR16);
|
||
Status = gEfiShellProtocol->ReadFile (Handle, &CharSize, &CharBuffer);
|
||
if ( EFI_ERROR(Status)
|
||
|| CharSize == 0
|
||
|| (CharBuffer == L'\n')
|
||
){
|
||
break;
|
||
}
|
||
//
|
||
// if we have space save it...
|
||
//
|
||
if ((CountSoFar+1)*sizeof(CHAR16) < *Size){
|
||
ASSERT(Buffer != NULL);
|
||
((CHAR16*)Buffer)[CountSoFar] = CharBuffer;
|
||
((CHAR16*)Buffer)[CountSoFar+1] = CHAR_NULL;
|
||
}
|
||
}
|
||
|
||
//
|
||
// if we ran out of space tell when...
|
||
//
|
||
if ((CountSoFar+1)*sizeof(CHAR16) > *Size){
|
||
*Size = (CountSoFar+1)*sizeof(CHAR16);
|
||
if (!Truncate) {
|
||
gEfiShellProtocol->SetFilePosition(Handle, OriginalFilePosition);
|
||
} else {
|
||
DEBUG((DEBUG_WARN, "The line was truncated in ShellFileHandleReadLine"));
|
||
}
|
||
return (EFI_BUFFER_TOO_SMALL);
|
||
}
|
||
while(Buffer[StrLen(Buffer)-1] == L'\r') {
|
||
Buffer[StrLen(Buffer)-1] = CHAR_NULL;
|
||
}
|
||
|
||
return (Status);
|
||
}
|
||
|
||
|
||
/**
|
||
Function to read a single line using StdIn from a SHELL_FILE_HANDLE. The \n is not included in the returned
|
||
buffer. The returned buffer must be callee freed.
|
||
|
||
If the position upon start is 0, then the Ascii Boolean will be set. This should be
|
||
maintained and not changed for all operations with the same file.
|
||
|
||
@param[in] Handle SHELL_FILE_HANDLE to read from.
|
||
|
||
@return The line of text from the file.
|
||
@retval NULL There was not enough memory available.
|
||
|
||
@sa ShellFileHandleReadLine
|
||
**/
|
||
CHAR16*
|
||
EFIAPI
|
||
ParseReturnStdInLine (
|
||
IN SHELL_FILE_HANDLE Handle
|
||
)
|
||
{
|
||
CHAR16 *RetVal;
|
||
UINTN Size;
|
||
EFI_STATUS Status;
|
||
|
||
Size = 0;
|
||
RetVal = NULL;
|
||
|
||
Status = ShellFileHandleReadStdInLine (Handle, RetVal, &Size, FALSE);
|
||
if (Status == EFI_BUFFER_TOO_SMALL) {
|
||
RetVal = AllocateZeroPool(Size);
|
||
if (RetVal == NULL) {
|
||
return (NULL);
|
||
}
|
||
Status = ShellFileHandleReadStdInLine (Handle, RetVal, &Size, FALSE);
|
||
|
||
}
|
||
if (EFI_ERROR(Status) && (RetVal != NULL)) {
|
||
FreePool(RetVal);
|
||
RetVal = NULL;
|
||
}
|
||
return (RetVal);
|
||
}
|
||
|
||
/**
|
||
Handle stings for SFO Output with escape character ^ in a string
|
||
1. Quotation marks in the string must be escaped by using a ^ character (i.e. ^<5E>).
|
||
2. The ^ character may be inserted using ^^.
|
||
|
||
@param[in] String The Unicode NULL-terminated string.
|
||
|
||
@retval NewString The new string handled for SFO.
|
||
**/
|
||
EFI_STRING
|
||
HandleStringWithEscapeCharForParse (
|
||
IN CHAR16 *String
|
||
)
|
||
{
|
||
EFI_STRING NewStr;
|
||
EFI_STRING StrWalker;
|
||
EFI_STRING ReturnStr;
|
||
|
||
if (String == NULL) {
|
||
return NULL;
|
||
}
|
||
|
||
//
|
||
// start to parse the input string.
|
||
//
|
||
NewStr = AllocateZeroPool (StrSize (String));
|
||
if (NewStr == NULL) {
|
||
return NULL;
|
||
}
|
||
ReturnStr = NewStr;
|
||
StrWalker = String;
|
||
while (*StrWalker != CHAR_NULL) {
|
||
if (*StrWalker == L'^' && (*(StrWalker + 1) == L'^' || *(StrWalker + 1) == L'"')) {
|
||
*NewStr = *(StrWalker + 1);
|
||
StrWalker++;
|
||
} else {
|
||
*NewStr = *StrWalker;
|
||
}
|
||
StrWalker++;
|
||
NewStr++;
|
||
}
|
||
|
||
return ReturnStr;
|
||
}
|
||
|
||
|
||
/**
|
||
Do the actual parsing of the file. the file should be SFO output from a
|
||
shell command or a similar format.
|
||
|
||
@param[in] FileName The filename to open.
|
||
@param[in] TableName The name of the table to find.
|
||
@param[in] ColumnIndex The column number to get.
|
||
@param[in] TableNameInstance Which instance of the table to get (row).
|
||
@param[in] ShellCommandInstance Which instance of the command to get.
|
||
@param[in] StreamingUnicode Indicates Input file is StdIn Unicode streaming data or not
|
||
|
||
@retval SHELL_NOT_FOUND The requested instance was not found.
|
||
@retval SHELL_SUCCESS The operation was successful.
|
||
**/
|
||
SHELL_STATUS
|
||
EFIAPI
|
||
PerformParsing(
|
||
IN CONST CHAR16 *FileName,
|
||
IN CONST CHAR16 *TableName,
|
||
IN CONST UINTN ColumnIndex,
|
||
IN CONST UINTN TableNameInstance,
|
||
IN CONST UINTN ShellCommandInstance,
|
||
IN BOOLEAN StreamingUnicode
|
||
)
|
||
{
|
||
SHELL_FILE_HANDLE FileHandle;
|
||
EFI_STATUS Status;
|
||
BOOLEAN Ascii;
|
||
UINTN LoopVariable;
|
||
UINTN ColumnLoop;
|
||
CHAR16 *TempLine;
|
||
CHAR16 *ColumnPointer;
|
||
SHELL_STATUS ShellStatus;
|
||
CHAR16 *TempSpot;
|
||
CHAR16 *SfoString;
|
||
|
||
ASSERT(FileName != NULL);
|
||
ASSERT(TableName != NULL);
|
||
|
||
ShellStatus = SHELL_SUCCESS;
|
||
|
||
Status = ShellOpenFileByName(FileName, &FileHandle, EFI_FILE_MODE_READ, 0);
|
||
if (EFI_ERROR(Status)) {
|
||
ShellPrintHiiEx(-1, -1, NULL, STRING_TOKEN (STR_GEN_FILE_OPEN_FAIL), gShellLevel2HiiHandle, L"parse", FileName);
|
||
ShellStatus = SHELL_NOT_FOUND;
|
||
} else if (!EFI_ERROR (FileHandleIsDirectory (FileHandle))) {
|
||
ShellPrintHiiEx (-1, -1, NULL, STRING_TOKEN (STR_GEN_NOT_FILE), gShellLevel2HiiHandle, L"parse", FileName);
|
||
ShellStatus = SHELL_NOT_FOUND;
|
||
} else {
|
||
for (LoopVariable = 0 ; LoopVariable < ShellCommandInstance && !ShellFileHandleEof(FileHandle);) {
|
||
if (StreamingUnicode) {
|
||
TempLine = ParseReturnStdInLine (FileHandle);
|
||
} else {
|
||
TempLine = ShellFileHandleReturnLine (FileHandle, &Ascii);
|
||
}
|
||
|
||
if ((TempLine == NULL) || (*TempLine == CHAR_NULL && StreamingUnicode == TRUE)) {
|
||
break;
|
||
}
|
||
|
||
//
|
||
// Search for "ShellCommand," in the file to start the SFO table
|
||
// for a given ShellCommand. The UEFI Shell spec does not specify
|
||
// a space after the comma.
|
||
//
|
||
if (StrStr (TempLine, L"ShellCommand,") == TempLine) {
|
||
LoopVariable++;
|
||
}
|
||
SHELL_FREE_NON_NULL(TempLine);
|
||
}
|
||
if (LoopVariable == ShellCommandInstance) {
|
||
LoopVariable = 0;
|
||
while(1) {
|
||
if (StreamingUnicode) {
|
||
TempLine = ParseReturnStdInLine (FileHandle);
|
||
} else {
|
||
TempLine = ShellFileHandleReturnLine (FileHandle, &Ascii);
|
||
}
|
||
if (TempLine == NULL
|
||
|| *TempLine == CHAR_NULL
|
||
|| StrStr (TempLine, L"ShellCommand,") == TempLine) {
|
||
SHELL_FREE_NON_NULL(TempLine);
|
||
break;
|
||
}
|
||
if (StrStr (TempLine, TableName) == TempLine) {
|
||
LoopVariable++;
|
||
if (LoopVariable == TableNameInstance
|
||
|| (TableNameInstance == (UINTN)-1)) {
|
||
for (ColumnLoop = 1, ColumnPointer = TempLine; ColumnLoop < ColumnIndex && ColumnPointer != NULL && *ColumnPointer != CHAR_NULL; ColumnLoop++) {
|
||
ColumnPointer = StrStr (ColumnPointer, L",\"");
|
||
if (ColumnPointer != NULL && *ColumnPointer != CHAR_NULL){
|
||
ColumnPointer++;
|
||
}
|
||
}
|
||
if (ColumnLoop == ColumnIndex) {
|
||
if (ColumnPointer == NULL) {
|
||
ShellPrintHiiEx(-1, -1, NULL, STRING_TOKEN (STR_GEN_NO_VALUE), gShellLevel2HiiHandle, L"parse", L"Column Index");
|
||
ShellStatus = SHELL_INVALID_PARAMETER;
|
||
} else {
|
||
TempSpot = StrStr (ColumnPointer, L",\"");
|
||
if (TempSpot != NULL) {
|
||
*TempSpot = CHAR_NULL;
|
||
}
|
||
while (ColumnPointer != NULL && *ColumnPointer != CHAR_NULL && ColumnPointer[0] == L' '){
|
||
ColumnPointer++;
|
||
}
|
||
if (ColumnPointer != NULL && *ColumnPointer != CHAR_NULL && ColumnPointer[0] == L'\"'){
|
||
ColumnPointer++;
|
||
}
|
||
if (ColumnPointer != NULL && *ColumnPointer != CHAR_NULL && ColumnPointer[StrLen (ColumnPointer) - 1] == L'\"'){
|
||
ColumnPointer[StrLen (ColumnPointer) - 1] = CHAR_NULL;
|
||
}
|
||
SfoString = HandleStringWithEscapeCharForParse (ColumnPointer);
|
||
if (SfoString != NULL) {
|
||
ShellPrintEx (-1, -1, L"%s\r\n", SfoString);
|
||
SHELL_FREE_NON_NULL (SfoString);
|
||
}
|
||
}
|
||
}
|
||
}
|
||
}
|
||
SHELL_FREE_NON_NULL(TempLine);
|
||
}
|
||
}
|
||
}
|
||
return (ShellStatus);
|
||
}
|
||
|
||
STATIC CONST SHELL_PARAM_ITEM ParamList[] = {
|
||
{L"-i", TypeValue},
|
||
{L"-s", TypeValue},
|
||
{NULL, TypeMax}
|
||
};
|
||
|
||
/**
|
||
Function for 'parse' command.
|
||
|
||
@param[in] ImageHandle Handle to the Image (NULL if Internal).
|
||
@param[in] SystemTable Pointer to the System Table (NULL if Internal).
|
||
**/
|
||
SHELL_STATUS
|
||
EFIAPI
|
||
ShellCommandRunParse (
|
||
IN EFI_HANDLE ImageHandle,
|
||
IN EFI_SYSTEM_TABLE *SystemTable
|
||
)
|
||
{
|
||
EFI_STATUS Status;
|
||
LIST_ENTRY *Package;
|
||
CHAR16 *ProblemParam;
|
||
CONST CHAR16 *FileName;
|
||
CONST CHAR16 *TableName;
|
||
CONST CHAR16 *ColumnString;
|
||
SHELL_STATUS ShellStatus;
|
||
UINTN ShellCommandInstance;
|
||
UINTN TableNameInstance;
|
||
BOOLEAN StreamingUnicode;
|
||
|
||
ShellStatus = SHELL_SUCCESS;
|
||
ProblemParam = NULL;
|
||
StreamingUnicode = FALSE;
|
||
|
||
//
|
||
// initialize the shell lib (we must be in non-auto-init...)
|
||
//
|
||
Status = ShellInitialize();
|
||
ASSERT_EFI_ERROR(Status);
|
||
|
||
//
|
||
// parse the command line
|
||
//
|
||
Status = ShellCommandLineParseEx (ParamList, &Package, &ProblemParam, TRUE, FALSE);
|
||
if (EFI_ERROR(Status)) {
|
||
if (Status == EFI_VOLUME_CORRUPTED && ProblemParam != NULL) {
|
||
ShellPrintHiiEx(-1, -1, NULL, STRING_TOKEN (STR_GEN_PROBLEM), gShellLevel2HiiHandle, L"parse", ProblemParam);
|
||
FreePool(ProblemParam);
|
||
ShellStatus = SHELL_INVALID_PARAMETER;
|
||
} else {
|
||
ASSERT(FALSE);
|
||
}
|
||
} else {
|
||
StreamingUnicode = IsStdInDataAvailable ();
|
||
if ((!StreamingUnicode && (ShellCommandLineGetCount(Package) < 4)) ||
|
||
(ShellCommandLineGetCount(Package) < 3)) {
|
||
ShellPrintHiiEx(-1, -1, NULL, STRING_TOKEN (STR_GEN_TOO_FEW), gShellLevel2HiiHandle, L"parse");
|
||
ShellStatus = SHELL_INVALID_PARAMETER;
|
||
} else if ((StreamingUnicode && (ShellCommandLineGetCount(Package) > 3)) ||
|
||
(ShellCommandLineGetCount(Package) > 4)) {
|
||
ShellPrintHiiEx(-1, -1, NULL, STRING_TOKEN (STR_GEN_TOO_MANY), gShellLevel2HiiHandle, L"parse");
|
||
ShellStatus = SHELL_INVALID_PARAMETER;
|
||
} else {
|
||
if (StreamingUnicode) {
|
||
FileName = L">i";
|
||
TableName = ShellCommandLineGetRawValue(Package, 1);
|
||
ColumnString = ShellCommandLineGetRawValue(Package, 2);
|
||
} else {
|
||
FileName = ShellCommandLineGetRawValue(Package, 1);
|
||
TableName = ShellCommandLineGetRawValue(Package, 2);
|
||
ColumnString = ShellCommandLineGetRawValue(Package, 3);
|
||
}
|
||
if (ShellCommandLineGetValue(Package, L"-i") == NULL) {
|
||
TableNameInstance = (UINTN)-1;
|
||
} else {
|
||
TableNameInstance = ShellStrToUintn(ShellCommandLineGetValue(Package, L"-i"));
|
||
}
|
||
if (ShellCommandLineGetValue(Package, L"-s") == NULL) {
|
||
ShellCommandInstance = 1;
|
||
} else {
|
||
ShellCommandInstance = ShellStrToUintn(ShellCommandLineGetValue(Package, L"-s"));
|
||
}
|
||
|
||
ShellStatus = PerformParsing(FileName, TableName, ShellStrToUintn(ColumnString), TableNameInstance, ShellCommandInstance, StreamingUnicode);
|
||
}
|
||
}
|
||
|
||
//
|
||
// free the command line package
|
||
//
|
||
ShellCommandLineFreeVarList (Package);
|
||
|
||
return (ShellStatus);
|
||
}
|
||
|