2010-09-14 07:18:09 +02:00
|
|
|
/** @file
|
|
|
|
Main file for NULL named library for level 2 shell command functions.
|
|
|
|
|
|
|
|
these functions are:
|
|
|
|
attrib,
|
|
|
|
cd,
|
|
|
|
cp,
|
|
|
|
date*,
|
|
|
|
time*,
|
|
|
|
load,
|
|
|
|
ls,
|
|
|
|
map,
|
|
|
|
mkdir,
|
|
|
|
mv,
|
|
|
|
parse,
|
|
|
|
rm,
|
|
|
|
reset,
|
|
|
|
set,
|
2011-03-25 21:58:08 +01:00
|
|
|
timezone*,
|
|
|
|
vol
|
2010-09-14 07:18:09 +02:00
|
|
|
|
|
|
|
* functions are non-interactive only
|
|
|
|
|
2014-08-28 21:56:08 +02:00
|
|
|
Copyright (c) 2014 Hewlett-Packard Development Company, L.P.
|
2018-01-26 09:43:22 +01: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"
|
|
|
|
|
|
|
|
CONST CHAR16 mFileName[] = L"ShellCommands";
|
ShellPkg: stop using EFI_HANDLE in place of EFI_HII_HANDLE
The UefiShell*CommandsLib instances have constructor functions that do
something like:
gHiiHandle = HiiAddPackages (...);
...
ShellCommandRegisterCommandName (..., gHiiHandle, ...);
and destructor functions that implement the following pattern:
HiiRemovePackages (gHiiHandle);
The -- semantic, not functional -- problem is that "gHiiHandle" is
declared with type EFI_HANDLE, and not EFI_HII_HANDLE, in all of these
library instances, even though HiiAddPackages() correctly returns
EFI_HII_HANDLE, and HiiRemovePackages() takes EFI_HII_HANDLE.
Once we fix the type of "gHiiHandle", it causes sort of a butterfly
effect, because it is passed around widely. Track down and update all of
those locations.
The DynamicCommand lib instances use a similar pattern, so they are
affected too.
NOTE: in practice, this patch is a no-op, as both EFI_HII_HANDLE and
EFI_HANDLE are typedefs to (VOID*). However, we shouldn't use EFI_HANDLE
where semantically EFI_HII_HANDLE is passed around.
Cc: Jaben Carsey <jaben.carsey@intel.com>
Cc: Ray Ni <ray.ni@intel.com>
Cc: Zhichao Gao <zhichao.gao@intel.com>
Signed-off-by: Laszlo Ersek <lersek@redhat.com>
Reviewed-by: Philippe Mathieu-Daude <philmd@redhat.com>
Reviewed-by: Zhichao Gao <zhichao.gao@intel.com>
2019-09-06 23:15:42 +02:00
|
|
|
EFI_HII_HANDLE gShellLevel2HiiHandle = NULL;
|
2010-09-14 07:18:09 +02:00
|
|
|
|
2011-03-25 21:58:08 +01:00
|
|
|
/**
|
|
|
|
Get the filename to get help text from if not using HII.
|
|
|
|
|
|
|
|
@retval The filename.
|
|
|
|
**/
|
2010-09-14 07:18:09 +02:00
|
|
|
CONST CHAR16 *
|
|
|
|
EFIAPI
|
|
|
|
ShellCommandGetManFileNameLevel2 (
|
|
|
|
VOID
|
|
|
|
)
|
|
|
|
{
|
|
|
|
return (mFileName);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
Constructor for the Shell Level 2 Commands library.
|
|
|
|
|
|
|
|
Install the handlers for level 2 UEFI Shell 2.0 commands.
|
|
|
|
|
|
|
|
@param ImageHandle the image handle of the process
|
|
|
|
@param SystemTable the EFI System Table pointer
|
|
|
|
|
2023-11-09 20:27:09 +01:00
|
|
|
@retval EFI_SUCCESS the shell command handlers were installed successfully
|
2010-09-14 07:18:09 +02:00
|
|
|
@retval EFI_UNSUPPORTED the shell level required was not found.
|
|
|
|
**/
|
|
|
|
EFI_STATUS
|
|
|
|
EFIAPI
|
|
|
|
ShellLevel2CommandsLibConstructor (
|
|
|
|
IN EFI_HANDLE ImageHandle,
|
|
|
|
IN EFI_SYSTEM_TABLE *SystemTable
|
|
|
|
)
|
|
|
|
{
|
|
|
|
//
|
|
|
|
// if shell level is less than 2 do nothing
|
|
|
|
//
|
|
|
|
if (PcdGet8 (PcdShellSupportLevel) < 2) {
|
2011-05-17 00:12:20 +02:00
|
|
|
return (EFI_SUCCESS);
|
2010-09-14 07:18:09 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
gShellLevel2HiiHandle = HiiAddPackages (&gShellLevel2HiiGuid, gImageHandle, UefiShellLevel2CommandsLibStrings, NULL);
|
|
|
|
if (gShellLevel2HiiHandle == NULL) {
|
|
|
|
return (EFI_DEVICE_ERROR);
|
|
|
|
}
|
|
|
|
|
|
|
|
//
|
|
|
|
// install our shell command handlers that are always installed
|
|
|
|
//
|
|
|
|
ShellCommandRegisterCommandName (L"attrib", ShellCommandRunAttrib, ShellCommandGetManFileNameLevel2, 2, L"", TRUE, gShellLevel2HiiHandle, STRING_TOKEN (STR_GET_HELP_ATTRIB));
|
|
|
|
ShellCommandRegisterCommandName (L"cd", ShellCommandRunCd, ShellCommandGetManFileNameLevel2, 2, L"", TRUE, gShellLevel2HiiHandle, STRING_TOKEN (STR_GET_HELP_CD));
|
|
|
|
ShellCommandRegisterCommandName (L"cp", ShellCommandRunCp, ShellCommandGetManFileNameLevel2, 2, L"", TRUE, gShellLevel2HiiHandle, STRING_TOKEN (STR_GET_HELP_CP));
|
|
|
|
ShellCommandRegisterCommandName (L"load", ShellCommandRunLoad, ShellCommandGetManFileNameLevel2, 2, L"", TRUE, gShellLevel2HiiHandle, STRING_TOKEN (STR_GET_HELP_LOAD));
|
|
|
|
ShellCommandRegisterCommandName (L"map", ShellCommandRunMap, ShellCommandGetManFileNameLevel2, 2, L"", TRUE, gShellLevel2HiiHandle, STRING_TOKEN (STR_GET_HELP_MAP));
|
|
|
|
ShellCommandRegisterCommandName (L"mkdir", ShellCommandRunMkDir, ShellCommandGetManFileNameLevel2, 2, L"", TRUE, gShellLevel2HiiHandle, STRING_TOKEN (STR_GET_HELP_MKDIR));
|
|
|
|
ShellCommandRegisterCommandName (L"mv", ShellCommandRunMv, ShellCommandGetManFileNameLevel2, 2, L"", TRUE, gShellLevel2HiiHandle, STRING_TOKEN (STR_GET_HELP_MV));
|
|
|
|
ShellCommandRegisterCommandName (L"parse", ShellCommandRunParse, ShellCommandGetManFileNameLevel2, 2, L"", TRUE, gShellLevel2HiiHandle, STRING_TOKEN (STR_GET_HELP_PARSE));
|
|
|
|
ShellCommandRegisterCommandName (L"reset", ShellCommandRunReset, ShellCommandGetManFileNameLevel2, 2, L"", TRUE, gShellLevel2HiiHandle, STRING_TOKEN (STR_GET_HELP_RESET));
|
2014-02-11 21:36:05 +01:00
|
|
|
ShellCommandRegisterCommandName (L"set", ShellCommandRunSet, ShellCommandGetManFileNameLevel2, 2, L"", FALSE, gShellLevel2HiiHandle, STRING_TOKEN (STR_GET_HELP_SET));
|
2010-09-14 07:18:09 +02:00
|
|
|
ShellCommandRegisterCommandName (L"ls", ShellCommandRunLs, ShellCommandGetManFileNameLevel2, 2, L"", TRUE, gShellLevel2HiiHandle, STRING_TOKEN (STR_GET_HELP_LS));
|
|
|
|
ShellCommandRegisterCommandName (L"rm", ShellCommandRunRm, ShellCommandGetManFileNameLevel2, 2, L"", TRUE, gShellLevel2HiiHandle, STRING_TOKEN (STR_GET_HELP_RM));
|
2011-03-25 21:58:08 +01:00
|
|
|
ShellCommandRegisterCommandName (L"vol", ShellCommandRunVol, ShellCommandGetManFileNameLevel2, 2, L"", TRUE, gShellLevel2HiiHandle, STRING_TOKEN (STR_GET_HELP_VOL));
|
2010-09-14 07:18:09 +02:00
|
|
|
|
|
|
|
//
|
2016-10-19 09:01:35 +02:00
|
|
|
// support for permanent (built in) aliases
|
2010-09-14 07:18:09 +02:00
|
|
|
//
|
|
|
|
ShellCommandRegisterAlias (L"rm", L"del");
|
|
|
|
ShellCommandRegisterAlias (L"ls", L"dir");
|
|
|
|
ShellCommandRegisterAlias (L"cp", L"copy");
|
|
|
|
ShellCommandRegisterAlias (L"mkdir", L"md");
|
|
|
|
ShellCommandRegisterAlias (L"cd ..", L"cd..");
|
|
|
|
ShellCommandRegisterAlias (L"cd \\", L"cd\\");
|
2014-08-28 21:56:08 +02:00
|
|
|
ShellCommandRegisterAlias (L"mv", L"ren");
|
2014-08-29 22:22:48 +02:00
|
|
|
ShellCommandRegisterAlias (L"mv", L"move");
|
|
|
|
ShellCommandRegisterAlias (L"map", L"mount");
|
2010-09-14 07:18:09 +02:00
|
|
|
//
|
|
|
|
// These are installed in level 2 or 3...
|
|
|
|
//
|
|
|
|
if ((PcdGet8 (PcdShellSupportLevel) == 2) || (PcdGet8 (PcdShellSupportLevel) == 3)) {
|
|
|
|
ShellCommandRegisterCommandName (L"date", ShellCommandRunDate, ShellCommandGetManFileNameLevel2, PcdGet8 (PcdShellSupportLevel), L"", TRUE, gShellLevel2HiiHandle, STRING_TOKEN (STR_GET_HELP_DATE));
|
|
|
|
ShellCommandRegisterCommandName (L"time", ShellCommandRunTime, ShellCommandGetManFileNameLevel2, PcdGet8 (PcdShellSupportLevel), L"", TRUE, gShellLevel2HiiHandle, STRING_TOKEN (STR_GET_HELP_TIME));
|
|
|
|
ShellCommandRegisterCommandName (L"timezone", ShellCommandRunTimeZone, ShellCommandGetManFileNameLevel2, PcdGet8 (PcdShellSupportLevel), L"", TRUE, gShellLevel2HiiHandle, STRING_TOKEN (STR_GET_HELP_TIMEZONE));
|
|
|
|
} else {
|
|
|
|
DEBUG_CODE_BEGIN ();
|
|
|
|
//
|
|
|
|
// we want to be able to test these so install them under a different name in debug mode...
|
|
|
|
//
|
|
|
|
ShellCommandRegisterCommandName (L"l2date", ShellCommandRunDate, ShellCommandGetManFileNameLevel2, 2, L"", TRUE, gShellLevel2HiiHandle, STRING_TOKEN (STR_GET_HELP_DATE));
|
|
|
|
ShellCommandRegisterCommandName (L"l2time", ShellCommandRunTime, ShellCommandGetManFileNameLevel2, 2, L"", TRUE, gShellLevel2HiiHandle, STRING_TOKEN (STR_GET_HELP_TIME));
|
|
|
|
ShellCommandRegisterCommandName (L"l2timezone", ShellCommandRunTimeZone, ShellCommandGetManFileNameLevel2, 2, L"", TRUE, gShellLevel2HiiHandle, STRING_TOKEN (STR_GET_HELP_TIMEZONE));
|
|
|
|
DEBUG_CODE_END ();
|
|
|
|
}
|
|
|
|
|
|
|
|
return (EFI_SUCCESS);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
Destructor for the library. free any resources.
|
|
|
|
|
|
|
|
@param ImageHandle The image handle of the process.
|
|
|
|
@param SystemTable The EFI System Table pointer.
|
|
|
|
|
|
|
|
@retval EFI_SUCCESS Always returned.
|
|
|
|
**/
|
|
|
|
EFI_STATUS
|
|
|
|
EFIAPI
|
|
|
|
ShellLevel2CommandsLibDestructor (
|
|
|
|
IN EFI_HANDLE ImageHandle,
|
|
|
|
IN EFI_SYSTEM_TABLE *SystemTable
|
|
|
|
)
|
|
|
|
{
|
|
|
|
if (gShellLevel2HiiHandle != NULL) {
|
|
|
|
HiiRemovePackages (gShellLevel2HiiHandle);
|
|
|
|
}
|
2021-12-05 23:54:13 +01:00
|
|
|
|
2010-09-14 07:18:09 +02:00
|
|
|
return (EFI_SUCCESS);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2023-11-09 20:27:09 +01:00
|
|
|
returns a fully qualified directory (contains a map drive at the beginning)
|
2010-09-14 07:18:09 +02:00
|
|
|
path from a unknown directory path.
|
|
|
|
|
2023-11-09 20:27:09 +01:00
|
|
|
If Path is already fully qualified this will return a duplicate otherwise this
|
2010-09-14 07:18:09 +02:00
|
|
|
will use get the current directory and use that to build the fully qualified
|
|
|
|
version.
|
|
|
|
|
|
|
|
if the return value is not NULL it must be caller freed.
|
|
|
|
|
|
|
|
@param[in] Path The unknown Path Value
|
|
|
|
|
|
|
|
@retval NULL A memory allocation failed
|
2011-10-17 19:49:21 +02:00
|
|
|
@retval NULL A fully qualified path could not be discovered.
|
2023-11-09 20:27:09 +01:00
|
|
|
@retval other An allocated pointer to a fully qualified path.
|
2010-09-14 07:18:09 +02:00
|
|
|
**/
|
|
|
|
CHAR16 *
|
|
|
|
GetFullyQualifiedPath (
|
|
|
|
IN CONST CHAR16 *Path
|
|
|
|
)
|
|
|
|
{
|
|
|
|
CHAR16 *PathToReturn;
|
|
|
|
UINTN Size;
|
|
|
|
CONST CHAR16 *CurDir;
|
|
|
|
|
|
|
|
PathToReturn = NULL;
|
|
|
|
Size = 0;
|
|
|
|
|
|
|
|
ASSERT ((PathToReturn == NULL && Size == 0) || (PathToReturn != NULL));
|
|
|
|
//
|
|
|
|
// convert a local path to an absolute path
|
|
|
|
//
|
|
|
|
if (StrStr (Path, L":") == NULL) {
|
|
|
|
CurDir = gEfiShellProtocol->GetCurDir (NULL);
|
|
|
|
StrnCatGrow (&PathToReturn, &Size, CurDir, 0);
|
2015-10-23 04:03:20 +02:00
|
|
|
StrnCatGrow (&PathToReturn, &Size, L"\\", 0);
|
2010-09-14 07:18:09 +02:00
|
|
|
if (*Path == L'\\') {
|
|
|
|
Path++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
StrnCatGrow (&PathToReturn, &Size, Path, 0);
|
2021-12-05 23:54:13 +01:00
|
|
|
|
2011-07-01 00:58:53 +02:00
|
|
|
PathCleanUpDirectories (PathToReturn);
|
2010-09-14 07:18:09 +02:00
|
|
|
|
2011-10-18 03:47:53 +02:00
|
|
|
if (PathToReturn == NULL) {
|
2011-10-17 19:49:21 +02:00
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2010-09-14 07:18:09 +02:00
|
|
|
while (PathToReturn[StrLen (PathToReturn)-1] == L'*') {
|
|
|
|
PathToReturn[StrLen (PathToReturn)-1] = CHAR_NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
return (PathToReturn);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
Function to verify all intermediate directories in the path.
|
|
|
|
|
|
|
|
@param[in] Path The pointer to the path to fix.
|
|
|
|
|
|
|
|
@retval EFI_SUCCESS The operation was successful.
|
|
|
|
**/
|
|
|
|
EFI_STATUS
|
|
|
|
VerifyIntermediateDirectories (
|
|
|
|
IN CONST CHAR16 *Path
|
|
|
|
)
|
|
|
|
{
|
|
|
|
EFI_STATUS Status;
|
|
|
|
CHAR16 *PathCopy;
|
|
|
|
CHAR16 *TempSpot;
|
|
|
|
SHELL_FILE_HANDLE FileHandle;
|
|
|
|
|
|
|
|
ASSERT (Path != NULL);
|
|
|
|
|
|
|
|
Status = EFI_SUCCESS;
|
|
|
|
PathCopy = NULL;
|
|
|
|
PathCopy = StrnCatGrow (&PathCopy, NULL, Path, 0);
|
|
|
|
FileHandle = NULL;
|
|
|
|
|
2011-10-14 21:21:13 +02:00
|
|
|
if (PathCopy == NULL) {
|
|
|
|
return (EFI_OUT_OF_RESOURCES);
|
|
|
|
}
|
|
|
|
|
2010-09-14 07:18:09 +02:00
|
|
|
for (TempSpot = &PathCopy[StrLen (PathCopy)-1]; *TempSpot != CHAR_NULL && *TempSpot != L'\\'; TempSpot = &PathCopy[StrLen (PathCopy)-1]) {
|
|
|
|
*TempSpot = CHAR_NULL;
|
|
|
|
}
|
2021-12-05 23:54:13 +01:00
|
|
|
|
2010-09-14 07:18:09 +02:00
|
|
|
if (*TempSpot == L'\\') {
|
|
|
|
*TempSpot = CHAR_NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ((PathCopy != NULL) && (*PathCopy != CHAR_NULL)) {
|
|
|
|
Status = VerifyIntermediateDirectories (PathCopy);
|
|
|
|
|
|
|
|
if (PathCopy[StrLen (PathCopy)-1] != L':') {
|
|
|
|
if (!EFI_ERROR (Status)) {
|
|
|
|
Status = ShellOpenFileByName (PathCopy, &FileHandle, EFI_FILE_MODE_READ, 0);
|
|
|
|
if (FileHandle != NULL) {
|
|
|
|
ShellCloseFile (&FileHandle);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
SHELL_FREE_NON_NULL (PathCopy);
|
|
|
|
|
|
|
|
return (Status);
|
|
|
|
}
|
|
|
|
|
2011-03-25 21:58:08 +01:00
|
|
|
/**
|
|
|
|
String comparison without regard to case for a limited number of characters.
|
|
|
|
|
|
|
|
@param[in] Source The first item to compare.
|
|
|
|
@param[in] Target The second item to compare.
|
|
|
|
@param[in] Count How many characters to compare.
|
|
|
|
|
2018-01-26 09:43:22 +01:00
|
|
|
@retval 0 Source and Target are identical strings without regard to case.
|
|
|
|
@retval !=0 Source is not identical to Target.
|
2018-06-27 15:13:38 +02:00
|
|
|
|
2011-03-25 21:58:08 +01:00
|
|
|
**/
|
2018-01-26 09:43:22 +01:00
|
|
|
INTN
|
2010-09-14 07:18:09 +02:00
|
|
|
StrniCmp (
|
|
|
|
IN CONST CHAR16 *Source,
|
|
|
|
IN CONST CHAR16 *Target,
|
|
|
|
IN CONST UINTN Count
|
|
|
|
)
|
|
|
|
{
|
2018-01-26 09:43:22 +01:00
|
|
|
CHAR16 *SourceCopy;
|
|
|
|
CHAR16 *TargetCopy;
|
|
|
|
UINTN SourceLength;
|
|
|
|
UINTN TargetLength;
|
|
|
|
INTN Result;
|
|
|
|
|
|
|
|
if (Count == 0) {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
SourceLength = StrLen (Source);
|
|
|
|
TargetLength = StrLen (Target);
|
|
|
|
SourceLength = MIN (SourceLength, Count);
|
|
|
|
TargetLength = MIN (TargetLength, Count);
|
|
|
|
SourceCopy = AllocateCopyPool ((SourceLength + 1) * sizeof (CHAR16), Source);
|
|
|
|
if (SourceCopy == NULL) {
|
|
|
|
return -1;
|
|
|
|
}
|
2021-12-05 23:54:13 +01:00
|
|
|
|
2018-01-26 09:43:22 +01:00
|
|
|
TargetCopy = AllocateCopyPool ((TargetLength + 1) * sizeof (CHAR16), Target);
|
|
|
|
if (TargetCopy == NULL) {
|
|
|
|
FreePool (SourceCopy);
|
|
|
|
return -1;
|
2010-09-14 07:18:09 +02:00
|
|
|
}
|
2018-06-27 15:13:38 +02:00
|
|
|
|
2018-01-26 09:43:22 +01:00
|
|
|
SourceCopy[SourceLength] = L'\0';
|
|
|
|
TargetCopy[TargetLength] = L'\0';
|
|
|
|
Result = gUnicodeCollation->StriColl (gUnicodeCollation, SourceCopy, TargetCopy);
|
|
|
|
FreePool (SourceCopy);
|
|
|
|
FreePool (TargetCopy);
|
|
|
|
return Result;
|
2010-09-14 07:18:09 +02:00
|
|
|
}
|
|
|
|
|
2014-09-17 09:58:31 +02:00
|
|
|
/**
|
|
|
|
Cleans off all the quotes in the string.
|
|
|
|
|
|
|
|
@param[in] OriginalString pointer to the string to be cleaned.
|
2018-06-27 15:13:38 +02:00
|
|
|
@param[out] CleanString The new string with all quotes removed.
|
|
|
|
Memory allocated in the function and free
|
2014-09-17 09:58:31 +02:00
|
|
|
by caller.
|
|
|
|
|
|
|
|
@retval EFI_SUCCESS The operation was successful.
|
|
|
|
**/
|
|
|
|
EFI_STATUS
|
|
|
|
ShellLevel2StripQuotes (
|
|
|
|
IN CONST CHAR16 *OriginalString,
|
|
|
|
OUT CHAR16 **CleanString
|
|
|
|
)
|
|
|
|
{
|
|
|
|
CHAR16 *Walker;
|
2018-06-27 15:13:38 +02:00
|
|
|
|
2014-09-17 09:58:31 +02:00
|
|
|
if ((OriginalString == NULL) || (CleanString == NULL)) {
|
|
|
|
return EFI_INVALID_PARAMETER;
|
|
|
|
}
|
|
|
|
|
|
|
|
*CleanString = AllocateCopyPool (StrSize (OriginalString), OriginalString);
|
|
|
|
if (*CleanString == NULL) {
|
|
|
|
return EFI_OUT_OF_RESOURCES;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (Walker = *CleanString; Walker != NULL && *Walker != CHAR_NULL; Walker++) {
|
|
|
|
if (*Walker == L'\"') {
|
|
|
|
CopyMem (Walker, Walker+1, StrSize (Walker) - sizeof (Walker[0]));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return EFI_SUCCESS;
|
|
|
|
}
|