blob: 7566887fc6503fc8a5954717f2d779ef0212cb0e [file] [log] [blame]
/*
* Copyright (c) 2018, 2019 ARM Limited
* All rights reserved
*
* The license below extends only to copyright in the software and shall
* not be construed as granting a license to any other intellectual
* property including but not limited to intellectual property relating
* to a hardware implementation of the functionality of the software
* licensed hereunder. You may use the software subject to the license
* terms below provided that you ensure that this notice is replicated
* unmodified and in its entirety in all distributions of the software,
* modified or unmodified, in source code or in binary form.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions are
* met: redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer;
* redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in the
* documentation and/or other materials provided with the distribution;
* neither the name of the copyright holders nor the names of its
* contributors may be used to endorse or promote products derived from
* this software without specific prior written permission.
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
* "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
* LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
* A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
* OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
* SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
* LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
* DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
* THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
* OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*/
#ifndef __ARCH_ARM_SEMIHOSTING_HH__
#define __ARCH_ARM_SEMIHOSTING_HH__
#include <cstdio>
#include <functional>
#include <map>
#include <memory>
#include <utility>
#include <vector>
#include "arch/arm/intregs.hh"
#include "arch/arm/utility.hh"
#include "cpu/thread_context.hh"
#include "mem/port_proxy.hh"
#include "sim/guest_abi.hh"
#include "sim/sim_object.hh"
struct ArmSemihostingParams;
class SerialDevice;
/**
* Semihosting for AArch32 and AArch64
*
* This class implements the Arm semihosting interface. This interface
* allows baremetal code access service, such as IO, from the
* simulator. It is conceptually a simplified version of gem5's more
* general syscall emulation mode.
*
* Exits calls (SYS_EXIT, SYS_EXIT_EXTENDED) from the guest get
* translated into simualtion exits. Well-known exit codes are
* translated to messages on the form 'semi:ADP_.*' while unknown
* codes are returned in hex ('semi:0x..'). The subcode is reported in
* the gem5 exit event.
*/
class ArmSemihosting : public SimObject
{
public:
enum {
// Standard ARM immediate values which trigger semihosting.
T32Imm = 0xAB,
A32Imm = 0x123456,
A64Imm = 0xF000,
// The immediate value which enables gem5 semihosting calls. Use the
// standard value for thumb.
Gem5Imm = 0x5D57
};
static PortProxy &portProxy(ThreadContext *tc);
struct AbiBase
{
template <typename Arg>
class StateBase
{
private:
Addr argPointer;
ByteOrder endian;
public:
StateBase(const ThreadContext *tc, Addr arg_pointer) :
argPointer(arg_pointer), endian(ArmISA::byteOrder(tc))
{}
/*
* These two methods are used to both read an argument or its
* address, and to move position on to the next location. Normally
* State would be more passive, but since it behaves almost the
* same no matter what the argument type is we can simplify and
* consolidate a little bit by centralizing these methods.
*/
// Return the address of an argument slot and move past it.
Addr
getAddr()
{
Addr addr = argPointer;
argPointer += sizeof(Arg);
return addr;
}
// Read the value in an argument slot and move past it.
Arg
get(ThreadContext *tc)
{
Arg arg = ArmSemihosting::portProxy(tc).read<Arg>(
argPointer, endian);
argPointer += sizeof(Arg);
return arg;
}
using ArgType = Arg;
};
};
struct Abi64 : public AbiBase
{
using UintPtr = uint64_t;
class State : public StateBase<uint64_t>
{
public:
// For 64 bit semihosting, the params are pointer to by X1.
explicit State(const ThreadContext *tc) :
StateBase<uint64_t>(tc, tc->readIntReg(ArmISA::INTREG_X1))
{}
};
};
struct Abi32 : public AbiBase
{
using UintPtr = uint32_t;
class State : public StateBase<uint64_t>
{
public:
// For 32 bit semihosting, the params are pointer to by R1.
explicit State(const ThreadContext *tc) :
StateBase<uint64_t>(tc, tc->readIntReg(ArmISA::INTREG_R1))
{}
};
};
// Use this argument type when you need to modify an argument in place.
// This will give you the address of the argument itself and the size of
// each argument slot, rather than the actual value of the argument.
struct InPlaceArg
{
Addr addr;
size_t size;
InPlaceArg(Addr _addr, size_t _size) : addr(_addr), size(_size) {}
// A helper function to read the argument since the guest ABI mechanism
// didn't do that for us.
uint64_t
read(ThreadContext *tc, ByteOrder endian)
{
auto &proxy = ArmSemihosting::portProxy(tc);
if (size == 8)
return proxy.read<uint64_t>(addr, endian);
else if (size == 4)
return proxy.read<uint32_t>(addr, endian);
else
panic("Unexpected semihosting argument size %d.", size);
}
// A helper function to write to the argument's slot in the params.
void
write(ThreadContext *tc, uint64_t val, ByteOrder endian)
{
auto &proxy = ArmSemihosting::portProxy(tc);
if (size == 8)
proxy.write<uint64_t>(addr, val, endian);
else if (size == 4)
proxy.write<uint32_t>(addr, val, endian);
else
panic("Unexpected semihosting argument size %d.", size);
}
};
enum Operation {
SYS_OPEN = 0x01,
SYS_CLOSE = 0x02,
SYS_WRITEC = 0x03,
SYS_WRITE0 = 0x04,
SYS_WRITE = 0x05,
SYS_READ = 0x06,
SYS_READC = 0x07,
SYS_ISERROR = 0x08,
SYS_ISTTY = 0x09,
SYS_SEEK = 0x0A,
SYS_FLEN = 0x0C,
SYS_TMPNAM = 0x0D,
SYS_REMOVE = 0x0E,
SYS_RENAME = 0x0F,
SYS_CLOCK = 0x10,
SYS_TIME = 0x11,
SYS_SYSTEM = 0x12,
SYS_ERRNO = 0x13,
SYS_GET_CMDLINE = 0x15,
SYS_HEAPINFO = 0x16,
SYS_EXIT = 0x18,
SYS_EXIT_EXTENDED = 0x20,
SYS_ELAPSED = 0x30,
SYS_TICKFREQ = 0x31,
MaxStandardOp = 0xFF,
SYS_GEM5_PSEUDO_OP = 0x100
};
ArmSemihosting(const ArmSemihostingParams &p);
/** Perform an Arm Semihosting call from aarch64 code. */
bool call64(ThreadContext *tc, bool gem5_ops);
/** Perform an Arm Semihosting call from aarch32 code. */
bool call32(ThreadContext *tc, bool gem5_ops);
public: // SimObject and related interfaces
void serialize(CheckpointOut &cp) const override;
void unserialize(CheckpointIn &cp) override;
protected: // Configuration
const std::string cmdLine;
const Addr memReserve;
const Addr stackSize;
/**
* Base time when the simulation started. This is used to
* calculate the time of date when the guest call SYS_TIME.
*/
const time_t timeBase;
/** Number of bits to right shift gem5 ticks to fit in a uint32_t */
const unsigned tickShift;
protected: // Internal state
typedef uint64_t SemiErrno;
SemiErrno semiErrno;
protected: // File IO
/**
* Internal state for open files
*
* This class describes the internal state of a file opened
* through the semihosting interface.
*
* A file instance is normally created using one of the
* ArmSemihosting::FileBase::create() factory methods. These
* methods handle some the magic file names in the Arm Semihosting
* specification and instantiate the right implementation. For the
* same, when unserializing a checkpoint, the create method must
* be used to unserialize a new instance of a file descriptor.
*/
class FileBase : public Serializable
{
public:
FileBase(ArmSemihosting &_parent, const char *name, const char *_mode)
: parent(_parent), _name(name), mode(_mode) {}
virtual ~FileBase() {};
FileBase() = delete;
FileBase(FileBase &) = delete;
static std::unique_ptr<FileBase> create(
ArmSemihosting &parent, const std::string &fname,
const char *mode);
static std::unique_ptr<FileBase> create(
ArmSemihosting &parent, CheckpointIn &cp, const std::string &sec);
void serialize(CheckpointOut &cp) const override;
void unserialize(CheckpointIn &cp) override;
const std::string &fileName() { return _name; }
public:
/** @{
* Semihosting file IO interfaces
*
* These interfaces implement common IO functionality in the
* Semihosting interface.
*
* All functions return a negative value that corresponds to a
* UNIX errno value when they fail and >=0 on success.
*/
/**
* Open the the file.
*
* @return <0 on error (-errno), 0 on success.
*/
virtual int64_t open() { return 0; }
/**
* Close the file.
*
* @return <0 on error (-errno), 0 on success.
*/
virtual int64_t close() { return 0; }
/**
* Check if a file corresponds to a TTY device.
*
* @return True if the file is a TTY, false otherwise.
*/
virtual bool isTTY() const { return false; }
/**
* Read data from file.
*
* @return <0 on error (-errno), bytes read on success (0 for EOF).
*/
virtual int64_t read(uint8_t *buffer, uint64_t size);
/**
* Write data to file.
*
* @return <0 on error (-errno), bytes written on success.
*/
virtual int64_t write(const uint8_t *buffer, uint64_t size);
/**
* Seek to an absolute position in the file.
*
* @param pos Byte offset from start of file.
* @return <0 on error (-errno), 0 on success.
*/
virtual int64_t seek(uint64_t pos);
/**
* Get the length of a file in bytes.
*
* @return <0 on error (-errno), length on success
*/
virtual int64_t flen();
/** @} */
protected:
ArmSemihosting &parent;
std::string _name;
std::string mode;
};
/** Implementation of the ':semihosting-features' magic file. */
class FileFeatures : public FileBase
{
public:
FileFeatures(ArmSemihosting &_parent,
const char *name, const char *mode);
void serialize(CheckpointOut &cp) const override;
void unserialize(CheckpointIn &cp) override;
int64_t read(uint8_t *buffer, uint64_t size) override;
int64_t seek(uint64_t pos) override;
protected:
size_t pos;
};
class File : public FileBase
{
public:
File(ArmSemihosting &_parent, const char *name, const char *mode);
~File();
void serialize(CheckpointOut &cp) const override;
void unserialize(CheckpointIn &cp) override;
int64_t open() override { return openImpl(false); }
int64_t close() override;
bool isTTY() const override;
int64_t read(uint8_t *buffer, uint64_t size) override;
int64_t write(const uint8_t *buffer, uint64_t size) override;
int64_t seek(uint64_t pos) override;
int64_t flen() override;
protected:
int64_t openImpl(bool unserialize);
bool needClose() const { return !isTTY(); }
FILE *file;
};
std::string filesRootDir;
std::vector<std::unique_ptr<FileBase>> files;
using Handle = size_t;
FILE *stdin;
FILE *stdout;
FILE *stderr;
protected: // Helper functions
unsigned
calcTickShift() const
{
int msb = findMsbSet(SimClock::Frequency);
return msb > 31 ? msb - 31 : 0;
}
uint64_t
semiTick(Tick tick) const
{
return tick >> tickShift;
}
void semiExit(uint64_t code, uint64_t subcode);
std::string readString(ThreadContext *tc, Addr ptr, size_t len);
public:
typedef std::pair<uint64_t, SemiErrno> RetErrno;
private:
static RetErrno
retError(SemiErrno e)
{
return RetErrno((uint64_t)-1, e);
}
static RetErrno
retOK(uint64_t r)
{
return RetErrno(r, 0);
}
/**
* Semihosting call information structure.
*
* This structure describes how a semi-hosting call is
* implemented. It contains debug information (e.g., the name of
* the call), and a way to invoke it in a particular context.
*/
struct SemiCall
{
/** Call name */
const char *name;
// A type for member functions implementing semihosting calls.
template <typename ...Args>
using Implementation =
RetErrno (ArmSemihosting::*)(ThreadContext *tc, Args... args);
// Since guest ABI doesn't know how to call member function pointers,
// this template builds a wrapper that takes care of that.
template <typename ...Args>
static inline std::function<RetErrno(ThreadContext *tc, Args... args)>
wrapImpl(ArmSemihosting *sh, Implementation<Args...> impl)
{
return [sh, impl](ThreadContext *tc, Args... args) {
return (sh->*impl)(tc, args...);
};
}
// A type for functions which dispatch semihosting calls through the
// guest ABI mechanism.
using Dispatcher =
std::function<RetErrno(ArmSemihosting *sh, ThreadContext *tc)>;
using Dumper = std::function<std::string(ThreadContext *tc)>;
// Dispatchers for 32 and 64 bits.
Dispatcher call32;
Dispatcher call64;
// Dumpers which print semihosting calls and their arguments.
Dumper dump32;
Dumper dump64;
// A function which builds a dispatcher for a semihosting call.
template <typename Abi, typename ...Args>
static inline Dispatcher
buildDispatcher(Implementation<Args...> impl)
{
// This lambda is the dispatcher we're building.
return [impl](ArmSemihosting *sh, ThreadContext *tc) {
auto wrapper = wrapImpl(sh, impl);
return invokeSimcall<Abi>(tc, wrapper);
};
}
// A function which builds a dumper for a semihosting call.
template <typename Abi, typename ...Args>
static inline Dumper
buildDumper(const char *name, Implementation<Args...> impl)
{
// This lambda is the dumper we're building.
return [name](ThreadContext *tc) -> std::string {
return dumpSimcall<Abi, RetErrno, Args...>(name, tc);
};
}
// When there's one implementation, use it for both 32 and 64 bits.
template <typename ...Args>
SemiCall(const char *_name, Implementation<Args...> common) :
name(_name), call32(buildDispatcher<Abi32>(common)),
call64(buildDispatcher<Abi64>(common)),
dump32(buildDumper<Abi32>(_name, common)),
dump64(buildDumper<Abi64>(_name, common))
{}
// When there are two, use one for 32 bits and one for 64 bits.
template <typename ...Args32, typename ...Args64>
SemiCall(const char *_name, Implementation<Args32...> impl32,
Implementation<Args64...> impl64) :
name(_name), call32(buildDispatcher<Abi32>(impl32)),
call64(buildDispatcher<Abi64>(impl64)),
dump32(buildDumper<Abi32>(_name, impl32)),
dump64(buildDumper<Abi64>(_name, impl64))
{}
};
RetErrno callOpen(ThreadContext *tc, const Addr name_base,
int fmode, size_t name_size);
RetErrno callClose(ThreadContext *tc, Handle handle);
RetErrno callWriteC(ThreadContext *tc, InPlaceArg c);
RetErrno callWrite0(ThreadContext *tc, InPlaceArg str);
RetErrno callWrite(ThreadContext *tc, Handle handle,
Addr buffer, size_t size);
RetErrno callRead(ThreadContext *tc, Handle handle,
Addr buffer, size_t size);
RetErrno callReadC(ThreadContext *tc);
RetErrno callIsError(ThreadContext *tc, int64_t status);
RetErrno callIsTTY(ThreadContext *tc, Handle handle);
RetErrno callSeek(ThreadContext *tc, Handle handle, uint64_t pos);
RetErrno callFLen(ThreadContext *tc, Handle handle);
RetErrno callTmpNam(ThreadContext *tc, Addr buffer,
uint64_t id, size_t size);
RetErrno callRemove(ThreadContext *tc, Addr name_base, size_t name_size);
RetErrno callRename(ThreadContext *tc, Addr from_addr, size_t from_size,
Addr to_addr, size_t to_size);
RetErrno callClock(ThreadContext *tc);
RetErrno callTime(ThreadContext *tc);
RetErrno callSystem(ThreadContext *tc, Addr cmd_addr, size_t cmd_size);
RetErrno callErrno(ThreadContext *tc);
RetErrno callGetCmdLine(ThreadContext *tc, Addr addr, InPlaceArg size_arg);
void gatherHeapInfo(ThreadContext *tc, bool aarch64,
Addr &heap_base, Addr &heap_limit,
Addr &stack_base, Addr &stack_limit);
RetErrno callHeapInfo32(ThreadContext *tc, Addr block_addr);
RetErrno callHeapInfo64(ThreadContext *tc, Addr block_addr);
RetErrno callExit32(ThreadContext *tc, InPlaceArg code);
RetErrno callExit64(ThreadContext *tc, uint64_t code, uint64_t subcode);
RetErrno callExitExtended(ThreadContext *tc, uint64_t code,
uint64_t subcode);
RetErrno callElapsed32(ThreadContext *tc, InPlaceArg low, InPlaceArg high);
RetErrno callElapsed64(ThreadContext *tc, InPlaceArg ticks);
RetErrno callTickFreq(ThreadContext *tc);
RetErrno callGem5PseudoOp32(ThreadContext *tc, uint32_t encoded_func);
RetErrno callGem5PseudoOp64(ThreadContext *tc, uint64_t encoded_func);
template <typename Abi>
void
unrecognizedCall(ThreadContext *tc, const char *format, uint64_t op)
{
warn(format, op);
std::function<RetErrno(ThreadContext *tc)> retErr =
[](ThreadContext *tc) { return retError(EINVAL); };
invokeSimcall<Abi>(tc, retErr);
}
static FILE *getSTDIO(const char *stream_name,
const std::string &name, const char *mode);
static const std::map<uint32_t, SemiCall> calls;
static const std::vector<const char *> fmodes;
static const std::map<uint64_t, const char *> exitCodes;
static const std::vector<uint8_t> features;
static const std::map<const std::string, FILE *> stdioMap;
// used in callTmpNam() to deterministically generate a temp filename
uint16_t tmpNameIndex = 0;
};
std::ostream &operator << (
std::ostream &os, const ArmSemihosting::InPlaceArg &ipa);
namespace GuestABI
{
template <typename Arg>
struct Argument<ArmSemihosting::Abi64, Arg,
typename std::enable_if_t<std::is_integral<Arg>::value>>
{
static Arg
get(ThreadContext *tc, ArmSemihosting::Abi64::State &state)
{
return state.get(tc);
}
};
template <typename Arg>
struct Argument<ArmSemihosting::Abi32, Arg,
typename std::enable_if_t<std::is_integral<Arg>::value>>
{
static Arg
get(ThreadContext *tc, ArmSemihosting::Abi32::State &state)
{
if (std::is_signed<Arg>::value)
return sext<32>(state.get(tc));
else
return state.get(tc);
}
};
template <typename Abi>
struct Argument<Abi, ArmSemihosting::InPlaceArg, typename std::enable_if_t<
std::is_base_of<ArmSemihosting::AbiBase, Abi>::value>>
{
static ArmSemihosting::InPlaceArg
get(ThreadContext *tc, typename Abi::State &state)
{
return ArmSemihosting::InPlaceArg(
state.getAddr(), sizeof(typename Abi::State::ArgType));
}
};
template <>
struct Result<ArmSemihosting::Abi32, ArmSemihosting::RetErrno>
{
static void
store(ThreadContext *tc, const ArmSemihosting::RetErrno &err)
{
tc->setIntReg(ArmISA::INTREG_R0, err.first);
}
};
template <>
struct Result<ArmSemihosting::Abi64, ArmSemihosting::RetErrno>
{
static void
store(ThreadContext *tc, const ArmSemihosting::RetErrno &err)
{
tc->setIntReg(ArmISA::INTREG_X0, err.first);
}
};
} // namespace GuestABI
#endif // __ARCH_ARM_SEMIHOSTING_HH__