2010-09-14 07:18:09 +02:00
|
|
|
/** @file
|
|
|
|
Main file for Parse shell level 2 function.
|
|
|
|
|
2015-02-04 23:25:01 +01:00
|
|
|
(C) Copyright 2013-2015 Hewlett-Packard Development Company, L.P.<BR>
|
2018-06-27 15:13:38 +02:00
|
|
|
Copyright (c) 2009 - 2018, Intel Corporation. All rights reserved.<BR>
|
2019-04-04 01:07:06 +02:00
|
|
|
SPDX-License-Identifier: BSD-2-Clause-Patent
|
2010-09-14 07:18:09 +02:00
|
|
|
|
|
|
|
**/
|
|
|
|
|
|
|
|
#include "UefiShellLevel2CommandsLib.h"
|
|
|
|
|
2015-06-03 22:33:41 +02:00
|
|
|
/**
|
|
|
|
Check if data is coming from StdIn output.
|
|
|
|
|
|
|
|
@param[in] None
|
2018-06-27 15:13:38 +02:00
|
|
|
|
|
|
|
@retval TRUE StdIn stream data available to parse
|
|
|
|
@retval FALSE StdIn stream data is not available to parse.
|
2015-06-03 22:33:41 +02:00
|
|
|
**/
|
|
|
|
BOOLEAN
|
|
|
|
IsStdInDataAvailable (
|
|
|
|
VOID
|
|
|
|
)
|
|
|
|
{
|
|
|
|
SHELL_FILE_HANDLE FileHandle;
|
|
|
|
EFI_STATUS Status;
|
2018-06-27 15:13:38 +02:00
|
|
|
CHAR16 CharBuffer;
|
2015-06-03 22:33:41 +02:00
|
|
|
UINTN CharSize;
|
|
|
|
UINT64 OriginalFilePosition;
|
|
|
|
|
2018-06-27 15:13:38 +02:00
|
|
|
Status = EFI_SUCCESS;
|
2015-06-03 22:33:41 +02:00
|
|
|
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;
|
|
|
|
}
|
2021-12-05 23:54:13 +01:00
|
|
|
|
2015-06-03 22:33:41 +02:00
|
|
|
gEfiShellProtocol->SetFilePosition (FileHandle, OriginalFilePosition);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (FileHandle == NULL) {
|
|
|
|
return FALSE;
|
|
|
|
} else {
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-06-03 22:34:48 +02:00
|
|
|
/**
|
|
|
|
Handle stings for SFO Output with escape character ^ in a string
|
2018-06-27 15:13:38 +02:00
|
|
|
1. Quotation marks in the string must be escaped by using a ^ character (i.e. ^").
|
2015-06-03 22:34:48 +02:00
|
|
|
2. The ^ character may be inserted using ^^.
|
|
|
|
|
|
|
|
@param[in] String The Unicode NULL-terminated string.
|
2018-06-27 15:13:38 +02:00
|
|
|
|
2015-06-03 22:34:48 +02:00
|
|
|
@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;
|
|
|
|
}
|
2018-06-27 15:13:38 +02:00
|
|
|
|
2015-06-03 22:34:48 +02:00
|
|
|
//
|
|
|
|
// start to parse the input string.
|
|
|
|
//
|
|
|
|
NewStr = AllocateZeroPool (StrSize (String));
|
|
|
|
if (NewStr == NULL) {
|
|
|
|
return NULL;
|
|
|
|
}
|
2021-12-05 23:54:13 +01:00
|
|
|
|
2015-06-03 22:34:48 +02:00
|
|
|
ReturnStr = NewStr;
|
|
|
|
StrWalker = String;
|
|
|
|
while (*StrWalker != CHAR_NULL) {
|
|
|
|
if ((*StrWalker == L'^') && ((*(StrWalker + 1) == L'^') || (*(StrWalker + 1) == L'"'))) {
|
|
|
|
*NewStr = *(StrWalker + 1);
|
|
|
|
StrWalker++;
|
|
|
|
} else {
|
|
|
|
*NewStr = *StrWalker;
|
|
|
|
}
|
2021-12-05 23:54:13 +01:00
|
|
|
|
2015-06-03 22:34:48 +02:00
|
|
|
StrWalker++;
|
|
|
|
NewStr++;
|
|
|
|
}
|
2018-06-27 15:13:38 +02:00
|
|
|
|
2015-06-03 22:34:48 +02:00
|
|
|
return ReturnStr;
|
|
|
|
}
|
|
|
|
|
2011-03-25 21:58:08 +01:00
|
|
|
/**
|
2018-06-27 15:13:38 +02:00
|
|
|
Do the actual parsing of the file. the file should be SFO output from a
|
2011-03-25 21:58:08 +01:00
|
|
|
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.
|
2015-06-03 22:33:41 +02:00
|
|
|
@param[in] StreamingUnicode Indicates Input file is StdIn Unicode streaming data or not
|
2011-03-25 21:58:08 +01:00
|
|
|
|
|
|
|
@retval SHELL_NOT_FOUND The requested instance was not found.
|
|
|
|
@retval SHELL_SUCCESS The operation was successful.
|
|
|
|
**/
|
2010-09-14 07:18:09 +02:00
|
|
|
SHELL_STATUS
|
|
|
|
PerformParsing (
|
|
|
|
IN CONST CHAR16 *FileName,
|
|
|
|
IN CONST CHAR16 *TableName,
|
|
|
|
IN CONST UINTN ColumnIndex,
|
|
|
|
IN CONST UINTN TableNameInstance,
|
2015-06-03 22:33:41 +02:00
|
|
|
IN CONST UINTN ShellCommandInstance,
|
|
|
|
IN BOOLEAN StreamingUnicode
|
2010-09-14 07:18:09 +02:00
|
|
|
)
|
|
|
|
{
|
|
|
|
SHELL_FILE_HANDLE FileHandle;
|
|
|
|
EFI_STATUS Status;
|
|
|
|
BOOLEAN Ascii;
|
|
|
|
UINTN LoopVariable;
|
|
|
|
UINTN ColumnLoop;
|
|
|
|
CHAR16 *TempLine;
|
|
|
|
CHAR16 *ColumnPointer;
|
|
|
|
SHELL_STATUS ShellStatus;
|
|
|
|
CHAR16 *TempSpot;
|
2015-06-03 22:34:48 +02:00
|
|
|
CHAR16 *SfoString;
|
2021-12-05 23:54:13 +01:00
|
|
|
|
2010-09-14 07:18:09 +02:00
|
|
|
ASSERT (FileName != NULL);
|
|
|
|
ASSERT (TableName != NULL);
|
2021-12-05 23:54:13 +01:00
|
|
|
|
2010-09-14 07:18:09 +02:00
|
|
|
ShellStatus = SHELL_SUCCESS;
|
2021-12-05 23:54:13 +01:00
|
|
|
|
2010-09-14 07:18:09 +02:00
|
|
|
Status = ShellOpenFileByName (FileName, &FileHandle, EFI_FILE_MODE_READ, 0);
|
|
|
|
if (EFI_ERROR (Status)) {
|
2018-06-27 15:13:38 +02:00
|
|
|
ShellPrintHiiEx (-1, -1, NULL, STRING_TOKEN (STR_GEN_FILE_OPEN_FAIL), gShellLevel2HiiHandle, L"parse", FileName);
|
2010-09-14 07:18:09 +02:00
|
|
|
ShellStatus = SHELL_NOT_FOUND;
|
2014-08-21 22:28:32 +02:00
|
|
|
} else if (!EFI_ERROR (FileHandleIsDirectory (FileHandle))) {
|
2018-06-27 15:13:38 +02:00
|
|
|
ShellPrintHiiEx (-1, -1, NULL, STRING_TOKEN (STR_GEN_NOT_FILE), gShellLevel2HiiHandle, L"parse", FileName);
|
2014-08-21 22:28:32 +02:00
|
|
|
ShellStatus = SHELL_NOT_FOUND;
|
2010-09-14 07:18:09 +02:00
|
|
|
} else {
|
|
|
|
for (LoopVariable = 0; LoopVariable < ShellCommandInstance && !ShellFileHandleEof (FileHandle);) {
|
2018-06-27 15:13:38 +02:00
|
|
|
TempLine = ShellFileHandleReturnLine (FileHandle, &Ascii);
|
2015-06-03 22:33:41 +02:00
|
|
|
|
2015-06-09 03:14:10 +02:00
|
|
|
if ((TempLine == NULL) || ((*TempLine == CHAR_NULL) && StreamingUnicode)) {
|
2015-06-03 22:33:41 +02:00
|
|
|
break;
|
2010-09-14 07:18:09 +02:00
|
|
|
}
|
2014-08-21 22:28:32 +02:00
|
|
|
|
|
|
|
//
|
|
|
|
// 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) {
|
2010-09-14 07:18:09 +02:00
|
|
|
LoopVariable++;
|
|
|
|
}
|
2021-12-05 23:54:13 +01:00
|
|
|
|
2010-09-14 07:18:09 +02:00
|
|
|
SHELL_FREE_NON_NULL (TempLine);
|
|
|
|
}
|
2021-12-05 23:54:13 +01:00
|
|
|
|
2010-09-14 07:18:09 +02:00
|
|
|
if (LoopVariable == ShellCommandInstance) {
|
|
|
|
LoopVariable = 0;
|
|
|
|
while (1) {
|
2018-06-27 15:13:38 +02:00
|
|
|
TempLine = ShellFileHandleReturnLine (FileHandle, &Ascii);
|
2014-08-21 22:28:32 +02:00
|
|
|
if ( (TempLine == NULL)
|
|
|
|
|| (*TempLine == CHAR_NULL)
|
|
|
|
|| (StrStr (TempLine, L"ShellCommand,") == TempLine))
|
|
|
|
{
|
2010-09-14 07:18:09 +02:00
|
|
|
SHELL_FREE_NON_NULL (TempLine);
|
|
|
|
break;
|
|
|
|
}
|
2021-12-05 23:54:13 +01:00
|
|
|
|
2014-08-21 22:28:32 +02:00
|
|
|
if (StrStr (TempLine, TableName) == TempLine) {
|
2010-09-14 07:18:09 +02:00
|
|
|
LoopVariable++;
|
2014-08-21 22:28:32 +02:00
|
|
|
if ( (LoopVariable == TableNameInstance)
|
|
|
|
|| (TableNameInstance == (UINTN)-1))
|
|
|
|
{
|
|
|
|
for (ColumnLoop = 1, ColumnPointer = TempLine; ColumnLoop < ColumnIndex && ColumnPointer != NULL && *ColumnPointer != CHAR_NULL; ColumnLoop++) {
|
2014-11-21 00:08:00 +01:00
|
|
|
ColumnPointer = StrStr (ColumnPointer, L",\"");
|
2014-08-21 22:28:32 +02:00
|
|
|
if ((ColumnPointer != NULL) && (*ColumnPointer != CHAR_NULL)) {
|
2012-01-02 22:17:20 +01:00
|
|
|
ColumnPointer++;
|
|
|
|
}
|
2014-08-21 22:28:32 +02:00
|
|
|
}
|
2021-12-05 23:54:13 +01:00
|
|
|
|
2014-08-21 22:28:32 +02:00
|
|
|
if (ColumnLoop == ColumnIndex) {
|
|
|
|
if (ColumnPointer == NULL) {
|
2018-06-27 15:13:38 +02:00
|
|
|
ShellPrintHiiEx (-1, -1, NULL, STRING_TOKEN (STR_GEN_NO_VALUE), gShellLevel2HiiHandle, L"parse", L"Column Index");
|
2014-08-21 22:28:32 +02:00
|
|
|
ShellStatus = SHELL_INVALID_PARAMETER;
|
|
|
|
} else {
|
2014-11-21 00:08:00 +01:00
|
|
|
TempSpot = StrStr (ColumnPointer, L",\"");
|
2014-08-21 22:28:32 +02:00
|
|
|
if (TempSpot != NULL) {
|
|
|
|
*TempSpot = CHAR_NULL;
|
|
|
|
}
|
2021-12-05 23:54:13 +01:00
|
|
|
|
2014-08-21 22:28:32 +02:00
|
|
|
while (ColumnPointer != NULL && *ColumnPointer != CHAR_NULL && ColumnPointer[0] == L' ') {
|
|
|
|
ColumnPointer++;
|
|
|
|
}
|
2021-12-05 23:54:13 +01:00
|
|
|
|
2014-08-21 22:28:32 +02:00
|
|
|
if ((ColumnPointer != NULL) && (*ColumnPointer != CHAR_NULL) && (ColumnPointer[0] == L'\"')) {
|
|
|
|
ColumnPointer++;
|
|
|
|
}
|
2021-12-05 23:54:13 +01:00
|
|
|
|
2014-08-21 22:28:32 +02:00
|
|
|
if ((ColumnPointer != NULL) && (*ColumnPointer != CHAR_NULL) && (ColumnPointer[StrLen (ColumnPointer) - 1] == L'\"')) {
|
|
|
|
ColumnPointer[StrLen (ColumnPointer) - 1] = CHAR_NULL;
|
|
|
|
}
|
2021-12-05 23:54:13 +01:00
|
|
|
|
2015-06-03 22:34:48 +02:00
|
|
|
SfoString = HandleStringWithEscapeCharForParse (ColumnPointer);
|
|
|
|
if (SfoString != NULL) {
|
|
|
|
ShellPrintEx (-1, -1, L"%s\r\n", SfoString);
|
|
|
|
SHELL_FREE_NON_NULL (SfoString);
|
|
|
|
}
|
2014-08-21 22:28:32 +02:00
|
|
|
}
|
2012-01-02 22:17:20 +01:00
|
|
|
}
|
2010-09-14 07:18:09 +02:00
|
|
|
}
|
|
|
|
}
|
2021-12-05 23:54:13 +01:00
|
|
|
|
2010-09-14 07:18:09 +02:00
|
|
|
SHELL_FREE_NON_NULL (TempLine);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2021-12-05 23:54:13 +01:00
|
|
|
|
2010-09-14 07:18:09 +02:00
|
|
|
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;
|
2015-06-03 22:33:41 +02:00
|
|
|
BOOLEAN StreamingUnicode;
|
2010-09-14 07:18:09 +02:00
|
|
|
|
2015-06-03 22:33:41 +02:00
|
|
|
ShellStatus = SHELL_SUCCESS;
|
|
|
|
ProblemParam = NULL;
|
|
|
|
StreamingUnicode = FALSE;
|
2010-09-14 07:18:09 +02:00
|
|
|
|
|
|
|
//
|
|
|
|
// initialize the shell lib (we must be in non-auto-init...)
|
|
|
|
//
|
|
|
|
Status = ShellInitialize ();
|
|
|
|
ASSERT_EFI_ERROR (Status);
|
|
|
|
|
|
|
|
//
|
|
|
|
// parse the command line
|
|
|
|
//
|
2015-06-03 22:33:41 +02:00
|
|
|
Status = ShellCommandLineParseEx (ParamList, &Package, &ProblemParam, TRUE, FALSE);
|
2010-09-14 07:18:09 +02:00
|
|
|
if (EFI_ERROR (Status)) {
|
|
|
|
if ((Status == EFI_VOLUME_CORRUPTED) && (ProblemParam != NULL)) {
|
2018-06-27 15:13:38 +02:00
|
|
|
ShellPrintHiiEx (-1, -1, NULL, STRING_TOKEN (STR_GEN_PROBLEM), gShellLevel2HiiHandle, L"parse", ProblemParam);
|
2010-09-14 07:18:09 +02:00
|
|
|
FreePool (ProblemParam);
|
|
|
|
ShellStatus = SHELL_INVALID_PARAMETER;
|
|
|
|
} else {
|
|
|
|
ASSERT (FALSE);
|
|
|
|
}
|
|
|
|
} else {
|
2015-06-03 22:33:41 +02:00
|
|
|
StreamingUnicode = IsStdInDataAvailable ();
|
|
|
|
if ((!StreamingUnicode && (ShellCommandLineGetCount (Package) < 4)) ||
|
|
|
|
(ShellCommandLineGetCount (Package) < 3))
|
|
|
|
{
|
2018-06-27 15:13:38 +02:00
|
|
|
ShellPrintHiiEx (-1, -1, NULL, STRING_TOKEN (STR_GEN_TOO_FEW), gShellLevel2HiiHandle, L"parse");
|
2010-09-14 07:18:09 +02:00
|
|
|
ShellStatus = SHELL_INVALID_PARAMETER;
|
2015-06-03 22:33:41 +02:00
|
|
|
} else if ((StreamingUnicode && (ShellCommandLineGetCount (Package) > 3)) ||
|
|
|
|
(ShellCommandLineGetCount (Package) > 4))
|
|
|
|
{
|
2018-06-27 15:13:38 +02:00
|
|
|
ShellPrintHiiEx (-1, -1, NULL, STRING_TOKEN (STR_GEN_TOO_MANY), gShellLevel2HiiHandle, L"parse");
|
2010-09-14 07:18:09 +02:00
|
|
|
ShellStatus = SHELL_INVALID_PARAMETER;
|
|
|
|
} else {
|
2015-06-03 22:33:41 +02:00
|
|
|
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);
|
|
|
|
}
|
2021-12-05 23:54:13 +01:00
|
|
|
|
2010-09-14 07:18:09 +02:00
|
|
|
if (ShellCommandLineGetValue (Package, L"-i") == NULL) {
|
|
|
|
TableNameInstance = (UINTN)-1;
|
|
|
|
} else {
|
|
|
|
TableNameInstance = ShellStrToUintn (ShellCommandLineGetValue (Package, L"-i"));
|
|
|
|
}
|
2021-12-05 23:54:13 +01:00
|
|
|
|
2010-09-14 07:18:09 +02:00
|
|
|
if (ShellCommandLineGetValue (Package, L"-s") == NULL) {
|
|
|
|
ShellCommandInstance = 1;
|
|
|
|
} else {
|
|
|
|
ShellCommandInstance = ShellStrToUintn (ShellCommandLineGetValue (Package, L"-s"));
|
|
|
|
}
|
|
|
|
|
2015-06-03 22:33:41 +02:00
|
|
|
ShellStatus = PerformParsing (FileName, TableName, ShellStrToUintn (ColumnString), TableNameInstance, ShellCommandInstance, StreamingUnicode);
|
2010-09-14 07:18:09 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
//
|
|
|
|
// free the command line package
|
|
|
|
//
|
|
|
|
ShellCommandLineFreeVarList (Package);
|
|
|
|
|
|
|
|
return (ShellStatus);
|
|
|
|
}
|