/** @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); }