/** @file | |
Copyright (c) 2008 - 2009, Apple Inc. All rights reserved.<BR> | |
Copyright (c) 2013 - 2014, ARM Ltd. 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 <Base.h> | |
#include <Library/BaseLib.h> | |
#include <Library/SemihostLib.h> | |
#include "SemihostPrivate.h" | |
BOOLEAN | |
SemihostConnectionSupported ( | |
VOID | |
) | |
{ | |
return SEMIHOST_SUPPORTED; | |
} | |
RETURN_STATUS | |
SemihostFileOpen ( | |
IN CHAR8 *FileName, | |
IN UINT32 Mode, | |
OUT UINTN *FileHandle | |
) | |
{ | |
SEMIHOST_FILE_OPEN_BLOCK OpenBlock; | |
INT32 Result; | |
if (FileHandle == NULL) { | |
return RETURN_INVALID_PARAMETER; | |
} | |
// Remove any leading separator (e.g.: '\'). EFI Shell adds one. | |
if (*FileName == '\\') { | |
FileName++; | |
} | |
OpenBlock.FileName = FileName; | |
OpenBlock.Mode = Mode; | |
OpenBlock.NameLength = AsciiStrLen(FileName); | |
Result = Semihost_SYS_OPEN(&OpenBlock); | |
if (Result == -1) { | |
return RETURN_NOT_FOUND; | |
} else { | |
*FileHandle = Result; | |
return RETURN_SUCCESS; | |
} | |
} | |
RETURN_STATUS | |
SemihostFileSeek ( | |
IN UINTN FileHandle, | |
IN UINTN Offset | |
) | |
{ | |
SEMIHOST_FILE_SEEK_BLOCK SeekBlock; | |
INT32 Result; | |
SeekBlock.Handle = FileHandle; | |
SeekBlock.Location = Offset; | |
Result = Semihost_SYS_SEEK(&SeekBlock); | |
// Semihosting does not behave as documented. It returns the offset on | |
// success. | |
if (Result < 0) { | |
return RETURN_ABORTED; | |
} else { | |
return RETURN_SUCCESS; | |
} | |
} | |
RETURN_STATUS | |
SemihostFileRead ( | |
IN UINTN FileHandle, | |
IN OUT UINTN *Length, | |
OUT VOID *Buffer | |
) | |
{ | |
SEMIHOST_FILE_READ_WRITE_BLOCK ReadBlock; | |
UINT32 Result; | |
if ((Length == NULL) || (Buffer == NULL)) { | |
return RETURN_INVALID_PARAMETER; | |
} | |
ReadBlock.Handle = FileHandle; | |
ReadBlock.Buffer = Buffer; | |
ReadBlock.Length = *Length; | |
Result = Semihost_SYS_READ(&ReadBlock); | |
if ((*Length != 0) && (Result == *Length)) { | |
return RETURN_ABORTED; | |
} else { | |
*Length -= Result; | |
return RETURN_SUCCESS; | |
} | |
} | |
RETURN_STATUS | |
SemihostFileWrite ( | |
IN UINTN FileHandle, | |
IN OUT UINTN *Length, | |
IN VOID *Buffer | |
) | |
{ | |
SEMIHOST_FILE_READ_WRITE_BLOCK WriteBlock; | |
if ((Length == NULL) || (Buffer == NULL)) { | |
return RETURN_INVALID_PARAMETER; | |
} | |
WriteBlock.Handle = FileHandle; | |
WriteBlock.Buffer = Buffer; | |
WriteBlock.Length = *Length; | |
*Length = Semihost_SYS_WRITE(&WriteBlock); | |
if (*Length != 0) | |
return RETURN_ABORTED; | |
else | |
return RETURN_SUCCESS; | |
} | |
RETURN_STATUS | |
SemihostFileClose ( | |
IN UINTN FileHandle | |
) | |
{ | |
INT32 Result = Semihost_SYS_CLOSE(&FileHandle); | |
if (Result == -1) { | |
return RETURN_INVALID_PARAMETER; | |
} else { | |
return RETURN_SUCCESS; | |
} | |
} | |
RETURN_STATUS | |
SemihostFileLength ( | |
IN UINTN FileHandle, | |
OUT UINTN *Length | |
) | |
{ | |
INT32 Result; | |
if (Length == NULL) { | |
return RETURN_INVALID_PARAMETER; | |
} | |
Result = Semihost_SYS_FLEN(&FileHandle); | |
if (Result == -1) { | |
return RETURN_ABORTED; | |
} else { | |
*Length = Result; | |
return RETURN_SUCCESS; | |
} | |
} | |
/** | |
Get a temporary name for a file from the host running the debug agent. | |
@param[out] Buffer Pointer to the buffer where the temporary name has to | |
be stored | |
@param[in] Identifier File name identifier (integer in the range 0 to 255) | |
@param[in] Length Length of the buffer to store the temporary name | |
@retval RETURN_SUCCESS Temporary name returned | |
@retval RETURN_INVALID_PARAMETER Invalid buffer address | |
@retval RETURN_ABORTED Temporary name not returned | |
**/ | |
RETURN_STATUS | |
SemihostFileTmpName( | |
OUT VOID *Buffer, | |
IN UINT8 Identifier, | |
IN UINTN Length | |
) | |
{ | |
SEMIHOST_FILE_TMPNAME_BLOCK TmpNameBlock; | |
INT32 Result; | |
if (Buffer == NULL) { | |
return RETURN_INVALID_PARAMETER; | |
} | |
TmpNameBlock.Buffer = Buffer; | |
TmpNameBlock.Identifier = Identifier; | |
TmpNameBlock.Length = Length; | |
Result = Semihost_SYS_TMPNAME (&TmpNameBlock); | |
if (Result != 0) { | |
return RETURN_ABORTED; | |
} else { | |
return RETURN_SUCCESS; | |
} | |
} | |
RETURN_STATUS | |
SemihostFileRemove ( | |
IN CHAR8 *FileName | |
) | |
{ | |
SEMIHOST_FILE_REMOVE_BLOCK RemoveBlock; | |
UINT32 Result; | |
// Remove any leading separator (e.g.: '\'). EFI Shell adds one. | |
if (*FileName == '\\') { | |
FileName++; | |
} | |
RemoveBlock.FileName = FileName; | |
RemoveBlock.NameLength = AsciiStrLen(FileName); | |
Result = Semihost_SYS_REMOVE(&RemoveBlock); | |
if (Result == 0) { | |
return RETURN_SUCCESS; | |
} else { | |
return RETURN_ABORTED; | |
} | |
} | |
/** | |
Rename a specified file. | |
@param[in] FileName Name of the file to rename. | |
@param[in] NewFileName The new name of the file. | |
@retval RETURN_SUCCESS File Renamed | |
@retval RETURN_INVALID_PARAMETER Either the current or the new name is not specified | |
@retval RETURN_ABORTED Rename failed | |
**/ | |
RETURN_STATUS | |
SemihostFileRename( | |
IN CHAR8 *FileName, | |
IN CHAR8 *NewFileName | |
) | |
{ | |
SEMIHOST_FILE_RENAME_BLOCK RenameBlock; | |
INT32 Result; | |
if ((FileName == NULL) || (NewFileName == NULL)) { | |
return RETURN_INVALID_PARAMETER; | |
} | |
RenameBlock.FileName = FileName; | |
RenameBlock.FileNameLength = AsciiStrLen (FileName); | |
RenameBlock.NewFileName = NewFileName; | |
RenameBlock.NewFileNameLength = AsciiStrLen (NewFileName); | |
Result = Semihost_SYS_RENAME (&RenameBlock); | |
if (Result != 0) { | |
return RETURN_ABORTED; | |
} else { | |
return RETURN_SUCCESS; | |
} | |
} | |
CHAR8 | |
SemihostReadCharacter ( | |
VOID | |
) | |
{ | |
return Semihost_SYS_READC(); | |
} | |
VOID | |
SemihostWriteCharacter ( | |
IN CHAR8 Character | |
) | |
{ | |
Semihost_SYS_WRITEC(&Character); | |
} | |
VOID | |
SemihostWriteString ( | |
IN CHAR8 *String | |
) | |
{ | |
Semihost_SYS_WRITE0(String); | |
} | |
UINT32 | |
SemihostSystem ( | |
IN CHAR8 *CommandLine | |
) | |
{ | |
SEMIHOST_SYSTEM_BLOCK SystemBlock; | |
SystemBlock.CommandLine = CommandLine; | |
SystemBlock.CommandLength = AsciiStrLen(CommandLine); | |
return Semihost_SYS_SYSTEM(&SystemBlock); | |
} |