mirror of https://github.com/acidanthera/audk.git
289 lines
11 KiB
C
289 lines
11 KiB
C
/** @file
|
|
Main file for attrib shell level 2 function.
|
|
|
|
(C) Copyright 2016 Hewlett Packard Enterprise Development LP<BR>
|
|
(C) Copyright 2015 Hewlett-Packard Development Company, L.P.<BR>
|
|
Copyright (c) 2009 - 2016, 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"
|
|
|
|
/**
|
|
Function for 'cd' 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
|
|
ShellCommandRunCd (
|
|
IN EFI_HANDLE ImageHandle,
|
|
IN EFI_SYSTEM_TABLE *SystemTable
|
|
)
|
|
{
|
|
EFI_STATUS Status;
|
|
LIST_ENTRY *Package;
|
|
CONST CHAR16 *Directory;
|
|
CHAR16 *Cwd;
|
|
CHAR16 *Path;
|
|
CHAR16 *Drive;
|
|
UINTN CwdSize;
|
|
UINTN DriveSize;
|
|
CHAR16 *ProblemParam;
|
|
SHELL_STATUS ShellStatus;
|
|
SHELL_FILE_HANDLE Handle;
|
|
CONST CHAR16 *Param1;
|
|
CHAR16 *Param1Copy;
|
|
CHAR16* Walker;
|
|
|
|
ProblemParam = NULL;
|
|
ShellStatus = SHELL_SUCCESS;
|
|
Drive = NULL;
|
|
DriveSize = 0;
|
|
|
|
Status = CommandInit();
|
|
ASSERT_EFI_ERROR(Status);
|
|
|
|
//
|
|
// initialize the shell lib (we must be in non-auto-init...)
|
|
//
|
|
Status = ShellInitialize();
|
|
ASSERT_EFI_ERROR(Status);
|
|
|
|
//
|
|
// parse the command line
|
|
//
|
|
Status = ShellCommandLineParse (EmptyParamList, &Package, &ProblemParam, TRUE);
|
|
if (EFI_ERROR(Status)) {
|
|
if (Status == EFI_VOLUME_CORRUPTED && ProblemParam != NULL) {
|
|
ShellPrintHiiEx(-1, -1, NULL, STRING_TOKEN (STR_GEN_PROBLEM), gShellLevel2HiiHandle, L"cd", ProblemParam);
|
|
FreePool(ProblemParam);
|
|
ShellStatus = SHELL_INVALID_PARAMETER;
|
|
} else {
|
|
ASSERT(FALSE);
|
|
}
|
|
}
|
|
|
|
//
|
|
// check for "-?"
|
|
//
|
|
if (ShellCommandLineGetFlag(Package, L"-?")) {
|
|
ASSERT(FALSE);
|
|
} else if (ShellCommandLineGetRawValue(Package, 2) != NULL) {
|
|
ShellPrintHiiEx(-1, -1, NULL, STRING_TOKEN (STR_GEN_TOO_MANY), gShellLevel2HiiHandle, L"cd");
|
|
ShellStatus = SHELL_INVALID_PARAMETER;
|
|
} else {
|
|
//
|
|
// remember that param 0 is the command name
|
|
// If there are 0 value parameters, then print the current directory
|
|
// else If there are 2 value parameters, then print the error message
|
|
// else If there is 1 value paramerer , then change the directory
|
|
//
|
|
Param1 = ShellCommandLineGetRawValue(Package, 1);
|
|
if (Param1 == NULL) {
|
|
//
|
|
// display the current directory
|
|
//
|
|
Directory = ShellGetCurrentDir(NULL);
|
|
if (Directory != NULL) {
|
|
ShellPrintHiiEx(-1, -1, NULL, STRING_TOKEN (STR_CD_PRINT), gShellLevel2HiiHandle, Directory);
|
|
} else {
|
|
ShellPrintHiiEx(-1, -1, NULL, STRING_TOKEN (STR_GEN_NO_CWD), gShellLevel2HiiHandle, L"cd");
|
|
ShellStatus = SHELL_NOT_FOUND;
|
|
}
|
|
} else {
|
|
Param1Copy = CatSPrint(NULL, L"%s", Param1, NULL);
|
|
for (Walker = Param1Copy; Walker != NULL && *Walker != CHAR_NULL ; Walker++) {
|
|
if (*Walker == L'\"') {
|
|
CopyMem(Walker, Walker+1, StrSize(Walker) - sizeof(Walker[0]));
|
|
}
|
|
}
|
|
|
|
if (Param1Copy != NULL) {
|
|
Param1Copy = PathCleanUpDirectories(Param1Copy);
|
|
}
|
|
if (Param1Copy != NULL) {
|
|
if (StrCmp(Param1Copy, L".") == 0) {
|
|
//
|
|
// nothing to do... change to current directory
|
|
//
|
|
} else if (StrCmp(Param1Copy, L"..") == 0) {
|
|
//
|
|
// Change up one directory...
|
|
//
|
|
Directory = ShellGetCurrentDir(NULL);
|
|
if (Directory == NULL) {
|
|
ShellPrintHiiEx(-1, -1, NULL, STRING_TOKEN (STR_GEN_NO_CWD), gShellLevel2HiiHandle, L"cd");
|
|
ShellStatus = SHELL_NOT_FOUND;
|
|
} else {
|
|
CwdSize = StrSize(Directory) + sizeof(CHAR16);
|
|
Cwd = AllocateZeroPool(CwdSize);
|
|
if (Cwd == NULL) {
|
|
ShellPrintHiiEx (-1, -1, NULL, STRING_TOKEN (STR_GEN_OUT_MEM), gShellLevel2HiiHandle, L"cd");
|
|
ShellStatus = SHELL_OUT_OF_RESOURCES;
|
|
} else {
|
|
StrCpyS (Cwd, StrSize (Directory) / sizeof (CHAR16) + 1, Directory);
|
|
StrCatS (Cwd, StrSize (Directory) / sizeof (CHAR16) + 1, L"\\");
|
|
Drive = GetFullyQualifiedPath (Cwd);
|
|
PathRemoveLastItem (Drive);
|
|
FreePool (Cwd);
|
|
}
|
|
}
|
|
if (ShellStatus == SHELL_SUCCESS && Drive != NULL) {
|
|
//
|
|
// change directory on current drive letter
|
|
//
|
|
Status = gEfiShellProtocol->SetCurDir(NULL, Drive);
|
|
if (Status == EFI_NOT_FOUND) {
|
|
ShellPrintHiiEx(-1, -1, NULL, STRING_TOKEN (STR_CD_NF), gShellLevel2HiiHandle, L"cd");
|
|
ShellStatus = SHELL_NOT_FOUND;
|
|
}
|
|
}
|
|
} else if (StrCmp(Param1Copy, L"\\") == 0) {
|
|
//
|
|
// Move to root of current drive
|
|
//
|
|
Directory = ShellGetCurrentDir(NULL);
|
|
if (Directory == NULL) {
|
|
ShellPrintHiiEx(-1, -1, NULL, STRING_TOKEN (STR_GEN_NO_CWD), gShellLevel2HiiHandle, L"cd");
|
|
ShellStatus = SHELL_NOT_FOUND;
|
|
} else {
|
|
CwdSize = StrSize(Directory) + sizeof(CHAR16);
|
|
Cwd = AllocateZeroPool(CwdSize);
|
|
if (Cwd == NULL) {
|
|
ShellPrintHiiEx (-1, -1, NULL, STRING_TOKEN (STR_GEN_OUT_MEM), gShellLevel2HiiHandle, L"cd");
|
|
ShellStatus = SHELL_OUT_OF_RESOURCES;
|
|
} else {
|
|
StrCpyS (Cwd, StrSize (Directory) / sizeof (CHAR16) + 1, Directory);
|
|
StrCatS (Cwd, StrSize (Directory) / sizeof (CHAR16) + 1, L"\\");
|
|
Drive = GetFullyQualifiedPath (Cwd);
|
|
while (PathRemoveLastItem (Drive)) {
|
|
//
|
|
// Check if Drive contains 'fsx:\' only or still points to a sub-directory.
|
|
// Don't remove trailing '\' from Drive if it points to the root directory.
|
|
//
|
|
Path = StrStr (Drive, L":\\");
|
|
if ((Path != NULL) && (*(Path + 2) == CHAR_NULL)) {
|
|
break;
|
|
}
|
|
}
|
|
FreePool (Cwd);
|
|
}
|
|
}
|
|
if (ShellStatus == SHELL_SUCCESS && Drive != NULL) {
|
|
//
|
|
// change directory on current drive letter
|
|
//
|
|
Status = gEfiShellProtocol->SetCurDir(NULL, Drive);
|
|
if (Status == EFI_NOT_FOUND) {
|
|
ShellPrintHiiEx(-1, -1, NULL, STRING_TOKEN (STR_CD_NF), gShellLevel2HiiHandle, L"cd");
|
|
ShellStatus = SHELL_NOT_FOUND;
|
|
}
|
|
}
|
|
} else if (StrStr(Param1Copy, L":") == NULL) {
|
|
//
|
|
// change directory without a drive identifier
|
|
//
|
|
if (ShellGetCurrentDir(NULL) == NULL) {
|
|
ShellPrintHiiEx(-1, -1, NULL, STRING_TOKEN (STR_GEN_NO_CWD), gShellLevel2HiiHandle, L"cd");
|
|
ShellStatus = SHELL_NOT_FOUND;
|
|
} else {
|
|
ASSERT((Drive == NULL && DriveSize == 0) || (Drive != NULL));
|
|
Drive = StrnCatGrow(&Drive, &DriveSize, ShellGetCurrentDir(NULL), 0);
|
|
Drive = StrnCatGrow(&Drive, &DriveSize, L"\\", 0);
|
|
if (*Param1Copy == L'\\') {
|
|
while (PathRemoveLastItem(Drive)) ;
|
|
Drive = StrnCatGrow(&Drive, &DriveSize, Param1Copy+1, 0);
|
|
} else {
|
|
Drive = StrnCatGrow(&Drive, &DriveSize, Param1Copy, 0);
|
|
}
|
|
//
|
|
// Verify that this is a valid directory
|
|
//
|
|
Status = gEfiShellProtocol->OpenFileByName(Drive, &Handle, EFI_FILE_MODE_READ);
|
|
if (EFI_ERROR(Status)) {
|
|
ShellPrintHiiEx(-1, -1, NULL, STRING_TOKEN (STR_GEN_DIR_NF), gShellLevel2HiiHandle, L"cd", Drive);
|
|
ShellStatus = SHELL_NOT_FOUND;
|
|
} else if (EFI_ERROR(FileHandleIsDirectory(Handle))) {
|
|
ShellPrintHiiEx(-1, -1, NULL, STRING_TOKEN (STR_GEN_NOT_DIR), gShellLevel2HiiHandle, L"cd", Drive);
|
|
ShellStatus = SHELL_NOT_FOUND;
|
|
}
|
|
if (ShellStatus == SHELL_SUCCESS && Drive != NULL) {
|
|
//
|
|
// change directory on current drive letter
|
|
//
|
|
Status = gEfiShellProtocol->SetCurDir(NULL, Drive);
|
|
if (Status == EFI_NOT_FOUND) {
|
|
ShellPrintHiiEx(-1, -1, NULL, STRING_TOKEN (STR_CD_NF), gShellLevel2HiiHandle, L"cd");
|
|
ShellStatus = SHELL_NOT_FOUND;
|
|
}
|
|
}
|
|
if (Handle != NULL) {
|
|
gEfiShellProtocol->CloseFile(Handle);
|
|
DEBUG_CODE(Handle = NULL;);
|
|
}
|
|
}
|
|
} else {
|
|
//
|
|
// change directory with a drive letter
|
|
//
|
|
Drive = AllocateCopyPool(StrSize(Param1Copy), Param1Copy);
|
|
if (Drive == NULL) {
|
|
ShellPrintHiiEx(-1, -1, NULL, STRING_TOKEN (STR_GEN_NO_MEM), gShellLevel2HiiHandle, L"cd");
|
|
ShellStatus = SHELL_OUT_OF_RESOURCES;
|
|
} else {
|
|
Path = StrStr(Drive, L":");
|
|
ASSERT(Path != NULL);
|
|
if (EFI_ERROR(ShellIsDirectory(Param1Copy))) {
|
|
ShellPrintHiiEx(-1, -1, NULL, STRING_TOKEN (STR_GEN_NOT_DIR), gShellLevel2HiiHandle, L"cd", Param1Copy);
|
|
ShellStatus = SHELL_NOT_FOUND;
|
|
} else if (*(Path+1) == CHAR_NULL) {
|
|
ShellPrintHiiEx(-1, -1, NULL, STRING_TOKEN (STR_CD_NF), gShellLevel2HiiHandle, L"cd");
|
|
ShellStatus = SHELL_NOT_FOUND;
|
|
} else {
|
|
*(Path+1) = CHAR_NULL;
|
|
if (Path == Drive + StrLen(Drive)) {
|
|
ShellPrintHiiEx(-1, -1, NULL, STRING_TOKEN (STR_CD_NF), gShellLevel2HiiHandle, L"cd");
|
|
ShellStatus = SHELL_NOT_FOUND;
|
|
} else {
|
|
Status = gEfiShellProtocol->SetCurDir(Drive, Path+2);
|
|
ShellPrintHiiEx(-1, -1, NULL, STRING_TOKEN (STR_CD_PRINT), gShellLevel2HiiHandle, ShellGetCurrentDir(Drive));
|
|
}
|
|
}
|
|
if (Status == EFI_NOT_FOUND) {
|
|
ShellPrintHiiEx(-1, -1, NULL, STRING_TOKEN (STR_CD_NF), gShellLevel2HiiHandle, L"cd");
|
|
Status = SHELL_NOT_FOUND;
|
|
} else if (EFI_ERROR(Status)) {
|
|
ShellPrintHiiEx(-1, -1, NULL, STRING_TOKEN (STR_GEN_DIR_NF), gShellLevel2HiiHandle, L"cd", Param1Copy);
|
|
Status = SHELL_NOT_FOUND;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
FreePool(Param1Copy);
|
|
}
|
|
}
|
|
|
|
if (Drive != NULL) {
|
|
FreePool(Drive);
|
|
}
|
|
//
|
|
// free the command line package
|
|
//
|
|
ShellCommandLineFreeVarList (Package);
|
|
|
|
//
|
|
// return the status
|
|
//
|
|
return (ShellStatus);
|
|
}
|
|
|